| Index: content/browser/media/capture/audio_mirroring_manager.h
|
| diff --git a/content/browser/media/capture/audio_mirroring_manager.h b/content/browser/media/capture/audio_mirroring_manager.h
|
| index 86a715e319c6510c00398dcc5ba94a4804b4e2a4..441c884ad92522a0d47f6b1894ca77e8c6ca12c9 100644
|
| --- a/content/browser/media/capture/audio_mirroring_manager.h
|
| +++ b/content/browser/media/capture/audio_mirroring_manager.h
|
| @@ -11,26 +11,33 @@
|
| //
|
| // 1. AudioRendererHost gets a CreateStream message from the render process
|
| // and, among other things, creates an AudioOutputController to control the
|
| -// audio data flow between the render and browser processes.
|
| -// 2. At some point, AudioRendererHost receives an "associate with render
|
| -// view" message. Among other actions, it registers the
|
| -// AudioOutputController with AudioMirroringManager (as a Diverter).
|
| -// 3. A user request to mirror all the audio for a single RenderView is made.
|
| -// A MirroringDestination is created, and StartMirroring() is called to
|
| -// begin the mirroring session. This causes AudioMirroringManager to
|
| -// instruct any matching Diverters to divert their audio data to the
|
| -// MirroringDestination.
|
| -//
|
| -// #2 and #3 above may occur in any order, as it is the job of
|
| -// AudioMirroringManager to realize when the players can be "matched up."
|
| +// audio data flow between the render and browser processes. More
|
| +// importantly, it registers the AudioOutputController with
|
| +// AudioMirroringManager (as a Diverter).
|
| +// 2. A user request to mirror all the audio for a WebContents is made. A
|
| +// MirroringDestination is created, and StartMirroring() is called to begin
|
| +// the mirroring session. The MirroringDestination is queried to determine
|
| +// which of all the known Diverters will re-route their audio to it.
|
| +// 3a. During a mirroring session, AudioMirroringManager may encounter new
|
| +// Diverters, and will query all the MirroringDestinations to determine
|
| +// which is a match, if any.
|
| +// 3b. During a mirroring session, a call to StartMirroring() can be made to
|
| +// request a "refresh" query on a MirroringDestination, and this will
|
| +// result in AudioMirroringManager starting/stopping only those Diverters
|
| +// that are not correctly routed to the destination.
|
| +// 3c. When a mirroring session is stopped, the remaining destinations will be
|
| +// queried to determine whether diverting should continue to a different
|
| +// destination.
|
|
|
| #ifndef CONTENT_BROWSER_MEDIA_CAPTURE_AUDIO_MIRRORING_MANAGER_H_
|
| #define CONTENT_BROWSER_MEDIA_CAPTURE_AUDIO_MIRRORING_MANAGER_H_
|
|
|
| -#include <map>
|
| +#include <set>
|
| #include <utility>
|
| +#include <vector>
|
|
|
| #include "base/basictypes.h"
|
| +#include "base/callback_forward.h"
|
| #include "base/threading/thread_checker.h"
|
| #include "content/common/content_export.h"
|
| #include "media/audio/audio_source_diverter.h"
|
| @@ -46,10 +53,25 @@ class CONTENT_EXPORT AudioMirroringManager {
|
| // Interface for diverting audio data to an alternative AudioOutputStream.
|
| typedef media::AudioSourceDiverter Diverter;
|
|
|
| + // A SourceFrameRef is a RenderFrameHost identified by a <render_process_id,
|
| + // render_frame_id> pair.
|
| + typedef std::pair<int, int> SourceFrameRef;
|
| +
|
| // Interface to be implemented by audio mirroring destinations. See comments
|
| // for StartMirroring() and StopMirroring() below.
|
| class MirroringDestination {
|
| public:
|
| + // Asynchronously query whether this MirroringDestination wants to consume
|
| + // audio sourced from each of the |candidates|. |results_callback| is run
|
| + // to indicate which of them (or none) should have audio routed to this
|
| + // MirroringDestination. |results_callback| must be run on the same thread
|
| + // as the one that called QueryForMatches().
|
| + typedef base::Callback<void(const std::set<SourceFrameRef>&)>
|
| + MatchesCallback;
|
| + virtual void QueryForMatches(
|
| + const std::set<SourceFrameRef>& candidates,
|
| + const MatchesCallback& results_callback) = 0;
|
| +
|
| // Create a consumer of audio data in the format specified by |params|, and
|
| // connect it as an input to mirroring. When Close() is called on the
|
| // returned AudioOutputStream, the input is disconnected and the object
|
| @@ -68,41 +90,68 @@ class CONTENT_EXPORT AudioMirroringManager {
|
| // Returns the global instance.
|
| static AudioMirroringManager* GetInstance();
|
|
|
| - // Add/Remove a diverter for an audio stream with a known RenderView target
|
| - // (represented by |render_process_id| + |render_view_id|). Multiple
|
| - // diverters may be added for the same target. |diverter| must live until
|
| - // after RemoveDiverter() is called.
|
| - //
|
| - // Re-entrancy warning: These methods should not be called by a Diverter
|
| - // during a Start/StopDiverting() invocation.
|
| - virtual void AddDiverter(int render_process_id, int render_view_id,
|
| + // Add/Remove a diverter for an audio stream with a known RenderFrame source
|
| + // (represented by |render_process_id| + |render_frame_id|). Multiple
|
| + // diverters may be added for the same source frame, but never the same
|
| + // diverter. |diverter| must live until after RemoveDiverter() is called.
|
| + virtual void AddDiverter(int render_process_id, int render_frame_id,
|
| Diverter* diverter);
|
| - virtual void RemoveDiverter(int render_process_id, int render_view_id,
|
| - Diverter* diverter);
|
| -
|
| - // Start/stop mirroring all audio output streams associated with a RenderView
|
| - // target (represented by |render_process_id| + |render_view_id|) to
|
| - // |destination|. |destination| must live until after StopMirroring() is
|
| - // called.
|
| - virtual void StartMirroring(int render_process_id, int render_view_id,
|
| - MirroringDestination* destination);
|
| - virtual void StopMirroring(int render_process_id, int render_view_id,
|
| - MirroringDestination* destination);
|
| + virtual void RemoveDiverter(Diverter* diverter);
|
| +
|
| + // (Re-)Start/Stop mirroring to the given |destination|. |destination| must
|
| + // live until after StopMirroring() is called. A client may request a
|
| + // re-start by calling StartMirroring() again; and this will cause
|
| + // AudioMirroringManager to query |destination| and only re-route those
|
| + // diverters that are missing/new to the returned set of matches.
|
| + virtual void StartMirroring(MirroringDestination* destination);
|
| + virtual void StopMirroring(MirroringDestination* destination);
|
|
|
| private:
|
| - // A mirroring target is a RenderView identified by a
|
| - // <render_process_id, render_view_id> pair.
|
| - typedef std::pair<int, int> Target;
|
| + friend class AudioMirroringManagerTest;
|
|
|
| - // Note: Objects in these maps are not owned.
|
| - typedef std::multimap<Target, Diverter*> DiverterMap;
|
| - typedef std::map<Target, MirroringDestination*> SessionMap;
|
| + struct StreamRoutingState {
|
| + // The source render frame associated with the audio stream.
|
| + SourceFrameRef source_render_frame;
|
|
|
| - // Currently-active divertable audio streams.
|
| - DiverterMap diverters_;
|
| + // The diverter for re-routing the audio stream.
|
| + Diverter* diverter;
|
| +
|
| + // If not NULL, the audio stream is currently being diverted to this
|
| + // destination.
|
| + MirroringDestination* destination;
|
| +
|
| + StreamRoutingState(const SourceFrameRef& source_frame,
|
| + Diverter* stream_diverter);
|
| + ~StreamRoutingState();
|
| + };
|
|
|
| - // Currently-active mirroring sessions.
|
| - SessionMap sessions_;
|
| + typedef std::vector<StreamRoutingState> StreamRoutes;
|
| + typedef std::vector<MirroringDestination*> Destinations;
|
| +
|
| + // Helper to find a destination other than |old_destination| for the given
|
| + // |candidates| to be diverted to.
|
| + void InitiateQueriesToFindNewDestination(
|
| + MirroringDestination* old_destination,
|
| + const std::set<SourceFrameRef>& candidates);
|
| +
|
| + // MirroringDestination query callback. |matches| contains all RenderFrame
|
| + // sources that will be diverted to |destination|. If |add_only| is false,
|
| + // then any Diverters currently routed to |destination| but not found in
|
| + // |matches| will be stopped.
|
| + void UpdateRoutesToDestination(MirroringDestination* destination,
|
| + bool add_only,
|
| + const std::set<SourceFrameRef>& matches);
|
| +
|
| + // Starts diverting audio to the |new_destination|, if not NULL. Otherwise,
|
| + // stops diverting audio.
|
| + static void ChangeRoute(StreamRoutingState* route,
|
| + MirroringDestination* new_destination);
|
| +
|
| + // Routing table. Contains one entry for each Diverter.
|
| + StreamRoutes routes_;
|
| +
|
| + // All active mirroring sessions.
|
| + Destinations sessions_;
|
|
|
| // Used to check that all AudioMirroringManager code runs on the same thread.
|
| base::ThreadChecker thread_checker_;
|
|
|