aboutsummaryrefslogtreecommitdiffstats
path: root/arduino/libraries/Bluefruit52Lib/src/utility
diff options
context:
space:
mode:
Diffstat (limited to 'arduino/libraries/Bluefruit52Lib/src/utility')
-rwxr-xr-xarduino/libraries/Bluefruit52Lib/src/utility/AdaMsg.cpp131
-rwxr-xr-xarduino/libraries/Bluefruit52Lib/src/utility/AdaMsg.h71
-rwxr-xr-xarduino/libraries/Bluefruit52Lib/src/utility/bonding.cpp348
-rwxr-xr-xarduino/libraries/Bluefruit52Lib/src/utility/bonding.h74
-rwxr-xr-xarduino/libraries/Bluefruit52Lib/src/utility/bootloader_util.c75
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);
+}