xref: /btstack/src/ad_parser.c (revision b12ad86731289e1fe57e6b39dca956855fa22842)
1*b12ad867SMatthias Ringwald /*
2*b12ad867SMatthias Ringwald  * Copyright (C) 2014 BlueKitchen GmbH
3*b12ad867SMatthias Ringwald  *
4*b12ad867SMatthias Ringwald  * Redistribution and use in source and binary forms, with or without
5*b12ad867SMatthias Ringwald  * modification, are permitted provided that the following conditions
6*b12ad867SMatthias Ringwald  * are met:
7*b12ad867SMatthias Ringwald  *
8*b12ad867SMatthias Ringwald  * 1. Redistributions of source code must retain the above copyright
9*b12ad867SMatthias Ringwald  *    notice, this list of conditions and the following disclaimer.
10*b12ad867SMatthias Ringwald  * 2. Redistributions in binary form must reproduce the above copyright
11*b12ad867SMatthias Ringwald  *    notice, this list of conditions and the following disclaimer in the
12*b12ad867SMatthias Ringwald  *    documentation and/or other materials provided with the distribution.
13*b12ad867SMatthias Ringwald  * 3. Neither the name of the copyright holders nor the names of
14*b12ad867SMatthias Ringwald  *    contributors may be used to endorse or promote products derived
15*b12ad867SMatthias Ringwald  *    from this software without specific prior written permission.
16*b12ad867SMatthias Ringwald  * 4. Any redistribution, use, or modification is done solely for
17*b12ad867SMatthias Ringwald  *    personal benefit and not for any commercial purpose or for
18*b12ad867SMatthias Ringwald  *    monetary gain.
19*b12ad867SMatthias Ringwald  *
20*b12ad867SMatthias Ringwald  * THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS
21*b12ad867SMatthias Ringwald  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
22*b12ad867SMatthias Ringwald  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
23*b12ad867SMatthias Ringwald  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL MATTHIAS
24*b12ad867SMatthias Ringwald  * RINGWALD OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
25*b12ad867SMatthias Ringwald  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
26*b12ad867SMatthias Ringwald  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
27*b12ad867SMatthias Ringwald  * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
28*b12ad867SMatthias Ringwald  * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
29*b12ad867SMatthias Ringwald  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
30*b12ad867SMatthias Ringwald  * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
31*b12ad867SMatthias Ringwald  * SUCH DAMAGE.
32*b12ad867SMatthias Ringwald  *
33*b12ad867SMatthias Ringwald  * Please inquire about commercial licensing options at
34*b12ad867SMatthias Ringwald  * [email protected]
35*b12ad867SMatthias Ringwald  *
36*b12ad867SMatthias Ringwald  */
37*b12ad867SMatthias Ringwald 
38*b12ad867SMatthias Ringwald 
39*b12ad867SMatthias Ringwald // *****************************************************************************
40*b12ad867SMatthias Ringwald //
41*b12ad867SMatthias Ringwald // Advertising Data Parser
42*b12ad867SMatthias Ringwald //
43*b12ad867SMatthias Ringwald // *****************************************************************************
44*b12ad867SMatthias Ringwald 
45*b12ad867SMatthias Ringwald #include <stdint.h>
46*b12ad867SMatthias Ringwald #include <stdio.h>
47*b12ad867SMatthias Ringwald #include <stdlib.h>
48*b12ad867SMatthias Ringwald #include <string.h>
49*b12ad867SMatthias Ringwald 
50*b12ad867SMatthias Ringwald #include "btstack_util.h"
51*b12ad867SMatthias Ringwald #include "classic/sdp_util.h"
52*b12ad867SMatthias Ringwald #include "hci_cmd.h"
53*b12ad867SMatthias Ringwald 
54*b12ad867SMatthias Ringwald #include "hci.h"
55*b12ad867SMatthias Ringwald #include "ble/ad_parser.h"
56*b12ad867SMatthias Ringwald 
57*b12ad867SMatthias Ringwald typedef enum {
58*b12ad867SMatthias Ringwald     IncompleteList16 = 0x02,
59*b12ad867SMatthias Ringwald     CompleteList16 = 0x03,
60*b12ad867SMatthias Ringwald     IncompleteList128 = 0x06,
61*b12ad867SMatthias Ringwald     CompleteList128 = 0x07
62*b12ad867SMatthias Ringwald } UUID_TYPE;
63*b12ad867SMatthias Ringwald 
64*b12ad867SMatthias Ringwald void ad_iterator_init(ad_context_t *context, uint8_t ad_len, const uint8_t * ad_data){
65*b12ad867SMatthias Ringwald     context->data = ad_data;
66*b12ad867SMatthias Ringwald     context->length = ad_len;
67*b12ad867SMatthias Ringwald     context->offset = 0;
68*b12ad867SMatthias Ringwald }
69*b12ad867SMatthias Ringwald 
70*b12ad867SMatthias Ringwald int  ad_iterator_has_more(const ad_context_t * context){
71*b12ad867SMatthias Ringwald     return context->offset < context->length;
72*b12ad867SMatthias Ringwald }
73*b12ad867SMatthias Ringwald 
74*b12ad867SMatthias Ringwald void ad_iterator_next(ad_context_t * context){
75*b12ad867SMatthias Ringwald     int chunk_len = context->data[context->offset];
76*b12ad867SMatthias Ringwald     int new_offset = context->offset + 1 + chunk_len;
77*b12ad867SMatthias Ringwald     // avoid uint8_t overrun
78*b12ad867SMatthias Ringwald     if (new_offset > 0xff){
79*b12ad867SMatthias Ringwald         new_offset = 0xff;
80*b12ad867SMatthias Ringwald     }
81*b12ad867SMatthias Ringwald     context->offset = new_offset;
82*b12ad867SMatthias Ringwald }
83*b12ad867SMatthias Ringwald 
84*b12ad867SMatthias Ringwald uint8_t   ad_iterator_get_data_len(const ad_context_t * context){
85*b12ad867SMatthias Ringwald     return context->data[context->offset] - 1;
86*b12ad867SMatthias Ringwald }
87*b12ad867SMatthias Ringwald 
88*b12ad867SMatthias Ringwald uint8_t   ad_iterator_get_data_type(const ad_context_t * context){
89*b12ad867SMatthias Ringwald     return context->data[context->offset + 1];
90*b12ad867SMatthias Ringwald }
91*b12ad867SMatthias Ringwald 
92*b12ad867SMatthias Ringwald const uint8_t * ad_iterator_get_data(const ad_context_t * context){
93*b12ad867SMatthias Ringwald     return &context->data[context->offset + 2];
94*b12ad867SMatthias Ringwald }
95*b12ad867SMatthias Ringwald 
96*b12ad867SMatthias Ringwald int ad_data_contains_uuid16(uint8_t ad_len, const uint8_t * ad_data, uint16_t uuid16){
97*b12ad867SMatthias Ringwald     ad_context_t context;
98*b12ad867SMatthias Ringwald     for (ad_iterator_init(&context, ad_len, ad_data) ; ad_iterator_has_more(&context) ; ad_iterator_next(&context)){
99*b12ad867SMatthias Ringwald         uint8_t data_type    = ad_iterator_get_data_type(&context);
100*b12ad867SMatthias Ringwald         uint8_t data_len     = ad_iterator_get_data_len(&context);
101*b12ad867SMatthias Ringwald         const uint8_t * data = ad_iterator_get_data(&context);
102*b12ad867SMatthias Ringwald 
103*b12ad867SMatthias Ringwald         int i;
104*b12ad867SMatthias Ringwald         uint8_t ad_uuid128[16], uuid128_bt[16];
105*b12ad867SMatthias Ringwald 
106*b12ad867SMatthias Ringwald         switch (data_type){
107*b12ad867SMatthias Ringwald             case IncompleteList16:
108*b12ad867SMatthias Ringwald             case CompleteList16:
109*b12ad867SMatthias Ringwald                 for (i=0; i<data_len; i+=2){
110*b12ad867SMatthias Ringwald                     uint16_t uuid = little_endian_read_16(data, i);
111*b12ad867SMatthias Ringwald                     if ( uuid == uuid16 ) return 1;
112*b12ad867SMatthias Ringwald                 }
113*b12ad867SMatthias Ringwald                 break;
114*b12ad867SMatthias Ringwald             case IncompleteList128:
115*b12ad867SMatthias Ringwald             case CompleteList128:
116*b12ad867SMatthias Ringwald                 uuid_add_bluetooth_prefix(ad_uuid128, uuid16);
117*b12ad867SMatthias Ringwald                 reverse_128(ad_uuid128, uuid128_bt);
118*b12ad867SMatthias Ringwald 
119*b12ad867SMatthias Ringwald                 for (i=0; i<data_len; i+=16){
120*b12ad867SMatthias Ringwald                     if (memcmp(uuid128_bt, &data[i], 16) == 0) return 1;
121*b12ad867SMatthias Ringwald                 }
122*b12ad867SMatthias Ringwald                 break;
123*b12ad867SMatthias Ringwald             default:
124*b12ad867SMatthias Ringwald                 break;
125*b12ad867SMatthias Ringwald         }
126*b12ad867SMatthias Ringwald     }
127*b12ad867SMatthias Ringwald     return 0;
128*b12ad867SMatthias Ringwald }
129*b12ad867SMatthias Ringwald 
130*b12ad867SMatthias Ringwald int ad_data_contains_uuid128(uint8_t ad_len, const uint8_t * ad_data, const uint8_t * uuid128){
131*b12ad867SMatthias Ringwald     ad_context_t context;
132*b12ad867SMatthias Ringwald     // input in big endian/network order, bluetooth data in little endian
133*b12ad867SMatthias Ringwald     uint8_t uuid128_le[16];
134*b12ad867SMatthias Ringwald     reverse_128(uuid128, uuid128_le);
135*b12ad867SMatthias Ringwald     for (ad_iterator_init(&context, ad_len, ad_data) ; ad_iterator_has_more(&context) ; ad_iterator_next(&context)){
136*b12ad867SMatthias Ringwald         uint8_t data_type = ad_iterator_get_data_type(&context);
137*b12ad867SMatthias Ringwald         uint8_t data_len  = ad_iterator_get_data_len(&context);
138*b12ad867SMatthias Ringwald         const uint8_t * data = ad_iterator_get_data(&context);
139*b12ad867SMatthias Ringwald 
140*b12ad867SMatthias Ringwald         int i;
141*b12ad867SMatthias Ringwald         uint8_t ad_uuid128[16];
142*b12ad867SMatthias Ringwald 
143*b12ad867SMatthias Ringwald 
144*b12ad867SMatthias Ringwald         switch (data_type){
145*b12ad867SMatthias Ringwald             case IncompleteList16:
146*b12ad867SMatthias Ringwald             case CompleteList16:
147*b12ad867SMatthias Ringwald                 for (i=0; i<data_len; i+=2){
148*b12ad867SMatthias Ringwald                     uint16_t uuid16 = little_endian_read_16(data, i);
149*b12ad867SMatthias Ringwald                     uuid_add_bluetooth_prefix(ad_uuid128, uuid16);
150*b12ad867SMatthias Ringwald 
151*b12ad867SMatthias Ringwald                     if (memcmp(ad_uuid128, uuid128_le, 16) == 0) return 1;
152*b12ad867SMatthias Ringwald                 }
153*b12ad867SMatthias Ringwald 
154*b12ad867SMatthias Ringwald                 break;
155*b12ad867SMatthias Ringwald             case IncompleteList128:
156*b12ad867SMatthias Ringwald             case CompleteList128:
157*b12ad867SMatthias Ringwald                 for (i=0; i<data_len; i+=16){
158*b12ad867SMatthias Ringwald                     if (memcmp(uuid128_le, &data[i], 16) == 0) return 1;
159*b12ad867SMatthias Ringwald                 }
160*b12ad867SMatthias Ringwald                 break;
161*b12ad867SMatthias Ringwald             default:
162*b12ad867SMatthias Ringwald                 break;
163*b12ad867SMatthias Ringwald         }
164*b12ad867SMatthias Ringwald     }
165*b12ad867SMatthias Ringwald     return 0;
166*b12ad867SMatthias Ringwald }
167*b12ad867SMatthias Ringwald 
168