|  | // Copyright 2014 The Chromium Authors | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #include "device/bluetooth/dbus/bluetooth_gatt_characteristic_client.h" | 
|  |  | 
|  | #include <stddef.h> | 
|  |  | 
|  | #include "base/containers/to_vector.h" | 
|  | #include "base/functional/bind.h" | 
|  | #include "base/logging.h" | 
|  | #include "base/memory/raw_ptr.h" | 
|  | #include "base/memory/weak_ptr.h" | 
|  | #include "base/observer_list.h" | 
|  | #include "base/values.h" | 
|  | #include "dbus/bus.h" | 
|  | #include "dbus/object_manager.h" | 
|  | #include "dbus/values_util.h" | 
|  | #include "third_party/cros_system_api/dbus/bluetooth/dbus-constants.h" | 
|  | #include "third_party/cros_system_api/dbus/service_constants.h" | 
|  |  | 
|  | namespace bluez { | 
|  |  | 
|  | // 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(bluetooth_gatt_characteristic::kValueProperty, &value); | 
|  | RegisterProperty(bluetooth_gatt_characteristic::kNotifyingProperty, | 
|  | ¬ifying); | 
|  | RegisterProperty(bluetooth_gatt_characteristic::kFlagsProperty, &flags); | 
|  | } | 
|  |  | 
|  | BluetoothGattCharacteristicClient::Properties::~Properties() = default; | 
|  |  | 
|  | // The BluetoothGattCharacteristicClient implementation used in production. | 
|  | class BluetoothGattCharacteristicClientImpl | 
|  | : public BluetoothGattCharacteristicClient, | 
|  | public dbus::ObjectManager::Interface { | 
|  | public: | 
|  | BluetoothGattCharacteristicClientImpl() : object_manager_(nullptr) {} | 
|  |  | 
|  | BluetoothGattCharacteristicClientImpl( | 
|  | const BluetoothGattCharacteristicClientImpl&) = delete; | 
|  | BluetoothGattCharacteristicClientImpl& operator=( | 
|  | const BluetoothGattCharacteristicClientImpl&) = delete; | 
|  |  | 
|  | ~BluetoothGattCharacteristicClientImpl() override { | 
|  | object_manager_->UnregisterInterface( | 
|  | bluetooth_gatt_characteristic::kBluetoothGattCharacteristicInterface); | 
|  | } | 
|  |  | 
|  | // BluetoothGattCharacteristicClient override. | 
|  | void AddObserver( | 
|  | BluetoothGattCharacteristicClient::Observer* observer) override { | 
|  | DCHECK(observer); | 
|  | observers_.AddObserver(observer); | 
|  | } | 
|  |  | 
|  | // BluetoothGattCharacteristicClient override. | 
|  | void RemoveObserver( | 
|  | BluetoothGattCharacteristicClient::Observer* observer) override { | 
|  | DCHECK(observer); | 
|  | observers_.RemoveObserver(observer); | 
|  | } | 
|  |  | 
|  | // BluetoothGattCharacteristicClient override. | 
|  | std::vector<dbus::ObjectPath> GetCharacteristics() override { | 
|  | DCHECK(object_manager_); | 
|  | return object_manager_->GetObjectsWithInterface( | 
|  | bluetooth_gatt_characteristic::kBluetoothGattCharacteristicInterface); | 
|  | } | 
|  |  | 
|  | // BluetoothGattCharacteristicClient override. | 
|  | 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. | 
|  | void ReadValue(const dbus::ObjectPath& object_path, | 
|  | ValueCallback callback, | 
|  | ErrorCallback error_callback) override { | 
|  | dbus::ObjectProxy* object_proxy = | 
|  | object_manager_->GetObjectProxy(object_path); | 
|  | if (!object_proxy) { | 
|  | std::move(error_callback).Run(kUnknownCharacteristicError, ""); | 
|  | return; | 
|  | } | 
|  |  | 
|  | dbus::MethodCall method_call( | 
|  | bluetooth_gatt_characteristic::kBluetoothGattCharacteristicInterface, | 
|  | bluetooth_gatt_characteristic::kReadValue); | 
|  |  | 
|  | // Append empty option dict | 
|  | dbus::MessageWriter writer(&method_call); | 
|  | dbus::AppendValueData(&writer, base::Value::Dict()); | 
|  |  | 
|  | object_proxy->CallMethodWithErrorResponse( | 
|  | &method_call, dbus::ObjectProxy::TIMEOUT_USE_DEFAULT, | 
|  | base::BindOnce(&BluetoothGattCharacteristicClientImpl::OnValueResponse, | 
|  | weak_ptr_factory_.GetWeakPtr(), std::move(callback), | 
|  | std::move(error_callback))); | 
|  | } | 
|  |  | 
|  | // BluetoothGattCharacteristicClient override. | 
|  | void WriteValue(const dbus::ObjectPath& object_path, | 
|  | base::span<const uint8_t> value, | 
|  | std::string_view type_option, | 
|  | base::OnceClosure callback, | 
|  | ErrorCallback error_callback) override { | 
|  | dbus::ObjectProxy* object_proxy = | 
|  | object_manager_->GetObjectProxy(object_path); | 
|  | if (!object_proxy) { | 
|  | std::move(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); | 
|  |  | 
|  | // Append option dict | 
|  | base::Value::Dict dict; | 
|  | if (!type_option.empty()) { | 
|  | // NB: the "type" option was added in BlueZ 5.51. Older versions of BlueZ | 
|  | // will ignore this option. | 
|  | dict.Set(bluetooth_gatt_characteristic::kOptionType, type_option); | 
|  | } | 
|  | dbus::AppendValueData(&writer, dict); | 
|  |  | 
|  | object_proxy->CallMethodWithErrorResponse( | 
|  | &method_call, dbus::ObjectProxy::TIMEOUT_USE_DEFAULT, | 
|  | base::BindOnce(&BluetoothGattCharacteristicClientImpl::OnMethodResponse, | 
|  | weak_ptr_factory_.GetWeakPtr(), std::move(callback), | 
|  | std::move(error_callback))); | 
|  | } | 
|  |  | 
|  | void PrepareWriteValue(const dbus::ObjectPath& object_path, | 
|  | base::span<const uint8_t> value, | 
|  | base::OnceClosure callback, | 
|  | ErrorCallback error_callback) override { | 
|  | dbus::ObjectProxy* object_proxy = | 
|  | object_manager_->GetObjectProxy(object_path); | 
|  | if (!object_proxy) { | 
|  | std::move(error_callback).Run(kUnknownCharacteristicError, ""); | 
|  | return; | 
|  | } | 
|  |  | 
|  | dbus::MethodCall method_call( | 
|  | bluetooth_gatt_characteristic::kBluetoothGattCharacteristicInterface, | 
|  | bluetooth_gatt_characteristic::kPrepareWriteValue); | 
|  | dbus::MessageWriter writer(&method_call); | 
|  | writer.AppendArrayOfBytes(value); | 
|  |  | 
|  | dbus::AppendValueData(&writer, base::Value::Dict()); | 
|  |  | 
|  | object_proxy->CallMethodWithErrorResponse( | 
|  | &method_call, dbus::ObjectProxy::TIMEOUT_USE_DEFAULT, | 
|  | base::BindOnce(&BluetoothGattCharacteristicClientImpl::OnMethodResponse, | 
|  | weak_ptr_factory_.GetWeakPtr(), std::move(callback), | 
|  | std::move(error_callback))); | 
|  | } | 
|  |  | 
|  | // BluetoothGattCharacteristicClient override. | 
|  | void StartNotify( | 
|  | const dbus::ObjectPath& object_path, | 
|  | #if BUILDFLAG(IS_CHROMEOS) | 
|  | device::BluetoothGattCharacteristic::NotificationType notification_type, | 
|  | #endif  // BUILDFLAG(IS_CHROMEOS) | 
|  | base::OnceClosure callback, | 
|  | ErrorCallback error_callback) override { | 
|  | dbus::ObjectProxy* object_proxy = | 
|  | object_manager_->GetObjectProxy(object_path); | 
|  | if (!object_proxy) { | 
|  | std::move(error_callback).Run(kUnknownCharacteristicError, ""); | 
|  | return; | 
|  | } | 
|  |  | 
|  | dbus::MethodCall method_call( | 
|  | bluetooth_gatt_characteristic::kBluetoothGattCharacteristicInterface, | 
|  | bluetooth_gatt_characteristic::kStartNotify); | 
|  | #if BUILDFLAG(IS_CHROMEOS) | 
|  | dbus::MessageWriter writer(&method_call); | 
|  | writer.AppendByte(static_cast<uint8_t>(notification_type)); | 
|  | #endif  // BUILDFLAG(IS_CHROMEOS) | 
|  |  | 
|  | object_proxy->CallMethodWithErrorResponse( | 
|  | &method_call, dbus::ObjectProxy::TIMEOUT_USE_DEFAULT, | 
|  | base::BindOnce(&BluetoothGattCharacteristicClientImpl::OnMethodResponse, | 
|  | weak_ptr_factory_.GetWeakPtr(), std::move(callback), | 
|  | std::move(error_callback))); | 
|  | } | 
|  |  | 
|  | // BluetoothGattCharacteristicClient override. | 
|  | void StopNotify(const dbus::ObjectPath& object_path, | 
|  | base::OnceClosure callback, | 
|  | ErrorCallback error_callback) override { | 
|  | dbus::ObjectProxy* object_proxy = | 
|  | object_manager_->GetObjectProxy(object_path); | 
|  | if (!object_proxy) { | 
|  | std::move(error_callback).Run(kUnknownCharacteristicError, ""); | 
|  | return; | 
|  | } | 
|  |  | 
|  | dbus::MethodCall method_call( | 
|  | bluetooth_gatt_characteristic::kBluetoothGattCharacteristicInterface, | 
|  | bluetooth_gatt_characteristic::kStopNotify); | 
|  |  | 
|  | object_proxy->CallMethodWithErrorResponse( | 
|  | &method_call, dbus::ObjectProxy::TIMEOUT_USE_DEFAULT, | 
|  | base::BindOnce(&BluetoothGattCharacteristicClientImpl::OnMethodResponse, | 
|  | weak_ptr_factory_.GetWeakPtr(), std::move(callback), | 
|  | std::move(error_callback))); | 
|  | } | 
|  |  | 
|  | // dbus::ObjectManager::Interface override. | 
|  | dbus::PropertySet* CreateProperties( | 
|  | dbus::ObjectProxy* object_proxy, | 
|  | const dbus::ObjectPath& object_path, | 
|  | const std::string& interface_name) override { | 
|  | return new Properties( | 
|  | object_proxy, interface_name, | 
|  | base::BindRepeating( | 
|  | &BluetoothGattCharacteristicClientImpl::OnPropertyChanged, | 
|  | weak_ptr_factory_.GetWeakPtr(), object_path)); | 
|  | } | 
|  |  | 
|  | // dbus::ObjectManager::Interface override. | 
|  | void ObjectAdded(const dbus::ObjectPath& object_path, | 
|  | const std::string& interface_name) override { | 
|  | DVLOG(2) << "Remote GATT characteristic added: " << object_path.value(); | 
|  | for (auto& observer : observers_) | 
|  | observer.GattCharacteristicAdded(object_path); | 
|  | } | 
|  |  | 
|  | // dbus::ObjectManager::Interface override. | 
|  | void ObjectRemoved(const dbus::ObjectPath& object_path, | 
|  | const std::string& interface_name) override { | 
|  | DVLOG(2) << "Remote GATT characteristic removed: " << object_path.value(); | 
|  | for (auto& observer : observers_) | 
|  | observer.GattCharacteristicRemoved(object_path); | 
|  | } | 
|  |  | 
|  | protected: | 
|  | // bluez::DBusClient override. | 
|  | void Init(dbus::Bus* bus, | 
|  | const std::string& bluetooth_service_name) override { | 
|  | object_manager_ = bus->GetObjectManager( | 
|  | bluetooth_service_name, | 
|  | 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) { | 
|  | DVLOG(2) << "Remote GATT characteristic property changed: " | 
|  | << object_path.value() << ": " << property_name; | 
|  | for (auto& observer : observers_) | 
|  | observer.GattCharacteristicPropertyChanged(object_path, property_name); | 
|  | } | 
|  |  | 
|  | void OnMethodResponse(base::OnceClosure callback, | 
|  | ErrorCallback error_callback, | 
|  | dbus::Response* response, | 
|  | dbus::ErrorResponse* error_response) { | 
|  | if (response) { | 
|  | std::move(callback).Run(); | 
|  | } else { | 
|  | OnError(std::move(error_callback), error_response); | 
|  | } | 
|  | } | 
|  |  | 
|  | void OnValueResponse(ValueCallback callback, | 
|  | ErrorCallback error_callback, | 
|  | dbus::Response* response, | 
|  | dbus::ErrorResponse* error_response) { | 
|  | if (!response) { | 
|  | OnError(std::move(error_callback), error_response); | 
|  | return; | 
|  | } | 
|  |  | 
|  | dbus::MessageReader reader(response); | 
|  | base::span<const uint8_t> bytes; | 
|  | if (!reader.PopArrayOfBytes(&bytes)) { | 
|  | DVLOG(2) << "Error reading array of bytes in ValueCallback"; | 
|  | } | 
|  |  | 
|  | std::move(callback).Run(/*error_code=*/std::nullopt, base::ToVector(bytes)); | 
|  | } | 
|  |  | 
|  | // Called when a response for a failed method call is received. | 
|  | void OnError(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 = ""; | 
|  | } | 
|  | std::move(error_callback).Run(error_name, error_message); | 
|  | } | 
|  |  | 
|  | raw_ptr<dbus::ObjectManager> object_manager_; | 
|  |  | 
|  | // List of observers interested in event notifications from us. | 
|  | base::ObserverList<BluetoothGattCharacteristicClient::Observer>::Unchecked | 
|  | 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_{ | 
|  | this}; | 
|  | }; | 
|  |  | 
|  | BluetoothGattCharacteristicClient::BluetoothGattCharacteristicClient() = | 
|  | default; | 
|  |  | 
|  | BluetoothGattCharacteristicClient::~BluetoothGattCharacteristicClient() = | 
|  | default; | 
|  |  | 
|  | // static | 
|  | BluetoothGattCharacteristicClient* BluetoothGattCharacteristicClient::Create() { | 
|  | return new BluetoothGattCharacteristicClientImpl(); | 
|  | } | 
|  |  | 
|  | }  // namespace bluez |