Index: media/filters/audio_decoder_selector.cc |
diff --git a/media/filters/audio_decoder_selector.cc b/media/filters/audio_decoder_selector.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..e3f76f5c6821c6f46dec120654bc41a8523091ed |
--- /dev/null |
+++ b/media/filters/audio_decoder_selector.cc |
@@ -0,0 +1,155 @@ |
+// Copyright (c) 2012 The Chromium Authors. All rights reserved. |
+// Use of this source code is governed by a BSD-style license that can be |
+// found in the LICENSE file. |
+ |
+#include "media/filters/audio_decoder_selector.h" |
+ |
+#include "base/bind.h" |
+#include "base/callback_helpers.h" |
+#include "base/logging.h" |
+#include "base/message_loop_proxy.h" |
+#include "media/base/audio_decoder_config.h" |
+#include "media/base/bind_to_loop.h" |
+#include "media/base/demuxer_stream.h" |
+#include "media/base/pipeline.h" |
+#include "media/filters/decrypting_audio_decoder.h" |
+#include "media/filters/decrypting_demuxer_stream.h" |
+ |
+namespace media { |
+ |
+AudioDecoderSelector::AudioDecoderSelector( |
+ const scoped_refptr<base::MessageLoopProxy>& message_loop, |
+ const AudioDecoderList& decoders, |
+ const SetDecryptorReadyCB& set_decryptor_ready_cb) |
+ : message_loop_(message_loop), |
+ decoders_(decoders), |
+ set_decryptor_ready_cb_(set_decryptor_ready_cb), |
+ ALLOW_THIS_IN_INITIALIZER_LIST(weak_ptr_factory_(this)) { |
+} |
+ |
+AudioDecoderSelector::~AudioDecoderSelector() {} |
+ |
+void AudioDecoderSelector::SelectAudioDecoder( |
+ const scoped_refptr<DemuxerStream>& stream, |
+ const StatisticsCB& statistics_cb, |
+ const SelectDecoderCB& select_decoder_cb) { |
+ DVLOG(2) << "SelectAudioDecoder()"; |
+ DCHECK(message_loop_->BelongsToCurrentThread()); |
+ DCHECK(stream); |
+ |
+ // Make sure the |selected_decoder_cb| on a different execution stack. |
scherkus (not reviewing)
2012/12/14 06:39:12
s/selected/select/
I'd probably reword this to sa
xhwang
2012/12/14 07:45:06
Done.
|
+ select_decoder_cb_ = BindToCurrentLoop(select_decoder_cb); |
+ |
+ const AudioDecoderConfig& config = stream->audio_decoder_config(); |
+ if (!config.IsValidConfig()) { |
+ DLOG(ERROR) << "Invalid audio stream config."; |
+ base::ResetAndReturn(&select_decoder_cb_).Run(NULL, NULL); |
+ return; |
+ } |
+ |
+ input_stream_ = stream; |
+ statistics_cb_ = statistics_cb; |
+ |
+ if (!config.is_encrypted()) { |
+ if (decoders_.empty()) { |
+ DLOG(ERROR) << "No audio decoder can be used to decode the input stream."; |
+ base::ResetAndReturn(&select_decoder_cb_).Run(NULL, NULL); |
+ return; |
+ } |
+ |
+ InitializeNextDecoder(); |
+ return; |
+ } |
+ |
+ // This could happen if Encrypted Media Extension (EME) is not enabled. |
+ if (set_decryptor_ready_cb_.is_null()) { |
+ base::ResetAndReturn(&select_decoder_cb_).Run(NULL, NULL); |
+ return; |
+ } |
+ |
+ audio_decoder_ = new DecryptingAudioDecoder(message_loop_, |
+ set_decryptor_ready_cb_); |
+ |
+ audio_decoder_->Initialize( |
+ input_stream_, |
+ BindToCurrentLoop(base::Bind( |
+ &AudioDecoderSelector::DecryptingAudioDecoderInitDone, |
+ weak_ptr_factory_.GetWeakPtr())), |
+ statistics_cb_); |
+} |
+ |
+void AudioDecoderSelector::DecryptingAudioDecoderInitDone( |
+ PipelineStatus status) { |
+ DCHECK(message_loop_->BelongsToCurrentThread()); |
+ |
+ if (status == PIPELINE_OK) { |
+ decoders_.clear(); |
+ base::ResetAndReturn(&select_decoder_cb_).Run(audio_decoder_, NULL); |
+ return; |
+ } |
+ |
+ audio_decoder_ = NULL; |
+ |
+ if (decoders_.empty()) { |
+ DLOG(ERROR) << "No audio decoder can be used to decode the input stream."; |
+ base::ResetAndReturn(&select_decoder_cb_).Run(NULL, NULL); |
+ return; |
+ } |
+ |
+ decrypted_stream_ = new DecryptingDemuxerStream( |
+ message_loop_, set_decryptor_ready_cb_); |
+ |
+ decrypted_stream_->Initialize( |
+ input_stream_, |
+ BindToCurrentLoop(base::Bind( |
+ &AudioDecoderSelector::DecryptingDemuxerStreamInitDone, |
+ weak_ptr_factory_.GetWeakPtr()))); |
+} |
+ |
+void AudioDecoderSelector::DecryptingDemuxerStreamInitDone( |
+ PipelineStatus status) { |
+ DCHECK(message_loop_->BelongsToCurrentThread()); |
+ |
+ if (status != PIPELINE_OK) { |
+ decrypted_stream_ = NULL; |
+ base::ResetAndReturn(&select_decoder_cb_).Run(NULL, NULL); |
+ return; |
+ } |
+ |
+ DCHECK(!decrypted_stream_->audio_decoder_config().is_encrypted()); |
+ input_stream_ = decrypted_stream_; |
+ InitializeNextDecoder(); |
+} |
+ |
+void AudioDecoderSelector::InitializeNextDecoder() { |
+ DCHECK(message_loop_->BelongsToCurrentThread()); |
+ DCHECK(!decoders_.empty()); |
+ |
+ audio_decoder_ = decoders_.front(); |
+ decoders_.pop_front(); |
+ DCHECK(audio_decoder_); |
+ audio_decoder_->Initialize( |
+ input_stream_, |
+ BindToCurrentLoop(base::Bind( |
+ &AudioDecoderSelector::DecoderInitDone, |
+ weak_ptr_factory_.GetWeakPtr())), |
+ statistics_cb_); |
+} |
+ |
+void AudioDecoderSelector::DecoderInitDone(PipelineStatus status) { |
+ DCHECK(message_loop_->BelongsToCurrentThread()); |
+ |
+ if (status != PIPELINE_OK) { |
+ if (!decoders_.empty()) |
+ InitializeNextDecoder(); |
+ else |
+ base::ResetAndReturn(&select_decoder_cb_).Run(NULL, NULL); |
+ return; |
+ } |
+ |
+ decoders_.clear(); |
+ base::ResetAndReturn(&select_decoder_cb_).Run(audio_decoder_, |
+ decrypted_stream_); |
+} |
+ |
+} // namespace media |