OLD | NEW |
---|---|
1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 // Implements a Demuxer that can switch among different data sources mid-stream. | 5 // Implements a Demuxer that can switch among different data sources mid-stream. |
6 // Uses FFmpegDemuxer under the covers, so see the caveats at the top of | 6 // Uses FFmpegDemuxer under the covers, so see the caveats at the top of |
7 // ffmpeg_demuxer.h. | 7 // ffmpeg_demuxer.h. |
8 | 8 |
9 #ifndef MEDIA_FILTERS_ADAPTIVE_DEMUXER_H_ | 9 #ifndef MEDIA_FILTERS_ADAPTIVE_DEMUXER_H_ |
10 #define MEDIA_FILTERS_ADAPTIVE_DEMUXER_H_ | 10 #define MEDIA_FILTERS_ADAPTIVE_DEMUXER_H_ |
11 | 11 |
12 #include <deque> | 12 #include <deque> |
13 #include <vector> | 13 #include <vector> |
14 | 14 |
15 #include "base/callback.h" | 15 #include "base/callback.h" |
16 #include "base/synchronization/lock.h" | 16 #include "base/synchronization/lock.h" |
17 #include "media/base/buffers.h" | 17 #include "media/base/buffers.h" |
18 #include "media/base/filter_factories.h" | 18 #include "media/base/filter_factories.h" |
19 #include "media/base/filters.h" | 19 #include "media/base/filters.h" |
20 #include "media/base/pipeline.h" | 20 #include "media/base/pipeline.h" |
21 #include "media/base/media_format.h" | 21 #include "media/base/media_format.h" |
22 | 22 |
23 namespace media { | 23 namespace media { |
24 | 24 |
25 class AdaptiveDemuxer; | 25 class AdaptiveDemuxer; |
26 class StreamSwitchManager; | |
26 | 27 |
27 class AdaptiveDemuxerStream : public DemuxerStream { | 28 class AdaptiveDemuxerStream : public DemuxerStream { |
28 public: | 29 public: |
29 typedef std::vector<scoped_refptr<DemuxerStream> > StreamVector; | 30 typedef std::vector<scoped_refptr<DemuxerStream> > StreamVector; |
31 typedef std::deque<ReadCallback> ReadCBQueue; | |
32 | |
33 // Typedefs used by the stream switching code to seek a stream to a desired | |
34 // location. The AdaptiveDemuxer constructs a SeekFunction for the stream | |
35 // being switched to and passes this function to | |
36 // AdaptiveDemuxerStream::ChangeCurrentStream(). For more information about | |
37 // how the SeekFunction is implemented, look at the comments for | |
38 // AdaptiveDemuxer::StartStreamSwitchSeek() below. | |
39 // | |
40 // Callback for SeekFunction. The PipelineStatus parameter indicates whether | |
41 // the seek was successful or not. The base::TimeDelta parameter indicates the | |
42 // time we actually seeked to. If the seek was successful this should be >= | |
43 // than the requested time. | |
44 typedef base::Callback<void(base::TimeDelta, PipelineStatus)> SeekFunctionCB; | |
45 | |
46 // Wraps a function that performs a seek on a specified stream. The | |
47 // base::TimeDelta parameter indicates what time to seek to. The | |
48 // SeekFunctionCBparameter specifies the callback we want called when the seek | |
49 // completes. | |
50 typedef base::Callback<void(base::TimeDelta, SeekFunctionCB)> SeekFunction; | |
30 | 51 |
31 // Keeps references to the passed-in streams. |streams| must be non-empty and | 52 // Keeps references to the passed-in streams. |streams| must be non-empty and |
32 // all the streams in it must agree on type and media_format (or be NULL). | 53 // all the streams in it must agree on type and media_format (or be NULL). |
33 // |initial_stream| must be a valid index into |streams| and specifies the | 54 // |initial_stream| must be a valid index into |streams| and specifies the |
34 // current stream on construction. | 55 // current stream on construction. |
35 AdaptiveDemuxerStream(StreamVector const& streams, int initial_stream); | 56 AdaptiveDemuxerStream(StreamVector const& streams, int initial_stream); |
36 virtual ~AdaptiveDemuxerStream(); | 57 virtual ~AdaptiveDemuxerStream(); |
37 | 58 |
59 // Notifies this stream that a Seek() was requested on the demuxer. | |
60 void OnAdaptiveDemuxerSeek(base::TimeDelta time); | |
Ami GONE FROM CHROMIUM
2011/05/25 05:39:01
s/time/seek_time/ to match elsewhere?
acolwell GONE FROM CHROMIUM
2011/05/25 21:43:46
Done.
| |
61 | |
38 // Change the stream to satisfy subsequent Read() requests from. The | 62 // Change the stream to satisfy subsequent Read() requests from. The |
39 // referenced pointer must not be NULL. | 63 // referenced pointer must not be NULL. The SeekFunction parameter |
40 void ChangeCurrentStream(int index); | 64 // provides a way to seek |streams_[index]| to a specific time. |
65 void ChangeCurrentStream(int index, const SeekFunction& seek_function, | |
66 const PipelineStatusCB& cb); | |
41 | 67 |
42 // DemuxerStream methods. | 68 // DemuxerStream methods. |
43 virtual void Read(const ReadCallback& read_callback); | 69 virtual void Read(const ReadCallback& read_callback); |
44 virtual Type type(); | 70 virtual Type type(); |
45 virtual const MediaFormat& media_format(); | 71 virtual const MediaFormat& media_format(); |
46 virtual void EnableBitstreamConverter(); | 72 virtual void EnableBitstreamConverter(); |
47 virtual AVStream* GetAVStream(); | 73 virtual AVStream* GetAVStream(); |
48 | 74 |
49 private: | 75 private: |
50 // Returns a pointer to the current stream. | 76 // Returns a pointer to the current stream. |
51 DemuxerStream* current_stream(); | 77 DemuxerStream* current_stream(); |
52 | 78 |
53 // DEBUG_MODE-only CHECK that the data members are in a reasonable state. | 79 // DEBUG_MODE-only CHECK that the data members are in a reasonable state. |
54 void DCheckSanity(); | 80 void DCheckSanity(); |
55 | 81 |
82 void OnReadDone(Buffer* buffer); | |
83 | |
84 bool IsSwitchPending_Locked() const; | |
85 bool CanStartSwitch_Locked() const; | |
86 | |
87 // Starts the stream switch. This method expects that no Read()s are | |
88 // outstanding on |streams_[current_stream_index_]|. | |
89 void StartSwitch(); | |
90 | |
91 // Called by the SeekFunction when it has completed the seek on | |
92 // |streams_[switch_index_]|. | |
93 void OnSwitchSeekDone(base::TimeDelta seek_time, PipelineStatus status); | |
94 | |
56 StreamVector streams_; | 95 StreamVector streams_; |
57 // Guards the members below. Only held for simple variable reads/writes, not | 96 // Guards the members below. Only held for simple variable reads/writes, not |
58 // during async operation. | 97 // during async operation. |
59 base::Lock lock_; | 98 base::Lock lock_; |
60 int current_stream_index_; | 99 int current_stream_index_; |
61 bool bitstream_converter_enabled_; | 100 bool bitstream_converter_enabled_; |
62 | 101 |
102 // The number of outstanding Read()'s on |streams_[current_stream_index_]|. | |
103 int pending_reads_; | |
104 | |
105 // A queue of callback objects passed to Read(). | |
106 ReadCBQueue read_cb_queue_; | |
107 | |
108 // Callback passed to ChangeCurrentStream(). This is called & reset when the | |
109 // stream switch has completed. Callback object is null when a stream switch | |
110 // is not in progress. | |
111 PipelineStatusCB switch_cb_; | |
112 | |
113 // The index of the stream we are switching to. During a stream switch | |
114 // 0 <= |switch_index_| < |streams_.size()| and | |
115 // |streams_[switch_index]| MUST NOT be null. |switch_index_| is set to -1 if | |
116 // a stream switch is not in progress. | |
117 int switch_index_; | |
118 | |
119 // Function used to seek |streams_[switch_index_]| to a specific time. The | |
120 // callback is null if a switch is not pending. | |
121 SeekFunction switch_seek_function_; | |
122 | |
123 // The timestamp of the last buffer returned via a Read() callback. | |
124 base::TimeDelta last_buffer_timestamp_; | |
125 | |
63 DISALLOW_IMPLICIT_CONSTRUCTORS(AdaptiveDemuxerStream); | 126 DISALLOW_IMPLICIT_CONSTRUCTORS(AdaptiveDemuxerStream); |
64 }; | 127 }; |
65 | 128 |
66 class AdaptiveDemuxer : public Demuxer { | 129 class AdaptiveDemuxer : public Demuxer { |
67 public: | 130 public: |
68 typedef std::vector<scoped_refptr<Demuxer> > DemuxerVector; | 131 typedef std::vector<scoped_refptr<Demuxer> > DemuxerVector; |
69 | 132 |
70 // |demuxers| must be non-empty, and the index arguments must be valid indexes | 133 // |demuxers| must be non-empty, and the index arguments must be valid indexes |
71 // into |demuxers|, or -1 to indicate no demuxer is serving that type. | 134 // into |demuxers|, or -1 to indicate no demuxer is serving that type. |
72 AdaptiveDemuxer(DemuxerVector const& demuxers, | 135 AdaptiveDemuxer(DemuxerVector const& demuxers, |
73 int initial_audio_demuxer_index, | 136 int initial_audio_demuxer_index, |
74 int initial_video_demuxer_index); | 137 int initial_video_demuxer_index); |
75 virtual ~AdaptiveDemuxer(); | 138 virtual ~AdaptiveDemuxer(); |
76 | 139 |
77 // Change which demuxers the streams will use. | 140 // Change to a different video stream. |
78 void ChangeCurrentDemuxer(int audio_index, int video_index); | 141 void ChangeVideoStream(int video_id, const PipelineStatusCB& done_cb); |
142 | |
143 // Get the ID of the current video stream. | |
144 int GetCurrentVideoId() const; | |
79 | 145 |
80 // Filter implementation. | 146 // Filter implementation. |
81 virtual void Stop(FilterCallback* callback); | 147 virtual void Stop(FilterCallback* callback); |
82 virtual void Seek(base::TimeDelta time, const FilterStatusCB& cb); | 148 virtual void Seek(base::TimeDelta time, const FilterStatusCB& cb); |
83 virtual void OnAudioRendererDisabled(); | 149 virtual void OnAudioRendererDisabled(); |
84 virtual void set_host(FilterHost* filter_host); | 150 virtual void set_host(FilterHost* filter_host); |
85 virtual void SetPlaybackRate(float playback_rate); | 151 virtual void SetPlaybackRate(float playback_rate); |
86 virtual void SetPreload(Preload preload); | 152 virtual void SetPreload(Preload preload); |
87 | 153 |
88 // Demuxer implementation. | 154 // Demuxer implementation. |
89 virtual scoped_refptr<DemuxerStream> GetStream(DemuxerStream::Type type); | 155 virtual scoped_refptr<DemuxerStream> GetStream(DemuxerStream::Type type); |
90 | 156 |
91 private: | 157 private: |
92 // Returns a pointer to the currently active demuxer of the given type. | 158 // Returns a pointer to the currently active demuxer of the given type. |
93 Demuxer* current_demuxer(DemuxerStream::Type type); | 159 Demuxer* current_demuxer(DemuxerStream::Type type); |
94 | 160 |
161 // Called when the AdaptiveDemuxerStream completes a stream switch. | |
162 void ChangeVideoStreamDone(int new_stream_index, | |
163 const PipelineStatusCB& done_cb, | |
164 PipelineStatus status); | |
165 | |
166 // Methods for SeekFunction. | |
167 // | |
168 // Seeking on a stream follows the flow below. | |
169 // | |
170 // SeekFunction.Run() -> StartStreamSwitchSeek() | |
171 // | | |
172 // V | |
173 // +--------------+ | |
174 // | Found Switch | | |
175 // | Point in | No (Seeking to get more index data.) | |
176 // | Index Data? |---> |demuxers_[stream_index]->Seek()| | |
177 // +--------------+ | | |
178 // |Yes V | |
179 // | OnIndexSeekDone() | |
180 // | | | |
181 // V V | |
182 // (Seeking stream to switch point.) Yes +--------------+ | |
183 // |demuxers_[stream_index]->Seek()| <------| Found Switch | | |
184 // | | Point in | | |
185 // | | Index Data? | | |
186 // | +--------------+ | |
187 // | | No | |
188 // V V | |
189 // OnStreamSeekDone() seek_cb(ERROR) | |
190 // | | |
191 // V | |
192 // seek_cb(OK | ERROR) | |
193 // | |
194 // When AdaptiveDemuxer creates a SeekFunction object it points to this | |
195 // method. | |
196 void StartStreamSwitchSeek( | |
197 DemuxerStream::Type type, | |
198 int stream_index, | |
199 base::TimeDelta seek_time, | |
200 const AdaptiveDemuxerStream::SeekFunctionCB& seek_cb); | |
201 | |
202 // Called when the seek for index data initiated by StartStreamSwitchSeek() | |
203 // completes. | |
204 void OnIndexSeekDone(DemuxerStream::Type type, | |
205 int stream_index, | |
206 base::TimeDelta seek_time, | |
207 const AdaptiveDemuxerStream::SeekFunctionCB& seek_cb, | |
208 PipelineStatus status); | |
209 | |
210 // Called when the stream seek initiated by StartStreamSwitchSeek() or | |
211 // OnIndexSeekDone() completes. | |
212 static void OnStreamSeekDone( | |
213 const AdaptiveDemuxerStream::SeekFunctionCB& seek_cb, | |
214 base::TimeDelta seek_point, | |
215 PipelineStatus status); | |
216 | |
95 DemuxerVector demuxers_; | 217 DemuxerVector demuxers_; |
96 scoped_refptr<AdaptiveDemuxerStream> audio_stream_; | 218 scoped_refptr<AdaptiveDemuxerStream> audio_stream_; |
97 scoped_refptr<AdaptiveDemuxerStream> video_stream_; | 219 scoped_refptr<AdaptiveDemuxerStream> video_stream_; |
98 // Guards the members below. Only held for simple variable reads/writes, not | 220 // Guards the members below. Only held for simple variable reads/writes, not |
99 // during async operation. | 221 // during async operation. |
100 base::Lock lock_; | 222 mutable base::Lock lock_; |
101 int current_audio_demuxer_index_; | 223 int current_audio_demuxer_index_; |
102 int current_video_demuxer_index_; | 224 int current_video_demuxer_index_; |
225 float playback_rate_; | |
226 bool switch_pending_; | |
227 scoped_refptr<StreamSwitchManager> stream_switch_manager_; | |
103 | 228 |
104 DISALLOW_IMPLICIT_CONSTRUCTORS(AdaptiveDemuxer); | 229 DISALLOW_IMPLICIT_CONSTRUCTORS(AdaptiveDemuxer); |
105 }; | 230 }; |
106 | 231 |
107 // AdaptiveDemuxerFactory wraps an underlying DemuxerFactory that knows how to | 232 // AdaptiveDemuxerFactory wraps an underlying DemuxerFactory that knows how to |
108 // build demuxers for a single URL, and implements a primitive (for now) version | 233 // build demuxers for a single URL, and implements a primitive (for now) version |
109 // of multi-file manifests. The manifest is encoded in the |url| parameter to | 234 // of multi-file manifests. The manifest is encoded in the |url| parameter to |
110 // Build() as: | 235 // Build() as: |
111 // x-adaptive:<initial_audio_index>:<initial_video_index>:<URL>[^<URL>]* where | 236 // x-adaptive:<initial_audio_index>:<initial_video_index>:<URL>[^<URL>]* where |
112 // <URL>'s are "real" media URLs which are passed to the underlying | 237 // <URL>'s are "real" media URLs which are passed to the underlying |
(...skipping 14 matching lines...) Expand all Loading... | |
127 | 252 |
128 private: | 253 private: |
129 scoped_ptr<DemuxerFactory> delegate_factory_; | 254 scoped_ptr<DemuxerFactory> delegate_factory_; |
130 | 255 |
131 DISALLOW_IMPLICIT_CONSTRUCTORS(AdaptiveDemuxerFactory); | 256 DISALLOW_IMPLICIT_CONSTRUCTORS(AdaptiveDemuxerFactory); |
132 }; | 257 }; |
133 | 258 |
134 } // namespace media | 259 } // namespace media |
135 | 260 |
136 #endif // MEDIA_FILTERS_ADAPTIVE_DEMUXER_H_ | 261 #endif // MEDIA_FILTERS_ADAPTIVE_DEMUXER_H_ |
OLD | NEW |