1 // Copyright 2014 The Chromium OS 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.
4
5 #include <brillo/dbus/dbus_object.h>
6
7 #include <vector>
8
9 #include <base/bind.h>
10 #include <base/logging.h>
11 #include <brillo/dbus/async_event_sequencer.h>
12 #include <brillo/dbus/exported_object_manager.h>
13 #include <brillo/dbus/exported_property_set.h>
14 #include <dbus/property.h>
15
16 namespace brillo {
17 namespace dbus_utils {
18
19 //////////////////////////////////////////////////////////////////////////////
20
DBusInterface(DBusObject * dbus_object,const std::string & interface_name)21 DBusInterface::DBusInterface(DBusObject* dbus_object,
22 const std::string& interface_name)
23 : dbus_object_(dbus_object), interface_name_(interface_name) {
24 }
25
AddProperty(const std::string & property_name,ExportedPropertyBase * prop_base)26 void DBusInterface::AddProperty(const std::string& property_name,
27 ExportedPropertyBase* prop_base) {
28 dbus_object_->property_set_.RegisterProperty(
29 interface_name_, property_name, prop_base);
30 }
31
ExportAsync(ExportedObjectManager * object_manager,dbus::Bus *,dbus::ExportedObject * exported_object,const dbus::ObjectPath & object_path,const AsyncEventSequencer::CompletionAction & completion_callback)32 void DBusInterface::ExportAsync(
33 ExportedObjectManager* object_manager,
34 dbus::Bus* /* bus */,
35 dbus::ExportedObject* exported_object,
36 const dbus::ObjectPath& object_path,
37 const AsyncEventSequencer::CompletionAction& completion_callback) {
38 VLOG(1) << "Registering D-Bus interface '" << interface_name_ << "' for '"
39 << object_path.value() << "'";
40 scoped_refptr<AsyncEventSequencer> sequencer(new AsyncEventSequencer());
41 for (const auto& pair : handlers_) {
42 std::string method_name = pair.first;
43 VLOG(1) << "Exporting method: " << interface_name_ << "." << method_name;
44 std::string export_error = "Failed exporting " + method_name + " method";
45 auto export_handler = sequencer->GetExportHandler(
46 interface_name_, method_name, export_error, true);
47 auto method_handler =
48 base::Bind(&DBusInterface::HandleMethodCall, base::Unretained(this));
49 exported_object->ExportMethod(
50 interface_name_, method_name, method_handler, export_handler);
51 }
52
53 std::vector<AsyncEventSequencer::CompletionAction> actions;
54 if (object_manager) {
55 auto property_writer_callback =
56 dbus_object_->property_set_.GetPropertyWriter(interface_name_);
57 actions.push_back(
58 base::Bind(&DBusInterface::ClaimInterface,
59 weak_factory_.GetWeakPtr(),
60 object_manager->AsWeakPtr(),
61 object_path,
62 property_writer_callback));
63 }
64 actions.push_back(completion_callback);
65 sequencer->OnAllTasksCompletedCall(actions);
66 }
67
ExportAndBlock(ExportedObjectManager * object_manager,dbus::Bus *,dbus::ExportedObject * exported_object,const dbus::ObjectPath & object_path)68 void DBusInterface::ExportAndBlock(
69 ExportedObjectManager* object_manager,
70 dbus::Bus* /* bus */,
71 dbus::ExportedObject* exported_object,
72 const dbus::ObjectPath& object_path) {
73 VLOG(1) << "Registering D-Bus interface '" << interface_name_ << "' for '"
74 << object_path.value() << "'";
75 for (const auto& pair : handlers_) {
76 std::string method_name = pair.first;
77 VLOG(1) << "Exporting method: " << interface_name_ << "." << method_name;
78 auto method_handler =
79 base::Bind(&DBusInterface::HandleMethodCall, base::Unretained(this));
80 if (!exported_object->ExportMethodAndBlock(
81 interface_name_, method_name, method_handler)) {
82 LOG(FATAL) << "Failed exporting " << method_name << " method";
83 }
84 }
85
86 if (object_manager) {
87 auto property_writer_callback =
88 dbus_object_->property_set_.GetPropertyWriter(interface_name_);
89 ClaimInterface(object_manager->AsWeakPtr(),
90 object_path,
91 property_writer_callback,
92 true);
93 }
94 }
95
ClaimInterface(base::WeakPtr<ExportedObjectManager> object_manager,const dbus::ObjectPath & object_path,const ExportedPropertySet::PropertyWriter & writer,bool all_succeeded)96 void DBusInterface::ClaimInterface(
97 base::WeakPtr<ExportedObjectManager> object_manager,
98 const dbus::ObjectPath& object_path,
99 const ExportedPropertySet::PropertyWriter& writer,
100 bool all_succeeded) {
101 if (!all_succeeded || !object_manager) {
102 LOG(ERROR) << "Skipping claiming interface: " << interface_name_;
103 return;
104 }
105 object_manager->ClaimInterface(object_path, interface_name_, writer);
106 release_interface_cb_.Reset(
107 base::Bind(&ExportedObjectManager::ReleaseInterface,
108 object_manager, object_path, interface_name_));
109 }
110
HandleMethodCall(dbus::MethodCall * method_call,ResponseSender sender)111 void DBusInterface::HandleMethodCall(dbus::MethodCall* method_call,
112 ResponseSender sender) {
113 std::string method_name = method_call->GetMember();
114 // Make a local copy of |interface_name_| because calling HandleMethod()
115 // can potentially kill this interface object...
116 std::string interface_name = interface_name_;
117 VLOG(1) << "Received method call request: " << interface_name << "."
118 << method_name << "(" << method_call->GetSignature() << ")";
119 auto pair = handlers_.find(method_name);
120 if (pair == handlers_.end()) {
121 auto response =
122 dbus::ErrorResponse::FromMethodCall(method_call,
123 DBUS_ERROR_UNKNOWN_METHOD,
124 "Unknown method: " + method_name);
125 sender.Run(std::move(response));
126 return;
127 }
128 VLOG(1) << "Dispatching DBus method call: " << method_name;
129 pair->second->HandleMethod(method_call, sender);
130 }
131
AddHandlerImpl(const std::string & method_name,std::unique_ptr<DBusInterfaceMethodHandlerInterface> handler)132 void DBusInterface::AddHandlerImpl(
133 const std::string& method_name,
134 std::unique_ptr<DBusInterfaceMethodHandlerInterface> handler) {
135 VLOG(1) << "Declaring method handler: " << interface_name_ << "."
136 << method_name;
137 auto res = handlers_.insert(std::make_pair(method_name, std::move(handler)));
138 CHECK(res.second) << "Method '" << method_name << "' already exists";
139 }
140
AddSignalImpl(const std::string & signal_name,const std::shared_ptr<DBusSignalBase> & signal)141 void DBusInterface::AddSignalImpl(
142 const std::string& signal_name,
143 const std::shared_ptr<DBusSignalBase>& signal) {
144 VLOG(1) << "Declaring a signal sink: " << interface_name_ << "."
145 << signal_name;
146 CHECK(signals_.insert(std::make_pair(signal_name, signal)).second)
147 << "The signal '" << signal_name << "' is already registered";
148 }
149
150 ///////////////////////////////////////////////////////////////////////////////
151
DBusObject(ExportedObjectManager * object_manager,const scoped_refptr<dbus::Bus> & bus,const dbus::ObjectPath & object_path)152 DBusObject::DBusObject(ExportedObjectManager* object_manager,
153 const scoped_refptr<dbus::Bus>& bus,
154 const dbus::ObjectPath& object_path)
155 : property_set_(bus.get()), bus_(bus), object_path_(object_path) {
156 if (object_manager)
157 object_manager_ = object_manager->AsWeakPtr();
158 }
159
~DBusObject()160 DBusObject::~DBusObject() {
161 if (exported_object_)
162 exported_object_->Unregister();
163 }
164
AddOrGetInterface(const std::string & interface_name)165 DBusInterface* DBusObject::AddOrGetInterface(
166 const std::string& interface_name) {
167 auto iter = interfaces_.find(interface_name);
168 if (iter == interfaces_.end()) {
169 VLOG(1) << "Adding an interface '" << interface_name << "' to object '"
170 << object_path_.value() << "'.";
171 // Interface doesn't exist yet. Create one...
172 std::unique_ptr<DBusInterface> new_itf(
173 new DBusInterface(this, interface_name));
174 iter = interfaces_.insert(std::make_pair(interface_name,
175 std::move(new_itf))).first;
176 }
177 return iter->second.get();
178 }
179
FindInterface(const std::string & interface_name) const180 DBusInterface* DBusObject::FindInterface(
181 const std::string& interface_name) const {
182 auto itf_iter = interfaces_.find(interface_name);
183 return (itf_iter == interfaces_.end()) ? nullptr : itf_iter->second.get();
184 }
185
RegisterAsync(const AsyncEventSequencer::CompletionAction & completion_callback)186 void DBusObject::RegisterAsync(
187 const AsyncEventSequencer::CompletionAction& completion_callback) {
188 VLOG(1) << "Registering D-Bus object '" << object_path_.value() << "'.";
189 CHECK(exported_object_ == nullptr) << "Object already registered.";
190 scoped_refptr<AsyncEventSequencer> sequencer(new AsyncEventSequencer());
191 exported_object_ = bus_->GetExportedObject(object_path_);
192
193 // Add the org.freedesktop.DBus.Properties interface to the object.
194 DBusInterface* prop_interface = AddOrGetInterface(dbus::kPropertiesInterface);
195 prop_interface->AddSimpleMethodHandler(
196 dbus::kPropertiesGetAll,
197 base::Unretained(&property_set_),
198 &ExportedPropertySet::HandleGetAll);
199 prop_interface->AddSimpleMethodHandlerWithError(
200 dbus::kPropertiesGet,
201 base::Unretained(&property_set_),
202 &ExportedPropertySet::HandleGet);
203 prop_interface->AddSimpleMethodHandlerWithError(
204 dbus::kPropertiesSet,
205 base::Unretained(&property_set_),
206 &ExportedPropertySet::HandleSet);
207 property_set_.OnPropertiesInterfaceExported(prop_interface);
208
209 // Export interface methods
210 for (const auto& pair : interfaces_) {
211 pair.second->ExportAsync(
212 object_manager_.get(),
213 bus_.get(),
214 exported_object_,
215 object_path_,
216 sequencer->GetHandler("Failed to export interface " + pair.first,
217 false));
218 }
219
220 sequencer->OnAllTasksCompletedCall({completion_callback});
221 }
222
RegisterAndBlock()223 void DBusObject::RegisterAndBlock() {
224 VLOG(1) << "Registering D-Bus object '" << object_path_.value() << "'.";
225 CHECK(exported_object_ == nullptr) << "Object already registered.";
226 exported_object_ = bus_->GetExportedObject(object_path_);
227
228 // Add the org.freedesktop.DBus.Properties interface to the object.
229 DBusInterface* prop_interface = AddOrGetInterface(dbus::kPropertiesInterface);
230 prop_interface->AddSimpleMethodHandler(
231 dbus::kPropertiesGetAll,
232 base::Unretained(&property_set_),
233 &ExportedPropertySet::HandleGetAll);
234 prop_interface->AddSimpleMethodHandlerWithError(
235 dbus::kPropertiesGet,
236 base::Unretained(&property_set_),
237 &ExportedPropertySet::HandleGet);
238 prop_interface->AddSimpleMethodHandlerWithError(
239 dbus::kPropertiesSet,
240 base::Unretained(&property_set_),
241 &ExportedPropertySet::HandleSet);
242 property_set_.OnPropertiesInterfaceExported(prop_interface);
243
244 // Export interface methods
245 for (const auto& pair : interfaces_) {
246 pair.second->ExportAndBlock(
247 object_manager_.get(),
248 bus_.get(),
249 exported_object_,
250 object_path_);
251 }
252 }
253
UnregisterAsync()254 void DBusObject::UnregisterAsync() {
255 VLOG(1) << "Unregistering D-Bus object '" << object_path_.value() << "'.";
256 CHECK(exported_object_ != nullptr) << "Object not registered.";
257
258 // This will unregister the object path from the bus.
259 exported_object_->Unregister();
260 // This will remove |exported_object_| from bus's object table. This function
261 // will also post a task to unregister |exported_object_| (same as the call
262 // above), which will be a no-op since it is already done by then.
263 // By doing both in here, the object path is guarantee to be reusable upon
264 // return from this function.
265 bus_->UnregisterExportedObject(object_path_);
266 exported_object_ = nullptr;
267 }
268
SendSignal(dbus::Signal * signal)269 bool DBusObject::SendSignal(dbus::Signal* signal) {
270 if (exported_object_) {
271 exported_object_->SendSignal(signal);
272 return true;
273 }
274 LOG(ERROR) << "Trying to send a signal from an object that is not exported";
275 return false;
276 }
277
278 } // namespace dbus_utils
279 } // namespace brillo
280