aboutsummaryrefslogtreecommitdiff
path: root/webrtc/video/video_decoder_unittest.cc
diff options
context:
space:
mode:
Diffstat (limited to 'webrtc/video/video_decoder_unittest.cc')
-rw-r--r--webrtc/video/video_decoder_unittest.cc165
1 files changed, 165 insertions, 0 deletions
diff --git a/webrtc/video/video_decoder_unittest.cc b/webrtc/video/video_decoder_unittest.cc
new file mode 100644
index 0000000000..be09b191ac
--- /dev/null
+++ b/webrtc/video/video_decoder_unittest.cc
@@ -0,0 +1,165 @@
+/*
+ * Copyright (c) 2015 The WebRTC project authors. All Rights Reserved.
+ *
+ * Use of this source code is governed by a BSD-style license
+ * that can be found in the LICENSE file in the root of the source
+ * tree. An additional intellectual property rights grant can be found
+ * in the file PATENTS. All contributing project authors may
+ * be found in the AUTHORS file in the root of the source tree.
+ */
+
+#include "webrtc/video_decoder.h"
+
+#include "testing/gtest/include/gtest/gtest.h"
+#include "webrtc/modules/video_coding/codecs/interface/video_error_codes.h"
+
+namespace webrtc {
+
+class VideoDecoderSoftwareFallbackWrapperTest : public ::testing::Test {
+ protected:
+ VideoDecoderSoftwareFallbackWrapperTest()
+ : fallback_wrapper_(kVideoCodecVP8, &fake_decoder_) {}
+
+ class CountingFakeDecoder : public VideoDecoder {
+ public:
+ int32_t InitDecode(const VideoCodec* codec_settings,
+ int32_t number_of_cores) override {
+ ++init_decode_count_;
+ return WEBRTC_VIDEO_CODEC_OK;
+ }
+
+ int32_t Decode(const EncodedImage& input_image,
+ bool missing_frames,
+ const RTPFragmentationHeader* fragmentation,
+ const CodecSpecificInfo* codec_specific_info,
+ int64_t render_time_ms) override {
+ ++decode_count_;
+ return decode_return_code_;
+ }
+
+ int32_t RegisterDecodeCompleteCallback(
+ DecodedImageCallback* callback) override {
+ decode_complete_callback_ = callback;
+ return WEBRTC_VIDEO_CODEC_OK;
+ }
+
+ int32_t Release() override {
+ ++release_count_;
+ return WEBRTC_VIDEO_CODEC_OK;
+ }
+
+ int32_t Reset() override {
+ ++reset_count_;
+ return WEBRTC_VIDEO_CODEC_OK;
+ }
+ int init_decode_count_ = 0;
+ int decode_count_ = 0;
+ int32_t decode_return_code_ = WEBRTC_VIDEO_CODEC_OK;
+ DecodedImageCallback* decode_complete_callback_ = nullptr;
+ int release_count_ = 0;
+ int reset_count_ = 0;
+ };
+ CountingFakeDecoder fake_decoder_;
+ VideoDecoderSoftwareFallbackWrapper fallback_wrapper_;
+};
+
+TEST_F(VideoDecoderSoftwareFallbackWrapperTest, InitializesDecoder) {
+ VideoCodec codec = {};
+ fallback_wrapper_.InitDecode(&codec, 2);
+ EXPECT_EQ(1, fake_decoder_.init_decode_count_);
+}
+
+TEST_F(VideoDecoderSoftwareFallbackWrapperTest,
+ CanRecoverFromSoftwareFallback) {
+ VideoCodec codec = {};
+ fallback_wrapper_.InitDecode(&codec, 2);
+ // Unfortunately faking a VP8 frame is hard. Rely on no Decode -> using SW
+ // decoder.
+ fake_decoder_.decode_return_code_ = WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE;
+ EncodedImage encoded_image;
+ fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
+ EXPECT_EQ(1, fake_decoder_.decode_count_);
+
+ // Fail -> fake_decoder shouldn't be used anymore.
+ fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
+ EXPECT_EQ(1, fake_decoder_.decode_count_)
+ << "Decoder used even though fallback should be active.";
+
+ // Should be able to recover on a keyframe.
+ encoded_image._frameType = kVideoFrameKey;
+ fake_decoder_.decode_return_code_ = WEBRTC_VIDEO_CODEC_OK;
+ fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
+ EXPECT_EQ(2, fake_decoder_.decode_count_)
+ << "Wrapper did not try to decode a keyframe using registered decoder.";
+
+ encoded_image._frameType = kVideoFrameDelta;
+ fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
+ EXPECT_EQ(3, fake_decoder_.decode_count_)
+ << "Decoder not used on future delta frames.";
+}
+
+TEST_F(VideoDecoderSoftwareFallbackWrapperTest, DoesNotFallbackOnEveryError) {
+ VideoCodec codec = {};
+ fallback_wrapper_.InitDecode(&codec, 2);
+ fake_decoder_.decode_return_code_ = WEBRTC_VIDEO_CODEC_ERROR;
+ EncodedImage encoded_image;
+ EXPECT_EQ(
+ fake_decoder_.decode_return_code_,
+ fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1));
+ EXPECT_EQ(1, fake_decoder_.decode_count_);
+
+ fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
+ EXPECT_EQ(2, fake_decoder_.decode_count_)
+ << "Decoder should be active even though previous decode failed.";
+}
+
+TEST_F(VideoDecoderSoftwareFallbackWrapperTest, ForwardsReleaseCall) {
+ VideoCodec codec = {};
+ fallback_wrapper_.InitDecode(&codec, 2);
+ fallback_wrapper_.Release();
+ EXPECT_EQ(1, fake_decoder_.release_count_);
+
+ fake_decoder_.decode_return_code_ = WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE;
+ EncodedImage encoded_image;
+ fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
+ EXPECT_EQ(1, fake_decoder_.release_count_)
+ << "Decoder should not be released during fallback.";
+ fallback_wrapper_.Release();
+ EXPECT_EQ(2, fake_decoder_.release_count_);
+}
+
+TEST_F(VideoDecoderSoftwareFallbackWrapperTest, ForwardsResetCall) {
+ VideoCodec codec = {};
+ fallback_wrapper_.InitDecode(&codec, 2);
+ fallback_wrapper_.Reset();
+ EXPECT_EQ(1, fake_decoder_.reset_count_);
+
+ fake_decoder_.decode_return_code_ = WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE;
+ EncodedImage encoded_image;
+ fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
+ fallback_wrapper_.Reset();
+ EXPECT_EQ(2, fake_decoder_.reset_count_)
+ << "Reset not forwarded during fallback.";
+}
+
+// TODO(pbos): Fake a VP8 frame well enough to actually receive a callback from
+// the software encoder.
+TEST_F(VideoDecoderSoftwareFallbackWrapperTest,
+ ForwardsRegisterDecodeCompleteCallback) {
+ class FakeDecodedImageCallback : public DecodedImageCallback {
+ int32_t Decoded(VideoFrame& decodedImage) override { return 0; }
+ } callback, callback2;
+
+ VideoCodec codec = {};
+ fallback_wrapper_.InitDecode(&codec, 2);
+ fallback_wrapper_.RegisterDecodeCompleteCallback(&callback);
+ EXPECT_EQ(&callback, fake_decoder_.decode_complete_callback_);
+
+ fake_decoder_.decode_return_code_ = WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE;
+ EncodedImage encoded_image;
+ fallback_wrapper_.Decode(encoded_image, false, nullptr, nullptr, -1);
+ fallback_wrapper_.RegisterDecodeCompleteCallback(&callback2);
+ EXPECT_EQ(&callback2, fake_decoder_.decode_complete_callback_);
+}
+
+} // namespace webrtc