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 23b12ad867SMatthias Ringwald * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL MATTHIAS 24b12ad867SMatthias Ringwald * RINGWALD 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" 52b12ad867SMatthias Ringwald #include "classic/sdp_util.h" 531d0cde9dSMatthias Ringwald #include "hci.h" 54b12ad867SMatthias Ringwald #include "hci_cmd.h" 55b12ad867SMatthias Ringwald 5666818fc6SMatthias Ringwald #include "ad_parser.h" 57b12ad867SMatthias Ringwald 58b12ad867SMatthias Ringwald void ad_iterator_init(ad_context_t *context, uint8_t ad_len, const uint8_t * ad_data){ 59b12ad867SMatthias Ringwald context->data = ad_data; 60b12ad867SMatthias Ringwald context->length = ad_len; 61b12ad867SMatthias Ringwald context->offset = 0; 62b12ad867SMatthias Ringwald } 63b12ad867SMatthias Ringwald 642baf01d2SMatthias Ringwald bool ad_iterator_has_more(const ad_context_t * context){ 6588949f84SMatthias Ringwald // assert chunk_len and chunk_type are withing buffer 66*0dd3c6c3SMatthias Ringwald if ((context->offset + 1u) >= context->length) { 67e0558b1bSMatthias Ringwald return false; 68e0558b1bSMatthias Ringwald } 6988949f84SMatthias Ringwald 7088949f84SMatthias Ringwald // assert chunk_len > 0 71e0558b1bSMatthias Ringwald uint8_t chunk_len = context->data[context->offset]; 72*0dd3c6c3SMatthias Ringwald if (chunk_len == 0u){ 73e0558b1bSMatthias Ringwald return false; 74e0558b1bSMatthias Ringwald } 7588949f84SMatthias Ringwald 7688949f84SMatthias Ringwald // assert complete chunk fits into buffer 77*0dd3c6c3SMatthias Ringwald if ((context->offset + 1u + chunk_len) > context->length) { 78e0558b1bSMatthias Ringwald return false; 79e0558b1bSMatthias Ringwald } 802baf01d2SMatthias Ringwald return true; 81b12ad867SMatthias Ringwald } 82b12ad867SMatthias Ringwald 8388949f84SMatthias Ringwald // pre: ad_iterator_has_more() == 1 84b12ad867SMatthias Ringwald void ad_iterator_next(ad_context_t * context){ 85e0558b1bSMatthias Ringwald uint8_t chunk_len = context->data[context->offset]; 86*0dd3c6c3SMatthias Ringwald context->offset += 1u + chunk_len; 87b12ad867SMatthias Ringwald } 88b12ad867SMatthias Ringwald 89b12ad867SMatthias Ringwald uint8_t ad_iterator_get_data_len(const ad_context_t * context){ 90*0dd3c6c3SMatthias Ringwald return context->data[context->offset] - 1u; 91b12ad867SMatthias Ringwald } 92b12ad867SMatthias Ringwald 93b12ad867SMatthias Ringwald uint8_t ad_iterator_get_data_type(const ad_context_t * context){ 94*0dd3c6c3SMatthias Ringwald return context->data[context->offset + 1u]; 95b12ad867SMatthias Ringwald } 96b12ad867SMatthias Ringwald 97b12ad867SMatthias Ringwald const uint8_t * ad_iterator_get_data(const ad_context_t * context){ 98*0dd3c6c3SMatthias Ringwald return &context->data[context->offset + 2u]; 99b12ad867SMatthias Ringwald } 100b12ad867SMatthias Ringwald 1012baf01d2SMatthias Ringwald bool ad_data_contains_uuid16(uint8_t ad_len, const uint8_t * ad_data, uint16_t uuid16){ 102b12ad867SMatthias Ringwald ad_context_t context; 103fb08d3bfSMatthias Ringwald ad_iterator_init(&context, ad_len, ad_data); 104fb08d3bfSMatthias Ringwald while ( ad_iterator_has_more(&context) ){ 105b12ad867SMatthias Ringwald uint8_t data_type = ad_iterator_get_data_type(&context); 106b12ad867SMatthias Ringwald uint8_t data_len = ad_iterator_get_data_len(&context); 107b12ad867SMatthias Ringwald const uint8_t * data = ad_iterator_get_data(&context); 108b12ad867SMatthias Ringwald 109e0558b1bSMatthias Ringwald uint8_t i; 110b12ad867SMatthias Ringwald uint8_t ad_uuid128[16], uuid128_bt[16]; 111b12ad867SMatthias Ringwald 112b12ad867SMatthias Ringwald switch (data_type){ 1131d0cde9dSMatthias Ringwald case BLUETOOTH_DATA_TYPE_INCOMPLETE_LIST_OF_16_BIT_SERVICE_CLASS_UUIDS: 1141d0cde9dSMatthias Ringwald case BLUETOOTH_DATA_TYPE_COMPLETE_LIST_OF_16_BIT_SERVICE_CLASS_UUIDS: 115*0dd3c6c3SMatthias Ringwald for (i=0u; (i + 2u) <= data_len; i+= 2u){ 116e0558b1bSMatthias Ringwald uint16_t uuid = (uint16_t) little_endian_read_16(data, (int) i); 117e0558b1bSMatthias Ringwald if ( uuid == uuid16 ) { 118e0558b1bSMatthias Ringwald return true; 119e0558b1bSMatthias Ringwald } 120b12ad867SMatthias Ringwald } 121b12ad867SMatthias Ringwald break; 1221d0cde9dSMatthias Ringwald case BLUETOOTH_DATA_TYPE_INCOMPLETE_LIST_OF_128_BIT_SERVICE_CLASS_UUIDS: 1231d0cde9dSMatthias Ringwald case BLUETOOTH_DATA_TYPE_COMPLETE_LIST_OF_128_BIT_SERVICE_CLASS_UUIDS: 124b12ad867SMatthias Ringwald uuid_add_bluetooth_prefix(ad_uuid128, uuid16); 125b12ad867SMatthias Ringwald reverse_128(ad_uuid128, uuid128_bt); 126*0dd3c6c3SMatthias Ringwald for (i=0u; (i + 16u) <= data_len; i += 16u){ 127e0558b1bSMatthias Ringwald if (memcmp(uuid128_bt, &data[i], 16) == 0){ 128e0558b1bSMatthias Ringwald return true; 129e0558b1bSMatthias Ringwald }; 130b12ad867SMatthias Ringwald } 131b12ad867SMatthias Ringwald break; 132b12ad867SMatthias Ringwald default: 133b12ad867SMatthias Ringwald break; 134b12ad867SMatthias Ringwald } 135fb08d3bfSMatthias Ringwald ad_iterator_next(&context); 136b12ad867SMatthias Ringwald } 1372baf01d2SMatthias Ringwald return false; 138b12ad867SMatthias Ringwald } 139b12ad867SMatthias Ringwald 1402baf01d2SMatthias Ringwald bool ad_data_contains_uuid128(uint8_t ad_len, const uint8_t * ad_data, const uint8_t * uuid128){ 141b12ad867SMatthias Ringwald ad_context_t context; 142b12ad867SMatthias Ringwald // input in big endian/network order, bluetooth data in little endian 143b12ad867SMatthias Ringwald uint8_t uuid128_le[16]; 144b12ad867SMatthias Ringwald reverse_128(uuid128, uuid128_le); 145fb08d3bfSMatthias Ringwald ad_iterator_init(&context, ad_len, ad_data); 146fb08d3bfSMatthias Ringwald while ( ad_iterator_has_more(&context) ){ 147b12ad867SMatthias Ringwald uint8_t data_type = ad_iterator_get_data_type(&context); 148b12ad867SMatthias Ringwald uint8_t data_len = ad_iterator_get_data_len(&context); 149b12ad867SMatthias Ringwald const uint8_t * data = ad_iterator_get_data(&context); 150b12ad867SMatthias Ringwald 151e0558b1bSMatthias Ringwald uint8_t i; 152b12ad867SMatthias Ringwald uint8_t ad_uuid128[16]; 153b12ad867SMatthias Ringwald 154b12ad867SMatthias Ringwald switch (data_type){ 1551d0cde9dSMatthias Ringwald case BLUETOOTH_DATA_TYPE_INCOMPLETE_LIST_OF_16_BIT_SERVICE_CLASS_UUIDS: 1561d0cde9dSMatthias Ringwald case BLUETOOTH_DATA_TYPE_COMPLETE_LIST_OF_16_BIT_SERVICE_CLASS_UUIDS: 157*0dd3c6c3SMatthias Ringwald for (i = 0u; (i+2u) <= data_len; i += 2u){ 158e0558b1bSMatthias Ringwald uint16_t uuid16 = little_endian_read_16(data, (int) i); 159b12ad867SMatthias Ringwald uuid_add_bluetooth_prefix(ad_uuid128, uuid16); 160b12ad867SMatthias Ringwald 161e0558b1bSMatthias Ringwald if (memcmp(ad_uuid128, uuid128_le, 16) == 0) { 162e0558b1bSMatthias Ringwald return true; 163e0558b1bSMatthias Ringwald } 164b12ad867SMatthias Ringwald } 165b12ad867SMatthias Ringwald 166b12ad867SMatthias Ringwald break; 1671d0cde9dSMatthias Ringwald case BLUETOOTH_DATA_TYPE_INCOMPLETE_LIST_OF_128_BIT_SERVICE_CLASS_UUIDS: 1681d0cde9dSMatthias Ringwald case BLUETOOTH_DATA_TYPE_COMPLETE_LIST_OF_128_BIT_SERVICE_CLASS_UUIDS: 169*0dd3c6c3SMatthias Ringwald for (i = 0u; (i + 16u) <= data_len; i += 16u){ 170e0558b1bSMatthias Ringwald if (memcmp(uuid128_le, &data[i], 16) == 0) { 171e0558b1bSMatthias Ringwald return true; 172e0558b1bSMatthias Ringwald } 173b12ad867SMatthias Ringwald } 174b12ad867SMatthias Ringwald break; 175b12ad867SMatthias Ringwald default: 176b12ad867SMatthias Ringwald break; 177b12ad867SMatthias Ringwald } 178fb08d3bfSMatthias Ringwald ad_iterator_next(&context); 179b12ad867SMatthias Ringwald } 1802baf01d2SMatthias Ringwald return false; 181b12ad867SMatthias Ringwald } 182b12ad867SMatthias Ringwald 183