test

Fork of nRF51822 by Nordic Semiconductor

Committer:
rgrover1
Date:
Fri Jun 19 15:55:22 2015 +0100
Revision:
245:3abc61d38db3
Parent:
244:57c98fe71376
Child:
246:1098730d84de
Synchronized with git rev 993a277c
Author: Rohit Grover
reorganized the division of helper functions between SD and NordicSD once again.

Who changed what in which revision?

UserRevisionLine numberNew contents of line
rgrover1 239:693a1f145b5a 1 /* mbed Microcontroller Library
rgrover1 239:693a1f145b5a 2 * Copyright (c) 2006-2013 ARM Limited
rgrover1 239:693a1f145b5a 3 *
rgrover1 239:693a1f145b5a 4 * Licensed under the Apache License, Version 2.0 (the "License");
rgrover1 239:693a1f145b5a 5 * you may not use this file except in compliance with the License.
rgrover1 239:693a1f145b5a 6 * You may obtain a copy of the License at
rgrover1 239:693a1f145b5a 7 *
rgrover1 239:693a1f145b5a 8 * http://www.apache.org/licenses/LICENSE-2.0
rgrover1 239:693a1f145b5a 9 *
rgrover1 239:693a1f145b5a 10 * Unless required by applicable law or agreed to in writing, software
rgrover1 239:693a1f145b5a 11 * distributed under the License is distributed on an "AS IS" BASIS,
rgrover1 239:693a1f145b5a 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
rgrover1 239:693a1f145b5a 13 * See the License for the specific language governing permissions and
rgrover1 239:693a1f145b5a 14 * limitations under the License.
rgrover1 239:693a1f145b5a 15 */
rgrover1 239:693a1f145b5a 16
rgrover1 239:693a1f145b5a 17 #ifndef _BTLE_DISCOVERY_H_
rgrover1 239:693a1f145b5a 18 #define _BTLE_DISCOVERY_H_
rgrover1 239:693a1f145b5a 19
rgrover1 239:693a1f145b5a 20 #include "blecommon.h"
rgrover1 239:693a1f145b5a 21 #include "UUID.h"
rgrover1 239:693a1f145b5a 22 #include "Gap.h"
rgrover1 239:693a1f145b5a 23 #include "ble_gattc.h"
rgrover1 239:693a1f145b5a 24 #include <stdio.h>
rgrover1 239:693a1f145b5a 25
rgrover1 239:693a1f145b5a 26 /**@brief Structure for holding information about the service and the characteristics found during
rgrover1 239:693a1f145b5a 27 * the discovery process.
rgrover1 239:693a1f145b5a 28 */
rgrover1 239:693a1f145b5a 29 struct DiscoveredService {
rgrover1 239:693a1f145b5a 30 void setup(ShortUUIDBytes_t uuidIn, Gap::Handle_t start, Gap::Handle_t end) {
rgrover1 239:693a1f145b5a 31 uuid = uuidIn;
rgrover1 239:693a1f145b5a 32 startHandle = start;
rgrover1 239:693a1f145b5a 33 endHandle = end;
rgrover1 239:693a1f145b5a 34 }
rgrover1 239:693a1f145b5a 35
rgrover1 239:693a1f145b5a 36 ShortUUIDBytes_t uuid; /**< UUID of the service. */
rgrover1 239:693a1f145b5a 37 Gap::Handle_t startHandle; /**< Service Handle Range. */
rgrover1 239:693a1f145b5a 38 Gap::Handle_t endHandle; /**< Service Handle Range. */
rgrover1 239:693a1f145b5a 39 };
rgrover1 239:693a1f145b5a 40
rgrover1 239:693a1f145b5a 41 /**@brief Structure for holding information about the service and the characteristics found during
rgrover1 239:693a1f145b5a 42 * the discovery process.
rgrover1 239:693a1f145b5a 43 */
rgrover1 239:693a1f145b5a 44 struct DiscoveredCharacteristic {
rgrover1 239:693a1f145b5a 45 struct Properties_t {
rgrover1 239:693a1f145b5a 46 static const uint8_t BROADCAST_PROPERTY_MASK = 0x01;
rgrover1 239:693a1f145b5a 47 static const uint8_t READ_PROPERTY_MASK = 0x02;
rgrover1 239:693a1f145b5a 48 static const uint8_t WRITE_WO_RESPONSE_PROPERTY_MASK = 0x04;
rgrover1 239:693a1f145b5a 49 static const uint8_t WRITE_PROPERTY_MASK = 0x08;
rgrover1 239:693a1f145b5a 50 static const uint8_t NOTIFY_PROPERTY_MASK = 0x10;
rgrover1 239:693a1f145b5a 51 static const uint8_t INDICATE_PROPERTY_MASK = 0x20;
rgrover1 239:693a1f145b5a 52 static const uint8_t AUTH_SIGNED_PROPERTY_MASK = 0x40;
rgrover1 239:693a1f145b5a 53
rgrover1 239:693a1f145b5a 54 Properties_t() : broadcast(0), read(0), write_wo_resp(0), write(0), notify(0), indicate(0), auth_signed_wr(0) {
rgrover1 239:693a1f145b5a 55 /* empty */
rgrover1 239:693a1f145b5a 56 }
rgrover1 239:693a1f145b5a 57
rgrover1 239:693a1f145b5a 58 Properties_t(uint8_t props) :
rgrover1 239:693a1f145b5a 59 broadcast(props & BROADCAST_PROPERTY_MASK),
rgrover1 239:693a1f145b5a 60 read(props & READ_PROPERTY_MASK),
rgrover1 239:693a1f145b5a 61 write_wo_resp(props & WRITE_WO_RESPONSE_PROPERTY_MASK),
rgrover1 239:693a1f145b5a 62 write(props & WRITE_PROPERTY_MASK),
rgrover1 239:693a1f145b5a 63 notify(props & NOTIFY_PROPERTY_MASK),
rgrover1 239:693a1f145b5a 64 indicate(props & INDICATE_PROPERTY_MASK),
rgrover1 239:693a1f145b5a 65 auth_signed_wr(props & AUTH_SIGNED_PROPERTY_MASK) {
rgrover1 239:693a1f145b5a 66 /* empty*/
rgrover1 239:693a1f145b5a 67 }
rgrover1 239:693a1f145b5a 68
rgrover1 239:693a1f145b5a 69 uint8_t broadcast :1; /**< Broadcasting of the value permitted. */
rgrover1 239:693a1f145b5a 70 uint8_t read :1; /**< Reading the value permitted. */
rgrover1 239:693a1f145b5a 71 uint8_t write_wo_resp :1; /**< Writing the value with Write Command permitted. */
rgrover1 239:693a1f145b5a 72 uint8_t write :1; /**< Writing the value with Write Request permitted. */
rgrover1 239:693a1f145b5a 73 uint8_t notify :1; /**< Notications of the value permitted. */
rgrover1 239:693a1f145b5a 74 uint8_t indicate :1; /**< Indications of the value permitted. */
rgrover1 239:693a1f145b5a 75 uint8_t auth_signed_wr :1; /**< Writing the value with Signed Write Command permitted. */
rgrover1 239:693a1f145b5a 76 };
rgrover1 239:693a1f145b5a 77
rgrover1 239:693a1f145b5a 78 void setup(ShortUUIDBytes_t uuidIn, Properties_t propsIn, Gap::Handle_t declHandleIn, Gap::Handle_t valueHandleIn) {
rgrover1 239:693a1f145b5a 79 uuid = uuidIn;
rgrover1 239:693a1f145b5a 80 props = propsIn;
rgrover1 239:693a1f145b5a 81 declHandle = declHandleIn;
rgrover1 239:693a1f145b5a 82 valueHandle = valueHandleIn;
rgrover1 239:693a1f145b5a 83 }
rgrover1 239:693a1f145b5a 84
rgrover1 239:693a1f145b5a 85 ShortUUIDBytes_t uuid;
rgrover1 239:693a1f145b5a 86 Properties_t props;
rgrover1 239:693a1f145b5a 87 Gap::Handle_t declHandle;
rgrover1 239:693a1f145b5a 88 Gap::Handle_t valueHandle;
rgrover1 239:693a1f145b5a 89 };
rgrover1 239:693a1f145b5a 90
rgrover1 239:693a1f145b5a 91 class ServiceDiscovery {
rgrover1 239:693a1f145b5a 92 public:
rgrover1 239:693a1f145b5a 93 static const unsigned BLE_DB_DISCOVERY_MAX_SRV = 4; /**< Maximum number of services supported by this module. This also indicates the maximum number of users allowed to be registered to this module. (one user per service). */
rgrover1 239:693a1f145b5a 94 static const unsigned BLE_DB_DISCOVERY_MAX_CHAR_PER_SRV = 4; /**< Maximum number of characteristics per service supported by this module. */
rgrover1 239:693a1f145b5a 95
rgrover1 239:693a1f145b5a 96 static const uint16_t SRV_DISC_START_HANDLE = 0x0001; /**< The start handle value used during service discovery. */
rgrover1 239:693a1f145b5a 97
rgrover1 239:693a1f145b5a 98 typedef void (*ServiceCallback_t)(void);
rgrover1 239:693a1f145b5a 99 typedef void (*CharacteristicCallback_t)(void);
rgrover1 239:693a1f145b5a 100
rgrover1 239:693a1f145b5a 101 public:
rgrover1 239:693a1f145b5a 102 static ble_error_t launch(Gap::Handle_t connectionHandle,
rgrover1 239:693a1f145b5a 103 ServiceCallback_t sc,
rgrover1 239:693a1f145b5a 104 CharacteristicCallback_t cc = NULL);
rgrover1 239:693a1f145b5a 105 static ble_error_t launch(Gap::Handle_t connectionHandle,
rgrover1 239:693a1f145b5a 106 UUID matchingServiceUUID,
rgrover1 239:693a1f145b5a 107 ServiceCallback_t sc,
rgrover1 239:693a1f145b5a 108 UUID matchingCharacteristicUUID = ShortUUIDBytes_t(BLE_UUID_UNKNOWN),
rgrover1 239:693a1f145b5a 109 CharacteristicCallback_t cc = NULL);
rgrover1 239:693a1f145b5a 110
rgrover1 239:693a1f145b5a 111 static ServiceDiscovery *getSingleton(void);
rgrover1 239:693a1f145b5a 112
rgrover1 239:693a1f145b5a 113 public:
rgrover1 239:693a1f145b5a 114 void terminate(void) {
rgrover1 242:73fc02cc20b1 115 sDiscoveryActive = false;
rgrover1 239:693a1f145b5a 116 printf("end of service discovery\r\n");
rgrover1 239:693a1f145b5a 117 }
rgrover1 239:693a1f145b5a 118
rgrover1 245:3abc61d38db3 119 void terminateCharacteristicDiscovery(void) {
rgrover1 245:3abc61d38db3 120 cDiscoveryActive = false;
rgrover1 245:3abc61d38db3 121 sDiscoveryActive = true;
rgrover1 245:3abc61d38db3 122 serviceIndex++; /* Progress service index to keep discovery alive. */
rgrover1 245:3abc61d38db3 123 }
rgrover1 245:3abc61d38db3 124
rgrover1 239:693a1f145b5a 125 void resetDiscoveredServices(void) {
rgrover1 242:73fc02cc20b1 126 numServices = 0;
rgrover1 242:73fc02cc20b1 127 serviceIndex = 0;
rgrover1 239:693a1f145b5a 128 memset(services, 0, sizeof(DiscoveredService) * BLE_DB_DISCOVERY_MAX_SRV);
rgrover1 239:693a1f145b5a 129 }
rgrover1 239:693a1f145b5a 130
rgrover1 245:3abc61d38db3 131 protected:
rgrover1 245:3abc61d38db3 132 void resetDiscoveredCharacteristics(void) {
rgrover1 245:3abc61d38db3 133 numCharacteristics = 0;
rgrover1 245:3abc61d38db3 134 characteristicIndex = 0;
rgrover1 245:3abc61d38db3 135 memset(characteristics, 0, sizeof(DiscoveredCharacteristic) * BLE_DB_DISCOVERY_MAX_CHAR_PER_SRV);
rgrover1 245:3abc61d38db3 136 }
rgrover1 245:3abc61d38db3 137
rgrover1 245:3abc61d38db3 138 public:
rgrover1 243:a966506d1e5b 139 void serviceDiscoveryStarted(Gap::Handle_t connectionHandle) {
rgrover1 243:a966506d1e5b 140 connHandle = connectionHandle;
rgrover1 243:a966506d1e5b 141 resetDiscoveredServices();
rgrover1 243:a966506d1e5b 142 sDiscoveryActive = true;
rgrover1 243:a966506d1e5b 143 cDiscoveryActive = false;
rgrover1 243:a966506d1e5b 144 }
rgrover1 243:a966506d1e5b 145
rgrover1 243:a966506d1e5b 146 protected:
rgrover1 245:3abc61d38db3 147 void characteristicDiscoveryStarted(Gap::Handle_t connectionHandle) {
rgrover1 245:3abc61d38db3 148 connHandle = connectionHandle;
rgrover1 245:3abc61d38db3 149 resetDiscoveredCharacteristics();
rgrover1 245:3abc61d38db3 150 cDiscoveryActive = true;
rgrover1 245:3abc61d38db3 151 sDiscoveryActive = false;
rgrover1 245:3abc61d38db3 152 }
rgrover1 245:3abc61d38db3 153
rgrover1 245:3abc61d38db3 154 protected:
rgrover1 243:a966506d1e5b 155 ServiceDiscovery() {
rgrover1 243:a966506d1e5b 156 /* empty */
rgrover1 243:a966506d1e5b 157 }
rgrover1 243:a966506d1e5b 158
rgrover1 243:a966506d1e5b 159 protected:
rgrover1 243:a966506d1e5b 160 DiscoveredService services[BLE_DB_DISCOVERY_MAX_SRV]; /**< Information related to the current service being discovered.
rgrover1 243:a966506d1e5b 161 * This is intended for internal use during service discovery. */
rgrover1 243:a966506d1e5b 162 DiscoveredCharacteristic characteristics[BLE_DB_DISCOVERY_MAX_CHAR_PER_SRV];
rgrover1 243:a966506d1e5b 163
rgrover1 243:a966506d1e5b 164 uint16_t connHandle; /**< Connection handle as provided by the SoftDevice. */
rgrover1 243:a966506d1e5b 165 uint8_t serviceIndex; /**< Index of the current service being discovered. This is intended for internal use during service discovery.*/
rgrover1 243:a966506d1e5b 166 uint8_t numServices; /**< Number of services at the peers GATT database.*/
rgrover1 243:a966506d1e5b 167 uint8_t characteristicIndex; /**< Index of the current characteristic being discovered. This is intended for internal use during service discovery.*/
rgrover1 243:a966506d1e5b 168 uint8_t numCharacteristics; /**< Number of characteristics within the service.*/
rgrover1 243:a966506d1e5b 169
rgrover1 243:a966506d1e5b 170 bool sDiscoveryActive;
rgrover1 243:a966506d1e5b 171 bool cDiscoveryActive;
rgrover1 243:a966506d1e5b 172 };
rgrover1 243:a966506d1e5b 173
rgrover1 243:a966506d1e5b 174 class NordicServiceDiscovery : public ServiceDiscovery {
rgrover1 243:a966506d1e5b 175 public:
rgrover1 243:a966506d1e5b 176 void setupDiscoveredServices(const ble_gattc_evt_prim_srvc_disc_rsp_t *response);
rgrover1 243:a966506d1e5b 177 void setupDiscoveredCharacteristics(const ble_gattc_evt_char_disc_rsp_t *response);
rgrover1 243:a966506d1e5b 178
rgrover1 244:57c98fe71376 179 public:
rgrover1 244:57c98fe71376 180 ble_error_t launchCharacteristicDiscovery(Gap::Handle_t connectionHandle, Gap::Handle_t startHandle, Gap::Handle_t endHandle);
rgrover1 244:57c98fe71376 181
rgrover1 243:a966506d1e5b 182 public:
rgrover1 239:693a1f145b5a 183 void progressCharacteristicDiscovery() {
rgrover1 242:73fc02cc20b1 184 while (cDiscoveryActive && (characteristicIndex < numCharacteristics)) {
rgrover1 239:693a1f145b5a 185 /* THIS IS WHERE THE CALLBACK WILL GO */
rgrover1 242:73fc02cc20b1 186 printf("%x [%u]\r\n", characteristics[characteristicIndex].uuid, characteristics[characteristicIndex].valueHandle);
rgrover1 239:693a1f145b5a 187
rgrover1 242:73fc02cc20b1 188 characteristicIndex++;
rgrover1 239:693a1f145b5a 189 }
rgrover1 239:693a1f145b5a 190
rgrover1 242:73fc02cc20b1 191 if (cDiscoveryActive) {
rgrover1 242:73fc02cc20b1 192 Gap::Handle_t startHandle = characteristics[characteristicIndex - 1].valueHandle + 1;
rgrover1 242:73fc02cc20b1 193 Gap::Handle_t endHandle = services[serviceIndex].endHandle;
rgrover1 239:693a1f145b5a 194 resetDiscoveredCharacteristics();
rgrover1 239:693a1f145b5a 195
rgrover1 239:693a1f145b5a 196 if (startHandle < endHandle) {
rgrover1 239:693a1f145b5a 197 ble_gattc_handle_range_t handleRange = {
rgrover1 239:693a1f145b5a 198 .start_handle = startHandle,
rgrover1 239:693a1f145b5a 199 .end_handle = endHandle
rgrover1 239:693a1f145b5a 200 };
rgrover1 239:693a1f145b5a 201 printf("char discovery returned %u\r\n", sd_ble_gattc_characteristics_discover(connHandle, &handleRange));
rgrover1 239:693a1f145b5a 202 } else {
rgrover1 239:693a1f145b5a 203 terminateCharacteristicDiscovery();
rgrover1 239:693a1f145b5a 204 }
rgrover1 239:693a1f145b5a 205 }
rgrover1 239:693a1f145b5a 206 }
rgrover1 239:693a1f145b5a 207
rgrover1 239:693a1f145b5a 208 void progressServiceDiscovery() {
rgrover1 242:73fc02cc20b1 209 while (sDiscoveryActive && (serviceIndex < numServices)) {
rgrover1 239:693a1f145b5a 210 /* THIS IS WHERE THE CALLBACK WILL GO */
rgrover1 242:73fc02cc20b1 211 printf("%x [%u %u]\r\n", services[serviceIndex].uuid, services[serviceIndex].startHandle, services[serviceIndex].endHandle);
rgrover1 239:693a1f145b5a 212
rgrover1 239:693a1f145b5a 213 if (true) { /* characteristic discovery is optional. */
rgrover1 242:73fc02cc20b1 214 launchCharacteristicDiscovery(connHandle, services[serviceIndex].startHandle, services[serviceIndex].endHandle);
rgrover1 239:693a1f145b5a 215 } else {
rgrover1 242:73fc02cc20b1 216 serviceIndex++; /* Progress service index to keep discovery alive. */
rgrover1 239:693a1f145b5a 217 }
rgrover1 239:693a1f145b5a 218 }
rgrover1 242:73fc02cc20b1 219 if (sDiscoveryActive && (numServices > 0) && (serviceIndex > 0)) {
rgrover1 242:73fc02cc20b1 220 Gap::Handle_t endHandle = services[serviceIndex - 1].endHandle;
rgrover1 239:693a1f145b5a 221 resetDiscoveredServices();
rgrover1 239:693a1f145b5a 222
rgrover1 239:693a1f145b5a 223 printf("services discover returned %u\r\n", sd_ble_gattc_primary_services_discover(connHandle, endHandle, NULL));
rgrover1 239:693a1f145b5a 224 }
rgrover1 239:693a1f145b5a 225 }
rgrover1 239:693a1f145b5a 226
rgrover1 240:75b69581d1dd 227 };
rgrover1 240:75b69581d1dd 228
rgrover1 239:693a1f145b5a 229 #endif /*_BTLE_DISCOVERY_H_*/