// Copyright (c) 2009 The Chromium OS Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef LIBBRILLO_BRILLO_GLIB_DBUS_H_ #define LIBBRILLO_BRILLO_GLIB_DBUS_H_ // IMPORTANT: Do not use this in new code. Instead, use libchrome's D-Bus // bindings. See https://goo.gl/EH3MmR for more details. #include <dbus/dbus-glib.h> #include <glib-object.h> #include <algorithm> #include <string> #include "base/logging.h" #include <brillo/brillo_export.h> #include <brillo/glib/object.h> struct DBusMessage; struct DBusConnection; namespace brillo { namespace dbus { // \brief BusConnection manages the ref-count for a ::DBusGConnection*. // // A BusConnection has reference semantics bound to a particular communication // bus. // // \models Copyable, Assignable // \related GetSystemBusConnection() class BRILLO_EXPORT BusConnection { public: typedef ::DBusGConnection* value_type; BusConnection(const BusConnection& x) : object_(x.object_) { if (object_) ::dbus_g_connection_ref(object_); } ~BusConnection() { if (object_) ::dbus_g_connection_unref(object_); } BusConnection& operator=(BusConnection x) { swap(*this, x); return *this; } const value_type& g_connection() const { DCHECK(object_) << "referencing an empty connection"; return object_; } operator bool() const { return object_; } bool HasConnection() const { return object_; } private: friend void swap(BusConnection& x, BusConnection& y); friend class Proxy; friend BusConnection GetSystemBusConnection(); friend BusConnection GetPrivateBusConnection(const char* address); // Constructor takes ownership BRILLO_PRIVATE explicit BusConnection(::DBusGConnection* x) : object_(x) {} value_type object_; }; inline void swap(BusConnection& x, BusConnection& y) { std::swap(x.object_, y.object_); } // \brief Proxy manages the ref-count for a ::DBusGProxy*. // // Proxy has reference semantics and represents a connection to on object on // the bus. A proxy object is constructed with a connection to a bus, a name // to an entity on the bus, a path to an object owned by the entity, and an // interface protocol name used to communicate with the object. class BRILLO_EXPORT Proxy { public: typedef ::DBusGProxy* value_type; Proxy(); // Set |connect_to_name_owner| true if you'd like to use // dbus_g_proxy_new_for_name_owner() rather than dbus_g_proxy_new_for_name(). Proxy(const BusConnection& connection, const char* name, const char* path, const char* interface, bool connect_to_name_owner); // Equivalent to Proxy(connection, name, path, interface, false). Proxy(const BusConnection& connection, const char* name, const char* path, const char* interface); // Creates a peer proxy using dbus_g_proxy_new_for_peer. Proxy(const BusConnection& connection, const char* path, const char* interface); Proxy(const Proxy& x); ~Proxy(); Proxy& operator=(Proxy x) { swap(*this, x); return *this; } const char* path() const { DCHECK(object_) << "referencing an empty proxy"; return ::dbus_g_proxy_get_path(object_); } // gproxy() returns a reference to the underlying ::DBusGProxy*. As this // library evolves, the gproxy() will be moved to be private. const value_type& gproxy() const { DCHECK(object_) << "referencing an empty proxy"; return object_; } operator bool() const { return object_; } private: BRILLO_PRIVATE static value_type GetGProxy(const BusConnection& connection, const char* name, const char* path, const char* interface, bool connect_to_name_owner); BRILLO_PRIVATE static value_type GetGPeerProxy( const BusConnection& connection, const char* path, const char* interface); BRILLO_PRIVATE operator int() const; // for safe bool cast friend void swap(Proxy& x, Proxy& y); value_type object_; }; inline void swap(Proxy& x, Proxy& y) { std::swap(x.object_, y.object_); } // \brief RegisterExclusiveService configures a GObject to run as a service on // a supplied ::BusConnection. // // RegisterExclusiveService encapsulates the process of configuring the // supplied \param object at \param service_path on the \param connection. // Exclusivity is ensured by replacing any existing services at that named // location and confirming that the connection is the primary owner. // // Type information for the \param object must be installed with // dbus_g_object_type_install_info prior to use. BRILLO_EXPORT bool RegisterExclusiveService(const BusConnection& connection, const char* interface_name, const char* service_name, const char* service_path, GObject* object); template<typename F> // F is a function signature class MonitorConnection; template<typename A1> class MonitorConnection<void(A1)> { public: MonitorConnection(const Proxy& proxy, const char* name, void (*monitor)(void*, A1), void* object) : proxy_(proxy), name_(name), monitor_(monitor), object_(object) {} static void Run(::DBusGProxy*, A1 x, MonitorConnection* self) { self->monitor_(self->object_, x); } const Proxy& proxy() const { return proxy_; } const std::string& name() const { return name_; } private: Proxy proxy_; std::string name_; void (*monitor_)(void*, A1); void* object_; }; template<typename A1, typename A2> class MonitorConnection<void(A1, A2)> { public: MonitorConnection(const Proxy& proxy, const char* name, void (*monitor)(void*, A1, A2), void* object) : proxy_(proxy), name_(name), monitor_(monitor), object_(object) {} static void Run(::DBusGProxy*, A1 x, A2 y, MonitorConnection* self) { self->monitor_(self->object_, x, y); } const Proxy& proxy() const { return proxy_; } const std::string& name() const { return name_; } private: Proxy proxy_; std::string name_; void (*monitor_)(void*, A1, A2); void* object_; }; template<typename A1, typename A2, typename A3> class MonitorConnection<void(A1, A2, A3)> { public: MonitorConnection(const Proxy& proxy, const char* name, void (*monitor)(void*, A1, A2, A3), void* object) : proxy_(proxy), name_(name), monitor_(monitor), object_(object) {} static void Run(::DBusGProxy*, A1 x, A2 y, A3 z, MonitorConnection* self) { self->monitor_(self->object_, x, y, z); } const Proxy& proxy() const { return proxy_; } const std::string& name() const { return name_; } private: Proxy proxy_; std::string name_; void (*monitor_)(void*, A1, A2, A3); void* object_; }; template<typename A1, typename A2, typename A3, typename A4> class MonitorConnection<void(A1, A2, A3, A4)> { public: MonitorConnection(const Proxy& proxy, const char* name, void (*monitor)(void*, A1, A2, A3, A4), void* object) : proxy_(proxy), name_(name), monitor_(monitor), object_(object) {} static void Run(::DBusGProxy*, A1 x, A2 y, A3 z, A4 w, MonitorConnection* self) { self->monitor_(self->object_, x, y, z, w); } const Proxy& proxy() const { return proxy_; } const std::string& name() const { return name_; } private: Proxy proxy_; std::string name_; void (*monitor_)(void*, A1, A2, A3, A4); void* object_; }; template<typename A1> MonitorConnection<void(A1)>* Monitor(const Proxy& proxy, const char* name, void (*monitor)(void*, A1), void* object) { typedef MonitorConnection<void(A1)> ConnectionType; ConnectionType* result = new ConnectionType(proxy, name, monitor, object); ::dbus_g_proxy_add_signal( proxy.gproxy(), name, glib::type_to_gtypeid<A1>(), G_TYPE_INVALID); ::dbus_g_proxy_connect_signal( proxy.gproxy(), name, G_CALLBACK(&ConnectionType::Run), result, nullptr); return result; } template<typename A1, typename A2> MonitorConnection<void(A1, A2)>* Monitor(const Proxy& proxy, const char* name, void (*monitor)(void*, A1, A2), void* object) { typedef MonitorConnection<void(A1, A2)> ConnectionType; ConnectionType* result = new ConnectionType(proxy, name, monitor, object); ::dbus_g_proxy_add_signal(proxy.gproxy(), name, glib::type_to_gtypeid<A1>(), glib::type_to_gtypeid<A2>(), G_TYPE_INVALID); ::dbus_g_proxy_connect_signal( proxy.gproxy(), name, G_CALLBACK(&ConnectionType::Run), result, nullptr); return result; } template<typename A1, typename A2, typename A3> MonitorConnection<void(A1, A2, A3)>* Monitor(const Proxy& proxy, const char* name, void (*monitor)(void*, A1, A2, A3), void* object) { typedef MonitorConnection<void(A1, A2, A3)> ConnectionType; ConnectionType* result = new ConnectionType(proxy, name, monitor, object); ::dbus_g_proxy_add_signal(proxy.gproxy(), name, glib::type_to_gtypeid<A1>(), glib::type_to_gtypeid<A2>(), glib::type_to_gtypeid<A3>(), G_TYPE_INVALID); ::dbus_g_proxy_connect_signal( proxy.gproxy(), name, G_CALLBACK(&ConnectionType::Run), result, nullptr); return result; } template<typename A1, typename A2, typename A3, typename A4> MonitorConnection<void(A1, A2, A3, A4)>* Monitor( const Proxy& proxy, const char* name, void (*monitor)(void*, A1, A2, A3, A4), void* object) { typedef MonitorConnection<void(A1, A2, A3, A4)> ConnectionType; ConnectionType* result = new ConnectionType(proxy, name, monitor, object); ::dbus_g_proxy_add_signal(proxy.gproxy(), name, glib::type_to_gtypeid<A1>(), glib::type_to_gtypeid<A2>(), glib::type_to_gtypeid<A3>(), glib::type_to_gtypeid<A4>(), G_TYPE_INVALID); ::dbus_g_proxy_connect_signal( proxy.gproxy(), name, G_CALLBACK(&ConnectionType::Run), result, nullptr); return result; } template<typename F> void Disconnect(MonitorConnection<F>* connection) { typedef MonitorConnection<F> ConnectionType; ::dbus_g_proxy_disconnect_signal(connection->proxy().gproxy(), connection->name().c_str(), G_CALLBACK(&ConnectionType::Run), connection); delete connection; } // \brief call_PtrArray() invokes a method on a proxy returning a // glib::PtrArray. // // CallPtrArray is the first instance of what is likely to be a general // way to make method calls to a proxy. It will likely be replaced with // something like Call(proxy, method, arg1, arg2, ..., ResultType*) in the // future. However, I don't yet have enough cases to generalize from. BRILLO_EXPORT bool CallPtrArray(const Proxy& proxy, const char* method, glib::ScopedPtrArray<const char*>* result); // \brief RetrieveProperty() retrieves a property of an object associated with a // proxy. // // Given a proxy to an object supporting the org.freedesktop.DBus.Properties // interface, the RetrieveProperty() call will retrieve a property of the // specified interface on the object storing it in \param result and returning // \true. If the dbus call fails or the object returned is not of type \param T, // then \false is returned and \param result is unchanged. // // \example // Proxy proxy(GetSystemBusConnection(), // "org.freedesktop.DeviceKit.Power", // A named entity on the bus // battery_name, // Path to a battery on the bus // "org.freedesktop.DBus.Properties") // Properties interface // // double x; // if (RetrieveProperty(proxy, // "org.freedesktop.DeviceKit.Power.Device", // "percentage") // std::cout << "Battery charge is " << x << "% of capacity."; // \end_example template<typename T> inline bool RetrieveProperty(const Proxy& proxy, const char* interface, const char* property, T* result) { glib::ScopedError error; glib::Value value; if (!::dbus_g_proxy_call(proxy.gproxy(), "Get", &Resetter(&error).lvalue(), G_TYPE_STRING, interface, G_TYPE_STRING, property, G_TYPE_INVALID, G_TYPE_VALUE, &value, G_TYPE_INVALID)) { LOG(ERROR) << "Getting property failed: " << (error->message ? error->message : "Unknown Error."); return false; } return glib::Retrieve(value, result); } // \brief RetrieveProperties returns a HashTable of all properties for the // specified interface. BRILLO_EXPORT bool RetrieveProperties(const Proxy& proxy, const char* interface, glib::ScopedHashTable* result); // \brief Returns a connection to the system bus. BRILLO_EXPORT BusConnection GetSystemBusConnection(); // \brief Returns a private connection to a bus at |address|. BRILLO_EXPORT BusConnection GetPrivateBusConnection(const char* address); // \brief Calls a method |method_name| with no arguments per the given |path| // and |interface_name|. Ignores return value. BRILLO_EXPORT void CallMethodWithNoArguments(const char* service_name, const char* path, const char* interface_name, const char* method_name); // \brief Low-level signal monitor base class. // // Used when there is no definite named signal sender (that Proxy // could be used for). class BRILLO_EXPORT SignalWatcher { public: SignalWatcher() {} ~SignalWatcher(); void StartMonitoring(const std::string& interface, const std::string& signal); private: // Callback invoked on the given signal arrival. virtual void OnSignal(DBusMessage* message) = 0; // Returns a string matching the D-Bus messages that we want to listen for. BRILLO_PRIVATE std::string GetDBusMatchString() const; // A D-Bus message filter to receive signals. BRILLO_PRIVATE static DBusHandlerResult FilterDBusMessage( DBusConnection* dbus_conn, DBusMessage* message, void* data); std::string interface_; std::string signal_; }; } // namespace dbus } // namespace brillo #endif // LIBBRILLO_BRILLO_GLIB_DBUS_H_