blob: 15f66374804d3a66f76d2a00839b21e402d61088 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
#include <inttypes.h>
#include <stdio.h>
#include <string.h>
#include <errno.h>
#include "os/os_mbuf.h"
#include "nimble/ble.h"
#include "ble_hs_priv.h"
#include "host/ble_uuid.h"
#define BLE_UUID16_STR_MAX_LEN 6
#define BLE_UUID32_STR_MAX_LEN 10
#define BLE_UUID128_STR_MAX_LEN 36
static uint8_t ble_uuid_base[16] = {
0xfb, 0x34, 0x9b, 0x5f, 0x80, 0x00, 0x00, 0x80,
0x00, 0x10, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00
};
#if MYNEWT_VAL(BLE_HS_DEBUG)
#define VERIFY_UUID(uuid) \
assert((uuid->type == BLE_UUID_TYPE_16) || \
(uuid->type == BLE_UUID_TYPE_32) || \
(uuid->type == BLE_UUID_TYPE_128))
#else
#define VERIFY_UUID(uuid)
#endif
static int
hex2val(char c, uint8_t *value)
{
if (c >= '0' && c <= '9') {
*value = c - '0';
} else if (c >= 'a' && c <= 'f') {
*value = c - 'a' + 10;
} else if (c >= 'A' && c <= 'F') {
*value = c - 'A' + 10;
} else {
return BLE_HS_EINVAL;
}
return 0;
}
static size_t
hex2bin(const char *hex, uint8_t *bin, size_t bin_len)
{
size_t len = 0;
uint8_t tmp_val;
int rc;
while (*hex && len < bin_len) {
rc = hex2val(*hex++, &tmp_val);
if (rc != 0) {
return 0;
}
bin[len] = tmp_val << 4;
if (!*hex) {
len++;
break;
}
rc = hex2val(*hex++, &tmp_val);
if (rc != 0) {
return 0;
}
bin[len++] |= tmp_val;
}
return len;
}
int
ble_uuid_init_from_buf(ble_uuid_any_t *uuid, const void *buf, size_t len)
{
switch (len) {
case 2:
uuid->u.type = BLE_UUID_TYPE_16;
uuid->u16.value = get_le16(buf);
return 0;
case 4:
uuid->u.type = BLE_UUID_TYPE_32;
uuid->u32.value = get_le32(buf);
return 0;
case 16:
uuid->u.type = BLE_UUID_TYPE_128;
memcpy(uuid->u128.value, buf, 16);
return 0;
}
return BLE_HS_EINVAL;
}
int
ble_uuid_cmp(const ble_uuid_t *uuid1, const ble_uuid_t *uuid2)
{
VERIFY_UUID(uuid1);
VERIFY_UUID(uuid2);
if (uuid1->type != uuid2->type) {
return uuid1->type - uuid2->type;
}
switch (uuid1->type) {
case BLE_UUID_TYPE_16:
return (int) BLE_UUID16(uuid1)->value - (int) BLE_UUID16(uuid2)->value;
case BLE_UUID_TYPE_32:
return (int) BLE_UUID32(uuid1)->value - (int) BLE_UUID32(uuid2)->value;
case BLE_UUID_TYPE_128:
return memcmp(BLE_UUID128(uuid1)->value, BLE_UUID128(uuid2)->value, 16);
}
BLE_HS_DBG_ASSERT(0);
return -1;
}
void
ble_uuid_copy(ble_uuid_any_t *dst, const ble_uuid_t *src)
{
VERIFY_UUID(src);
switch (src->type) {
case BLE_UUID_TYPE_16:
dst->u16 = *(const ble_uuid16_t *)src;
break;
case BLE_UUID_TYPE_32:
dst->u32 = *(const ble_uuid32_t *)src;
break;
case BLE_UUID_TYPE_128:
dst->u128 = *(const ble_uuid128_t *)src;
break;
default:
BLE_HS_DBG_ASSERT(0);
break;
}
}
char *
ble_uuid_to_str(const ble_uuid_t *uuid, char *dst)
{
const uint8_t *u8p;
switch (uuid->type) {
case BLE_UUID_TYPE_16:
sprintf(dst, "0x%04" PRIx16, BLE_UUID16(uuid)->value);
break;
case BLE_UUID_TYPE_32:
sprintf(dst, "0x%08" PRIx32, BLE_UUID32(uuid)->value);
break;
case BLE_UUID_TYPE_128:
u8p = BLE_UUID128(uuid)->value;
sprintf(dst, "%02x%02x%02x%02x-%02x%02x-%02x%02x-%02x%02x-"
"%02x%02x%02x%02x%02x%02x",
u8p[15], u8p[14], u8p[13], u8p[12],
u8p[11], u8p[10], u8p[9], u8p[8],
u8p[7], u8p[6], u8p[5], u8p[4],
u8p[3], u8p[2], u8p[1], u8p[0]);
break;
default:
dst[0] = '\0';
break;
}
return dst;
}
int
ble_uuid_from_str(ble_uuid_any_t *uuid, const char *str)
{
uint8_t tmp_rslt = 0;
uint8_t *u8p;
const char *str_ptr;
uint16_t u16 = 0;
uint32_t u32 = 0;
int len = (int) strlen(str);
if ((len < 4) || (len % 2 != 0)) {
return BLE_HS_EINVAL;
}
str_ptr = &str[len - 2];
if (len <= BLE_UUID16_STR_MAX_LEN) {
uuid->u.type = BLE_UUID_TYPE_16;
} else if (len <= BLE_UUID32_STR_MAX_LEN) {
uuid->u.type = BLE_UUID_TYPE_32;
} else if (len <= BLE_UUID128_STR_MAX_LEN) {
uuid->u.type = BLE_UUID_TYPE_128;
} else {
return BLE_HS_EINVAL;
}
switch (uuid->u.type) {
case BLE_UUID_TYPE_128:
uuid->u.type = BLE_UUID_TYPE_128;
u8p = uuid->u128.value;
for (int i = 0; i < 16; i++) {
if (hex2bin(str_ptr, u8p, 1) != 1) {
return BLE_HS_EINVAL;
}
/* Check if string end */
if (str_ptr == str) {
break;
}
/* Remove '-' */
if (*(str_ptr - 1) == '-') {
str_ptr--;
}
str_ptr -= 2;
u8p++;
}
if (memcmp(ble_uuid_base, uuid->u128.value, 12) == 0) {
uint8_t *tmp_ptr = &uuid->u128.value[12];
uint32_t tmp_val32 = 0;
for (int i = 0; i < 4; i++) {
tmp_val32 |= ((uint32_t)(*tmp_ptr++) << 8 * i);
}
if (tmp_val32 <= UINT16_MAX) {
uuid->u.type = BLE_UUID_TYPE_16;
uuid->u16.value = (uint16_t) tmp_val32;
} else {
uuid->u.type = BLE_UUID_TYPE_32;
uuid->u32.value = tmp_val32;
}
}
break;
case BLE_UUID_TYPE_32:
for (int i = 0; i < 4; i++) {
if (hex2bin(str_ptr, &tmp_rslt, 1) != 1) {
return BLE_HS_EINVAL;
}
u32 |= ((uint32_t) tmp_rslt) << (i * 8);
if (str_ptr == str) {
break;
}
str_ptr -= 2;
}
uuid->u32.value = u32;
break;
case BLE_UUID_TYPE_16:
for (int i = 0; i < 2; i++) {
if (hex2bin(str_ptr, &tmp_rslt, 1) != 1) {
return BLE_HS_EINVAL;
}
u16 |= ((uint32_t) tmp_rslt) << (i * 8);
if (str_ptr == str) {
break;
}
str_ptr -= 2;
}
uuid->u16.value = u16;
break;
default:
return BLE_HS_EINVAL;
}
return 0;
}
uint16_t
ble_uuid_u16(const ble_uuid_t *uuid)
{
VERIFY_UUID(uuid);
return uuid->type == BLE_UUID_TYPE_16 ? BLE_UUID16(uuid)->value : 0;
}
/* APIs below are private (ble_uuid_priv.h) */
int
ble_uuid_init_from_att_mbuf(ble_uuid_any_t *uuid, struct os_mbuf *om, int off,
int len)
{
uint8_t val[16];
int rc;
rc = os_mbuf_copydata(om, off, len, val);
if (rc != 0) {
return rc;
}
rc = ble_uuid_init_from_att_buf(uuid, val, len);
return rc;
}
int
ble_uuid_init_from_att_buf(ble_uuid_any_t *uuid, const void *buf, size_t len)
{
int rc = 0;
if (len == 2) {
uuid->u.type = BLE_UUID_TYPE_16;
uuid->u16.value = get_le16(buf);
} else if (len == 16) {
uuid->u.type = BLE_UUID_TYPE_128;
memcpy(uuid->u128.value, buf, 16);
} else {
rc = BLE_HS_EINVAL;
}
return rc;
}
int
ble_uuid_to_any(const ble_uuid_t *uuid, ble_uuid_any_t *uuid_any)
{
VERIFY_UUID(uuid);
uuid_any->u.type = uuid->type;
switch (uuid->type) {
case BLE_UUID_TYPE_16:
uuid_any->u16.value = BLE_UUID16(uuid)->value;
break;
case BLE_UUID_TYPE_32:
uuid_any->u32.value = BLE_UUID32(uuid)->value;
break;
case BLE_UUID_TYPE_128:
memcpy(uuid_any->u128.value, BLE_UUID128(uuid)->value, 16);
break;
default:
return BLE_HS_EINVAL;
}
return 0;
}
int
ble_uuid_to_mbuf(const ble_uuid_t *uuid, struct os_mbuf *om)
{
int len;
void *buf;
VERIFY_UUID(uuid);
len = ble_uuid_length(uuid);
buf = os_mbuf_extend(om, len);
if (buf == NULL) {
return BLE_HS_ENOMEM;
}
ble_uuid_flat(uuid, buf);
return 0;
}
int
ble_uuid_flat(const ble_uuid_t *uuid, void *dst)
{
VERIFY_UUID(uuid);
switch (uuid->type) {
case BLE_UUID_TYPE_16:
put_le16(dst, BLE_UUID16(uuid)->value);
break;
case BLE_UUID_TYPE_32:
memcpy(dst, ble_uuid_base, 16);
put_le32(dst + 12, BLE_UUID32(uuid)->value);
break;
case BLE_UUID_TYPE_128:
memcpy(dst, BLE_UUID128(uuid)->value, 16);
break;
default:
return BLE_HS_EINVAL;
}
return 0;
}
int
ble_uuid_length(const ble_uuid_t *uuid)
{
VERIFY_UUID(uuid);
return uuid->type >> 3;
}