//
// Copyright (C) 2015 Google, Inc.
//
// 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 "service/common/bluetooth/uuid.h"
#include <algorithm>
#include <array>
#include <stack>
#include <string>
#include <base/rand_util.h>
#include <base/strings/stringprintf.h>
#include <base/strings/string_split.h>
#include <base/strings/string_util.h>
namespace bluetooth {
namespace {
const UUID::UUID128Bit kSigBaseUUID = {
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10, 0x00,
0x80, 0x00, 0x00, 0x80, 0x5f, 0x9b, 0x34, 0xfb }
};
} // namespace
// static
UUID UUID::GetRandom() {
UUID128Bit bytes;
base::RandBytes(bytes.data(), bytes.size());
return UUID(bytes);
}
// static
UUID UUID::GetNil() {
UUID128Bit bytes;
bytes.fill(0);
return UUID(bytes);
}
// static
UUID UUID::GetMax() {
UUID128Bit bytes;
bytes.fill(1);
return UUID(bytes);
}
void UUID::InitializeDefault() {
// Initialize to Bluetooth SIG base UUID.
id_ = kSigBaseUUID;
is_valid_ = true;
}
UUID::UUID() {
InitializeDefault();
}
UUID::UUID(std::string uuid) {
InitializeDefault();
is_valid_ = false;
if (uuid.empty())
return;
if (uuid.size() < 11 && uuid.find("0x") == 0)
uuid = uuid.substr(2);
if (uuid.size() != 4 && uuid.size() != 8 && uuid.size() != 36)
return;
if (uuid.size() == 36) {
if (uuid[8] != '-')
return;
if (uuid[13] != '-')
return;
if (uuid[18] != '-')
return;
if (uuid[23] != '-')
return;
std::vector<std::string> tokens = base::SplitString(
uuid, "-", base::TRIM_WHITESPACE, base::SPLIT_WANT_ALL);
if (tokens.size() != 5)
return;
uuid = base::JoinString(tokens, "");
}
const int start_index = uuid.size() == 4 ? 2 : 0;
const size_t copy_size = std::min(id_.size(), uuid.size() / 2);
for (size_t i = 0; i < copy_size; ++i) {
std::string octet_text(uuid, i * 2, 2);
char* temp = nullptr;
id_[start_index + i] = strtol(octet_text.c_str(), &temp, 16);
if (*temp != '\0')
return;
}
is_valid_ = true;
}
UUID::UUID(const bt_uuid_t& uuid) {
std::reverse_copy(uuid.uu, uuid.uu + sizeof(uuid.uu), id_.begin());
is_valid_ = true;
}
UUID::UUID(const UUID16Bit& uuid) {
InitializeDefault();
std::copy(uuid.begin(), uuid.end(), id_.begin() + kNumBytes16);
}
UUID::UUID(const UUID32Bit& uuid) {
InitializeDefault();
std::copy(uuid.begin(), uuid.end(), id_.begin());
}
UUID::UUID(const UUID128Bit& uuid) : id_(uuid), is_valid_(true) {}
UUID::UUID128Bit UUID::GetFullBigEndian() const {
return id_;
}
UUID::UUID128Bit UUID::GetFullLittleEndian() const {
UUID::UUID128Bit ret;
std::reverse_copy(id_.begin(), id_.end(), ret.begin());
return ret;
}
bt_uuid_t UUID::GetBlueDroid() const {
bt_uuid_t ret;
std::reverse_copy(id_.begin(), id_.end(), ret.uu);
return ret;
}
std::string UUID::ToString() const {
return base::StringPrintf(
"%02x%02x%02x%02x-%02x%02x-%02x%02x-%02x%02x-%02x%02x%02x%02x%02x%02x",
id_[0], id_[1], id_[2], id_[3],
id_[4], id_[5], id_[6], id_[7],
id_[8], id_[9], id_[10], id_[11],
id_[12], id_[13], id_[14], id_[15]);
}
size_t UUID::GetShortestRepresentationSize() const {
if (memcmp(id_.data() + 4, kSigBaseUUID.data() + 4, id_.size() - 4) != 0)
return kNumBytes128;
if (id_[0] == 0 && id_[1] == 0)
return kNumBytes16;
return kNumBytes32;
}
bool UUID::operator<(const UUID& rhs) const {
return std::lexicographical_compare(id_.begin(), id_.end(), rhs.id_.begin(),
rhs.id_.end());
}
bool UUID::operator==(const UUID& rhs) const {
return std::equal(id_.begin(), id_.end(), rhs.id_.begin());
}
} // namespace bluetooth