Skip to content

Commit 4ff6944

Browse files
BenjaminLawsonCQ Bot Account
authored and
CQ Bot Account
committed
pw_bluetooth_sapphire: Implement Central2::Connect
Implement Central2::Connect and disconnect functionality for Sapphire. This CL does not implement all of the Connection2 methods. Rename Connection2::Disconnect to Release for consistency with other pw_bluetooth classes and for clarity (the destructor will initiate the disconnect). Move UuidFrom() into its own header so it can be easily reused. Bug: b/396449684 Change-Id: Id5309cca6089cdd12a6361939b93abdcf44f860d Reviewed-on: https://pigweed-review.googlesource.com/c/pigweed/pigweed/+/267692 Commit-Queue: Ben Lawson <benlawson@google.com> Lint: Lint 🤖 <android-build-ayeaye@system.gserviceaccount.com> Docs-Not-Needed: Ben Lawson <benlawson@google.com> Reviewed-by: Faraaz Sareshwala <fsareshwala@google.com>
1 parent 86ff4cd commit 4ff6944

File tree

15 files changed

+519
-35
lines changed

15 files changed

+519
-35
lines changed

pw_bluetooth/public/pw_bluetooth/low_energy/connection2.h

+4-4
Original file line numberDiff line numberDiff line change
@@ -185,15 +185,15 @@ class Connection2 {
185185
ConnectL2capParameters parameters) = 0;
186186

187187
private:
188-
/// Request to disconnect this connection. This method is called by the
189-
/// ~Connection::Ptr() when it goes out of scope, the API client should never
188+
/// Request to destroy this connection. This method is called by the
189+
/// ~Connection2::Ptr() when it goes out of scope, the API client should never
190190
/// call this method.
191-
virtual void Disconnect() = 0;
191+
virtual void Release() = 0;
192192

193193
public:
194194
/// Movable `Connection2` smart pointer. When `Connection::Ptr` is destroyed
195195
/// the `Connection2` will disconnect automatically.
196-
using Ptr = internal::RaiiPtr<Connection2, &Connection2::Disconnect>;
196+
using Ptr = internal::RaiiPtr<Connection2, &Connection2::Release>;
197197
};
198198

199199
} // namespace pw::bluetooth::low_energy

pw_bluetooth_sapphire/BUILD.bazel

+36
Original file line numberDiff line numberDiff line change
@@ -45,12 +45,48 @@ cc_library(
4545
strip_include_prefix = "public",
4646
tags = ["noclangtidy"],
4747
deps = [
48+
":connection",
49+
":uuid",
4850
"//pw_bluetooth:pw_bluetooth2",
4951
"//pw_bluetooth_sapphire/host/gap",
5052
"//pw_multibuf:allocator",
5153
],
5254
)
5355

56+
cc_library(
57+
name = "connection",
58+
srcs = [
59+
"connection.cc",
60+
"connection_options.cc",
61+
],
62+
hdrs = [
63+
"public/pw_bluetooth_sapphire/internal/connection.h",
64+
"public/pw_bluetooth_sapphire/internal/connection_options.h",
65+
],
66+
strip_include_prefix = "public",
67+
tags = ["noclangtidy"],
68+
visibility = ["//visibility:private"],
69+
deps = [
70+
":uuid",
71+
"//pw_bluetooth:pw_bluetooth2",
72+
"//pw_bluetooth_sapphire/host/gap",
73+
],
74+
)
75+
76+
cc_library(
77+
name = "uuid",
78+
hdrs = [
79+
"public/pw_bluetooth_sapphire/internal/uuid.h",
80+
],
81+
strip_include_prefix = "public",
82+
visibility = ["//visibility:private"],
83+
deps = [
84+
"//pw_bluetooth:pw_bluetooth2",
85+
"//pw_bluetooth_sapphire/host/common",
86+
"//pw_span",
87+
],
88+
)
89+
5490
pw_cc_test(
5591
name = "central_test",
5692
srcs = ["central_test.cc"],

pw_bluetooth_sapphire/BUILD.gn

+30
Original file line numberDiff line numberDiff line change
@@ -49,13 +49,43 @@ pw_source_set("central") {
4949
public = [ "public/pw_bluetooth_sapphire/central.h" ]
5050
sources = [ "central.cc" ]
5151
public_deps = [
52+
":connection",
53+
":uuid",
5254
"$dir_pw_bluetooth:pw_bluetooth2",
5355
"$dir_pw_multibuf:allocator",
5456
"host/gap",
5557
]
5658
public_configs = [ ":public_include_path" ]
5759
}
5860

61+
pw_source_set("connection") {
62+
visibility = [ ":*" ]
63+
public = [
64+
"public/pw_bluetooth_sapphire/internal/connection.h",
65+
"public/pw_bluetooth_sapphire/internal/connection_options.h",
66+
]
67+
sources = [
68+
"connection.cc",
69+
"connection_options.cc",
70+
]
71+
public_deps = [
72+
":uuid",
73+
"$dir_pw_bluetooth:pw_bluetooth2",
74+
"host/gap",
75+
]
76+
public_configs = [ ":public_include_path" ]
77+
}
78+
79+
pw_source_set("uuid") {
80+
visibility = [ ":*" ]
81+
public = [ "public/pw_bluetooth_sapphire/internal/uuid.h" ]
82+
public_configs = [ ":public_include_path" ]
83+
deps = [
84+
"//pw_bluetooth_sapphire/host/common",
85+
"//pw_span",
86+
]
87+
}
88+
5989
pw_test("central_test") {
6090
enable_if = pw_bluetooth_sapphire_ENABLED
6191
sources = [ "central_test.cc" ]

pw_bluetooth_sapphire/central.cc

+64-14
Original file line numberDiff line numberDiff line change
@@ -14,23 +14,22 @@
1414

1515
#include "pw_bluetooth_sapphire/central.h"
1616

17+
#include "pw_bluetooth_sapphire/internal/connection_options.h"
18+
#include "pw_bluetooth_sapphire/internal/uuid.h"
19+
1720
namespace pw::bluetooth_sapphire {
1821
namespace {
1922

2023
pw::sync::Mutex g_peripheral_lock;
2124

22-
bt::UUID UuidFrom(const pw::bluetooth::Uuid& uuid) {
23-
return bt::UUID(bt::BufferView(pw::as_bytes(uuid.As128BitSpan())));
24-
}
25-
2625
bt::gap::DiscoveryFilter DiscoveryFilterFrom(const Central::ScanFilter& in) {
2726
bt::gap::DiscoveryFilter out;
2827
if (in.service_uuid.has_value()) {
29-
bt::UUID uuid = UuidFrom(in.service_uuid.value());
28+
bt::UUID uuid = internal::UuidFrom(in.service_uuid.value());
3029
out.set_service_uuids(std::vector<bt::UUID>{std::move(uuid)});
3130
}
3231
if (in.service_data_uuid.has_value()) {
33-
bt::UUID uuid = UuidFrom(in.service_data_uuid.value());
32+
bt::UUID uuid = internal::UuidFrom(in.service_data_uuid.value());
3433
out.set_service_data_uuids(std::vector<bt::UUID>{std::move(uuid)});
3534
}
3635
if (in.manufacturer_id.has_value()) {
@@ -46,7 +45,7 @@ bt::gap::DiscoveryFilter DiscoveryFilterFrom(const Central::ScanFilter& in) {
4645
out.set_pathloss(in.max_path_loss.value());
4746
}
4847
if (in.solicitation_uuid.has_value()) {
49-
bt::UUID uuid = UuidFrom(in.solicitation_uuid.value());
48+
bt::UUID uuid = internal::UuidFrom(in.solicitation_uuid.value());
5049
out.set_solicitation_uuids(std::vector<bt::UUID>{std::move(uuid)});
5150
}
5251
return out;
@@ -108,6 +107,7 @@ Central::Central(bt::gap::Adapter::WeakPtr adapter,
108107
pw::multibuf::MultiBufAllocator& allocator)
109108
: adapter_(std::move(adapter)),
110109
dispatcher_(dispatcher),
110+
heap_dispatcher_(dispatcher),
111111
allocator_(allocator),
112112
weak_factory_(this),
113113
self_(weak_factory_.GetWeakPtr()) {}
@@ -118,11 +118,41 @@ Central::~Central() {
118118
}
119119

120120
async2::OnceReceiver<Central::ConnectResult> Central::Connect(
121-
pw::bluetooth::PeerId,
122-
bluetooth::low_energy::Connection2::ConnectionOptions) {
123-
// TODO: https://pwbug.dev/377301546 - Implement Connect
124-
return async2::OnceReceiver<ConnectResult>(
125-
pw::unexpected(ConnectError::kCouldNotBeEstablished));
121+
pw::bluetooth::PeerId peer_id,
122+
bluetooth::low_energy::Connection2::ConnectionOptions options) {
123+
bt::PeerId internal_peer_id(peer_id);
124+
bt::gap::LowEnergyConnectionOptions connection_options =
125+
internal::ConnectionOptionsFrom(options);
126+
127+
auto [result_sender, result_receiver] =
128+
async2::MakeOnceSenderAndReceiver<ConnectResult>();
129+
130+
bt::gap::Adapter::LowEnergy::ConnectionResultCallback result_cb =
131+
[self = self_,
132+
peer = internal_peer_id,
133+
sender = std::move(result_sender)](
134+
bt::gap::Adapter::LowEnergy::ConnectionResult result) mutable {
135+
if (!self.is_alive()) {
136+
return;
137+
}
138+
self->OnConnectionResult(peer, std::move(result), std::move(sender));
139+
};
140+
141+
async::TaskFunction task_fn = [self = self_,
142+
internal_peer_id,
143+
connection_options,
144+
cb = std::move(result_cb)](
145+
async::Context&, Status status) mutable {
146+
if (!status.ok() || !self.is_alive()) {
147+
return;
148+
}
149+
self->adapter_->le()->Connect(
150+
internal_peer_id, std::move(cb), connection_options);
151+
};
152+
Status post_status = heap_dispatcher_.Post(std::move(task_fn));
153+
PW_CHECK_OK(post_status);
154+
155+
return std::move(result_receiver);
126156
}
127157

128158
async2::OnceReceiver<Central::ScanStartResult> Central::Scan(
@@ -190,7 +220,7 @@ async2::OnceReceiver<Central::ScanStartResult> Central::Scan(
190220
self->adapter_->le()->StartDiscovery(active, std::move(cb));
191221
}
192222
};
193-
Status post_status = dispatcher_.Post(std::move(task_fn));
223+
Status post_status = heap_dispatcher_.Post(std::move(task_fn));
194224
PW_CHECK_OK(post_status);
195225

196226
return std::move(result_receiver);
@@ -295,7 +325,7 @@ void Central::StopScanLocked(uint16_t scan_id) {
295325
}
296326
iter->second.OnScanHandleDestroyedLocked();
297327

298-
pw::Status post_status = dispatcher_.Post(
328+
pw::Status post_status = heap_dispatcher_.Post(
299329
[self = self_, scan_id](pw::async::Context, pw::Status status) {
300330
if (!status.ok() || !self.is_alive()) {
301331
return;
@@ -306,4 +336,24 @@ void Central::StopScanLocked(uint16_t scan_id) {
306336
PW_CHECK(post_status.ok());
307337
}
308338

339+
void Central::OnConnectionResult(
340+
bt::PeerId peer_id,
341+
bt::gap::Adapter::LowEnergy::ConnectionResult result,
342+
async2::OnceSender<ConnectResult> result_sender) {
343+
if (result.is_error()) {
344+
if (result.error_value() == bt::HostError::kNotFound) {
345+
result_sender.emplace(pw::unexpected(ConnectError::kUnknownPeer));
346+
} else {
347+
result_sender.emplace(
348+
pw::unexpected(ConnectError::kCouldNotBeEstablished));
349+
}
350+
return;
351+
}
352+
353+
pw::bluetooth::low_energy::Connection2::Ptr connection_ptr(
354+
new internal::Connection(
355+
peer_id, std::move(result.value()), dispatcher_));
356+
result_sender.emplace(std::move(connection_ptr));
357+
}
358+
309359
} // namespace pw::bluetooth_sapphire

pw_bluetooth_sapphire/central_test.cc

+87
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,8 @@ using pw::async2::PendFuncTask;
3636
using pw::async2::Poll;
3737
using pw::chrono::SystemClock;
3838
using ScanFilter = Central::ScanFilter;
39+
using DisconnectReason =
40+
pw::bluetooth::low_energy::Connection2::DisconnectReason;
3941

4042
const bt::DeviceAddress kAddress0(bt::DeviceAddress::Type::kLEPublic, {0});
4143
const bt::StaticByteBuffer kAdvDataWithName(0x05, // length
@@ -108,6 +110,8 @@ class CentralTest : public ::testing::Test {
108110

109111
void DestroyCentral() { central_.reset(); }
110112

113+
auto connections() { return adapter().fake_le()->connections(); }
114+
111115
bt::gap::testing::FakeAdapter& adapter() { return adapter_; }
112116
bt::gap::PeerCache& peer_cache() { return *adapter_.peer_cache(); }
113117
pw::bluetooth_sapphire::Central& central() { return central_.value(); }
@@ -480,4 +484,87 @@ TEST_F(CentralTest, CentralDestroyedBeforeScanHandle) {
480484
scan_handle.reset();
481485
}
482486

487+
TEST_F(CentralTest, ConnectAndDisconnectSuccess) {
488+
bt::gap::Peer* peer = peer_cache().NewPeer(kAddress0, /*connectable=*/true);
489+
pw::bluetooth::low_energy::Connection2::ConnectionOptions options;
490+
std::optional<pw::Result<Central::ConnectResult>> connect_result;
491+
pw::async2::OnceReceiver<Central::ConnectResult> receiver =
492+
central().Connect(peer->identifier().value(), options);
493+
PendFuncTask connect_task =
494+
PendFuncTask([&connect_result, &receiver](Context& cx) -> Poll<> {
495+
Poll<pw::Result<Central::ConnectResult>> poll = receiver.Pend(cx);
496+
if (poll.IsPending()) {
497+
return Pending();
498+
}
499+
connect_result = std::move(poll->value());
500+
return Ready();
501+
});
502+
async2_dispatcher().Post(connect_task);
503+
EXPECT_TRUE(async2_dispatcher().RunUntilStalled().IsPending());
504+
async_dispatcher().RunUntilIdle();
505+
EXPECT_TRUE(async2_dispatcher().RunUntilStalled().IsReady());
506+
ASSERT_TRUE(connect_result.has_value());
507+
ASSERT_TRUE(connect_result->ok());
508+
ASSERT_TRUE(connect_result->value());
509+
ASSERT_EQ(adapter().fake_le()->connections().count(peer->identifier()), 1u);
510+
pw::bluetooth::low_energy::Connection2::Ptr connection =
511+
std::move(connect_result->value().value());
512+
513+
// Disconnect
514+
connection.reset();
515+
ASSERT_EQ(connections().count(peer->identifier()), 1u);
516+
async_dispatcher().RunUntilIdle();
517+
ASSERT_EQ(connections().count(peer->identifier()), 0u);
518+
}
519+
520+
TEST_F(CentralTest, PendDisconnect) {
521+
bt::gap::Peer* peer = peer_cache().NewPeer(kAddress0, /*connectable=*/true);
522+
pw::bluetooth::low_energy::Connection2::ConnectionOptions options;
523+
std::optional<pw::Result<Central::ConnectResult>> connect_result;
524+
pw::async2::OnceReceiver<Central::ConnectResult> receiver =
525+
central().Connect(peer->identifier().value(), options);
526+
PendFuncTask connect_task =
527+
PendFuncTask([&connect_result, &receiver](Context& cx) -> Poll<> {
528+
Poll<pw::Result<Central::ConnectResult>> poll = receiver.Pend(cx);
529+
if (poll.IsPending()) {
530+
return Pending();
531+
}
532+
connect_result = std::move(poll->value());
533+
return Ready();
534+
});
535+
async2_dispatcher().Post(connect_task);
536+
EXPECT_TRUE(async2_dispatcher().RunUntilStalled().IsPending());
537+
async_dispatcher().RunUntilIdle();
538+
EXPECT_TRUE(async2_dispatcher().RunUntilStalled().IsReady());
539+
ASSERT_TRUE(connect_result.has_value());
540+
ASSERT_TRUE(connect_result->ok());
541+
ASSERT_TRUE(connect_result->value());
542+
ASSERT_EQ(adapter().fake_le()->connections().count(peer->identifier()), 1u);
543+
pw::bluetooth::low_energy::Connection2::Ptr connection =
544+
std::move(connect_result->value().value());
545+
546+
std::optional<DisconnectReason> disconnect_reason;
547+
PendFuncTask disconnect_task =
548+
PendFuncTask([&connection, &disconnect_reason](Context& cx) -> Poll<> {
549+
Poll<DisconnectReason> poll = connection->PendDisconnect(cx);
550+
if (poll.IsPending()) {
551+
return Pending();
552+
}
553+
disconnect_reason = poll.value();
554+
return Ready();
555+
});
556+
async2_dispatcher().Post(disconnect_task);
557+
EXPECT_TRUE(async2_dispatcher().RunUntilStalled().IsPending());
558+
ASSERT_FALSE(disconnect_reason.has_value());
559+
560+
ASSERT_TRUE(adapter().fake_le()->Disconnect(peer->identifier()));
561+
ASSERT_EQ(adapter().fake_le()->connections().count(peer->identifier()), 0u);
562+
EXPECT_TRUE(async2_dispatcher().RunUntilStalled().IsReady());
563+
ASSERT_TRUE(disconnect_reason.has_value());
564+
EXPECT_EQ(disconnect_reason.value(), DisconnectReason::kFailure);
565+
566+
connection.reset();
567+
async_dispatcher().RunUntilIdle();
568+
}
569+
483570
} // namespace

0 commit comments

Comments
 (0)