diff options
Diffstat (limited to 'arduino/libraries/Bluefruit52Lib/src/utility')
5 files changed, 699 insertions, 0 deletions
diff --git a/arduino/libraries/Bluefruit52Lib/src/utility/AdaMsg.cpp b/arduino/libraries/Bluefruit52Lib/src/utility/AdaMsg.cpp new file mode 100755 index 0000000..5c0f43b --- /dev/null +++ b/arduino/libraries/Bluefruit52Lib/src/utility/AdaMsg.cpp @@ -0,0 +1,131 @@ +/**************************************************************************/ +/*! + @file AdaMsg.cpp + @author hathach (tinyusb.org) + + @section LICENSE + + Software License Agreement (BSD License) + + Copyright (c) 2018, Adafruit Industries (adafruit.com) + All rights reserved. + + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are met: + 1. Redistributions of source code must retain the above copyright + notice, this list of conditions and the following disclaimer. + 2. Redistributions in binary form must reproduce the above copyright + notice, this list of conditions and the following disclaimer in the + documentation and/or other materials provided with the distribution. + 3. Neither the name of the copyright holders nor the + names of its contributors may be used to endorse or promote products + derived from this software without specific prior written permission. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS ''AS IS'' AND ANY + EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED + WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE + DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE FOR ANY + DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; + LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND + ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS + SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +*/ +/**************************************************************************/ + +#include "AdaMsg.h" + +void AdaMsg::_init(void) +{ + _dynamic = true; + _waiting = false; + _sem = NULL; + + buffer = NULL; + remaining = xferlen = 0; +} + +AdaMsg::AdaMsg(void) +{ + _init(); +} + +// dynamic mean semaphore is malloced and freed only when in action +void AdaMsg::begin(bool dynamic) +{ + _dynamic = dynamic; + if ( !_dynamic ) + { + _sem = xSemaphoreCreateCounting(10, 0); + } +} + +void AdaMsg::stop(void) +{ + if (!_dynamic) vSemaphoreDelete(_sem); + _init(); +} + +void AdaMsg::prepare(void* buf, uint16_t bufsize) +{ + buffer = (uint8_t*) buf; + remaining = bufsize; + xferlen = 0; +} + +/** + * + * @param ms + * @return -1 if timeout + */ +int32_t AdaMsg::waitUntilComplete(uint32_t ms) +{ + if (_dynamic) + { + _sem = xSemaphoreCreateBinary(); + VERIFY(_sem, -1); + } + + int result = -1; + + _waiting = true; + if ( xSemaphoreTake(_sem, ms2tick(ms) ) ) + { + result = xferlen; + } + _waiting = false; + + if (_dynamic) + { + vSemaphoreDelete(_sem); + _sem = NULL; + } + + return result; +} + +bool AdaMsg::isWaiting(void) +{ + return _waiting; +} + +uint16_t AdaMsg::feed(void* data, uint16_t len) +{ + len = min16(len, remaining); + + // pass NULL to skip copy + if ( data ) memcpy(buffer, data, len); + + buffer += len; + remaining -= len; + xferlen += len; + + return len; +} + +void AdaMsg::complete(void) +{ + if(_sem) xSemaphoreGive(_sem); +} + diff --git a/arduino/libraries/Bluefruit52Lib/src/utility/AdaMsg.h b/arduino/libraries/Bluefruit52Lib/src/utility/AdaMsg.h new file mode 100755 index 0000000..78fd99a --- /dev/null +++ b/arduino/libraries/Bluefruit52Lib/src/utility/AdaMsg.h @@ -0,0 +1,71 @@ +/**************************************************************************/ +/*! + @file AdaMsg.h + @author hathach (tinyusb.org) + + @section LICENSE + + Software License Agreement (BSD License) + + Copyright (c) 2018, Adafruit Industries (adafruit.com) + All rights reserved. + + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are met: + 1. Redistributions of source code must retain the above copyright + notice, this list of conditions and the following disclaimer. + 2. Redistributions in binary form must reproduce the above copyright + notice, this list of conditions and the following disclaimer in the + documentation and/or other materials provided with the distribution. + 3. Neither the name of the copyright holders nor the + names of its contributors may be used to endorse or promote products + derived from this software without specific prior written permission. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS ''AS IS'' AND ANY + EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED + WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE + DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE FOR ANY + DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; + LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND + ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS + SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +*/ +/**************************************************************************/ +#ifndef ADAMSG_H_ +#define ADAMSG_H_ + +#include "Arduino.h" + +class AdaMsg +{ + private: + bool _dynamic; + volatile bool _waiting; + SemaphoreHandle_t _sem; + + void _init(void); + + public: + uint8_t* buffer; + uint16_t remaining; + volatile uint16_t xferlen; + + AdaMsg(void); + + // dynamic mean semaphore is malloced and freed only when in action + void begin(bool dynamic = true); + void stop(void); + + void prepare(void* buf, uint16_t bufsize); + int32_t waitUntilComplete(uint32_t ms); + bool isWaiting(void); + + uint16_t feed(void* data, uint16_t len); + void complete(void); +}; + + + +#endif /* ADAMSG_H_ */ diff --git a/arduino/libraries/Bluefruit52Lib/src/utility/bonding.cpp b/arduino/libraries/Bluefruit52Lib/src/utility/bonding.cpp new file mode 100755 index 0000000..e9f9289 --- /dev/null +++ b/arduino/libraries/Bluefruit52Lib/src/utility/bonding.cpp @@ -0,0 +1,348 @@ +/**************************************************************************/ +/*! + @file bonding.cpp + @author hathach (tinyusb.org) + + @section LICENSE + + Software License Agreement (BSD License) + + Copyright (c) 2018, Adafruit Industries (adafruit.com) + All rights reserved. + + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are met: + 1. Redistributions of source code must retain the above copyright + notice, this list of conditions and the following disclaimer. + 2. Redistributions in binary form must reproduce the above copyright + notice, this list of conditions and the following disclaimer in the + documentation and/or other materials provided with the distribution. + 3. Neither the name of the copyright holders nor the + names of its contributors may be used to endorse or promote products + derived from this software without specific prior written permission. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS ''AS IS'' AND ANY + EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED + WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE + DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE FOR ANY + DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; + LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND + ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS + SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +*/ +/**************************************************************************/ + +#include <Arduino.h> +#include "Bluefruit_FileIO.h" +#include "bonding.h" +#include "bluefruit.h" + +#define BOND_DEBUG 0 + +#if (CFG_DEBUG == 1 && BOND_DEBUG == 1) || (CFG_DEBUG >= 2) +#define BOND_LOG(...) LOG_LV1("BOND", __VA_ARGS__) +#else +#define BOND_LOG(...) +#endif + +/*------------------------------------------------------------------*/ +/* Bond Key is saved in following layout + * - Bond Data : 80 bytes + * - Name : variable (including null char) + * - CCCD : variable + * + * Each field has an 1-byte preceding length + *------------------------------------------------------------------*/ +#define SVC_CONTEXT_FLAG (BLE_GATTS_SYS_ATTR_FLAG_SYS_SRVCS | BLE_GATTS_SYS_ATTR_FLAG_USR_SRVCS) +#define BOND_FNAME_LEN max(sizeof(BOND_FNAME_PRPH), sizeof(BOND_FNAME_CNTR)) + +static void get_fname (char* fname, uint8_t role, uint16_t ediv) +{ + sprintf(fname, (role == BLE_GAP_ROLE_PERIPH) ? BOND_FNAME_PRPH : BOND_FNAME_CNTR, ediv); +} + +static bool bdata_skip_field(File* file) +{ + int len = file->read(); + VERIFY(len > 0); + + file->seek(len + file->position()); + return true; +} + +static void bdata_write(File* file, void const* buffer, uint16_t bufsize) +{ + file->write( (uint8_t) bufsize ); + file->write( (uint8_t const*) buffer, bufsize); +} + +void bond_init(void) +{ + InternalFS.begin(); + + // Create prph and central bond folder if not existed + if ( !InternalFS.exists(BOND_DIR_PRPH) ) InternalFS.mkdir(BOND_DIR_PRPH); + if ( !InternalFS.exists(BOND_DIR_CNTR) ) InternalFS.mkdir(BOND_DIR_CNTR); +} + +/*------------------------------------------------------------------*/ +/* Keys + *------------------------------------------------------------------*/ +static void bond_save_keys_dfr (uint8_t role, uint16_t conn_hdl, bond_keys_t* bkeys) +{ + uint16_t const ediv = (role == BLE_GAP_ROLE_PERIPH) ? bkeys->own_enc.master_id.ediv : bkeys->peer_enc.master_id.ediv; + + char filename[BOND_FNAME_LEN]; + get_fname(filename, role, ediv); + + // delete if file already exists + if ( InternalFS.exists(filename) ) InternalFS.remove(filename); + + File file(filename, FILE_WRITE, InternalFS); + VERIFY(file,); + + //------------- save keys -------------// + bdata_write(&file, bkeys, sizeof(bond_keys_t)); + + //------------- save device name -------------// + char devname[CFG_MAX_DEVNAME_LEN] = { 0 }; + Bluefruit.Gap.getPeerName(conn_hdl, devname, CFG_MAX_DEVNAME_LEN); + + // If couldn't get devname then use peer mac address + if ( !devname[0] ) + { + uint8_t* mac = bkeys->peer_id.id_addr_info.addr; + sprintf(devname, "%02X:%02X:%02X:%02X:%02X:%02X", mac[5], mac[4], mac[3], mac[2], mac[1], mac[0]); + } + + bdata_write(&file, devname, strlen(devname)+1); // save also null char + + BOND_LOG("Saved keys for \"%s\" to file %s ( %d bytes )", devname, filename, file.size()); + + file.close(); +} + +bool bond_save_keys (uint8_t role, uint16_t conn_hdl, bond_keys_t* bkeys) +{ + uint8_t* buf = (uint8_t*) rtos_malloc( sizeof(bond_keys_t) ); + VERIFY(buf); + + memcpy(buf, bkeys, sizeof(bond_keys_t)); + + // queue to execute in Ada Callback thread + ada_callback(buf, bond_save_keys_dfr, role, conn_hdl, buf); + + return true; +} + +bool bond_load_keys(uint8_t role, uint16_t ediv, bond_keys_t* bkeys) +{ + char filename[BOND_FNAME_LEN]; + get_fname(filename, role, ediv); + + File file(filename, FILE_READ, InternalFS); + VERIFY(file); + + int keylen = file.read(); + VERIFY(keylen == sizeof(bond_keys_t)); + + file.read(bkeys, keylen); + file.close(); + + BOND_LOG("Loaded keys from file %s", filename); + + return true; +} + + +/*------------------------------------------------------------------*/ +/* CCCD + *------------------------------------------------------------------*/ +static void bond_save_cccd_dfr (uint8_t role, uint16_t conn_hdl, uint16_t ediv) +{ + uint16_t len=0; + sd_ble_gatts_sys_attr_get(conn_hdl, NULL, &len, SVC_CONTEXT_FLAG); + VERIFY(len, ); + + uint8_t sys_attr[len]; + VERIFY_STATUS(sd_ble_gatts_sys_attr_get(conn_hdl, sys_attr, &len, SVC_CONTEXT_FLAG),); + + char filename[BOND_FNAME_LEN]; + get_fname(filename, role, ediv); + + File file(filename, FILE_WRITE, InternalFS); + VERIFY(file,); + + file.seek(0); // write mode start at the end, seek to beginning + bdata_skip_field(&file); // skip key + bdata_skip_field(&file); // skip name + + bdata_write(&file, sys_attr, len); + + BOND_LOG("Saved CCCD setting to file %s ( offset = %d, len = %d bytes )", filename, file.size() - (len + 1), len); + + file.close(); +} + +bool bond_save_cccd (uint8_t role, uint16_t conn_hdl, uint16_t ediv) +{ + VERIFY(ediv != 0xFFFF); + + // queue to execute in Ada Callback thread + ada_callback(NULL, bond_save_cccd_dfr, role, conn_hdl, ediv); + + return true; +} + +bool bond_load_cccd(uint8_t role, uint16_t conn_hdl, uint16_t ediv) +{ + bool loaded = false; + + if ( ediv != 0xFFFF ) + { + char filename[BOND_FNAME_LEN]; + get_fname(filename, role, ediv); + + File file(filename, FILE_READ, InternalFS); + + if ( file ) + { + bdata_skip_field(&file); // skip key + bdata_skip_field(&file); // skip name + + int len = file.read(); + if ( len > 0 ) + { + uint8_t sys_attr[len]; + + file.read(sys_attr, len); + + if ( ERROR_NONE == sd_ble_gatts_sys_attr_set(conn_hdl, sys_attr, len, SVC_CONTEXT_FLAG) ) + { + loaded = true; + BOND_LOG("Loaded CCCD from file %s ( offset = %d, len = %d bytes )", filename, file.size() - (len + 1), len); + } + } + } + + file.close(); + } + + if ( !loaded ) + { + LOG_LV1("BOND", "CCCD setting not found"); + } + + return loaded; +} + +void bond_print_list(uint8_t role) +{ + char const * dpath = (role == BLE_GAP_ROLE_PERIPH ? BOND_DIR_PRPH : BOND_DIR_CNTR); + + File dir(dpath, FILE_READ, InternalFS); + File file(InternalFS); + + while ( (file = dir.openNextFile(FILE_READ)) ) + { + if ( !file.isDirectory() && bdata_skip_field(&file) ) // skip key + { + int len = file.read(); + if ( len > 0 ) + { + char devname[len]; + file.read(devname, len); + + printf(" %s : %s (%d bytes)\n", file.name(), devname, file.size()); + } + } + + file.close(); + } + + printf("\n"); + + file.close(); + dir.close(); +} + + +bool bond_find_cntr(ble_gap_addr_t* addr, bond_keys_t* bkeys) +{ + bool found = false; + + File dir(BOND_DIR_CNTR, FILE_READ, InternalFS); + File file(InternalFS); + + while ( (file = dir.openNextFile(FILE_READ)) ) + { + // Read bond data of each stored file + int keylen = file.read(); + if ( keylen == sizeof(bond_keys_t) ) + { + file.read((uint8_t*) bkeys, keylen); + + // Compare static address + if ( !memcmp(addr->addr, bkeys->peer_id.id_addr_info.addr, 6) ) + { + found = true; + } + else if ( addr->addr_type == BLE_GAP_ADDR_TYPE_RANDOM_PRIVATE_RESOLVABLE ) + { + // Resolving private address + } + } + + file.close(); + + if ( found ) break; + } + + file.close(); + dir.close(); + + return found; +} + +/*------------------------------------------------------------------*/ +/* DELETE + *------------------------------------------------------------------*/ +void bond_clear_prph(void) +{ + // delete bonds dir + InternalFS.rmdir_r(BOND_DIR_PRPH); + + // Create an empty one + InternalFS.mkdir(BOND_DIR_PRPH); +} + +void bond_clear_cntr(void) +{ + // delete bonds dir + InternalFS.rmdir_r(BOND_DIR_CNTR); + + // Create an empty one + InternalFS.mkdir(BOND_DIR_CNTR); + +} + +void bond_clear_all(void) +{ + // delete bonds dir + InternalFS.rmdir_r(BOND_DIR_PRPH); + InternalFS.rmdir_r(BOND_DIR_CNTR); + + // Create an empty one + InternalFS.mkdir(BOND_DIR_PRPH); + InternalFS.mkdir(BOND_DIR_CNTR); +} + +void bond_remove_key(uint8_t role, uint16_t ediv) +{ + char filename[BOND_FNAME_LEN]; + get_fname(filename, role, ediv); + + InternalFS.remove(filename); +} diff --git a/arduino/libraries/Bluefruit52Lib/src/utility/bonding.h b/arduino/libraries/Bluefruit52Lib/src/utility/bonding.h new file mode 100755 index 0000000..0980369 --- /dev/null +++ b/arduino/libraries/Bluefruit52Lib/src/utility/bonding.h @@ -0,0 +1,74 @@ +/**************************************************************************/ +/*! + @file bonding.h + @author hathach (tinyusb.org) + + @section LICENSE + + Software License Agreement (BSD License) + + Copyright (c) 2018, Adafruit Industries (adafruit.com) + All rights reserved. + + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are met: + 1. Redistributions of source code must retain the above copyright + notice, this list of conditions and the following disclaimer. + 2. Redistributions in binary form must reproduce the above copyright + notice, this list of conditions and the following disclaimer in the + documentation and/or other materials provided with the distribution. + 3. Neither the name of the copyright holders nor the + names of its contributors may be used to endorse or promote products + derived from this software without specific prior written permission. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS ''AS IS'' AND ANY + EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED + WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE + DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE FOR ANY + DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; + LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND + ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS + SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +*/ +/**************************************************************************/ +#ifndef BONDING_H_ +#define BONDING_H_ + +#include "bluefruit_common.h" + +#define BOND_DIR_PRPH "/adafruit/bond_prph" +#define BOND_DIR_CNTR "/adafruit/bond_cntr" + +#define BOND_FNAME_PRPH BOND_DIR_PRPH "/%04x" +#define BOND_FNAME_CNTR BOND_DIR_CNTR "/%04x" + +// Shared keys with bonded device, size = 80 bytes +typedef struct +{ + ble_gap_enc_key_t own_enc; + ble_gap_enc_key_t peer_enc; + ble_gap_id_key_t peer_id; +} bond_keys_t; + +void bond_init(void); +void bond_clear_prph(void); +void bond_clear_cntr(void); +void bond_clear_all(void); + +void bond_remove_key(uint8_t role, uint16_t ediv); + +bool bond_save_keys (uint8_t role, uint16_t conn_hdl, bond_keys_t* bkeys); +bool bond_load_keys(uint8_t role, uint16_t ediv, bond_keys_t* bkeys); + +bool bond_save_cccd (uint8_t role, uint16_t conn_hdl, uint16_t ediv); +bool bond_load_cccd (uint8_t role, uint16_t conn_hdl, uint16_t ediv); + +void bond_print_list(uint8_t role); + +bool bond_find_cntr(ble_gap_addr_t* addr, bond_keys_t* bkeys); + + + +#endif /* BONDING_H_ */ diff --git a/arduino/libraries/Bluefruit52Lib/src/utility/bootloader_util.c b/arduino/libraries/Bluefruit52Lib/src/utility/bootloader_util.c new file mode 100755 index 0000000..470666c --- /dev/null +++ b/arduino/libraries/Bluefruit52Lib/src/utility/bootloader_util.c @@ -0,0 +1,75 @@ +/* Copyright (c) 2013 Nordic Semiconductor. All Rights Reserved. + * + * The information contained herein is property of Nordic Semiconductor ASA. + * Terms and conditions of usage are described in detail in NORDIC + * SEMICONDUCTOR STANDARD SOFTWARE LICENSE AGREEMENT. + * + * Licensees are granted free, non-transferable use of the information. NO + * WARRANTY of ANY KIND is provided. This heading must NOT be removed from + * the file. + * + */ + +#include <stdint.h> +#include <string.h> + + +/** + * @brief Function for aborting current application/bootloader jump to to other app/bootloader. + * + * @details This functions will use the address provide to swap the stack pointer and then load + * the address of the reset handler to be executed. It will check current system mode + * (thread/handler) and if in thread mode it will reset into other application. + * If in handler mode \ref isr_abort will be executed to ensure correct exit of handler + * mode and jump into reset handler of other application. + * + * @param[in] start_addr Start address of other application. This address must point to the + initial stack pointer of the application. + * + * @note This function will never return but issue a reset into provided application. + */ +static inline void bootloader_util_reset (uint32_t start_addr) __attribute__ ((optimize("-fomit-frame-pointer"))); +static inline void bootloader_util_reset(uint32_t start_addr) +{ + __asm volatile( + "ldr r0, [%0]\t\n" // Get App initial MSP for bootloader. + "msr msp, r0\t\n" // Set the main stack pointer to the applications MSP. + "ldr r0, [%0, #0x04]\t\n" // Load Reset handler into R0. + + "movs r4, #0xFF\t\n" // Move ones to R4. + "sxtb r4, r4\t\n" // Sign extend R4 to obtain 0xFFFFFFFF instead of 0xFF. + + "mrs r5, IPSR\t\n" // Load IPSR to R5 to check for handler or thread mode. + "cmp r5, #0x00\t\n" // Compare, if 0 then we are in thread mode and can continue to reset handler of bootloader. + "bne isr_abort\t\n" // If not zero we need to exit current ISR and jump to reset handler of bootloader. + + "mov lr, r4\t\n" // Clear the link register and set to ones to ensure no return. + "bx r0\t\n" // Branch to reset handler of bootloader. + + "isr_abort: \t\n" + + "mov r5, r4\t\n" // Fill with ones before jumping to reset handling. Will be popped as LR when exiting ISR. Ensures no return to application. + "mov r6, r0\t\n" // Move address of reset handler to R6. Will be popped as PC when exiting ISR. Ensures the reset handler will be executed when exist ISR. + "movs r7, #0x21\t\n" // Move MSB reset value of xPSR to R7. Will be popped as xPSR when exiting ISR. xPSR is 0x21000000 thus MSB is 0x21. + "rev r7, r7\t\n" // Reverse byte order to put 0x21 as MSB. + "push {r4-r7}\t\n" // Push everything to new stack to allow interrupt handler to fetch it on exiting the ISR. + + "movs r4, #0x00\t\n" // Fill with zeros before jumping to reset handling. We be popped as R0 when exiting ISR (Cleaning up of the registers). + "movs r5, #0x00\t\n" // Fill with zeros before jumping to reset handling. We be popped as R1 when exiting ISR (Cleaning up of the registers). + "movs r6, #0x00\t\n" // Fill with zeros before jumping to reset handling. We be popped as R2 when exiting ISR (Cleaning up of the registers). + "movs r7, #0x00\t\n" // Fill with zeros before jumping to reset handling. We be popped as R3 when exiting ISR (Cleaning up of the registers). + "push {r4-r7}\t\n" // Push zeros (R4-R7) to stack to prepare for exiting the interrupt routine. + + "movs r0, #0xF9\t\n" // Move the execution return command into register, 0xFFFFFFF9. + "sxtb r0, r0\t\n" // Sign extend R0 to obtain 0xFFFFFFF9 instead of 0xF9. + "bx r0\t\n" // No return - Handler mode will be exited. Stack will be popped and execution will continue in reset handler initializing other application. + ".align\t\n" + :: "r" (start_addr) // Argument list for the gcc assembly. start_addr is %0. + : "r0", "r4", "r5", "r6", "r7" // List of register maintained manually. + ); +} + +void bootloader_util_app_start(uint32_t start_addr) +{ + bootloader_util_reset(start_addr); +} |