| // Copyright 2014 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 MOJO_EDK_SYSTEM_CHANNEL_MANAGER_H_ | 
 | #define MOJO_EDK_SYSTEM_CHANNEL_MANAGER_H_ | 
 |  | 
 | #include <stdint.h> | 
 |  | 
 | #include "base/callback_forward.h" | 
 | #include "base/containers/hash_tables.h" | 
 | #include "base/macros.h" | 
 | #include "base/memory/ref_counted.h" | 
 | #include "base/synchronization/lock.h" | 
 | #include "mojo/edk/embedder/scoped_platform_handle.h" | 
 | #include "mojo/edk/system/channel_info.h" | 
 |  | 
 | namespace base { | 
 | class TaskRunner; | 
 | } | 
 |  | 
 | namespace mojo { | 
 |  | 
 | namespace embedder { | 
 | class PlatformSupport; | 
 | } | 
 |  | 
 | namespace system { | 
 |  | 
 | class Channel; | 
 | class ChannelEndpoint; | 
 | class MessagePipeDispatcher; | 
 |  | 
 | // IDs for |Channel|s managed by a |ChannelManager|. (IDs should be thought of | 
 | // as specific to a given |ChannelManager|.) 0 is never a valid ID. | 
 | typedef uint64_t ChannelId; | 
 |  | 
 | const ChannelId kInvalidChannelId = 0; | 
 |  | 
 | // This class manages and "owns" |Channel|s (which typically connect to other | 
 | // processes) for a given process. This class is thread-safe, except as | 
 | // specifically noted. | 
 | class MOJO_SYSTEM_IMPL_EXPORT ChannelManager { | 
 |  public: | 
 |   // |io_thread_task_runner| should be the |TaskRunner| for the I/O thread, on | 
 |   // which this channel manager will create all channels. |platform_support| and | 
 |   // |io_thread_task_runner| should remain remain alive at least until after the | 
 |   // shutdown callback has been called. | 
 |   ChannelManager(embedder::PlatformSupport* platform_support, | 
 |                  scoped_refptr<base::TaskRunner> io_thread_task_runner); | 
 |   ~ChannelManager(); | 
 |  | 
 |   // Shuts down the channel manager, including shutting down all channels (as if | 
 |   // |ShutdownChannelOnIOThread()| were called for each channel). This must be | 
 |   // called from the I/O thread (given to the constructor) and completes | 
 |   // synchronously. This, or |Shutdown()|, must be called before destroying this | 
 |   // object. | 
 |   void ShutdownOnIOThread(); | 
 |  | 
 |   // Like |ShutdownOnIOThread()|, but may be called from any thread. On | 
 |   // completion, will call |callback| ("on" |io_thread_task_runner| if | 
 |   // |callback_thread_task_runner| is null else by posted using | 
 |   // |callback_thread_task_runner|). Note: This will always post a task to the | 
 |   // I/O thread, even it is the current thread. | 
 |   // TODO(vtl): Consider if this is really necessary, since it only has one use | 
 |   // (in tests). | 
 |   void Shutdown(const base::Closure& callback, | 
 |                 scoped_refptr<base::TaskRunner> callback_thread_task_runner); | 
 |  | 
 |   // Creates a |Channel| and adds it to the set of channels managed by this | 
 |   // |ChannelManager|. This must be called from the I/O thread (given to the | 
 |   // constructor). |channel_id| should be a valid |ChannelId| (i.e., nonzero) | 
 |   // not "assigned" to any other |Channel| being managed by this | 
 |   // |ChannelManager|. | 
 |   scoped_refptr<MessagePipeDispatcher> CreateChannelOnIOThread( | 
 |       ChannelId channel_id, | 
 |       embedder::ScopedPlatformHandle platform_handle); | 
 |  | 
 |   // Like |CreateChannelOnIOThread()|, but may be called from any thread. On | 
 |   // completion, will call |callback| ("on" |io_thread_task_runner| if | 
 |   // |callback_thread_task_runner| is null else by posted using | 
 |   // |callback_thread_task_runner|). Note: This will always post a task to the | 
 |   // I/O thread, even if |io_thread_task_runner| is the task runner for the | 
 |   // current thread. | 
 |   // TODO(vtl): The |io_thread_task_runner| argument is temporary (we should use | 
 |   // the channel manager's I/O thread). | 
 |   scoped_refptr<MessagePipeDispatcher> CreateChannel( | 
 |       ChannelId channel_id, | 
 |       embedder::ScopedPlatformHandle platform_handle, | 
 |       scoped_refptr<base::TaskRunner> io_thread_task_runner, | 
 |       const base::Closure& callback, | 
 |       scoped_refptr<base::TaskRunner> callback_thread_task_runner); | 
 |  | 
 |   // Gets the |Channel| with the given ID (which must exist). | 
 |   scoped_refptr<Channel> GetChannel(ChannelId channel_id) const; | 
 |  | 
 |   // Informs the channel manager (and thus channel) that it will be shutdown | 
 |   // soon (by calling |ShutdownChannel()|). Calling this is optional (and may in | 
 |   // fact be called multiple times) but it will suppress certain warnings (e.g., | 
 |   // for the channel being broken) and enable others (if messages are written to | 
 |   // the channel). | 
 |   void WillShutdownChannel(ChannelId channel_id); | 
 |  | 
 |   // Shuts down the channel specified by the given ID. This, or | 
 |   // |ShutdownChannel()|, should be called once per channel (created using | 
 |   // |CreateChannelOnIOThread()| or |CreateChannel()|). This must be called from | 
 |   // the channel's "channel thread", and completes synchronously. | 
 |   // TODO(vtl): "channel thread" will become "this object's I/O thread". | 
 |   void ShutdownChannelOnIOThread(ChannelId channel_id); | 
 |  | 
 |   // Like |ShutdownChannelOnIOThread()|, but may be called from any thread. It | 
 |   // will always post a task to the channel's "channel thread", and post | 
 |   // |callback| to |callback_thread_task_runner| (or execute it directly on the | 
 |   // "channel thread" if |callback_thread_task_runner| is null) on completion. | 
 |   // TODO(vtl): "channel thread" will become "this object's I/O thread". | 
 |   void ShutdownChannel( | 
 |       ChannelId channel_id, | 
 |       const base::Closure& callback, | 
 |       scoped_refptr<base::TaskRunner> callback_thread_task_runner); | 
 |  | 
 |  private: | 
 |   // Used by |Shutdown()|. Called on the I/O thread. | 
 |   void ShutdownHelper( | 
 |       const base::Closure& callback, | 
 |       scoped_refptr<base::TaskRunner> callback_thread_task_runner); | 
 |  | 
 |   // Used by |CreateChannelOnIOThread()| and |CreateChannelHelper()|. Called on | 
 |   // the I/O thread. | 
 |   void CreateChannelOnIOThreadHelper( | 
 |       ChannelId channel_id, | 
 |       embedder::ScopedPlatformHandle platform_handle, | 
 |       scoped_refptr<system::ChannelEndpoint> bootstrap_channel_endpoint); | 
 |  | 
 |   // Used by |CreateChannel()|. Called on the I/O thread. | 
 |   void CreateChannelHelper( | 
 |       ChannelId channel_id, | 
 |       embedder::ScopedPlatformHandle platform_handle, | 
 |       scoped_refptr<system::ChannelEndpoint> bootstrap_channel_endpoint, | 
 |       const base::Closure& callback, | 
 |       scoped_refptr<base::TaskRunner> callback_thread_task_runner); | 
 |  | 
 |   embedder::PlatformSupport* const platform_support_; | 
 |   const scoped_refptr<base::TaskRunner> io_thread_task_runner_; | 
 |  | 
 |   // Note: |Channel| methods should not be called under |lock_|. | 
 |   mutable base::Lock lock_;  // Protects the members below. | 
 |  | 
 |   // TODO(vtl): Once we give the |ChannelManager| one single I/O thread, we can | 
 |   // get rid of |ChannelInfo| (and just have ref pointers to |Channel|s). | 
 |   base::hash_map<ChannelId, ChannelInfo> channel_infos_; | 
 |  | 
 |   DISALLOW_COPY_AND_ASSIGN(ChannelManager); | 
 | }; | 
 |  | 
 | }  // namespace system | 
 | }  // namespace mojo | 
 |  | 
 | #endif  // MOJO_EDK_SYSTEM_CHANNEL_MANAGER_H_ |