Important changes to repositories hosted on mbed.com
Mbed hosted mercurial repositories are deprecated and are due to be permanently deleted in July 2026.
To keep a copy of this software download the repository Zip archive or clone locally using Mercurial.
It is also possible to export all your personal repositories from the account settings page.
Dependencies: BLE_API BLE_HeartRate mbed nRF51822
Fork of BLE_HeartRate by
Revision 77:d0c01a3d966f, committed 2015-11-30
- Comitter:
- finneyj
- Date:
- Mon Nov 30 13:37:37 2015 +0000
- Parent:
- 76:8b36a7c45cd2
- Commit message:
- Repro of BLE disconnect bug under pure mbed
Changed in this revision
--- a/BLE_API.lib Thu Nov 26 14:35:42 2015 +0000 +++ b/BLE_API.lib Mon Nov 30 13:37:37 2015 +0000 @@ -1,1 +1,1 @@ -http://mbed.org/teams/Bluetooth-Low-Energy/code/BLE_API/#3cb4b0e44d8f +http://mbed.org/teams/Bluetooth-Low-Energy/code/BLE_API/#c6c50b28ccd2
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/BLE_HeartRate.lib Mon Nov 30 13:37:37 2015 +0000 @@ -0,0 +1,1 @@ +http://developer.mbed.org/teams/Bluetooth-Low-Energy/code/BLE_HeartRate/#3a7d497a3e03
--- /dev/null Thu Jan 01 00:00:00 1970 +0000
+++ b/MicroBitAccelerometerService.cpp Mon Nov 30 13:37:37 2015 +0000
@@ -0,0 +1,76 @@
+/**
+ * Class definition for the custom MicroBit Accelerometer Service.
+ * Provides a BLE service to remotely read the state of the accelerometer, and configure its behaviour.
+ */
+
+#include "mbed.h"
+#include "ble/BLE.h"
+#include "ble/UUID.h"
+#include "MicroBitAccelerometerService.h"
+
+GattAttribute::Handle_t accelerometerDataCharacteristicHandleGlobal;
+
+/**
+ * Constructor.
+ * Create a representation of the AccelerometerService
+ * @param _ble The instance of a BLE device that we're running on.
+ */
+MicroBitAccelerometerService::MicroBitAccelerometerService(BLEDevice &_ble) :
+ ble(_ble)
+{
+ // Create the data structures that represent each of our characteristics in Soft Device.
+ GattCharacteristic *accelerometerDataCharacteristic = new GattCharacteristic(MicroBitAccelerometerServiceDataUUID, (uint8_t *)accelerometerDataCharacteristicBuffer, 0,
+ sizeof(accelerometerDataCharacteristicBuffer), GattCharacteristic::BLE_GATT_CHAR_PROPERTIES_READ | GattCharacteristic::BLE_GATT_CHAR_PROPERTIES_NOTIFY);
+
+ GattCharacteristic *accelerometerPeriodCharacteristic = new GattCharacteristic(MicroBitAccelerometerServicePeriodUUID, (uint8_t *)&accelerometerPeriodCharacteristicBuffer, 0,
+ sizeof(accelerometerPeriodCharacteristicBuffer),
+ GattCharacteristic::BLE_GATT_CHAR_PROPERTIES_READ | GattCharacteristic::BLE_GATT_CHAR_PROPERTIES_WRITE);
+
+ // Initialise our characteristic values.
+ accelerometerDataCharacteristicBuffer[0] = 0;
+ accelerometerDataCharacteristicBuffer[1] = 0;
+ accelerometerDataCharacteristicBuffer[2] = 0;
+ accelerometerPeriodCharacteristicBuffer = 0;
+
+ // Set default security requirements
+ accelerometerDataCharacteristic->requireSecurity(SecurityManager::SECURITY_MODE_ENCRYPTION_WITH_MITM);
+ accelerometerPeriodCharacteristic->requireSecurity(SecurityManager::SECURITY_MODE_ENCRYPTION_WITH_MITM);
+
+ GattCharacteristic *characteristics[] = {accelerometerDataCharacteristic, accelerometerPeriodCharacteristic};
+ GattService service(MicroBitAccelerometerServiceUUID, characteristics, sizeof(characteristics) / sizeof(GattCharacteristic *));
+
+ ble.addService(service);
+
+ accelerometerDataCharacteristicHandle = accelerometerDataCharacteristic->getValueHandle();
+ accelerometerDataCharacteristicHandleGlobal = accelerometerDataCharacteristic->getValueHandle();
+ accelerometerPeriodCharacteristicHandle = accelerometerPeriodCharacteristic->getValueHandle();
+
+ ble.gattServer().write(accelerometerDataCharacteristicHandle,(uint8_t *)accelerometerDataCharacteristicBuffer, sizeof(accelerometerDataCharacteristicBuffer));
+ ble.gattServer().write(accelerometerPeriodCharacteristicHandle, (const uint8_t *)&accelerometerPeriodCharacteristicBuffer, sizeof(accelerometerPeriodCharacteristicBuffer));
+
+ ble.onDataWritten(this, &MicroBitAccelerometerService::onDataWritten);
+}
+
+/**
+ * Callback. Invoked when any of our attributes are written via BLE.
+ */
+void MicroBitAccelerometerService::onDataWritten(const GattWriteCallbackParams *params)
+{
+ if (params->handle == accelerometerPeriodCharacteristicHandle && params->len >= sizeof(accelerometerPeriodCharacteristicBuffer))
+ {
+ accelerometerPeriodCharacteristicBuffer = *((uint16_t *)params->data);
+ }
+}
+
+const uint8_t MicroBitAccelerometerServiceUUID[] = {
+ 0xe9,0x5d,0x07,0x53,0x25,0x1d,0x47,0x0a,0xa0,0x62,0xfa,0x19,0x22,0xdf,0xa9,0xa8
+};
+
+const uint8_t MicroBitAccelerometerServiceDataUUID[] = {
+ 0xe9,0x5d,0xca,0x4b,0x25,0x1d,0x47,0x0a,0xa0,0x62,0xfa,0x19,0x22,0xdf,0xa9,0xa8
+};
+
+const uint8_t MicroBitAccelerometerServicePeriodUUID[] = {
+ 0xe9,0x5d,0xfb,0x24,0x25,0x1d,0x47,0x0a,0xa0,0x62,0xfa,0x19,0x22,0xdf,0xa9,0xa8
+};
+
--- /dev/null Thu Jan 01 00:00:00 1970 +0000
+++ b/MicroBitAccelerometerService.h Mon Nov 30 13:37:37 2015 +0000
@@ -0,0 +1,48 @@
+#ifndef MICROBIT_ACCELEROMETER_SERVICE_H
+#define MICROBIT_ACCELEROMETER_SERVICE_H
+
+extern GattAttribute::Handle_t accelerometerDataCharacteristicHandleGlobal;
+
+// UUIDs for our service and characteristics
+extern const uint8_t MicroBitAccelerometerServiceUUID[];
+extern const uint8_t MicroBitAccelerometerServiceDataUUID[];
+extern const uint8_t MicroBitAccelerometerServicePeriodUUID[];
+
+
+/**
+ * Class definition for a MicroBit BLE Accelerometer Service.
+ * Provides access to live accelerometer data via BLE, and provides basic configuration options.
+ */
+class MicroBitAccelerometerService
+{
+ public:
+
+ /**
+ * Constructor.
+ * Create a representation of the AccelerometerService
+ * @param _ble The instance of a BLE device that we're running on.
+ */
+ MicroBitAccelerometerService(BLEDevice &_ble);
+
+
+ private:
+
+ /**
+ * Callback. Invoked when any of our attributes are written via BLE.
+ */
+ void onDataWritten(const GattWriteCallbackParams *params);
+
+ // Bluetooth stack we're running on.
+ BLEDevice &ble;
+
+ // memory for our 8 bit control characteristics.
+ uint16_t accelerometerDataCharacteristicBuffer[3];
+ uint16_t accelerometerPeriodCharacteristicBuffer;
+
+ // Handles to access each characteristic when they are held by Soft Device.
+ GattAttribute::Handle_t accelerometerDataCharacteristicHandle;
+ GattAttribute::Handle_t accelerometerPeriodCharacteristicHandle;
+};
+
+
+#endif
--- a/main.cpp Thu Nov 26 14:35:42 2015 +0000
+++ b/main.cpp Mon Nov 30 13:37:37 2015 +0000
@@ -16,21 +16,23 @@
#include "mbed.h"
#include "ble/BLE.h"
-#include "ble/services/HeartRateService.h"
-#include "ble/services/BatteryService.h"
#include "ble/services/DeviceInformationService.h"
-
-DigitalOut led1(LED1);
+#include "MicroBitAccelerometerService.h"
+#include "nrf_soc.h"
-const static char DEVICE_NAME[] = "HRM1";
-static const uint16_t uuid16_list[] = {GattService::UUID_HEART_RATE_SERVICE,
- GattService::UUID_DEVICE_INFORMATION_SERVICE};
-static volatile bool triggerSensorPolling = false;
+const char* DEVICE_NAME = "BBC micro:bit [zevug]";
+const char* MICROBIT_BLE_MANUFACTURER = "The Cast of W1A";
+const char* MICROBIT_BLE_MODEL = "BBC micro:bit";
+const char* MICROBIT_BLE_HARDWARE_VERSION = "1.0";
+const char* MICROBIT_BLE_FIRMWARE_VERSION = "BODGE";
+const char* MICROBIT_BLE_SOFTWARE_VERSION = "1.0";
-uint8_t hrmCounter = 100; // init HRM to 100bps
+#define MICROBIT_BLE_ENABLE_BONDING true
+#define MICROBIT_BLE_REQUIRE_MITM true
-HeartRateService *hrService;
+MicroBitAccelerometerService *accelService;
DeviceInformationService *deviceInfo;
+Serial pc(USBTX, USBRX);
void disconnectionCallback(const Gap::DisconnectionCallbackParams_t *params)
{
@@ -39,11 +41,23 @@
void periodicCallback(void)
{
- led1 = !led1; /* Do blinky on LED1 while we're waiting for BLE events */
+ pc.printf(".");
+ wait_ms(3);
+}
- /* Note that the periodicCallback() executes in interrupt context, so it is safer to do
- * heavy-weight sensor polling from the main thread. */
- triggerSensorPolling = true;
+static void passkeyDisplayCallback(Gap::Handle_t handle, const SecurityManager::Passkey_t passkey)
+{
+ pc.printf("\nPASSKEY: ");
+
+ for (int i = 0; i<SecurityManager::PASSKEY_LEN; i++)
+ pc.printf("%c", passkey[i]);
+
+ pc.printf("\n");
+}
+
+static void securitySetupCompletedCallback(Gap::Handle_t handle, SecurityManager::SecurityCompletionStatus_t status)
+{
+ pc.printf("paring complete\n");
}
void bleInitComplete(BLE::InitializationCompleteCallbackContext *params)
@@ -57,51 +71,98 @@
ble.gap().onDisconnection(disconnectionCallback);
- /* Setup primary service. */
- hrService = new HeartRateService(ble, hrmCounter, HeartRateService::LOCATION_FINGER);
+ // Setup our security requirements.
+ ble.securityManager().onPasskeyDisplay(passkeyDisplayCallback);
+ ble.securityManager().onSecuritySetupCompleted(securitySetupCompletedCallback);
+ ble.securityManager().init(MICROBIT_BLE_ENABLE_BONDING, MICROBIT_BLE_REQUIRE_MITM, SecurityManager::IO_CAPS_DISPLAY_ONLY);
/* Setup auxiliary service. */
- deviceInfo = new DeviceInformationService(ble, "ARM", "Model1", "SN1", "hw-rev1", "fw-rev1", "soft-rev1");
+ deviceInfo = new DeviceInformationService (ble, MICROBIT_BLE_MANUFACTURER, MICROBIT_BLE_MODEL, "SN1", MICROBIT_BLE_HARDWARE_VERSION, MICROBIT_BLE_FIRMWARE_VERSION, MICROBIT_BLE_SOFTWARE_VERSION);
+
+ /* Setup primary service. */
+ accelService = new MicroBitAccelerometerService(ble);
+
+ // Configure for high speed mode where possible.
+ Gap::ConnectionParams_t fast;
+ ble.getPreferredConnectionParams(&fast);
+ fast.minConnectionInterval = 8; // 10 ms
+ fast.maxConnectionInterval = 16; // 20 ms
+ fast.slaveLatency = 0;
+ ble.setPreferredConnectionParams(&fast);
/* Setup advertising. */
- ble.gap().accumulateAdvertisingPayload(GapAdvertisingData::BREDR_NOT_SUPPORTED | GapAdvertisingData::LE_GENERAL_DISCOVERABLE);
- ble.gap().accumulateAdvertisingPayload(GapAdvertisingData::COMPLETE_LIST_16BIT_SERVICE_IDS, (uint8_t *)uuid16_list, sizeof(uuid16_list));
- ble.gap().accumulateAdvertisingPayload(GapAdvertisingData::GENERIC_HEART_RATE_SENSOR);
- ble.gap().accumulateAdvertisingPayload(GapAdvertisingData::COMPLETE_LOCAL_NAME, (uint8_t *)DEVICE_NAME, sizeof(DEVICE_NAME));
- ble.gap().setAdvertisingType(GapAdvertisingParams::ADV_CONNECTABLE_UNDIRECTED);
- ble.gap().setAdvertisingInterval(1000); /* 1000ms */
- ble.gap().startAdvertising();
+ //ble.gap().accumulateAdvertisingPayload(GapAdvertisingData::BREDR_NOT_SUPPORTED | GapAdvertisingData::LE_GENERAL_DISCOVERABLE);
+ //ble.gap().accumulateAdvertisingPayload(GapAdvertisingData::COMPLETE_LIST_16BIT_SERVICE_IDS, (uint8_t *)uuid16_list, sizeof(uuid16_list));
+ //ble.gap().accumulateAdvertisingPayload(GapAdvertisingData::GENERIC_HEART_RATE_SENSOR);
+ //ble.gap().accumulateAdvertisingPayload(GapAdvertisingData::COMPLETE_LOCAL_NAME, (uint8_t *)DEVICE_NAME, sizeof(DEVICE_NAME));
+ //ble.gap().setAdvertisingType(GapAdvertisingParams::ADV_CONNECTABLE_UNDIRECTED);
+ //ble.gap().setAdvertisingInterval(200); /* 1000ms */
+ //ble.gap().startAdvertising();
+
+ // Setup advertising.
+ ble.accumulateAdvertisingPayload(GapAdvertisingData::BREDR_NOT_SUPPORTED | GapAdvertisingData::LE_GENERAL_DISCOVERABLE);
+ ble.accumulateAdvertisingPayload(GapAdvertisingData::COMPLETE_LOCAL_NAME, (uint8_t *)DEVICE_NAME, strlen(DEVICE_NAME));
+ ble.setAdvertisingType(GapAdvertisingParams::ADV_CONNECTABLE_UNDIRECTED);
+ ble.setAdvertisingInterval(40);
+ ble.startAdvertising();
}
int main(void)
{
- led1 = 1;
Ticker ticker;
- ticker.attach(periodicCallback, 1); // blink LED every second
+
+ pc.baud(115200);
+
+ ticker.attach(periodicCallback, 0.006); // 6ms
BLE& ble = BLE::Instance(BLE::DEFAULT_INSTANCE);
ble.init(bleInitComplete);
/* SpinWait for initialization to complete. This is necessary because the
* BLE object is used in the main loop below. */
- while (ble.hasInitialized() == false) { /* spin loop */ }
+ while (ble.hasInitialized() == false) { /* spin loop */ }
+
+ // TEST
+
+ uint16_t i=0;
+ uint16_t accelerometerDataCharacteristicBuffer[3];
+
+ // Disable all the contents of the ISR.
+ //uBit.systemTicker.detach();
- // infinite loop
- while (1) {
- // check for trigger from periodicCallback()
- if (triggerSensorPolling && ble.getGapState().connected) {
- triggerSensorPolling = false;
+ for (int irq = 0; irq < 23; irq++)
+ sd_nvic_SetPriority((IRQn_Type)irq, 3);
+
+ while(1)
+ {
+ /*
+ if (i % 500 == 0)
+ {
+ int sp = (int) __get_MSP();
+ uBit.serial.printf("MSP: 0x%.8x\n", sp);
- // Do blocking calls or whatever is necessary for sensor polling.
- // In our case, we simply update the HRM measurement.
- hrmCounter++;
- if (hrmCounter == 175) { // 100 <= HRM bps <=175
- hrmCounter = 100;
+ uint32_t pm = __get_PRIMASK();
+ uBit.serial.printf("PG: 0x%.8x\n", pm);
+
+ nrf_app_irq_priority_t pri;
+
+ for (int irq = 0; irq <= 25; irq++)
+ {
+ sd_nvic_GetPriority((IRQn_Type)irq, &pri);
+ uBit.serial.printf("irq: %2d priority: %d\n", irq, pri);
}
- hrService->updateHeartRate(hrmCounter);
- } else {
- ble.waitForEvent(); // low power wait for event
+ }
+ */
+
+ wait_ms(20);
+ accelerometerDataCharacteristicBuffer[0] = i;
+ i++;
+
+ if (ble.getGapState().connected)
+ {
+ ble.gattServer().write(accelerometerDataCharacteristicHandleGlobal,(uint8_t *)accelerometerDataCharacteristicBuffer, sizeof(accelerometerDataCharacteristicBuffer));
}
}
}
+
--- a/mbed.bld Thu Nov 26 14:35:42 2015 +0000 +++ b/mbed.bld Mon Nov 30 13:37:37 2015 +0000 @@ -1,1 +1,1 @@ -http://mbed.org/users/mbed_official/code/mbed/builds/9296ab0bfc11 \ No newline at end of file +http://mbed.org/users/mbed_official/code/mbed/builds/165afa46840b \ No newline at end of file
--- a/nRF51822.lib Thu Nov 26 14:35:42 2015 +0000 +++ b/nRF51822.lib Mon Nov 30 13:37:37 2015 +0000 @@ -1,1 +1,1 @@ -http://mbed.org/teams/Nordic-Semiconductor/code/nRF51822/#5de80a383c85 +http://mbed.org/teams/Nordic-Semiconductor/code/nRF51822/#1762975cb3cb
