Hi,
In my current project I have indicating LEDs connected via TWI bus. In this case, I can't use BSP_LED_0/BSP_LED_1 outputs to indicate states as it implemented in nrf_ble_dfu.c
Unfortunately, I see no ways to propose patch set for SDK, so please forward my change to SDK developers.
In this version we have __WEAK functions for LED control and user can re-define indication in his project.
/**
* Copyright (c) 2016 - 2017, Nordic Semiconductor ASA
*
* 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, except as embedded into a Nordic
* Semiconductor ASA integrated circuit in a product or a software update for
* such product, 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 Nordic Semiconductor ASA nor the names of its
* contributors may be used to endorse or promote products derived from this
* software without specific prior written permission.
*
* 4. This software, with or without modification, must only be used with a
* Nordic Semiconductor ASA integrated circuit.
*
* 5. Any software provided in binary form under this license must not be reverse
* engineered, decompiled, modified and/or disassembled.
*
* THIS SOFTWARE IS PROVIDED BY NORDIC SEMICONDUCTOR ASA "AS IS" AND ANY EXPRESS
* OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
* OF MERCHANTABILITY, NONINFRINGEMENT, AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL NORDIC SEMICONDUCTOR ASA OR CONTRIBUTORS 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 "nrf_ble_dfu.h"
#include <stddef.h>
#include "boards.h"
#include "sdk_common.h"
#include "nrf_dfu_transport.h"
#include "nrf_dfu_req_handler.h"
#include "nrf_dfu_handling_error.h"
#include "nrf_dfu_mbr.h"
#include "nrf_bootloader_info.h"
#include "ble_conn_params.h"
#include "ble_hci.h"
#include "nrf_sdh.h"
#include "nrf_sdh_ble.h"
#include "nrf_log.h"
#include "nrf_delay.h"
#include "app_timer.h"
#include "nrf_dfu_svci.h"
#include "nrf_dfu_settings.h"
#ifdef BSP_LED_0
#define ADVERTISING_LED_PIN_NO BSP_LED_0 /**< Is on when device is advertising. */
#endif
#ifdef BSP_LED_1
#define CONNECTED_LED_PIN_NO BSP_LED_1 /**< Is on when device has connected. */
#endif
#define DEVICE_NAME "DfuTarg" /**< Name of device included in the advertising data. Must be shorter than or equal to 20 bytes. */
#define MANUFACTURER_NAME "NordicSemiconductor" /**< Manufacturer. Will be passed to Device Information Service. */
#define APP_BLE_CONN_CFG_TAG 1 /**< A tag identifying the SoftDevice BLE configuration. */
#define MIN_CONN_INTERVAL (uint16_t)(MSEC_TO_UNITS(7.5, UNIT_1_25_MS)) /**< Minimum acceptable connection interval. */
#define MAX_CONN_INTERVAL (uint16_t)(MSEC_TO_UNITS(30, UNIT_1_25_MS)) /**< Maximum acceptable connection interval. */
#define SLAVE_LATENCY 0 /**< Slave latency. */
#define CONN_SUP_TIMEOUT (4 * 100) /**< Connection supervisory timeout (4 seconds). */
#define MAX_ADV_NAME_LENGTH 20 /**< Maximum length of advertising name. */
#define APP_ADV_DATA_HEADER_SIZE (9) /**< Size of encoded advertisement data header (not including device name). */
#define APP_ADV_INTERVAL MSEC_TO_UNITS(25, UNIT_0_625_MS) /**< The advertising interval (25 ms.). */
#define APP_ADV_TIMEOUT_IN_SECONDS BLE_GAP_ADV_TIMEOUT_GENERAL_UNLIMITED /**< The advertising timeout in units of seconds. This is set to @ref BLE_GAP_ADV_TIMEOUT_GENERAL_UNLIMITED so that the advertisement is done as long as there there is a call to @ref dfu_transport_close function.*/
#define APP_FEATURE_NOT_SUPPORTED BLE_GATT_STATUS_ATTERR_APP_BEGIN + 2 /**< Reply when unsupported features are requested. */
#define SEC_PARAM_BOND 0 /**< Perform bonding. */
#define SEC_PARAM_MITM 0 /**< Man In The Middle protection not required. */
#define SEC_PARAM_IO_CAPABILITIES BLE_GAP_IO_CAPS_NONE /**< No I/O capabilities. */
#define SEC_PARAM_OOB 0 /**< Out Of Band data not available. */
#define SEC_PARAM_MIN_KEY_SIZE 7 /**< Minimum encryption key size. */
#define SEC_PARAM_MAX_KEY_SIZE 16 /**< Maximum encryption key size. */
#define OPCODE_LEN 1 /**< Length of opcode inside Heart Rate Measurement packet. */
#define HANDLE_LEN 2 /**< Length of handle inside Heart Rate Measurement packet. */
#define MAX_DFU_PKT_LEN NRF_SDH_BLE_GATT_MAX_MTU_SIZE - OPCODE_LEN - HANDLE_LEN /**< Maximum length (in bytes) of the DFU Packet characteristic. */
STATIC_ASSERT(MAX_DFU_PKT_LEN <= FLASH_BUFFER_LENGTH); /**< Assert to prevent buffer overflow. Every write to the DFU Packet control point is buffered and written to flash. */
#define PKT_CREATE_PARAM_LEN (6) /**< Length (in bytes) of the parameters for Create Object request. */
#define PKT_SET_PRN_PARAM_LEN (3) /**< Length (in bytes) of the parameters for Set Packet Receipt Notification request. */
#define PKT_READ_OBJECT_INFO_PARAM_LEN (2) /**< Length (in bytes) of the parameters for Read Object Info request. */
#define MAX_RESPONSE_LEN (17) /**< Maximum length (in bytes) of the response to a Control Point command. */
#define DFU_BLE_FLAG_NONE (0)
#define DFU_BLE_FLAG_SERVICE_INITIALIZED (1 << 0) /**< Flag to check if the DFU service was initialized by the application.*/
#define DFU_BLE_FLAG_USE_ADV_NAME (1 << 1) /**< Flag to indicate that advertisement name is to be used. */
#define DFU_BLE_RESETTING_SOON (1 << 2) /**< Flag to indicate that the device will reset soon. */
#define BLE_OBSERVER_PRIO 2 /**< BLE observer priority. Controls the priority for BLE event handler. */
DFU_TRANSPORT_REGISTER(nrf_dfu_transport_t const dfu_trans) =
{
.init_func = ble_dfu_transport_init,
.close_func = ble_dfu_transport_close
};
static ble_dfu_t m_dfu; /**< Structure used to identify the Device Firmware Update service. */
static uint16_t m_pkt_notif_target; /**< Number of packets of firmware data to be received before transmitting the next Packet Receipt Notification to the DFU Controller. */
static uint16_t m_pkt_notif_target_cnt; /**< Number of packets of firmware data received after sending last Packet Receipt Notification or since the receipt of a @ref BLE_DFU_PKT_RCPT_NOTIF_ENABLED event from the DFU service, which ever occurs later.*/
static uint16_t m_conn_handle = BLE_CONN_HANDLE_INVALID; /**< Handle of the current connection. */
static uint32_t m_flags;
static uint8_t m_notif_buffer[MAX_RESPONSE_LEN]; /**< Buffer used for sending notifications to peer. */
#if defined(NRF_DFU_BLE_REQUIRES_BONDS) && (NRF_DFU_BLE_REQUIRES_BONDS == 1)
static nrf_dfu_peer_data_t m_peer_data;
static ble_gap_addr_t const * m_whitelist[1]; /**< List of peers in whitelist (only one) */
static ble_gap_id_key_t const * m_gap_ids[1];
#else
static nrf_dfu_adv_name_t m_adv_name;
#endif
__WEAK void nrf_dfu_advertising_led(uint8_t state)
{
#ifdef ADVERTISING_LED_PIN_NO
if (state)
nrf_gpio_pin_set(ADVERTISING_LED_PIN_NO);
else
nrf_gpio_pin_clear(ADVERTISING_LED_PIN_NO);
#endif
}
__WEAK void nrf_dfu_connected_led(uint8_t state)
{
#ifdef CONNECTED_LED_PIN_NO
if (state)
nrf_gpio_pin_set(CONNECTED_LED_PIN_NO);
else
nrf_gpio_pin_clear(CONNECTED_LED_PIN_NO);
#endif
}
/**@brief Function for the Advertising functionality initialization.
*
* @details Encodes the required advertising data and passes it to the stack.
* The advertising data encoded here is specific for DFU.
* Setting advertising data can by done by calling @ref ble_advdata_set.
*/
static uint32_t advertising_init(uint8_t adv_flags)
{
uint32_t err_code;
uint16_t len_advdata = APP_ADV_DATA_HEADER_SIZE;
uint16_t actual_device_name_length = BLE_GAP_ADV_MAX_SIZE - APP_ADV_DATA_HEADER_SIZE;
uint8_t p_encoded_advdata[BLE_GAP_ADV_MAX_SIZE];
// Encode flags.
p_encoded_advdata[0] = 0x2;
p_encoded_advdata[1] = BLE_GAP_AD_TYPE_FLAGS;
p_encoded_advdata[2] = adv_flags;
// Encode 'more available' UUID list.
p_encoded_advdata[3] = 0x3;
p_encoded_advdata[4] = BLE_GAP_AD_TYPE_16BIT_SERVICE_UUID_MORE_AVAILABLE;
p_encoded_advdata[5] = LSB_16(BLE_DFU_SERVICE_UUID);
p_encoded_advdata[6] = MSB_16(BLE_DFU_SERVICE_UUID);
// Get GAP device name and length.
err_code = sd_ble_gap_device_name_get(&p_encoded_advdata[9], &actual_device_name_length);
if (err_code != NRF_SUCCESS)
{
return err_code;
}
// Set GAP device in advertising data.
p_encoded_advdata[7] = actual_device_name_length + 1; // (actual_length + ADV_AD_TYPE_FIELD_SIZE(1))
p_encoded_advdata[8] = BLE_GAP_AD_TYPE_COMPLETE_LOCAL_NAME;
len_advdata += actual_device_name_length;
return sd_ble_gap_adv_data_set(p_encoded_advdata, len_advdata, NULL, 0);
}
/**@brief Function for starting advertising.
*/
static uint32_t advertising_start(void)
{
uint32_t err_code;
ble_gap_adv_params_t adv_params =
{
.type = BLE_GAP_ADV_TYPE_ADV_IND,
.p_peer_addr = NULL,
.fp = BLE_GAP_ADV_FP_ANY,
.interval = APP_ADV_INTERVAL,
.timeout = APP_ADV_TIMEOUT_IN_SECONDS,
};
uint8_t adv_flag = BLE_GAP_ADV_FLAGS_LE_ONLY_GENERAL_DISC_MODE;
#if defined(NRF_DFU_BLE_REQUIRES_BONDS) && (NRF_DFU_BLE_REQUIRES_BONDS == 1)
ble_gap_irk_t empty_irk = {{0}};
NRF_LOG_DEBUG("##### Setting adv with peer data ####");
if (memcmp(m_peer_data.ble_id.id_info.irk, empty_irk.irk, sizeof(ble_gap_irk_t)) == 0)
{
NRF_LOG_DEBUG("##### No IRK, General discovery");
}
else
{
NRF_LOG_DEBUG("##### IRK Found. Setting whitelist ####");
// API versions lower than 3 are not supported.
STATIC_ASSERT(NRF_SD_BLE_API_VERSION >= 3);
m_whitelist[0] = &m_peer_data.ble_id.id_addr_info;
err_code = sd_ble_gap_whitelist_set(m_whitelist, 1);
VERIFY_SUCCESS(err_code);
m_gap_ids[0] = &m_peer_data.ble_id;
sd_ble_gap_device_identities_set(m_gap_ids, NULL, 1);
VERIFY_SUCCESS(err_code);
adv_flag = BLE_GAP_ADV_FLAG_BR_EDR_NOT_SUPPORTED;
adv_params.fp = BLE_GAP_ADV_FP_FILTER_CONNREQ;
}
#else
{
NRF_LOG_DEBUG("#### Advertising NO BONDING ####");
}
#endif // NRF_DFU_BLE_REQUIRES_BONDS
err_code = advertising_init(adv_flag);
VERIFY_SUCCESS(err_code);
(void) sd_ble_gap_adv_stop();
err_code = sd_ble_gap_adv_start(&adv_params, APP_BLE_CONN_CFG_TAG);
VERIFY_SUCCESS(err_code);
nrf_dfu_advertising_led(0);
nrf_dfu_connected_led(1);
return NRF_SUCCESS;
}
/**@brief Function for stopping advertising.
*/
static void advertising_stop(void)
{
(void)sd_ble_gap_adv_stop();
nrf_dfu_advertising_led(1);
}
static bool is_cccd_configured(ble_dfu_t * p_dfu)
{
uint8_t cccd_val_buf[BLE_CCCD_VALUE_LEN];
ble_gatts_value_t gatts_value = {0};
gatts_value.len = BLE_CCCD_VALUE_LEN;
gatts_value.p_value = cccd_val_buf;
// Check the CCCD Value of DFU Control Point.
uint32_t err_code = sd_ble_gatts_value_get(m_conn_handle,
p_dfu->dfu_ctrl_pt_handles.cccd_handle,
&gatts_value);
VERIFY_SUCCESS(err_code);
return ble_srv_is_notification_enabled(cccd_val_buf);
}
static uint32_t send_hvx(uint16_t conn_handle, uint16_t value_handle, uint16_t len)
{
ble_gatts_hvx_params_t hvx_params = {0};
hvx_params.handle = value_handle;
hvx_params.type = BLE_GATT_HVX_NOTIFICATION;
hvx_params.p_len = &len;
hvx_params.p_data = m_notif_buffer;
return sd_ble_gatts_hvx(conn_handle, &hvx_params);
}
#if defined(NRF_DFU_BLE_REQUIRES_BONDS) && (NRF_DFU_BLE_REQUIRES_BONDS == 1)
static uint32_t service_changed_send(void)
{
uint32_t err_code;
NRF_LOG_DEBUG("Sending Service Changed indication");
err_code = sd_ble_gatts_sys_attr_set(m_conn_handle,
m_peer_data.sys_serv_attr,
sizeof(m_peer_data.sys_serv_attr),
BLE_GATTS_SYS_ATTR_FLAG_SYS_SRVCS);
VERIFY_SUCCESS(err_code);
err_code = sd_ble_gatts_sys_attr_set(m_conn_handle,
NULL,
0,
BLE_GATTS_SYS_ATTR_FLAG_USR_SRVCS);
VERIFY_SUCCESS(err_code);
err_code = sd_ble_gatts_service_changed(m_conn_handle, m_dfu.service_handle, 0xFFFF);
if ((err_code == BLE_ERROR_INVALID_CONN_HANDLE) ||
(err_code == NRF_ERROR_INVALID_STATE) ||
(err_code == NRF_ERROR_BUSY))
{
// Those errors can be expected when sending trying to send Service Changed indication
// if the CCCD is not set to indicate. Thus set the returning error code to success.
NRF_LOG_WARNING("Client did not have the Service Changed indication set to enabled. Error: 0x%08x", err_code);
err_code = NRF_SUCCESS;
}
return err_code;
}
#endif
static uint32_t response_send(ble_dfu_t * p_dfu,
uint8_t op_code,
nrf_dfu_res_code_t resp_val)
{
uint16_t index = 0;
NRF_LOG_DEBUG("Sending Response: [0x%01x, 0x%01x]", op_code, resp_val);
ASSERT(p_dfu != NULL);
if ((m_conn_handle == BLE_CONN_HANDLE_INVALID) || (m_flags & DFU_BLE_FLAG_SERVICE_INITIALIZED) == 0)
{
return NRF_ERROR_INVALID_STATE;
}
m_notif_buffer[index++] = BLE_DFU_OP_CODE_RESPONSE;
// Encode the Request Op code.
m_notif_buffer[index++] = op_code;
// Encode the Response Value.
m_notif_buffer[index++] = (uint8_t)resp_val;
// If the error was an extended error code, add the error to the response.
if (resp_val == NRF_DFU_RES_CODE_EXT_ERROR)
{
m_notif_buffer[index++] = ext_error_get();
// Clear the last extended error code.
(void) ext_error_set(NRF_DFU_EXT_ERROR_NO_ERROR);
}
return send_hvx(m_conn_handle, p_dfu->dfu_ctrl_pt_handles.value_handle, index);
}
static uint32_t response_crc_cmd_send(ble_dfu_t * p_dfu,
uint32_t offset,
uint32_t crc)
{
uint16_t index = 0;
NRF_LOG_DEBUG("Sending CRC: [0x60, 0x03, 0x01, 0:x%08x, CRC:0x%08x]", offset, crc);
ASSERT(p_dfu != NULL);
if ((m_conn_handle == BLE_CONN_HANDLE_INVALID) || (m_flags & DFU_BLE_FLAG_SERVICE_INITIALIZED) == 0)
{
return NRF_ERROR_INVALID_STATE;
}
m_notif_buffer[index++] = BLE_DFU_OP_CODE_RESPONSE;
// Encode the Request Op code
m_notif_buffer[index++] = BLE_DFU_OP_CODE_CALCULATE_CRC;
// Encode the Response Value.
m_notif_buffer[index++] = (uint8_t)NRF_DFU_RES_CODE_SUCCESS;
// Encode the Offset Value.
index += uint32_encode(offset, &m_notif_buffer[index]);
// Encode the Crc Value.
index += uint32_encode(crc, &m_notif_buffer[index]);
return send_hvx(m_conn_handle, p_dfu->dfu_ctrl_pt_handles.value_handle, index);
}
static uint32_t response_select_object_cmd_send(ble_dfu_t * p_dfu,
uint32_t max_size,
uint32_t offset,
uint32_t crc)
{
uint16_t index = 0;
NRF_LOG_DEBUG("Sending Object Info: [0x60, 0x06, 0x01 max: 0:x%08x 0:x%08x, CRC:0x%08x]",
max_size, offset, crc);
ASSERT(p_dfu != NULL);
if ((m_conn_handle == BLE_CONN_HANDLE_INVALID) || (m_flags & DFU_BLE_FLAG_SERVICE_INITIALIZED) == 0)
{
return NRF_ERROR_INVALID_STATE;
}
m_notif_buffer[index++] = BLE_DFU_OP_CODE_RESPONSE;
// Encode the Request Op code.
m_notif_buffer[index++] = BLE_DFU_OP_CODE_SELECT_OBJECT;
// Encode the Success Response Value.
m_notif_buffer[index++] = (uint8_t)NRF_DFU_RES_CODE_SUCCESS;
// Encode the Max Size Value.
index += uint32_encode(max_size, &m_notif_buffer[index]);
// Encode the Offset Value.
index += uint32_encode(offset, &m_notif_buffer[index]);
// Encode the CRC Value.
index += uint32_encode(crc, &m_notif_buffer[index]);
return send_hvx(m_conn_handle, p_dfu->dfu_ctrl_pt_handles.value_handle, index);
}
/**@brief Function for handling a Write event on the Control Point characteristic.
*
* @param[in] p_dfu DFU Service Structure.
* @param[in] p_ble_write_evt Pointer to the write event received from BLE stack.
*
* @return NRF_SUCCESS on successful processing of control point write. Otherwise an error code.
*/
static uint32_t on_ctrl_pt_write(ble_dfu_t * p_dfu, ble_gatts_evt_write_t const * p_ble_write_evt)
{
nrf_dfu_res_code_t res_code;
nrf_dfu_req_t dfu_req;
nrf_dfu_res_t dfu_res = {{{0}}};
memset(&dfu_req, 0, sizeof(nrf_dfu_req_t));
switch (p_ble_write_evt->data[0])
{
case BLE_DFU_OP_CODE_CREATE_OBJECT:
if (p_ble_write_evt->len != PKT_CREATE_PARAM_LEN)
{
return response_send(p_dfu,
BLE_DFU_OP_CODE_CREATE_OBJECT,
NRF_DFU_RES_CODE_INVALID_PARAMETER);
}
NRF_LOG_DEBUG("Received create object");
// Reset the packet receipt notification on create object
m_pkt_notif_target_cnt = m_pkt_notif_target;
// Get type parameter
//lint -save -e415
dfu_req.obj_type = p_ble_write_evt->data[1];
//lint -restore
// Get length value
//lint -save -e416
dfu_req.object_size = uint32_decode(&(p_ble_write_evt->data[2]));
//lint -restore
// Set req type
dfu_req.req_type = NRF_DFU_OBJECT_OP_CREATE;
res_code = nrf_dfu_req_handler_on_req(NULL, &dfu_req, &dfu_res);
return response_send(p_dfu, BLE_DFU_OP_CODE_CREATE_OBJECT, res_code);
case BLE_DFU_OP_CODE_EXECUTE_OBJECT:
NRF_LOG_DEBUG("Received execute object");
// Set req type
dfu_req.req_type = NRF_DFU_OBJECT_OP_EXECUTE;
res_code = nrf_dfu_req_handler_on_req(NULL, &dfu_req, &dfu_res);
return response_send(p_dfu, BLE_DFU_OP_CODE_EXECUTE_OBJECT, res_code);
case BLE_DFU_OP_CODE_SET_RECEIPT_NOTIF:
NRF_LOG_DEBUG("Set receipt notif");
if (p_ble_write_evt->len != PKT_SET_PRN_PARAM_LEN)
{
return (response_send(p_dfu,
BLE_DFU_OP_CODE_SET_RECEIPT_NOTIF,
NRF_DFU_RES_CODE_INVALID_PARAMETER));
}
//lint -save -e415
m_pkt_notif_target = uint16_decode(&(p_ble_write_evt->data[1]));
//lint -restore
m_pkt_notif_target_cnt = m_pkt_notif_target;
return response_send(p_dfu, BLE_DFU_OP_CODE_SET_RECEIPT_NOTIF, NRF_DFU_RES_CODE_SUCCESS);
case BLE_DFU_OP_CODE_CALCULATE_CRC:
NRF_LOG_DEBUG("Received calculate CRC");
dfu_req.req_type = NRF_DFU_OBJECT_OP_CRC;
res_code = nrf_dfu_req_handler_on_req(NULL, &dfu_req, &dfu_res);
if (res_code == NRF_DFU_RES_CODE_SUCCESS)
{
return response_crc_cmd_send(p_dfu, dfu_res.offset, dfu_res.crc);
}
else
{
return response_send(p_dfu, BLE_DFU_OP_CODE_CALCULATE_CRC, res_code);
}
case BLE_DFU_OP_CODE_SELECT_OBJECT:
NRF_LOG_DEBUG("Received select object");
if (p_ble_write_evt->len != PKT_READ_OBJECT_INFO_PARAM_LEN)
{
return response_send(p_dfu,
BLE_DFU_OP_CODE_SELECT_OBJECT,
NRF_DFU_RES_CODE_INVALID_PARAMETER);
}
// Set object type to read info about
//lint -save -e415
dfu_req.obj_type = p_ble_write_evt->data[1];
//lint -restore
dfu_req.req_type = NRF_DFU_OBJECT_OP_SELECT;
res_code = nrf_dfu_req_handler_on_req(NULL, &dfu_req, &dfu_res);
if (res_code == NRF_DFU_RES_CODE_SUCCESS)
{
return response_select_object_cmd_send(p_dfu, dfu_res.max_size, dfu_res.offset, dfu_res.crc);
}
else
{
return response_send(p_dfu, BLE_DFU_OP_CODE_SELECT_OBJECT, res_code);
}
default:
NRF_LOG_WARNING("Received unsupported OP code");
// Unsupported op code.
return response_send(p_dfu,
p_ble_write_evt->data[0],
NRF_DFU_RES_CODE_INVALID_PARAMETER);
}
}
/**@brief Function for handling the @ref BLE_GATTS_EVT_RW_AUTHORIZE_REQUEST event from the
* SoftDevice.
*
* @param[in] p_dfu DFU Service Structure.
* @param[in] p_ble_evt Pointer to the event received from BLE stack.
*/
static bool on_rw_authorize_req(ble_dfu_t * p_dfu, ble_evt_t const * p_ble_evt)
{
uint32_t err_code;
ble_gatts_rw_authorize_reply_params_t auth_reply = {0};
ble_gatts_evt_rw_authorize_request_t const * p_authorize_request;
ble_gatts_evt_write_t const * p_ble_write_evt;
p_authorize_request = &(p_ble_evt->evt.gatts_evt.params.authorize_request);
p_ble_write_evt = &(p_ble_evt->evt.gatts_evt.params.authorize_request.request.write);
if ((p_authorize_request->type == BLE_GATTS_AUTHORIZE_TYPE_WRITE) &&
(p_authorize_request->request.write.handle == p_dfu->dfu_ctrl_pt_handles.value_handle) &&
(p_authorize_request->request.write.op != BLE_GATTS_OP_PREP_WRITE_REQ) &&
(p_authorize_request->request.write.op != BLE_GATTS_OP_EXEC_WRITE_REQ_NOW) &&
(p_authorize_request->request.write.op != BLE_GATTS_OP_EXEC_WRITE_REQ_CANCEL) )
{
auth_reply.type = BLE_GATTS_AUTHORIZE_TYPE_WRITE;
auth_reply.params.write.update = 1;
auth_reply.params.write.offset = p_ble_write_evt->offset;
auth_reply.params.write.len = p_ble_write_evt->len;
auth_reply.params.write.p_data = p_ble_write_evt->data;
if (!is_cccd_configured(p_dfu))
{
// Send an error response to the peer indicating that the CCCD is improperly configured.
auth_reply.params.write.gatt_status = BLE_GATT_STATUS_ATTERR_CPS_CCCD_CONFIG_ERROR;
// Ignore response of auth reply
(void)sd_ble_gatts_rw_authorize_reply(m_conn_handle, &auth_reply);
return false;
}
auth_reply.params.write.gatt_status = BLE_GATT_STATUS_SUCCESS;
err_code = sd_ble_gatts_rw_authorize_reply(m_conn_handle, &auth_reply);
return err_code == NRF_SUCCESS ? true: false;
}
else
{
return false;
}
}
/**@brief Function for handling the @ref BLE_GATTS_EVT_WRITE event from the SoftDevice.
*
* @param[in] p_dfu DFU Service Structure.
* @param[in] p_ble_evt Pointer to the event received from BLE stack.
*/
static void on_write(ble_dfu_t * p_dfu, ble_evt_t const * p_ble_evt)
{
if (p_ble_evt->evt.gatts_evt.params.write.handle == p_dfu->dfu_pkt_handles.value_handle)
{
nrf_dfu_res_code_t res_code;
nrf_dfu_req_t dfu_req;
nrf_dfu_res_t dfu_res = {{{0}}};
memset(&dfu_req, 0, sizeof(nrf_dfu_req_t));
// Set req type
dfu_req.req_type = NRF_DFU_OBJECT_OP_WRITE;
// Set data and length
dfu_req.p_req = (uint8_t*)p_ble_evt->evt.gatts_evt.params.write.data;
dfu_req.req_len = p_ble_evt->evt.gatts_evt.params.write.len;
res_code = nrf_dfu_req_handler_on_req(NULL, &dfu_req, &dfu_res);
if (res_code != NRF_DFU_RES_CODE_SUCCESS)
{
NRF_LOG_ERROR("Failure to run packet write");
}
// Check if a packet receipt notification is needed to be sent.
if (m_pkt_notif_target != 0 && --m_pkt_notif_target_cnt == 0)
{
(void)response_crc_cmd_send(p_dfu, dfu_res.offset, dfu_res.crc);
// Reset the counter for the number of firmware packets.
m_pkt_notif_target_cnt = m_pkt_notif_target;
}
}
}
/**@brief Function for handling the HVC event.
*
* @details Handles HVC events from the BLE stack.
*
* @param[in] p_dfu DFU Service structure.
* @param[in] p_ble_evt Event received from the BLE stack.
*/
static void on_hvc(ble_dfu_t * p_dfu, ble_evt_t const * p_ble_evt)
{
ble_gatts_evt_hvc_t const * p_hvc = &p_ble_evt->evt.gatts_evt.params.hvc;
if (p_hvc->handle == BLE_UUID_GATT_CHARACTERISTIC_SERVICE_CHANGED)
{
NRF_LOG_DEBUG("on_hvx: service_changed handle");
}
else
{
NRF_LOG_DEBUG("on_hvx: Handle: 0x%04x", p_hvc->handle);
}
}
/**@brief Function for the Application's SoftDevice event handler.
*
* @param[in] p_ble_evt SoftDevice event.
*/
static void ble_evt_handler(ble_evt_t const * p_ble_evt, void * p_context)
{
uint32_t err_code = NRF_SUCCESS;
switch (p_ble_evt->header.evt_id)
{
case BLE_GAP_EVT_CONNECTED:
{
nrf_dfu_connected_led(0);
nrf_dfu_advertising_led(1);
m_conn_handle = p_ble_evt->evt.gap_evt.conn_handle;
} break;
case BLE_GAP_EVT_DISCONNECTED:
{
// Restart advertising so that the DFU Controller can reconnect if possible.
nrf_dfu_connected_led(1);
if ((m_flags & DFU_BLE_RESETTING_SOON) == 0)
{
err_code = advertising_start();
APP_ERROR_CHECK(err_code);
}
m_conn_handle = BLE_CONN_HANDLE_INVALID;
} break;
case BLE_GATTS_EVT_HVN_TX_COMPLETE:
{
nrf_dfu_req_handler_reset_if_dfu_complete();
} break;
case BLE_GAP_EVT_SEC_PARAMS_REQUEST:
{
NRF_LOG_DEBUG("Received: BLE_GAP_EVT_SEC_PARAMS_REQUEST");
uint16_t cccd;
ble_gatts_value_t value =
{
.len = BLE_CCCD_VALUE_LEN,
.offset = 0,
.p_value = (uint8_t*)&cccd
};
err_code = sd_ble_gatts_value_get(m_conn_handle, BLE_UUID_GATT_CHARACTERISTIC_SERVICE_CHANGED, &value);
APP_ERROR_CHECK(err_code);
NRF_LOG_DEBUG("CCCD for the service changed is 0x%04x", cccd);
ble_gap_sec_keyset_t * p_keys = NULL;
ble_gap_sec_params_t * p_sec_params = NULL;
err_code = sd_ble_gap_sec_params_reply(m_conn_handle,
BLE_GAP_SEC_STATUS_PAIRING_NOT_SUPP,
p_sec_params,
p_keys);
APP_ERROR_CHECK(err_code);
} break;
case BLE_GAP_EVT_CONN_PARAM_UPDATE:
{
NRF_LOG_DEBUG("Received: BLE_GAP_EVT_CONN_PARAM_UPDATE");
NRF_LOG_DEBUG("conn_sup_timeout: %d\r\nmax_conn_interval: %d\r\nmin_conn_interval: %d\r\nslave_latency %d",
p_ble_evt->evt.gap_evt.params.conn_param_update.conn_params.conn_sup_timeout,
p_ble_evt->evt.gap_evt.params.conn_param_update.conn_params.max_conn_interval,
p_ble_evt->evt.gap_evt.params.conn_param_update.conn_params.min_conn_interval,
p_ble_evt->evt.gap_evt.params.conn_param_update.conn_params.slave_latency);
} break;
case BLE_GAP_EVT_CONN_PARAM_UPDATE_REQUEST:
{
NRF_LOG_DEBUG("Received: BLE_GAP_EVT_CONN_PARAM_UPDATE_REQUEST");
err_code = sd_ble_gap_conn_param_update(m_conn_handle, &p_ble_evt->evt.gap_evt.params.conn_param_update_request.conn_params);
if(err_code != NRF_SUCCESS)
{
NRF_LOG_ERROR("Failure to update connection parameter request: 0x%x", err_code);
}
APP_ERROR_CHECK(err_code);
} break;
#ifndef S140
case BLE_GAP_EVT_PHY_UPDATE_REQUEST:
{
NRF_LOG_DEBUG("PHY update request.");
ble_gap_phys_t const phys =
{
.rx_phys = BLE_GAP_PHY_AUTO,
.tx_phys = BLE_GAP_PHY_AUTO,
};
err_code = sd_ble_gap_phy_update(p_ble_evt->evt.gap_evt.conn_handle, &phys);
APP_ERROR_CHECK(err_code);
} break;
#endif
case BLE_GATTS_EVT_TIMEOUT:
{
if (p_ble_evt->evt.gatts_evt.params.timeout.src == BLE_GATT_TIMEOUT_SRC_PROTOCOL)
{
err_code = sd_ble_gap_disconnect(m_conn_handle,
BLE_HCI_REMOTE_USER_TERMINATED_CONNECTION);
APP_ERROR_CHECK(err_code);
}
} break;
case BLE_EVT_USER_MEM_REQUEST:
{
err_code = sd_ble_user_mem_reply(m_conn_handle, NULL);
APP_ERROR_CHECK(err_code);
} break;
case BLE_GATTS_EVT_RW_AUTHORIZE_REQUEST:
{
if (p_ble_evt->evt.gatts_evt.params.authorize_request.type
!= BLE_GATTS_AUTHORIZE_TYPE_INVALID)
{
if (on_rw_authorize_req(&m_dfu, p_ble_evt))
{
err_code = on_ctrl_pt_write(&m_dfu,
&(p_ble_evt->evt.gatts_evt.params.authorize_request.request.write));
#ifdef NRF_DFU_DEBUG_VERSION
if (err_code != NRF_SUCCESS)
{
NRF_LOG_ERROR("Could not handle on_ctrl_pt_write. err_code: 0x%04x", err_code);
}
#else
// Swallow result
(void) err_code;
#endif
}
}
} break;
case BLE_GAP_EVT_SEC_INFO_REQUEST:
{
NRF_LOG_DEBUG("== conn sec update request");
ble_gap_enc_info_t * p_enc_info = NULL;
ble_gap_irk_t * p_id_info = NULL;
#if defined(NRF_DFU_BLE_REQUIRES_BONDS) && (NRF_DFU_BLE_REQUIRES_BONDS == 1)
// If there is a match in diversifier, then set the correct keys.
if (p_ble_evt->evt.gap_evt.params.sec_info_request.master_id.ediv ==
m_peer_data.enc_key.master_id.ediv)
{
p_enc_info = &m_peer_data.enc_key.enc_info;
}
p_id_info = &m_peer_data.ble_id.id_info;
#endif
err_code = sd_ble_gap_sec_info_reply(p_ble_evt->evt.gap_evt.conn_handle,
p_enc_info,
p_id_info,
NULL);
APP_ERROR_CHECK(err_code);
} break;
case BLE_GAP_EVT_CONN_SEC_UPDATE:
case BLE_GATTS_EVT_SYS_ATTR_MISSING:
{
#if defined(NRF_DFU_BLE_REQUIRES_BONDS) && (NRF_DFU_BLE_REQUIRES_BONDS == 1)
err_code = service_changed_send();
#else
err_code = sd_ble_gatts_sys_attr_set(p_ble_evt->evt.gap_evt.conn_handle, NULL, 0, 0);
#endif
APP_ERROR_CHECK(err_code);
NRF_LOG_DEBUG("== We are finished handling conn sec update");
} break;
case BLE_GATTS_EVT_WRITE:
{
on_write(&m_dfu, p_ble_evt);
} break;
#if (NRF_SD_BLE_API_VERSION >= 3)
case BLE_GATTS_EVT_EXCHANGE_MTU_REQUEST:
{
err_code = sd_ble_gatts_exchange_mtu_reply(p_ble_evt->evt.gatts_evt.conn_handle,
NRF_SDH_BLE_GATT_MAX_MTU_SIZE);
APP_ERROR_CHECK(err_code);
} break; // BLE_GATTS_EVT_EXCHANGE_MTU_REQUEST
case BLE_GAP_EVT_DATA_LENGTH_UPDATE_REQUEST:
{
ble_gap_data_length_params_t const dlp =
{
.max_rx_octets = BLE_GAP_DATA_LENGTH_AUTO,
.max_tx_octets = BLE_GAP_DATA_LENGTH_AUTO,
};
err_code = sd_ble_gap_data_length_update(p_ble_evt->evt.gatts_evt.conn_handle,
&dlp, NULL);
APP_ERROR_CHECK(err_code);
} break; // BLE_GAP_EVT_DATA_LENGTH_UPDATE_REQUEST
#endif
case BLE_GATTS_EVT_HVC:
on_hvc(&m_dfu, p_ble_evt);
break;
case BLE_GATTS_EVT_SC_CONFIRM:
NRF_LOG_DEBUG("Service Changed was handled");
break;
default:
// No implementation needed.
break;
}
}
/**@brief Function for the LEDs initialization.
*
* @details Initializes all LEDs used by this application.
*/
static void leds_init(void)
{
#ifdef ADVERTISING_LED_PIN_NO
nrf_gpio_cfg_output(ADVERTISING_LED_PIN_NO);
nrf_gpio_pin_set(ADVERTISING_LED_PIN_NO);
#endif
#ifdef CONNECTED_LED_PIN_NO
nrf_gpio_cfg_output(CONNECTED_LED_PIN_NO);
nrf_gpio_pin_set(CONNECTED_LED_PIN_NO);
#endif
}
#if !defined(NRF_DFU_BLE_REQUIRES_BONDS) || (NRF_DFU_BLE_REQUIRES_BONDS == 0)
static uint32_t gap_address_change(void)
{
uint32_t err_code;
ble_gap_addr_t addr;
#if (NRF_SD_BLE_API_VERSION < 3)
err_code = sd_ble_gap_address_get(&addr);
#else
err_code = sd_ble_gap_addr_get(&addr);
#endif
VERIFY_SUCCESS(err_code);
// Increase the BLE address by one when advertising openly.
addr.addr[0] += 1;
#if (NRF_SD_BLE_API_VERSION < 3)
err_code = sd_ble_gap_address_set(BLE_GAP_ADDR_CYCLE_MODE_NONE, &addr);
#else
err_code = sd_ble_gap_addr_set(&addr);
#endif
VERIFY_SUCCESS(err_code);
return NRF_SUCCESS;
}
#endif
/**@brief Function for GAP initialization.
*
* @details This function will set up all necessary GAP (Generic Access Profile) parameters of
* the device. It also sets the permissions and appearance.
*/
static uint32_t gap_params_init(void)
{
uint32_t err_code;
ble_gap_conn_params_t gap_conn_params = {0};
ble_gap_conn_sec_mode_t sec_mode;
uint8_t const * device_name;
uint32_t name_len;
BLE_GAP_CONN_SEC_MODE_SET_OPEN(&sec_mode);
#if !defined(NRF_DFU_BLE_REQUIRES_BONDS) || (NRF_DFU_BLE_REQUIRES_BONDS == 0)
err_code = gap_address_change();
VERIFY_SUCCESS(err_code);
if ((m_flags & DFU_BLE_FLAG_USE_ADV_NAME) != 0)
{
NRF_LOG_DEBUG("Setting adv name: %s, length: %d", (uint32_t)m_adv_name.name, m_adv_name.len);
device_name = m_adv_name.name;
name_len = m_adv_name.len;
}
else
#endif
{
NRF_LOG_DEBUG("Regular adv name");
device_name = (uint8_t const *)DEVICE_NAME;
name_len = strlen(DEVICE_NAME);
}
err_code = sd_ble_gap_device_name_set(&sec_mode, device_name, name_len);
VERIFY_SUCCESS(err_code);
gap_conn_params.min_conn_interval = MIN_CONN_INTERVAL;
gap_conn_params.max_conn_interval = MAX_CONN_INTERVAL;
gap_conn_params.slave_latency = SLAVE_LATENCY;
gap_conn_params.conn_sup_timeout = CONN_SUP_TIMEOUT;
err_code = sd_ble_gap_ppcp_set(&gap_conn_params);
return err_code;
}
static uint32_t ble_stack_init(bool init_softdevice)
{
uint32_t err_code;
uint32_t ram_start = 0;
ble_cfg_t ble_cfg = {{0}};
#if defined(NRF_DFU_BLE_REQUIRES_BONDS) && (NRF_DFU_BLE_REQUIRES_BONDS == 1)
ble_cfg_t ble_gatts_cfg_service_changed = {{0}};
#endif
if (init_softdevice)
{
err_code = nrf_dfu_mbr_init_sd();
VERIFY_SUCCESS(err_code);
}
NRF_LOG_DEBUG("vector table: 0x%08x", BOOTLOADER_START_ADDR);
err_code = sd_softdevice_vector_table_base_set(BOOTLOADER_START_ADDR);
VERIFY_SUCCESS(err_code);
NRF_LOG_DEBUG("vector table: 0x%08x", BOOTLOADER_START_ADDR);
NRF_LOG_DEBUG("Error code - sd_softdevice_vector_table_base_set: 0x%08x", err_code);
err_code = nrf_sdh_enable_request();
APP_ERROR_CHECK(err_code);
// Fetch the start address of the application RAM.
err_code = nrf_sdh_ble_app_ram_start_get(&ram_start);
VERIFY_SUCCESS(err_code);
// Configure the maximum number of connections.
memset(&ble_cfg, 0, sizeof(ble_cfg));
ble_cfg.gap_cfg.role_count_cfg.periph_role_count = 1;
ble_cfg.gap_cfg.role_count_cfg.central_role_count = 0;
ble_cfg.gap_cfg.role_count_cfg.central_sec_count = 0;
err_code = sd_ble_cfg_set(BLE_GAP_CFG_ROLE_COUNT, &ble_cfg, ram_start);
memset(&ble_cfg, 0x00, sizeof(ble_cfg));
ble_cfg.conn_cfg.conn_cfg_tag = APP_BLE_CONN_CFG_TAG;
ble_cfg.conn_cfg.params.gatt_conn_cfg.att_mtu = NRF_SDH_BLE_GATT_MAX_MTU_SIZE;
err_code = sd_ble_cfg_set(BLE_CONN_CFG_GATT, &ble_cfg, ram_start);
if (err_code != NRF_SUCCESS)
{
NRF_LOG_ERROR("Error code - sd_ble_cfg_set: 0x%08x", err_code);
return err_code;
}
#if defined(NRF_DFU_BLE_REQUIRES_BONDS) && (NRF_DFU_BLE_REQUIRES_BONDS == 1)
NRF_LOG_DEBUG("Running Service Changed config");
ble_gatts_cfg_service_changed.gatts_cfg.service_changed.service_changed = 1;
err_code = sd_ble_cfg_set(BLE_GATTS_CFG_SERVICE_CHANGED, &ble_gatts_cfg_service_changed, ram_start);
VERIFY_SUCCESS(err_code);
NRF_LOG_DEBUG("Finished running Service Changed config");
#endif
NRF_LOG_DEBUG("Enabling SoftDevice.");
// Enable BLE stack.
err_code = nrf_sdh_ble_enable(&ram_start);
if (err_code != NRF_SUCCESS)
{
NRF_LOG_ERROR("Failed softdevice_enable: 0x%08x", err_code);
}
else
{
NRF_LOG_DEBUG("SoftDevice enabled.");
}
NRF_SDH_BLE_OBSERVER(m_ble_evt_observer, BLE_OBSERVER_PRIO, ble_evt_handler, NULL);
return err_code;
}
/**@brief Function for adding DFU Packet characteristic to the BLE Stack.
*
* @param[in] p_dfu DFU Service structure.
*
* @return NRF_SUCCESS on success. Otherwise an error code.
*/
static uint32_t dfu_pkt_char_add(ble_dfu_t * const p_dfu)
{
ble_gatts_char_md_t char_md = {{0}};
ble_gatts_attr_t attr_char_value = {0};
ble_gatts_attr_md_t attr_md = {{0}};
ble_uuid_t char_uuid;
char_md.char_props.write_wo_resp = 1;
char_uuid.type = p_dfu->uuid_type;
char_uuid.uuid = BLE_DFU_PKT_CHAR_UUID;
BLE_GAP_CONN_SEC_MODE_SET_NO_ACCESS(&attr_md.read_perm);
#if defined(NRF_DFU_BLE_REQUIRES_BONDS) && (NRF_DFU_BLE_REQUIRES_BONDS == 1)
BLE_GAP_CONN_SEC_MODE_SET_ENC_NO_MITM(&attr_md.write_perm);
#else
BLE_GAP_CONN_SEC_MODE_SET_OPEN(&attr_md.write_perm);
#endif
attr_md.vloc = BLE_GATTS_VLOC_STACK;
attr_md.vlen = 1;
attr_char_value.p_uuid = &char_uuid;
attr_char_value.p_attr_md = &attr_md;
attr_char_value.max_len = MAX_DFU_PKT_LEN;
attr_char_value.p_value = NULL;
return sd_ble_gatts_characteristic_add(p_dfu->service_handle,
&char_md,
&attr_char_value,
&p_dfu->dfu_pkt_handles);
}
/**@brief Function for adding DFU Control Point characteristic to the BLE Stack.
*
* @param[in] p_dfu DFU Service structure.
*
* @return NRF_SUCCESS on success. Otherwise an error code.
*/
static uint32_t dfu_ctrl_pt_add(ble_dfu_t * const p_dfu)
{
ble_gatts_char_md_t char_md = {{0}};
ble_gatts_attr_t attr_char_value = {0};
ble_gatts_attr_md_t attr_md = {{0}};
ble_uuid_t char_uuid;
char_md.char_props.write = 1;
char_md.char_props.notify = 1;
char_uuid.type = p_dfu->uuid_type;
char_uuid.uuid = BLE_DFU_CTRL_PT_UUID;
BLE_GAP_CONN_SEC_MODE_SET_NO_ACCESS(&attr_md.read_perm);
#if defined(NRF_DFU_BLE_REQUIRES_BONDS) && (NRF_DFU_BLE_REQUIRES_BONDS == 1)
BLE_GAP_CONN_SEC_MODE_SET_ENC_NO_MITM(&attr_md.write_perm);
#else
BLE_GAP_CONN_SEC_MODE_SET_OPEN(&attr_md.write_perm);
#endif
attr_md.vloc = BLE_GATTS_VLOC_STACK;
attr_md.wr_auth = 1;
attr_md.vlen = 1;
attr_char_value.p_uuid = &char_uuid;
attr_char_value.p_attr_md = &attr_md;
attr_char_value.max_len = BLE_GATT_ATT_MTU_DEFAULT;
attr_char_value.p_value = NULL;
return sd_ble_gatts_characteristic_add(p_dfu->service_handle,
&char_md,
&attr_char_value,
&p_dfu->dfu_ctrl_pt_handles);
}
/**@brief Function for checking if the CCCD of DFU Control point is configured for Notification.
*
* @details This function checks if the CCCD of DFU Control Point characteristic is configured
* for Notification by the DFU Controller.
*
* @param[in] p_dfu DFU Service structure.
*
* @return True if the CCCD of DFU Control Point characteristic is configured for Notification.
* False otherwise.
*/
uint32_t ble_dfu_init(ble_dfu_t * p_dfu)
{
ble_uuid_t service_uuid;
uint32_t err_code;
ASSERT(p_dfu != NULL);
m_conn_handle = BLE_CONN_HANDLE_INVALID;
BLE_UUID_BLE_ASSIGN(service_uuid, BLE_DFU_SERVICE_UUID);
err_code = sd_ble_gatts_service_add(BLE_GATTS_SRVC_TYPE_PRIMARY,
&service_uuid,
&(p_dfu->service_handle));
VERIFY_SUCCESS(err_code);
ble_uuid128_t const base_uuid128 =
{
{
0x50, 0xEA, 0xDA, 0x30, 0x88, 0x83, 0xB8, 0x9F,
0x60, 0x4F, 0x15, 0xF3, 0x00, 0x00, 0xC9, 0x8E
}
};
err_code = sd_ble_uuid_vs_add(&base_uuid128, &p_dfu->uuid_type);
VERIFY_SUCCESS(err_code);
err_code = dfu_pkt_char_add(p_dfu);
VERIFY_SUCCESS(err_code);
err_code = dfu_ctrl_pt_add(p_dfu);
VERIFY_SUCCESS(err_code);
m_flags |= DFU_BLE_FLAG_SERVICE_INITIALIZED;
return NRF_SUCCESS;
}
uint32_t ble_dfu_transport_init(void)
{
uint32_t err_code;
NRF_LOG_DEBUG("Initializing BLE DFU transport");
leds_init();
#if defined(NRF_DFU_BLE_REQUIRES_BONDS) && (NRF_DFU_BLE_REQUIRES_BONDS == 1)
// Copy out the peer data if bonds are required
if (nrf_dfu_settings_peer_data_is_valid())
{
NRF_LOG_DEBUG("Copying peer data");
(void)nrf_dfu_settings_peer_data_copy(&m_peer_data);
}
else
{
APP_ERROR_HANDLER(NRF_ERROR_INTERNAL);
}
#endif
err_code = ble_stack_init(true);
VERIFY_SUCCESS(err_code);
#if !defined(NRF_DFU_BLE_REQUIRES_BONDS) || (NRF_DFU_BLE_REQUIRES_BONDS == 0)
// Copy out the new advertisement name when bonds are not required and name is set.
if (nrf_dfu_settings_adv_name_is_valid())
{
(void)nrf_dfu_settings_adv_name_copy(&m_adv_name);
// Set flags for advertisement name that is to be used
m_flags |= DFU_BLE_FLAG_USE_ADV_NAME;
NRF_LOG_DEBUG("nrf_dfu_settings_adv_name_is_valid TRUE");
}
else
{
NRF_LOG_DEBUG("nrf_dfu_settings_adv_name_is_valid FALSE");
}
#endif
err_code = gap_params_init();
VERIFY_SUCCESS(err_code);
// Initialize the Device Firmware Update Service.
err_code = ble_dfu_init(&m_dfu);
VERIFY_SUCCESS(err_code);
err_code = advertising_start();
VERIFY_SUCCESS(err_code);
NRF_LOG_DEBUG("Finished initializing BLE DFU transport");
return NRF_SUCCESS;
}
uint32_t ble_dfu_transport_close(void)
{
uint32_t err_code = NRF_SUCCESS;
NRF_LOG_DEBUG("Shutting down BLE transport.");
if (m_conn_handle != BLE_CONN_HANDLE_INVALID)
{
NRF_LOG_DEBUG("Disconnecting.");
// Set flag to prevent advertisement from starting
m_flags |= DFU_BLE_RESETTING_SOON;
// Disconnect from peer.
err_code = sd_ble_gap_disconnect(m_conn_handle, BLE_HCI_REMOTE_USER_TERMINATED_CONNECTION);
VERIFY_SUCCESS(err_code);
// Wait a bit for the disconnect event to be sent on air.
nrf_delay_ms(200);
}
else
{
advertising_stop();
}
NRF_LOG_DEBUG("BLE transport shut down.");
return err_code;
}
Hope you can include this change into next SDK release.
Thank you.