Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(8)

Side by Side Diff: chromeos/dbus/bluetooth_le_advertisement_service_provider.cc

Issue 1052363005: Add DBus bindings for BLE Advertisement. (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: Created 5 years, 8 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
(Empty)
1 // Copyright 2015 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "bluetooth_le_advertisement_service_provider.h"
6
7 #include "base/bind.h"
8 #include "base/logging.h"
9 #include "base/memory/ref_counted.h"
10 #include "base/stl_util.h"
11 #include "base/threading/platform_thread.h"
12 #include "chromeos/dbus/dbus_thread_manager.h"
13 #include "dbus/exported_object.h"
14 #include "dbus/message.h"
15 #include "fake_bluetooth_le_advertisement_service_provider.h"
16 #include "third_party/cros_system_api/dbus/service_constants.h"
17
18 namespace chromeos {
19
20 namespace {
21 const char kErrorInvalidArgs[] = "org.freedesktop.DBus.Error.InvalidArgs";
22 } // namespace
23
24 // TODO(rkc): Remove these once the service constants header is updated.
25 namespace bluetooth_advertisement {
26 const char kBluetoothAdvertisementInterface[] = "";
27 const char kRelease[] = "";
28
29 const char kTypeProperty[] = "";
30 const char kServiceUUIDsProperty[] = "";
31 const char kManufacturerDataProperty[] = "";
32 const char kSolicitUUIDsProperty[] = "";
33 const char kServiceDataProperty[] = "";
34 }
35
36 // The BluetoothAdvertisementServiceProvider implementation used in production.
37 class BluetoothAdvertisementServiceProviderImpl
38 : public BluetoothLEAdvertisementServiceProvider {
39 public:
40 BluetoothAdvertisementServiceProviderImpl(
41 dbus::Bus* bus,
42 const dbus::ObjectPath& object_path,
43 Delegate* delegate,
44 AdvertisementType type,
45 scoped_ptr<UUIDList> service_uuids,
46 scoped_ptr<ManufacturerData> manufacturer_data,
47 scoped_ptr<UUIDList> solicit_uuids,
48 scoped_ptr<ServiceData> service_data)
49 : origin_thread_id_(base::PlatformThread::CurrentId()),
50 bus_(bus),
51 delegate_(delegate),
52 object_path_(object_path),
53 type_(type),
54 service_uuids_(service_uuids.Pass()),
55 manufacturer_data_(manufacturer_data.Pass()),
56 solicit_uuids_(solicit_uuids.Pass()),
57 service_data_(service_data.Pass()),
58 weak_ptr_factory_(this) {
59 DCHECK(bus);
60 DCHECK(delegate);
61
62 VLOG(1) << "Creating Bluetooth Advertisement: " << object_path_.value();
63
64 exported_object_ = bus_->GetExportedObject(object_path_);
65
66 // Export Bluetooth Advertisement interface methods.
67 exported_object_->ExportMethod(
68 bluetooth_advertisement::kBluetoothAdvertisementInterface,
69 bluetooth_advertisement::kRelease,
70 base::Bind(&BluetoothAdvertisementServiceProviderImpl::Release,
71 weak_ptr_factory_.GetWeakPtr()),
72 base::Bind(&BluetoothAdvertisementServiceProviderImpl::OnExported,
73 weak_ptr_factory_.GetWeakPtr()));
74
75 // Export dbus property methods.
76 exported_object_->ExportMethod(
77 dbus::kDBusPropertiesInterface, dbus::kDBusPropertiesGet,
78 base::Bind(&BluetoothAdvertisementServiceProviderImpl::Get,
79 weak_ptr_factory_.GetWeakPtr()),
80 base::Bind(&BluetoothAdvertisementServiceProviderImpl::OnExported,
81 weak_ptr_factory_.GetWeakPtr()));
82
83 exported_object_->ExportMethod(
84 dbus::kDBusPropertiesInterface, dbus::kDBusPropertiesGetAll,
85 base::Bind(&BluetoothAdvertisementServiceProviderImpl::GetAll,
86 weak_ptr_factory_.GetWeakPtr()),
87 base::Bind(&BluetoothAdvertisementServiceProviderImpl::OnExported,
88 weak_ptr_factory_.GetWeakPtr()));
89 }
90
91 ~BluetoothAdvertisementServiceProviderImpl() override {
92 VLOG(1) << "Cleaning up Bluetooth Advertisement: " << object_path_.value();
93
94 // Unregister the object path so we can reuse with a new agent.
95 bus_->UnregisterExportedObject(object_path_);
96 }
97
98 private:
99 // Returns true if the current thread is on the origin thread.
100 bool OnOriginThread() {
101 return base::PlatformThread::CurrentId() == origin_thread_id_;
102 }
103
104 // Called by dbus:: when this advertisement is unregistered from the Bluetooth
105 // daemon, generally by our request.
106 void Release(dbus::MethodCall* method_call,
107 dbus::ExportedObject::ResponseSender response_sender) {
108 DCHECK(OnOriginThread());
109 DCHECK(delegate_);
110
111 delegate_->Released();
112 }
113
114 // Called by dbus:: when the Bluetooth daemon fetches a single property of
115 // the descriptor.
116 void Get(dbus::MethodCall* method_call,
117 dbus::ExportedObject::ResponseSender response_sender) {
118 VLOG(2) << "BluetoothAdvertisementServiceProvider::Get: "
119 << object_path_.value();
120 DCHECK(OnOriginThread());
121
122 dbus::MessageReader reader(method_call);
123
124 std::string interface_name;
125 std::string property_name;
126 if (!reader.PopString(&interface_name) ||
127 !reader.PopString(&property_name) || reader.HasMoreData()) {
128 scoped_ptr<dbus::ErrorResponse> error_response =
129 dbus::ErrorResponse::FromMethodCall(method_call, kErrorInvalidArgs,
130 "Expected 'ss'.");
131 response_sender.Run(error_response.Pass());
132 return;
133 }
134
135 // Only the advertisement interface is supported.
136 if (interface_name !=
137 bluetooth_advertisement::kBluetoothAdvertisementInterface) {
138 scoped_ptr<dbus::ErrorResponse> error_response =
139 dbus::ErrorResponse::FromMethodCall(
140 method_call, kErrorInvalidArgs,
141 "No such interface: '" + interface_name + "'.");
142 response_sender.Run(error_response.Pass());
143 return;
144 }
145
146 scoped_ptr<dbus::Response> response =
147 dbus::Response::FromMethodCall(method_call);
148 dbus::MessageWriter writer(response.get());
149 dbus::MessageWriter variant_writer(NULL);
150
151 if (property_name == bluetooth_advertisement::kTypeProperty) {
152 writer.OpenVariant("s", &variant_writer);
153 if (type_ == ADVERTISEMENT_TYPE_BROADCAST) {
154 variant_writer.AppendString("broadcast");
155 } else {
156 variant_writer.AppendString("peripheral");
157 }
158 } else if ((property_name ==
159 bluetooth_advertisement::kServiceUUIDsProperty) &&
160 service_uuids_) {
161 writer.OpenVariant("as", &variant_writer);
162 variant_writer.AppendArrayOfStrings(*service_uuids_);
163 } else if ((property_name ==
164 bluetooth_advertisement::kSolicitUUIDsProperty) &&
165 solicit_uuids_) {
166 writer.OpenVariant("as", &variant_writer);
167 variant_writer.AppendArrayOfStrings(*solicit_uuids_);
168 } else if ((property_name ==
169 bluetooth_advertisement::kManufacturerDataProperty) &&
170 manufacturer_data_) {
171 writer.OpenVariant("o", &variant_writer);
172 AppendManufacturerDataVariant(&variant_writer);
173 } else if ((property_name ==
174 bluetooth_advertisement::kServiceDataProperty) &&
175 service_data_) {
176 writer.OpenVariant("o", &variant_writer);
177 AppendServiceDataVariant(&variant_writer);
178 } else {
179 scoped_ptr<dbus::ErrorResponse> error_response =
180 dbus::ErrorResponse::FromMethodCall(
181 method_call, kErrorInvalidArgs,
182 "No such property: '" + property_name + "'.");
183 response_sender.Run(error_response.Pass());
184 }
185
186 writer.CloseContainer(&variant_writer);
187 response_sender.Run(response.Pass());
188 }
189
190 // Called by dbus:: when the Bluetooth daemon fetches all properties of the
191 // descriptor.
192 void GetAll(dbus::MethodCall* method_call,
193 dbus::ExportedObject::ResponseSender response_sender) {
194 VLOG(2) << "BluetoothAdvertisementServiceProvider::GetAll: "
195 << object_path_.value();
196 DCHECK(OnOriginThread());
197
198 dbus::MessageReader reader(method_call);
199
200 std::string interface_name;
201 if (!reader.PopString(&interface_name) || reader.HasMoreData()) {
202 scoped_ptr<dbus::ErrorResponse> error_response =
203 dbus::ErrorResponse::FromMethodCall(method_call, kErrorInvalidArgs,
204 "Expected 's'.");
205 response_sender.Run(error_response.Pass());
206 return;
207 }
208
209 // Only the advertisement interface is supported.
210 if (interface_name !=
211 bluetooth_advertisement::kBluetoothAdvertisementInterface) {
212 scoped_ptr<dbus::ErrorResponse> error_response =
213 dbus::ErrorResponse::FromMethodCall(
214 method_call, kErrorInvalidArgs,
215 "No such interface: '" + interface_name + "'.");
216 response_sender.Run(error_response.Pass());
217 return;
218 }
219
220 response_sender.Run(CreateGetAllResponse(method_call).Pass());
221 }
222
223 // Called by dbus:: when a method is exported.
224 void OnExported(const std::string& interface_name,
225 const std::string& method_name,
226 bool success) {
227 LOG_IF(WARNING, !success) << "Failed to export " << interface_name << "."
228 << method_name;
229 }
230
231 // Helper for populating the DBus response with the advertisement data.
232 scoped_ptr<dbus::Response> CreateGetAllResponse(
233 dbus::MethodCall* method_call) {
234 VLOG(2) << "Descriptor value obtained from delegate. Responding to "
235 << "GetAll.";
236
237 scoped_ptr<dbus::Response> response =
238 dbus::Response::FromMethodCall(method_call);
239
240 dbus::MessageWriter writer(response.get());
241 dbus::MessageWriter array_writer(NULL);
242
243 writer.OpenArray("{sv}", &array_writer);
244
245 AppendType(&array_writer);
246 AppendServiceUUIDs(&array_writer);
247 AppendManufacturerData(&array_writer);
248 AppendSolicitUUIDs(&array_writer);
249 AppendServiceData(&array_writer);
250
251 writer.CloseContainer(&array_writer);
252 return response;
253 }
254
255 // Called by the Delegate in response to a successful method call to get the
256 // descriptor value.
257 void OnGet(dbus::MethodCall* method_call,
258 dbus::ExportedObject::ResponseSender response_sender,
259 const std::vector<uint8>& value) {
260 VLOG(2) << "Returning descriptor value obtained from delegate.";
261 scoped_ptr<dbus::Response> response =
262 dbus::Response::FromMethodCall(method_call);
263 dbus::MessageWriter writer(response.get());
264 dbus::MessageWriter variant_writer(NULL);
265
266 writer.OpenVariant("ay", &variant_writer);
267 variant_writer.AppendArrayOfBytes(value.data(), value.size());
268 writer.CloseContainer(&variant_writer);
269
270 response_sender.Run(response.Pass());
271 }
272
273 void AppendType(dbus::MessageWriter* array_writer) {
274 dbus::MessageWriter dict_entry_writer(NULL);
275 array_writer->OpenDictEntry(&dict_entry_writer);
276 dict_entry_writer.AppendString(bluetooth_advertisement::kTypeProperty);
277 if (type_ == ADVERTISEMENT_TYPE_BROADCAST) {
278 dict_entry_writer.AppendString("broadcast");
279 } else {
280 dict_entry_writer.AppendString("peripheral");
281 }
282 array_writer->CloseContainer(&dict_entry_writer);
283 }
284
285 void AppendServiceUUIDs(dbus::MessageWriter* array_writer) {
286 dbus::MessageWriter dict_entry_writer(NULL);
287 array_writer->OpenDictEntry(&dict_entry_writer);
288 dict_entry_writer.AppendString(
289 bluetooth_advertisement::kServiceUUIDsProperty);
290 dict_entry_writer.AppendArrayOfStrings(*service_uuids_);
291 array_writer->CloseContainer(&dict_entry_writer);
292 }
293
294 void AppendManufacturerData(dbus::MessageWriter* array_writer) {
295 dbus::MessageWriter dict_entry_writer(NULL);
296 array_writer->OpenDictEntry(&dict_entry_writer);
297 dict_entry_writer.AppendString(
298 bluetooth_advertisement::kManufacturerDataProperty);
299 dbus::MessageWriter variant_writer(NULL);
300 dict_entry_writer.OpenVariant("a{qay}", &variant_writer);
301 AppendManufacturerDataVariant(&variant_writer);
302 dict_entry_writer.CloseContainer(&variant_writer);
303 array_writer->CloseContainer(&dict_entry_writer);
304 }
305
306 void AppendSolicitUUIDs(dbus::MessageWriter* array_writer) {
307 dbus::MessageWriter dict_entry_writer(NULL);
308 array_writer->OpenDictEntry(&dict_entry_writer);
309 dict_entry_writer.AppendString(
310 bluetooth_advertisement::kSolicitUUIDsProperty);
311 dict_entry_writer.AppendArrayOfStrings(*solicit_uuids_);
312 array_writer->CloseContainer(&dict_entry_writer);
313 }
314
315 void AppendServiceData(dbus::MessageWriter* array_writer) {
316 dbus::MessageWriter dict_entry_writer(NULL);
317 array_writer->OpenDictEntry(&dict_entry_writer);
318 dict_entry_writer.AppendString(
319 bluetooth_advertisement::kServiceDataProperty);
320 dbus::MessageWriter variant_writer(NULL);
321 dict_entry_writer.OpenVariant("a{say}", &variant_writer);
322 AppendServiceDataVariant(&variant_writer);
323 dict_entry_writer.CloseContainer(&variant_writer);
324 array_writer->CloseContainer(&dict_entry_writer);
325 }
326
327 void AppendManufacturerDataVariant(dbus::MessageWriter* writer) {
328 dbus::MessageWriter array_writer(NULL);
329 writer->OpenArray("{qay}", &array_writer);
330 for (const auto& m : *manufacturer_data_) {
331 dbus::MessageWriter entry_writer(NULL);
332
333 array_writer.OpenDictEntry(&entry_writer);
334
335 entry_writer.AppendUint32(m.first);
336 entry_writer.AppendArrayOfBytes(vector_as_array(&m.second),
337 m.second.size());
338
339 array_writer.CloseContainer(&entry_writer);
340 }
341 writer->CloseContainer(&array_writer);
342 }
343
344 void AppendServiceDataVariant(dbus::MessageWriter* writer) {
345 dbus::MessageWriter array_writer(NULL);
346 writer->OpenArray("{say}", &array_writer);
347 for (const auto& m : *service_data_) {
348 dbus::MessageWriter entry_writer(NULL);
349
350 array_writer.OpenDictEntry(&entry_writer);
351
352 entry_writer.AppendString(m.first);
353 entry_writer.AppendArrayOfBytes(vector_as_array(&m.second),
354 m.second.size());
355
356 array_writer.CloseContainer(&entry_writer);
357 }
358 writer->CloseContainer(&array_writer);
359 }
360
361 // Origin thread (i.e. the UI thread in production).
362 base::PlatformThreadId origin_thread_id_;
363
364 // D-Bus bus object is exported on, not owned by this object and must
365 // outlive it.
366 dbus::Bus* bus_;
367
368 // All incoming method calls are passed on to the Delegate and a callback
369 // passed to generate the reply. |delegate_| is generally the object that
370 // owns this one, and must outlive it.
371 Delegate* delegate_;
372
373 // D-Bus object path of object we are exporting, kept so we can unregister
374 // again in our destructor.
375 dbus::ObjectPath object_path_;
376
377 // Advertisement data that needs to be provided to BlueZ when requested.
378 AdvertisementType type_;
379 scoped_ptr<UUIDList> service_uuids_;
380 scoped_ptr<ManufacturerData> manufacturer_data_;
381 scoped_ptr<UUIDList> solicit_uuids_;
382 scoped_ptr<ServiceData> service_data_;
383
384 // D-Bus object we are exporting, owned by this object.
385 scoped_refptr<dbus::ExportedObject> exported_object_;
386
387 // Weak pointer factory for generating 'this' pointers that might live longer
388 // than we do.
389 // Note: This should remain the last member so it'll be destroyed and
390 // invalidate its weak pointers before any other members are destroyed.
391 base::WeakPtrFactory<BluetoothAdvertisementServiceProviderImpl>
392 weak_ptr_factory_;
393
394 DISALLOW_COPY_AND_ASSIGN(BluetoothAdvertisementServiceProviderImpl);
395 };
396
397 BluetoothLEAdvertisementServiceProvider::
398 BluetoothLEAdvertisementServiceProvider() {
399 }
400
401 BluetoothLEAdvertisementServiceProvider::
402 ~BluetoothLEAdvertisementServiceProvider() {
403 }
404
405 // static
406 BluetoothLEAdvertisementServiceProvider*
407 BluetoothLEAdvertisementServiceProvider::Create(
408 dbus::Bus* bus,
409 const dbus::ObjectPath& object_path,
410 Delegate* delegate,
411 AdvertisementType type,
412 scoped_ptr<UUIDList> service_uuids,
413 scoped_ptr<ManufacturerData> manufacturer_data,
414 scoped_ptr<UUIDList> solicit_uuids,
415 scoped_ptr<ServiceData> service_data) {
416 if (!DBusThreadManager::Get()->IsUsingStub(DBusClientBundle::BLUETOOTH)) {
417 return new BluetoothAdvertisementServiceProviderImpl(
418 bus, object_path, delegate, type, service_uuids.Pass(),
419 manufacturer_data.Pass(), solicit_uuids.Pass(), service_data.Pass());
420 } else {
421 return new FakeBluetoothAdvertisementServiceProvider(object_path, delegate);
422 }
423 }
424
425 } // namespace chromeos
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698