nordic

Fork of nRF51822 by Nordic Semiconductor

Committer:
rgrover1
Date:
Thu Nov 26 15:02:39 2015 +0000
Revision:
480:f6e4a3f0a09a
Parent:
479:b5b54bf15533
Child:
483:0efda47cc288
Synchronized with git rev 167ddd63
Author: Andres Amaya Garcia
Execute radio notification in low priority context

The ble-nrf51822 implementation of the BLE API executes callbacks for radio
notification events at very high priority. This functionality is replaced by
executing the radio notification callback at a lower priority. When using
mbed OS the callback is posted through minar. In mbed classic the callback is
executed directly, but from a lower priority. Note that minar or callback
execution in mbed classic could not be posted/generated directly from the
radio notification handler because this causes race conditions that may lead
to a hard-fault. Alternatively, a Timeout was used to post the callback in
another context with lower priority.

Who changed what in which revision?

UserRevisionLine numberNew contents of line
Rohit Grover 22:c6ee8136847e 1 /* mbed Microcontroller Library
Rohit Grover 22:c6ee8136847e 2 * Copyright (c) 2006-2013 ARM Limited
Rohit Grover 22:c6ee8136847e 3 *
Rohit Grover 22:c6ee8136847e 4 * Licensed under the Apache License, Version 2.0 (the "License");
Rohit Grover 22:c6ee8136847e 5 * you may not use this file except in compliance with the License.
Rohit Grover 22:c6ee8136847e 6 * You may obtain a copy of the License at
Rohit Grover 22:c6ee8136847e 7 *
Rohit Grover 22:c6ee8136847e 8 * http://www.apache.org/licenses/LICENSE-2.0
Rohit Grover 22:c6ee8136847e 9 *
Rohit Grover 22:c6ee8136847e 10 * Unless required by applicable law or agreed to in writing, software
Rohit Grover 22:c6ee8136847e 11 * distributed under the License is distributed on an "AS IS" BASIS,
Rohit Grover 22:c6ee8136847e 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Rohit Grover 22:c6ee8136847e 13 * See the License for the specific language governing permissions and
Rohit Grover 22:c6ee8136847e 14 * limitations under the License.
Rohit Grover 22:c6ee8136847e 15 */
Rohit Grover 52:120bd37b9d0d 16
Rohit Grover 22:c6ee8136847e 17 #include "common/common.h"
Rohit Grover 37:c29c330d942c 18 #include "nordic_common.h"
Rohit Grover 22:c6ee8136847e 19
Rohit Grover 22:c6ee8136847e 20 #include "btle.h"
Rohit Grover 22:c6ee8136847e 21
Rohit Grover 22:c6ee8136847e 22 #include "ble_flash.h"
Rohit Grover 22:c6ee8136847e 23 #include "ble_conn_params.h"
Rohit Grover 22:c6ee8136847e 24
Rohit Grover 22:c6ee8136847e 25 #include "btle_gap.h"
Rohit Grover 22:c6ee8136847e 26 #include "btle_advertising.h"
Rohit Grover 22:c6ee8136847e 27 #include "custom/custom_helper.h"
Rohit Grover 22:c6ee8136847e 28
rgrover1 372:8f7d2137727a 29 #include "ble/GapEvents.h"
rgrover1 394:0f7c5048efb3 30 #include "nRF5xGap.h"
rgrover1 394:0f7c5048efb3 31 #include "nRF5xGattServer.h"
rgrover1 394:0f7c5048efb3 32 #include "nRF5xSecurityManager.h"
rgrover1 372:8f7d2137727a 33
rgrover1 480:f6e4a3f0a09a 34 extern "C" {
rgrover1 480:f6e4a3f0a09a 35 #include "pstorage.h"
rgrover1 125:664d4257a9f8 36 #include "device_manager.h"
rgrover1 480:f6e4a3f0a09a 37 #include "softdevice_handler.h"
rgrover1 480:f6e4a3f0a09a 38 #include "ble_stack_handler_types.h"
rgrover1 480:f6e4a3f0a09a 39 }
Rohit Grover 22:c6ee8136847e 40
Rohit Grover 56:a1071b629aa3 41 #include "ble_hci.h"
rgrover1 372:8f7d2137727a 42 #include "btle_discovery.h"
Rohit Grover 56:a1071b629aa3 43
Rohit Grover 56:a1071b629aa3 44 extern "C" void assert_nrf_callback(uint16_t line_num, const uint8_t *p_file_name);
Rohit Grover 56:a1071b629aa3 45 void app_error_handler(uint32_t error_code, uint32_t line_num, const uint8_t *p_file_name);
Rohit Grover 22:c6ee8136847e 46
Rohit Grover 22:c6ee8136847e 47 static void btle_handler(ble_evt_t *p_ble_evt);
Rohit Grover 22:c6ee8136847e 48
Rohit Grover 22:c6ee8136847e 49 static void sys_evt_dispatch(uint32_t sys_evt)
Rohit Grover 22:c6ee8136847e 50 {
Rohit Grover 22:c6ee8136847e 51 pstorage_sys_event_handler(sys_evt);
Rohit Grover 22:c6ee8136847e 52 }
Rohit Grover 22:c6ee8136847e 53
rgrover1 432:95158d6d493f 54 /**
rgrover1 432:95158d6d493f 55 * This function is called in interrupt context to handle BLE events; i.e. pull
rgrover1 432:95158d6d493f 56 * system and user events out of the pending events-queue of the BLE stack. The
rgrover1 432:95158d6d493f 57 * BLE stack signals the availability of events by the triggering the SWI2
rgrover1 432:95158d6d493f 58 * interrupt, which forwards the handling to this function.
rgrover1 432:95158d6d493f 59 *
rgrover1 432:95158d6d493f 60 * The event processing loop is implemented in intern_softdevice_events_execute().
rgrover1 432:95158d6d493f 61 *
rgrover1 432:95158d6d493f 62 * In mbed OS, a callback for intern_softdevice_events_execute() is posted
rgrover1 432:95158d6d493f 63 * to the scheduler, which then executes in thread mode. In mbed-classic,
rgrover1 432:95158d6d493f 64 * event processing happens right-away in interrupt context (which is more
rgrover1 432:95158d6d493f 65 * risk-prone). In either case, the logic of event processing is identical.
rgrover1 432:95158d6d493f 66 */
rgrover1 432:95158d6d493f 67 static uint32_t eventHandler()
rgrover1 432:95158d6d493f 68 {
rgrover1 432:95158d6d493f 69 #ifdef YOTTA_CFG_MBED_OS
rgrover1 432:95158d6d493f 70 minar::Scheduler::postCallback(intern_softdevice_events_execute);
rgrover1 432:95158d6d493f 71 #else
rgrover1 432:95158d6d493f 72 intern_softdevice_events_execute();
rgrover1 432:95158d6d493f 73 #endif
rgrover1 432:95158d6d493f 74
rgrover1 432:95158d6d493f 75 return NRF_SUCCESS;
rgrover1 432:95158d6d493f 76 }
rgrover1 432:95158d6d493f 77
Rohit Grover 22:c6ee8136847e 78 error_t btle_init(void)
Rohit Grover 22:c6ee8136847e 79 {
rgrover1 431:db7edc9ad0bc 80 nrf_clock_lfclksrc_t clockSource;
rgrover1 405:fccbe137c4ac 81 if (NRF_CLOCK->LFCLKSRC & (CLOCK_LFCLKSRC_SRC_Xtal << CLOCK_LFCLKSRC_SRC_Pos)) {
rgrover1 431:db7edc9ad0bc 82 clockSource = NRF_CLOCK_LFCLKSRC_XTAL_20_PPM;
rgrover1 405:fccbe137c4ac 83 } else {
rgrover1 431:db7edc9ad0bc 84 clockSource = NRF_CLOCK_LFCLKSRC_RC_250_PPM_4000MS_CALIBRATION;
rgrover1 405:fccbe137c4ac 85 }
rgrover1 432:95158d6d493f 86 SOFTDEVICE_HANDLER_INIT(clockSource, eventHandler);
Rohit Grover 56:a1071b629aa3 87
Rohit Grover 56:a1071b629aa3 88 // Enable BLE stack
Rohit Grover 56:a1071b629aa3 89 /**
Rohit Grover 56:a1071b629aa3 90 * Using this call, the application can select whether to include the
Rohit Grover 56:a1071b629aa3 91 * Service Changed characteristic in the GATT Server. The default in all
Rohit Grover 56:a1071b629aa3 92 * previous releases has been to include the Service Changed characteristic,
Rohit Grover 56:a1071b629aa3 93 * but this affects how GATT clients behave. Specifically, it requires
Rohit Grover 56:a1071b629aa3 94 * clients to subscribe to this attribute and not to cache attribute handles
Rohit Grover 56:a1071b629aa3 95 * between connections unless the devices are bonded. If the application
Rohit Grover 56:a1071b629aa3 96 * does not need to change the structure of the GATT server attributes at
Rohit Grover 56:a1071b629aa3 97 * runtime this adds unnecessary complexity to the interaction with peer
Rohit Grover 56:a1071b629aa3 98 * clients. If the SoftDevice is enabled with the Service Changed
Rohit Grover 56:a1071b629aa3 99 * Characteristics turned off, then clients are allowed to cache attribute
Rohit Grover 56:a1071b629aa3 100 * handles making applications simpler on both sides.
Rohit Grover 56:a1071b629aa3 101 */
Rohit Grover 68:936d81c963fe 102 static const bool IS_SRVC_CHANGED_CHARACT_PRESENT = true;
Rohit Grover 56:a1071b629aa3 103 ble_enable_params_t enableParams = {
Rohit Grover 56:a1071b629aa3 104 .gatts_enable_params = {
Rohit Grover 56:a1071b629aa3 105 .service_changed = IS_SRVC_CHANGED_CHARACT_PRESENT
Rohit Grover 56:a1071b629aa3 106 }
Rohit Grover 56:a1071b629aa3 107 };
Rohit Grover 56:a1071b629aa3 108 if (sd_ble_enable(&enableParams) != NRF_SUCCESS) {
Rohit Grover 56:a1071b629aa3 109 return ERROR_INVALID_PARAM;
Rohit Grover 56:a1071b629aa3 110 }
Rohit Grover 56:a1071b629aa3 111
Rohit Grover 56:a1071b629aa3 112 ble_gap_addr_t addr;
Rohit Grover 56:a1071b629aa3 113 if (sd_ble_gap_address_get(&addr) != NRF_SUCCESS) {
Rohit Grover 56:a1071b629aa3 114 return ERROR_INVALID_PARAM;
Rohit Grover 56:a1071b629aa3 115 }
Rohit Grover 56:a1071b629aa3 116 if (sd_ble_gap_address_set(BLE_GAP_ADDR_CYCLE_MODE_NONE, &addr) != NRF_SUCCESS) {
Rohit Grover 56:a1071b629aa3 117 return ERROR_INVALID_PARAM;
Rohit Grover 56:a1071b629aa3 118 }
Rohit Grover 22:c6ee8136847e 119
Rohit Grover 22:c6ee8136847e 120 ASSERT_STATUS( softdevice_ble_evt_handler_set(btle_handler));
Rohit Grover 22:c6ee8136847e 121 ASSERT_STATUS( softdevice_sys_evt_handler_set(sys_evt_dispatch));
Rohit Grover 22:c6ee8136847e 122
rgrover1 128:bd1e1fe607e0 123 btle_gap_init();
rgrover1 128:bd1e1fe607e0 124
rgrover1 128:bd1e1fe607e0 125 return ERROR_NONE;
rgrover1 128:bd1e1fe607e0 126 }
rgrover1 128:bd1e1fe607e0 127
Rohit Grover 22:c6ee8136847e 128 static void btle_handler(ble_evt_t *p_ble_evt)
Rohit Grover 22:c6ee8136847e 129 {
Rohit Grover 22:c6ee8136847e 130 /* Library service handlers */
Rohit Grover 56:a1071b629aa3 131 #if SDK_CONN_PARAMS_MODULE_ENABLE
Rohit Grover 22:c6ee8136847e 132 ble_conn_params_on_ble_evt(p_ble_evt);
Rohit Grover 56:a1071b629aa3 133 #endif
Rohit Grover 22:c6ee8136847e 134
rgrover1 126:35e4f65364bc 135 dm_ble_evt_handler(p_ble_evt);
rgrover1 126:35e4f65364bc 136
rgrover1 456:e33de7c4574c 137 #if !defined(TARGET_MCU_NRF51_16K_S110) && !defined(TARGET_MCU_NRF51_32K_S110)
rgrover1 372:8f7d2137727a 138 bleGattcEventHandler(p_ble_evt);
rgrover1 431:db7edc9ad0bc 139 #endif
rgrover1 372:8f7d2137727a 140
Rohit Grover 22:c6ee8136847e 141 /* Custom event handler */
Rohit Grover 22:c6ee8136847e 142 switch (p_ble_evt->header.evt_id) {
Rohit Grover 41:6e66cd970659 143 case BLE_GAP_EVT_CONNECTED: {
Rohit Grover 41:6e66cd970659 144 Gap::Handle_t handle = p_ble_evt->evt.gap_evt.conn_handle;
rgrover1 456:e33de7c4574c 145 #if defined(TARGET_MCU_NRF51_16K_S110) || defined(TARGET_MCU_NRF51_32K_S110)
rgrover1 400:a471c3da4806 146 /* Only peripheral role is supported by S110 */
rgrover1 400:a471c3da4806 147 Gap::Role_t role = Gap::PERIPHERAL;
rgrover1 400:a471c3da4806 148 #else
rgrover1 400:a471c3da4806 149 Gap::Role_t role = static_cast<Gap::Role_t>(p_ble_evt->evt.gap_evt.params.connected.role);
rgrover1 400:a471c3da4806 150 #endif
rgrover1 394:0f7c5048efb3 151 nRF5xGap::getInstance().setConnectionHandle(handle);
Rohit Grover 56:a1071b629aa3 152 const Gap::ConnectionParams_t *params = reinterpret_cast<Gap::ConnectionParams_t *>(&(p_ble_evt->evt.gap_evt.params.connected.conn_params));
rgrover1 77:9886b2865631 153 const ble_gap_addr_t *peer = &p_ble_evt->evt.gap_evt.params.connected.peer_addr;
rgrover1 113:737b08b3b995 154 const ble_gap_addr_t *own = &p_ble_evt->evt.gap_evt.params.connected.own_addr;
rgrover1 394:0f7c5048efb3 155 nRF5xGap::getInstance().processConnectionEvent(handle,
rgrover1 400:a471c3da4806 156 role,
rgrover1 180:10e4c13360d9 157 static_cast<Gap::AddressType_t>(peer->addr_type), peer->addr,
rgrover1 180:10e4c13360d9 158 static_cast<Gap::AddressType_t>(own->addr_type), own->addr,
rgrover1 113:737b08b3b995 159 params);
Rohit Grover 22:c6ee8136847e 160 break;
Rohit Grover 41:6e66cd970659 161 }
Rohit Grover 22:c6ee8136847e 162
Rohit Grover 41:6e66cd970659 163 case BLE_GAP_EVT_DISCONNECTED: {
Rohit Grover 41:6e66cd970659 164 Gap::Handle_t handle = p_ble_evt->evt.gap_evt.conn_handle;
Rohit Grover 22:c6ee8136847e 165 // Since we are not in a connection and have not started advertising,
Rohit Grover 22:c6ee8136847e 166 // store bonds
rgrover1 394:0f7c5048efb3 167 nRF5xGap::getInstance().setConnectionHandle (BLE_CONN_HANDLE_INVALID);
Rohit Grover 56:a1071b629aa3 168
Rohit Grover 65:98215c4f3a25 169 Gap::DisconnectionReason_t reason;
Rohit Grover 65:98215c4f3a25 170 switch (p_ble_evt->evt.gap_evt.params.disconnected.reason) {
Rohit Grover 65:98215c4f3a25 171 case BLE_HCI_LOCAL_HOST_TERMINATED_CONNECTION:
Rohit Grover 65:98215c4f3a25 172 reason = Gap::LOCAL_HOST_TERMINATED_CONNECTION;
Rohit Grover 65:98215c4f3a25 173 break;
Rohit Grover 65:98215c4f3a25 174 case BLE_HCI_REMOTE_USER_TERMINATED_CONNECTION:
Rohit Grover 65:98215c4f3a25 175 reason = Gap::REMOTE_USER_TERMINATED_CONNECTION;
Rohit Grover 65:98215c4f3a25 176 break;
Rohit Grover 65:98215c4f3a25 177 case BLE_HCI_CONN_INTERVAL_UNACCEPTABLE:
Rohit Grover 65:98215c4f3a25 178 reason = Gap::CONN_INTERVAL_UNACCEPTABLE;
Rohit Grover 65:98215c4f3a25 179 break;
Rohit Grover 65:98215c4f3a25 180 default:
Rohit Grover 65:98215c4f3a25 181 /* Please refer to the underlying transport library for an
Rohit Grover 65:98215c4f3a25 182 * interpretion of this reason's value. */
Rohit Grover 65:98215c4f3a25 183 reason = static_cast<Gap::DisconnectionReason_t>(p_ble_evt->evt.gap_evt.params.disconnected.reason);
Rohit Grover 65:98215c4f3a25 184 break;
Rohit Grover 56:a1071b629aa3 185 }
rgrover1 394:0f7c5048efb3 186 nRF5xGap::getInstance().processDisconnectionEvent(handle, reason);
Rohit Grover 22:c6ee8136847e 187 break;
Rohit Grover 41:6e66cd970659 188 }
Rohit Grover 22:c6ee8136847e 189
rgrover1 162:6712855a0107 190 case BLE_GAP_EVT_PASSKEY_DISPLAY:
rgrover1 394:0f7c5048efb3 191 nRF5xSecurityManager::getInstance().processPasskeyDisplayEvent(p_ble_evt->evt.gap_evt.conn_handle, p_ble_evt->evt.gap_evt.params.passkey_display.passkey);
rgrover1 162:6712855a0107 192 break;
rgrover1 162:6712855a0107 193
Rohit Grover 22:c6ee8136847e 194 case BLE_GAP_EVT_TIMEOUT:
rgrover1 394:0f7c5048efb3 195 nRF5xGap::getInstance().processTimeoutEvent(static_cast<Gap::TimeoutSource_t>(p_ble_evt->evt.gap_evt.params.timeout.src));
Rohit Grover 22:c6ee8136847e 196 break;
Rohit Grover 22:c6ee8136847e 197
Rohit Grover 22:c6ee8136847e 198 case BLE_GATTC_EVT_TIMEOUT:
Rohit Grover 22:c6ee8136847e 199 case BLE_GATTS_EVT_TIMEOUT:
Rohit Grover 22:c6ee8136847e 200 // Disconnect on GATT Server and Client timeout events.
Rohit Grover 22:c6ee8136847e 201 // ASSERT_STATUS_RET_VOID (sd_ble_gap_disconnect(m_conn_handle,
Rohit Grover 22:c6ee8136847e 202 // BLE_HCI_REMOTE_USER_TERMINATED_CONNECTION));
Rohit Grover 22:c6ee8136847e 203 break;
Rohit Grover 22:c6ee8136847e 204
rgrover1 166:67fcad70785a 205 case BLE_GAP_EVT_ADV_REPORT: {
rgrover1 166:67fcad70785a 206 const ble_gap_evt_adv_report_t *advReport = &p_ble_evt->evt.gap_evt.params.adv_report;
rgrover1 394:0f7c5048efb3 207 nRF5xGap::getInstance().processAdvertisementReport(advReport->peer_addr.addr,
rgrover1 168:66ee4f876396 208 advReport->rssi,
rgrover1 168:66ee4f876396 209 advReport->scan_rsp,
rgrover1 372:8f7d2137727a 210 static_cast<GapAdvertisingParams::AdvertisingType_t>(advReport->type),
rgrover1 168:66ee4f876396 211 advReport->dlen,
rgrover1 168:66ee4f876396 212 advReport->data);
rgrover1 168:66ee4f876396 213 break;
rgrover1 166:67fcad70785a 214 }
rgrover1 166:67fcad70785a 215
Rohit Grover 22:c6ee8136847e 216 default:
Rohit Grover 22:c6ee8136847e 217 break;
Rohit Grover 22:c6ee8136847e 218 }
Rohit Grover 22:c6ee8136847e 219
rgrover1 394:0f7c5048efb3 220 nRF5xGattServer::getInstance().hwCallback(p_ble_evt);
Rohit Grover 22:c6ee8136847e 221 }
Rohit Grover 22:c6ee8136847e 222
rgrover1 161:7d04579fdb2a 223 /*! @brief Callback when an error occurs inside the SoftDevice */
Rohit Grover 22:c6ee8136847e 224 void assert_nrf_callback(uint16_t line_num, const uint8_t *p_file_name)
Rohit Grover 22:c6ee8136847e 225 {
Rohit Grover 22:c6ee8136847e 226 ASSERT(false, (void) 0);
Rohit Grover 22:c6ee8136847e 227 }
Rohit Grover 22:c6ee8136847e 228
Rohit Grover 22:c6ee8136847e 229 /*!
Rohit Grover 22:c6ee8136847e 230 @brief Handler for general errors above the SoftDevice layer.
Rohit Grover 22:c6ee8136847e 231 Typically we can' recover from this so we do a reset.
Rohit Grover 22:c6ee8136847e 232 */
rgrover1 161:7d04579fdb2a 233 void app_error_handler(uint32_t error_code, uint32_t line_num, const uint8_t *p_file_name)
Rohit Grover 22:c6ee8136847e 234 {
Rohit Grover 22:c6ee8136847e 235 ASSERT_STATUS_RET_VOID( error_code );
Rohit Grover 22:c6ee8136847e 236 NVIC_SystemReset();
rgrover1 77:9886b2865631 237 }