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
3 // found in the LICENSE file.
5 #include "base/synchronization/waitable_event.h"
6 #include "base/task_runner_util.h"
7 #include "base/threading/thread.h"
8 #include "content/renderer/media/rtc_video_decoder_factory_tv.h"
9 #include "media/base/decoder_buffer.h"
10 #include "media/base/video_decoder_config.h"
11 #include "testing/gmock/include/gmock/gmock.h"
12 #include "testing/gtest/include/gtest/gtest.h"
13 #include "third_party/webrtc/modules/video_coding/codecs/interface/mock/mock_video_codec_interface.h"
14 #include "ui/gfx/rect.h"
17 using ::testing::Return;
21 class RTCVideoDecoderFactoryTvTest : public ::testing::Test {
23 RTCVideoDecoderFactoryTvTest()
24 : factory_(new RTCVideoDecoderFactoryTv),
26 is_demuxer_acquired_(false),
29 input_image_(&data_, sizeof(data_), sizeof(data_)),
31 read_event_(false, false),
32 decoder_thread_("Test decoder thread"),
33 decoder_thread_event_(false, false) {
34 memset(&codec_, 0, sizeof(codec_));
35 message_loop_proxy_ = base::MessageLoopProxy::current();
36 input_image_._frameType = webrtc::kKeyFrame;
37 input_image_._encodedWidth = size_.width();
38 input_image_._encodedHeight = size_.height();
39 input_image_._completeFrame = true;
40 decoder_thread_.Start();
43 virtual ~RTCVideoDecoderFactoryTvTest() {
44 if (is_demuxer_acquired_) {
45 factory_->ReleaseDemuxer();
46 is_demuxer_acquired_ = false;
49 factory_->DestroyVideoDecoder(decoder_);
53 decoder_thread_.Stop();
56 void ReadCallback(media::DemuxerStream::Status status,
57 const scoped_refptr<media::DecoderBuffer>& decoder_buffer) {
59 case media::DemuxerStream::kOk:
60 EXPECT_TRUE(decoder_buffer);
62 case media::DemuxerStream::kConfigChanged:
63 case media::DemuxerStream::kAborted:
64 EXPECT_FALSE(decoder_buffer);
67 last_decoder_buffer_ = decoder_buffer;
71 void ExpectEqualsAndSignal(int32_t expected, int32_t actual) {
72 EXPECT_EQ(expected, actual);
73 decoder_thread_event_.Signal();
76 void ExpectNotEqualsAndSignal(int32_t unexpected, int32_t actual) {
77 EXPECT_NE(unexpected, actual);
78 decoder_thread_event_.Signal();
82 base::Callback<void(int32_t)> BindExpectEquals(int32_t expected) {
83 return base::Bind(&RTCVideoDecoderFactoryTvTest::ExpectEqualsAndSignal,
84 base::Unretained(this),
88 base::Callback<void(int32_t)> BindExpectNotEquals(int32_t unexpected) {
89 return base::Bind(&RTCVideoDecoderFactoryTvTest::ExpectNotEqualsAndSignal,
90 base::Unretained(this),
94 base::Callback<int32_t(void)> BindInitDecode(const webrtc::VideoCodec* codec,
96 return base::Bind(&webrtc::VideoDecoder::InitDecode,
97 base::Unretained(decoder_),
102 base::Callback<int32_t(void)> BindDecode(
103 const webrtc::EncodedImage& input_image,
105 const webrtc::RTPFragmentationHeader* fragmentation,
106 const webrtc::CodecSpecificInfo* info,
107 int64_t render_time_ms) {
108 return base::Bind(&webrtc::VideoDecoder::Decode,
109 base::Unretained(decoder_),
117 void CreateDecoderAndAcquireDemuxer() {
118 decoder_ = factory_->CreateVideoDecoder(webrtc::kVideoCodecVP8);
119 ASSERT_TRUE(decoder_);
120 ASSERT_TRUE(factory_->AcquireDemuxer());
121 is_demuxer_acquired_ = true;
125 codec_.codecType = webrtc::kVideoCodecVP8;
126 codec_.width = size_.width();
127 codec_.height = size_.height();
128 base::PostTaskAndReplyWithResult(decoder_thread_.message_loop_proxy(),
130 BindInitDecode(&codec_, 1),
131 BindExpectEquals(WEBRTC_VIDEO_CODEC_OK));
132 decoder_thread_event_.Wait();
133 base::PostTaskAndReplyWithResult(
134 decoder_thread_.message_loop_proxy(),
136 base::Bind(&webrtc::VideoDecoder::RegisterDecodeCompleteCallback,
137 base::Unretained(decoder_),
138 &decode_complete_callback_),
139 BindExpectEquals(WEBRTC_VIDEO_CODEC_OK));
140 decoder_thread_event_.Wait();
143 void GetVideoStream() {
144 video_stream_ = factory_->GetStream(media::DemuxerStream::VIDEO);
145 ASSERT_TRUE(video_stream_);
146 EXPECT_EQ(media::kCodecVP8, video_stream_->video_decoder_config().codec());
147 EXPECT_EQ(size_, video_stream_->video_decoder_config().coded_size());
148 EXPECT_EQ(gfx::Rect(size_),
149 video_stream_->video_decoder_config().visible_rect());
150 EXPECT_EQ(size_, video_stream_->video_decoder_config().natural_size());
153 void PostDecodeAndWait(int32_t expected,
154 const webrtc::EncodedImage& input_image,
156 const webrtc::RTPFragmentationHeader* fragmentation,
157 const webrtc::CodecSpecificInfo* info,
158 int64_t render_time_ms) {
159 base::PostTaskAndReplyWithResult(
160 decoder_thread_.message_loop_proxy(),
163 input_image, missing_frames, fragmentation, info, render_time_ms),
164 BindExpectEquals(expected));
165 decoder_thread_event_.Wait();
168 RTCVideoDecoderFactoryTv* factory_;
169 webrtc::VideoDecoder* decoder_;
170 bool is_demuxer_acquired_;
171 base::MessageLoopProxy* message_loop_proxy_;
172 media::DemuxerStream* video_stream_;
173 webrtc::VideoCodec codec_;
175 webrtc::EncodedImage input_image_;
177 webrtc::MockDecodedImageCallback decode_complete_callback_;
178 base::WaitableEvent read_event_;
179 base::Thread decoder_thread_;
180 base::WaitableEvent decoder_thread_event_;
181 scoped_refptr<media::DecoderBuffer> last_decoder_buffer_;
184 TEST_F(RTCVideoDecoderFactoryTvTest, CreateAndDestroyDecoder) {
185 // Only VP8 decoder is supported.
186 ASSERT_FALSE(factory_->CreateVideoDecoder(webrtc::kVideoCodecI420));
187 decoder_ = factory_->CreateVideoDecoder(webrtc::kVideoCodecVP8);
188 ASSERT_TRUE(decoder_);
189 // Only one decoder at a time will be created.
190 ASSERT_FALSE(factory_->CreateVideoDecoder(webrtc::kVideoCodecVP8));
191 factory_->DestroyVideoDecoder(decoder_);
194 TEST_F(RTCVideoDecoderFactoryTvTest, AcquireDemuxerAfterCreateDecoder) {
195 decoder_ = factory_->CreateVideoDecoder(webrtc::kVideoCodecVP8);
196 ASSERT_TRUE(decoder_);
197 ASSERT_TRUE(factory_->AcquireDemuxer());
198 is_demuxer_acquired_ = true;
199 // Demuxer can be acquired only once.
200 ASSERT_FALSE(factory_->AcquireDemuxer());
203 TEST_F(RTCVideoDecoderFactoryTvTest, AcquireDemuxerBeforeCreateDecoder) {
204 ASSERT_TRUE(factory_->AcquireDemuxer());
205 is_demuxer_acquired_ = true;
206 decoder_ = factory_->CreateVideoDecoder(webrtc::kVideoCodecVP8);
207 ASSERT_TRUE(decoder_);
210 TEST_F(RTCVideoDecoderFactoryTvTest, InitDecodeReturnsErrorOnNonVP8Codec) {
211 CreateDecoderAndAcquireDemuxer();
212 codec_.codecType = webrtc::kVideoCodecI420;
213 base::PostTaskAndReplyWithResult(decoder_thread_.message_loop_proxy(),
215 BindInitDecode(&codec_, 1),
216 BindExpectNotEquals(WEBRTC_VIDEO_CODEC_OK));
217 decoder_thread_event_.Wait();
220 TEST_F(RTCVideoDecoderFactoryTvTest, InitDecodeReturnsErrorOnFeedbackMode) {
221 CreateDecoderAndAcquireDemuxer();
222 codec_.codecType = webrtc::kVideoCodecVP8;
223 codec_.codecSpecific.VP8.feedbackModeOn = true;
224 base::PostTaskAndReplyWithResult(decoder_thread_.message_loop_proxy(),
226 BindInitDecode(&codec_, 1),
227 BindExpectNotEquals(WEBRTC_VIDEO_CODEC_OK));
228 decoder_thread_event_.Wait();
231 TEST_F(RTCVideoDecoderFactoryTvTest, DecodeReturnsErrorBeforeInitDecode) {
232 CreateDecoderAndAcquireDemuxer();
234 WEBRTC_VIDEO_CODEC_UNINITIALIZED, input_image_, false, NULL, NULL, 0);
237 TEST_F(RTCVideoDecoderFactoryTvTest, DecodeReturnsErrorOnDamagedBitstream) {
238 CreateDecoderAndAcquireDemuxer();
240 input_image_._completeFrame = false;
242 WEBRTC_VIDEO_CODEC_ERROR, input_image_, false, NULL, NULL, 0);
245 TEST_F(RTCVideoDecoderFactoryTvTest, DecodeReturnsErrorOnMissingFrames) {
246 CreateDecoderAndAcquireDemuxer();
249 WEBRTC_VIDEO_CODEC_ERROR, input_image_, true, NULL, NULL, 0);
252 TEST_F(RTCVideoDecoderFactoryTvTest, GetNonVideoStreamFails) {
253 CreateDecoderAndAcquireDemuxer();
255 EXPECT_FALSE(factory_->GetStream(media::DemuxerStream::AUDIO));
256 EXPECT_FALSE(factory_->GetStream(media::DemuxerStream::UNKNOWN));
259 TEST_F(RTCVideoDecoderFactoryTvTest, GetVideoStreamSucceeds) {
260 CreateDecoderAndAcquireDemuxer();
265 TEST_F(RTCVideoDecoderFactoryTvTest, DecodeReturnsErrorOnNonKeyFrameAtFirst) {
266 CreateDecoderAndAcquireDemuxer();
269 input_image_._frameType = webrtc::kDeltaFrame;
271 WEBRTC_VIDEO_CODEC_ERROR, input_image_, false, NULL, NULL, 0);
274 TEST_F(RTCVideoDecoderFactoryTvTest, DecodeUpdatesVideoSizeOnKeyFrame) {
275 CreateDecoderAndAcquireDemuxer();
278 gfx::Size new_size(320, 240);
279 input_image_._encodedWidth = new_size.width();
280 input_image_._encodedHeight = new_size.height();
281 PostDecodeAndWait(WEBRTC_VIDEO_CODEC_OK, input_image_, false, NULL, NULL, 0);
282 EXPECT_EQ(new_size, video_stream_->video_decoder_config().coded_size());
283 EXPECT_EQ(gfx::Rect(new_size),
284 video_stream_->video_decoder_config().visible_rect());
285 EXPECT_EQ(new_size, video_stream_->video_decoder_config().natural_size());
288 TEST_F(RTCVideoDecoderFactoryTvTest, DecodeAdjustsTimestampFromZero) {
289 CreateDecoderAndAcquireDemuxer();
293 WEBRTC_VIDEO_CODEC_OK, input_image_, false, NULL, NULL, 10000);
294 video_stream_->Read(base::Bind(&RTCVideoDecoderFactoryTvTest::ReadCallback,
295 base::Unretained(this)));
297 EXPECT_EQ(base::TimeDelta::FromMilliseconds(0),
298 last_decoder_buffer_->GetTimestamp());
300 WEBRTC_VIDEO_CODEC_OK, input_image_, false, NULL, NULL, 10033);
301 video_stream_->Read(base::Bind(&RTCVideoDecoderFactoryTvTest::ReadCallback,
302 base::Unretained(this)));
304 EXPECT_EQ(base::TimeDelta::FromMilliseconds(33),
305 last_decoder_buffer_->GetTimestamp());
308 TEST_F(RTCVideoDecoderFactoryTvTest, DecodePassesDataCorrectly) {
309 CreateDecoderAndAcquireDemuxer();
312 video_stream_->Read(base::Bind(&RTCVideoDecoderFactoryTvTest::ReadCallback,
313 base::Unretained(this)));
314 PostDecodeAndWait(WEBRTC_VIDEO_CODEC_OK, input_image_, false, NULL, NULL, 0);
316 EXPECT_EQ(static_cast<int>(sizeof(data_)),
317 last_decoder_buffer_->GetDataSize());
318 EXPECT_EQ(data_, last_decoder_buffer_->GetData()[0]);
321 TEST_F(RTCVideoDecoderFactoryTvTest, NextReadTriggersDecodeCompleteCallback) {
322 EXPECT_CALL(decode_complete_callback_, Decoded(_))
323 .WillOnce(Return(WEBRTC_VIDEO_CODEC_OK));
325 CreateDecoderAndAcquireDemuxer();
328 video_stream_->Read(base::Bind(&RTCVideoDecoderFactoryTvTest::ReadCallback,
329 base::Unretained(this)));
330 PostDecodeAndWait(WEBRTC_VIDEO_CODEC_OK, input_image_, false, NULL, NULL, 0);
332 video_stream_->Read(base::Bind(&RTCVideoDecoderFactoryTvTest::ReadCallback,
333 base::Unretained(this)));
336 TEST_F(RTCVideoDecoderFactoryTvTest, ResetReturnsOk) {
337 CreateDecoderAndAcquireDemuxer();
339 EXPECT_EQ(WEBRTC_VIDEO_CODEC_OK, decoder_->Reset());
342 TEST_F(RTCVideoDecoderFactoryTvTest, ReleaseReturnsOk) {
343 CreateDecoderAndAcquireDemuxer();
345 EXPECT_EQ(WEBRTC_VIDEO_CODEC_OK, decoder_->Release());