1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
55
60
61
62
63
64
65
66
67
68
73
74
75
76
81
83
84
85
86
87
88
89
90
91
92
105
106
107
109
110
111
112
113
114
116
118
119
120
121
125
126
127
128
129
130
133
134
138
139
140
141
142
143
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
200
201
208
209
216
217
224
225
226
227
228
229
230
231
232
233
234
235
242
243
244
245
246
247
250
251
252
258
259
265
266
269
272
275
278
279
280
281
282
283
284
285
286
287
288
289
290
293
294
295
296
297
298
299
300
301
302
303
304
305
308
309
310
311
317
318
324
325
328
329
332
333
334
335
336
337
338
339
340
341
342
343
355
356
357
358
359
362
363
368
369
370
371
374
375
376
377
378
379
380
384
385
386
387
391
392
/* ... */
#define BTSTACK_FILE__ "att_db_util.c"
#include <string.h>
#include <stdlib.h>
#include "ble/att_db_util.h"
#include "ble/att_db.h"
#include "ble/core.h"
#include "btstack_util.h"
#include "btstack_debug.h"
#include "bluetooth.h"
8 includes
#ifdef HAVE_MALLOC
#define ATT_DB_BUFFER_INCREMENT 128
/* ... */#else
#ifdef MAX_ATT_DB_SIZE
static uint8_t att_db_storage[MAX_ATT_DB_SIZE];
#else
#error Neither HAVE_MALLOC nor MAX_ATT_DB_SIZE is defined.
#endif/* ... */
#endif
static uint8_t * att_db;
static uint16_t att_db_size;
static uint16_t att_db_max_size;
static uint16_t att_db_next_handle;
static uint16_t att_db_hash_len;
static void att_db_util_set_end_tag(void){
att_db[att_db_size] = 0u;
att_db[att_db_size+1u] = 0u;
}{ ... }
void att_db_util_init(void){
#ifdef HAVE_MALLOC
if (att_db == NULL){
att_db = (uint8_t*) malloc((size_t)ATT_DB_BUFFER_INCREMENT);
att_db_max_size = (uint16_t)ATT_DB_BUFFER_INCREMENT;
}if (att_db == NULL) { ... }
/* ... */#else
att_db = att_db_storage;
att_db_max_size = sizeof(att_db_storage);/* ... */
#endif
att_db[0] = (uint8_t)ATT_DB_VERSION;
att_db_size = 1u;
att_db_next_handle = 1u;
att_db_hash_len = 0u;
att_db_util_set_end_tag();
}{ ... }
static bool att_db_util_hash_include_with_value(uint16_t uuid16){
switch (uuid16){
case GATT_PRIMARY_SERVICE_UUID:
case GATT_SECONDARY_SERVICE_UUID:
case GATT_INCLUDE_SERVICE_UUID:
case GATT_CHARACTERISTICS_UUID:
case GATT_CHARACTERISTIC_EXTENDED_PROPERTIES:
return true;case GATT_CHARACTERISTIC_EXTENDED_PROPERTIES:
default:
return false;default
}switch (uuid16) { ... }
}{ ... }
static bool att_db_util_hash_include_without_value(uint16_t uuid16){
/* ... */
switch (uuid16){
case GATT_CHARACTERISTIC_USER_DESCRIPTION:
case GATT_CLIENT_CHARACTERISTICS_CONFIGURATION:
case GATT_SERVER_CHARACTERISTICS_CONFIGURATION:
case GATT_CHARACTERISTIC_PRESENTATION_FORMAT:
case GATT_CHARACTERISTIC_AGGREGATE_FORMAT:
return true;case GATT_CHARACTERISTIC_AGGREGATE_FORMAT:
default:
return false;default
}switch (uuid16) { ... }
}{ ... }
/* ... */
static int att_db_util_assert_space(uint16_t size){
uint16_t required_size = att_db_size + size + 2u;
if (required_size <= att_db_max_size) return 1;
#ifdef HAVE_MALLOC
uint16_t new_size = att_db_max_size;
while (new_size < required_size){
new_size += (uint16_t)ATT_DB_BUFFER_INCREMENT;
}while (new_size < required_size) { ... }
uint8_t * new_db = (uint8_t*) realloc(att_db, new_size);
if (!new_db) {
log_error("att_db: realloc failed");
return 0;
}if (!new_db) { ... }
att_db = new_db;
att_db_max_size = new_size;
att_set_db(att_db);
return 1;/* ... */
#else
log_error("att_db: out of memory");
return 0;/* ... */
#endif
}{ ... }
static void att_db_util_add_attribute_uuid16(uint16_t uuid16, uint16_t flags, uint8_t * data, uint16_t data_len){
int size = 2u + 2u + 2u + 2u + data_len;
if (!att_db_util_assert_space(size)) return;
little_endian_store_16(att_db, att_db_size, size);
att_db_size += 2u;
little_endian_store_16(att_db, att_db_size, flags);
att_db_size += 2u;
little_endian_store_16(att_db, att_db_size, att_db_next_handle);
att_db_size += 2u;
att_db_next_handle++;
little_endian_store_16(att_db, att_db_size, uuid16);
att_db_size += 2u;
(void)memcpy(&att_db[att_db_size], data, data_len);
att_db_size += data_len;
att_db_util_set_end_tag();
if (att_db_util_hash_include_with_value(uuid16)){
att_db_hash_len += 4u + data_len;
}if (att_db_util_hash_include_with_value(uuid16)) { ... } else if (att_db_util_hash_include_without_value(uuid16)){
att_db_hash_len += 4u;
}else if (att_db_util_hash_include_without_value(uuid16)) { ... }
}{ ... }
static void att_db_util_add_attribute_uuid128(const uint8_t * uuid128, uint16_t flags, uint8_t * data, uint16_t data_len){
int size = 2u + 2u + 2u + 16u + data_len;
if (!att_db_util_assert_space(size)) return;
uint16_t flags_to_store = flags | (uint16_t)ATT_PROPERTY_UUID128;
little_endian_store_16(att_db, att_db_size, size);
att_db_size += 2u;
little_endian_store_16(att_db, att_db_size, flags_to_store);
att_db_size += 2u;
little_endian_store_16(att_db, att_db_size, att_db_next_handle);
att_db_size += 2u;
att_db_next_handle++;
reverse_128(uuid128, &att_db[att_db_size]);
att_db_size += 16u;
(void)memcpy(&att_db[att_db_size], data, data_len);
att_db_size += data_len;
att_db_util_set_end_tag();
}{ ... }
uint16_t att_db_util_add_service_uuid16(uint16_t uuid16){
uint8_t buffer[2];
little_endian_store_16(buffer, 0, uuid16);
uint16_t service_handle = att_db_next_handle;
att_db_util_add_attribute_uuid16(GATT_PRIMARY_SERVICE_UUID, ATT_PROPERTY_READ, buffer, 2);
return service_handle;
}{ ... }
uint16_t att_db_util_add_service_uuid128(const uint8_t * uuid128){
uint8_t buffer[16];
reverse_128(uuid128, buffer);
uint16_t service_handle = att_db_next_handle;
att_db_util_add_attribute_uuid16(GATT_PRIMARY_SERVICE_UUID, ATT_PROPERTY_READ, buffer, 16);
return service_handle;
}{ ... }
uint16_t att_db_util_add_secondary_service_uuid16(uint16_t uuid16){
uint8_t buffer[2];
little_endian_store_16(buffer, 0, uuid16);
uint16_t service_handle = att_db_next_handle;
att_db_util_add_attribute_uuid16(GATT_SECONDARY_SERVICE_UUID, ATT_PROPERTY_READ, buffer, 2);
return service_handle;
}{ ... }
uint16_t att_db_util_add_secondary_service_uuid128(const uint8_t * uuid128){
uint8_t buffer[16];
reverse_128(uuid128, buffer);
uint16_t service_handle = att_db_next_handle;
att_db_util_add_attribute_uuid16(GATT_SECONDARY_SERVICE_UUID, ATT_PROPERTY_READ, buffer, 16);
return service_handle;
}{ ... }
uint16_t att_db_util_add_included_service_uuid16(uint16_t start_group_handle, uint16_t end_group_handle, uint16_t uuid16){
uint8_t buffer[6];
little_endian_store_16(buffer, 0, start_group_handle);
little_endian_store_16(buffer, 2, end_group_handle);
little_endian_store_16(buffer, 4, uuid16);
uint16_t service_handle = att_db_next_handle;
att_db_util_add_attribute_uuid16(GATT_INCLUDE_SERVICE_UUID, ATT_PROPERTY_READ, buffer, sizeof(buffer));
return service_handle;
}{ ... }
static void att_db_util_add_client_characteristic_configuration(uint16_t flags){
uint8_t buffer[2];
uint16_t flags_to_store = (flags & 0x1f391u) | (uint16_t)ATT_PROPERTY_READ | (uint16_t)ATT_PROPERTY_WRITE | (uint16_t)ATT_PROPERTY_DYNAMIC;
little_endian_store_16(buffer, 0, 0);
att_db_util_add_attribute_uuid16(GATT_CLIENT_CHARACTERISTICS_CONFIGURATION, flags_to_store, buffer, 2);
}{ ... }
static uint16_t att_db_util_encode_permissions(uint16_t properties, uint8_t read_permission, uint8_t write_permission){
uint16_t flags = properties & 0xfff4eu;
if ((read_permission > (uint8_t)ATT_SECURITY_NONE) || (write_permission > (uint8_t)ATT_SECURITY_NONE)){
flags |= 0xf000u;
}if ((read_permission > (uint8_t)ATT_SECURITY_NONE) || (write_permission > (uint8_t)ATT_SECURITY_NONE)) { ... }
uint8_t final_read_permission;
if (read_permission == (uint8_t)ATT_SECURITY_AUTHENTICATED_SC){
final_read_permission = (uint8_t)ATT_SECURITY_AUTHENTICATED;
flags |= (uint16_t)ATT_PROPERTY_READ_PERMISSION_SC;
}if (read_permission == (uint8_t)ATT_SECURITY_AUTHENTICATED_SC) { ... } else {
final_read_permission = read_permission;
}else { ... }
uint8_t final_write_permission;
if (write_permission == (uint8_t)ATT_SECURITY_AUTHENTICATED_SC){
final_write_permission = (uint8_t)ATT_SECURITY_AUTHENTICATED;
flags |= (uint16_t)ATT_PROPERTY_WRITE_PERMISSION_SC;
}if (write_permission == (uint8_t)ATT_SECURITY_AUTHENTICATED_SC) { ... } else {
final_write_permission = write_permission;
}else { ... }
if ((final_read_permission & 1u) != 0u){
flags |= (uint16_t)ATT_PROPERTY_READ_PERMISSION_BIT_0;
}if ((final_read_permission & 1u) != 0u) { ... }
if ((final_read_permission & 2u) != 0u){
flags |= (uint16_t)ATT_PROPERTY_READ_PERMISSION_BIT_1;
}if ((final_read_permission & 2u) != 0u) { ... }
if ((final_write_permission & 1u) != 0u){
flags |= (uint16_t)ATT_PROPERTY_WRITE_PERMISSION_BIT_0;
}if ((final_write_permission & 1u) != 0u) { ... }
if ((final_write_permission & 2u) != 0u){
flags |= (uint16_t)ATT_PROPERTY_WRITE_PERMISSION_BIT_1;
}if ((final_write_permission & 2u) != 0u) { ... }
return flags;
}{ ... }
uint16_t att_db_util_add_characteristic_uuid16(uint16_t uuid16, uint16_t properties, uint8_t read_permission, uint8_t write_permission, uint8_t * data, uint16_t data_len){
uint8_t buffer[5];
buffer[0] = (uint8_t) (properties & 0xff);
little_endian_store_16(buffer, 1u, att_db_next_handle + 1u);
little_endian_store_16(buffer, 3, uuid16);
att_db_util_add_attribute_uuid16(GATT_CHARACTERISTICS_UUID, ATT_PROPERTY_READ, buffer, sizeof(buffer));
uint16_t flags = att_db_util_encode_permissions(properties, read_permission, write_permission);
uint16_t value_handle = att_db_next_handle;
att_db_util_add_attribute_uuid16(uuid16, flags, data, data_len);
if ((properties & (uint8_t)(ATT_PROPERTY_NOTIFY | ATT_PROPERTY_INDICATE)) != 0u){
att_db_util_add_client_characteristic_configuration(flags);
}if ((properties & (uint8_t)(ATT_PROPERTY_NOTIFY | ATT_PROPERTY_INDICATE)) != 0u) { ... }
return value_handle;
}{ ... }
uint16_t att_db_util_add_characteristic_uuid128(const uint8_t * uuid128, uint16_t properties, uint8_t read_permission, uint8_t write_permission, uint8_t * data, uint16_t data_len){
uint8_t buffer[19];
buffer[0] = (uint8_t) (properties & 0xff);
little_endian_store_16(buffer, 1u, att_db_next_handle + 1u);
reverse_128(uuid128, &buffer[3]);
att_db_util_add_attribute_uuid16(GATT_CHARACTERISTICS_UUID, ATT_PROPERTY_READ, buffer, sizeof(buffer));
uint16_t flags = att_db_util_encode_permissions(properties, read_permission, write_permission);
uint16_t value_handle = att_db_next_handle;
att_db_util_add_attribute_uuid128(uuid128, flags, data, data_len);
if ((properties & (uint8_t)(ATT_PROPERTY_NOTIFY | ATT_PROPERTY_INDICATE)) != 0u){
att_db_util_add_client_characteristic_configuration(flags);
}if ((properties & (uint8_t)(ATT_PROPERTY_NOTIFY | ATT_PROPERTY_INDICATE)) != 0u) { ... }
return value_handle;
}{ ... }
uint16_t att_db_util_add_descriptor_uuid16(uint16_t uuid16, uint16_t properties, uint8_t read_permission, uint8_t write_permission, uint8_t * data, uint16_t data_len){
uint16_t descriptor_handler = att_db_next_handle;
uint16_t flags = att_db_util_encode_permissions(properties, read_permission, write_permission);
att_db_util_add_attribute_uuid16(uuid16, flags, data, data_len);
return descriptor_handler;
}{ ... }
uint16_t att_db_util_add_descriptor_uuid128(const uint8_t * uuid128, uint16_t properties, uint8_t read_permission, uint8_t write_permission, uint8_t * data, uint16_t data_len){
uint16_t descriptor_handler = att_db_next_handle;
uint16_t flags = att_db_util_encode_permissions(properties, read_permission, write_permission);
att_db_util_add_attribute_uuid128(uuid128, flags, data, data_len);
return descriptor_handler;
}{ ... }
uint8_t * att_db_util_get_address(void){
return att_db;
}{ ... }
uint16_t att_db_util_get_size(void){
return att_db_size + 2u;
}{ ... }
static uint8_t * att_db_util_hash_att_ptr;
static uint16_t att_db_util_hash_offset;
static uint16_t att_db_util_hash_bytes_available;
static void att_db_util_hash_fetch_next_attribute(void){
while (true){
uint16_t size = little_endian_read_16(att_db_util_hash_att_ptr, 0);
btstack_assert(size != 0);
UNUSED(size);
uint16_t flags = little_endian_read_16(att_db_util_hash_att_ptr, 2);
if ((flags & (uint16_t)ATT_PROPERTY_UUID128) == 0u) {
uint16_t uuid16 = little_endian_read_16(att_db_util_hash_att_ptr, 6);
if (att_db_util_hash_include_with_value(uuid16)){
att_db_util_hash_offset = 4;
att_db_util_hash_bytes_available = size - 4u;
return;
}if (att_db_util_hash_include_with_value(uuid16)) { ... } else if (att_db_util_hash_include_without_value(uuid16)){
att_db_util_hash_offset = 4;
att_db_util_hash_bytes_available = 4;
return;
}else if (att_db_util_hash_include_without_value(uuid16)) { ... }
}if ((flags & (uint16_t)ATT_PROPERTY_UUID128) == 0u) { ... }
att_db_util_hash_att_ptr += size;
}while (true) { ... }
}{ ... }
uint16_t att_db_util_hash_len(void){
return att_db_hash_len;
}{ ... }
void att_db_util_hash_init(void){
att_db_util_hash_att_ptr = &att_db[1];
att_db_util_hash_bytes_available = 0u;
}{ ... }
uint8_t att_db_util_hash_get_next(void){
if (att_db_util_hash_bytes_available == 0u){
att_db_util_hash_fetch_next_attribute();
}if (att_db_util_hash_bytes_available == 0u) { ... }
uint8_t next = att_db_util_hash_att_ptr[att_db_util_hash_offset++];
att_db_util_hash_bytes_available--;
if (att_db_util_hash_bytes_available == 0u){
uint16_t size = little_endian_read_16(att_db_util_hash_att_ptr, 0);
att_db_util_hash_att_ptr += size;
}if (att_db_util_hash_bytes_available == 0u) { ... }
return next;
}{ ... }
static uint8_t att_db_util_hash_get(uint16_t offset){
UNUSED(offset);
return att_db_util_hash_get_next();
}{ ... }
void att_db_util_hash_calc(btstack_crypto_aes128_cmac_t * request, uint8_t * db_hash, void (* callback)(void * arg), void * callback_arg){
static const uint8_t zero_key[16] = { 0 };
att_db_util_hash_init();
btstack_crypto_aes128_cmac_generator(request, zero_key, att_db_hash_len, &att_db_util_hash_get, db_hash, callback, callback_arg);
}{ ... }