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