Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright 2014 The Chromium Authors. All rights reserved. | 1 // Copyright 2014 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/mojo/services/mojo_renderer_service.h" | 5 #include "media/mojo/services/mojo_renderer_service.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/memory/scoped_vector.h" | 9 #include "base/memory/scoped_vector.h" |
| 10 #include "media/base/audio_decoder.h" | 10 #include "media/base/audio_decoder.h" |
| 11 #include "media/base/audio_renderer.h" | 11 #include "media/base/audio_renderer.h" |
| 12 #include "media/base/audio_renderer_sink.h" | 12 #include "media/base/audio_renderer_sink.h" |
| 13 #include "media/base/decryptor.h" | 13 #include "media/base/decryptor.h" |
| 14 #include "media/base/media_log.h" | 14 #include "media/base/media_log.h" |
| 15 #include "media/base/serial_runner.h" | |
|
xhwang
2014/10/29 00:22:13
not used?
DaleCurtis
2014/10/30 23:16:56
Done.
| |
| 15 #include "media/base/video_renderer.h" | 16 #include "media/base/video_renderer.h" |
| 16 #include "media/filters/audio_renderer_impl.h" | 17 #include "media/filters/audio_renderer_impl.h" |
| 17 #include "media/filters/renderer_impl.h" | 18 #include "media/filters/renderer_impl.h" |
| 19 #include "media/filters/video_renderer_impl.h" | |
| 18 #include "media/mojo/services/mojo_demuxer_stream_adapter.h" | 20 #include "media/mojo/services/mojo_demuxer_stream_adapter.h" |
| 19 #include "media/mojo/services/renderer_config.h" | 21 #include "media/mojo/services/renderer_config.h" |
| 20 #include "mojo/application/application_runner_chromium.h" | 22 #include "mojo/application/application_runner_chromium.h" |
| 21 #include "mojo/public/c/system/main.h" | 23 #include "mojo/public/c/system/main.h" |
| 22 #include "mojo/public/cpp/application/application_connection.h" | 24 #include "mojo/public/cpp/application/application_connection.h" |
| 23 #include "mojo/public/cpp/application/application_delegate.h" | 25 #include "mojo/public/cpp/application/application_delegate.h" |
| 24 #include "mojo/public/cpp/application/interface_factory_impl.h" | 26 #include "mojo/public/cpp/application/interface_factory_impl.h" |
| 25 | 27 |
| 28 // TODO(dalecurtis): Put this in renderer config. | |
| 29 #include "media/filters/ffmpeg_video_decoder.h" | |
| 30 | |
| 26 namespace media { | 31 namespace media { |
| 27 | 32 |
| 28 // Time interval to update media time. | 33 // Time interval to update media time. |
| 29 const int kTimeUpdateIntervalMs = 50; | 34 const int kTimeUpdateIntervalMs = 50; |
| 30 | 35 |
| 31 static void LogMediaSourceError(const scoped_refptr<MediaLog>& media_log, | 36 static void LogMediaSourceError(const scoped_refptr<MediaLog>& media_log, |
| 32 const std::string& error) { | 37 const std::string& error) { |
| 33 media_log->AddEvent(media_log->CreateMediaSourceErrorEvent(error)); | 38 media_log->AddEvent(media_log->CreateMediaSourceErrorEvent(error)); |
| 34 } | 39 } |
| 35 | 40 |
| 41 static void PaintNothingDropEverything(const scoped_refptr<VideoFrame>& frame) { | |
|
xhwang
2014/10/29 00:22:13
PaintNothing == DropEverything ? Pick one of them?
DaleCurtis
2014/10/30 23:16:56
Done.
| |
| 42 } | |
| 43 | |
| 36 // Shim DemuxerStreamProvider wrapper for a single DemuxerStream. | 44 // Shim DemuxerStreamProvider wrapper for a single DemuxerStream. |
| 37 // TODO(dalecurtis): Once we support more than one DemuxerStream we'll need a | 45 // TODO(dalecurtis): Once we support more than one DemuxerStream we'll need a |
| 38 // more complicated shim which can handle a mojo::Array<DemuxerStream>. | 46 // more complicated shim which can handle a mojo::Array<DemuxerStream>. |
| 47 // TODO(dalecurtis): This should probably be in its own file now. | |
| 39 class DemuxerStreamProviderShim : public DemuxerStreamProvider { | 48 class DemuxerStreamProviderShim : public DemuxerStreamProvider { |
| 40 public: | 49 public: |
| 41 DemuxerStreamProviderShim(scoped_ptr<MojoDemuxerStreamAdapter> stream) | 50 DemuxerStreamProviderShim(mojo::DemuxerStreamPtr audio, |
| 42 : stream_(stream.Pass()) {} | 51 mojo::DemuxerStreamPtr video, |
| 52 const base::Closure& demuxer_ready_cb) | |
| 53 : demuxer_ready_cb_(demuxer_ready_cb), | |
| 54 streams_ready_(0), | |
| 55 weak_factory_(this) { | |
| 56 DCHECK(audio || video); | |
| 57 DCHECK(!demuxer_ready_cb_.is_null()); | |
| 58 | |
| 59 if (audio) { | |
| 60 streams_.push_back(new MojoDemuxerStreamAdapter( | |
| 61 audio.Pass(), | |
| 62 base::Bind(&DemuxerStreamProviderShim::OnStreamReady, | |
| 63 weak_factory_.GetWeakPtr()), | |
| 64 DemuxerStream::AUDIO)); | |
| 65 } | |
| 66 | |
| 67 if (video) { | |
| 68 streams_.push_back(new MojoDemuxerStreamAdapter( | |
| 69 video.Pass(), | |
| 70 base::Bind(&DemuxerStreamProviderShim::OnStreamReady, | |
| 71 weak_factory_.GetWeakPtr()), | |
| 72 DemuxerStream::VIDEO)); | |
| 73 } | |
|
xhwang
2014/10/29 00:22:13
see above comments
DaleCurtis
2014/10/30 23:16:56
Moved out.
| |
| 74 } | |
| 43 | 75 |
| 44 ~DemuxerStreamProviderShim() override {} | 76 ~DemuxerStreamProviderShim() override {} |
| 45 | 77 |
| 46 DemuxerStream* GetStream(DemuxerStream::Type type) override { | 78 DemuxerStream* GetStream(DemuxerStream::Type type) override { |
| 47 return type != stream_->type() ? nullptr : stream_.get(); | 79 DCHECK(demuxer_ready_cb_.is_null()); |
| 48 }; | 80 for (auto* stream : streams_) { |
| 81 if (stream->type() == type) | |
| 82 return stream; | |
| 83 } | |
| 84 | |
| 85 return nullptr; | |
| 86 } | |
| 49 | 87 |
| 50 Liveness GetLiveness() const override { | 88 Liveness GetLiveness() const override { |
| 89 // TODO(dalecurtis): What to do here?? Probably we need an API for this. | |
|
xhwang
2014/10/29 00:22:13
See https://code.google.com/p/chromium/issues/deta
DaleCurtis
2014/10/30 23:16:57
Acknowledged.
| |
| 51 return DemuxerStreamProvider::LIVENESS_UNKNOWN; | 90 return DemuxerStreamProvider::LIVENESS_UNKNOWN; |
| 52 } | 91 } |
| 53 | 92 |
| 54 private: | 93 private: |
| 55 scoped_ptr<MojoDemuxerStreamAdapter> stream_; | 94 void OnStreamReady() { |
| 95 if (++streams_ready_ == streams_.size()) { | |
|
xhwang
2014/10/29 00:22:13
I like this!
| |
| 96 demuxer_ready_cb_.Run(); | |
| 97 demuxer_ready_cb_.Reset(); | |
|
xhwang
2014/10/29 00:22:13
base::ResetAndReturn(&demuxer_ready_cb).Run()?
DaleCurtis
2014/10/30 23:16:56
Done.
| |
| 98 } | |
| 99 } | |
| 100 | |
| 101 base::Closure demuxer_ready_cb_; | |
| 102 ScopedVector<MojoDemuxerStreamAdapter> streams_; | |
| 103 size_t streams_ready_; | |
| 104 | |
| 105 base::WeakPtrFactory<DemuxerStreamProviderShim> weak_factory_; | |
| 56 | 106 |
| 57 DISALLOW_COPY_AND_ASSIGN(DemuxerStreamProviderShim); | 107 DISALLOW_COPY_AND_ASSIGN(DemuxerStreamProviderShim); |
| 58 }; | 108 }; |
| 59 | 109 |
| 60 class MojoRendererApplication | 110 class MojoRendererApplication |
| 61 : public mojo::ApplicationDelegate, | 111 : public mojo::ApplicationDelegate, |
| 62 public mojo::InterfaceFactory<mojo::MediaRenderer> { | 112 public mojo::InterfaceFactory<mojo::MediaRenderer> { |
| 63 public: | 113 public: |
| 64 // mojo::ApplicationDelegate implementation. | 114 // mojo::ApplicationDelegate implementation. |
| 65 bool ConfigureIncomingConnection( | 115 bool ConfigureIncomingConnection( |
| (...skipping 28 matching lines...) Expand all Loading... | |
| 94 | 144 |
| 95 scoped_ptr<AudioRenderer> audio_renderer(new AudioRendererImpl( | 145 scoped_ptr<AudioRenderer> audio_renderer(new AudioRendererImpl( |
| 96 task_runner, | 146 task_runner, |
| 97 audio_renderer_sink_.get(), | 147 audio_renderer_sink_.get(), |
| 98 renderer_config->GetAudioDecoders( | 148 renderer_config->GetAudioDecoders( |
| 99 task_runner, | 149 task_runner, |
| 100 base::Bind(&LogMediaSourceError, media_log)).Pass(), | 150 base::Bind(&LogMediaSourceError, media_log)).Pass(), |
| 101 SetDecryptorReadyCB(), | 151 SetDecryptorReadyCB(), |
| 102 renderer_config->GetAudioHardwareConfig(), | 152 renderer_config->GetAudioHardwareConfig(), |
| 103 media_log)); | 153 media_log)); |
| 104 scoped_ptr<VideoRenderer> video_renderer(nullptr); | 154 |
| 155 | |
| 156 // Create our video decoders and renderer. | |
| 157 ScopedVector<VideoDecoder> video_decoders; | |
| 158 video_decoders.push_back(new FFmpegVideoDecoder(task_runner)); | |
| 159 | |
| 160 scoped_ptr<VideoRenderer> video_renderer( | |
| 161 new VideoRendererImpl(task_runner, | |
| 162 video_decoders.Pass(), | |
| 163 SetDecryptorReadyCB(), | |
| 164 base::Bind(&PaintNothingDropEverything), | |
| 165 true, | |
| 166 media_log)); | |
| 105 | 167 |
| 106 // Create renderer. | 168 // Create renderer. |
| 107 renderer_.reset(new RendererImpl( | 169 renderer_.reset(new RendererImpl( |
| 108 task_runner, audio_renderer.Pass(), video_renderer.Pass())); | 170 task_runner, audio_renderer.Pass(), video_renderer.Pass())); |
| 109 } | 171 } |
| 110 | 172 |
| 111 MojoRendererService::~MojoRendererService() { | 173 MojoRendererService::~MojoRendererService() { |
| 112 } | 174 } |
| 113 | 175 |
| 114 void MojoRendererService::Initialize(mojo::DemuxerStreamPtr audio, | 176 void MojoRendererService::Initialize(mojo::DemuxerStreamPtr audio, |
| 115 mojo::DemuxerStreamPtr video, | 177 mojo::DemuxerStreamPtr video, |
| 116 const mojo::Closure& callback) { | 178 const mojo::Closure& callback) { |
| 117 DVLOG(1) << __FUNCTION__; | 179 DVLOG(1) << __FUNCTION__; |
| 118 DCHECK_EQ(state_, STATE_UNINITIALIZED); | 180 DCHECK_EQ(state_, STATE_UNINITIALIZED); |
| 119 DCHECK(client()); | 181 DCHECK(client()); |
| 120 | 182 |
| 121 state_ = STATE_INITIALIZING; | 183 state_ = STATE_INITIALIZING; |
| 122 stream_provider_.reset(new DemuxerStreamProviderShim( | 184 stream_provider_.reset(new DemuxerStreamProviderShim( |
| 123 make_scoped_ptr(new MojoDemuxerStreamAdapter( | 185 audio.Pass(), |
| 124 audio.Pass(), | 186 video.Pass(), |
| 125 base::Bind(&MojoRendererService::OnStreamReady, | 187 base::Bind(&MojoRendererService::OnStreamReady, weak_this_, callback))); |
| 126 weak_this_, | |
| 127 callback))).Pass())); | |
| 128 } | 188 } |
| 129 | 189 |
| 130 void MojoRendererService::Flush(const mojo::Closure& callback) { | 190 void MojoRendererService::Flush(const mojo::Closure& callback) { |
| 131 DVLOG(2) << __FUNCTION__; | 191 DVLOG(2) << __FUNCTION__; |
| 132 DCHECK_EQ(state_, STATE_PLAYING); | 192 DCHECK_EQ(state_, STATE_PLAYING); |
| 133 | 193 |
| 134 state_ = STATE_FLUSHING; | 194 state_ = STATE_FLUSHING; |
| 135 renderer_->Flush(base::Bind(&MojoTrampoline, callback)); | 195 renderer_->Flush(base::Bind(&MojoTrampoline, callback)); |
| 136 } | 196 } |
| 137 | 197 |
| (...skipping 82 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 220 state_ = STATE_ERROR; | 280 state_ = STATE_ERROR; |
| 221 client()->OnError(); | 281 client()->OnError(); |
| 222 } | 282 } |
| 223 | 283 |
| 224 } // namespace media | 284 } // namespace media |
| 225 | 285 |
| 226 MojoResult MojoMain(MojoHandle shell_handle) { | 286 MojoResult MojoMain(MojoHandle shell_handle) { |
| 227 mojo::ApplicationRunnerChromium runner(new media::MojoRendererApplication); | 287 mojo::ApplicationRunnerChromium runner(new media::MojoRendererApplication); |
| 228 return runner.Run(shell_handle); | 288 return runner.Run(shell_handle); |
| 229 } | 289 } |
| OLD | NEW |