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

Side by Side Diff: content/renderer/media/rtc_video_decoder_factory_tv.cc

Issue 14247018: Implement WebRTC in Chrome for TV (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: build fix Created 7 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
OLDNEW
(Empty)
1 // Copyright (c) 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "content/renderer/media/rtc_video_decoder_factory_tv.h"
6
7 #include "base/callback_helpers.h"
8 #include "content/renderer/media/rtc_video_decoder_bridge_tv.h"
9 #include "media/base/audio_decoder_config.h"
10 #include "media/base/bind_to_loop.h"
11 #include "media/base/decoder_buffer.h"
12 #include "media/base/video_decoder_config.h"
13 #include "third_party/libjingle/source/talk/base/ratetracker.h"
14
15 using media::DemuxerStream;
16
17 namespace content {
18
19 // RTCDemuxerStream ------------------------------------------------------------
20
21 class RTCDemuxerStream : public DemuxerStream {
22 public:
23 explicit RTCDemuxerStream(const gfx::Size& size);
24 virtual ~RTCDemuxerStream();
25
26 // DemuxerStream implementation.
27 virtual void Read(const ReadCB& read_cb) OVERRIDE;
28 virtual const media::AudioDecoderConfig& audio_decoder_config() OVERRIDE;
29 virtual const media::VideoDecoderConfig& video_decoder_config() OVERRIDE;
30 virtual Type type() OVERRIDE;
31 virtual void EnableBitstreamConverter() OVERRIDE;
32
33 void QueueBuffer(scoped_refptr<media::DecoderBuffer> buffer,
34 const base::Closure& done_cb,
35 const gfx::Size& new_size);
36 void Destroy();
37
38 private:
39 struct BufferEntry {
40 BufferEntry(const scoped_refptr<media::DecoderBuffer>& decoder_buffer_param,
41 const base::Closure& done_cb_param,
42 const gfx::Size& new_size_param)
43 : decoder_buffer(decoder_buffer_param),
44 done_cb(done_cb_param),
45 new_size(new_size_param) {}
46
47 scoped_refptr<media::DecoderBuffer> decoder_buffer;
48 base::Closure done_cb;
49 // When |!new_size.isEmpty()|, it means that config change with new size
50 // |new_size| happened.
51 gfx::Size new_size;
52 };
53
54 void RunReadCallback_Locked();
55
56 base::Lock lock_;
57 bool is_destroyed_;
58 std::queue<BufferEntry> buffer_queue_;
59 ReadCB read_cb_;
60 base::Closure pending_done_cb_;
61
62 media::AudioDecoderConfig dummy_audio_decoder_config_;
63 media::VideoDecoderConfig video_decoder_config_;
64 talk_base::RateTracker frame_rate_tracker_;
65 };
66
67 RTCDemuxerStream::RTCDemuxerStream(const gfx::Size& size)
68 : is_destroyed_(false),
69 video_decoder_config_(media::kCodecVP8,
70 media::VP8PROFILE_MAIN,
71 media::VideoFrame::NATIVE_TEXTURE,
72 size,
73 gfx::Rect(size),
74 size,
75 NULL,
76 0,
77 false) {}
78
79 RTCDemuxerStream::~RTCDemuxerStream() { DCHECK(is_destroyed_); }
80
81 const media::AudioDecoderConfig& RTCDemuxerStream::audio_decoder_config() {
82 NOTIMPLEMENTED() << "Does not support audio.";
83 return dummy_audio_decoder_config_;
84 }
85
86 const media::VideoDecoderConfig& RTCDemuxerStream::video_decoder_config() {
87 base::AutoLock lock(lock_);
88 return video_decoder_config_;
89 }
90
91 DemuxerStream::Type RTCDemuxerStream::type() { return DemuxerStream::VIDEO; }
92
93 void RTCDemuxerStream::EnableBitstreamConverter() { NOTREACHED(); }
94
95 void RTCDemuxerStream::QueueBuffer(scoped_refptr<media::DecoderBuffer> buffer,
96 const base::Closure& done_cb,
97 const gfx::Size& new_size) {
98 base::AutoLock lock(lock_);
99 if (is_destroyed_)
100 return;
101 buffer_queue_.push(BufferEntry(buffer, done_cb, new_size));
102 if (buffer)
103 frame_rate_tracker_.Update(1);
104 DVLOG(1) << "frame rate received : " << frame_rate_tracker_.units_second();
105 RunReadCallback_Locked();
106 }
107
108 void RTCDemuxerStream::Read(const ReadCB& read_cb) {
109 base::AutoLock lock(lock_);
110 DCHECK(read_cb_.is_null());
111 if (is_destroyed_) {
112 media::BindToLoop(base::MessageLoopProxy::current(), read_cb)
113 .Run(DemuxerStream::kAborted, NULL);
114 return;
115 }
116 // A call to |Read| operation means that |MediaSourceDelegate| is done with
117 // the previous buffer.
118 if (!pending_done_cb_.is_null())
119 base::ResetAndReturn(&pending_done_cb_).Run();
120 read_cb_ = media::BindToLoop(base::MessageLoopProxy::current(), read_cb);
121 RunReadCallback_Locked();
122 }
123
124 void RTCDemuxerStream::Destroy() {
125 base::AutoLock lock(lock_);
126 DCHECK(!is_destroyed_);
127 is_destroyed_ = true;
128 if (!read_cb_.is_null())
129 base::ResetAndReturn(&read_cb_).Run(DemuxerStream::kAborted, NULL);
130 pending_done_cb_.Reset();
131 while (!buffer_queue_.empty())
132 buffer_queue_.pop();
133 }
134
135 void RTCDemuxerStream::RunReadCallback_Locked() {
136 if (read_cb_.is_null() || buffer_queue_.empty())
137 return;
138
139 BufferEntry& front = buffer_queue_.front();
140 if (!front.new_size.IsEmpty()) {
141 // No VideoFrame actually reaches GL renderer in Google TV case. We just
142 // make coded_size == visible_rect == natural_size here.
143 video_decoder_config_.Initialize(media::kCodecVP8,
144 media::VP8PROFILE_MAIN,
145 media::VideoFrame::NATIVE_TEXTURE,
146 front.new_size,
147 gfx::Rect(front.new_size),
148 front.new_size,
149 NULL,
150 0,
151 false,
152 false);
153 base::ResetAndReturn(&read_cb_).Run(DemuxerStream::kConfigChanged, NULL);
154 front.new_size.SetSize(0, 0);
155 return;
156 }
157 DCHECK(pending_done_cb_.is_null());
158 pending_done_cb_ = front.done_cb;
159 base::ResetAndReturn(&read_cb_).Run(DemuxerStream::kOk, front.decoder_buffer);
160 buffer_queue_.pop();
161 }
162
163 // RTCVideoDecoderFactoryTv ----------------------------------------------------
164
165 RTCVideoDecoderFactoryTv::RTCVideoDecoderFactoryTv() : is_acquired_(false) {}
166 RTCVideoDecoderFactoryTv::~RTCVideoDecoderFactoryTv() {}
167
168 webrtc::VideoDecoder* RTCVideoDecoderFactoryTv::CreateVideoDecoder(
169 webrtc::VideoCodecType type) {
170 base::AutoLock lock(lock_);
171 // One decoder at a time!
172 if (decoder_)
173 return NULL;
174 // Only VP8 is supported --- returning NULL will make WebRTC fall back to SW
175 // decoder.
176 if (type != webrtc::kVideoCodecVP8)
177 return NULL;
178 decoder_.reset(new RTCVideoDecoderBridgeTv(this));
179 return decoder_.get();
180 }
181
182 void RTCVideoDecoderFactoryTv::DestroyVideoDecoder(
183 webrtc::VideoDecoder* decoder) {
184 base::AutoLock lock(lock_);
185 DCHECK(decoder_.get() == decoder);
186 decoder_.reset();
187 }
188
189 bool RTCVideoDecoderFactoryTv::AcquireDemuxer() {
190 base::AutoLock lock(lock_);
191 if (is_acquired_)
192 return false;
193 is_acquired_ = true;
194 return true;
195 }
196
197 void RTCVideoDecoderFactoryTv::ReleaseDemuxer() {
198 base::AutoLock lock(lock_);
199 DCHECK(is_acquired_);
200 is_acquired_ = false;
201 // Clean up internal state as a demuxer.
202 init_cb_.Reset();
203 if (stream_) {
204 stream_->Destroy();
205 stream_.reset();
206 }
207 }
208
209 void RTCVideoDecoderFactoryTv::Initialize(media::DemuxerHost*,
210 const media::PipelineStatusCB& cb) {
211 base::AutoLock lock(lock_);
212 init_cb_ = media::BindToLoop(base::MessageLoopProxy::current(), cb);
213 if (stream_)
214 base::ResetAndReturn(&init_cb_).Run(media::PIPELINE_OK);
215 }
216
217 DemuxerStream* RTCVideoDecoderFactoryTv::GetStream(DemuxerStream::Type type) {
218 base::AutoLock lock(lock_);
219 if (type == DemuxerStream::VIDEO)
220 return stream_.get();
221 return NULL;
222 }
223
224 base::TimeDelta RTCVideoDecoderFactoryTv::GetStartTime() const {
225 return base::TimeDelta();
226 }
227
228 void RTCVideoDecoderFactoryTv::InitializeStream(const gfx::Size& size) {
229 base::AutoLock lock(lock_);
230 DCHECK(!stream_);
231 stream_.reset(new RTCDemuxerStream(size));
232 if (!init_cb_.is_null())
233 base::ResetAndReturn(&init_cb_).Run(media::PIPELINE_OK);
234 }
235
236 void RTCVideoDecoderFactoryTv::QueueBuffer(
237 scoped_refptr<media::DecoderBuffer> buffer,
238 const base::Closure& done_cb,
239 const gfx::Size& new_size) {
240 base::AutoLock lock(lock_);
241 DCHECK(stream_);
242 stream_->QueueBuffer(buffer, done_cb, new_size);
243 }
244
245 } // namespace content
OLDNEW
« no previous file with comments | « content/renderer/media/rtc_video_decoder_factory_tv.h ('k') | content/renderer/media/rtc_video_decoder_factory_tv_unittest.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698