1 // Copyright 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "chromeos/dbus/nfc_client_helpers.h"
7 #include "base/stl_util.h"
8 #include "dbus/values_util.h"
11 namespace nfc_client_helpers
{
13 const char kNoResponseError
[] = "org.chromium.Error.NoResponse";
14 const char kUnknownObjectError
[] = "org.chromium.Error.UnknownObject";
16 void OnSuccess(const base::Closure
& callback
, dbus::Response
* response
) {
21 void OnError(const ErrorCallback
& error_callback
,
22 dbus::ErrorResponse
* response
) {
23 // Error response has optional error message argument.
24 std::string error_name
;
25 std::string error_message
;
27 dbus::MessageReader
reader(response
);
28 error_name
= response
->GetErrorName();
29 reader
.PopString(&error_message
);
31 error_name
= kNoResponseError
;
34 error_callback
.Run(error_name
, error_message
);
37 DBusObjectMap::DBusObjectMap(const std::string
& service_name
,
41 service_name_(service_name
),
47 DBusObjectMap::~DBusObjectMap() {
48 // Clean up the Properties structures. We don't explicitly delete the object
49 // proxies, as they are owned by dbus::Bus.
50 for (ObjectMap::iterator iter
= object_map_
.begin();
51 iter
!= object_map_
.end(); ++iter
) {
52 const dbus::ObjectPath
& object_path
= iter
->first
;
53 ObjectPropertyPair pair
= iter
->second
;
54 delegate_
->ObjectRemoved(object_path
);
55 CleanUpObjectPropertyPair(pair
);
59 dbus::ObjectProxy
* DBusObjectMap::GetObjectProxy(
60 const dbus::ObjectPath
& object_path
) {
61 return GetObjectPropertyPair(object_path
).first
;
64 NfcPropertySet
* DBusObjectMap::GetObjectProperties(
65 const dbus::ObjectPath
& object_path
) {
66 return GetObjectPropertyPair(object_path
).second
;
69 void DBusObjectMap::UpdateObjects(const ObjectPathVector
& object_paths
) {
70 // This set is used to query if an object path was removed, while updating
71 // the removed paths below. The iterator is used as a hint to accelerate
73 std::set
<dbus::ObjectPath
> object_path_set
;
74 std::set
<dbus::ObjectPath
>::iterator object_path_set_iter
=
75 object_path_set
.begin();
78 for (ObjectPathVector::const_iterator iter
= object_paths
.begin();
79 iter
!= object_paths
.end(); ++iter
) {
80 const dbus::ObjectPath
&object_path
= *iter
;
81 AddObject(object_path
);
82 // Neard usually returns object paths in ascending order. Give a hint here
83 // to make insertion amortized constant.
84 object_path_set_iter
=
85 object_path_set
.insert(object_path_set_iter
, object_path
);
88 // Remove all objects that are not in |object_paths|.
89 ObjectMap::const_iterator iter
= object_map_
.begin();
90 while (iter
!= object_map_
.end()) {
91 // It is safe to use a const reference here, as DBusObjectMap::RemoveObject
92 // won't access it after the iterator becomes invalidated.
93 const dbus::ObjectPath
&object_path
= iter
->first
;
95 if (!ContainsKey(object_path_set
, object_path
))
96 RemoveObject(object_path
);
100 bool DBusObjectMap::AddObject(const dbus::ObjectPath
& object_path
) {
101 ObjectMap::iterator iter
= object_map_
.find(object_path
);
102 if (iter
!= object_map_
.end())
107 dbus::ObjectProxy
* object_proxy
= bus_
->GetObjectProxy(service_name_
,
110 // Create the properties structure.
111 NfcPropertySet
* properties
= delegate_
->CreateProperties(object_proxy
);
112 properties
->ConnectSignals();
113 properties
->GetAll();
114 ObjectPropertyPair object
= std::make_pair(object_proxy
, properties
);
115 object_map_
[object_path
] = object
;
116 VLOG(1) << "Created proxy for object with Path: " << object_path
.value()
117 << ", Service: " << service_name_
;
118 delegate_
->ObjectAdded(object_path
);
122 void DBusObjectMap::RemoveObject(const dbus::ObjectPath
& object_path
) {
125 ObjectMap::iterator iter
= object_map_
.find(object_path
);
126 if (iter
== object_map_
.end())
129 // Notify the delegate, so that it can perform any clean up that is
131 delegate_
->ObjectRemoved(object_path
);
133 // Clean up the object proxy and the properties structure.
134 ObjectPropertyPair pair
= iter
->second
;
135 CleanUpObjectPropertyPair(pair
);
136 object_map_
.erase(iter
);
137 VLOG(1) << "Object proxy removed for object path: "
138 << object_path
.value();
141 void DBusObjectMap::RefreshProperties(const dbus::ObjectPath
& object_path
) {
142 NfcPropertySet
* properties
= GetObjectProperties(object_path
);
144 properties
->GetAll();
147 void DBusObjectMap::RefreshAllProperties() {
148 for (ObjectMap::const_iterator iter
= object_map_
.begin();
149 iter
!= object_map_
.end(); ++iter
) {
150 const dbus::ObjectPath
& object_path
= iter
->first
;
151 RefreshProperties(object_path
);
155 ObjectPathVector
DBusObjectMap::GetObjectPaths() {
156 std::vector
<dbus::ObjectPath
> object_paths
;
157 for (ObjectMap::const_iterator iter
= object_map_
.begin();
158 iter
!= object_map_
.end(); ++iter
) {
159 const dbus::ObjectPath
& object_path
= iter
->first
;
160 object_paths
.push_back(object_path
);
165 DBusObjectMap::ObjectPropertyPair
DBusObjectMap::GetObjectPropertyPair(
166 const dbus::ObjectPath
& object_path
) {
167 ObjectMap::iterator iter
= object_map_
.find(object_path
);
168 if (iter
!= object_map_
.end())
170 return std::make_pair(static_cast<dbus::ObjectProxy
*>(NULL
),
171 static_cast<NfcPropertySet
*>(NULL
));
174 void DBusObjectMap::CleanUpObjectPropertyPair(const ObjectPropertyPair
& pair
) {
175 // Don't remove the object proxy. There is a bug in dbus::Bus that causes a
176 // crash when object proxies are removed, due to the proxy objects not being
177 // properly reference counted. (See crbug.com/170182 and crbug.com/328264).
178 NfcPropertySet
* properties
= pair
.second
;
182 ObjectProxyTree::ObjectProxyTree() {
185 ObjectProxyTree::~ObjectProxyTree() {
186 for (PathsToObjectMapsType::iterator iter
= paths_to_object_maps_
.begin();
187 iter
!= paths_to_object_maps_
.end(); ++iter
) {
188 DBusObjectMap
* object_map
= iter
->second
;
193 bool ObjectProxyTree::CreateObjectMap(const dbus::ObjectPath
& object_path
,
194 const std::string
& service_name
,
195 DBusObjectMap::Delegate
* delegate
,
197 if (ContainsKey(paths_to_object_maps_
, object_path
)) {
198 LOG(ERROR
) << "Mapping already exists for object path: "
199 << object_path
.value();
202 paths_to_object_maps_
[object_path
] =
203 new DBusObjectMap(service_name
, delegate
, bus
);
207 void ObjectProxyTree::RemoveObjectMap(const dbus::ObjectPath
& object_path
) {
208 PathsToObjectMapsType::iterator iter
=
209 paths_to_object_maps_
.find(object_path
);
210 if (iter
== paths_to_object_maps_
.end())
212 DBusObjectMap
* object_map
= iter
->second
;
213 paths_to_object_maps_
.erase(iter
);
217 DBusObjectMap
* ObjectProxyTree::GetObjectMap(
218 const dbus::ObjectPath
& object_path
) {
219 PathsToObjectMapsType::iterator iter
=
220 paths_to_object_maps_
.find(object_path
);
221 if (iter
== paths_to_object_maps_
.end())
226 dbus::ObjectProxy
* ObjectProxyTree::FindObjectProxy(
227 const dbus::ObjectPath
& object_proxy_path
) {
228 for (PathsToObjectMapsType::iterator iter
= paths_to_object_maps_
.begin();
229 iter
!= paths_to_object_maps_
.end(); ++iter
) {
230 DBusObjectMap
* object_map
= iter
->second
;
231 dbus::ObjectProxy
* object_proxy
=
232 object_map
->GetObjectProxy(object_proxy_path
);
239 NfcPropertySet
* ObjectProxyTree::FindObjectProperties(
240 const dbus::ObjectPath
& object_proxy_path
) {
241 for (PathsToObjectMapsType::iterator iter
= paths_to_object_maps_
.begin();
242 iter
!= paths_to_object_maps_
.end(); ++iter
) {
243 DBusObjectMap
* object_map
= iter
->second
;
244 NfcPropertySet
* properties
=
245 object_map
->GetObjectProperties(object_proxy_path
);
252 } // namespace nfc_client_helpers
253 } // namespace chromeos