1b12ad867SMatthias Ringwald /*
2b12ad867SMatthias Ringwald * Copyright (C) 2014 BlueKitchen GmbH
3b12ad867SMatthias Ringwald *
4b12ad867SMatthias Ringwald * Redistribution and use in source and binary forms, with or without
5b12ad867SMatthias Ringwald * modification, are permitted provided that the following conditions
6b12ad867SMatthias Ringwald * are met:
7b12ad867SMatthias Ringwald *
8b12ad867SMatthias Ringwald * 1. Redistributions of source code must retain the above copyright
9b12ad867SMatthias Ringwald * notice, this list of conditions and the following disclaimer.
10b12ad867SMatthias Ringwald * 2. Redistributions in binary form must reproduce the above copyright
11b12ad867SMatthias Ringwald * notice, this list of conditions and the following disclaimer in the
12b12ad867SMatthias Ringwald * documentation and/or other materials provided with the distribution.
13b12ad867SMatthias Ringwald * 3. Neither the name of the copyright holders nor the names of
14b12ad867SMatthias Ringwald * contributors may be used to endorse or promote products derived
15b12ad867SMatthias Ringwald * from this software without specific prior written permission.
16b12ad867SMatthias Ringwald * 4. Any redistribution, use, or modification is done solely for
17b12ad867SMatthias Ringwald * personal benefit and not for any commercial purpose or for
18b12ad867SMatthias Ringwald * monetary gain.
19b12ad867SMatthias Ringwald *
20b12ad867SMatthias Ringwald * THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS
21b12ad867SMatthias Ringwald * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
22b12ad867SMatthias Ringwald * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
23*2fca4dadSMilanka Ringwald * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL BLUEKITCHEN
24*2fca4dadSMilanka Ringwald * GMBH OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
25b12ad867SMatthias Ringwald * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
26b12ad867SMatthias Ringwald * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
27b12ad867SMatthias Ringwald * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
28b12ad867SMatthias Ringwald * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
29b12ad867SMatthias Ringwald * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
30b12ad867SMatthias Ringwald * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
31b12ad867SMatthias Ringwald * SUCH DAMAGE.
32b12ad867SMatthias Ringwald *
33b12ad867SMatthias Ringwald * Please inquire about commercial licensing options at
34b12ad867SMatthias Ringwald * [email protected]
35b12ad867SMatthias Ringwald *
36b12ad867SMatthias Ringwald */
37b12ad867SMatthias Ringwald
38e501bae0SMatthias Ringwald #define BTSTACK_FILE__ "ad_parser.c"
39ab2c6ae4SMatthias Ringwald
40b12ad867SMatthias Ringwald
41b12ad867SMatthias Ringwald // *****************************************************************************
42b12ad867SMatthias Ringwald //
43b12ad867SMatthias Ringwald // Advertising Data Parser
44b12ad867SMatthias Ringwald //
45b12ad867SMatthias Ringwald // *****************************************************************************
46b12ad867SMatthias Ringwald
47b12ad867SMatthias Ringwald #include <stdint.h>
48b12ad867SMatthias Ringwald #include <string.h>
49b12ad867SMatthias Ringwald
501d0cde9dSMatthias Ringwald #include "bluetooth_data_types.h"
51b12ad867SMatthias Ringwald #include "btstack_util.h"
521d0cde9dSMatthias Ringwald #include "hci.h"
53b12ad867SMatthias Ringwald #include "hci_cmd.h"
54b12ad867SMatthias Ringwald
5566818fc6SMatthias Ringwald #include "ad_parser.h"
56b12ad867SMatthias Ringwald
ad_iterator_init(ad_context_t * context,uint8_t ad_len,const uint8_t * ad_data)57b12ad867SMatthias Ringwald void ad_iterator_init(ad_context_t *context, uint8_t ad_len, const uint8_t * ad_data){
58b12ad867SMatthias Ringwald context->data = ad_data;
59b12ad867SMatthias Ringwald context->length = ad_len;
60b12ad867SMatthias Ringwald context->offset = 0;
61b12ad867SMatthias Ringwald }
62b12ad867SMatthias Ringwald
ad_iterator_has_more(const ad_context_t * context)632baf01d2SMatthias Ringwald bool ad_iterator_has_more(const ad_context_t * context){
6488949f84SMatthias Ringwald // assert chunk_len and chunk_type are withing buffer
650dd3c6c3SMatthias Ringwald if ((context->offset + 1u) >= context->length) {
66e0558b1bSMatthias Ringwald return false;
67e0558b1bSMatthias Ringwald }
6888949f84SMatthias Ringwald
6988949f84SMatthias Ringwald // assert chunk_len > 0
70e0558b1bSMatthias Ringwald uint8_t chunk_len = context->data[context->offset];
710dd3c6c3SMatthias Ringwald if (chunk_len == 0u){
72e0558b1bSMatthias Ringwald return false;
73e0558b1bSMatthias Ringwald }
7488949f84SMatthias Ringwald
7588949f84SMatthias Ringwald // assert complete chunk fits into buffer
760dd3c6c3SMatthias Ringwald if ((context->offset + 1u + chunk_len) > context->length) {
77e0558b1bSMatthias Ringwald return false;
78e0558b1bSMatthias Ringwald }
792baf01d2SMatthias Ringwald return true;
80b12ad867SMatthias Ringwald }
81b12ad867SMatthias Ringwald
8288949f84SMatthias Ringwald // pre: ad_iterator_has_more() == 1
ad_iterator_next(ad_context_t * context)83b12ad867SMatthias Ringwald void ad_iterator_next(ad_context_t * context){
84e0558b1bSMatthias Ringwald uint8_t chunk_len = context->data[context->offset];
850dd3c6c3SMatthias Ringwald context->offset += 1u + chunk_len;
86b12ad867SMatthias Ringwald }
87b12ad867SMatthias Ringwald
ad_iterator_get_data_len(const ad_context_t * context)88b12ad867SMatthias Ringwald uint8_t ad_iterator_get_data_len(const ad_context_t * context){
890dd3c6c3SMatthias Ringwald return context->data[context->offset] - 1u;
90b12ad867SMatthias Ringwald }
91b12ad867SMatthias Ringwald
ad_iterator_get_data_type(const ad_context_t * context)92b12ad867SMatthias Ringwald uint8_t ad_iterator_get_data_type(const ad_context_t * context){
930dd3c6c3SMatthias Ringwald return context->data[context->offset + 1u];
94b12ad867SMatthias Ringwald }
95b12ad867SMatthias Ringwald
ad_iterator_get_data(const ad_context_t * context)96b12ad867SMatthias Ringwald const uint8_t * ad_iterator_get_data(const ad_context_t * context){
970dd3c6c3SMatthias Ringwald return &context->data[context->offset + 2u];
98b12ad867SMatthias Ringwald }
99b12ad867SMatthias Ringwald
ad_data_contains_uuid16(uint8_t ad_len,const uint8_t * ad_data,uint16_t uuid16)1002baf01d2SMatthias Ringwald bool ad_data_contains_uuid16(uint8_t ad_len, const uint8_t * ad_data, uint16_t uuid16){
101b12ad867SMatthias Ringwald ad_context_t context;
102fb08d3bfSMatthias Ringwald ad_iterator_init(&context, ad_len, ad_data);
103fb08d3bfSMatthias Ringwald while ( ad_iterator_has_more(&context) ){
104b12ad867SMatthias Ringwald uint8_t data_type = ad_iterator_get_data_type(&context);
105b12ad867SMatthias Ringwald uint8_t data_len = ad_iterator_get_data_len(&context);
106b12ad867SMatthias Ringwald const uint8_t * data = ad_iterator_get_data(&context);
107b12ad867SMatthias Ringwald
108e0558b1bSMatthias Ringwald uint8_t i;
109b12ad867SMatthias Ringwald uint8_t ad_uuid128[16], uuid128_bt[16];
110b12ad867SMatthias Ringwald
111b12ad867SMatthias Ringwald switch (data_type){
1121d0cde9dSMatthias Ringwald case BLUETOOTH_DATA_TYPE_INCOMPLETE_LIST_OF_16_BIT_SERVICE_CLASS_UUIDS:
1131d0cde9dSMatthias Ringwald case BLUETOOTH_DATA_TYPE_COMPLETE_LIST_OF_16_BIT_SERVICE_CLASS_UUIDS:
1140dd3c6c3SMatthias Ringwald for (i=0u; (i + 2u) <= data_len; i+= 2u){
115e0558b1bSMatthias Ringwald uint16_t uuid = (uint16_t) little_endian_read_16(data, (int) i);
116e0558b1bSMatthias Ringwald if ( uuid == uuid16 ) {
117e0558b1bSMatthias Ringwald return true;
118e0558b1bSMatthias Ringwald }
119b12ad867SMatthias Ringwald }
120b12ad867SMatthias Ringwald break;
1211d0cde9dSMatthias Ringwald case BLUETOOTH_DATA_TYPE_INCOMPLETE_LIST_OF_128_BIT_SERVICE_CLASS_UUIDS:
1221d0cde9dSMatthias Ringwald case BLUETOOTH_DATA_TYPE_COMPLETE_LIST_OF_128_BIT_SERVICE_CLASS_UUIDS:
123b12ad867SMatthias Ringwald uuid_add_bluetooth_prefix(ad_uuid128, uuid16);
124b12ad867SMatthias Ringwald reverse_128(ad_uuid128, uuid128_bt);
1250dd3c6c3SMatthias Ringwald for (i=0u; (i + 16u) <= data_len; i += 16u){
126e0558b1bSMatthias Ringwald if (memcmp(uuid128_bt, &data[i], 16) == 0){
127e0558b1bSMatthias Ringwald return true;
128e0558b1bSMatthias Ringwald };
129b12ad867SMatthias Ringwald }
130b12ad867SMatthias Ringwald break;
131b12ad867SMatthias Ringwald default:
132b12ad867SMatthias Ringwald break;
133b12ad867SMatthias Ringwald }
134fb08d3bfSMatthias Ringwald ad_iterator_next(&context);
135b12ad867SMatthias Ringwald }
1362baf01d2SMatthias Ringwald return false;
137b12ad867SMatthias Ringwald }
138b12ad867SMatthias Ringwald
ad_data_contains_uuid128(uint8_t ad_len,const uint8_t * ad_data,const uint8_t * uuid128)1392baf01d2SMatthias Ringwald bool ad_data_contains_uuid128(uint8_t ad_len, const uint8_t * ad_data, const uint8_t * uuid128){
140b12ad867SMatthias Ringwald ad_context_t context;
141b12ad867SMatthias Ringwald // input in big endian/network order, bluetooth data in little endian
142b12ad867SMatthias Ringwald uint8_t uuid128_le[16];
143b12ad867SMatthias Ringwald reverse_128(uuid128, uuid128_le);
144fb08d3bfSMatthias Ringwald ad_iterator_init(&context, ad_len, ad_data);
145fb08d3bfSMatthias Ringwald while ( ad_iterator_has_more(&context) ){
146b12ad867SMatthias Ringwald uint8_t data_type = ad_iterator_get_data_type(&context);
147b12ad867SMatthias Ringwald uint8_t data_len = ad_iterator_get_data_len(&context);
148b12ad867SMatthias Ringwald const uint8_t * data = ad_iterator_get_data(&context);
149b12ad867SMatthias Ringwald
150e0558b1bSMatthias Ringwald uint8_t i;
151b12ad867SMatthias Ringwald uint8_t ad_uuid128[16];
152b12ad867SMatthias Ringwald
153b12ad867SMatthias Ringwald switch (data_type){
1541d0cde9dSMatthias Ringwald case BLUETOOTH_DATA_TYPE_INCOMPLETE_LIST_OF_16_BIT_SERVICE_CLASS_UUIDS:
1551d0cde9dSMatthias Ringwald case BLUETOOTH_DATA_TYPE_COMPLETE_LIST_OF_16_BIT_SERVICE_CLASS_UUIDS:
1560dd3c6c3SMatthias Ringwald for (i = 0u; (i+2u) <= data_len; i += 2u){
157e0558b1bSMatthias Ringwald uint16_t uuid16 = little_endian_read_16(data, (int) i);
158b12ad867SMatthias Ringwald uuid_add_bluetooth_prefix(ad_uuid128, uuid16);
159b12ad867SMatthias Ringwald
160e0558b1bSMatthias Ringwald if (memcmp(ad_uuid128, uuid128_le, 16) == 0) {
161e0558b1bSMatthias Ringwald return true;
162e0558b1bSMatthias Ringwald }
163b12ad867SMatthias Ringwald }
164b12ad867SMatthias Ringwald
165b12ad867SMatthias Ringwald break;
1661d0cde9dSMatthias Ringwald case BLUETOOTH_DATA_TYPE_INCOMPLETE_LIST_OF_128_BIT_SERVICE_CLASS_UUIDS:
1671d0cde9dSMatthias Ringwald case BLUETOOTH_DATA_TYPE_COMPLETE_LIST_OF_128_BIT_SERVICE_CLASS_UUIDS:
1680dd3c6c3SMatthias Ringwald for (i = 0u; (i + 16u) <= data_len; i += 16u){
169e0558b1bSMatthias Ringwald if (memcmp(uuid128_le, &data[i], 16) == 0) {
170e0558b1bSMatthias Ringwald return true;
171e0558b1bSMatthias Ringwald }
172b12ad867SMatthias Ringwald }
173b12ad867SMatthias Ringwald break;
174b12ad867SMatthias Ringwald default:
175b12ad867SMatthias Ringwald break;
176b12ad867SMatthias Ringwald }
177fb08d3bfSMatthias Ringwald ad_iterator_next(&context);
178b12ad867SMatthias Ringwald }
1792baf01d2SMatthias Ringwald return false;
180b12ad867SMatthias Ringwald }
181b12ad867SMatthias Ringwald
182