// Copyright 2014 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/bluetooth_gatt_characteristic_client.h" #include "base/bind.h" #include "base/memory/weak_ptr.h" #include "base/observer_list.h" #include "dbus/bus.h" #include "dbus/object_manager.h" #include "third_party/cros_system_api/dbus/service_constants.h" namespace chromeos { namespace { // TODO(armansito): Move these to service_constants.h later. const char kNotifyingProperty[] = "Notifying"; const char kStartNotify[] = "StartNotify"; const char kStopNotify[] = "StopNotify"; } // namespace // static const char BluetoothGattCharacteristicClient::kNoResponseError[] = "org.chromium.Error.NoResponse"; // static const char BluetoothGattCharacteristicClient::kUnknownCharacteristicError[] = "org.chromium.Error.UnknownCharacteristic"; BluetoothGattCharacteristicClient::Properties::Properties( dbus::ObjectProxy* object_proxy, const std::string& interface_name, const PropertyChangedCallback& callback) : dbus::PropertySet(object_proxy, interface_name, callback) { RegisterProperty(bluetooth_gatt_characteristic::kUUIDProperty, &uuid); RegisterProperty(bluetooth_gatt_characteristic::kServiceProperty, &service); RegisterProperty(kNotifyingProperty, ¬ifying); RegisterProperty(bluetooth_gatt_characteristic::kFlagsProperty, &flags); } BluetoothGattCharacteristicClient::Properties::~Properties() { } // The BluetoothGattCharacteristicClient implementation used in production. class BluetoothGattCharacteristicClientImpl : public BluetoothGattCharacteristicClient, public dbus::ObjectManager::Interface { public: BluetoothGattCharacteristicClientImpl() : object_manager_(NULL), weak_ptr_factory_(this) { } virtual ~BluetoothGattCharacteristicClientImpl() { object_manager_->UnregisterInterface( bluetooth_gatt_characteristic::kBluetoothGattCharacteristicInterface); } // BluetoothGattCharacteristicClient override. virtual void AddObserver( BluetoothGattCharacteristicClient::Observer* observer) OVERRIDE { DCHECK(observer); observers_.AddObserver(observer); } // BluetoothGattCharacteristicClient override. virtual void RemoveObserver( BluetoothGattCharacteristicClient::Observer* observer) OVERRIDE { DCHECK(observer); observers_.RemoveObserver(observer); } // BluetoothGattCharacteristicClient override. virtual std::vector<dbus::ObjectPath> GetCharacteristics() OVERRIDE { DCHECK(object_manager_); return object_manager_->GetObjectsWithInterface( bluetooth_gatt_characteristic::kBluetoothGattCharacteristicInterface); } // BluetoothGattCharacteristicClient override. virtual Properties* GetProperties( const dbus::ObjectPath& object_path) OVERRIDE { DCHECK(object_manager_); return static_cast<Properties*>( object_manager_->GetProperties( object_path, bluetooth_gatt_characteristic:: kBluetoothGattCharacteristicInterface)); } // BluetoothGattCharacteristicClient override. virtual void ReadValue(const dbus::ObjectPath& object_path, const ValueCallback& callback, const ErrorCallback& error_callback) OVERRIDE { dbus::ObjectProxy* object_proxy = object_manager_->GetObjectProxy(object_path); if (!object_proxy) { error_callback.Run(kUnknownCharacteristicError, ""); return; } dbus::MethodCall method_call( bluetooth_gatt_characteristic::kBluetoothGattCharacteristicInterface, bluetooth_gatt_characteristic::kReadValue); object_proxy->CallMethodWithErrorCallback( &method_call, dbus::ObjectProxy::TIMEOUT_USE_DEFAULT, base::Bind(&BluetoothGattCharacteristicClientImpl::OnValueSuccess, weak_ptr_factory_.GetWeakPtr(), callback), base::Bind(&BluetoothGattCharacteristicClientImpl::OnError, weak_ptr_factory_.GetWeakPtr(), error_callback)); } // BluetoothGattCharacteristicClient override. virtual void WriteValue(const dbus::ObjectPath& object_path, const std::vector<uint8>& value, const base::Closure& callback, const ErrorCallback& error_callback) OVERRIDE { dbus::ObjectProxy* object_proxy = object_manager_->GetObjectProxy(object_path); if (!object_proxy) { error_callback.Run(kUnknownCharacteristicError, ""); return; } dbus::MethodCall method_call( bluetooth_gatt_characteristic::kBluetoothGattCharacteristicInterface, bluetooth_gatt_characteristic::kWriteValue); dbus::MessageWriter writer(&method_call); writer.AppendArrayOfBytes(value.data(), value.size()); object_proxy->CallMethodWithErrorCallback( &method_call, dbus::ObjectProxy::TIMEOUT_USE_DEFAULT, base::Bind(&BluetoothGattCharacteristicClientImpl::OnSuccess, weak_ptr_factory_.GetWeakPtr(), callback), base::Bind(&BluetoothGattCharacteristicClientImpl::OnError, weak_ptr_factory_.GetWeakPtr(), error_callback)); } // BluetoothGattCharacteristicClient override. virtual void StartNotify(const dbus::ObjectPath& object_path, const base::Closure& callback, const ErrorCallback& error_callback) OVERRIDE { dbus::ObjectProxy* object_proxy = object_manager_->GetObjectProxy(object_path); if (!object_proxy) { error_callback.Run(kUnknownCharacteristicError, ""); return; } dbus::MethodCall method_call( bluetooth_gatt_characteristic::kBluetoothGattCharacteristicInterface, kStartNotify); object_proxy->CallMethodWithErrorCallback( &method_call, dbus::ObjectProxy::TIMEOUT_USE_DEFAULT, base::Bind(&BluetoothGattCharacteristicClientImpl::OnSuccess, weak_ptr_factory_.GetWeakPtr(), callback), base::Bind(&BluetoothGattCharacteristicClientImpl::OnError, weak_ptr_factory_.GetWeakPtr(), error_callback)); } // BluetoothGattCharacteristicClient override. virtual void StopNotify(const dbus::ObjectPath& object_path, const base::Closure& callback, const ErrorCallback& error_callback) OVERRIDE { dbus::ObjectProxy* object_proxy = object_manager_->GetObjectProxy(object_path); if (!object_proxy) { error_callback.Run(kUnknownCharacteristicError, ""); return; } dbus::MethodCall method_call( bluetooth_gatt_characteristic::kBluetoothGattCharacteristicInterface, kStopNotify); object_proxy->CallMethodWithErrorCallback( &method_call, dbus::ObjectProxy::TIMEOUT_USE_DEFAULT, base::Bind(&BluetoothGattCharacteristicClientImpl::OnSuccess, weak_ptr_factory_.GetWeakPtr(), callback), base::Bind(&BluetoothGattCharacteristicClientImpl::OnError, weak_ptr_factory_.GetWeakPtr(), error_callback)); } // dbus::ObjectManager::Interface override. virtual dbus::PropertySet* CreateProperties( dbus::ObjectProxy *object_proxy, const dbus::ObjectPath& object_path, const std::string& interface_name) OVERRIDE { Properties* properties = new Properties( object_proxy, interface_name, base::Bind(&BluetoothGattCharacteristicClientImpl::OnPropertyChanged, weak_ptr_factory_.GetWeakPtr(), object_path)); return static_cast<dbus::PropertySet*>(properties); } // dbus::ObjectManager::Interface override. virtual void ObjectAdded(const dbus::ObjectPath& object_path, const std::string& interface_name) OVERRIDE { VLOG(2) << "Remote GATT characteristic added: " << object_path.value(); FOR_EACH_OBSERVER(BluetoothGattCharacteristicClient::Observer, observers_, GattCharacteristicAdded(object_path)); // Connect the "ValueUpdated" signal. dbus::ObjectProxy* object_proxy = object_manager_->GetObjectProxy(object_path); DCHECK(object_proxy); object_proxy->ConnectToSignal( bluetooth_gatt_characteristic::kBluetoothGattCharacteristicInterface, bluetooth_gatt_characteristic::kValueUpdatedSignal, base::Bind(&BluetoothGattCharacteristicClientImpl::ValueUpdatedReceived, weak_ptr_factory_.GetWeakPtr(), object_path), base::Bind( &BluetoothGattCharacteristicClientImpl::ValueUpdatedConnected, weak_ptr_factory_.GetWeakPtr())); } // dbus::ObjectManager::Interface override. virtual void ObjectRemoved(const dbus::ObjectPath& object_path, const std::string& interface_name) OVERRIDE { VLOG(2) << "Remote GATT characteristic removed: " << object_path.value(); FOR_EACH_OBSERVER(BluetoothGattCharacteristicClient::Observer, observers_, GattCharacteristicRemoved(object_path)); } protected: // chromeos::DBusClient override. virtual void Init(dbus::Bus* bus) OVERRIDE { object_manager_ = bus->GetObjectManager( bluetooth_object_manager::kBluetoothObjectManagerServiceName, dbus::ObjectPath( bluetooth_object_manager::kBluetoothObjectManagerServicePath)); object_manager_->RegisterInterface( bluetooth_gatt_characteristic::kBluetoothGattCharacteristicInterface, this); } private: // Called by dbus::PropertySet when a property value is changed, either by // result of a signal or response to a GetAll() or Get() call. Informs // observers. virtual void OnPropertyChanged(const dbus::ObjectPath& object_path, const std::string& property_name) { VLOG(2) << "Remote GATT characteristic property changed: " << object_path.value() << ": " << property_name; FOR_EACH_OBSERVER(BluetoothGattCharacteristicClient::Observer, observers_, GattCharacteristicPropertyChanged(object_path, property_name)); } // Called by dbus:: when a "ValueUpdated" signal is received. void ValueUpdatedReceived(const dbus::ObjectPath& object_path, dbus::Signal* signal) { DCHECK(signal); const uint8* bytes = NULL; size_t length = 0; dbus::MessageReader reader(signal); if (!reader.PopArrayOfBytes(&bytes, &length)) { LOG(WARNING) << "ValueUpdated signal has incorrect parameters: " << signal->ToString(); return; } std::vector<uint8> value; if (bytes) value.assign(bytes, bytes + length); FOR_EACH_OBSERVER(BluetoothGattCharacteristicClient::Observer, observers_, GattCharacteristicValueUpdated(object_path, value)); } // Called by dbus:: when the "ValueUpdated" signal is initially connected. void ValueUpdatedConnected(const std::string& interface_name, const std::string& signal_name, bool success) { LOG_IF(WARNING, !success) << "Failed to connect to the ValueUpdated signal"; } // Called when a response for successful method call is received. void OnSuccess(const base::Closure& callback, dbus::Response* response) { DCHECK(response); callback.Run(); } // Called when a characteristic value response for a successful method call // is received. void OnValueSuccess(const ValueCallback& callback, dbus::Response* response) { DCHECK(response); dbus::MessageReader reader(response); const uint8* bytes = NULL; size_t length = 0; if (!reader.PopArrayOfBytes(&bytes, &length)) VLOG(2) << "Error reading array of bytes in ValueCallback"; std::vector<uint8> value; if (bytes) value.assign(bytes, bytes + length); callback.Run(value); } // Called when a response for a failed method call is received. void OnError(const ErrorCallback& error_callback, dbus::ErrorResponse* response) { // Error response has optional error message argument. std::string error_name; std::string error_message; if (response) { dbus::MessageReader reader(response); error_name = response->GetErrorName(); reader.PopString(&error_message); } else { error_name = kNoResponseError; error_message = ""; } error_callback.Run(error_name, error_message); } dbus::ObjectManager* object_manager_; // List of observers interested in event notifications from us. ObserverList<BluetoothGattCharacteristicClient::Observer> observers_; // Weak pointer factory for generating 'this' pointers that might live longer // than we do. // Note: This should remain the last member so it'll be destroyed and // invalidate its weak pointers before any other members are destroyed. base::WeakPtrFactory<BluetoothGattCharacteristicClientImpl> weak_ptr_factory_; DISALLOW_COPY_AND_ASSIGN(BluetoothGattCharacteristicClientImpl); }; BluetoothGattCharacteristicClient::BluetoothGattCharacteristicClient() { } BluetoothGattCharacteristicClient::~BluetoothGattCharacteristicClient() { } // static BluetoothGattCharacteristicClient* BluetoothGattCharacteristicClient::Create() { return new BluetoothGattCharacteristicClientImpl(); } } // namespace chromeos