1 // Copyright (c) 2012 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. 4 5 #ifndef DBUS_BUS_H_ 6 #define DBUS_BUS_H_ 7 8 #include <dbus/dbus.h> 9 #include <stdint.h> 10 11 #include <map> 12 #include <set> 13 #include <string> 14 #include <utility> 15 #include <vector> 16 17 #include "base/callback.h" 18 #include "base/macros.h" 19 #include "base/memory/ref_counted.h" 20 #include "base/synchronization/waitable_event.h" 21 #include "base/threading/platform_thread.h" 22 #include "dbus/dbus_export.h" 23 #include "dbus/object_path.h" 24 25 namespace base { 26 class SequencedTaskRunner; 27 class SingleThreadTaskRunner; 28 class TaskRunner; 29 } 30 31 namespace dbus { 32 33 class ExportedObject; 34 class ObjectManager; 35 class ObjectProxy; 36 37 // Bus is used to establish a connection with D-Bus, create object 38 // proxies, and export objects. 39 // 40 // For asynchronous operations such as an asynchronous method call, the 41 // bus object will use a task runner to monitor the underlying file 42 // descriptor used for D-Bus communication. By default, the bus will use 43 // the current thread's task runner. If |dbus_task_runner| option is 44 // specified, the bus will use that task runner instead. 45 // 46 // THREADING 47 // 48 // In the D-Bus library, we use the two threads: 49 // 50 // - The origin thread: the thread that created the Bus object. 51 // - The D-Bus thread: the thread servicing |dbus_task_runner|. 52 // 53 // The origin thread is usually Chrome's UI thread. The D-Bus thread is 54 // usually a dedicated thread for the D-Bus library. 55 // 56 // BLOCKING CALLS 57 // 58 // Functions that issue blocking calls are marked "BLOCKING CALL" and 59 // these functions should be called in the D-Bus thread (if 60 // supplied). AssertOnDBusThread() is placed in these functions. 61 // 62 // Note that it's hard to tell if a libdbus function is actually blocking 63 // or not (ex. dbus_bus_request_name() internally calls 64 // dbus_connection_send_with_reply_and_block(), which is a blocking 65 // call). To err on the safe side, we consider all libdbus functions that 66 // deal with the connection to dbus-daemon to be blocking. 67 // 68 // SHUTDOWN 69 // 70 // The Bus object must be shut down manually by ShutdownAndBlock() and 71 // friends. We require the manual shutdown to make the operation explicit 72 // rather than doing it silently in the destructor. 73 // 74 // EXAMPLE USAGE: 75 // 76 // Synchronous method call: 77 // 78 // dbus::Bus::Options options; 79 // // Set up the bus options here. 80 // ... 81 // dbus::Bus bus(options); 82 // 83 // dbus::ObjectProxy* object_proxy = 84 // bus.GetObjectProxy(service_name, object_path); 85 // 86 // dbus::MethodCall method_call(interface_name, method_name); 87 // std::unique_ptr<dbus::Response> response( 88 // object_proxy.CallMethodAndBlock(&method_call, timeout_ms)); 89 // if (response.get() != nullptr) { // Success. 90 // ... 91 // } 92 // 93 // Asynchronous method call: 94 // 95 // void OnResponse(dbus::Response* response) { 96 // // response is NULL if the method call failed. 97 // if (!response) 98 // return; 99 // } 100 // 101 // ... 102 // object_proxy.CallMethod(&method_call, timeout_ms, 103 // base::Bind(&OnResponse)); 104 // 105 // Exporting a method: 106 // 107 // void Echo(dbus::MethodCall* method_call, 108 // dbus::ExportedObject::ResponseSender response_sender) { 109 // // Do something with method_call. 110 // Response* response = Response::FromMethodCall(method_call); 111 // // Build response here. 112 // // Can send an immediate response here to implement a synchronous service 113 // // or store the response_sender and send a response later to implement an 114 // // asynchronous service. 115 // response_sender.Run(response); 116 // } 117 // 118 // void OnExported(const std::string& interface_name, 119 // const ObjectPath& object_path, 120 // bool success) { 121 // // success is true if the method was exported successfully. 122 // } 123 // 124 // ... 125 // dbus::ExportedObject* exported_object = 126 // bus.GetExportedObject(service_name, object_path); 127 // exported_object.ExportMethod(interface_name, method_name, 128 // base::Bind(&Echo), 129 // base::Bind(&OnExported)); 130 // 131 // WHY IS THIS A REF COUNTED OBJECT? 132 // 133 // Bus is a ref counted object, to ensure that |this| of the object is 134 // alive when callbacks referencing |this| are called. However, after the 135 // bus is shut down, |connection_| can be NULL. Hence, callbacks should 136 // not rely on that |connection_| is alive. 137 class CHROME_DBUS_EXPORT Bus : public base::RefCountedThreadSafe<Bus> { 138 public: 139 // Specifies the bus type. SESSION is used to communicate with per-user 140 // services like GNOME applications. SYSTEM is used to communicate with 141 // system-wide services like NetworkManager. CUSTOM_ADDRESS is used to 142 // communicate with an user specified address. 143 enum BusType { 144 SESSION = DBUS_BUS_SESSION, 145 SYSTEM = DBUS_BUS_SYSTEM, 146 CUSTOM_ADDRESS, 147 }; 148 149 // Specifies the connection type. PRIVATE should usually be used unless 150 // you are sure that SHARED is safe for you, which is unlikely the case 151 // in Chrome. 152 // 153 // PRIVATE gives you a private connection, that won't be shared with 154 // other Bus objects. 155 // 156 // SHARED gives you a connection shared among other Bus objects, which 157 // is unsafe if the connection is shared with multiple threads. 158 enum ConnectionType { 159 PRIVATE, 160 SHARED, 161 }; 162 163 // Specifies whether the GetServiceOwnerAndBlock call should report or 164 // suppress errors. 165 enum GetServiceOwnerOption { 166 REPORT_ERRORS, 167 SUPPRESS_ERRORS, 168 }; 169 170 // Specifies service ownership options. 171 // 172 // REQUIRE_PRIMARY indicates that you require primary ownership of the 173 // service name. 174 // 175 // ALLOW_REPLACEMENT indicates that you'll allow another connection to 176 // steal ownership of this service name from you. 177 // 178 // REQUIRE_PRIMARY_ALLOW_REPLACEMENT does the obvious. 179 enum ServiceOwnershipOptions { 180 REQUIRE_PRIMARY = (DBUS_NAME_FLAG_DO_NOT_QUEUE | 181 DBUS_NAME_FLAG_REPLACE_EXISTING), 182 REQUIRE_PRIMARY_ALLOW_REPLACEMENT = (REQUIRE_PRIMARY | 183 DBUS_NAME_FLAG_ALLOW_REPLACEMENT), 184 }; 185 186 // Options used to create a Bus object. 187 struct CHROME_DBUS_EXPORT Options { 188 Options(); 189 ~Options(); 190 191 BusType bus_type; // SESSION by default. 192 ConnectionType connection_type; // PRIVATE by default. 193 // If dbus_task_runner is set, the bus object will use that 194 // task runner to process asynchronous operations. 195 // 196 // The thread servicing the task runner should meet the following 197 // requirements: 198 // 1) Already running. 199 // 2) Has a MessageLoopForIO. 200 scoped_refptr<base::SequencedTaskRunner> dbus_task_runner; 201 202 // Specifies the server addresses to be connected. If you want to 203 // communicate with non dbus-daemon such as ibus-daemon, set |bus_type| to 204 // CUSTOM_ADDRESS, and |address| to the D-Bus server address you want to 205 // connect to. The format of this address value is the dbus address style 206 // which is described in 207 // http://dbus.freedesktop.org/doc/dbus-specification.html#addresses 208 // 209 // EXAMPLE USAGE: 210 // dbus::Bus::Options options; 211 // options.bus_type = CUSTOM_ADDRESS; 212 // options.address.assign("unix:path=/tmp/dbus-XXXXXXX"); 213 // // Set up other options 214 // dbus::Bus bus(options); 215 // 216 // // Do something. 217 // 218 std::string address; 219 }; 220 221 // Creates a Bus object. The actual connection will be established when 222 // Connect() is called. 223 explicit Bus(const Options& options); 224 225 // Called when an ownership request is complete. 226 // Parameters: 227 // - the requested service name. 228 // - whether ownership has been obtained or not. 229 typedef base::Callback<void (const std::string&, bool)> OnOwnershipCallback; 230 231 // Called when GetServiceOwner() completes. 232 // |service_owner| is the return value from GetServiceOwnerAndBlock(). 233 typedef base::Callback<void (const std::string& service_owner)> 234 GetServiceOwnerCallback; 235 236 // TODO(satorux): Remove the service name parameter as the caller of 237 // RequestOwnership() knows the service name. 238 239 // Gets the object proxy for the given service name and the object path. 240 // The caller must not delete the returned object. 241 // 242 // Returns an existing object proxy if the bus object already owns the 243 // object proxy for the given service name and the object path. 244 // Never returns NULL. 245 // 246 // The bus will own all object proxies created by the bus, to ensure 247 // that the object proxies are detached from remote objects at the 248 // shutdown time of the bus. 249 // 250 // The object proxy is used to call methods of remote objects, and 251 // receive signals from them. 252 // 253 // |service_name| looks like "org.freedesktop.NetworkManager", and 254 // |object_path| looks like "/org/freedesktop/NetworkManager/Devices/0". 255 // 256 // Must be called in the origin thread. 257 virtual ObjectProxy* GetObjectProxy(const std::string& service_name, 258 const ObjectPath& object_path); 259 260 // Same as above, but also takes a bitfield of ObjectProxy::Options. 261 // See object_proxy.h for available options. 262 virtual ObjectProxy* GetObjectProxyWithOptions( 263 const std::string& service_name, 264 const ObjectPath& object_path, 265 int options); 266 267 // Removes the previously created object proxy for the given service 268 // name and the object path and releases its memory. 269 // 270 // If and object proxy for the given service name and object was 271 // created with GetObjectProxy, this function removes it from the 272 // bus object and detaches the ObjectProxy, invalidating any pointer 273 // previously acquired for it with GetObjectProxy. A subsequent call 274 // to GetObjectProxy will return a new object. 275 // 276 // All the object proxies are detached from remote objects at the 277 // shutdown time of the bus, but they can be detached early to reduce 278 // memory footprint and used match rules for the bus connection. 279 // 280 // |service_name| looks like "org.freedesktop.NetworkManager", and 281 // |object_path| looks like "/org/freedesktop/NetworkManager/Devices/0". 282 // |callback| is called when the object proxy is successfully removed and 283 // detached. 284 // 285 // The function returns true when there is an object proxy matching the 286 // |service_name| and |object_path| to remove, and calls |callback| when it 287 // is removed. Otherwise, it returns false and the |callback| function is 288 // never called. The |callback| argument must not be null. 289 // 290 // Must be called in the origin thread. 291 virtual bool RemoveObjectProxy(const std::string& service_name, 292 const ObjectPath& object_path, 293 const base::Closure& callback); 294 295 // Same as above, but also takes a bitfield of ObjectProxy::Options. 296 // See object_proxy.h for available options. 297 virtual bool RemoveObjectProxyWithOptions( 298 const std::string& service_name, 299 const ObjectPath& object_path, 300 int options, 301 const base::Closure& callback); 302 303 // Gets the exported object for the given object path. 304 // The caller must not delete the returned object. 305 // 306 // Returns an existing exported object if the bus object already owns 307 // the exported object for the given object path. Never returns NULL. 308 // 309 // The bus will own all exported objects created by the bus, to ensure 310 // that the exported objects are unregistered at the shutdown time of 311 // the bus. 312 // 313 // The exported object is used to export methods of local objects, and 314 // send signal from them. 315 // 316 // Must be called in the origin thread. 317 virtual ExportedObject* GetExportedObject(const ObjectPath& object_path); 318 319 // Unregisters the exported object for the given object path |object_path|. 320 // 321 // Getting an exported object for the same object path after this call 322 // will return a new object, method calls on any remaining copies of the 323 // previous object will not be called. 324 // 325 // Must be called in the origin thread. 326 virtual void UnregisterExportedObject(const ObjectPath& object_path); 327 328 329 // Gets an object manager for the given remote object path |object_path| 330 // exported by the service |service_name|. 331 // 332 // Returns an existing object manager if the bus object already owns a 333 // matching object manager, never returns NULL. 334 // 335 // The caller must not delete the returned object, the bus retains ownership 336 // of all object managers. 337 // 338 // Must be called in the origin thread. 339 virtual ObjectManager* GetObjectManager(const std::string& service_name, 340 const ObjectPath& object_path); 341 342 // Unregisters the object manager for the given remote object path 343 // |object_path| exported by the srevice |service_name|. 344 // 345 // Getting an object manager for the same remote object after this call 346 // will return a new object, method calls on any remaining copies of the 347 // previous object are not permitted. 348 // 349 // This method will asynchronously clean up any match rules that have been 350 // added for the object manager and invoke |callback| when the operation is 351 // complete. If this method returns false, then |callback| is never called. 352 // The |callback| argument must not be null. 353 // 354 // Must be called in the origin thread. 355 virtual bool RemoveObjectManager(const std::string& service_name, 356 const ObjectPath& object_path, 357 const base::Closure& callback); 358 359 // Shuts down the bus and blocks until it's done. More specifically, this 360 // function does the following: 361 // 362 // - Unregisters the object paths 363 // - Releases the service names 364 // - Closes the connection to dbus-daemon. 365 // 366 // This function can be called multiple times and it is no-op for the 2nd time 367 // calling. 368 // 369 // BLOCKING CALL. 370 virtual void ShutdownAndBlock(); 371 372 // Similar to ShutdownAndBlock(), but this function is used to 373 // synchronously shut down the bus that uses the D-Bus thread. This 374 // function is intended to be used at the very end of the browser 375 // shutdown, where it makes more sense to shut down the bus 376 // synchronously, than trying to make it asynchronous. 377 // 378 // BLOCKING CALL, but must be called in the origin thread. 379 virtual void ShutdownOnDBusThreadAndBlock(); 380 381 // Returns true if the shutdown has been completed. shutdown_completed()382 bool shutdown_completed() { return shutdown_completed_; } 383 384 // 385 // The public functions below are not intended to be used in client 386 // code. These are used to implement ObjectProxy and ExportedObject. 387 // 388 389 // Connects the bus to the dbus-daemon. 390 // Returns true on success, or the bus is already connected. 391 // 392 // BLOCKING CALL. 393 virtual bool Connect(); 394 395 // Disconnects the bus from the dbus-daemon. 396 // Safe to call multiple times and no operation after the first call. 397 // Do not call for shared connection it will be released by libdbus. 398 // 399 // BLOCKING CALL. 400 virtual void ClosePrivateConnection(); 401 402 // Requests the ownership of the service name given by |service_name|. 403 // See also RequestOwnershipAndBlock(). 404 // 405 // |on_ownership_callback| is called when the service name is obtained 406 // or failed to be obtained, in the origin thread. 407 // 408 // Must be called in the origin thread. 409 virtual void RequestOwnership(const std::string& service_name, 410 ServiceOwnershipOptions options, 411 OnOwnershipCallback on_ownership_callback); 412 413 // Requests the ownership of the given service name. 414 // Returns true on success, or the the service name is already obtained. 415 // 416 // Note that it's important to expose methods before requesting a service 417 // name with this method. See also ExportedObject::ExportMethodAndBlock() 418 // for details. 419 // 420 // BLOCKING CALL. 421 virtual bool RequestOwnershipAndBlock(const std::string& service_name, 422 ServiceOwnershipOptions options); 423 424 // Releases the ownership of the given service name. 425 // Returns true on success. 426 // 427 // BLOCKING CALL. 428 virtual bool ReleaseOwnership(const std::string& service_name); 429 430 // Sets up async operations. 431 // Returns true on success, or it's already set up. 432 // This function needs to be called before starting async operations. 433 // 434 // BLOCKING CALL. 435 virtual bool SetUpAsyncOperations(); 436 437 // Sends a message to the bus and blocks until the response is 438 // received. Used to implement synchronous method calls. 439 // 440 // BLOCKING CALL. 441 virtual DBusMessage* SendWithReplyAndBlock(DBusMessage* request, 442 int timeout_ms, 443 DBusError* error); 444 445 // Requests to send a message to the bus. The reply is handled with 446 // |pending_call| at a later time. 447 // 448 // BLOCKING CALL. 449 virtual void SendWithReply(DBusMessage* request, 450 DBusPendingCall** pending_call, 451 int timeout_ms); 452 453 // Requests to send a message to the bus. The message serial number will 454 // be stored in |serial|. 455 // 456 // BLOCKING CALL. 457 virtual void Send(DBusMessage* request, uint32_t* serial); 458 459 // Adds the message filter function. |filter_function| will be called 460 // when incoming messages are received. 461 // 462 // When a new incoming message arrives, filter functions are called in 463 // the order that they were added until the the incoming message is 464 // handled by a filter function. 465 // 466 // The same filter function associated with the same user data cannot be 467 // added more than once. 468 // 469 // BLOCKING CALL. 470 virtual void AddFilterFunction(DBusHandleMessageFunction filter_function, 471 void* user_data); 472 473 // Removes the message filter previously added by AddFilterFunction(). 474 // 475 // BLOCKING CALL. 476 virtual void RemoveFilterFunction(DBusHandleMessageFunction filter_function, 477 void* user_data); 478 479 // Adds the match rule. Messages that match the rule will be processed 480 // by the filter functions added by AddFilterFunction(). 481 // 482 // You cannot specify which filter function to use for a match rule. 483 // Instead, you should check if an incoming message is what you are 484 // interested in, in the filter functions. 485 // 486 // The same match rule can be added more than once and should be removed 487 // as many times as it was added. 488 // 489 // The match rule looks like: 490 // "type='signal', interface='org.chromium.SomeInterface'". 491 // 492 // See "Message Bus Message Routing" section in the D-Bus specification 493 // for details about match rules: 494 // http://dbus.freedesktop.org/doc/dbus-specification.html#message-bus-routing 495 // 496 // BLOCKING CALL. 497 virtual void AddMatch(const std::string& match_rule, DBusError* error); 498 499 // Removes the match rule previously added by AddMatch(). 500 // Returns false if the requested match rule is unknown or has already been 501 // removed. Otherwise, returns true and sets |error| accordingly. 502 // 503 // BLOCKING CALL. 504 virtual bool RemoveMatch(const std::string& match_rule, DBusError* error); 505 506 // Tries to register the object path. Returns true on success. 507 // Returns false if the object path is already registered. 508 // 509 // |message_function| in |vtable| will be called every time when a new 510 // |message sent to the object path arrives. 511 // 512 // The same object path must not be added more than once. 513 // 514 // See also documentation of |dbus_connection_try_register_object_path| at 515 // http://dbus.freedesktop.org/doc/api/html/group__DBusConnection.html 516 // 517 // BLOCKING CALL. 518 virtual bool TryRegisterObjectPath(const ObjectPath& object_path, 519 const DBusObjectPathVTable* vtable, 520 void* user_data, 521 DBusError* error); 522 523 // Unregister the object path. 524 // 525 // BLOCKING CALL. 526 virtual void UnregisterObjectPath(const ObjectPath& object_path); 527 528 // Returns the task runner of the D-Bus thread. 529 virtual base::TaskRunner* GetDBusTaskRunner(); 530 531 // Returns the task runner of the thread that created the bus. 532 virtual base::TaskRunner* GetOriginTaskRunner(); 533 534 // Returns true if the bus has the D-Bus thread. 535 virtual bool HasDBusThread(); 536 537 // Check whether the current thread is on the origin thread (the thread 538 // that created the bus). If not, DCHECK will fail. 539 virtual void AssertOnOriginThread(); 540 541 // Check whether the current thread is on the D-Bus thread. If not, 542 // DCHECK will fail. If the D-Bus thread is not supplied, it calls 543 // AssertOnOriginThread(). 544 virtual void AssertOnDBusThread(); 545 546 // Gets the owner for |service_name| via org.freedesktop.DBus.GetNameOwner. 547 // Returns the owner name, if any, or an empty string on failure. 548 // |options| specifies where to printing error messages or not. 549 // 550 // BLOCKING CALL. 551 virtual std::string GetServiceOwnerAndBlock(const std::string& service_name, 552 GetServiceOwnerOption options); 553 554 // A non-blocking version of GetServiceOwnerAndBlock(). 555 // Must be called in the origin thread. 556 virtual void GetServiceOwner(const std::string& service_name, 557 const GetServiceOwnerCallback& callback); 558 559 // Whenever the owner for |service_name| changes, run |callback| with the 560 // name of the new owner. If the owner goes away, then |callback| receives 561 // an empty string. 562 // 563 // Any unique (service_name, callback) can be used. Duplicate are ignored. 564 // |service_name| must not be empty and |callback| must not be null. 565 // 566 // Must be called in the origin thread. 567 virtual void ListenForServiceOwnerChange( 568 const std::string& service_name, 569 const GetServiceOwnerCallback& callback); 570 571 // Stop listening for |service_name| owner changes for |callback|. 572 // Any unique (service_name, callback) can be used. Non-registered callbacks 573 // for a given service name are ignored. 574 // |service_name| must not be empty and |callback| must not be null. 575 // 576 // Must be called in the origin thread. 577 virtual void UnlistenForServiceOwnerChange( 578 const std::string& service_name, 579 const GetServiceOwnerCallback& callback); 580 581 // Return the unique name of the bus connnection if it is connected to 582 // D-BUS. Otherwise, return an empty string. 583 std::string GetConnectionName(); 584 585 // Returns true if the bus is connected to D-Bus. is_connected()586 bool is_connected() { return connection_ != nullptr; } 587 588 protected: 589 // This is protected, so we can define sub classes. 590 virtual ~Bus(); 591 592 private: 593 friend class base::RefCountedThreadSafe<Bus>; 594 595 // Helper function used for RemoveObjectProxy(). 596 void RemoveObjectProxyInternal(scoped_refptr<dbus::ObjectProxy> object_proxy, 597 const base::Closure& callback); 598 599 // Helper functions used for RemoveObjectManager(). 600 void RemoveObjectManagerInternal( 601 scoped_refptr<dbus::ObjectManager> object_manager, 602 const base::Closure& callback); 603 void RemoveObjectManagerInternalHelper( 604 scoped_refptr<dbus::ObjectManager> object_manager, 605 const base::Closure& callback); 606 607 // Helper function used for UnregisterExportedObject(). 608 void UnregisterExportedObjectInternal( 609 scoped_refptr<dbus::ExportedObject> exported_object); 610 611 // Helper function used for ShutdownOnDBusThreadAndBlock(). 612 void ShutdownOnDBusThreadAndBlockInternal(); 613 614 // Helper function used for RequestOwnership(). 615 void RequestOwnershipInternal(const std::string& service_name, 616 ServiceOwnershipOptions options, 617 OnOwnershipCallback on_ownership_callback); 618 619 // Helper function used for GetServiceOwner(). 620 void GetServiceOwnerInternal(const std::string& service_name, 621 const GetServiceOwnerCallback& callback); 622 623 // Helper function used for ListenForServiceOwnerChange(). 624 void ListenForServiceOwnerChangeInternal( 625 const std::string& service_name, 626 const GetServiceOwnerCallback& callback); 627 628 // Helper function used for UnListenForServiceOwnerChange(). 629 void UnlistenForServiceOwnerChangeInternal( 630 const std::string& service_name, 631 const GetServiceOwnerCallback& callback); 632 633 // Processes the all incoming data to the connection, if any. 634 // 635 // BLOCKING CALL. 636 void ProcessAllIncomingDataIfAny(); 637 638 // Called when a watch object is added. Used to start monitoring the 639 // file descriptor used for D-Bus communication. 640 dbus_bool_t OnAddWatch(DBusWatch* raw_watch); 641 642 // Called when a watch object is removed. 643 void OnRemoveWatch(DBusWatch* raw_watch); 644 645 // Called when the "enabled" status of |raw_watch| is toggled. 646 void OnToggleWatch(DBusWatch* raw_watch); 647 648 // Called when a timeout object is added. Used to start monitoring 649 // timeout for method calls. 650 dbus_bool_t OnAddTimeout(DBusTimeout* raw_timeout); 651 652 // Called when a timeout object is removed. 653 void OnRemoveTimeout(DBusTimeout* raw_timeout); 654 655 // Called when the "enabled" status of |raw_timeout| is toggled. 656 void OnToggleTimeout(DBusTimeout* raw_timeout); 657 658 // Called when the dispatch status (i.e. if any incoming data is 659 // available) is changed. 660 void OnDispatchStatusChanged(DBusConnection* connection, 661 DBusDispatchStatus status); 662 663 // Called when a service owner change occurs. 664 void OnServiceOwnerChanged(DBusMessage* message); 665 666 // Callback helper functions. Redirects to the corresponding member function. 667 static dbus_bool_t OnAddWatchThunk(DBusWatch* raw_watch, void* data); 668 static void OnRemoveWatchThunk(DBusWatch* raw_watch, void* data); 669 static void OnToggleWatchThunk(DBusWatch* raw_watch, void* data); 670 static dbus_bool_t OnAddTimeoutThunk(DBusTimeout* raw_timeout, void* data); 671 static void OnRemoveTimeoutThunk(DBusTimeout* raw_timeout, void* data); 672 static void OnToggleTimeoutThunk(DBusTimeout* raw_timeout, void* data); 673 static void OnDispatchStatusChangedThunk(DBusConnection* connection, 674 DBusDispatchStatus status, 675 void* data); 676 677 // Calls OnConnectionDisconnected if the Disconnected signal is received. 678 static DBusHandlerResult OnConnectionDisconnectedFilter( 679 DBusConnection* connection, 680 DBusMessage* message, 681 void* user_data); 682 683 // Calls OnServiceOwnerChanged for a NameOwnerChanged signal. 684 static DBusHandlerResult OnServiceOwnerChangedFilter( 685 DBusConnection* connection, 686 DBusMessage* message, 687 void* user_data); 688 689 const BusType bus_type_; 690 const ConnectionType connection_type_; 691 scoped_refptr<base::SequencedTaskRunner> dbus_task_runner_; 692 base::WaitableEvent on_shutdown_; 693 DBusConnection* connection_; 694 695 scoped_refptr<base::SingleThreadTaskRunner> origin_task_runner_; 696 base::PlatformThreadId origin_thread_id_; 697 698 std::set<std::string> owned_service_names_; 699 // The following sets are used to check if rules/object_paths/filters 700 // are properly cleaned up before destruction of the bus object. 701 // Since it's not an error to add the same match rule twice, the repeated 702 // match rules are counted in a map. 703 std::map<std::string, int> match_rules_added_; 704 std::set<ObjectPath> registered_object_paths_; 705 std::set<std::pair<DBusHandleMessageFunction, void*>> filter_functions_added_; 706 707 // ObjectProxyTable is used to hold the object proxies created by the 708 // bus object. Key is a pair; the first part is a concatenated string of 709 // service name + object path, like 710 // "org.chromium.TestService/org/chromium/TestObject". 711 // The second part is the ObjectProxy::Options for the proxy. 712 typedef std::map<std::pair<std::string, int>, 713 scoped_refptr<dbus::ObjectProxy>> ObjectProxyTable; 714 ObjectProxyTable object_proxy_table_; 715 716 // ExportedObjectTable is used to hold the exported objects created by 717 // the bus object. Key is a concatenated string of service name + 718 // object path, like "org.chromium.TestService/org/chromium/TestObject". 719 typedef std::map<const dbus::ObjectPath, 720 scoped_refptr<dbus::ExportedObject>> ExportedObjectTable; 721 ExportedObjectTable exported_object_table_; 722 723 // ObjectManagerTable is used to hold the object managers created by the 724 // bus object. Key is a concatenated string of service name + object path, 725 // like "org.chromium.TestService/org/chromium/TestObject". 726 typedef std::map<std::string, 727 scoped_refptr<dbus::ObjectManager>> ObjectManagerTable; 728 ObjectManagerTable object_manager_table_; 729 730 // A map of NameOwnerChanged signals to listen for and the callbacks to run 731 // on the origin thread when the owner changes. 732 // Only accessed on the DBus thread. 733 // Key: Service name 734 // Value: Vector of callbacks. Unique and expected to be small. Not using 735 // std::set here because base::Callbacks don't have a '<' operator. 736 typedef std::map<std::string, std::vector<GetServiceOwnerCallback>> 737 ServiceOwnerChangedListenerMap; 738 ServiceOwnerChangedListenerMap service_owner_changed_listener_map_; 739 740 bool async_operations_set_up_; 741 bool shutdown_completed_; 742 743 // Counters to make sure that OnAddWatch()/OnRemoveWatch() and 744 // OnAddTimeout()/OnRemoveTimeou() are balanced. 745 int num_pending_watches_; 746 int num_pending_timeouts_; 747 748 std::string address_; 749 750 DISALLOW_COPY_AND_ASSIGN(Bus); 751 }; 752 753 } // namespace dbus 754 755 #endif // DBUS_BUS_H_ 756