Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(307)

Side by Side Diff: media/filters/ffmpeg_demuxer.h

Issue 335273002: Fix seeking when the start time is non-zero. (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: Comments. Created 6 years, 6 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « media/filters/ffmpeg_audio_decoder.cc ('k') | media/filters/ffmpeg_demuxer.cc » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. 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 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 the Demuxer interface using FFmpeg's libavformat. At this time 5 // Implements the Demuxer interface using FFmpeg's libavformat. At this time
6 // will support demuxing any audio/video format thrown at it. The streams 6 // will support demuxing any audio/video format thrown at it. The streams
7 // output mime types audio/x-ffmpeg and video/x-ffmpeg and include an integer 7 // output mime types audio/x-ffmpeg and video/x-ffmpeg and include an integer
8 // key FFmpegCodecID which contains the CodecID enumeration value. The CodecIDs 8 // key FFmpegCodecID which contains the CodecID enumeration value. The CodecIDs
9 // can be used to create and initialize the corresponding FFmpeg decoder. 9 // can be used to create and initialize the corresponding FFmpeg decoder.
10 // 10 //
11 // FFmpegDemuxer sets the duration of pipeline during initialization by using 11 // FFmpegDemuxer sets the duration of pipeline during initialization by using
12 // the duration of the longest audio/video stream. 12 // the duration of the longest audio/video stream.
13 // 13 //
14 // NOTE: since FFmpegDemuxer reads packets sequentially without seeking, media 14 // NOTE: since FFmpegDemuxer reads packets sequentially without seeking, media
15 // files with very large drift between audio/video streams may result in 15 // files with very large drift between audio/video streams may result in
16 // excessive memory consumption. 16 // excessive memory consumption.
17 // 17 //
18 // When stopped, FFmpegDemuxer and FFmpegDemuxerStream release all callbacks 18 // When stopped, FFmpegDemuxer and FFmpegDemuxerStream release all callbacks
19 // and buffered packets. Reads from a stopped FFmpegDemuxerStream will not be 19 // and buffered packets. Reads from a stopped FFmpegDemuxerStream will not be
20 // replied to. 20 // replied to.
21 21
22 #ifndef MEDIA_FILTERS_FFMPEG_DEMUXER_H_ 22 #ifndef MEDIA_FILTERS_FFMPEG_DEMUXER_H_
23 #define MEDIA_FILTERS_FFMPEG_DEMUXER_H_ 23 #define MEDIA_FILTERS_FFMPEG_DEMUXER_H_
24 24
25 #include <string> 25 #include <string>
26 #include <utility>
26 #include <vector> 27 #include <vector>
27 28
28 #include "base/callback.h" 29 #include "base/callback.h"
29 #include "base/gtest_prod_util.h" 30 #include "base/gtest_prod_util.h"
30 #include "base/memory/scoped_vector.h" 31 #include "base/memory/scoped_vector.h"
31 #include "base/threading/thread.h" 32 #include "base/threading/thread.h"
32 #include "media/base/audio_decoder_config.h" 33 #include "media/base/audio_decoder_config.h"
33 #include "media/base/decoder_buffer.h" 34 #include "media/base/decoder_buffer.h"
34 #include "media/base/decoder_buffer_queue.h" 35 #include "media/base/decoder_buffer_queue.h"
35 #include "media/base/demuxer.h" 36 #include "media/base/demuxer.h"
(...skipping 12 matching lines...) Expand all
48 49
49 class MediaLog; 50 class MediaLog;
50 class FFmpegDemuxer; 51 class FFmpegDemuxer;
51 class FFmpegGlue; 52 class FFmpegGlue;
52 class FFmpegH264ToAnnexBBitstreamConverter; 53 class FFmpegH264ToAnnexBBitstreamConverter;
53 54
54 typedef scoped_ptr<AVPacket, ScopedPtrAVFreePacket> ScopedAVPacket; 55 typedef scoped_ptr<AVPacket, ScopedPtrAVFreePacket> ScopedAVPacket;
55 56
56 class FFmpegDemuxerStream : public DemuxerStream { 57 class FFmpegDemuxerStream : public DemuxerStream {
57 public: 58 public:
58 // Keeps a copy of |demuxer| and initializes itself using information 59 // Keeps a copy of |demuxer| and initializes itself using information inside
59 // inside |stream|. Both parameters must outlive |this|. 60 // |stream|. Both parameters must outlive |this|.
60 FFmpegDemuxerStream(FFmpegDemuxer* demuxer, AVStream* stream); 61 // |discard_negative_timestamps| tells the DemuxerStream that all packets with
62 // negative timestamps should be marked for post-decode discard. All decoded
63 // data before time zero will be discarded.
64 FFmpegDemuxerStream(FFmpegDemuxer* demuxer,
65 AVStream* stream,
66 bool discard_negative_timestamps);
61 virtual ~FFmpegDemuxerStream(); 67 virtual ~FFmpegDemuxerStream();
62 68
63 // Enqueues the given AVPacket. It is invalid to queue a |packet| after 69 // Enqueues the given AVPacket. It is invalid to queue a |packet| after
64 // SetEndOfStream() has been called. 70 // SetEndOfStream() has been called.
65 void EnqueuePacket(ScopedAVPacket packet); 71 void EnqueuePacket(ScopedAVPacket packet);
66 72
67 // Enters the end of stream state. After delivering remaining queued buffers 73 // Enters the end of stream state. After delivering remaining queued buffers
68 // only end of stream buffers will be delivered. 74 // only end of stream buffers will be delivered.
69 void SetEndOfStream(); 75 void SetEndOfStream();
70 76
71 // Drops queued buffers and clears end of stream state. 77 // Drops queued buffers and clears end of stream state.
72 void FlushBuffers(); 78 void FlushBuffers();
73 79
74 // Empties the queues and ignores any additional calls to Read(). 80 // Empties the queues and ignores any additional calls to Read().
75 void Stop(); 81 void Stop();
76 82
77 // Returns the duration of this stream. 83 base::TimeDelta duration() const { return duration_; }
78 base::TimeDelta duration();
79 84
80 // DemuxerStream implementation. 85 // DemuxerStream implementation.
81 virtual Type type() OVERRIDE; 86 virtual Type type() OVERRIDE;
82 virtual void Read(const ReadCB& read_cb) OVERRIDE; 87 virtual void Read(const ReadCB& read_cb) OVERRIDE;
83 virtual void EnableBitstreamConverter() OVERRIDE; 88 virtual void EnableBitstreamConverter() OVERRIDE;
84 virtual bool SupportsConfigChanges() OVERRIDE; 89 virtual bool SupportsConfigChanges() OVERRIDE;
85 virtual AudioDecoderConfig audio_decoder_config() OVERRIDE; 90 virtual AudioDecoderConfig audio_decoder_config() OVERRIDE;
86 virtual VideoDecoderConfig video_decoder_config() OVERRIDE; 91 virtual VideoDecoderConfig video_decoder_config() OVERRIDE;
87 92
88 // Returns the range of buffered data in this stream. 93 // Returns the range of buffered data in this stream.
(...skipping 40 matching lines...) Expand 10 before | Expand all | Expand 10 after
129 DecoderBufferQueue buffer_queue_; 134 DecoderBufferQueue buffer_queue_;
130 ReadCB read_cb_; 135 ReadCB read_cb_;
131 136
132 #if defined(USE_PROPRIETARY_CODECS) 137 #if defined(USE_PROPRIETARY_CODECS)
133 scoped_ptr<FFmpegH264ToAnnexBBitstreamConverter> bitstream_converter_; 138 scoped_ptr<FFmpegH264ToAnnexBBitstreamConverter> bitstream_converter_;
134 #endif 139 #endif
135 140
136 bool bitstream_converter_enabled_; 141 bool bitstream_converter_enabled_;
137 142
138 std::string encryption_key_id_; 143 std::string encryption_key_id_;
144 const bool discard_negative_timestamps_;
139 145
140 DISALLOW_COPY_AND_ASSIGN(FFmpegDemuxerStream); 146 DISALLOW_COPY_AND_ASSIGN(FFmpegDemuxerStream);
141 }; 147 };
142 148
143 class MEDIA_EXPORT FFmpegDemuxer : public Demuxer { 149 class MEDIA_EXPORT FFmpegDemuxer : public Demuxer {
144 public: 150 public:
145 FFmpegDemuxer(const scoped_refptr<base::SingleThreadTaskRunner>& task_runner, 151 FFmpegDemuxer(const scoped_refptr<base::SingleThreadTaskRunner>& task_runner,
146 DataSource* data_source, 152 DataSource* data_source,
147 const NeedKeyCB& need_key_cb, 153 const NeedKeyCB& need_key_cb,
148 const scoped_refptr<MediaLog>& media_log); 154 const scoped_refptr<MediaLog>& media_log);
149 virtual ~FFmpegDemuxer(); 155 virtual ~FFmpegDemuxer();
150 156
151 // Demuxer implementation. 157 // Demuxer implementation.
152 virtual void Initialize(DemuxerHost* host, 158 virtual void Initialize(DemuxerHost* host,
153 const PipelineStatusCB& status_cb, 159 const PipelineStatusCB& status_cb,
154 bool enable_text_tracks) OVERRIDE; 160 bool enable_text_tracks) OVERRIDE;
155 virtual void Stop(const base::Closure& callback) OVERRIDE; 161 virtual void Stop(const base::Closure& callback) OVERRIDE;
156 virtual void Seek(base::TimeDelta time, const PipelineStatusCB& cb) OVERRIDE; 162 virtual void Seek(base::TimeDelta time, const PipelineStatusCB& cb) OVERRIDE;
157 virtual DemuxerStream* GetStream(DemuxerStream::Type type) OVERRIDE; 163 virtual DemuxerStream* GetStream(DemuxerStream::Type type) OVERRIDE;
158 virtual base::TimeDelta GetStartTime() const OVERRIDE;
159 virtual base::Time GetTimelineOffset() const OVERRIDE; 164 virtual base::Time GetTimelineOffset() const OVERRIDE;
160 virtual Liveness GetLiveness() const OVERRIDE; 165 virtual Liveness GetLiveness() const OVERRIDE;
161 166
162 // Calls |need_key_cb_| with the initialization data encountered in the file. 167 // Calls |need_key_cb_| with the initialization data encountered in the file.
163 void FireNeedKey(const std::string& init_data_type, 168 void FireNeedKey(const std::string& init_data_type,
164 const std::string& encryption_key_id); 169 const std::string& encryption_key_id);
165 170
166 // Allow FFmpegDemuxerStream to notify us when there is updated information 171 // Allow FFmpegDemuxerStream to notify us when there is updated information
167 // about capacity and what buffered data is available. 172 // about capacity and what buffered data is available.
168 void NotifyCapacityAvailable(); 173 void NotifyCapacityAvailable();
169 void NotifyBufferingChanged(); 174 void NotifyBufferingChanged();
170 175
176 // The lowest demuxed timestamp. DemuxerStream's must use this to adjust
177 // packet timestamps such that external clients see a zero-based timeline.
178 base::TimeDelta start_time() const { return start_time_; }
179
171 private: 180 private:
172 // To allow tests access to privates. 181 // To allow tests access to privates.
173 friend class FFmpegDemuxerTest; 182 friend class FFmpegDemuxerTest;
174 183
175 // FFmpeg callbacks during initialization. 184 // FFmpeg callbacks during initialization.
176 void OnOpenContextDone(const PipelineStatusCB& status_cb, bool result); 185 void OnOpenContextDone(const PipelineStatusCB& status_cb, bool result);
177 void OnFindStreamInfoDone(const PipelineStatusCB& status_cb, int result); 186 void OnFindStreamInfoDone(const PipelineStatusCB& status_cb, int result);
178 187
179 // FFmpeg callbacks during seeking. 188 // FFmpeg callbacks during seeking.
180 void OnSeekFrameDone(const PipelineStatusCB& cb, int result); 189 void OnSeekFrameDone(const PipelineStatusCB& cb, int result);
(...skipping 36 matching lines...) Expand 10 before | Expand all | Expand 10 after
217 // Tracks if there's an outstanding av_read_frame() operation. 226 // Tracks if there's an outstanding av_read_frame() operation.
218 // 227 //
219 // TODO(scherkus): Allow more than one read in flight for higher read 228 // TODO(scherkus): Allow more than one read in flight for higher read
220 // throughput using demuxer_bench to verify improvements. 229 // throughput using demuxer_bench to verify improvements.
221 bool pending_read_; 230 bool pending_read_;
222 231
223 // Tracks if there's an outstanding av_seek_frame() operation. Used to discard 232 // Tracks if there's an outstanding av_seek_frame() operation. Used to discard
224 // results of pre-seek av_read_frame() operations. 233 // results of pre-seek av_read_frame() operations.
225 bool pending_seek_; 234 bool pending_seek_;
226 235
227 // |streams_| mirrors the AVStream array in |format_context_|. It contains 236 // |streams_| mirrors the AVStream array in AVFormatContext. It contains
228 // FFmpegDemuxerStreams encapsluating AVStream objects at the same index. 237 // FFmpegDemuxerStreams encapsluating AVStream objects at the same index.
229 // 238 //
230 // Since we only support a single audio and video stream, |streams_| will 239 // Since we only support a single audio and video stream, |streams_| will
231 // contain NULL entries for additional audio/video streams as well as for 240 // contain NULL entries for additional audio/video streams as well as for
232 // stream types that we do not currently support. 241 // stream types that we do not currently support.
233 // 242 //
234 // Once initialized, operations on FFmpegDemuxerStreams should be carried out 243 // Once initialized, operations on FFmpegDemuxerStreams should be carried out
235 // on the demuxer thread. 244 // on the demuxer thread.
236 typedef ScopedVector<FFmpegDemuxerStream> StreamVector; 245 typedef ScopedVector<FFmpegDemuxerStream> StreamVector;
237 StreamVector streams_; 246 StreamVector streams_;
238 247
239 // Provides asynchronous IO to this demuxer. Consumed by |url_protocol_| to 248 // Provides asynchronous IO to this demuxer. Consumed by |url_protocol_| to
240 // integrate with libavformat. 249 // integrate with libavformat.
241 DataSource* data_source_; 250 DataSource* data_source_;
242 251
243 scoped_refptr<MediaLog> media_log_; 252 scoped_refptr<MediaLog> media_log_;
244 253
245 // Derived bitrate after initialization has completed. 254 // Derived bitrate after initialization has completed.
246 int bitrate_; 255 int bitrate_;
247 256
248 // The first timestamp of the opened media file. This is used to set the 257 // The first timestamp of the audio or video stream, whichever is lower. This
249 // starting clock value to match the timestamps in the media file. Default 258 // is used to adjust timestamps so that external consumers always see a zero
250 // is 0. 259 // based timeline.
251 base::TimeDelta start_time_; 260 base::TimeDelta start_time_;
252 261
262 // The index and start time of the preferred streams for seeking. Filled upon
263 // completion of OnFindStreamInfoDone(). Each info entry represents an index
264 // into |streams_| and the start time of that stream.
265 //
266 // Seek() will attempt to use |preferred_stream_for_seeking_| if the seek
267 // point occurs after its associated start time. Otherwise it will use
268 // |fallback_stream_for_seeking_|.
269 typedef std::pair<int, base::TimeDelta> StreamSeekInfo;
270 StreamSeekInfo preferred_stream_for_seeking_;
271 StreamSeekInfo fallback_stream_for_seeking_;
272
253 // The Time associated with timestamp 0. Set to a null 273 // The Time associated with timestamp 0. Set to a null
254 // time if the file doesn't have an association to Time. 274 // time if the file doesn't have an association to Time.
255 base::Time timeline_offset_; 275 base::Time timeline_offset_;
256 276
257 // Liveness of the stream. 277 // Liveness of the stream.
258 Liveness liveness_; 278 Liveness liveness_;
259 279
260 // Whether text streams have been enabled for this demuxer. 280 // Whether text streams have been enabled for this demuxer.
261 bool text_enabled_; 281 bool text_enabled_;
262 282
263 // Set if we know duration of the audio stream. Used when processing end of 283 // Set if we know duration of the audio stream. Used when processing end of
264 // stream -- at this moment we definitely know duration. 284 // stream -- at this moment we definitely know duration.
265 bool duration_known_; 285 bool duration_known_;
266 286
267 // FFmpegURLProtocol implementation and corresponding glue bits. 287 // FFmpegURLProtocol implementation and corresponding glue bits.
268 scoped_ptr<BlockingUrlProtocol> url_protocol_; 288 scoped_ptr<BlockingUrlProtocol> url_protocol_;
269 scoped_ptr<FFmpegGlue> glue_; 289 scoped_ptr<FFmpegGlue> glue_;
270 290
271 const NeedKeyCB need_key_cb_; 291 const NeedKeyCB need_key_cb_;
272 292
273 // NOTE: Weak pointers must be invalidated before all other member variables. 293 // NOTE: Weak pointers must be invalidated before all other member variables.
274 base::WeakPtrFactory<FFmpegDemuxer> weak_factory_; 294 base::WeakPtrFactory<FFmpegDemuxer> weak_factory_;
275 295
276 DISALLOW_COPY_AND_ASSIGN(FFmpegDemuxer); 296 DISALLOW_COPY_AND_ASSIGN(FFmpegDemuxer);
277 }; 297 };
278 298
279 } // namespace media 299 } // namespace media
280 300
281 #endif // MEDIA_FILTERS_FFMPEG_DEMUXER_H_ 301 #endif // MEDIA_FILTERS_FFMPEG_DEMUXER_H_
OLDNEW
« no previous file with comments | « media/filters/ffmpeg_audio_decoder.cc ('k') | media/filters/ffmpeg_demuxer.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698