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_profile_service_provider.h"
6
7 #include <string>
8
9 #include "base/bind.h"
10 #include "base/logging.h"
11 #include "base/memory/ref_counted.h"
12 #include "base/sys_info.h"
13 #include "base/threading/platform_thread.h"
14 #include "chromeos/dbus/fake_bluetooth_profile_service_provider.h"
15 #include "dbus/bus.h"
16 #include "dbus/exported_object.h"
17 #include "dbus/message.h"
18 #include "dbus/object_path.h"
19 #include "third_party/cros_system_api/dbus/service_constants.h"
20
21 namespace chromeos {
22
23 // The BluetoothProfileServiceProvider implementation used in production.
24 class BluetoothProfileServiceProviderImpl
25 : public BluetoothProfileServiceProvider {
26 public:
BluetoothProfileServiceProviderImpl(dbus::Bus * bus,const dbus::ObjectPath & object_path,Delegate * delegate)27 BluetoothProfileServiceProviderImpl(dbus::Bus* bus,
28 const dbus::ObjectPath& object_path,
29 Delegate* delegate)
30 : origin_thread_id_(base::PlatformThread::CurrentId()),
31 bus_(bus),
32 delegate_(delegate),
33 object_path_(object_path),
34 weak_ptr_factory_(this) {
35 VLOG(1) << "Creating Bluetooth Profile: " << object_path_.value();
36
37 exported_object_ = bus_->GetExportedObject(object_path_);
38
39 exported_object_->ExportMethod(
40 bluetooth_profile::kBluetoothProfileInterface,
41 bluetooth_profile::kRelease,
42 base::Bind(&BluetoothProfileServiceProviderImpl::Release,
43 weak_ptr_factory_.GetWeakPtr()),
44 base::Bind(&BluetoothProfileServiceProviderImpl::OnExported,
45 weak_ptr_factory_.GetWeakPtr()));
46
47 exported_object_->ExportMethod(
48 bluetooth_profile::kBluetoothProfileInterface,
49 bluetooth_profile::kNewConnection,
50 base::Bind(&BluetoothProfileServiceProviderImpl::NewConnection,
51 weak_ptr_factory_.GetWeakPtr()),
52 base::Bind(&BluetoothProfileServiceProviderImpl::OnExported,
53 weak_ptr_factory_.GetWeakPtr()));
54
55 exported_object_->ExportMethod(
56 bluetooth_profile::kBluetoothProfileInterface,
57 bluetooth_profile::kRequestDisconnection,
58 base::Bind(&BluetoothProfileServiceProviderImpl::RequestDisconnection,
59 weak_ptr_factory_.GetWeakPtr()),
60 base::Bind(&BluetoothProfileServiceProviderImpl::OnExported,
61 weak_ptr_factory_.GetWeakPtr()));
62
63 exported_object_->ExportMethod(
64 bluetooth_profile::kBluetoothProfileInterface,
65 bluetooth_profile::kCancel,
66 base::Bind(&BluetoothProfileServiceProviderImpl::Cancel,
67 weak_ptr_factory_.GetWeakPtr()),
68 base::Bind(&BluetoothProfileServiceProviderImpl::OnExported,
69 weak_ptr_factory_.GetWeakPtr()));
70 }
71
~BluetoothProfileServiceProviderImpl()72 virtual ~BluetoothProfileServiceProviderImpl() {
73 VLOG(1) << "Cleaning up Bluetooth Profile: " << object_path_.value();
74
75 // Unregister the object path so we can reuse with a new agent.
76 bus_->UnregisterExportedObject(object_path_);
77 }
78
79 private:
80 // Returns true if the current thread is on the origin thread.
OnOriginThread()81 bool OnOriginThread() {
82 return base::PlatformThread::CurrentId() == origin_thread_id_;
83 }
84
85 // Called by dbus:: when the profile is unregistered from the Bluetooth
86 // daemon, generally by our request.
Release(dbus::MethodCall * method_call,dbus::ExportedObject::ResponseSender response_sender)87 void Release(dbus::MethodCall* method_call,
88 dbus::ExportedObject::ResponseSender response_sender) {
89 DCHECK(OnOriginThread());
90 DCHECK(delegate_);
91
92 delegate_->Release();
93
94 response_sender.Run(dbus::Response::FromMethodCall(method_call));
95 }
96
97 // Called by dbus:: when the Bluetooth daemon establishes a new connection
98 // to the profile.
NewConnection(dbus::MethodCall * method_call,dbus::ExportedObject::ResponseSender response_sender)99 void NewConnection(dbus::MethodCall* method_call,
100 dbus::ExportedObject::ResponseSender response_sender) {
101 DCHECK(OnOriginThread());
102 DCHECK(delegate_);
103
104 dbus::MessageReader reader(method_call);
105 dbus::ObjectPath device_path;
106 scoped_ptr<dbus::FileDescriptor> fd(new dbus::FileDescriptor());
107 dbus::MessageReader array_reader(NULL);
108 if (!reader.PopObjectPath(&device_path) ||
109 !reader.PopFileDescriptor(fd.get()) ||
110 !reader.PopArray(&array_reader)) {
111 LOG(WARNING) << "NewConnection called with incorrect paramters: "
112 << method_call->ToString();
113 return;
114 }
115
116 Delegate::Options options;
117 while (array_reader.HasMoreData()) {
118 dbus::MessageReader dict_entry_reader(NULL);
119 std::string key;
120 if (!array_reader.PopDictEntry(&dict_entry_reader) ||
121 !dict_entry_reader.PopString(&key)) {
122 LOG(WARNING) << "NewConnection called with incorrect paramters: "
123 << method_call->ToString();
124 } else {
125 if (key == bluetooth_profile::kVersionProperty)
126 dict_entry_reader.PopVariantOfUint16(&options.version);
127 else if (key == bluetooth_profile::kFeaturesProperty)
128 dict_entry_reader.PopVariantOfUint16(&options.features);
129 }
130 }
131
132 Delegate::ConfirmationCallback callback = base::Bind(
133 &BluetoothProfileServiceProviderImpl::OnConfirmation,
134 weak_ptr_factory_.GetWeakPtr(),
135 method_call,
136 response_sender);
137
138 delegate_->NewConnection(device_path, fd.Pass(), options, callback);
139 }
140
141 // Called by dbus:: when the Bluetooth daemon is about to disconnect the
142 // profile.
RequestDisconnection(dbus::MethodCall * method_call,dbus::ExportedObject::ResponseSender response_sender)143 void RequestDisconnection(
144 dbus::MethodCall* method_call,
145 dbus::ExportedObject::ResponseSender response_sender) {
146 DCHECK(OnOriginThread());
147 DCHECK(delegate_);
148
149 dbus::MessageReader reader(method_call);
150 dbus::ObjectPath device_path;
151 if (!reader.PopObjectPath(&device_path)) {
152 LOG(WARNING) << "RequestDisconnection called with incorrect paramters: "
153 << method_call->ToString();
154 return;
155 }
156
157 Delegate::ConfirmationCallback callback = base::Bind(
158 &BluetoothProfileServiceProviderImpl::OnConfirmation,
159 weak_ptr_factory_.GetWeakPtr(),
160 method_call,
161 response_sender);
162
163 delegate_->RequestDisconnection(device_path, callback);
164 }
165
166 // Called by dbus:: when the request failed before a reply was returned
167 // from the device.
Cancel(dbus::MethodCall * method_call,dbus::ExportedObject::ResponseSender response_sender)168 void Cancel(dbus::MethodCall* method_call,
169 dbus::ExportedObject::ResponseSender response_sender) {
170 DCHECK(OnOriginThread());
171 DCHECK(delegate_);
172
173 delegate_->Cancel();
174
175 response_sender.Run(dbus::Response::FromMethodCall(method_call));
176 }
177
178 // Called by dbus:: when a method is exported.
OnExported(const std::string & interface_name,const std::string & method_name,bool success)179 void OnExported(const std::string& interface_name,
180 const std::string& method_name,
181 bool success) {
182 LOG_IF(WARNING, !success) << "Failed to export "
183 << interface_name << "." << method_name;
184 }
185
186 // Called by the Delegate in response to a method requiring confirmation.
OnConfirmation(dbus::MethodCall * method_call,dbus::ExportedObject::ResponseSender response_sender,Delegate::Status status)187 void OnConfirmation(dbus::MethodCall* method_call,
188 dbus::ExportedObject::ResponseSender response_sender,
189 Delegate::Status status) {
190 DCHECK(OnOriginThread());
191
192 switch (status) {
193 case Delegate::SUCCESS: {
194 response_sender.Run(dbus::Response::FromMethodCall(method_call));
195 break;
196 }
197 case Delegate::REJECTED: {
198 response_sender.Run(
199 dbus::ErrorResponse::FromMethodCall(
200 method_call, bluetooth_profile::kErrorRejected, "rejected")
201 .PassAs<dbus::Response>());
202 break;
203 }
204 case Delegate::CANCELLED: {
205 response_sender.Run(
206 dbus::ErrorResponse::FromMethodCall(
207 method_call, bluetooth_profile::kErrorCanceled, "canceled")
208 .PassAs<dbus::Response>());
209 break;
210 }
211 default:
212 NOTREACHED() << "Unexpected status code from delegate: " << status;
213 }
214 }
215
216 // Origin thread (i.e. the UI thread in production).
217 base::PlatformThreadId origin_thread_id_;
218
219 // D-Bus bus object is exported on, not owned by this object and must
220 // outlive it.
221 dbus::Bus* bus_;
222
223 // All incoming method calls are passed on to the Delegate and a callback
224 // passed to generate the reply. |delegate_| is generally the object that
225 // owns this one, and must outlive it.
226 Delegate* delegate_;
227
228 // D-Bus object path of object we are exporting, kept so we can unregister
229 // again in our destructor.
230 dbus::ObjectPath object_path_;
231
232 // D-Bus object we are exporting, owned by this object.
233 scoped_refptr<dbus::ExportedObject> exported_object_;
234
235 // Weak pointer factory for generating 'this' pointers that might live longer
236 // than we do.
237 // Note: This should remain the last member so it'll be destroyed and
238 // invalidate its weak pointers before any other members are destroyed.
239 base::WeakPtrFactory<BluetoothProfileServiceProviderImpl> weak_ptr_factory_;
240
241 DISALLOW_COPY_AND_ASSIGN(BluetoothProfileServiceProviderImpl);
242 };
243
BluetoothProfileServiceProvider()244 BluetoothProfileServiceProvider::BluetoothProfileServiceProvider() {
245 }
246
~BluetoothProfileServiceProvider()247 BluetoothProfileServiceProvider::~BluetoothProfileServiceProvider() {
248 }
249
250 // static
Create(dbus::Bus * bus,const dbus::ObjectPath & object_path,Delegate * delegate)251 BluetoothProfileServiceProvider* BluetoothProfileServiceProvider::Create(
252 dbus::Bus* bus,
253 const dbus::ObjectPath& object_path,
254 Delegate* delegate) {
255 if (base::SysInfo::IsRunningOnChromeOS()) {
256 return new BluetoothProfileServiceProviderImpl(bus, object_path, delegate);
257 } else {
258 return new FakeBluetoothProfileServiceProvider(object_path, delegate);
259 }
260 }
261
262 } // namespace chromeos
263