| Index: components/arc/arc_bridge_service.h | 
| diff --git a/components/arc/arc_bridge_service.h b/components/arc/arc_bridge_service.h | 
| new file mode 100644 | 
| index 0000000000000000000000000000000000000000..b0bc9bb4da8f80f00aa64cf439d0ab386530870e | 
| --- /dev/null | 
| +++ b/components/arc/arc_bridge_service.h | 
| @@ -0,0 +1,202 @@ | 
| +// Copyright 2015 The Chromium 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 COMPONENTS_ARC_ARC_BRIDGE_SERVICE_H_ | 
| +#define COMPONENTS_ARC_ARC_BRIDGE_SERVICE_H_ | 
| + | 
| +#include "base/macros.h" | 
| +#include "base/observer_list.h" | 
| +#include "base/sequenced_task_runner.h" | 
| +#include "ipc/ipc_channel_proxy.h" | 
| +#include "ipc/ipc_listener.h" | 
| +#include "ipc/ipc_message.h" | 
| + | 
| +namespace arc { | 
| + | 
| +class ArcBridgeTest; | 
| + | 
| +// The Chrome-side service that handles ARC instances and ARC bridge creation. | 
| +// This service handles the lifetime of ARC instances and sets up the | 
| +// communication channel (the ARC bridge) used to send and receive messages. | 
| +class ArcBridgeService : public IPC::Listener { | 
| + public: | 
| +  // The possible states of the bridge.  In the normal flow, the state changes | 
| +  // in the following sequence: | 
| +  // | 
| +  // STOPPED | 
| +  //   SetEnabled(true) + HandleStartup() -> SocketConnect() -> | 
| +  // CONNECTING | 
| +  //   Connect() -> | 
| +  // CONNECTED | 
| +  //   SocketConnectAfterSetSocketPermissions() -> | 
| +  // STARTING | 
| +  //   StartInstance() -> OnInstanceReady() -> | 
| +  // READY | 
| +  // | 
| +  // When Shutdown() is called, the state changes depending on the state it was | 
| +  // at: | 
| +  // | 
| +  // CONNECTED/CONNECTING -> STOPPED | 
| +  // STARTING/READY -> STOPPING -> StopInstance() -> STOPPED | 
| +  enum class State { | 
| +    // ARC is not currently running. | 
| +    STOPPED, | 
| + | 
| +    // The request to connect has been sent. | 
| +    CONNECTING, | 
| + | 
| +    // The bridge has connected to the socket, but has not started the ARC | 
| +    // instance. | 
| +    CONNECTED, | 
| + | 
| +    // The ARC bridge has been set up and ARC is starting up. | 
| +    STARTING, | 
| + | 
| +    // The ARC instance has been fully initialized and is now ready for user | 
| +    // interaction. | 
| +    READY, | 
| + | 
| +    // The ARC instance has started shutting down. | 
| +    STOPPING, | 
| +  }; | 
| + | 
| +  class Observer { | 
| +   public: | 
| +    // Called whenever the state of the bridge has changed. | 
| +    virtual void OnStateChanged(State state) {} | 
| + | 
| +    // Called whenever ARC's enabled status has changed for this session. | 
| +    virtual void OnEnabledChanged(bool enabled) {} | 
| + | 
| +    // Called whenever ARC's availability has changed for this system. | 
| +    virtual void OnAvailableChanged(bool available) {} | 
| + | 
| +   protected: | 
| +    virtual ~Observer() {} | 
| +  }; | 
| + | 
| +  ArcBridgeService( | 
| +      const scoped_refptr<base::SingleThreadTaskRunner>& ipc_task_runner, | 
| +      const scoped_refptr<base::SequencedTaskRunner>& file_task_runner); | 
| +  ~ArcBridgeService() override; | 
| + | 
| +  // Gets the global instance of the ARC Bridge Service. | 
| +  static ArcBridgeService* Get(); | 
| + | 
| +  // DetectAvailability() should be called once D-Bus is available. It will | 
| +  // call CheckArcAvailability() on the session_manager. | 
| +  void DetectAvailability(); | 
| + | 
| +  // HandleStartup() should be called upon profile startup.  This will only | 
| +  // launch an instance if the instance service is available and it is enabled. | 
| +  void HandleStartup(); | 
| + | 
| +  // Shutdown() should be called when the browser is shutting down. | 
| +  void Shutdown(); | 
| + | 
| +  // Adds or removes observers. | 
| +  void AddObserver(Observer* observer); | 
| +  void RemoveObserver(Observer* observer); | 
| + | 
| +  // Gets the current state of the bridge service. | 
| +  State state() const { return state_; } | 
| + | 
| +  // Enables or disables the bridge for this session. | 
| +  void SetEnabled(bool enabled); | 
| + | 
| +  // Gets if the bridge is enabled for this session. | 
| +  bool enabled() const { return enabled_; } | 
| + | 
| +  // Gets if ARC is available in this system. | 
| +  bool available() const { return available_; } | 
| + | 
| +  // Requests registration of an input device on the ARC instance. | 
| +  // TODO(denniskempin): Make this interface more typesafe. | 
| +  // |name| should be the displayable name of the emulated device (e.g. "Chrome | 
| +  // OS Keyboard"), |device_type| the name of the device type (e.g. "keyboard") | 
| +  // and |fd| a file descriptor that emulates the kernel events of the device. | 
| +  bool RegisterInputDevice(const std::string& name, | 
| +                           const std::string& device_type, | 
| +                           base::ScopedFD fd); | 
| + | 
| + private: | 
| +  friend class arc::ArcBridgeTest; | 
| +  FRIEND_TEST_ALL_PREFIXES(ArcBridgeTest, Basic); | 
| +  FRIEND_TEST_ALL_PREFIXES(ArcBridgeTest, ShutdownWhenDisabled); | 
| + | 
| +  // If all pre-requisites are true (ARC is available, it has been enabled, and | 
| +  // the session has started), and ARC is stopped, start ARC. If ARC is running | 
| +  // and the pre-requisites stop being true, stop ARC. | 
| +  void PrerequisitesChanged(); | 
| + | 
| +  // Binds to the socket specified by |socket_path|. | 
| +  void SocketConnect(const base::FilePath& socket_path); | 
| + | 
| +  // Binds to the socket specified by |socket_path| after creating its parent | 
| +  // directory is present. | 
| +  void SocketConnectAfterEnsureParentDirectory( | 
| +      const base::FilePath& socket_path, | 
| +      bool directory_present); | 
| + | 
| +  // Internal connection method. Separated to make testing easier. | 
| +  bool Connect(const IPC::ChannelHandle& handle, IPC::Channel::Mode mode); | 
| + | 
| +  // Finishes connecting after setting socket permissions. | 
| +  void SocketConnectAfterSetSocketPermissions(const base::FilePath& socket_path, | 
| +                                              bool socket_permissions_success); | 
| + | 
| +  // Stops the running instance. | 
| +  void StopInstance(); | 
| + | 
| +  // Called when the ARC instance has finished setting up and is ready for user | 
| +  // interaction. | 
| +  void OnInstanceReady(); | 
| + | 
| +  // Changes the current state and notifies all observers. | 
| +  void SetState(State state); | 
| + | 
| +  // IPC::Listener: | 
| +  bool OnMessageReceived(const IPC::Message& message) override; | 
| + | 
| +  // DBus callbacks. | 
| +  void OnArcAvailable(bool available); | 
| +  void OnInstanceStarted(bool success); | 
| +  void OnInstanceStopped(bool success); | 
| + | 
| +  // Task runner on which incoming messages are handled. | 
| +  scoped_refptr<base::SequencedTaskRunner> origin_task_runner_; | 
| + | 
| +  // Task runner on which ipc operations are performed. | 
| +  scoped_refptr<base::SingleThreadTaskRunner> ipc_task_runner_; | 
| + | 
| +  // Task runner on which file operations are performed. | 
| +  scoped_refptr<base::SequencedTaskRunner> file_task_runner_; | 
| + | 
| +  // The channel through which messages are sent. | 
| +  scoped_ptr<IPC::ChannelProxy> ipc_channel_; | 
| + | 
| +  // List of observers to notify of state changes. | 
| +  base::ObserverList<Observer> observer_list_; | 
| + | 
| +  // If the user's session has started. | 
| +  bool session_started_; | 
| + | 
| +  // If the ARC instance service is available. | 
| +  bool available_; | 
| + | 
| +  // If ARC has been enabled by policy and user choice. | 
| +  bool enabled_; | 
| + | 
| +  // The current state of the bridge. | 
| +  ArcBridgeService::State state_; | 
| + | 
| +  // WeakPtrFactory to use callbacks. | 
| +  base::WeakPtrFactory<ArcBridgeService> weak_factory_; | 
| + | 
| +  DISALLOW_COPY_AND_ASSIGN(ArcBridgeService); | 
| +}; | 
| + | 
| +}  // namespace arc | 
| + | 
| +#endif  // COMPONENTS_ARC_ARC_BRIDGE_SERVICE_H_ | 
|  |