// Copyright (c) 2013 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "chromeos/dbus/fake_bluetooth_adapter_client.h" #include "base/bind.h" #include "base/logging.h" #include "base/message_loop/message_loop.h" #include "base/time/time.h" #include "chromeos/dbus/dbus_thread_manager.h" #include "chromeos/dbus/fake_bluetooth_device_client.h" #include "dbus/object_path.h" #include "third_party/cros_system_api/dbus/service_constants.h" namespace chromeos { namespace { // Default interval for delayed tasks. const int kSimulationIntervalMs = 750; } // namespace const char FakeBluetoothAdapterClient::kAdapterPath[] = "/fake/hci0"; const char FakeBluetoothAdapterClient::kAdapterName[] = "Fake Adapter"; const char FakeBluetoothAdapterClient::kAdapterAddress[] = "01:1A:2B:1A:2B:03"; const char FakeBluetoothAdapterClient::kSecondAdapterPath[] = "/fake/hci1"; const char FakeBluetoothAdapterClient::kSecondAdapterName[] = "Second Fake Adapter"; const char FakeBluetoothAdapterClient::kSecondAdapterAddress[] = "00:DE:51:10:01:00"; FakeBluetoothAdapterClient::Properties::Properties( const PropertyChangedCallback& callback) : BluetoothAdapterClient::Properties( NULL, bluetooth_adapter::kBluetoothAdapterInterface, callback) { } FakeBluetoothAdapterClient::Properties::~Properties() { } void FakeBluetoothAdapterClient::Properties::Get( dbus::PropertyBase* property, dbus::PropertySet::GetCallback callback) { VLOG(1) << "Get " << property->name(); callback.Run(false); } void FakeBluetoothAdapterClient::Properties::GetAll() { VLOG(1) << "GetAll"; } void FakeBluetoothAdapterClient::Properties::Set( dbus::PropertyBase *property, dbus::PropertySet::SetCallback callback) { VLOG(1) << "Set " << property->name(); if (property->name() == powered.name() || property->name() == alias.name() || property->name() == discoverable.name() || property->name() == discoverable_timeout.name()) { callback.Run(true); property->ReplaceValueWithSetValue(); } else { callback.Run(false); } } FakeBluetoothAdapterClient::FakeBluetoothAdapterClient() : visible_(true), second_visible_(false), discovering_count_(0), simulation_interval_ms_(kSimulationIntervalMs) { properties_.reset(new Properties(base::Bind( &FakeBluetoothAdapterClient::OnPropertyChanged, base::Unretained(this)))); properties_->address.ReplaceValue(kAdapterAddress); properties_->name.ReplaceValue("Fake Adapter (Name)"); properties_->alias.ReplaceValue(kAdapterName); properties_->pairable.ReplaceValue(true); second_properties_.reset(new Properties(base::Bind( &FakeBluetoothAdapterClient::OnPropertyChanged, base::Unretained(this)))); second_properties_->address.ReplaceValue(kSecondAdapterAddress); second_properties_->name.ReplaceValue("Second Fake Adapter (Name)"); second_properties_->alias.ReplaceValue(kSecondAdapterName); second_properties_->pairable.ReplaceValue(true); } FakeBluetoothAdapterClient::~FakeBluetoothAdapterClient() { } void FakeBluetoothAdapterClient::Init(dbus::Bus* bus) { } void FakeBluetoothAdapterClient::AddObserver(Observer* observer) { observers_.AddObserver(observer); } void FakeBluetoothAdapterClient::RemoveObserver(Observer* observer) { observers_.RemoveObserver(observer); } std::vector<dbus::ObjectPath> FakeBluetoothAdapterClient::GetAdapters() { std::vector<dbus::ObjectPath> object_paths; if (visible_) object_paths.push_back(dbus::ObjectPath(kAdapterPath)); if (second_visible_) object_paths.push_back(dbus::ObjectPath(kSecondAdapterPath)); return object_paths; } FakeBluetoothAdapterClient::Properties* FakeBluetoothAdapterClient::GetProperties(const dbus::ObjectPath& object_path) { if (object_path == dbus::ObjectPath(kAdapterPath)) return properties_.get(); else if (object_path == dbus::ObjectPath(kSecondAdapterPath)) return second_properties_.get(); else return NULL; } void FakeBluetoothAdapterClient::StartDiscovery( const dbus::ObjectPath& object_path, const base::Closure& callback, const ErrorCallback& error_callback) { if (object_path != dbus::ObjectPath(kAdapterPath)) { PostDelayedTask(base::Bind(error_callback, kNoResponseError, "")); return; } ++discovering_count_; VLOG(1) << "StartDiscovery: " << object_path.value() << ", " << "count is now " << discovering_count_; PostDelayedTask(callback); if (discovering_count_ == 1) { properties_->discovering.ReplaceValue(true); FakeBluetoothDeviceClient* device_client = static_cast<FakeBluetoothDeviceClient*>( DBusThreadManager::Get()->GetBluetoothDeviceClient()); device_client->BeginDiscoverySimulation(dbus::ObjectPath(kAdapterPath)); } } void FakeBluetoothAdapterClient::StopDiscovery( const dbus::ObjectPath& object_path, const base::Closure& callback, const ErrorCallback& error_callback) { if (object_path != dbus::ObjectPath(kAdapterPath)) { PostDelayedTask(base::Bind(error_callback, kNoResponseError, "")); return; } if (!discovering_count_) { LOG(WARNING) << "StopDiscovery called when not discovering"; PostDelayedTask(base::Bind(error_callback, kNoResponseError, "")); return; } --discovering_count_; VLOG(1) << "StopDiscovery: " << object_path.value() << ", " << "count is now " << discovering_count_; PostDelayedTask(callback); if (discovering_count_ == 0) { FakeBluetoothDeviceClient* device_client = static_cast<FakeBluetoothDeviceClient*>( DBusThreadManager::Get()->GetBluetoothDeviceClient()); device_client->EndDiscoverySimulation(dbus::ObjectPath(kAdapterPath)); if (simulation_interval_ms_ > 100) { device_client->BeginIncomingPairingSimulation( dbus::ObjectPath(kAdapterPath)); } properties_->discovering.ReplaceValue(false); } } void FakeBluetoothAdapterClient::RemoveDevice( const dbus::ObjectPath& object_path, const dbus::ObjectPath& device_path, const base::Closure& callback, const ErrorCallback& error_callback) { if (object_path != dbus::ObjectPath(kAdapterPath)) { error_callback.Run(kNoResponseError, ""); return; } VLOG(1) << "RemoveDevice: " << object_path.value() << " " << device_path.value(); callback.Run(); FakeBluetoothDeviceClient* device_client = static_cast<FakeBluetoothDeviceClient*>( DBusThreadManager::Get()->GetBluetoothDeviceClient()); device_client->RemoveDevice(dbus::ObjectPath(kAdapterPath), device_path); } void FakeBluetoothAdapterClient::SetSimulationIntervalMs(int interval_ms) { simulation_interval_ms_ = interval_ms; } void FakeBluetoothAdapterClient::SetVisible( bool visible) { if (visible && !visible_) { // Adapter becoming visible visible_ = visible; FOR_EACH_OBSERVER(BluetoothAdapterClient::Observer, observers_, AdapterAdded(dbus::ObjectPath(kAdapterPath))); } else if (visible_ && !visible) { // Adapter becoming invisible visible_ = visible; FOR_EACH_OBSERVER(BluetoothAdapterClient::Observer, observers_, AdapterRemoved(dbus::ObjectPath(kAdapterPath))); } } void FakeBluetoothAdapterClient::SetSecondVisible( bool visible) { if (visible && !second_visible_) { // Second adapter becoming visible second_visible_ = visible; FOR_EACH_OBSERVER(BluetoothAdapterClient::Observer, observers_, AdapterAdded(dbus::ObjectPath(kSecondAdapterPath))); } else if (second_visible_ && !visible) { // Second adapter becoming invisible second_visible_ = visible; FOR_EACH_OBSERVER(BluetoothAdapterClient::Observer, observers_, AdapterRemoved(dbus::ObjectPath(kSecondAdapterPath))); } } void FakeBluetoothAdapterClient::OnPropertyChanged( const std::string& property_name) { if (property_name == properties_->powered.name() && !properties_->powered.value()) { VLOG(1) << "Adapter powered off"; if (discovering_count_) { discovering_count_ = 0; properties_->discovering.ReplaceValue(false); } } FOR_EACH_OBSERVER(BluetoothAdapterClient::Observer, observers_, AdapterPropertyChanged(dbus::ObjectPath(kAdapterPath), property_name)); } void FakeBluetoothAdapterClient::PostDelayedTask( const base::Closure& callback) { base::MessageLoop::current()->PostDelayedTask( FROM_HERE, callback, base::TimeDelta::FromMilliseconds(simulation_interval_ms_)); } } // namespace chromeos