// 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