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

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: fix bugs Created 7 years, 7 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 // A call to |Read| operation means that |MediaSourceDelegate| is done with
112 // the previous buffer.
113 if (!pending_done_cb_.is_null())
114 base::ResetAndReturn(&pending_done_cb_).Run();
115 read_cb_ = media::BindToLoop(base::MessageLoopProxy::current(), read_cb);
116 RunReadCallback_Locked();
acolwell GONE FROM CHROMIUM 2013/05/20 21:23:22 Either put a DCHECK(!is_destroyed_) at the top or
wonsik 2013/05/21 10:09:29 Done.
117 }
118
119 void RTCDemuxerStream::Destroy() {
120 base::AutoLock lock(lock_);
121 DCHECK(!is_destroyed_);
122 is_destroyed_ = true;
123 if (!read_cb_.is_null())
124 base::ResetAndReturn(&read_cb_).Run(DemuxerStream::kAborted, NULL);
125 }
acolwell GONE FROM CHROMIUM 2013/05/20 21:23:22 Should pending_done_cb_ and/or buffer_queue_ be cl
wonsik 2013/05/21 10:09:29 Done.
126
127 void RTCDemuxerStream::RunReadCallback_Locked() {
128 if (read_cb_.is_null() || buffer_queue_.empty())
129 return;
130
131 BufferEntry& front = buffer_queue_.front();
132 if (!front.new_size.IsEmpty()) {
133 // No VideoFrame actually reaches GL renderer in Google TV case. We just
134 // make coded_size == visible_rect == natural_size here.
135 video_decoder_config_.Initialize(media::kCodecVP8,
136 media::VP8PROFILE_MAIN,
137 media::VideoFrame::NATIVE_TEXTURE,
138 front.new_size,
139 gfx::Rect(front.new_size),
140 front.new_size,
141 NULL,
142 0,
143 false,
144 false);
145 base::ResetAndReturn(&read_cb_).Run(DemuxerStream::kConfigChanged, NULL);
146 front.new_size.SetSize(0, 0);
147 return;
148 }
149 DCHECK(pending_done_cb_.is_null());
150 pending_done_cb_ = front.done_cb;
151 base::ResetAndReturn(&read_cb_).Run(DemuxerStream::kOk, front.decoder_buffer);
152 buffer_queue_.pop();
153 }
154
155 // RTCVideoDecoderFactoryTv ----------------------------------------------------
156
157 RTCVideoDecoderFactoryTv::RTCVideoDecoderFactoryTv() : is_acquired_(false) {}
158 RTCVideoDecoderFactoryTv::~RTCVideoDecoderFactoryTv() {}
159
160 webrtc::VideoDecoder* RTCVideoDecoderFactoryTv::CreateVideoDecoder(
161 webrtc::VideoCodecType type) {
162 base::AutoLock lock(lock_);
163 // One decoder at a time!
164 if (decoder_)
165 return NULL;
166 // Only VP8 is supported --- returning NULL will make WebRTC fall back to SW
167 // decoder.
168 if (type != webrtc::kVideoCodecVP8)
169 return NULL;
170 decoder_.reset(new RTCVideoDecoderBridgeTv(this));
171 return decoder_.get();
172 }
173
174 void RTCVideoDecoderFactoryTv::DestroyVideoDecoder(
175 webrtc::VideoDecoder* decoder) {
176 base::AutoLock lock(lock_);
177 DCHECK(decoder_.get() == decoder);
178 decoder_.reset();
179 }
180
181 bool RTCVideoDecoderFactoryTv::AcquireDemuxer() {
182 base::AutoLock lock(lock_);
183 if (is_acquired_)
184 return false;
185 is_acquired_ = true;
186 return true;
187 }
188
189 void RTCVideoDecoderFactoryTv::ReleaseDemuxer() {
190 base::AutoLock lock(lock_);
191 DCHECK(is_acquired_);
192 is_acquired_ = false;
193 // Clean up internal state as a demuxer.
194 init_cb_.Reset();
195 if (stream_) {
196 stream_->Destroy();
197 stream_.reset();
198 }
199 }
200
201 void RTCVideoDecoderFactoryTv::Initialize(media::DemuxerHost*,
202 const media::PipelineStatusCB& cb) {
203 base::AutoLock lock(lock_);
204 init_cb_ = cb;
205 if (stream_)
206 base::ResetAndReturn(&init_cb_).Run(media::PIPELINE_OK);
207 }
208
209 DemuxerStream* RTCVideoDecoderFactoryTv::GetStream(DemuxerStream::Type type) {
210 base::AutoLock lock(lock_);
211 if (type == DemuxerStream::VIDEO)
212 return stream_.get();
213 return NULL;
214 }
215
216 base::TimeDelta RTCVideoDecoderFactoryTv::GetStartTime() const {
217 return base::TimeDelta();
218 }
219
220 void RTCVideoDecoderFactoryTv::InitializeStream(const gfx::Size& size) {
221 base::AutoLock lock(lock_);
222 DCHECK(!stream_);
223 stream_.reset(new RTCDemuxerStream(size));
224 if (!init_cb_.is_null())
225 base::ResetAndReturn(&init_cb_).Run(media::PIPELINE_OK);
226 }
227
228 void RTCVideoDecoderFactoryTv::QueueBuffer(
229 scoped_refptr<media::DecoderBuffer> buffer,
230 const base::Closure& done_cb,
231 const gfx::Size& new_size) {
232 base::AutoLock lock(lock_);
233 DCHECK(stream_);
234 stream_->QueueBuffer(buffer, done_cb, new_size);
235 }
236
237 } // namespace content
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698