| // Copyright 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/nfc_adapter_client.h" |
| |
| #include <map> |
| #include <utility> |
| |
| #include "base/bind.h" |
| #include "base/memory/scoped_ptr.h" |
| #include "base/observer_list.h" |
| #include "base/strings/stringprintf.h" |
| #include "chromeos/dbus/nfc_manager_client.h" |
| #include "dbus/bus.h" |
| #include "dbus/message.h" |
| #include "third_party/cros_system_api/dbus/service_constants.h" |
| |
| namespace chromeos { |
| |
| NfcAdapterClient::Properties::Properties( |
| dbus::ObjectProxy* object_proxy, |
| const PropertyChangedCallback& callback) |
| : NfcPropertySet(object_proxy, |
| nfc_adapter::kNfcAdapterInterface, |
| callback) { |
| RegisterProperty(nfc_adapter::kModeProperty, &mode); |
| RegisterProperty(nfc_adapter::kPoweredProperty, &powered); |
| RegisterProperty(nfc_adapter::kPollingProperty, &polling); |
| RegisterProperty(nfc_adapter::kProtocolsProperty, &protocols); |
| RegisterProperty(nfc_adapter::kTagsProperty, &tags); |
| RegisterProperty(nfc_adapter::kDevicesProperty, &devices); |
| } |
| |
| NfcAdapterClient::Properties::~Properties() { |
| } |
| |
| // The NfcAdapterClient implementation used in production. |
| class NfcAdapterClientImpl |
| : public NfcAdapterClient, |
| public NfcManagerClient::Observer, |
| public nfc_client_helpers::DBusObjectMap::Delegate { |
| public: |
| explicit NfcAdapterClientImpl(NfcManagerClient* manager_client) |
| : bus_(NULL), |
| manager_client_(manager_client), |
| weak_ptr_factory_(this) { |
| DCHECK(manager_client); |
| } |
| |
| virtual ~NfcAdapterClientImpl() { |
| manager_client_->RemoveObserver(this); |
| } |
| |
| // NfcAdapterClient override. |
| virtual void AddObserver(NfcAdapterClient::Observer* observer) OVERRIDE { |
| DCHECK(observer); |
| observers_.AddObserver(observer); |
| } |
| |
| // NfcAdapterClient override. |
| virtual void RemoveObserver(NfcAdapterClient::Observer* observer) OVERRIDE { |
| DCHECK(observer); |
| observers_.RemoveObserver(observer); |
| } |
| |
| // NfcAdapterClient override. |
| virtual std::vector<dbus::ObjectPath> GetAdapters() OVERRIDE { |
| return object_map_->GetObjectPaths(); |
| } |
| |
| // NfcAdapterClient override. |
| virtual Properties* GetProperties(const dbus::ObjectPath& object_path) |
| OVERRIDE { |
| return static_cast<Properties*>( |
| object_map_->GetObjectProperties(object_path)); |
| } |
| |
| // NfcAdapterClient override. |
| virtual void StartPollLoop( |
| const dbus::ObjectPath& object_path, |
| const std::string& mode, |
| const base::Closure& callback, |
| const nfc_client_helpers::ErrorCallback& error_callback) OVERRIDE { |
| dbus::ObjectProxy* object_proxy = object_map_->GetObjectProxy(object_path); |
| if (!object_proxy) { |
| std::string error_message = |
| base::StringPrintf("NFC adapter with object path \"%s\" does not " |
| "exist.", object_path.value().c_str()); |
| LOG(ERROR) << error_message; |
| error_callback.Run(nfc_client_helpers::kUnknownObjectError, |
| error_message); |
| return; |
| } |
| dbus::MethodCall method_call(nfc_adapter::kNfcAdapterInterface, |
| nfc_adapter::kStartPollLoop); |
| dbus::MessageWriter writer(&method_call); |
| writer.AppendString(mode); |
| object_proxy->CallMethodWithErrorCallback( |
| &method_call, |
| dbus::ObjectProxy::TIMEOUT_USE_DEFAULT, |
| base::Bind(&nfc_client_helpers::OnSuccess, callback), |
| base::Bind(&nfc_client_helpers::OnError, error_callback)); |
| } |
| |
| // NfcAdapterClient override. |
| virtual void StopPollLoop( |
| const dbus::ObjectPath& object_path, |
| const base::Closure& callback, |
| const nfc_client_helpers::ErrorCallback& error_callback) OVERRIDE { |
| dbus::ObjectProxy* object_proxy = object_map_->GetObjectProxy(object_path); |
| if (!object_proxy) { |
| std::string error_message = |
| base::StringPrintf("NFC adapter with object path \"%s\" does not " |
| "exist.", object_path.value().c_str()); |
| LOG(ERROR) << error_message; |
| error_callback.Run(nfc_client_helpers::kUnknownObjectError, |
| error_message); |
| return; |
| } |
| dbus::MethodCall method_call(nfc_adapter::kNfcAdapterInterface, |
| nfc_adapter::kStopPollLoop); |
| object_proxy->CallMethodWithErrorCallback( |
| &method_call, |
| dbus::ObjectProxy::TIMEOUT_USE_DEFAULT, |
| base::Bind(&nfc_client_helpers::OnSuccess, callback), |
| base::Bind(&nfc_client_helpers::OnError, error_callback)); |
| } |
| |
| protected: |
| // DBusClient override. |
| virtual void Init(dbus::Bus* bus) OVERRIDE { |
| VLOG(1) << "Creating NfcAdapterClientImpl"; |
| DCHECK(bus); |
| bus_ = bus; |
| object_map_.reset(new nfc_client_helpers::DBusObjectMap( |
| nfc_adapter::kNfcAdapterServiceName, this, bus)); |
| DCHECK(manager_client_); |
| manager_client_->AddObserver(this); |
| } |
| |
| private: |
| // NfcManagerClient::Observer override. |
| virtual void ManagerPropertyChanged( |
| const std::string& property_name) OVERRIDE { |
| // Update the adapter proxies. |
| DCHECK(manager_client_); |
| NfcManagerClient::Properties* manager_properties = |
| manager_client_->GetProperties(); |
| |
| // Ignore changes to properties other than "Adapters". |
| if (property_name != manager_properties->adapters.name()) |
| return; |
| |
| // Update the known adapters. |
| VLOG(1) << "NFC adapters changed."; |
| const std::vector<dbus::ObjectPath>& received_adapters = |
| manager_properties->adapters.value(); |
| object_map_->UpdateObjects(received_adapters); |
| } |
| |
| // nfc_client_helpers::DBusObjectMap::Delegate override. |
| virtual NfcPropertySet* CreateProperties( |
| dbus::ObjectProxy* object_proxy) OVERRIDE { |
| return new Properties( |
| object_proxy, |
| base::Bind(&NfcAdapterClientImpl::OnPropertyChanged, |
| weak_ptr_factory_.GetWeakPtr(), |
| object_proxy->object_path())); |
| } |
| |
| // nfc_client_helpers::DBusObjectMap::Delegate override. |
| virtual void ObjectAdded(const dbus::ObjectPath& object_path) OVERRIDE { |
| FOR_EACH_OBSERVER(NfcAdapterClient::Observer, observers_, |
| AdapterAdded(object_path)); |
| } |
| |
| // nfc_client_helpers::DBusObjectMap::Delegate override. |
| virtual void ObjectRemoved(const dbus::ObjectPath& object_path) OVERRIDE { |
| FOR_EACH_OBSERVER(NfcAdapterClient::Observer, observers_, |
| AdapterRemoved(object_path)); |
| } |
| |
| // Called by NfcPropertySet when a property value is changed, either by |
| // result of a signal or response to a GetAll() or Get() call. |
| void OnPropertyChanged(const dbus::ObjectPath& object_path, |
| const std::string& property_name) { |
| VLOG(1) << "Adapter property changed; Path: " << object_path.value() |
| << " Property: " << property_name; |
| FOR_EACH_OBSERVER(NfcAdapterClient::Observer, observers_, |
| AdapterPropertyChanged(object_path, property_name)); |
| } |
| |
| // We maintain a pointer to the bus to be able to request proxies for |
| // new NFC adapters that appear. |
| dbus::Bus* bus_; |
| |
| // List of observers interested in event notifications. |
| ObserverList<NfcAdapterClient::Observer> observers_; |
| |
| // Mapping from object paths to object proxies and properties structures that |
| // were already created by us. |
| scoped_ptr<nfc_client_helpers::DBusObjectMap> object_map_; |
| |
| // The manager client that we listen to events notifications from. |
| NfcManagerClient* manager_client_; |
| |
| // 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<NfcAdapterClientImpl> weak_ptr_factory_; |
| |
| DISALLOW_COPY_AND_ASSIGN(NfcAdapterClientImpl); |
| }; |
| |
| NfcAdapterClient::NfcAdapterClient() { |
| } |
| |
| NfcAdapterClient::~NfcAdapterClient() { |
| } |
| |
| NfcAdapterClient* NfcAdapterClient::Create(NfcManagerClient* manager_client) { |
| return new NfcAdapterClientImpl(manager_client); |
| } |
| |
| } // namespace chromeos |