1 // Copyright (c) 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.
5 #include "chromeos/dbus/fake_bluetooth_adapter_client.h"
7 #include "base/location.h"
8 #include "base/logging.h"
9 #include "base/single_thread_task_runner.h"
10 #include "base/thread_task_runner_handle.h"
11 #include "base/time/time.h"
12 #include "chromeos/dbus/dbus_thread_manager.h"
13 #include "chromeos/dbus/fake_bluetooth_device_client.h"
14 #include "third_party/cros_system_api/dbus/service_constants.h"
20 // Default interval for delayed tasks.
21 const int kSimulationIntervalMs
= 750;
25 const char FakeBluetoothAdapterClient::kAdapterPath
[] =
27 const char FakeBluetoothAdapterClient::kAdapterName
[] =
29 const char FakeBluetoothAdapterClient::kAdapterAddress
[] =
32 const char FakeBluetoothAdapterClient::kSecondAdapterPath
[] =
34 const char FakeBluetoothAdapterClient::kSecondAdapterName
[] =
35 "Second Fake Adapter";
36 const char FakeBluetoothAdapterClient::kSecondAdapterAddress
[] =
39 FakeBluetoothAdapterClient::Properties::Properties(
40 const PropertyChangedCallback
& callback
)
41 : BluetoothAdapterClient::Properties(
43 bluetooth_adapter::kBluetoothAdapterInterface
,
47 FakeBluetoothAdapterClient::Properties::~Properties() {
50 void FakeBluetoothAdapterClient::Properties::Get(
51 dbus::PropertyBase
* property
,
52 dbus::PropertySet::GetCallback callback
) {
53 VLOG(1) << "Get " << property
->name();
57 void FakeBluetoothAdapterClient::Properties::GetAll() {
61 void FakeBluetoothAdapterClient::Properties::Set(
62 dbus::PropertyBase
*property
,
63 dbus::PropertySet::SetCallback callback
) {
64 VLOG(1) << "Set " << property
->name();
65 if (property
->name() == powered
.name() ||
66 property
->name() == alias
.name() ||
67 property
->name() == discoverable
.name() ||
68 property
->name() == discoverable_timeout
.name()) {
70 property
->ReplaceValueWithSetValue();
76 FakeBluetoothAdapterClient::FakeBluetoothAdapterClient()
78 second_visible_(false),
79 discovering_count_(0),
80 set_discovery_filter_should_fail_(false),
81 simulation_interval_ms_(kSimulationIntervalMs
) {
82 properties_
.reset(new Properties(base::Bind(
83 &FakeBluetoothAdapterClient::OnPropertyChanged
, base::Unretained(this))));
85 properties_
->address
.ReplaceValue(kAdapterAddress
);
86 properties_
->name
.ReplaceValue("Fake Adapter (Name)");
87 properties_
->alias
.ReplaceValue(kAdapterName
);
88 properties_
->pairable
.ReplaceValue(true);
90 second_properties_
.reset(new Properties(base::Bind(
91 &FakeBluetoothAdapterClient::OnPropertyChanged
, base::Unretained(this))));
93 second_properties_
->address
.ReplaceValue(kSecondAdapterAddress
);
94 second_properties_
->name
.ReplaceValue("Second Fake Adapter (Name)");
95 second_properties_
->alias
.ReplaceValue(kSecondAdapterName
);
96 second_properties_
->pairable
.ReplaceValue(true);
99 FakeBluetoothAdapterClient::~FakeBluetoothAdapterClient() {
102 void FakeBluetoothAdapterClient::Init(dbus::Bus
* bus
) {
105 void FakeBluetoothAdapterClient::AddObserver(Observer
* observer
) {
106 observers_
.AddObserver(observer
);
109 void FakeBluetoothAdapterClient::RemoveObserver(Observer
* observer
) {
110 observers_
.RemoveObserver(observer
);
113 std::vector
<dbus::ObjectPath
> FakeBluetoothAdapterClient::GetAdapters() {
114 std::vector
<dbus::ObjectPath
> object_paths
;
116 object_paths
.push_back(dbus::ObjectPath(kAdapterPath
));
118 object_paths
.push_back(dbus::ObjectPath(kSecondAdapterPath
));
122 FakeBluetoothAdapterClient::Properties
*
123 FakeBluetoothAdapterClient::GetProperties(const dbus::ObjectPath
& object_path
) {
124 if (object_path
== dbus::ObjectPath(kAdapterPath
))
125 return properties_
.get();
126 else if (object_path
== dbus::ObjectPath(kSecondAdapterPath
))
127 return second_properties_
.get();
132 void FakeBluetoothAdapterClient::StartDiscovery(
133 const dbus::ObjectPath
& object_path
,
134 const base::Closure
& callback
,
135 const ErrorCallback
& error_callback
) {
136 if (object_path
!= dbus::ObjectPath(kAdapterPath
)) {
137 PostDelayedTask(base::Bind(error_callback
, kNoResponseError
, ""));
141 ++discovering_count_
;
142 VLOG(1) << "StartDiscovery: " << object_path
.value() << ", "
143 << "count is now " << discovering_count_
;
144 PostDelayedTask(callback
);
146 if (discovering_count_
== 1) {
147 properties_
->discovering
.ReplaceValue(true);
149 FakeBluetoothDeviceClient
* device_client
=
150 static_cast<FakeBluetoothDeviceClient
*>(
151 DBusThreadManager::Get()->GetBluetoothDeviceClient());
152 device_client
->BeginDiscoverySimulation(dbus::ObjectPath(kAdapterPath
));
156 void FakeBluetoothAdapterClient::StopDiscovery(
157 const dbus::ObjectPath
& object_path
,
158 const base::Closure
& callback
,
159 const ErrorCallback
& error_callback
) {
160 if (object_path
!= dbus::ObjectPath(kAdapterPath
)) {
161 PostDelayedTask(base::Bind(error_callback
, kNoResponseError
, ""));
165 if (!discovering_count_
) {
166 LOG(WARNING
) << "StopDiscovery called when not discovering";
167 PostDelayedTask(base::Bind(error_callback
, kNoResponseError
, ""));
171 --discovering_count_
;
172 VLOG(1) << "StopDiscovery: " << object_path
.value() << ", "
173 << "count is now " << discovering_count_
;
174 PostDelayedTask(callback
);
176 if (discovering_count_
== 0) {
177 FakeBluetoothDeviceClient
* device_client
=
178 static_cast<FakeBluetoothDeviceClient
*>(
179 DBusThreadManager::Get()->GetBluetoothDeviceClient());
180 device_client
->EndDiscoverySimulation(dbus::ObjectPath(kAdapterPath
));
182 if (simulation_interval_ms_
> 100) {
183 device_client
->BeginIncomingPairingSimulation(
184 dbus::ObjectPath(kAdapterPath
));
187 discovery_filter_
.reset();
188 properties_
->discovering
.ReplaceValue(false);
192 void FakeBluetoothAdapterClient::RemoveDevice(
193 const dbus::ObjectPath
& object_path
,
194 const dbus::ObjectPath
& device_path
,
195 const base::Closure
& callback
,
196 const ErrorCallback
& error_callback
) {
197 if (object_path
!= dbus::ObjectPath(kAdapterPath
)) {
198 error_callback
.Run(kNoResponseError
, "");
202 VLOG(1) << "RemoveDevice: " << object_path
.value()
203 << " " << device_path
.value();
206 FakeBluetoothDeviceClient
* device_client
=
207 static_cast<FakeBluetoothDeviceClient
*>(
208 DBusThreadManager::Get()->GetBluetoothDeviceClient());
209 device_client
->RemoveDevice(dbus::ObjectPath(kAdapterPath
), device_path
);
212 void FakeBluetoothAdapterClient::MakeSetDiscoveryFilterFail() {
213 set_discovery_filter_should_fail_
= true;
216 void FakeBluetoothAdapterClient::SetDiscoveryFilter(
217 const dbus::ObjectPath
& object_path
,
218 const DiscoveryFilter
& discovery_filter
,
219 const base::Closure
& callback
,
220 const ErrorCallback
& error_callback
) {
221 if (object_path
!= dbus::ObjectPath(kAdapterPath
)) {
222 PostDelayedTask(base::Bind(error_callback
, kNoResponseError
, ""));
225 VLOG(1) << "SetDiscoveryFilter: " << object_path
.value();
227 if (set_discovery_filter_should_fail_
) {
228 PostDelayedTask(base::Bind(error_callback
, kNoResponseError
, ""));
229 set_discovery_filter_should_fail_
= false;
233 discovery_filter_
.reset(new DiscoveryFilter());
234 discovery_filter_
->CopyFrom(discovery_filter
);
235 PostDelayedTask(callback
);
238 void FakeBluetoothAdapterClient::SetSimulationIntervalMs(int interval_ms
) {
239 simulation_interval_ms_
= interval_ms
;
242 BluetoothAdapterClient::DiscoveryFilter
*
243 FakeBluetoothAdapterClient::GetDiscoveryFilter() {
244 return discovery_filter_
.get();
247 void FakeBluetoothAdapterClient::SetVisible(
249 if (visible
&& !visible_
) {
250 // Adapter becoming visible
253 FOR_EACH_OBSERVER(BluetoothAdapterClient::Observer
, observers_
,
254 AdapterAdded(dbus::ObjectPath(kAdapterPath
)));
256 } else if (visible_
&& !visible
) {
257 // Adapter becoming invisible
260 FOR_EACH_OBSERVER(BluetoothAdapterClient::Observer
, observers_
,
261 AdapterRemoved(dbus::ObjectPath(kAdapterPath
)));
265 void FakeBluetoothAdapterClient::SetSecondVisible(
267 if (visible
&& !second_visible_
) {
268 // Second adapter becoming visible
269 second_visible_
= visible
;
271 FOR_EACH_OBSERVER(BluetoothAdapterClient::Observer
, observers_
,
272 AdapterAdded(dbus::ObjectPath(kSecondAdapterPath
)));
274 } else if (second_visible_
&& !visible
) {
275 // Second adapter becoming invisible
276 second_visible_
= visible
;
278 FOR_EACH_OBSERVER(BluetoothAdapterClient::Observer
, observers_
,
279 AdapterRemoved(dbus::ObjectPath(kSecondAdapterPath
)));
283 void FakeBluetoothAdapterClient::OnPropertyChanged(
284 const std::string
& property_name
) {
285 if (property_name
== properties_
->powered
.name() &&
286 !properties_
->powered
.value()) {
287 VLOG(1) << "Adapter powered off";
289 if (discovering_count_
) {
290 discovering_count_
= 0;
291 properties_
->discovering
.ReplaceValue(false);
295 FOR_EACH_OBSERVER(BluetoothAdapterClient::Observer
, observers_
,
296 AdapterPropertyChanged(dbus::ObjectPath(kAdapterPath
),
300 void FakeBluetoothAdapterClient::PostDelayedTask(
301 const base::Closure
& callback
) {
302 base::ThreadTaskRunnerHandle::Get()->PostDelayedTask(
304 base::TimeDelta::FromMilliseconds(simulation_interval_ms_
));
307 } // namespace chromeos