blob: dfd388be13d499263fe72670a8dc4379e105313c [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
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020011#include "media/engine/videodecodersoftwarefallbackwrapper.h"
12#include "api/video_codecs/video_decoder.h"
13#include "modules/video_coding/include/video_error_codes.h"
14#include "rtc_base/checks.h"
15#include "test/gtest.h"
Peter Boström7252a2b2015-05-18 19:42:03 +020016
17namespace webrtc {
18
19class VideoDecoderSoftwareFallbackWrapperTest : public ::testing::Test {
20 protected:
21 VideoDecoderSoftwareFallbackWrapperTest()
andersc063f0c02017-09-11 11:50:51 -070022 : fake_decoder_(new CountingFakeDecoder()),
Christian Fremerey267d84b2017-11-08 23:26:44 +000023 fallback_wrapper_(kVideoCodecVP8,
andersc063f0c02017-09-11 11:50:51 -070024 std::unique_ptr<VideoDecoder>(fake_decoder_)) {}
Peter Boström7252a2b2015-05-18 19:42:03 +020025
26 class CountingFakeDecoder : public VideoDecoder {
27 public:
28 int32_t InitDecode(const VideoCodec* codec_settings,
29 int32_t number_of_cores) override {
30 ++init_decode_count_;
Peter Boströmf812e452017-02-13 17:11:08 -050031 return init_decode_return_code_;
Peter Boström7252a2b2015-05-18 19:42:03 +020032 }
33
34 int32_t Decode(const EncodedImage& input_image,
35 bool missing_frames,
36 const RTPFragmentationHeader* fragmentation,
37 const CodecSpecificInfo* codec_specific_info,
38 int64_t render_time_ms) override {
39 ++decode_count_;
40 return decode_return_code_;
41 }
42
43 int32_t RegisterDecodeCompleteCallback(
44 DecodedImageCallback* callback) override {
45 decode_complete_callback_ = callback;
46 return WEBRTC_VIDEO_CODEC_OK;
47 }
48
49 int32_t Release() override {
50 ++release_count_;
51 return WEBRTC_VIDEO_CODEC_OK;
52 }
53
Peter Boströmb7d9a972015-12-18 16:01:11 +010054 const char* ImplementationName() const override {
55 return "fake-decoder";
56 }
57
Peter Boström7252a2b2015-05-18 19:42:03 +020058 int init_decode_count_ = 0;
59 int decode_count_ = 0;
Peter Boströmf812e452017-02-13 17:11:08 -050060 int32_t init_decode_return_code_ = WEBRTC_VIDEO_CODEC_OK;
Peter Boström7252a2b2015-05-18 19:42:03 +020061 int32_t decode_return_code_ = WEBRTC_VIDEO_CODEC_OK;
62 DecodedImageCallback* decode_complete_callback_ = nullptr;
63 int release_count_ = 0;
64 int reset_count_ = 0;
65 };
andersc063f0c02017-09-11 11:50:51 -070066 // |fake_decoder_| is owned and released by |fallback_wrapper_|.
67 CountingFakeDecoder* fake_decoder_;
Peter Boström7252a2b2015-05-18 19:42:03 +020068 VideoDecoderSoftwareFallbackWrapper fallback_wrapper_;
69};
70
71TEST_F(VideoDecoderSoftwareFallbackWrapperTest, InitializesDecoder) {
72 VideoCodec codec = {};
73 fallback_wrapper_.InitDecode(&codec, 2);
andersc063f0c02017-09-11 11:50:51 -070074 EXPECT_EQ(1, fake_decoder_->init_decode_count_);
Peter Boströmf812e452017-02-13 17:11:08 -050075
76 EncodedImage encoded_image;
77 encoded_image._frameType = kVideoFrameKey;
78 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
andersc063f0c02017-09-11 11:50:51 -070079 EXPECT_EQ(1, fake_decoder_->init_decode_count_)
Peter Boströmf812e452017-02-13 17:11:08 -050080 << "Initialized decoder should not be reinitialized.";
andersc063f0c02017-09-11 11:50:51 -070081 EXPECT_EQ(1, fake_decoder_->decode_count_);
Peter Boströmf812e452017-02-13 17:11:08 -050082}
83
84TEST_F(VideoDecoderSoftwareFallbackWrapperTest,
Emircan Uysaler874d9a52017-10-31 17:15:22 -070085 UsesFallbackDecoderAfterAnyInitDecodeFailure) {
Peter Boströmf812e452017-02-13 17:11:08 -050086 VideoCodec codec = {};
andersc063f0c02017-09-11 11:50:51 -070087 fake_decoder_->init_decode_return_code_ =
Emircan Uysaler874d9a52017-10-31 17:15:22 -070088 WEBRTC_VIDEO_CODEC_UNINITIALIZED;
Peter Boströmf812e452017-02-13 17:11:08 -050089 fallback_wrapper_.InitDecode(&codec, 2);
andersc063f0c02017-09-11 11:50:51 -070090 EXPECT_EQ(1, fake_decoder_->init_decode_count_);
Peter Boströmf812e452017-02-13 17:11:08 -050091
92 EncodedImage encoded_image;
93 encoded_image._frameType = kVideoFrameKey;
94 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
andersc063f0c02017-09-11 11:50:51 -070095 EXPECT_EQ(2, fake_decoder_->init_decode_count_)
Peter Boströmf812e452017-02-13 17:11:08 -050096 << "Should have attempted reinitializing the fallback decoder on "
97 "keyframe.";
98 // Unfortunately faking a VP8 frame is hard. Rely on no Decode -> using SW
99 // decoder.
andersc063f0c02017-09-11 11:50:51 -0700100 EXPECT_EQ(0, fake_decoder_->decode_count_)
Peter Boströmf812e452017-02-13 17:11:08 -0500101 << "Decoder used even though no InitDecode had succeeded.";
Peter Boström7252a2b2015-05-18 19:42:03 +0200102}
103
104TEST_F(VideoDecoderSoftwareFallbackWrapperTest,
105 CanRecoverFromSoftwareFallback) {
106 VideoCodec codec = {};
107 fallback_wrapper_.InitDecode(&codec, 2);
108 // Unfortunately faking a VP8 frame is hard. Rely on no Decode -> using SW
109 // decoder.
andersc063f0c02017-09-11 11:50:51 -0700110 fake_decoder_->decode_return_code_ = WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE;
Peter Boström7252a2b2015-05-18 19:42:03 +0200111 EncodedImage encoded_image;
112 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
andersc063f0c02017-09-11 11:50:51 -0700113 EXPECT_EQ(1, fake_decoder_->decode_count_);
Peter Boström7252a2b2015-05-18 19:42:03 +0200114
115 // Fail -> fake_decoder shouldn't be used anymore.
116 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
andersc063f0c02017-09-11 11:50:51 -0700117 EXPECT_EQ(1, fake_decoder_->decode_count_)
Peter Boström7252a2b2015-05-18 19:42:03 +0200118 << "Decoder used even though fallback should be active.";
119
120 // Should be able to recover on a keyframe.
Peter Boström49e196a2015-10-23 15:58:18 +0200121 encoded_image._frameType = kVideoFrameKey;
andersc063f0c02017-09-11 11:50:51 -0700122 fake_decoder_->decode_return_code_ = WEBRTC_VIDEO_CODEC_OK;
Peter Boström7252a2b2015-05-18 19:42:03 +0200123 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
andersc063f0c02017-09-11 11:50:51 -0700124 EXPECT_EQ(2, fake_decoder_->decode_count_)
Peter Boström7252a2b2015-05-18 19:42:03 +0200125 << "Wrapper did not try to decode a keyframe using registered decoder.";
126
Peter Boström49e196a2015-10-23 15:58:18 +0200127 encoded_image._frameType = kVideoFrameDelta;
Peter Boström7252a2b2015-05-18 19:42:03 +0200128 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
andersc063f0c02017-09-11 11:50:51 -0700129 EXPECT_EQ(3, fake_decoder_->decode_count_)
Peter Boström7252a2b2015-05-18 19:42:03 +0200130 << "Decoder not used on future delta frames.";
131}
132
133TEST_F(VideoDecoderSoftwareFallbackWrapperTest, DoesNotFallbackOnEveryError) {
134 VideoCodec codec = {};
135 fallback_wrapper_.InitDecode(&codec, 2);
andersc063f0c02017-09-11 11:50:51 -0700136 fake_decoder_->decode_return_code_ = WEBRTC_VIDEO_CODEC_ERROR;
Peter Boström7252a2b2015-05-18 19:42:03 +0200137 EncodedImage encoded_image;
138 EXPECT_EQ(
andersc063f0c02017-09-11 11:50:51 -0700139 fake_decoder_->decode_return_code_,
Peter Boström7252a2b2015-05-18 19:42:03 +0200140 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1));
andersc063f0c02017-09-11 11:50:51 -0700141 EXPECT_EQ(1, fake_decoder_->decode_count_);
Peter Boström7252a2b2015-05-18 19:42:03 +0200142
143 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
andersc063f0c02017-09-11 11:50:51 -0700144 EXPECT_EQ(2, fake_decoder_->decode_count_)
Peter Boström7252a2b2015-05-18 19:42:03 +0200145 << "Decoder should be active even though previous decode failed.";
146}
147
148TEST_F(VideoDecoderSoftwareFallbackWrapperTest, ForwardsReleaseCall) {
149 VideoCodec codec = {};
150 fallback_wrapper_.InitDecode(&codec, 2);
151 fallback_wrapper_.Release();
andersc063f0c02017-09-11 11:50:51 -0700152 EXPECT_EQ(1, fake_decoder_->release_count_);
Peter Boström7252a2b2015-05-18 19:42:03 +0200153
andersc063f0c02017-09-11 11:50:51 -0700154 fake_decoder_->decode_return_code_ = WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE;
Peter Boström7252a2b2015-05-18 19:42:03 +0200155 EncodedImage encoded_image;
156 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
andersc063f0c02017-09-11 11:50:51 -0700157 EXPECT_EQ(1, fake_decoder_->release_count_)
Peter Boström7252a2b2015-05-18 19:42:03 +0200158 << "Decoder should not be released during fallback.";
159 fallback_wrapper_.Release();
andersc063f0c02017-09-11 11:50:51 -0700160 EXPECT_EQ(2, fake_decoder_->release_count_);
Peter Boström7252a2b2015-05-18 19:42:03 +0200161}
162
Peter Boström7252a2b2015-05-18 19:42:03 +0200163// TODO(pbos): Fake a VP8 frame well enough to actually receive a callback from
Peter Boströmb7d9a972015-12-18 16:01:11 +0100164// the software decoder.
Peter Boström7252a2b2015-05-18 19:42:03 +0200165TEST_F(VideoDecoderSoftwareFallbackWrapperTest,
166 ForwardsRegisterDecodeCompleteCallback) {
167 class FakeDecodedImageCallback : public DecodedImageCallback {
Miguel Casas-Sanchez47650702015-05-29 17:21:40 -0700168 int32_t Decoded(VideoFrame& decodedImage) override { return 0; }
Per327d8ba2015-11-10 14:00:27 +0100169 int32_t Decoded(
170 webrtc::VideoFrame& decodedImage, int64_t decode_time_ms) override {
171 RTC_NOTREACHED();
172 return -1;
173 }
sakalcc452e12017-02-09 04:53:45 -0800174 void Decoded(webrtc::VideoFrame& decodedImage,
175 rtc::Optional<int32_t> decode_time_ms,
176 rtc::Optional<uint8_t> qp) override {
177 RTC_NOTREACHED();
178 }
Peter Boström7252a2b2015-05-18 19:42:03 +0200179 } callback, callback2;
180
181 VideoCodec codec = {};
182 fallback_wrapper_.InitDecode(&codec, 2);
183 fallback_wrapper_.RegisterDecodeCompleteCallback(&callback);
andersc063f0c02017-09-11 11:50:51 -0700184 EXPECT_EQ(&callback, fake_decoder_->decode_complete_callback_);
Peter Boström7252a2b2015-05-18 19:42:03 +0200185
andersc063f0c02017-09-11 11:50:51 -0700186 fake_decoder_->decode_return_code_ = WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE;
Peter Boström7252a2b2015-05-18 19:42:03 +0200187 EncodedImage encoded_image;
188 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
189 fallback_wrapper_.RegisterDecodeCompleteCallback(&callback2);
andersc063f0c02017-09-11 11:50:51 -0700190 EXPECT_EQ(&callback2, fake_decoder_->decode_complete_callback_);
Peter Boström7252a2b2015-05-18 19:42:03 +0200191}
192
Peter Boströmb7d9a972015-12-18 16:01:11 +0100193TEST_F(VideoDecoderSoftwareFallbackWrapperTest,
194 ReportsFallbackImplementationName) {
195 VideoCodec codec = {};
196 fallback_wrapper_.InitDecode(&codec, 2);
197
andersc063f0c02017-09-11 11:50:51 -0700198 fake_decoder_->decode_return_code_ = WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE;
Peter Boströmb7d9a972015-12-18 16:01:11 +0100199 EncodedImage encoded_image;
200 fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
201 // Hard coded expected value since libvpx is the software implementation name
202 // for VP8. Change accordingly if the underlying implementation does.
203 EXPECT_STREQ("libvpx (fallback from: fake-decoder)",
204 fallback_wrapper_.ImplementationName());
205 fallback_wrapper_.Release();
206}
207
Peter Boström7252a2b2015-05-18 19:42:03 +0200208} // namespace webrtc