blob: 49c178c0ebedddfd6213f8d20510e0f49b7bb8e8 [file] [log] [blame]
// Copyright 2024 The Pigweed Authors
//
// 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
//
// https://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 "pw_bluetooth_proxy/proxy_host.h"
#include "pw_assert/check.h" // IWYU pragma: keep
#include "pw_bluetooth/hci_common.emb.h"
#include "pw_bluetooth/hci_h4.emb.h"
#include "pw_bluetooth_proxy/emboss_util.h"
#include "pw_bluetooth_proxy/h4_packet.h"
#include "pw_log/log.h"
namespace pw::bluetooth::proxy {
ProxyHost::ProxyHost(
pw::Function<void(H4PacketWithHci&& packet)>&& send_to_host_fn,
pw::Function<void(H4PacketWithH4&& packet)>&& send_to_controller_fn,
uint16_t le_acl_credits_to_reserve)
: outward_send_to_host_fn_(std::move(send_to_host_fn)),
outward_send_to_controller_fn_(std::move(send_to_controller_fn)),
acl_data_channel_{le_acl_credits_to_reserve} {}
void ProxyHost::HandleH4HciFromHost(H4PacketWithH4&& h4_packet) {
SendToController(std::move(h4_packet));
}
void ProxyHost::ProcessH4HciFromController(H4PacketWithHci&& h4_packet) {
if (h4_packet.GetHciSpan().empty()) {
PW_LOG_ERROR("Received empty H4 buffer. So will not process.");
return;
}
if (h4_packet.GetH4Type() != emboss::H4PacketType::EVENT) {
return;
}
pw::span hci_buffer = h4_packet.GetHciSpan();
auto event = MakeEmboss<emboss::EventHeaderView>(hci_buffer);
if (!event.IsComplete()) {
PW_LOG_ERROR("Buffer is too small for EventHeader. So will not process.");
return;
}
if (event.event_code_enum().Read() != emboss::EventCode::COMMAND_COMPLETE) {
return;
}
auto command_complete_event =
MakeEmboss<emboss::CommandCompleteEventView>(hci_buffer);
if (!command_complete_event.IsComplete()) {
PW_LOG_ERROR(
"Buffer is too small for COMMAND_COMPLETE event. So will not process.");
return;
}
PW_MODIFY_DIAGNOSTICS_PUSH();
PW_MODIFY_DIAGNOSTIC(ignored, "-Wswitch-enum");
switch (command_complete_event.command_opcode_enum().Read()) {
case emboss::OpCode::LE_READ_BUFFER_SIZE_V1: {
auto read_event =
MakeEmboss<emboss::LEReadBufferSizeV1CommandCompleteEventWriter>(
hci_buffer);
if (!read_event.IsComplete()) {
PW_LOG_ERROR(
"Buffer is too small for LE_READ_BUFFER_SIZE_V1 command complete "
"event. So will not process.");
return;
}
acl_data_channel_.ProcessLEReadBufferSizeCommandCompleteEvent(read_event);
break;
}
case emboss::OpCode::LE_READ_BUFFER_SIZE_V2: {
auto read_event =
MakeEmboss<emboss::LEReadBufferSizeV2CommandCompleteEventWriter>(
hci_buffer);
if (!read_event.IsComplete()) {
PW_LOG_ERROR(
"Buffer is too small for LE_READ_BUFFER_SIZE_V2 command complete "
"event. So will not process.");
return;
}
acl_data_channel_.ProcessLEReadBufferSizeCommandCompleteEvent(read_event);
break;
}
default:
// Nothing to process
break;
}
PW_MODIFY_DIAGNOSTICS_POP();
}
void ProxyHost::HandleH4HciFromController(H4PacketWithHci&& h4_packet) {
ProcessH4HciFromController(std::move(h4_packet));
SendToHost(std::move(h4_packet));
}
void ProxyHost::SendToHost(H4PacketWithHci&& h4_packet) {
PW_DCHECK(outward_send_to_host_fn_ != nullptr);
outward_send_to_host_fn_(std::move(h4_packet));
}
void ProxyHost::SendToController(H4PacketWithH4&& h4_packet) {
PW_DCHECK(outward_send_to_controller_fn_ != nullptr);
outward_send_to_controller_fn_(std::move(h4_packet));
}
bool ProxyHost::HasSendAclCapability() const {
return acl_data_channel_.GetLeAclCreditsToReserve() > 0;
}
uint16_t ProxyHost::GetNumFreeLeAclPackets() const {
return acl_data_channel_.GetNumFreeLeAclPackets();
}
} // namespace pw::bluetooth::proxy