blob: 16991d21b7cea8c330efddf59d1fb13c4a94d798 [file] [log] [blame]
Peter Boström7252a2b2015-05-18 19:42:03 +02001/*
2 * Copyright (c) 2015 The WebRTC project authors. All Rights Reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
guidou716d7ac2017-04-03 09:15:52 -070011#include "webrtc/video_decoder.h"
12
Per327d8ba2015-11-10 14:00:27 +010013#include "webrtc/base/checks.h"
magjedf6acc2a2016-11-22 01:43:03 -080014#include "webrtc/media/engine/videodecodersoftwarefallbackwrapper.h"
Henrik Kjellander2557b862015-11-18 22:00:21 +010015#include "webrtc/modules/video_coding/include/video_error_codes.h"
kwibergac9f8762016-09-30 22:29:43 -070016#include "webrtc/test/gtest.h"
Peter Boström7252a2b2015-05-18 19:42:03 +020017
18namespace webrtc {
19
20class VideoDecoderSoftwareFallbackWrapperTest : public ::testing::Test {
21 protected:
22 VideoDecoderSoftwareFallbackWrapperTest()
23 : fallback_wrapper_(kVideoCodecVP8, &fake_decoder_) {}
24
25 class CountingFakeDecoder : public VideoDecoder {
26 public:
27 int32_t InitDecode(const VideoCodec* codec_settings,
28 int32_t number_of_cores) override {
29 ++init_decode_count_;
Peter Boströmf812e452017-02-13 17:11:08 -050030 return init_decode_return_code_;
Peter Boström7252a2b2015-05-18 19:42:03 +020031 }
32
33 int32_t Decode(const EncodedImage& input_image,
34 bool missing_frames,
35 const RTPFragmentationHeader* fragmentation,
36 const CodecSpecificInfo* codec_specific_info,
37 int64_t render_time_ms) override {
38 ++decode_count_;
39 return decode_return_code_;
40 }
41
42 int32_t RegisterDecodeCompleteCallback(
43 DecodedImageCallback* callback) override {
44 decode_complete_callback_ = callback;
45 return WEBRTC_VIDEO_CODEC_OK;
46 }
47
48 int32_t Release() override {
49 ++release_count_;
50 return WEBRTC_VIDEO_CODEC_OK;
51 }
52
Peter Boströmb7d9a972015-12-18 16:01:11 +010053 const char* ImplementationName() const override {
54 return "fake-decoder";
55 }
56
Peter Boström7252a2b2015-05-18 19:42:03 +020057 int init_decode_count_ = 0;
58 int decode_count_ = 0;
Peter Boströmf812e452017-02-13 17:11:08 -050059 int32_t init_decode_return_code_ = WEBRTC_VIDEO_CODEC_OK;
Peter Boström7252a2b2015-05-18 19:42:03 +020060 int32_t decode_return_code_ = WEBRTC_VIDEO_CODEC_OK;
61 DecodedImageCallback* decode_complete_callback_ = nullptr;
62 int release_count_ = 0;
63 int reset_count_ = 0;
64 };
65 CountingFakeDecoder fake_decoder_;
66 VideoDecoderSoftwareFallbackWrapper fallback_wrapper_;
67};
68
69TEST_F(VideoDecoderSoftwareFallbackWrapperTest, InitializesDecoder) {
70 VideoCodec codec = {};
71 fallback_wrapper_.InitDecode(&codec, 2);
72 EXPECT_EQ(1, fake_decoder_.init_decode_count_);
Peter Boströmf812e452017-02-13 17:11:08 -050073
74 EncodedImage encoded_image;
75 encoded_image._frameType = kVideoFrameKey;
76 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
77 EXPECT_EQ(1, fake_decoder_.init_decode_count_)
78 << "Initialized decoder should not be reinitialized.";
79 EXPECT_EQ(1, fake_decoder_.decode_count_);
80}
81
82TEST_F(VideoDecoderSoftwareFallbackWrapperTest,
83 UsesFallbackDecoderAfterOnInitDecodeFailure) {
84 VideoCodec codec = {};
85 fake_decoder_.init_decode_return_code_ = WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE;
86 fallback_wrapper_.InitDecode(&codec, 2);
87 EXPECT_EQ(1, fake_decoder_.init_decode_count_);
88
89 EncodedImage encoded_image;
90 encoded_image._frameType = kVideoFrameKey;
91 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
92 EXPECT_EQ(2, fake_decoder_.init_decode_count_)
93 << "Should have attempted reinitializing the fallback decoder on "
94 "keyframe.";
95 // Unfortunately faking a VP8 frame is hard. Rely on no Decode -> using SW
96 // decoder.
97 EXPECT_EQ(0, fake_decoder_.decode_count_)
98 << "Decoder used even though no InitDecode had succeeded.";
Peter Boström7252a2b2015-05-18 19:42:03 +020099}
100
101TEST_F(VideoDecoderSoftwareFallbackWrapperTest,
102 CanRecoverFromSoftwareFallback) {
103 VideoCodec codec = {};
104 fallback_wrapper_.InitDecode(&codec, 2);
105 // Unfortunately faking a VP8 frame is hard. Rely on no Decode -> using SW
106 // decoder.
107 fake_decoder_.decode_return_code_ = WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE;
108 EncodedImage encoded_image;
109 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
110 EXPECT_EQ(1, fake_decoder_.decode_count_);
111
112 // Fail -> fake_decoder shouldn't be used anymore.
113 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
114 EXPECT_EQ(1, fake_decoder_.decode_count_)
115 << "Decoder used even though fallback should be active.";
116
117 // Should be able to recover on a keyframe.
Peter Boström49e196a2015-10-23 15:58:18 +0200118 encoded_image._frameType = kVideoFrameKey;
Peter Boström7252a2b2015-05-18 19:42:03 +0200119 fake_decoder_.decode_return_code_ = WEBRTC_VIDEO_CODEC_OK;
120 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
121 EXPECT_EQ(2, fake_decoder_.decode_count_)
122 << "Wrapper did not try to decode a keyframe using registered decoder.";
123
Peter Boström49e196a2015-10-23 15:58:18 +0200124 encoded_image._frameType = kVideoFrameDelta;
Peter Boström7252a2b2015-05-18 19:42:03 +0200125 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
126 EXPECT_EQ(3, fake_decoder_.decode_count_)
127 << "Decoder not used on future delta frames.";
128}
129
130TEST_F(VideoDecoderSoftwareFallbackWrapperTest, DoesNotFallbackOnEveryError) {
131 VideoCodec codec = {};
132 fallback_wrapper_.InitDecode(&codec, 2);
133 fake_decoder_.decode_return_code_ = WEBRTC_VIDEO_CODEC_ERROR;
134 EncodedImage encoded_image;
135 EXPECT_EQ(
136 fake_decoder_.decode_return_code_,
137 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1));
138 EXPECT_EQ(1, fake_decoder_.decode_count_);
139
140 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
141 EXPECT_EQ(2, fake_decoder_.decode_count_)
142 << "Decoder should be active even though previous decode failed.";
143}
144
145TEST_F(VideoDecoderSoftwareFallbackWrapperTest, ForwardsReleaseCall) {
146 VideoCodec codec = {};
147 fallback_wrapper_.InitDecode(&codec, 2);
148 fallback_wrapper_.Release();
149 EXPECT_EQ(1, fake_decoder_.release_count_);
150
151 fake_decoder_.decode_return_code_ = WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE;
152 EncodedImage encoded_image;
153 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
154 EXPECT_EQ(1, fake_decoder_.release_count_)
155 << "Decoder should not be released during fallback.";
156 fallback_wrapper_.Release();
157 EXPECT_EQ(2, fake_decoder_.release_count_);
158}
159
Peter Boström7252a2b2015-05-18 19:42:03 +0200160// TODO(pbos): Fake a VP8 frame well enough to actually receive a callback from
Peter Boströmb7d9a972015-12-18 16:01:11 +0100161// the software decoder.
Peter Boström7252a2b2015-05-18 19:42:03 +0200162TEST_F(VideoDecoderSoftwareFallbackWrapperTest,
163 ForwardsRegisterDecodeCompleteCallback) {
164 class FakeDecodedImageCallback : public DecodedImageCallback {
Miguel Casas-Sanchez47650702015-05-29 17:21:40 -0700165 int32_t Decoded(VideoFrame& decodedImage) override { return 0; }
Per327d8ba2015-11-10 14:00:27 +0100166 int32_t Decoded(
167 webrtc::VideoFrame& decodedImage, int64_t decode_time_ms) override {
168 RTC_NOTREACHED();
169 return -1;
170 }
sakalcc452e12017-02-09 04:53:45 -0800171 void Decoded(webrtc::VideoFrame& decodedImage,
172 rtc::Optional<int32_t> decode_time_ms,
173 rtc::Optional<uint8_t> qp) override {
174 RTC_NOTREACHED();
175 }
Peter Boström7252a2b2015-05-18 19:42:03 +0200176 } callback, callback2;
177
178 VideoCodec codec = {};
179 fallback_wrapper_.InitDecode(&codec, 2);
180 fallback_wrapper_.RegisterDecodeCompleteCallback(&callback);
181 EXPECT_EQ(&callback, fake_decoder_.decode_complete_callback_);
182
183 fake_decoder_.decode_return_code_ = WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE;
184 EncodedImage encoded_image;
185 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
186 fallback_wrapper_.RegisterDecodeCompleteCallback(&callback2);
187 EXPECT_EQ(&callback2, fake_decoder_.decode_complete_callback_);
188}
189
Peter Boströmb7d9a972015-12-18 16:01:11 +0100190TEST_F(VideoDecoderSoftwareFallbackWrapperTest,
191 ReportsFallbackImplementationName) {
192 VideoCodec codec = {};
193 fallback_wrapper_.InitDecode(&codec, 2);
194
195 fake_decoder_.decode_return_code_ = WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE;
196 EncodedImage encoded_image;
197 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
198 // Hard coded expected value since libvpx is the software implementation name
199 // for VP8. Change accordingly if the underlying implementation does.
200 EXPECT_STREQ("libvpx (fallback from: fake-decoder)",
201 fallback_wrapper_.ImplementationName());
202 fallback_wrapper_.Release();
203}
204
Peter Boström7252a2b2015-05-18 19:42:03 +0200205} // namespace webrtc