Files
MuditaOS/module-bluetooth/Bluetooth/BtKeysStorage.cpp
SP2FET f9b80ac26c [EGD-5764] Implement handling multiple BT profiles
To be able to connect to the BT device with more than one audio
profile, we need something to manage those profiles. Created here
ProfileManager made it convenient
To be able to work properly with BT, CPU freq has to be at least
at Level_5 - to be fixed!
2021-02-25 14:03:44 +01:00

156 lines
5.5 KiB
C++

// Copyright (c) 2017-2020, Mudita Sp. z.o.o. All rights reserved.
// For licensing, see https://github.com/mudita/MuditaOS/LICENSE.md
#include <algorithm>
#include "BtKeysStorage.hpp"
#include <log/log.hpp>
json11::Json bluetooth::KeyStorage::keysJson = json11::Json();
btstack_link_key_db_t bluetooth::KeyStorage::keyStorage;
json11::Json::array bluetooth::KeyStorage::keys;
std::string bluetooth::KeyStorage::keysEntry;
std::shared_ptr<bluetooth::SettingsHolder> bluetooth::KeyStorage::settings = nullptr;
namespace bluetooth
{
namespace strings
{
constexpr inline auto keys = "keys";
constexpr inline auto link_key = "link_key";
constexpr inline auto bd_addr = "bd_addr";
constexpr inline auto type = "type";
} // namespace strings
auto KeyStorage::getKeyStorage() -> btstack_link_key_db_t *
{
keyStorage.open = openStorage;
keyStorage.set_local_bd_addr = set_local_bd_addr;
keyStorage.close = closeStorage;
keyStorage.get_link_key = getLinkKey;
keyStorage.put_link_key = putLinkKey;
keyStorage.delete_link_key = deleteLinkKey;
keyStorage.iterator_init = iterator_init;
keyStorage.iterator_get_next = iterator_get_next;
keyStorage.iterator_done = iterator_done;
return &keyStorage;
}
void KeyStorage::openStorage()
{
LOG_INFO("opening storage from API");
if (settings) {
keysEntry = std::visit(bluetooth::StringVisitor(), settings->getValue(bluetooth::Settings::BtKeys));
}
else {
LOG_ERROR("failed opening settings for BT!");
return;
}
if (keysEntry.empty()) {
LOG_WARN("opening empty key entry!");
return;
}
std::string err;
keysJson = json11::Json::parse(keysEntry.c_str(), err);
if (!err.empty()) {
LOG_ERROR("Error while parsing json: %s", err.c_str());
return;
}
keys = std::move(keysJson[strings::keys].array_items());
LOG_INFO("Imported keys: %d", static_cast<unsigned int>(keys.size()));
}
void KeyStorage::closeStorage()
{
LOG_INFO("closing storage from API");
writeSettings();
}
//
auto KeyStorage::getLinkKey(uint8_t *bd_addr, link_key_t link_key, link_key_type_t *type) -> int
{
if (type != nullptr && bd_addr != nullptr) {
LOG_INFO("getting key for address %s from API", bd_addr_to_str(bd_addr));
if (keys.empty()) {
LOG_ERROR("Keys empty!");
return 0;
}
for (auto key : keys) {
bd_addr_t addr;
sscanf_bd_addr(key[strings::bd_addr].string_value().c_str(), addr);
if (bd_addr_cmp(addr, bd_addr) == 0) {
auto foundLinkKey = key[strings::link_key].string_value().c_str();
memcpy(link_key, reinterpret_cast<const uint8_t *>(foundLinkKey), sizeof(link_key_t));
LOG_INFO("Getting key: %s", foundLinkKey);
*type = static_cast<link_key_type_t>(key[strings::type].int_value());
return 1;
}
LOG_ERROR("Can't find key for this address!");
}
}
return 0;
}
void KeyStorage::putLinkKey(uint8_t *bd_addr, uint8_t *link_key, link_key_type_t type)
{
LOG_INFO("putting key for address %s from API", bd_addr_to_str(bd_addr));
auto keyEntry = json11::Json::object{{strings::bd_addr, bd_addr_to_str(bd_addr)},
{strings::link_key, std::string(reinterpret_cast<char *>(link_key))},
{strings::type, type}};
keys.emplace_back(keyEntry);
if (settings->onLinkKeyAdded) {
settings->onLinkKeyAdded(bd_addr_to_str(bd_addr));
}
writeSettings();
LOG_INFO("keys written to the file");
LOG_INFO("Keys in file: %d", (int)keys.size());
}
void KeyStorage::deleteLinkKey(uint8_t *bd_addr)
{
auto keysSize = keys.size();
LOG_INFO("deleting key for address %s from API", bd_addr_to_str(bd_addr));
auto end = std::remove_if(keys.begin(), keys.end(), [&](auto &key) {
bd_addr_t addr;
sscanf_bd_addr(key[strings::bd_addr].string_value().c_str(), addr);
return bd_addr_cmp(addr, bd_addr) == 0;
});
keys.erase(end, keys.end());
if (keysSize != keys.size()) {
LOG_INFO("Key successfully deleted");
}
writeSettings();
}
//
auto KeyStorage::iterator_init(btstack_link_key_iterator_t *it) -> int
{
return 0;
}
auto KeyStorage::iterator_get_next(btstack_link_key_iterator_t *it,
uint8_t *bd_addr,
uint8_t *link_key,
link_key_type_t *type) -> int
{
return 0;
}
void KeyStorage::iterator_done(btstack_link_key_iterator_t *it)
{}
void KeyStorage::set_local_bd_addr(bd_addr_t bd_addr)
{}
void KeyStorage::writeSettings()
{
json11::Json finalJson = json11::Json::object{{strings::keys, keys}};
keysEntry = finalJson.dump();
if (settings) {
settings->setValue(bluetooth::Settings::BtKeys, keysEntry);
}
else {
LOG_ERROR("failed to open settings to write!");
return;
}
}
} // namespace Bt