// Copyright 2012 The Chromium Authors // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef DBUS_BUS_H_ #define DBUS_BUS_H_ #include <dbus/dbus.h> #include <stdint.h> #include <map> #include <memory> #include <set> #include <string> #include <utility> #include <vector> #include "base/functional/callback.h" #include "base/memory/raw_ptr.h" #include "base/memory/ref_counted.h" #include "base/synchronization/waitable_event.h" #include "base/threading/platform_thread.h" #include "base/types/expected.h" #include "dbus/dbus_export.h" #include "dbus/error.h" #include "dbus/object_path.h" namespace base { class SequencedTaskRunner; } namespace dbus { class ExportedObject; class ObjectManager; class ObjectProxy; class Response; // Bus is used to establish a connection with D-Bus, create object // proxies, and export objects. // // For asynchronous operations such as an asynchronous method call, the // bus object will use a task runner to monitor the underlying file // descriptor used for D-Bus communication. By default, the bus will use // the current thread's task runner. If |dbus_task_runner| option is // specified, the bus will use that task runner instead. // // THREADING // // In the D-Bus library, we use the two threads: // // - The origin thread: the thread that created the Bus object. // - The D-Bus thread: the thread servicing |dbus_task_runner|. // // The origin thread is usually Chrome's UI thread. The D-Bus thread is // usually a dedicated thread for the D-Bus library. // // BLOCKING CALLS // // Functions that issue blocking calls are marked "BLOCKING CALL" and // these functions should be called in the D-Bus thread (if // supplied). AssertOnDBusThread() is placed in these functions. // // Note that it's hard to tell if a libdbus function is actually blocking // or not (ex. dbus_bus_request_name() internally calls // dbus_connection_send_with_reply_and_block(), which is a blocking // call). To err on the safe side, we consider all libdbus functions that // deal with the connection to dbus-daemon to be blocking. // // SHUTDOWN // // The Bus object must be shut down manually by ShutdownAndBlock() and // friends. We require the manual shutdown to make the operation explicit // rather than doing it silently in the destructor. // // EXAMPLE USAGE: // // Synchronous method call: // // dbus::Bus::Options options; // // Set up the bus options here. // ... // dbus::Bus bus(options); // // dbus::ObjectProxy* object_proxy = // bus.GetObjectProxy(service_name, object_path); // // dbus::MethodCall method_call(interface_name, method_name); // std::unique_ptr<dbus::Response> response( // object_proxy.CallMethodAndBlock(&method_call, timeout_ms)); // if (response.get() != nullptr) { // Success. // ... // } // // Asynchronous method call: // // void OnResponse(dbus::Response* response) { // // response is NULL if the method call failed. // if (!response) // return; // } // // ... // object_proxy.CallMethod(&method_call, timeout_ms, // base::BindOnce(&OnResponse)); // // Exporting a method: // // void Echo(dbus::MethodCall* method_call, // dbus::ExportedObject::ResponseSender response_sender) { // // Do something with method_call. // Response* response = Response::FromMethodCall(method_call); // // Build response here. // // Can send an immediate response here to implement a synchronous service // // or store the response_sender and send a response later to implement an // // asynchronous service. // std::move(response_sender).Run(response); // } // // void OnExported(const std::string& interface_name, // const ObjectPath& object_path, // bool success) { // // success is true if the method was exported successfully. // } // // ... // dbus::ExportedObject* exported_object = // bus.GetExportedObject(service_name, object_path); // exported_object.ExportMethod(interface_name, method_name, // base::BindRepeating(&Echo), // base::BindOnce(&OnExported)); // // WHY IS THIS A REF COUNTED OBJECT? // // Bus is a ref counted object, to ensure that |this| of the object is // alive when callbacks referencing |this| are called. However, after the // bus is shut down, |connection_| can be NULL. Hence, callbacks should // not rely on that |connection_| is alive. class CHROME_DBUS_EXPORT Bus : public base::RefCountedThreadSafe<Bus> { … }; } // namespace dbus #endif // DBUS_BUS_H_