blob: 4f227fd375ea9dad4e4369790716e22beeab7ca2 [file] [log] [blame]
// 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,
&notifying);
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