OLD | NEW |
| (Empty) |
1 // Copyright 2013 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 "chromeos/dbus/bluetooth_agent_service_provider.h" | |
6 | |
7 | |
8 #include "base/bind.h" | |
9 #include "base/logging.h" | |
10 #include "base/memory/ref_counted.h" | |
11 #include "base/threading/platform_thread.h" | |
12 #include "chromeos/dbus/dbus_thread_manager.h" | |
13 #include "chromeos/dbus/fake_bluetooth_agent_service_provider.h" | |
14 #include "dbus/exported_object.h" | |
15 #include "dbus/message.h" | |
16 #include "third_party/cros_system_api/dbus/service_constants.h" | |
17 | |
18 namespace chromeos { | |
19 | |
20 // The BluetoothAgentServiceProvider implementation used in production. | |
21 class BluetoothAgentServiceProviderImpl | |
22 : public BluetoothAgentServiceProvider { | |
23 public: | |
24 BluetoothAgentServiceProviderImpl(dbus::Bus* bus, | |
25 const dbus::ObjectPath& object_path, | |
26 Delegate* delegate) | |
27 : origin_thread_id_(base::PlatformThread::CurrentId()), | |
28 bus_(bus), | |
29 delegate_(delegate), | |
30 object_path_(object_path), | |
31 weak_ptr_factory_(this) { | |
32 VLOG(1) << "Creating Bluetooth Agent: " << object_path_.value(); | |
33 | |
34 exported_object_ = bus_->GetExportedObject(object_path_); | |
35 | |
36 exported_object_->ExportMethod( | |
37 bluetooth_agent::kBluetoothAgentInterface, | |
38 bluetooth_agent::kRelease, | |
39 base::Bind(&BluetoothAgentServiceProviderImpl::Release, | |
40 weak_ptr_factory_.GetWeakPtr()), | |
41 base::Bind(&BluetoothAgentServiceProviderImpl::OnExported, | |
42 weak_ptr_factory_.GetWeakPtr())); | |
43 | |
44 exported_object_->ExportMethod( | |
45 bluetooth_agent::kBluetoothAgentInterface, | |
46 bluetooth_agent::kRequestPinCode, | |
47 base::Bind(&BluetoothAgentServiceProviderImpl::RequestPinCode, | |
48 weak_ptr_factory_.GetWeakPtr()), | |
49 base::Bind(&BluetoothAgentServiceProviderImpl::OnExported, | |
50 weak_ptr_factory_.GetWeakPtr())); | |
51 | |
52 exported_object_->ExportMethod( | |
53 bluetooth_agent::kBluetoothAgentInterface, | |
54 bluetooth_agent::kDisplayPinCode, | |
55 base::Bind(&BluetoothAgentServiceProviderImpl::DisplayPinCode, | |
56 weak_ptr_factory_.GetWeakPtr()), | |
57 base::Bind(&BluetoothAgentServiceProviderImpl::OnExported, | |
58 weak_ptr_factory_.GetWeakPtr())); | |
59 | |
60 exported_object_->ExportMethod( | |
61 bluetooth_agent::kBluetoothAgentInterface, | |
62 bluetooth_agent::kRequestPasskey, | |
63 base::Bind(&BluetoothAgentServiceProviderImpl::RequestPasskey, | |
64 weak_ptr_factory_.GetWeakPtr()), | |
65 base::Bind(&BluetoothAgentServiceProviderImpl::OnExported, | |
66 weak_ptr_factory_.GetWeakPtr())); | |
67 | |
68 exported_object_->ExportMethod( | |
69 bluetooth_agent::kBluetoothAgentInterface, | |
70 bluetooth_agent::kDisplayPasskey, | |
71 base::Bind(&BluetoothAgentServiceProviderImpl::DisplayPasskey, | |
72 weak_ptr_factory_.GetWeakPtr()), | |
73 base::Bind(&BluetoothAgentServiceProviderImpl::OnExported, | |
74 weak_ptr_factory_.GetWeakPtr())); | |
75 | |
76 exported_object_->ExportMethod( | |
77 bluetooth_agent::kBluetoothAgentInterface, | |
78 bluetooth_agent::kRequestConfirmation, | |
79 base::Bind(&BluetoothAgentServiceProviderImpl::RequestConfirmation, | |
80 weak_ptr_factory_.GetWeakPtr()), | |
81 base::Bind(&BluetoothAgentServiceProviderImpl::OnExported, | |
82 weak_ptr_factory_.GetWeakPtr())); | |
83 | |
84 exported_object_->ExportMethod( | |
85 bluetooth_agent::kBluetoothAgentInterface, | |
86 bluetooth_agent::kRequestAuthorization, | |
87 base::Bind(&BluetoothAgentServiceProviderImpl::RequestAuthorization, | |
88 weak_ptr_factory_.GetWeakPtr()), | |
89 base::Bind(&BluetoothAgentServiceProviderImpl::OnExported, | |
90 weak_ptr_factory_.GetWeakPtr())); | |
91 | |
92 exported_object_->ExportMethod( | |
93 bluetooth_agent::kBluetoothAgentInterface, | |
94 bluetooth_agent::kAuthorizeService, | |
95 base::Bind(&BluetoothAgentServiceProviderImpl::AuthorizeService, | |
96 weak_ptr_factory_.GetWeakPtr()), | |
97 base::Bind(&BluetoothAgentServiceProviderImpl::OnExported, | |
98 weak_ptr_factory_.GetWeakPtr())); | |
99 | |
100 exported_object_->ExportMethod( | |
101 bluetooth_agent::kBluetoothAgentInterface, | |
102 bluetooth_agent::kCancel, | |
103 base::Bind(&BluetoothAgentServiceProviderImpl::Cancel, | |
104 weak_ptr_factory_.GetWeakPtr()), | |
105 base::Bind(&BluetoothAgentServiceProviderImpl::OnExported, | |
106 weak_ptr_factory_.GetWeakPtr())); | |
107 } | |
108 | |
109 ~BluetoothAgentServiceProviderImpl() override { | |
110 VLOG(1) << "Cleaning up Bluetooth Agent: " << object_path_.value(); | |
111 | |
112 // Unregister the object path so we can reuse with a new agent. | |
113 bus_->UnregisterExportedObject(object_path_); | |
114 } | |
115 | |
116 private: | |
117 // Returns true if the current thread is on the origin thread. | |
118 bool OnOriginThread() { | |
119 return base::PlatformThread::CurrentId() == origin_thread_id_; | |
120 } | |
121 | |
122 // Called by dbus:: when the agent is unregistered from the Bluetooth | |
123 // daemon, generally at the end of a pairing request. | |
124 void Release(dbus::MethodCall* method_call, | |
125 dbus::ExportedObject::ResponseSender response_sender) { | |
126 DCHECK(OnOriginThread()); | |
127 DCHECK(delegate_); | |
128 | |
129 delegate_->Released(); | |
130 | |
131 response_sender.Run(dbus::Response::FromMethodCall(method_call)); | |
132 } | |
133 | |
134 // Called by dbus:: when the Bluetooth daemon requires a PIN Code for | |
135 // device authentication. | |
136 void RequestPinCode(dbus::MethodCall* method_call, | |
137 dbus::ExportedObject::ResponseSender response_sender) { | |
138 DCHECK(OnOriginThread()); | |
139 DCHECK(delegate_); | |
140 | |
141 dbus::MessageReader reader(method_call); | |
142 dbus::ObjectPath device_path; | |
143 if (!reader.PopObjectPath(&device_path)) { | |
144 LOG(WARNING) << "RequestPinCode called with incorrect paramters: " | |
145 << method_call->ToString(); | |
146 return; | |
147 } | |
148 | |
149 Delegate::PinCodeCallback callback = base::Bind( | |
150 &BluetoothAgentServiceProviderImpl::OnPinCode, | |
151 weak_ptr_factory_.GetWeakPtr(), | |
152 method_call, | |
153 response_sender); | |
154 | |
155 delegate_->RequestPinCode(device_path, callback); | |
156 } | |
157 | |
158 // Called by dbus:: when the Bluetooth daemon requires that the user | |
159 // enter a PIN Code into the remote device so that it may be | |
160 // authenticated. | |
161 void DisplayPinCode(dbus::MethodCall* method_call, | |
162 dbus::ExportedObject::ResponseSender response_sender) { | |
163 DCHECK(OnOriginThread()); | |
164 DCHECK(delegate_); | |
165 | |
166 dbus::MessageReader reader(method_call); | |
167 dbus::ObjectPath device_path; | |
168 std::string pincode; | |
169 if (!reader.PopObjectPath(&device_path) || | |
170 !reader.PopString(&pincode)) { | |
171 LOG(WARNING) << "DisplayPinCode called with incorrect paramters: " | |
172 << method_call->ToString(); | |
173 return; | |
174 } | |
175 | |
176 delegate_->DisplayPinCode(device_path, pincode); | |
177 | |
178 response_sender.Run(dbus::Response::FromMethodCall(method_call)); | |
179 } | |
180 | |
181 // Called by dbus:: when the Bluetooth daemon requires a Passkey for | |
182 // device authentication. | |
183 void RequestPasskey(dbus::MethodCall* method_call, | |
184 dbus::ExportedObject::ResponseSender response_sender) { | |
185 DCHECK(OnOriginThread()); | |
186 DCHECK(delegate_); | |
187 | |
188 dbus::MessageReader reader(method_call); | |
189 dbus::ObjectPath device_path; | |
190 if (!reader.PopObjectPath(&device_path)) { | |
191 LOG(WARNING) << "RequestPasskey called with incorrect paramters: " | |
192 << method_call->ToString(); | |
193 return; | |
194 } | |
195 | |
196 Delegate::PasskeyCallback callback = base::Bind( | |
197 &BluetoothAgentServiceProviderImpl::OnPasskey, | |
198 weak_ptr_factory_.GetWeakPtr(), | |
199 method_call, | |
200 response_sender); | |
201 | |
202 delegate_->RequestPasskey(device_path, callback); | |
203 } | |
204 | |
205 // Called by dbus:: when the Bluetooth daemon requires that the user | |
206 // enter a Passkey into the remote device so that it may be | |
207 // authenticated. | |
208 void DisplayPasskey(dbus::MethodCall* method_call, | |
209 dbus::ExportedObject::ResponseSender response_sender) { | |
210 DCHECK(OnOriginThread()); | |
211 DCHECK(delegate_); | |
212 | |
213 dbus::MessageReader reader(method_call); | |
214 dbus::ObjectPath device_path; | |
215 uint32 passkey; | |
216 uint16 entered; | |
217 if (!reader.PopObjectPath(&device_path) || | |
218 !reader.PopUint32(&passkey) || | |
219 !reader.PopUint16(&entered)) { | |
220 LOG(WARNING) << "DisplayPasskey called with incorrect paramters: " | |
221 << method_call->ToString(); | |
222 return; | |
223 } | |
224 | |
225 delegate_->DisplayPasskey(device_path, passkey, entered); | |
226 | |
227 response_sender.Run(dbus::Response::FromMethodCall(method_call)); | |
228 } | |
229 | |
230 // Called by dbus:: when the Bluetooth daemon requires that the user | |
231 // confirm that a Passkey is displayed on the screen of the remote | |
232 // device so that it may be authenticated. | |
233 void RequestConfirmation( | |
234 dbus::MethodCall* method_call, | |
235 dbus::ExportedObject::ResponseSender response_sender) { | |
236 DCHECK(OnOriginThread()); | |
237 DCHECK(delegate_); | |
238 | |
239 dbus::MessageReader reader(method_call); | |
240 dbus::ObjectPath device_path; | |
241 uint32 passkey; | |
242 if (!reader.PopObjectPath(&device_path) || | |
243 !reader.PopUint32(&passkey)) { | |
244 LOG(WARNING) << "RequestConfirmation called with incorrect paramters: " | |
245 << method_call->ToString(); | |
246 return; | |
247 } | |
248 | |
249 Delegate::ConfirmationCallback callback = base::Bind( | |
250 &BluetoothAgentServiceProviderImpl::OnConfirmation, | |
251 weak_ptr_factory_.GetWeakPtr(), | |
252 method_call, | |
253 response_sender); | |
254 | |
255 delegate_->RequestConfirmation(device_path, passkey, callback); | |
256 } | |
257 | |
258 // Called by dbus:: when the Bluetooth daemon requires that the user | |
259 // confirm an incoming just-works pairing. | |
260 void RequestAuthorization( | |
261 dbus::MethodCall* method_call, | |
262 dbus::ExportedObject::ResponseSender response_sender) { | |
263 DCHECK(OnOriginThread()); | |
264 DCHECK(delegate_); | |
265 | |
266 dbus::MessageReader reader(method_call); | |
267 dbus::ObjectPath device_path; | |
268 if (!reader.PopObjectPath(&device_path)) { | |
269 LOG(WARNING) << "RequestAuthorization called with incorrect paramters: " | |
270 << method_call->ToString(); | |
271 return; | |
272 } | |
273 | |
274 Delegate::ConfirmationCallback callback = base::Bind( | |
275 &BluetoothAgentServiceProviderImpl::OnConfirmation, | |
276 weak_ptr_factory_.GetWeakPtr(), | |
277 method_call, | |
278 response_sender); | |
279 | |
280 delegate_->RequestAuthorization(device_path, callback); | |
281 } | |
282 | |
283 // Called by dbus:: when the Bluetooth daemon requires that the user | |
284 // confirm that that a remote device is authorized to connect to a service | |
285 // UUID. | |
286 void AuthorizeService(dbus::MethodCall* method_call, | |
287 dbus::ExportedObject::ResponseSender response_sender) { | |
288 DCHECK(OnOriginThread()); | |
289 DCHECK(delegate_); | |
290 | |
291 dbus::MessageReader reader(method_call); | |
292 dbus::ObjectPath device_path; | |
293 std::string uuid; | |
294 if (!reader.PopObjectPath(&device_path) || | |
295 !reader.PopString(&uuid)) { | |
296 LOG(WARNING) << "AuthorizeService called with incorrect paramters: " | |
297 << method_call->ToString(); | |
298 return; | |
299 } | |
300 | |
301 Delegate::ConfirmationCallback callback = base::Bind( | |
302 &BluetoothAgentServiceProviderImpl::OnConfirmation, | |
303 weak_ptr_factory_.GetWeakPtr(), | |
304 method_call, | |
305 response_sender); | |
306 | |
307 delegate_->AuthorizeService(device_path, uuid, callback); | |
308 } | |
309 | |
310 // Called by dbus:: when the request failed before a reply was returned | |
311 // from the device. | |
312 void Cancel(dbus::MethodCall* method_call, | |
313 dbus::ExportedObject::ResponseSender response_sender) { | |
314 DCHECK(OnOriginThread()); | |
315 DCHECK(delegate_); | |
316 | |
317 delegate_->Cancel(); | |
318 | |
319 response_sender.Run(dbus::Response::FromMethodCall(method_call)); | |
320 } | |
321 | |
322 // Called by dbus:: when a method is exported. | |
323 void OnExported(const std::string& interface_name, | |
324 const std::string& method_name, | |
325 bool success) { | |
326 LOG_IF(WARNING, !success) << "Failed to export " | |
327 << interface_name << "." << method_name; | |
328 } | |
329 | |
330 // Called by the Delegate to response to a method requesting a PIN code. | |
331 void OnPinCode(dbus::MethodCall* method_call, | |
332 dbus::ExportedObject::ResponseSender response_sender, | |
333 Delegate::Status status, | |
334 const std::string& pincode) { | |
335 DCHECK(OnOriginThread()); | |
336 | |
337 switch (status) { | |
338 case Delegate::SUCCESS: { | |
339 scoped_ptr<dbus::Response> response( | |
340 dbus::Response::FromMethodCall(method_call)); | |
341 dbus::MessageWriter writer(response.get()); | |
342 writer.AppendString(pincode); | |
343 response_sender.Run(response.Pass()); | |
344 break; | |
345 } | |
346 case Delegate::REJECTED: { | |
347 response_sender.Run(dbus::ErrorResponse::FromMethodCall( | |
348 method_call, bluetooth_agent::kErrorRejected, "rejected")); | |
349 break; | |
350 } | |
351 case Delegate::CANCELLED: { | |
352 response_sender.Run(dbus::ErrorResponse::FromMethodCall( | |
353 method_call, bluetooth_agent::kErrorCanceled, "canceled")); | |
354 break; | |
355 } | |
356 default: | |
357 NOTREACHED() << "Unexpected status code from delegate: " << status; | |
358 } | |
359 } | |
360 | |
361 // Called by the Delegate to response to a method requesting a Passkey. | |
362 void OnPasskey(dbus::MethodCall* method_call, | |
363 dbus::ExportedObject::ResponseSender response_sender, | |
364 Delegate::Status status, | |
365 uint32 passkey) { | |
366 DCHECK(OnOriginThread()); | |
367 | |
368 switch (status) { | |
369 case Delegate::SUCCESS: { | |
370 scoped_ptr<dbus::Response> response( | |
371 dbus::Response::FromMethodCall(method_call)); | |
372 dbus::MessageWriter writer(response.get()); | |
373 writer.AppendUint32(passkey); | |
374 response_sender.Run(response.Pass()); | |
375 break; | |
376 } | |
377 case Delegate::REJECTED: { | |
378 response_sender.Run(dbus::ErrorResponse::FromMethodCall( | |
379 method_call, bluetooth_agent::kErrorRejected, "rejected")); | |
380 break; | |
381 } | |
382 case Delegate::CANCELLED: { | |
383 response_sender.Run(dbus::ErrorResponse::FromMethodCall( | |
384 method_call, bluetooth_agent::kErrorCanceled, "canceled")); | |
385 break; | |
386 } | |
387 default: | |
388 NOTREACHED() << "Unexpected status code from delegate: " << status; | |
389 } | |
390 } | |
391 | |
392 // Called by the Delegate in response to a method requiring confirmation. | |
393 void OnConfirmation(dbus::MethodCall* method_call, | |
394 dbus::ExportedObject::ResponseSender response_sender, | |
395 Delegate::Status status) { | |
396 DCHECK(OnOriginThread()); | |
397 | |
398 switch (status) { | |
399 case Delegate::SUCCESS: { | |
400 response_sender.Run(dbus::Response::FromMethodCall(method_call)); | |
401 break; | |
402 } | |
403 case Delegate::REJECTED: { | |
404 response_sender.Run(dbus::ErrorResponse::FromMethodCall( | |
405 method_call, bluetooth_agent::kErrorRejected, "rejected")); | |
406 break; | |
407 } | |
408 case Delegate::CANCELLED: { | |
409 response_sender.Run(dbus::ErrorResponse::FromMethodCall( | |
410 method_call, bluetooth_agent::kErrorCanceled, "canceled")); | |
411 break; | |
412 } | |
413 default: | |
414 NOTREACHED() << "Unexpected status code from delegate: " << status; | |
415 } | |
416 } | |
417 | |
418 // Origin thread (i.e. the UI thread in production). | |
419 base::PlatformThreadId origin_thread_id_; | |
420 | |
421 // D-Bus bus object is exported on, not owned by this object and must | |
422 // outlive it. | |
423 dbus::Bus* bus_; | |
424 | |
425 // All incoming method calls are passed on to the Delegate and a callback | |
426 // passed to generate the reply. |delegate_| is generally the object that | |
427 // owns this one, and must outlive it. | |
428 Delegate* delegate_; | |
429 | |
430 // D-Bus object path of object we are exporting, kept so we can unregister | |
431 // again in our destructor. | |
432 dbus::ObjectPath object_path_; | |
433 | |
434 // D-Bus object we are exporting, owned by this object. | |
435 scoped_refptr<dbus::ExportedObject> exported_object_; | |
436 | |
437 // Weak pointer factory for generating 'this' pointers that might live longer | |
438 // than we do. | |
439 // Note: This should remain the last member so it'll be destroyed and | |
440 // invalidate its weak pointers before any other members are destroyed. | |
441 base::WeakPtrFactory<BluetoothAgentServiceProviderImpl> weak_ptr_factory_; | |
442 | |
443 DISALLOW_COPY_AND_ASSIGN(BluetoothAgentServiceProviderImpl); | |
444 }; | |
445 | |
446 BluetoothAgentServiceProvider::BluetoothAgentServiceProvider() { | |
447 } | |
448 | |
449 BluetoothAgentServiceProvider::~BluetoothAgentServiceProvider() { | |
450 } | |
451 | |
452 // static | |
453 BluetoothAgentServiceProvider* BluetoothAgentServiceProvider::Create( | |
454 dbus::Bus* bus, | |
455 const dbus::ObjectPath& object_path, | |
456 Delegate* delegate) { | |
457 if (!DBusThreadManager::Get()->IsUsingStub(DBusClientBundle::BLUETOOTH)) { | |
458 return new BluetoothAgentServiceProviderImpl(bus, object_path, delegate); | |
459 } else { | |
460 return new FakeBluetoothAgentServiceProvider(object_path, delegate); | |
461 } | |
462 } | |
463 | |
464 } // namespace chromeos | |
OLD | NEW |