blob: 4ac84856f2f1f043c9c2032417aaedbb4180e228 [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
ilnikc42f5402017-04-03 08:37:32 -070011#include "webrtc/api/video_codecs/video_decoder.h"
Per327d8ba2015-11-10 14:00:27 +010012#include "webrtc/base/checks.h"
magjedf6acc2a2016-11-22 01:43:03 -080013#include "webrtc/media/engine/videodecodersoftwarefallbackwrapper.h"
Henrik Kjellander2557b862015-11-18 22:00:21 +010014#include "webrtc/modules/video_coding/include/video_error_codes.h"
kwibergac9f8762016-09-30 22:29:43 -070015#include "webrtc/test/gtest.h"
Peter Boström7252a2b2015-05-18 19:42:03 +020016
17namespace webrtc {
18
19class VideoDecoderSoftwareFallbackWrapperTest : public ::testing::Test {
20 protected:
21 VideoDecoderSoftwareFallbackWrapperTest()
22 : fallback_wrapper_(kVideoCodecVP8, &fake_decoder_) {}
23
24 class CountingFakeDecoder : public VideoDecoder {
25 public:
26 int32_t InitDecode(const VideoCodec* codec_settings,
27 int32_t number_of_cores) override {
28 ++init_decode_count_;
Peter Boströmf812e452017-02-13 17:11:08 -050029 return init_decode_return_code_;
Peter Boström7252a2b2015-05-18 19:42:03 +020030 }
31
32 int32_t Decode(const EncodedImage& input_image,
33 bool missing_frames,
34 const RTPFragmentationHeader* fragmentation,
35 const CodecSpecificInfo* codec_specific_info,
36 int64_t render_time_ms) override {
37 ++decode_count_;
38 return decode_return_code_;
39 }
40
41 int32_t RegisterDecodeCompleteCallback(
42 DecodedImageCallback* callback) override {
43 decode_complete_callback_ = callback;
44 return WEBRTC_VIDEO_CODEC_OK;
45 }
46
47 int32_t Release() override {
48 ++release_count_;
49 return WEBRTC_VIDEO_CODEC_OK;
50 }
51
Peter Boströmb7d9a972015-12-18 16:01:11 +010052 const char* ImplementationName() const override {
53 return "fake-decoder";
54 }
55
Peter Boström7252a2b2015-05-18 19:42:03 +020056 int init_decode_count_ = 0;
57 int decode_count_ = 0;
Peter Boströmf812e452017-02-13 17:11:08 -050058 int32_t init_decode_return_code_ = WEBRTC_VIDEO_CODEC_OK;
Peter Boström7252a2b2015-05-18 19:42:03 +020059 int32_t decode_return_code_ = WEBRTC_VIDEO_CODEC_OK;
60 DecodedImageCallback* decode_complete_callback_ = nullptr;
61 int release_count_ = 0;
62 int reset_count_ = 0;
63 };
64 CountingFakeDecoder fake_decoder_;
65 VideoDecoderSoftwareFallbackWrapper fallback_wrapper_;
66};
67
68TEST_F(VideoDecoderSoftwareFallbackWrapperTest, InitializesDecoder) {
69 VideoCodec codec = {};
70 fallback_wrapper_.InitDecode(&codec, 2);
71 EXPECT_EQ(1, fake_decoder_.init_decode_count_);
Peter Boströmf812e452017-02-13 17:11:08 -050072
73 EncodedImage encoded_image;
74 encoded_image._frameType = kVideoFrameKey;
75 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
76 EXPECT_EQ(1, fake_decoder_.init_decode_count_)
77 << "Initialized decoder should not be reinitialized.";
78 EXPECT_EQ(1, fake_decoder_.decode_count_);
79}
80
81TEST_F(VideoDecoderSoftwareFallbackWrapperTest,
82 UsesFallbackDecoderAfterOnInitDecodeFailure) {
83 VideoCodec codec = {};
84 fake_decoder_.init_decode_return_code_ = WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE;
85 fallback_wrapper_.InitDecode(&codec, 2);
86 EXPECT_EQ(1, fake_decoder_.init_decode_count_);
87
88 EncodedImage encoded_image;
89 encoded_image._frameType = kVideoFrameKey;
90 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
91 EXPECT_EQ(2, fake_decoder_.init_decode_count_)
92 << "Should have attempted reinitializing the fallback decoder on "
93 "keyframe.";
94 // Unfortunately faking a VP8 frame is hard. Rely on no Decode -> using SW
95 // decoder.
96 EXPECT_EQ(0, fake_decoder_.decode_count_)
97 << "Decoder used even though no InitDecode had succeeded.";
Peter Boström7252a2b2015-05-18 19:42:03 +020098}
99
100TEST_F(VideoDecoderSoftwareFallbackWrapperTest,
101 CanRecoverFromSoftwareFallback) {
102 VideoCodec codec = {};
103 fallback_wrapper_.InitDecode(&codec, 2);
104 // Unfortunately faking a VP8 frame is hard. Rely on no Decode -> using SW
105 // decoder.
106 fake_decoder_.decode_return_code_ = WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE;
107 EncodedImage encoded_image;
108 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
109 EXPECT_EQ(1, fake_decoder_.decode_count_);
110
111 // Fail -> fake_decoder shouldn't be used anymore.
112 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
113 EXPECT_EQ(1, fake_decoder_.decode_count_)
114 << "Decoder used even though fallback should be active.";
115
116 // Should be able to recover on a keyframe.
Peter Boström49e196a2015-10-23 15:58:18 +0200117 encoded_image._frameType = kVideoFrameKey;
Peter Boström7252a2b2015-05-18 19:42:03 +0200118 fake_decoder_.decode_return_code_ = WEBRTC_VIDEO_CODEC_OK;
119 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
120 EXPECT_EQ(2, fake_decoder_.decode_count_)
121 << "Wrapper did not try to decode a keyframe using registered decoder.";
122
Peter Boström49e196a2015-10-23 15:58:18 +0200123 encoded_image._frameType = kVideoFrameDelta;
Peter Boström7252a2b2015-05-18 19:42:03 +0200124 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
125 EXPECT_EQ(3, fake_decoder_.decode_count_)
126 << "Decoder not used on future delta frames.";
127}
128
129TEST_F(VideoDecoderSoftwareFallbackWrapperTest, DoesNotFallbackOnEveryError) {
130 VideoCodec codec = {};
131 fallback_wrapper_.InitDecode(&codec, 2);
132 fake_decoder_.decode_return_code_ = WEBRTC_VIDEO_CODEC_ERROR;
133 EncodedImage encoded_image;
134 EXPECT_EQ(
135 fake_decoder_.decode_return_code_,
136 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1));
137 EXPECT_EQ(1, fake_decoder_.decode_count_);
138
139 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
140 EXPECT_EQ(2, fake_decoder_.decode_count_)
141 << "Decoder should be active even though previous decode failed.";
142}
143
144TEST_F(VideoDecoderSoftwareFallbackWrapperTest, ForwardsReleaseCall) {
145 VideoCodec codec = {};
146 fallback_wrapper_.InitDecode(&codec, 2);
147 fallback_wrapper_.Release();
148 EXPECT_EQ(1, fake_decoder_.release_count_);
149
150 fake_decoder_.decode_return_code_ = WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE;
151 EncodedImage encoded_image;
152 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
153 EXPECT_EQ(1, fake_decoder_.release_count_)
154 << "Decoder should not be released during fallback.";
155 fallback_wrapper_.Release();
156 EXPECT_EQ(2, fake_decoder_.release_count_);
157}
158
Peter Boström7252a2b2015-05-18 19:42:03 +0200159// TODO(pbos): Fake a VP8 frame well enough to actually receive a callback from
Peter Boströmb7d9a972015-12-18 16:01:11 +0100160// the software decoder.
Peter Boström7252a2b2015-05-18 19:42:03 +0200161TEST_F(VideoDecoderSoftwareFallbackWrapperTest,
162 ForwardsRegisterDecodeCompleteCallback) {
163 class FakeDecodedImageCallback : public DecodedImageCallback {
Miguel Casas-Sanchez47650702015-05-29 17:21:40 -0700164 int32_t Decoded(VideoFrame& decodedImage) override { return 0; }
Per327d8ba2015-11-10 14:00:27 +0100165 int32_t Decoded(
166 webrtc::VideoFrame& decodedImage, int64_t decode_time_ms) override {
167 RTC_NOTREACHED();
168 return -1;
169 }
sakalcc452e12017-02-09 04:53:45 -0800170 void Decoded(webrtc::VideoFrame& decodedImage,
171 rtc::Optional<int32_t> decode_time_ms,
172 rtc::Optional<uint8_t> qp) override {
173 RTC_NOTREACHED();
174 }
Peter Boström7252a2b2015-05-18 19:42:03 +0200175 } callback, callback2;
176
177 VideoCodec codec = {};
178 fallback_wrapper_.InitDecode(&codec, 2);
179 fallback_wrapper_.RegisterDecodeCompleteCallback(&callback);
180 EXPECT_EQ(&callback, fake_decoder_.decode_complete_callback_);
181
182 fake_decoder_.decode_return_code_ = WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE;
183 EncodedImage encoded_image;
184 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
185 fallback_wrapper_.RegisterDecodeCompleteCallback(&callback2);
186 EXPECT_EQ(&callback2, fake_decoder_.decode_complete_callback_);
187}
188
Peter Boströmb7d9a972015-12-18 16:01:11 +0100189TEST_F(VideoDecoderSoftwareFallbackWrapperTest,
190 ReportsFallbackImplementationName) {
191 VideoCodec codec = {};
192 fallback_wrapper_.InitDecode(&codec, 2);
193
194 fake_decoder_.decode_return_code_ = WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE;
195 EncodedImage encoded_image;
196 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
197 // Hard coded expected value since libvpx is the software implementation name
198 // for VP8. Change accordingly if the underlying implementation does.
199 EXPECT_STREQ("libvpx (fallback from: fake-decoder)",
200 fallback_wrapper_.ImplementationName());
201 fallback_wrapper_.Release();
202}
203
Peter Boström7252a2b2015-05-18 19:42:03 +0200204} // namespace webrtc