Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright 2013 The Chromium Authors. All rights reserved. | 1 // Copyright 2013 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 "media/base/android/media_decoder_job.h" | 5 #include "media/base/android/media_decoder_job.h" |
| 6 | 6 |
| 7 #include "base/bind.h" | 7 #include "base/bind.h" |
| 8 #include "base/callback_helpers.h" | 8 #include "base/callback_helpers.h" |
| 9 #include "base/debug/trace_event.h" | 9 #include "base/debug/trace_event.h" |
| 10 #include "base/message_loop/message_loop_proxy.h" | 10 #include "base/message_loop/message_loop_proxy.h" |
| 11 #include "media/base/android/media_codec_bridge.h" | 11 #include "media/base/android/media_codec_bridge.h" |
| 12 #include "media/base/bind_to_current_loop.h" | 12 #include "media/base/bind_to_current_loop.h" |
| 13 #include "media/base/buffers.h" | 13 #include "media/base/buffers.h" |
| 14 | 14 |
| 15 namespace media { | 15 namespace media { |
| 16 | 16 |
| 17 // Timeout value for media codec operations. Because the first | 17 // Timeout value for media codec operations. Because the first |
| 18 // DequeInputBuffer() can take about 150 milliseconds, use 250 milliseconds | 18 // DequeInputBuffer() can take about 150 milliseconds, use 250 milliseconds |
| 19 // here. See http://b/9357571. | 19 // here. See http://b/9357571. |
| 20 static const int kMediaCodecTimeoutInMilliseconds = 250; | 20 static const int kMediaCodecTimeoutInMilliseconds = 250; |
| 21 | 21 |
| 22 static void CopyDemuxerConfig(DemuxerConfigs* src, DemuxerConfigs* dest) { | |
| 23 dest->audio_codec = src->audio_codec; | |
| 24 dest->audio_channels = src->audio_channels; | |
| 25 dest->audio_sampling_rate = src->audio_sampling_rate; | |
| 26 dest->is_audio_encrypted = src->is_audio_encrypted; | |
| 27 dest->audio_extra_data = src->audio_extra_data; | |
| 28 dest->video_codec = src->video_codec; | |
| 29 dest->video_size = src->video_size; | |
| 30 dest->is_video_encrypted = src->is_video_encrypted; | |
| 31 dest->video_extra_data = src->video_extra_data; | |
| 32 } | |
| 33 | |
| 22 MediaDecoderJob::MediaDecoderJob( | 34 MediaDecoderJob::MediaDecoderJob( |
| 23 const scoped_refptr<base::SingleThreadTaskRunner>& decoder_task_runner, | 35 const scoped_refptr<base::SingleThreadTaskRunner>& decoder_task_runner, |
| 24 MediaCodecBridge* media_codec_bridge, | 36 MediaCodecBridge* media_codec_bridge, |
| 25 const base::Closure& request_data_cb) | 37 const base::Closure& request_data_cb) |
| 26 : ui_task_runner_(base::MessageLoopProxy::current()), | 38 : ui_task_runner_(base::MessageLoopProxy::current()), |
| 27 decoder_task_runner_(decoder_task_runner), | 39 decoder_task_runner_(decoder_task_runner), |
| 28 media_codec_bridge_(media_codec_bridge), | 40 media_codec_bridge_(media_codec_bridge), |
| 29 needs_flush_(false), | 41 needs_flush_(false), |
| 30 input_eos_encountered_(false), | 42 input_eos_encountered_(false), |
| 31 output_eos_encountered_(false), | 43 output_eos_encountered_(false), |
| (...skipping 63 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 95 if (HasData()) { | 107 if (HasData()) { |
| 96 DVLOG(1) << __FUNCTION__ << " : using previously received data"; | 108 DVLOG(1) << __FUNCTION__ << " : using previously received data"; |
| 97 ui_task_runner_->PostTask(FROM_HERE, prefetch_cb); | 109 ui_task_runner_->PostTask(FROM_HERE, prefetch_cb); |
| 98 return; | 110 return; |
| 99 } | 111 } |
| 100 | 112 |
| 101 DVLOG(1) << __FUNCTION__ << " : requesting data"; | 113 DVLOG(1) << __FUNCTION__ << " : requesting data"; |
| 102 RequestData(prefetch_cb); | 114 RequestData(prefetch_cb); |
| 103 } | 115 } |
| 104 | 116 |
| 105 bool MediaDecoderJob::Decode( | 117 DemuxerConfigs* MediaDecoderJob::Decode( |
| 106 base::TimeTicks start_time_ticks, | 118 base::TimeTicks start_time_ticks, |
| 107 base::TimeDelta start_presentation_timestamp, | 119 base::TimeDelta start_presentation_timestamp, |
| 108 const DecoderCallback& callback) { | 120 const DecoderCallback& callback) { |
| 109 DCHECK(decode_cb_.is_null()); | 121 DCHECK(decode_cb_.is_null()); |
| 110 DCHECK(on_data_received_cb_.is_null()); | 122 DCHECK(on_data_received_cb_.is_null()); |
| 111 DCHECK(ui_task_runner_->BelongsToCurrentThread()); | 123 DCHECK(ui_task_runner_->BelongsToCurrentThread()); |
| 112 | 124 |
| 113 decode_cb_ = callback; | 125 decode_cb_ = callback; |
| 114 | 126 |
| 115 if (!HasData()) { | 127 if (!HasData()) { |
| 116 RequestData(base::Bind(&MediaDecoderJob::DecodeCurrentAccessUnit, | 128 RequestData(base::Bind(&MediaDecoderJob::DecodeCurrentAccessUnit, |
| 117 base::Unretained(this), | 129 base::Unretained(this), |
| 118 start_time_ticks, | 130 start_time_ticks, |
| 119 start_presentation_timestamp)); | 131 start_presentation_timestamp)); |
| 120 return true; | 132 return NULL; |
| 121 } | 133 } |
| 122 | 134 |
| 123 if (DemuxerStream::kConfigChanged == CurrentAccessUnit().status) { | 135 if (DemuxerStream::kConfigChanged == CurrentAccessUnit().status) { |
| 124 // Clear received data because we need to handle a config change. | 136 // Clear received data because we need to handle a config change. |
| 125 decode_cb_.Reset(); | 137 decode_cb_.Reset(); |
| 126 ClearData(); | 138 size_t index = CurrentReceivedDataChunkIndex(); |
| 127 return false; | 139 CHECK_EQ(1u, received_data_[index].demuxer_configs.size()); |
| 140 DemuxerConfigs* config = new DemuxerConfigs(); | |
| 141 CopyDemuxerConfig(&(received_data_[index].demuxer_configs[0]), config); | |
|
wolenetz
2014/05/06 19:06:11
Perhaps my C++ fu is failing me, but:
Since Demux
qinmin
2014/05/06 19:29:34
Done. passing a scoped_ptr to the caller now
On
| |
| 142 return config; | |
| 128 } | 143 } |
| 129 | 144 |
| 130 DecodeCurrentAccessUnit(start_time_ticks, start_presentation_timestamp); | 145 DecodeCurrentAccessUnit(start_time_ticks, start_presentation_timestamp); |
| 131 return true; | 146 return NULL; |
| 132 } | 147 } |
| 133 | 148 |
| 134 void MediaDecoderJob::StopDecode() { | 149 void MediaDecoderJob::StopDecode() { |
| 135 DCHECK(ui_task_runner_->BelongsToCurrentThread()); | 150 DCHECK(ui_task_runner_->BelongsToCurrentThread()); |
| 136 DCHECK(is_decoding()); | 151 DCHECK(is_decoding()); |
| 137 stop_decode_pending_ = true; | 152 stop_decode_pending_ = true; |
| 138 } | 153 } |
| 139 | 154 |
| 140 void MediaDecoderJob::Flush() { | 155 void MediaDecoderJob::Flush() { |
| 141 DCHECK(decode_cb_.is_null()); | 156 DCHECK(decode_cb_.is_null()); |
| (...skipping 313 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 455 }; | 470 }; |
| 456 | 471 |
| 457 stop_decode_pending_ = false; | 472 stop_decode_pending_ = false; |
| 458 base::ResetAndReturn(&decode_cb_).Run( | 473 base::ResetAndReturn(&decode_cb_).Run( |
| 459 status, current_presentation_timestamp, max_presentation_timestamp); | 474 status, current_presentation_timestamp, max_presentation_timestamp); |
| 460 } | 475 } |
| 461 | 476 |
| 462 const AccessUnit& MediaDecoderJob::CurrentAccessUnit() const { | 477 const AccessUnit& MediaDecoderJob::CurrentAccessUnit() const { |
| 463 DCHECK(ui_task_runner_->BelongsToCurrentThread()); | 478 DCHECK(ui_task_runner_->BelongsToCurrentThread()); |
| 464 DCHECK(HasData()); | 479 DCHECK(HasData()); |
| 465 int index = NoAccessUnitsRemainingInChunk(true) ? | 480 size_t index = CurrentReceivedDataChunkIndex(); |
| 481 return received_data_[index].access_units[access_unit_index_[index]]; | |
| 482 } | |
| 483 | |
| 484 size_t MediaDecoderJob::CurrentReceivedDataChunkIndex() const { | |
| 485 return NoAccessUnitsRemainingInChunk(true) ? | |
| 466 inactive_demuxer_data_index() : current_demuxer_data_index_; | 486 inactive_demuxer_data_index() : current_demuxer_data_index_; |
| 467 return received_data_[index].access_units[access_unit_index_[index]]; | |
| 468 } | 487 } |
| 469 | 488 |
| 470 bool MediaDecoderJob::NoAccessUnitsRemainingInChunk( | 489 bool MediaDecoderJob::NoAccessUnitsRemainingInChunk( |
| 471 bool is_active_chunk) const { | 490 bool is_active_chunk) const { |
| 472 DCHECK(ui_task_runner_->BelongsToCurrentThread()); | 491 DCHECK(ui_task_runner_->BelongsToCurrentThread()); |
| 473 size_t index = is_active_chunk ? current_demuxer_data_index_ : | 492 size_t index = is_active_chunk ? current_demuxer_data_index_ : |
| 474 inactive_demuxer_data_index(); | 493 inactive_demuxer_data_index(); |
| 475 return received_data_[index].access_units.size() <= access_unit_index_[index]; | 494 return received_data_[index].access_units.size() <= access_unit_index_[index]; |
| 476 } | 495 } |
| 477 | 496 |
| (...skipping 25 matching lines...) Expand all Loading... | |
| 503 } | 522 } |
| 504 | 523 |
| 505 void MediaDecoderJob::InitializeReceivedData() { | 524 void MediaDecoderJob::InitializeReceivedData() { |
| 506 for (size_t i = 0; i < 2; ++i) { | 525 for (size_t i = 0; i < 2; ++i) { |
| 507 received_data_[i] = DemuxerData(); | 526 received_data_[i] = DemuxerData(); |
| 508 access_unit_index_[i] = 0; | 527 access_unit_index_[i] = 0; |
| 509 } | 528 } |
| 510 } | 529 } |
| 511 | 530 |
| 512 } // namespace media | 531 } // namespace media |
| OLD | NEW |