blob: 551bc2b9eb4da5449b4ed8d1393a7a34b2bf727f [file] [log] [blame]
/******************************************************************************
*
* Copyright (C) 2017 The Android Open Source Project
*
* Licensed 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 "hci/uuid.h"
#include <openssl/rand.h>
#include <string.h>
#include <algorithm>
namespace bluetooth {
namespace hci {
using UUID128Bit = Uuid::UUID128Bit;
const Uuid Uuid::kEmpty = Uuid::From128BitBE(UUID128Bit{{0x00}});
namespace {
Uuid kBase = Uuid::From128BitBE(
UUID128Bit{{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10, 0x00, 0x80, 0x00, 0x00, 0x80, 0x5f, 0x9b, 0x34, 0xfb}});
} // namespace
size_t Uuid::GetShortestRepresentationSize() const {
if (memcmp(uu.data() + kNumBytes32, kBase.uu.data() + kNumBytes32, kNumBytes128 - kNumBytes32) != 0) {
return kNumBytes128;
}
if (uu[0] == 0 && uu[1] == 0) {
return kNumBytes16;
}
return kNumBytes32;
}
bool Uuid::Is16Bit() const {
return GetShortestRepresentationSize() == kNumBytes16;
}
uint16_t Uuid::As16Bit() const {
return (((uint16_t)uu[2]) << 8) + uu[3];
}
uint32_t Uuid::As32Bit() const {
return (((uint32_t)uu[0]) << 24) + (((uint32_t)uu[1]) << 16) + (((uint32_t)uu[2]) << 8) + uu[3];
}
std::optional<Uuid> Uuid::FromString(const std::string& uuid) {
if (uuid.empty()) {
return std::nullopt;
}
Uuid ret = kBase;
uint8_t* p = ret.uu.data();
if (uuid.size() == kString128BitLen) {
if (uuid[8] != '-' || uuid[13] != '-' || uuid[18] != '-' || uuid[23] != '-') {
return std::nullopt;
}
int c;
int rc = sscanf(
uuid.c_str(),
"%02hhx%02hhx%02hhx%02hhx-%02hhx%02hhx-%02hhx%02hhx"
"-%02hhx%02hhx-%02hhx%02hhx%02hhx%02hhx%02hhx%02hhx%n",
&p[0],
&p[1],
&p[2],
&p[3],
&p[4],
&p[5],
&p[6],
&p[7],
&p[8],
&p[9],
&p[10],
&p[11],
&p[12],
&p[13],
&p[14],
&p[15],
&c);
if (rc != 16) {
return std::nullopt;
}
if (c != kString128BitLen) {
return std::nullopt;
}
} else if (uuid.size() == 8) {
int c;
int rc = sscanf(uuid.c_str(), "%02hhx%02hhx%02hhx%02hhx%n", &p[0], &p[1], &p[2], &p[3], &c);
if (rc != 4) {
return std::nullopt;
}
if (c != 8) {
return std::nullopt;
}
} else if (uuid.size() == 4) {
int c;
int rc = sscanf(uuid.c_str(), "%02hhx%02hhx%n", &p[2], &p[3], &c);
if (rc != 2) {
return std::nullopt;
}
if (c != 4) {
return std::nullopt;
}
} else {
return std::nullopt;
}
return ret;
}
std::optional<Uuid> Uuid::FromLegacyConfigString(const std::string& uuid) {
return FromString(uuid);
}
Uuid Uuid::From16Bit(uint16_t uuid16) {
Uuid u = kBase;
u.uu[2] = (uint8_t)((0xFF00 & uuid16) >> 8);
u.uu[3] = (uint8_t)(0x00FF & uuid16);
return u;
}
Uuid Uuid::From32Bit(uint32_t uuid32) {
Uuid u = kBase;
u.uu[0] = (uint8_t)((0xFF000000 & uuid32) >> 24);
u.uu[1] = (uint8_t)((0x00FF0000 & uuid32) >> 16);
u.uu[2] = (uint8_t)((0x0000FF00 & uuid32) >> 8);
u.uu[3] = (uint8_t)(0x000000FF & uuid32);
return u;
}
Uuid Uuid::From128BitBE(const uint8_t* uuid) {
UUID128Bit tmp;
memcpy(tmp.data(), uuid, kNumBytes128);
return From128BitBE(tmp);
}
Uuid Uuid::From128BitLE(const UUID128Bit& uuid) {
Uuid u;
std::reverse_copy(uuid.data(), uuid.data() + kNumBytes128, u.uu.begin());
return u;
}
Uuid Uuid::From128BitLE(const uint8_t* uuid) {
UUID128Bit tmp;
memcpy(tmp.data(), uuid, kNumBytes128);
return From128BitLE(tmp);
}
UUID128Bit Uuid::To128BitLE() const {
UUID128Bit le;
std::reverse_copy(uu.data(), uu.data() + kNumBytes128, le.begin());
return le;
}
const UUID128Bit& Uuid::To128BitBE() const {
return uu;
}
Uuid Uuid::GetRandom() {
Uuid uuid;
RAND_bytes(uuid.uu.data(), uuid.uu.size());
return uuid;
}
bool Uuid::IsEmpty() const {
return *this == kEmpty;
}
bool Uuid::operator<(const Uuid& rhs) const {
return std::lexicographical_compare(uu.begin(), uu.end(), rhs.uu.begin(), rhs.uu.end());
}
bool Uuid::operator==(const Uuid& rhs) const {
return uu == rhs.uu;
}
bool Uuid::operator!=(const Uuid& rhs) const {
return uu != rhs.uu;
}
std::string Uuid::ToString() const {
char buf[kString128BitLen + 1] = {};
std::snprintf(
buf,
sizeof(buf),
"%02x%02x%02x%02x-%02x%02x-%02x%02x-%02x%02x-%02x%02x%02x%02x%02x%02x",
uu[0],
uu[1],
uu[2],
uu[3],
uu[4],
uu[5],
uu[6],
uu[7],
uu[8],
uu[9],
uu[10],
uu[11],
uu[12],
uu[13],
uu[14],
uu[15]);
return std::string(buf);
}
std::string Uuid::ToLegacyConfigString() const {
return ToString();
}
} // namespace hci
} // namespace bluetooth