| 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 #include "base/bind.h" | 5 #include "base/bind.h" |
| 6 #include "base/callback.h" | 6 #include "base/callback.h" |
| 7 #include "base/command_line.h" | 7 #include "base/command_line.h" |
| 8 #include "base/memory/scoped_ptr.h" | 8 #include "base/memory/scoped_ptr.h" |
| 9 #include "base/message_loop.h" | 9 #include "base/message_loop.h" |
| 10 #include "base/stl_util.h" | 10 #include "base/stl_util.h" |
| 11 #include "base/string_util.h" | 11 #include "base/string_util.h" |
| 12 #include "base/time.h" | 12 #include "base/time.h" |
| 13 #include "media/base/data_buffer.h" | 13 #include "media/base/data_buffer.h" |
| 14 #include "media/base/filter_host.h" | |
| 15 #include "media/base/limits.h" | 14 #include "media/base/limits.h" |
| 16 #include "media/base/media_switches.h" | 15 #include "media/base/media_switches.h" |
| 17 #include "media/ffmpeg/ffmpeg_common.h" | 16 #include "media/ffmpeg/ffmpeg_common.h" |
| 18 #include "media/filters/bitstream_converter.h" | 17 #include "media/filters/bitstream_converter.h" |
| 19 #include "media/filters/ffmpeg_demuxer.h" | 18 #include "media/filters/ffmpeg_demuxer.h" |
| 20 #include "media/filters/ffmpeg_glue.h" | 19 #include "media/filters/ffmpeg_glue.h" |
| 21 #include "media/filters/ffmpeg_h264_bitstream_converter.h" | 20 #include "media/filters/ffmpeg_h264_bitstream_converter.h" |
| 22 | 21 |
| 23 namespace media { | 22 namespace media { |
| 24 | 23 |
| (...skipping 287 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 312 | 311 |
| 313 void FFmpegDemuxer::Stop(const base::Closure& callback) { | 312 void FFmpegDemuxer::Stop(const base::Closure& callback) { |
| 314 // Post a task to notify the streams to stop as well. | 313 // Post a task to notify the streams to stop as well. |
| 315 message_loop_->PostTask(FROM_HERE, | 314 message_loop_->PostTask(FROM_HERE, |
| 316 base::Bind(&FFmpegDemuxer::StopTask, this, callback)); | 315 base::Bind(&FFmpegDemuxer::StopTask, this, callback)); |
| 317 | 316 |
| 318 // Then wakes up the thread from reading. | 317 // Then wakes up the thread from reading. |
| 319 SignalReadCompleted(DataSource::kReadError); | 318 SignalReadCompleted(DataSource::kReadError); |
| 320 } | 319 } |
| 321 | 320 |
| 322 void FFmpegDemuxer::Seek(base::TimeDelta time, const FilterStatusCB& cb) { | 321 void FFmpegDemuxer::Seek(base::TimeDelta time, const PipelineStatusCB& cb) { |
| 323 message_loop_->PostTask(FROM_HERE, | 322 message_loop_->PostTask(FROM_HERE, |
| 324 base::Bind(&FFmpegDemuxer::SeekTask, this, time, cb)); | 323 base::Bind(&FFmpegDemuxer::SeekTask, this, time, cb)); |
| 325 } | 324 } |
| 326 | 325 |
| 327 void FFmpegDemuxer::SetPlaybackRate(float playback_rate) { | 326 void FFmpegDemuxer::SetPlaybackRate(float playback_rate) { |
| 328 DCHECK(data_source_.get()); | 327 DCHECK(data_source_.get()); |
| 329 data_source_->SetPlaybackRate(playback_rate); | 328 data_source_->SetPlaybackRate(playback_rate); |
| 330 } | 329 } |
| 331 | 330 |
| 332 void FFmpegDemuxer::SetPreload(Preload preload) { | 331 void FFmpegDemuxer::SetPreload(Preload preload) { |
| 333 DCHECK(data_source_.get()); | 332 DCHECK(data_source_.get()); |
| 334 data_source_->SetPreload(preload); | 333 data_source_->SetPreload(preload); |
| 335 } | 334 } |
| 336 | 335 |
| 337 void FFmpegDemuxer::OnAudioRendererDisabled() { | 336 void FFmpegDemuxer::OnAudioRendererDisabled() { |
| 338 message_loop_->PostTask(FROM_HERE, base::Bind( | 337 message_loop_->PostTask(FROM_HERE, base::Bind( |
| 339 &FFmpegDemuxer::DisableAudioStreamTask, this)); | 338 &FFmpegDemuxer::DisableAudioStreamTask, this)); |
| 340 } | 339 } |
| 341 | 340 |
| 342 void FFmpegDemuxer::set_host(FilterHost* filter_host) { | 341 void FFmpegDemuxer::set_host(DemuxerHost* demuxer_host) { |
| 343 Demuxer::set_host(filter_host); | 342 Demuxer::set_host(demuxer_host); |
| 344 if (data_source_) | 343 if (data_source_) |
| 345 data_source_->set_host(filter_host); | 344 data_source_->set_host(demuxer_host); |
| 346 if (max_duration_.InMicroseconds() >= 0) | 345 if (max_duration_.InMicroseconds() >= 0) |
| 347 host()->SetDuration(max_duration_); | 346 host()->SetDuration(max_duration_); |
| 348 if (read_position_ > 0) | 347 if (read_position_ > 0) |
| 349 host()->SetCurrentReadPosition(read_position_); | 348 host()->SetCurrentReadPosition(read_position_); |
| 350 if (deferred_status_ != PIPELINE_OK) | 349 if (deferred_status_ != PIPELINE_OK) |
| 351 host()->SetError(deferred_status_); | 350 host()->OnDemuxerError(deferred_status_); |
| 352 } | 351 } |
| 353 | 352 |
| 354 void FFmpegDemuxer::Initialize(DataSource* data_source, | 353 void FFmpegDemuxer::Initialize(DataSource* data_source, |
| 355 const PipelineStatusCB& callback) { | 354 const PipelineStatusCB& callback) { |
| 356 message_loop_->PostTask( | 355 message_loop_->PostTask( |
| 357 FROM_HERE, | 356 FROM_HERE, |
| 358 base::Bind(&FFmpegDemuxer::InitializeTask, this, | 357 base::Bind(&FFmpegDemuxer::InitializeTask, this, |
| 359 make_scoped_refptr(data_source), | 358 make_scoped_refptr(data_source), |
| 360 callback)); | 359 callback)); |
| 361 } | 360 } |
| (...skipping 26 matching lines...) Expand all Loading... |
| 388 // Asynchronous read from data source. | 387 // Asynchronous read from data source. |
| 389 data_source_->Read(read_position_, size, data, | 388 data_source_->Read(read_position_, size, data, |
| 390 base::Bind(&FFmpegDemuxer::OnReadCompleted, this)); | 389 base::Bind(&FFmpegDemuxer::OnReadCompleted, this)); |
| 391 | 390 |
| 392 // TODO(hclam): The method is called on the demuxer thread and this method | 391 // TODO(hclam): The method is called on the demuxer thread and this method |
| 393 // call will block the thread. We need to implemented an additional thread to | 392 // call will block the thread. We need to implemented an additional thread to |
| 394 // let FFmpeg demuxer methods to run on. | 393 // let FFmpeg demuxer methods to run on. |
| 395 size_t last_read_bytes = WaitForRead(); | 394 size_t last_read_bytes = WaitForRead(); |
| 396 if (last_read_bytes == DataSource::kReadError) { | 395 if (last_read_bytes == DataSource::kReadError) { |
| 397 if (host()) | 396 if (host()) |
| 398 host()->SetError(PIPELINE_ERROR_READ); | 397 host()->OnDemuxerError(PIPELINE_ERROR_READ); |
| 399 else | 398 else |
| 400 deferred_status_ = PIPELINE_ERROR_READ; | 399 deferred_status_ = PIPELINE_ERROR_READ; |
| 401 | 400 |
| 402 // Returns with a negative number to signal an error to FFmpeg. | 401 // Returns with a negative number to signal an error to FFmpeg. |
| 403 read_has_failed_ = true; | 402 read_has_failed_ = true; |
| 404 return AVERROR(EIO); | 403 return AVERROR(EIO); |
| 405 } | 404 } |
| 406 read_position_ += last_read_bytes; | 405 read_position_ += last_read_bytes; |
| 407 | 406 |
| 408 if (host()) | 407 if (host()) |
| (...skipping 165 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 574 } | 573 } |
| 575 | 574 |
| 576 bool FFmpegDemuxer::IsLocalSource() { | 575 bool FFmpegDemuxer::IsLocalSource() { |
| 577 return local_source_; | 576 return local_source_; |
| 578 } | 577 } |
| 579 | 578 |
| 580 bool FFmpegDemuxer::IsSeekable() { | 579 bool FFmpegDemuxer::IsSeekable() { |
| 581 return !IsStreaming(); | 580 return !IsStreaming(); |
| 582 } | 581 } |
| 583 | 582 |
| 584 void FFmpegDemuxer::SeekTask(base::TimeDelta time, const FilterStatusCB& cb) { | 583 void FFmpegDemuxer::SeekTask(base::TimeDelta time, const PipelineStatusCB& cb) { |
| 585 DCHECK_EQ(MessageLoop::current(), message_loop_); | 584 DCHECK_EQ(MessageLoop::current(), message_loop_); |
| 586 | 585 |
| 587 // TODO(scherkus): remove this by separating Seek() from Flush() from | 586 // TODO(scherkus): remove this by separating Seek() from Flush() from |
| 588 // Preroll() states (i.e., the implicit Seek(0) should really be a Preroll()). | 587 // Preroll() states (i.e., the implicit Seek(0) should really be a Preroll()). |
| 589 if (first_seek_hack_) { | 588 if (first_seek_hack_) { |
| 590 first_seek_hack_ = false; | 589 first_seek_hack_ = false; |
| 591 | 590 |
| 592 if (time == start_time_) { | 591 if (time == start_time_) { |
| 593 cb.Run(PIPELINE_OK); | 592 cb.Run(PIPELINE_OK); |
| 594 return; | 593 return; |
| (...skipping 138 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 733 read_event_.Wait(); | 732 read_event_.Wait(); |
| 734 return last_read_bytes_; | 733 return last_read_bytes_; |
| 735 } | 734 } |
| 736 | 735 |
| 737 void FFmpegDemuxer::SignalReadCompleted(size_t size) { | 736 void FFmpegDemuxer::SignalReadCompleted(size_t size) { |
| 738 last_read_bytes_ = size; | 737 last_read_bytes_ = size; |
| 739 read_event_.Signal(); | 738 read_event_.Signal(); |
| 740 } | 739 } |
| 741 | 740 |
| 742 } // namespace media | 741 } // namespace media |
| OLD | NEW |