14
14
15
15
#include " pw_bluetooth_sapphire/central.h"
16
16
17
+ #include " pw_bluetooth_sapphire/internal/connection_options.h"
18
+ #include " pw_bluetooth_sapphire/internal/uuid.h"
19
+
17
20
namespace pw ::bluetooth_sapphire {
18
21
namespace {
19
22
20
23
pw::sync::Mutex g_peripheral_lock;
21
24
22
- bt::UUID UuidFrom (const pw::bluetooth::Uuid& uuid) {
23
- return bt::UUID (bt::BufferView (pw::as_bytes (uuid.As128BitSpan ())));
24
- }
25
-
26
25
bt::gap::DiscoveryFilter DiscoveryFilterFrom (const Central::ScanFilter& in) {
27
26
bt::gap::DiscoveryFilter out;
28
27
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 ());
30
29
out.set_service_uuids (std::vector<bt::UUID>{std::move (uuid)});
31
30
}
32
31
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 ());
34
33
out.set_service_data_uuids (std::vector<bt::UUID>{std::move (uuid)});
35
34
}
36
35
if (in.manufacturer_id .has_value ()) {
@@ -46,7 +45,7 @@ bt::gap::DiscoveryFilter DiscoveryFilterFrom(const Central::ScanFilter& in) {
46
45
out.set_pathloss (in.max_path_loss .value ());
47
46
}
48
47
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 ());
50
49
out.set_solicitation_uuids (std::vector<bt::UUID>{std::move (uuid)});
51
50
}
52
51
return out;
@@ -108,6 +107,7 @@ Central::Central(bt::gap::Adapter::WeakPtr adapter,
108
107
pw::multibuf::MultiBufAllocator& allocator)
109
108
: adapter_(std::move(adapter)),
110
109
dispatcher_ (dispatcher),
110
+ heap_dispatcher_(dispatcher),
111
111
allocator_(allocator),
112
112
weak_factory_(this ),
113
113
self_(weak_factory_.GetWeakPtr()) {}
@@ -118,11 +118,41 @@ Central::~Central() {
118
118
}
119
119
120
120
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);
126
156
}
127
157
128
158
async2::OnceReceiver<Central::ScanStartResult> Central::Scan (
@@ -190,7 +220,7 @@ async2::OnceReceiver<Central::ScanStartResult> Central::Scan(
190
220
self->adapter_ ->le ()->StartDiscovery (active, std::move (cb));
191
221
}
192
222
};
193
- Status post_status = dispatcher_ .Post (std::move (task_fn));
223
+ Status post_status = heap_dispatcher_ .Post (std::move (task_fn));
194
224
PW_CHECK_OK (post_status);
195
225
196
226
return std::move (result_receiver);
@@ -295,7 +325,7 @@ void Central::StopScanLocked(uint16_t scan_id) {
295
325
}
296
326
iter->second .OnScanHandleDestroyedLocked ();
297
327
298
- pw::Status post_status = dispatcher_ .Post (
328
+ pw::Status post_status = heap_dispatcher_ .Post (
299
329
[self = self_, scan_id](pw::async::Context, pw::Status status) {
300
330
if (!status.ok () || !self.is_alive ()) {
301
331
return ;
@@ -306,4 +336,24 @@ void Central::StopScanLocked(uint16_t scan_id) {
306
336
PW_CHECK (post_status.ok ());
307
337
}
308
338
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
+
309
359
} // namespace pw::bluetooth_sapphire
0 commit comments