pw_bluetooth_proxy#

Lightweight proxy for augmenting Bluetooth functionality

Unstable C++17

The pw_bluetooth_proxy module provides a lightweight proxy host that can be placed between a Bluetooth host and Bluetooth controller to add additional functionality or inspection. All without modifying the host or the controller.

An example use case could be offloading some functionality from a main host located on the application processor to instead be handled on the MCU (to reduce power usage).

The proxy acts as a proxy of all host controller interface (HCI) packets between the host and the controller.

pw::bluetooth::proxy::ProxyHost acts as the main coordinator for proxy functionality.

#include "pw_bluetooth_proxy/proxy_host.h"

  // Container creates ProxyHost .
  ProxyHost proxy = ProxyHost(std::move(container_send_to_host_fn),
                              std::move(container_send_to_controller_fn),
                              2);

  // Container passes H4 packets from host through proxy. Proxy will in turn
  // call the container-provided `container_send_to_controller_fn` to pass them
  // on to the controller. Some packets may be modified, added, or removed.
  proxy.HandleH4HciFromHost(std::move(h4_packet_from_host));

  // Container passes H4 packets from controller through proxy. Proxy will in
  // turn call the container-provided `container_send_to_host_fn` to pass them
  // on to the controller. Some packets may be modified, added, or removed.
  proxy.HandleH4HciFromController(std::move(h4_packet_from_controller));
Get Started

How to set up in your build system

API Reference

Reference information about the API

Roadmap

Upcoming plans

Code size analysis

Understand code footprint and savings potential

Get started#

  1. Add Emboss to your project as described in Emboss.

Bazel isn’t supported yet.

2. Then add $dir_pw_bluetooth_proxy to the deps list in your pw_executable() build target:

pw_executable("...") {
  # ...
  deps = [
    # ...
    "$dir_pw_bluetooth_proxy",
    # ...
  ]
}

2. Then add pw_bluetooth_proxy to the DEPS list in your cmake target:

API reference#

pw::bluetooth::proxy::ProxyHost#

class ProxyHost#

ProxyHost acts as the main coordinator for proxy functionality. After creation, the container then passes packets through the proxy.

Public Functions

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,
uint16_t br_edr_acl_credits_to_reserve = 0,
)#

Creates an ProxyHost that will process HCI packets.

Parameters:
  • send_to_host_fn[in] Callback that will be called when proxy wants to send HCI packet towards the host.

  • send_to_controller_fn[in] - Callback that will be called when proxy wants to send HCI packet towards the controller.

void HandleH4HciFromHost(H4PacketWithH4 &&h4_packet)#

Called by container to ask proxy to handle a H4 HCI packet sent from the host side towards the controller side. Proxy will in turn call the send_to_controller_fn provided during construction to pass the packet on to the controller. Some packets may be modified, added, or removed.

The proxy host currently does not require any from-host packets to support its current functionality. It will pass on all packets, so containers can choose to just pass all from-host packets through it.

Container is required to call this function synchronously (one packet at a time).

void HandleH4HciFromController(H4PacketWithHci &&h4_packet)#

Called by container to ask proxy to handle a H4 packet sent from the controller side towards the host side. Proxy will in turn call the send_to_host_fn provided during construction to pass the packet on to the host. Some packets may be modified, added, or removed.

To support all of its current functionality, the proxy host needs at least the following from-controller packets passed through it. It will pass on all other packets, so containers can choose to just pass all from-controller packets through the proxy host.

All packets of this type:

  • L2CAP over ACL packets (specifically those addressed to channels managed by the proxy host, including signaling packets)

HCI_Command_Complete events (7.7.14) containing return parameters for these commands:

  • HCI_LE_Read_Buffer_Size [v1] command (7.8.2)

  • HCI_LE_Read_Buffer_Size [v2] command (7.8.2)

These HCI event packets:

  • HCI_Number_Of_Completed_Packets event (7.7.19)

  • HCI_Disconnection_Complete event (7.7.5)

Container is required to call this function synchronously (one packet at a time).

void Reset()#

Called by container to notify proxy that the Bluetooth system is being reset, so the proxy can reset its internal state. Warning: Outstanding H4 packets are not invalidated upon reset. If they are destructed post-reset, packets generated post-reset are liable to be overwritten prematurely.

pw::Result<L2capCoc> AcquireL2capCoc(
uint16_t connection_handle,
L2capCoc::CocConfig rx_config,
L2capCoc::CocConfig tx_config,
pw::Function<void(pw::span<uint8_t> payload)> &&receive_fn,
pw::Function<void(L2capCoc::Event event)> &&event_fn,
)#

Returns an L2CAP connection-oriented channel that supports writing to and reading from a remote peer.

Parameters:
  • connection_handle[in] The connection handle of the remote peer.

  • rx_config[in] Parameters applying to reading packets. See l2cap_coc.h for details.

  • tx_config[in] Parameters applying to writing packets. See l2cap_coc.h for details.

  • receive_fn[in] Read callback to be invoked on Rx SDUs.

  • event_fn[in] Handle asynchronous events such as errors encountered by the channel.

Returns:

Code

Description

INVALID_ARGUMENT

If arguments are invalid (check logs). UNAVAILABLE: If channel could not be created because no memory was available to accommodate an additional ACL connection.

pw::Status SendGattNotify(uint16_t connection_handle, uint16_t attribute_handle, pw::span<const uint8_t> attribute_value)#

Send a GATT Notify to the indicated connection.

Parameters:
  • connection_handle[in] The connection handle of the peer to notify. Maximum valid connection handle is 0x0EFF.

  • attribute_handle[in] The attribute handle the notify should be sent on. Cannot be 0.

  • attribute_value[in] The data to be sent. Data will be copied before function completes.

Returns:

Code

Description

OK

If notify was successfully queued for send. UNAVAILABLE: If CHRE doesn’t have resources to queue the send at this time (transient error). INVALID_ARGUMENT: If arguments are invalid (check logs).

bool HasSendLeAclCapability() const#

Indicates whether the proxy has the capability of sending LE ACL packets. Note that this indicates intention, so it can be true even if the proxy has not yet or has been unable to reserve credits from the host.

inline bool HasSendAclCapability() const#

Deprecated:

Use HasSendLeAclCapability

bool HasSendBrEdrAclCapability() const#

Indicates whether the proxy has the capability of sending BR/EDR ACL packets. Note that this indicates intention, so it can be true even if the proxy has not yet or has been unable to reserve credits from the host.

uint16_t GetNumFreeLeAclPackets() const#

Returns the number of available LE ACL send credits for the proxy. Can be zero if the controller has not yet been initialized by the host.

uint16_t GetNumFreeBrEdrAclPackets() const#

Returns the number of available BR/EDR ACL send credits for the proxy. Can be zero if the controller has not yet been initialized by the host.

Public Static Functions

static inline constexpr size_t GetNumSimultaneousAclSendsSupported()#

Returns the max number of LE ACL sends that can be in-flight at one time. That is, ACL packets that have been sent and not yet released.

static inline constexpr size_t GetMaxAclSendSize()#

Returns the max LE ACL packet size supported to be sent.

static inline constexpr size_t GetMaxNumLeAclConnections()#

Returns the max number of simultaneous LE ACL connections supported.

Code size analysis#

Delta when constructing a proxy and just sending packets through.

Label

Segment

Delta

Create and use proxy as a simple passthrough

FLASH

-762

_printf_float

DEL

-596

__aeabi_dmul

DEL

-464

__aeabi_ddiv

+30

[section .code]

DEL

-206

__cvt

DEL

-204

pw::tokenizer::EncodeArgs()

DEL

-197

__aeabi_dsub

-2

__aeabi_d2f

DEL

-140

__gtdf2

DEL

-120

pw::bloat::BloatThisBinary()

DEL

-118

__exponent

DEL

-108

__floatundidf

-2

__aeabi_d2iz

DEL

-72

__libc_init_array

DEL

-68

__extendsfdf2

DEL

-68

_pw_boot_Entry

DEL

-64

pw_sys_io_lm3s6965evb_Init

+1

pw_assert_tokenized_HandleAssertFailure::_pw_tokenizer_string_entry_60_5

DEL

-48

pw_varint_Encode64

DEL

-42

_pw_log_tokenized_EncodeTokenizedLog

DEL

-36

__aeabi_i2d

DEL

-32

__aeabi_cdrcmple

DEL

-32

__aeabi_ui2d

DEL

-32

pw::sys_io::WriteBytes()

DEL

-30

pw::tokenizer::EncodedMessage<>::EncodedMessage()

DEL

-28

pw_tokenizer_EncodeInt64()

DEL

-24

pw_boot_PreStaticMemoryInit

DEL

-20

__aeabi_dcmpeq

-12

__aeabi_dcmpge

-2

__aeabi_dcmpgt

DEL

-20

__aeabi_dcmple

DEL

-20

__aeabi_dcmplt

DEL

-20

pw::sys_io::WriteByte()

DEL

-20

pw_assert_HandleFailure

-4

p05.0

DEL

-16

pw_boot_PostMain

+38

main

DEL

-8

pw_boot_Entry

DEL

-8

pw_log_tokenized_HandleLog

DEL

-6

__aeabi_memclr

DEL

-6

pw_boot_PreMainInit

DEL

-4

DefaultFaultHandler

DEL

-4

__aeabi_memcpy

DEL

-4

_exit

DEL

-2

pw_boot_PreStaticConstructorInit

NEW

+272

pw::bluetooth::proxy::L2capReadChannel::operator=()::_pw_tokenizer_string_entry_34_5

NEW

+239

pw::bluetooth::proxy::L2capAclUSignalingChannel::OnCFramePayload()::_pw_tokenizer_string_entry_54_7

NEW

+226

pw::bluetooth::proxy::AclDataChannel::Credits::Reserve()::_pw_tokenizer_string_entry_44_5

NEW

+216

pw::bluetooth::proxy::H4Storage::ReleaseH4Buff()::_pw_tokenizer_string_entry_49_1

NEW

+215

pw::bluetooth::proxy::AclDataChannel::HandleDisconnectionCompleteEvent()::_pw_tokenizer_string_entry_231_33

NEW

+213

pw::bluetooth::proxy::AclDataChannel::Credits::Reserve()::_pw_tokenizer_string_entry_60_13

NEW

+213

pw::bluetooth::proxy::L2capCoc::Create()::_pw_tokenizer_string_entry_116_11

NEW

+208

pw::bluetooth::proxy::L2capCoc::OnPduReceived()::_pw_tokenizer_string_entry_159_23

NEW

+203

pw::bluetooth::proxy::AclDataChannel::HandleDisconnectionCompleteEvent()::_pw_tokenizer_string_entry_219_29

NEW

+202

pw::bluetooth::proxy::L2capLeUSignalingChannel::OnCFramePayload()::_pw_tokenizer_string_entry_52_7

NEW

+201

pw::bluetooth::proxy::L2capLeUSignalingChannel::OnCFramePayload()::_pw_tokenizer_string_entry_62_11

NEW

+192

pw::bluetooth::proxy::L2capSignalingChannel::HandleFlowControlCreditInd()::_pw_tokenizer_string_entry_83_11

NEW

+191

pw::bluetooth::proxy::L2capAclUSignalingChannel::OnCFramePayload()::_pw_tokenizer_string_entry_41_3

NEW

+184

pw::bluetooth::proxy::L2capCoc::OnPduReceived()::_pw_tokenizer_string_entry_174_27

NEW

+183

pw::containers::internal::CheckIntrusiveItemIsUncontained()::_pw_tokenizer_string_entry_28_3

NEW

+182

pw::bluetooth::proxy::L2capCoc::OnPduReceived()::_pw_tokenizer_string_entry_200_35

NEW

+177

pw::bluetooth::proxy::L2capChannelManager::DrainWriteChannelQueues()::_pw_tokenizer_string_entry_117_13

NEW

+177

pw::bluetooth::proxy::L2capCoc::OnPduReceived()::_pw_tokenizer_string_entry_211_39

NEW

+174

pw::bluetooth::proxy::ProxyHost::HandleAclFromController()::_pw_tokenizer_string_entry_143_21

NEW

+173

pw::bluetooth::proxy::L2capCoc::OnPduReceived()::_pw_tokenizer_string_entry_187_31

NEW

+172

pw::bluetooth::proxy::ProxyHost::HandleCommandCompleteEvent()::_pw_tokenizer_string_entry_211_39

NEW

+172

pw::bluetooth::proxy::ProxyHost::HandleCommandCompleteEvent()::_pw_tokenizer_string_entry_225_43

NEW

+170

pw::bluetooth::proxy::L2capCoc::OnFragmentedPduReceived()::_pw_tokenizer_string_entry_247_43

NEW

+170

pw::bluetooth::proxy::L2capSignalingChannel::OnPduReceived()::_pw_tokenizer_string_entry_47_7

NEW

+167

pw::containers::internal::CheckIntrusiveContainerIsEmpty()::_pw_tokenizer_string_entry_22_1

NEW

+166

pw::bluetooth::proxy::AclDataChannel::HandleNumberOfCompletedPacketsEvent()::_pw_tokenizer_string_entry_131_21

NEW

+164

pw::bluetooth::proxy::L2capSignalingChannel::OnFragmentedPduReceived()::_pw_tokenizer_string_entry_105_15

NEW

+163

pw::bluetooth::proxy::ProxyHost::HandleCommandCompleteEvent()::_pw_tokenizer_string_entry_198_35

NEW

+161

pw::bluetooth::proxy::AclDataChannel::HandleDisconnectionCompleteEvent()::_pw_tokenizer_string_entry_198_25

NEW

+161

pw::bluetooth::proxy::L2capAclUSignalingChannel::OnCFramePayload()::_pw_tokenizer_string_entry_45_5

NEW

+161

pw::bluetooth::proxy::ProxyHost::HandleEventFromController()::_pw_tokenizer_string_entry_67_7

NEW

+159

pw::bluetooth::proxy::ProxyHost::HandleAclFromController()::_pw_tokenizer_string_entry_126_13

NEW

+158

pw::bluetooth::proxy::L2capWriteChannel::AreValidParameters()::_pw_tokenizer_string_entry_87_7

NEW

+154

pw::bluetooth::proxy::L2capChannelManager::GetTxH4Packet()::_pw_tokenizer_string_entry_67_7

NEW

+154

pw::bluetooth::proxy::L2capCoc::OnPduReceived()::_pw_tokenizer_string_entry_145_15

NEW

+149

pw::bluetooth::proxy::ProxyHost::HandleCommandCompleteEvent()::_pw_tokenizer_string_entry_183_31

NEW

+146

pw::bluetooth::proxy::L2capCoc::AddCredits()::_pw_tokenizer_string_entry_280_47

NEW

+146

pw::bluetooth::proxy::ProxyHost::AcquireL2capCoc()::_pw_tokenizer_string_entry_254_45

NEW

+144

pw::bluetooth::proxy::L2capCoc::Write()::_pw_tokenizer_string_entry_57_3

NEW

+144

pw::bluetooth::proxy::L2capCoc::Write()::_pw_tokenizer_string_entry_67_7

NEW

+142

pw::bluetooth::proxy::AclDataChannel::Credits::MarkCompleted()::_pw_tokenizer_string_entry_78_17

NEW

+142

pw::bluetooth::proxy::GattNotifyChannel::Write()::_pw_tokenizer_string_entry_30_3

NEW

+142

pw::bluetooth::proxy::L2capCoc::OnPduReceived()::_pw_tokenizer_string_entry_150_19

NEW

+141

pw::bluetooth::proxy::ProxyHost::HandleAclFromController()::_pw_tokenizer_string_entry_103_11

NEW

+140

pw::bluetooth::proxy::AclDataChannel::SendAcl()::_pw_tokenizer_string_entry_270_41

NEW

+139

pw::bluetooth::proxy::AclDataChannel::SendAcl()::_pw_tokenizer_string_entry_277_45

NEW

+137

pw::bluetooth::proxy::ProxyHost::HandleAclFromController()::_pw_tokenizer_string_entry_158_25

NEW

+135

pw::bluetooth::proxy::AclDataChannel::SendAcl()::_pw_tokenizer_string_entry_263_37

NEW

+134

pw::bluetooth::proxy::AclDataChannel::Credits::Reserve()::_pw_tokenizer_string_entry_52_9

NEW

+133

pw::bluetooth::proxy::ProxyHost::HandleAclFromController()::_pw_tokenizer_string_entry_129_17

NEW

+130

pw::bluetooth::proxy::L2capWriteChannel::AreValidParameters()::_pw_tokenizer_string_entry_91_11

NEW

+122

pw::bluetooth::proxy::L2capLeUSignalingChannel::OnCFramePayload()::_pw_tokenizer_string_entry_39_3

NEW

+117

pw::bluetooth::proxy::GattNotifyChannel::Create()::_pw_tokenizer_string_entry_77_7

NEW

+116

pw::bluetooth::proxy::L2capChannelManager::GetTxH4Packet()::_pw_tokenizer_string_entry_73_11

NEW

+76

pw::bluetooth::proxy::ProxyHost::HandleAclFromController()::_pw_tokenizer_string_entry_167_27

NEW

+56

pw_assert_HandleFailure::_pw_tokenizer_string_entry_35_1

NEW

+40

fit::internal::target<>::ops

NEW

+37

pw::bluetooth::proxy::AclDataChannel::Credits::MarkCompleted()::_pw_tokenizer_string_entry_78_15

NEW

+37

pw::bluetooth::proxy::AclDataChannel::Credits::Reserve()::_pw_tokenizer_string_entry_60_11

NEW

+37

pw::bluetooth::proxy::AclDataChannel::HandleDisconnectionCompleteEvent()::_pw_tokenizer_string_entry_198_23

NEW

+37

pw::bluetooth::proxy::AclDataChannel::HandleDisconnectionCompleteEvent()::_pw_tokenizer_string_entry_219_27

NEW

+37

pw::bluetooth::proxy::AclDataChannel::HandleDisconnectionCompleteEvent()::_pw_tokenizer_string_entry_231_31

NEW

+37

pw::bluetooth::proxy::AclDataChannel::HandleNumberOfCompletedPacketsEvent()::_pw_tokenizer_string_entry_131_19

NEW

+37

pw::bluetooth::proxy::AclDataChannel::SendAcl()::_pw_tokenizer_string_entry_263_35

NEW

+37

pw::bluetooth::proxy::AclDataChannel::SendAcl()::_pw_tokenizer_string_entry_270_39

NEW

+37

pw::bluetooth::proxy::AclDataChannel::SendAcl()::_pw_tokenizer_string_entry_277_43

NEW

+37

pw::bluetooth::proxy::GattNotifyChannel::Create()::_pw_tokenizer_string_entry_77_5

NEW

+37

pw::bluetooth::proxy::GattNotifyChannel::Write()::_pw_tokenizer_string_entry_30_1

NEW

+37

pw::bluetooth::proxy::L2capAclUSignalingChannel::OnCFramePayload()::_pw_tokenizer_string_entry_41_1

NEW

+37

pw::bluetooth::proxy::L2capChannelManager::GetTxH4Packet()::_pw_tokenizer_string_entry_67_5

NEW

+37

pw::bluetooth::proxy::L2capChannelManager::GetTxH4Packet()::_pw_tokenizer_string_entry_73_9

NEW

+37

pw::bluetooth::proxy::L2capCoc::AddCredits()::_pw_tokenizer_string_entry_280_45

NEW

+37

pw::bluetooth::proxy::L2capCoc::Create()::_pw_tokenizer_string_entry_116_9

NEW

+37

pw::bluetooth::proxy::L2capCoc::OnFragmentedPduReceived()::_pw_tokenizer_string_entry_247_41

NEW

+37

pw::bluetooth::proxy::L2capCoc::OnPduReceived()::_pw_tokenizer_string_entry_145_13

NEW

+37

pw::bluetooth::proxy::L2capCoc::OnPduReceived()::_pw_tokenizer_string_entry_150_17

NEW

+37

pw::bluetooth::proxy::L2capCoc::OnPduReceived()::_pw_tokenizer_string_entry_159_21

NEW

+37

pw::bluetooth::proxy::L2capCoc::OnPduReceived()::_pw_tokenizer_string_entry_174_25

NEW

+37

pw::bluetooth::proxy::L2capCoc::OnPduReceived()::_pw_tokenizer_string_entry_187_29

NEW

+37

pw::bluetooth::proxy::L2capCoc::OnPduReceived()::_pw_tokenizer_string_entry_200_33

NEW

+37

pw::bluetooth::proxy::L2capCoc::OnPduReceived()::_pw_tokenizer_string_entry_211_37

NEW

+37

pw::bluetooth::proxy::L2capCoc::Write()::_pw_tokenizer_string_entry_57_1

NEW

+37

pw::bluetooth::proxy::L2capCoc::Write()::_pw_tokenizer_string_entry_67_5

NEW

+37

pw::bluetooth::proxy::L2capLeUSignalingChannel::OnCFramePayload()::_pw_tokenizer_string_entry_39_1

NEW

+37

pw::bluetooth::proxy::L2capLeUSignalingChannel::OnCFramePayload()::_pw_tokenizer_string_entry_62_9

NEW

+37

pw::bluetooth::proxy::L2capSignalingChannel::HandleFlowControlCreditInd()::_pw_tokenizer_string_entry_83_9

NEW

+37

pw::bluetooth::proxy::L2capSignalingChannel::OnFragmentedPduReceived()::_pw_tokenizer_string_entry_105_13

NEW

+37

pw::bluetooth::proxy::L2capWriteChannel::AreValidParameters()::_pw_tokenizer_string_entry_87_5

NEW

+37

pw::bluetooth::proxy::L2capWriteChannel::AreValidParameters()::_pw_tokenizer_string_entry_91_9

NEW

+37

pw::bluetooth::proxy::ProxyHost::HandleAclFromController()::_pw_tokenizer_string_entry_103_9

NEW

+37

pw::bluetooth::proxy::ProxyHost::HandleAclFromController()::_pw_tokenizer_string_entry_129_15

NEW

+37

pw::bluetooth::proxy::ProxyHost::HandleAclFromController()::_pw_tokenizer_string_entry_143_19

NEW

+37

pw::bluetooth::proxy::ProxyHost::HandleAclFromController()::_pw_tokenizer_string_entry_158_23

NEW

+37

pw::bluetooth::proxy::ProxyHost::HandleCommandCompleteEvent()::_pw_tokenizer_string_entry_183_29

NEW

+37

pw::bluetooth::proxy::ProxyHost::HandleCommandCompleteEvent()::_pw_tokenizer_string_entry_198_33

NEW

+37

pw::bluetooth::proxy::ProxyHost::HandleCommandCompleteEvent()::_pw_tokenizer_string_entry_211_37

NEW

+37

pw::bluetooth::proxy::ProxyHost::HandleCommandCompleteEvent()::_pw_tokenizer_string_entry_225_41

NEW

+37

pw::bluetooth::proxy::ProxyHost::HandleEventFromController()::_pw_tokenizer_string_entry_67_5

NEW

+34

OUTLINED_FUNCTION_1

NEW

+30

OUTLINED_FUNCTION_0

NEW

+28

pw::bluetooth::proxy::H4PacketWithH4

NEW

+28

pw::bluetooth::proxy::H4PacketWithHci

NEW

+27

pw::bluetooth::proxy::L2capSignalingChannel::OnPduReceived()::_pw_tokenizer_string_entry_47_5

NEW

+24

pw::bluetooth::proxy::L2capReadChannel

NEW

+20

OUTLINED_FUNCTION_2

NEW

+20

OUTLINED_FUNCTION_3

NEW

+20

__cxa_pure_virtual

NEW

+20

fit::internal::null_target<>::ops

NEW

+20

write

NEW

+16

OUTLINED_FUNCTION_4

NEW

+16

free

NEW

+16

std::get_terminate()

NEW

+10

OUTLINED_FUNCTION_10

NEW

+10

OUTLINED_FUNCTION_9

NEW

+10

std::terminate()

NEW

+8

OUTLINED_FUNCTION_5

NEW

+8

OUTLINED_FUNCTION_6

NEW

+8

__cxxabiv1::__terminate()

NEW

+8

operator delete()

NEW

+6

OUTLINED_FUNCTION_11

NEW

+6

OUTLINED_FUNCTION_12

NEW

+6

OUTLINED_FUNCTION_7

NEW

+6

OUTLINED_FUNCTION_8

NEW

+6

__aeabi_memclr4

NEW

+6

pw::bluetooth::proxy::HciTransport::SendToHost()::_pw_tokenizer_string_entry_35_1

NEW

+2

fit::internal::target<>::invoke()

NEW

+2

pw::bluetooth::proxy::HciTransport::SendToController()::_pw_tokenizer_string_entry_41_3

+7,600

Roadmap#

  • ACL flow control

  • Sending GATT notifications

  • CMake support

  • Receiving GATT notifications

  • Taking ownership of a L2CAP channel

  • Bazel support

  • And more…