Files
MuditaOS/module-bluetooth/Bluetooth/CommandHandler.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

149 lines
4.8 KiB
C++

// Copyright (c) 2017-2021, Mudita Sp. z.o.o. All rights reserved.
// For licensing, see https://github.com/mudita/MuditaOS/LICENSE.md
#include "CommandHandler.hpp"
#include <utility>
#include <service-bluetooth/ServiceBluetooth.hpp>
#include <service-bluetooth/SettingsHolder.hpp>
#include <Service/Service.hpp>
#include "Device.hpp"
#include "BtCommand.hpp"
extern "C"
{
#include <module-bluetooth/lib/btstack/src/btstack_util.h>
}
namespace bluetooth
{
namespace
{
[[nodiscard]] auto toString(bluetooth::Error::Code code) -> std::string
{
return utils::enumToString(code);
}
} // namespace
CommandHandler::CommandHandler(sys::Service *service,
std::shared_ptr<bluetooth::SettingsHolder> settings,
std::shared_ptr<bluetooth::ProfileManager> profileManager,
std::shared_ptr<bluetooth::AbstractDriver> driver)
: service{service}, settings{std::move(settings)}, profileManager{std::move(profileManager)}, driver{std::move(
driver)}
{}
Error::Code CommandHandler::handle(Command command)
{
switch (command.getType()) {
case bluetooth::Command::PowerOn:
return Error::Success;
case bluetooth::Command::StartScan:
return scan();
case bluetooth::Command::StopScan:
return stopScan();
case bluetooth::Command::StartPan:
return startPan();
case bluetooth::Command::Pair:
return pair(command.getAddress());
case bluetooth::Command::VisibilityOn:
return setVisibility(true);
case bluetooth::Command::VisibilityOff:
return setVisibility(false);
case bluetooth::Command::ConnectAudio:
return establishAudioConnection(command.getAddress());
case bluetooth::Command::DisconnectAudio:
return disconnectAudioConnection();
case bluetooth::Command::PowerOff:
return Error::Success;
case bluetooth::Command::SwitchProfile:
return switchAudioProfile();
case bluetooth::Command::None:
return Error::Success;
case Command::StartStream:
profileManager->start();
return Error::Success;
case Command::StopStream:
profileManager->stop();
return Error::Success;
}
return Error::LibraryError;
}
Error::Code CommandHandler::scan()
{
if (const auto ret = driver->scan(); ret.err != bluetooth::Error::Success) {
LOG_ERROR("Cant start scan!: %s %" PRIu32 "", toString(ret.err).c_str(), ret.lib_code);
return ret.err;
}
LOG_INFO("Scan started!");
static_cast<ServiceBluetooth *>(service)->scanStartedCallback();
// open new scan window
return Error::Success;
}
Error::Code CommandHandler::stopScan()
{
driver->stopScan();
static_cast<ServiceBluetooth *>(service)->scanStoppedCallback();
return Error::Success;
}
Error::Code CommandHandler::startPan()
{
bluetooth::PAN::bnep_setup();
if (const auto err = bluetooth::PAN::bnep_start(); err.err != bluetooth::Error::Success) {
LOG_ERROR("PAN setup error: %s %" PRIu32, toString(err.err).c_str(), err.lib_code);
return err.err;
}
return bluetooth::Error::Success;
}
Error::Code CommandHandler::setVisibility(bool visibility)
{
driver->setVisibility(visibility);
settings->setValue(bluetooth::Settings::Visibility, static_cast<int>(visibility));
return Error::Success;
}
Error::Code CommandHandler::establishAudioConnection(uint8_t *addr)
{
profileManager->init();
LOG_INFO("Connecting audio with %s", bd_addr_to_str(addr));
profileManager->connect(addr);
return Error::Success;
}
Error::Code CommandHandler::disconnectAudioConnection()
{
profileManager->disconnect();
return Error::Success;
}
Error::Code CommandHandler::pair(bd_addr_t addr)
{
LOG_INFO("Pairing with %s", bd_addr_to_str(addr));
return driver->pair(addr) ? Error::Success : Error::LibraryError;
}
Error::Code CommandHandler::switchAudioProfile()
{
static auto profile = AudioProfile::A2DP;
if (profile == AudioProfile::A2DP) {
profile = AudioProfile::HSP;
LOG_INFO("New profile: HSP");
}
else {
profile = AudioProfile::A2DP;
LOG_INFO("New profile: A2DP");
}
profileManager->switchProfile(profile);
return Error::Success;
}
} // namespace bluetooth