A library for easier setup and prototyping of IoT devices (pucks), by collecting everything that is common for all pucks in one place.

Dependencies:   BLE_API nRF51822

Dependents:   ir-puck display-puck ir-puck2 BLE_ScoringDevice ... more

/media/uploads/stiaje/header.jpg

Introduction

Raspberry Pi took the maker community by storm when it launched in 2012. With its internet access it allowed small projects to be internet-of-things enabled. We have created a platform to take this one step further.

Our platform, called the Puck platform, is an internet of things platform for mbed. mbed makes it easy to program embedded hardware for people new to embedded systems. Our platform is built upon the first mbed chip with Bluetooth, the nRF51822 created by Nordic Semiconductor. We hope to create a community around these BLE devices where people contribute to the project, and share their designs with each other. Everything is open-source, of course, with lots of supporting materials.

We make it easy to rapidly prototype and develop Bluetooth LE enabled devices - get up and running in under 10 lines of code.

Tutorials and in-depth documentation is available at the project's GitHub page

Pucks

We've developed a handful of awesome examples to demonstrate the platform. These examples are named 'Pucks'. By talking to the internet through your smartphone, the barrier to creating your own Internet of Things device is lower than ever.

Committer:
cristea
Date:
Thu Jul 24 09:25:42 2014 +0000
Revision:
7:c07c01c2a741
Parent:
6:211ffef3b88e
Child:
8:49ffd38fb401
Child:
9:ba0527c6b6d0
Hoist logging out of Puck.h; ; There was a problem with multiple definitions of functions in Puck.h if; used in two different files. The logger has been moved out so that we; have independent loggers for each file, which can have different verbosity.

Who changed what in which revision?

UserRevisionLine numberNew contents of line
sigveseb 3:5432b38585ea 1 #ifndef __PUCK_HPP__
sigveseb 3:5432b38585ea 2 #define __PUCK_HPP__
sigveseb 3:5432b38585ea 3
sigveseb 3:5432b38585ea 4 #include "BLEDevice.h"
sigveseb 3:5432b38585ea 5 #include <vector>
cristea 7:c07c01c2a741 6 #include "Log.h"
sigveseb 3:5432b38585ea 7
sigveseb 3:5432b38585ea 8 enum PuckState {
sigveseb 3:5432b38585ea 9 CONNECTING,
sigveseb 3:5432b38585ea 10 CONNECTED,
sigveseb 3:5432b38585ea 11 ADVERTISING,
sigveseb 3:5432b38585ea 12 DISCONNECTED
sigveseb 3:5432b38585ea 13 };
sigveseb 3:5432b38585ea 14
sigveseb 3:5432b38585ea 15 const UUID stringToUUID(const char* str);
sigveseb 3:5432b38585ea 16
sigveseb 3:5432b38585ea 17 typedef void (*CharacteristicWriteCallback)(uint8_t* value);
sigveseb 3:5432b38585ea 18
sigveseb 3:5432b38585ea 19 typedef struct {
sigveseb 3:5432b38585ea 20 const UUID* uuid;
sigveseb 3:5432b38585ea 21 std::vector<CharacteristicWriteCallback>* callbacks;
sigveseb 3:5432b38585ea 22 } CharacteristicWriteCallbacks;
sigveseb 3:5432b38585ea 23
sigveseb 3:5432b38585ea 24
sigveseb 3:5432b38585ea 25 class Puck {
sigveseb 3:5432b38585ea 26 private:
sigveseb 3:5432b38585ea 27 Puck() {}
sigveseb 3:5432b38585ea 28 Puck(const Puck&);
sigveseb 3:5432b38585ea 29 Puck& operator=(const Puck&);
sigveseb 3:5432b38585ea 30
sigveseb 3:5432b38585ea 31 BLEDevice ble;
sigveseb 3:5432b38585ea 32 uint8_t beaconPayload[25];
sigveseb 3:5432b38585ea 33 PuckState state;
sigveseb 3:5432b38585ea 34 std::vector<GattService*> services;
sigveseb 3:5432b38585ea 35 std::vector<GattCharacteristic*> characteristics;
sigveseb 3:5432b38585ea 36 std::vector<CharacteristicWriteCallbacks*> writeCallbacks;
sigveseb 3:5432b38585ea 37 std::vector<CharacteristicWriteCallback> pendingCallbackStack;
sigveseb 3:5432b38585ea 38 std::vector<uint8_t*> pendingCallbackParameterStack;
sigveseb 3:5432b38585ea 39
stiaje 4:91506772210d 40 GattCharacteristic **previousCharacteristics;
stiaje 4:91506772210d 41
sigveseb 3:5432b38585ea 42 public:
sigveseb 3:5432b38585ea 43 static Puck &getPuck();
sigveseb 3:5432b38585ea 44
stiaje 5:2f2a2ac6b231 45 BLEDevice &getBle() { return ble; }
sigveseb 3:5432b38585ea 46 PuckState getState() { return state; }
sigveseb 3:5432b38585ea 47 void setState(PuckState state);
sigveseb 3:5432b38585ea 48 void init(uint16_t minor);
sigveseb 3:5432b38585ea 49 void startAdvertising();
sigveseb 3:5432b38585ea 50 void stopAdvertising();
sigveseb 3:5432b38585ea 51 bool drive();
sigveseb 3:5432b38585ea 52 void onDataWritten(uint16_t handle);
sigveseb 3:5432b38585ea 53 void addCharacteristic(const UUID serviceUuid, const UUID characteristicUuid, int bytes, int properties = 0xA);
sigveseb 3:5432b38585ea 54 void onCharacteristicWrite(const UUID uuid, CharacteristicWriteCallback callback);
sigveseb 3:5432b38585ea 55 uint8_t* getCharacteristicValue(const UUID uuid);
sigveseb 3:5432b38585ea 56 };
sigveseb 3:5432b38585ea 57
sigveseb 3:5432b38585ea 58 Puck &Puck::getPuck() {
sigveseb 3:5432b38585ea 59 static Puck _puckSingletonInstance;
sigveseb 3:5432b38585ea 60 return _puckSingletonInstance;
sigveseb 3:5432b38585ea 61 }
sigveseb 3:5432b38585ea 62
sigveseb 3:5432b38585ea 63
sigveseb 3:5432b38585ea 64 void onDisconnection(void) {
sigveseb 3:5432b38585ea 65 LOG_INFO("Disconnected.\n");
sigveseb 3:5432b38585ea 66 Puck::getPuck().setState(DISCONNECTED);
sigveseb 3:5432b38585ea 67 }
sigveseb 3:5432b38585ea 68
sigveseb 3:5432b38585ea 69 void onConnection(void) {
sigveseb 3:5432b38585ea 70 LOG_INFO("Connected.\n");
sigveseb 3:5432b38585ea 71 Puck::getPuck().setState(CONNECTED);
sigveseb 3:5432b38585ea 72 }
sigveseb 3:5432b38585ea 73
sigveseb 3:5432b38585ea 74 void onDataWrittenCallback(uint16_t handle) {
sigveseb 3:5432b38585ea 75 Puck::getPuck().onDataWritten(handle);
sigveseb 3:5432b38585ea 76 }
sigveseb 3:5432b38585ea 77
sigveseb 3:5432b38585ea 78 bool isEqualUUID(const UUID* uuidA, const UUID uuidB) {
sigveseb 3:5432b38585ea 79 const uint8_t* uuidABase = uuidA->getBaseUUID();
sigveseb 3:5432b38585ea 80 const uint8_t* uuidBBase = uuidB.getBaseUUID();
sigveseb 3:5432b38585ea 81 if(uuidA->getShortUUID() != uuidB.getShortUUID()) {
sigveseb 3:5432b38585ea 82 return false;
sigveseb 3:5432b38585ea 83 }
sigveseb 3:5432b38585ea 84 for(int i = 0; i < 16; i++) {
sigveseb 3:5432b38585ea 85 if(uuidABase[i] != uuidBBase[i]) {
sigveseb 3:5432b38585ea 86 return false;
sigveseb 3:5432b38585ea 87 }
sigveseb 3:5432b38585ea 88 }
sigveseb 3:5432b38585ea 89 return true;
sigveseb 3:5432b38585ea 90 }
sigveseb 3:5432b38585ea 91
sigveseb 3:5432b38585ea 92 const UUID stringToUUID(const char* str) {
sigveseb 3:5432b38585ea 93 uint8_t array[16];
sigveseb 3:5432b38585ea 94 for(int i = 0; i < 16; i++) {
sigveseb 3:5432b38585ea 95 array[i] = str[i];
sigveseb 3:5432b38585ea 96 }
sigveseb 3:5432b38585ea 97 return UUID(array);
sigveseb 3:5432b38585ea 98 }
sigveseb 3:5432b38585ea 99
sigveseb 3:5432b38585ea 100 void Puck::setState(PuckState state) {
sigveseb 3:5432b38585ea 101 LOG_DEBUG("Changed state to %i\n", state);
sigveseb 3:5432b38585ea 102 this->state = state;
sigveseb 3:5432b38585ea 103 }
sigveseb 3:5432b38585ea 104
sigveseb 3:5432b38585ea 105 void Puck::init(uint16_t minor) {
sigveseb 3:5432b38585ea 106 /*
sigveseb 3:5432b38585ea 107 * The Beacon payload (encapsulated within the MSD advertising data structure)
sigveseb 3:5432b38585ea 108 * has the following composition:
sigveseb 3:5432b38585ea 109 * 128-Bit UUID = E2 0A 39 F4 73 F5 4B C4 A1 2F 17 D1 AD 07 A9 61
sigveseb 3:5432b38585ea 110 * Major/Minor = 1337 / XXXX
sigveseb 3:5432b38585ea 111 * Tx Power = C8
sigveseb 3:5432b38585ea 112 */
sigveseb 3:5432b38585ea 113 uint8_t beaconPayloadTemplate[] = {
sigveseb 3:5432b38585ea 114 0x00, 0x00, // Company identifier code (0x004C == Apple)
sigveseb 3:5432b38585ea 115 0x02, // ID
sigveseb 3:5432b38585ea 116 0x15, // length of the remaining payload
sigveseb 3:5432b38585ea 117 0xE2, 0x0A, 0x39, 0xF4, 0x73, 0xF5, 0x4B, 0xC4, // UUID
sigveseb 3:5432b38585ea 118 0xA1, 0x2F, 0x17, 0xD1, 0xAD, 0x07, 0xA9, 0x61,
sigveseb 3:5432b38585ea 119 0x13, 0x37, // the major value to differenciate a location (Our app requires 1337 as major number)
sigveseb 3:5432b38585ea 120 0x00, 0x00, // the minor value to differenciate a location (Change this to differentiate location pucks)
sigveseb 3:5432b38585ea 121 0xC8 // 2's complement of the Tx power (-56dB)
sigveseb 3:5432b38585ea 122 };
sigveseb 3:5432b38585ea 123 beaconPayloadTemplate[22] = minor >> 8;
sigveseb 3:5432b38585ea 124 beaconPayloadTemplate[23] = minor & 255;
sigveseb 3:5432b38585ea 125
sigveseb 3:5432b38585ea 126 for (int i=0; i < 25; i++) {
sigveseb 3:5432b38585ea 127 beaconPayload[i] = beaconPayloadTemplate[i];
sigveseb 3:5432b38585ea 128 }
sigveseb 3:5432b38585ea 129
sigveseb 3:5432b38585ea 130 ble.init();
sigveseb 3:5432b38585ea 131 LOG_VERBOSE("Inited BLEDevice.\n");
sigveseb 3:5432b38585ea 132 setState(DISCONNECTED);
sigveseb 3:5432b38585ea 133
sigveseb 3:5432b38585ea 134 ble.accumulateAdvertisingPayload(GapAdvertisingData::BREDR_NOT_SUPPORTED);
sigveseb 3:5432b38585ea 135 LOG_VERBOSE("Accumulate advertising payload: BREDR_NOT_SUPPORTED.\n");
sigveseb 3:5432b38585ea 136
sigveseb 3:5432b38585ea 137 ble.accumulateAdvertisingPayload(GapAdvertisingData::MANUFACTURER_SPECIFIC_DATA, beaconPayload, sizeof(beaconPayload));
sigveseb 3:5432b38585ea 138 LOG_VERBOSE("Accumulate advertising payload: beacon data.\n");
sigveseb 3:5432b38585ea 139
sigveseb 3:5432b38585ea 140 ble.setAdvertisingType(GapAdvertisingParams::ADV_CONNECTABLE_UNDIRECTED);
sigveseb 3:5432b38585ea 141 LOG_VERBOSE("Setting advertising type: ADV_CONNECTABLE_UNDIRECTED.\n");
sigveseb 3:5432b38585ea 142
sigveseb 3:5432b38585ea 143 int hundredMillisecondsInAdvertisingIntervalFormat = 160;
sigveseb 3:5432b38585ea 144 ble.setAdvertisingInterval(hundredMillisecondsInAdvertisingIntervalFormat);
sigveseb 3:5432b38585ea 145 LOG_VERBOSE("Set advertising interval: 160 (100 ms).\n");
sigveseb 3:5432b38585ea 146
sigveseb 3:5432b38585ea 147 ble.onDisconnection(onDisconnection);
sigveseb 3:5432b38585ea 148 ble.onConnection(onConnection);
sigveseb 3:5432b38585ea 149 ble.onDataWritten(onDataWrittenCallback);
sigveseb 3:5432b38585ea 150
sigveseb 3:5432b38585ea 151 for(int i = 0; i < services.size(); i++) {
sigveseb 3:5432b38585ea 152 ble.addService(*services[i]);
sigveseb 3:5432b38585ea 153 LOG_VERBOSE("Added service %x to BLEDevice\n", services[i]);
sigveseb 3:5432b38585ea 154 }
sigveseb 3:5432b38585ea 155
sigveseb 3:5432b38585ea 156 LOG_INFO("Inited puck as 0x%X.\n", minor);
sigveseb 3:5432b38585ea 157 }
sigveseb 3:5432b38585ea 158
sigveseb 3:5432b38585ea 159 void Puck::startAdvertising() {
sigveseb 3:5432b38585ea 160 ble.startAdvertising();
sigveseb 3:5432b38585ea 161 LOG_INFO("Starting to advertise.\n");
sigveseb 3:5432b38585ea 162 setState(ADVERTISING);
sigveseb 3:5432b38585ea 163 }
sigveseb 3:5432b38585ea 164
sigveseb 3:5432b38585ea 165 void Puck::stopAdvertising() {
sigveseb 3:5432b38585ea 166 if(state == ADVERTISING) {
sigveseb 3:5432b38585ea 167 ble.stopAdvertising();
sigveseb 3:5432b38585ea 168 LOG_INFO("Stopped advertising.\n");
sigveseb 3:5432b38585ea 169 setState(DISCONNECTED);
sigveseb 3:5432b38585ea 170 } else {
sigveseb 3:5432b38585ea 171 LOG_WARN("Tried to stop advertising, but advertising is already stopped!\n");
sigveseb 3:5432b38585ea 172 }
sigveseb 3:5432b38585ea 173 }
sigveseb 3:5432b38585ea 174
sigveseb 3:5432b38585ea 175
sigveseb 3:5432b38585ea 176 void Puck::addCharacteristic(const UUID serviceUuid, const UUID characteristicUuid, int bytes, int properties) {
sigveseb 3:5432b38585ea 177 MBED_ASSERT(bytes <= 20);
sigveseb 3:5432b38585ea 178
sigveseb 3:5432b38585ea 179 uint16_t size = sizeof(uint8_t) * bytes;
sigveseb 3:5432b38585ea 180 uint8_t* value = (uint8_t*) malloc(size);
sigveseb 3:5432b38585ea 181 GattCharacteristic* characteristic = new GattCharacteristic(characteristicUuid, value, size, size, properties);
sigveseb 3:5432b38585ea 182 characteristics.push_back(characteristic);
sigveseb 3:5432b38585ea 183 GattService* service = NULL;
sigveseb 3:5432b38585ea 184 int removeIndex = -1;
sigveseb 3:5432b38585ea 185 for(int i = 0; i < services.size(); i++) {
sigveseb 3:5432b38585ea 186 if(isEqualUUID(&services[i]->getUUID(), serviceUuid)) {
sigveseb 3:5432b38585ea 187 service = services[i];
sigveseb 3:5432b38585ea 188 removeIndex = i;
sigveseb 3:5432b38585ea 189 break;
sigveseb 3:5432b38585ea 190 }
sigveseb 3:5432b38585ea 191 }
sigveseb 3:5432b38585ea 192 GattCharacteristic** characteristics = NULL;
sigveseb 3:5432b38585ea 193 int characteristicsLength = 0;
sigveseb 3:5432b38585ea 194 if(service != NULL) {
sigveseb 3:5432b38585ea 195 characteristicsLength = service->getCharacteristicCount() + 1;
sigveseb 3:5432b38585ea 196 characteristics = (GattCharacteristic**) malloc(sizeof(GattCharacteristic*) * characteristicsLength);
sigveseb 3:5432b38585ea 197 for(int i = 0; i < characteristicsLength; i++) {
sigveseb 3:5432b38585ea 198 characteristics[i] = service->getCharacteristic(i);
sigveseb 3:5432b38585ea 199 }
sigveseb 3:5432b38585ea 200 services.erase(services.begin() + removeIndex);
sigveseb 3:5432b38585ea 201 delete service;
stiaje 4:91506772210d 202 free(previousCharacteristics);
sigveseb 3:5432b38585ea 203 } else {
sigveseb 3:5432b38585ea 204 characteristicsLength = 1;
sigveseb 3:5432b38585ea 205 characteristics = (GattCharacteristic**) malloc(sizeof(GattCharacteristic*) * characteristicsLength);
sigveseb 3:5432b38585ea 206 }
sigveseb 3:5432b38585ea 207 characteristics[characteristicsLength - 1] = characteristic;
stiaje 4:91506772210d 208 previousCharacteristics = characteristics;
sigveseb 3:5432b38585ea 209 service = new GattService(serviceUuid, characteristics, characteristicsLength);
sigveseb 3:5432b38585ea 210 services.push_back(service);
sigveseb 3:5432b38585ea 211 }
sigveseb 3:5432b38585ea 212
sigveseb 3:5432b38585ea 213 bool Puck::drive() {
sigveseb 3:5432b38585ea 214 ble.waitForEvent();
sigveseb 3:5432b38585ea 215 if(state == DISCONNECTED) {
sigveseb 3:5432b38585ea 216 startAdvertising();
sigveseb 3:5432b38585ea 217 }
sigveseb 3:5432b38585ea 218 while(pendingCallbackStack.size() > 0) {
sigveseb 3:5432b38585ea 219 LOG_VERBOSE("PendingCallbackStack size: %i\n", pendingCallbackStack.size());
sigveseb 3:5432b38585ea 220 pendingCallbackStack.back()(pendingCallbackParameterStack.back());
sigveseb 3:5432b38585ea 221 pendingCallbackStack.pop_back();
sigveseb 3:5432b38585ea 222 pendingCallbackParameterStack.pop_back();
sigveseb 3:5432b38585ea 223 LOG_VERBOSE("Callback fired\n");
sigveseb 3:5432b38585ea 224 }
sigveseb 3:5432b38585ea 225 return true;
sigveseb 3:5432b38585ea 226 }
sigveseb 3:5432b38585ea 227
sigveseb 3:5432b38585ea 228
sigveseb 3:5432b38585ea 229 void Puck::onCharacteristicWrite(const UUID uuid, CharacteristicWriteCallback callback) {
sigveseb 3:5432b38585ea 230 CharacteristicWriteCallbacks* cb = NULL;
sigveseb 3:5432b38585ea 231 for(int i = 0; i< writeCallbacks.size(); i++) {
sigveseb 3:5432b38585ea 232 if(isEqualUUID(writeCallbacks[i]->uuid, uuid)) {
sigveseb 3:5432b38585ea 233 cb = writeCallbacks[i];
sigveseb 3:5432b38585ea 234 break;
sigveseb 3:5432b38585ea 235 }
sigveseb 3:5432b38585ea 236 }
sigveseb 3:5432b38585ea 237 if(cb == NULL) {
sigveseb 3:5432b38585ea 238 cb = (CharacteristicWriteCallbacks*) malloc(sizeof(CharacteristicWriteCallbacks));
sigveseb 3:5432b38585ea 239 cb->uuid = &uuid;
sigveseb 3:5432b38585ea 240 cb->callbacks = new std::vector<CharacteristicWriteCallback>();
sigveseb 3:5432b38585ea 241 writeCallbacks.push_back(cb);
sigveseb 3:5432b38585ea 242 }
sigveseb 3:5432b38585ea 243 cb->callbacks->push_back(callback);
sigveseb 3:5432b38585ea 244 LOG_VERBOSE("Bound characteristic write callback (uuid: %x, callback: %x)\n", uuid, callback);
sigveseb 3:5432b38585ea 245 }
sigveseb 3:5432b38585ea 246
sigveseb 3:5432b38585ea 247
sigveseb 3:5432b38585ea 248 uint8_t* Puck::getCharacteristicValue(const UUID uuid) {
sigveseb 3:5432b38585ea 249 LOG_VERBOSE("Reading characteristic value for UUID %x\n", uuid);
sigveseb 3:5432b38585ea 250 for(int i = 0; i < characteristics.size(); i++) {
sigveseb 3:5432b38585ea 251 GattCharacteristic* characteristic = characteristics[i];
sigveseb 3:5432b38585ea 252 if(isEqualUUID(&characteristic->getUUID(), uuid)) {
sigveseb 3:5432b38585ea 253 return characteristic->getValuePtr();
sigveseb 3:5432b38585ea 254 }
sigveseb 3:5432b38585ea 255 }
sigveseb 3:5432b38585ea 256 LOG_WARN("Tried to read an unknown characteristic!");
sigveseb 3:5432b38585ea 257 return NULL;
sigveseb 3:5432b38585ea 258 }
sigveseb 3:5432b38585ea 259
sigveseb 3:5432b38585ea 260
sigveseb 3:5432b38585ea 261 void Puck::onDataWritten(uint16_t handle) {
sigveseb 3:5432b38585ea 262 for (int i = 0; i < characteristics.size(); i++) {
sigveseb 3:5432b38585ea 263 GattCharacteristic* characteristic = characteristics[i];
sigveseb 3:5432b38585ea 264 if (characteristic->getHandle() == handle) {
sigveseb 3:5432b38585ea 265 uint16_t maxLength = characteristic->getMaxLength();
sigveseb 3:5432b38585ea 266 ble.readCharacteristicValue(handle, characteristic->getValuePtr(), &maxLength);
sigveseb 3:5432b38585ea 267 for(int j = 0; j < writeCallbacks.size(); j++) {
sigveseb 3:5432b38585ea 268 CharacteristicWriteCallbacks* characteristicWriteCallbacks = writeCallbacks[j];
sigveseb 3:5432b38585ea 269 if(isEqualUUID(characteristicWriteCallbacks->uuid, characteristic->getUUID())) {
sigveseb 3:5432b38585ea 270 for(int k = 0; k < characteristicWriteCallbacks->callbacks->size(); k++) {
sigveseb 3:5432b38585ea 271 pendingCallbackStack.push_back(characteristicWriteCallbacks->callbacks->at(k));
sigveseb 3:5432b38585ea 272 pendingCallbackParameterStack.push_back(characteristic->getValuePtr());
sigveseb 3:5432b38585ea 273 }
sigveseb 3:5432b38585ea 274 }
sigveseb 3:5432b38585ea 275 return;
sigveseb 3:5432b38585ea 276 }
sigveseb 3:5432b38585ea 277 }
sigveseb 3:5432b38585ea 278 }
sigveseb 3:5432b38585ea 279 }
sigveseb 3:5432b38585ea 280
sigveseb 3:5432b38585ea 281
sigveseb 3:5432b38585ea 282
sigveseb 3:5432b38585ea 283 #endif // __PUCK_HPP__