blob: 7e45d89d574a43be151239ea655c95bc7cbcd136 [file] [log] [blame]
/*
* Copyright 2019 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 "hal/hci_hal.h"
#include <stdlib.h>
#include <vector>
#include <future>
#include <android/hardware/bluetooth/1.0/IBluetoothHci.h>
#include <android/hardware/bluetooth/1.0/IBluetoothHciCallbacks.h>
#include <android/hardware/bluetooth/1.0/types.h>
#include "hal/snoop_logger.h"
#include "os/log.h"
using ::android::hardware::hidl_vec;
using ::android::hardware::Return;
using ::android::hardware::Void;
using ::android::hardware::bluetooth::V1_0::IBluetoothHci;
using ::android::hardware::bluetooth::V1_0::IBluetoothHciCallbacks;
using HidlStatus = ::android::hardware::bluetooth::V1_0::Status;
namespace bluetooth {
namespace hal {
namespace {
class HciDeathRecipient : public ::android::hardware::hidl_death_recipient {
public:
virtual void serviceDied(uint64_t /*cookie*/, const android::wp<::android::hidl::base::V1_0::IBase>& /*who*/) {
LOG_ERROR("Bluetooth HAL service died!");
abort();
}
};
android::sp<HciDeathRecipient> hci_death_recipient_ = new HciDeathRecipient();
class InternalHciCallbacks : public IBluetoothHciCallbacks {
public:
InternalHciCallbacks(SnoopLogger* btsnoop_logger)
: btsnoop_logger_(btsnoop_logger) {
init_promise_ = new std::promise<void>();
}
void SetCallback(HciHalCallbacks* callback) {
ASSERT(callback_ == nullptr && callback != nullptr);
callback_ = callback;
}
void ResetCallback() {
callback_ = nullptr;
}
std::promise<void>* GetInitPromise() {
return init_promise_;
}
Return<void> initializationComplete(HidlStatus status) {
ASSERT(status == HidlStatus::SUCCESS);
init_promise_->set_value();
return Void();
}
Return<void> hciEventReceived(const hidl_vec<uint8_t>& event) {
std::vector<uint8_t> received_hci_packet(event.begin(), event.end());
btsnoop_logger_->capture(received_hci_packet, SnoopLogger::Direction::INCOMING,
SnoopLogger::PacketType::EVT);
if (callback_ != nullptr) {
callback_->hciEventReceived(std::move(received_hci_packet));
}
return Void();
}
Return<void> aclDataReceived(const hidl_vec<uint8_t>& data) {
std::vector<uint8_t> received_hci_packet(data.begin(), data.end());
btsnoop_logger_->capture(received_hci_packet, SnoopLogger::Direction::INCOMING,
SnoopLogger::PacketType::ACL);
if (callback_ != nullptr) {
callback_->aclDataReceived(std::move(received_hci_packet));
}
return Void();
}
Return<void> scoDataReceived(const hidl_vec<uint8_t>& data) {
std::vector<uint8_t> received_hci_packet(data.begin(), data.end());
btsnoop_logger_->capture(received_hci_packet, SnoopLogger::Direction::INCOMING,
SnoopLogger::PacketType::SCO);
if (callback_ != nullptr) {
callback_->scoDataReceived(std::move(received_hci_packet));
}
return Void();
}
private:
std::promise<void>* init_promise_ = nullptr;
HciHalCallbacks* callback_ = nullptr;
SnoopLogger* btsnoop_logger_ = nullptr;
};
} // namespace
const std::string SnoopLogger::DefaultFilePath = "/data/misc/bluetooth/logs/btsnoop_hci.log";
const bool SnoopLogger::AlwaysFlush = false;
class HciHalHidl : public HciHal {
public:
void registerIncomingPacketCallback(HciHalCallbacks* callback) override {
callbacks_->SetCallback(callback);
}
void unregisterIncomingPacketCallback() override {
callbacks_->ResetCallback();
}
void sendHciCommand(HciPacket command) override {
btsnoop_logger_->capture(command, SnoopLogger::Direction::OUTGOING, SnoopLogger::PacketType::CMD);
bt_hci_->sendHciCommand(command);
}
void sendAclData(HciPacket packet) override {
btsnoop_logger_->capture(packet, SnoopLogger::Direction::OUTGOING, SnoopLogger::PacketType::ACL);
bt_hci_->sendAclData(packet);
}
void sendScoData(HciPacket packet) override {
btsnoop_logger_->capture(packet, SnoopLogger::Direction::OUTGOING, SnoopLogger::PacketType::SCO);
bt_hci_->sendScoData(packet);
}
protected:
void ListDependencies(ModuleList* list) override {
list->add<SnoopLogger>();
}
void Start() override {
btsnoop_logger_ = GetDependency<SnoopLogger>();
bt_hci_ = IBluetoothHci::getService();
ASSERT(bt_hci_ != nullptr);
auto death_link = bt_hci_->linkToDeath(hci_death_recipient_, 0);
ASSERT_LOG(death_link.isOk(), "Unable to set the death recipient for the Bluetooth HAL");
// Block allows allocation of a variable that might be bypassed by goto.
{
callbacks_ = new InternalHciCallbacks(btsnoop_logger_);
bt_hci_->initialize(callbacks_);
// Don't timeout here, time out at a higher layer
callbacks_->GetInitPromise()->get_future().wait();
}
}
void Stop() override {
ASSERT(bt_hci_ != nullptr);
auto death_unlink = bt_hci_->unlinkToDeath(hci_death_recipient_);
if (!death_unlink.isOk()) {
LOG_ERROR("Error unlinking death recipient from the Bluetooth HAL");
}
bt_hci_->close();
callbacks_->ResetCallback();
bt_hci_ = nullptr;
}
private:
android::sp<InternalHciCallbacks> callbacks_;
android::sp<IBluetoothHci> bt_hci_;
SnoopLogger* btsnoop_logger_;
};
const ModuleFactory HciHal::Factory = ModuleFactory([]() {
return new HciHalHidl();
});
} // namespace hal
} // namespace bluetooth