You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
239 lines
7.7 KiB
239 lines
7.7 KiB
/*
|
|
* Copyright (c) 2020 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 "video/video_stream_decoder_impl.h"
|
|
|
|
#include <vector>
|
|
|
|
#include "api/video/i420_buffer.h"
|
|
#include "test/gmock.h"
|
|
#include "test/gtest.h"
|
|
#include "test/time_controller/simulated_time_controller.h"
|
|
|
|
namespace webrtc {
|
|
namespace {
|
|
using ::testing::_;
|
|
using ::testing::ByMove;
|
|
using ::testing::NiceMock;
|
|
using ::testing::Return;
|
|
|
|
class MockVideoStreamDecoderCallbacks
|
|
: public VideoStreamDecoderInterface::Callbacks {
|
|
public:
|
|
MOCK_METHOD(void, OnNonDecodableState, (), (override));
|
|
MOCK_METHOD(void,
|
|
OnContinuousUntil,
|
|
(const video_coding::VideoLayerFrameId& key),
|
|
(override));
|
|
MOCK_METHOD(void,
|
|
OnDecodedFrame,
|
|
(VideoFrame decodedImage,
|
|
absl::optional<int> decode_time_ms,
|
|
absl::optional<int> qp),
|
|
(override));
|
|
};
|
|
|
|
class StubVideoDecoder : public VideoDecoder {
|
|
public:
|
|
MOCK_METHOD(int32_t,
|
|
InitDecode,
|
|
(const VideoCodec*, int32_t number_of_cores),
|
|
(override));
|
|
|
|
int32_t Decode(const EncodedImage& input_image,
|
|
bool missing_frames,
|
|
int64_t render_time_ms) override {
|
|
int32_t ret_code = DecodeCall(input_image, missing_frames, render_time_ms);
|
|
if (ret_code == WEBRTC_VIDEO_CODEC_OK ||
|
|
ret_code == WEBRTC_VIDEO_CODEC_OK_REQUEST_KEYFRAME) {
|
|
VideoFrame frame = VideoFrame::Builder()
|
|
.set_video_frame_buffer(I420Buffer::Create(1, 1))
|
|
.build();
|
|
callback_->Decoded(frame);
|
|
}
|
|
return ret_code;
|
|
}
|
|
|
|
MOCK_METHOD(int32_t,
|
|
DecodeCall,
|
|
(const EncodedImage& input_image,
|
|
bool missing_frames,
|
|
int64_t render_time_ms),
|
|
());
|
|
|
|
int32_t Release() override { return 0; }
|
|
|
|
int32_t RegisterDecodeCompleteCallback(
|
|
DecodedImageCallback* callback) override {
|
|
callback_ = callback;
|
|
return 0;
|
|
}
|
|
|
|
private:
|
|
DecodedImageCallback* callback_;
|
|
};
|
|
|
|
class WrappedVideoDecoder : public VideoDecoder {
|
|
public:
|
|
explicit WrappedVideoDecoder(StubVideoDecoder* decoder) : decoder_(decoder) {}
|
|
|
|
int32_t InitDecode(const VideoCodec* codec_settings,
|
|
int32_t number_of_cores) override {
|
|
return decoder_->InitDecode(codec_settings, number_of_cores);
|
|
}
|
|
int32_t Decode(const EncodedImage& input_image,
|
|
bool missing_frames,
|
|
int64_t render_time_ms) override {
|
|
return decoder_->Decode(input_image, missing_frames, render_time_ms);
|
|
}
|
|
int32_t Release() override { return decoder_->Release(); }
|
|
|
|
int32_t RegisterDecodeCompleteCallback(
|
|
DecodedImageCallback* callback) override {
|
|
return decoder_->RegisterDecodeCompleteCallback(callback);
|
|
}
|
|
|
|
private:
|
|
StubVideoDecoder* decoder_;
|
|
};
|
|
|
|
class FakeVideoDecoderFactory : public VideoDecoderFactory {
|
|
public:
|
|
std::vector<SdpVideoFormat> GetSupportedFormats() const override {
|
|
return {};
|
|
}
|
|
std::unique_ptr<VideoDecoder> CreateVideoDecoder(
|
|
const SdpVideoFormat& format) override {
|
|
if (format.name == "VP8") {
|
|
return std::make_unique<WrappedVideoDecoder>(&vp8_decoder_);
|
|
}
|
|
|
|
if (format.name == "AV1") {
|
|
return std::make_unique<WrappedVideoDecoder>(&av1_decoder_);
|
|
}
|
|
|
|
return {};
|
|
}
|
|
|
|
StubVideoDecoder& Vp8Decoder() { return vp8_decoder_; }
|
|
StubVideoDecoder& Av1Decoder() { return av1_decoder_; }
|
|
|
|
private:
|
|
NiceMock<StubVideoDecoder> vp8_decoder_;
|
|
NiceMock<StubVideoDecoder> av1_decoder_;
|
|
};
|
|
|
|
class FakeEncodedFrame : public video_coding::EncodedFrame {
|
|
public:
|
|
int64_t ReceivedTime() const override { return 0; }
|
|
int64_t RenderTime() const override { return 0; }
|
|
|
|
// Setters for protected variables.
|
|
void SetPayloadType(int payload_type) { _payloadType = payload_type; }
|
|
};
|
|
|
|
class FrameBuilder {
|
|
public:
|
|
FrameBuilder() : frame_(std::make_unique<FakeEncodedFrame>()) {}
|
|
|
|
FrameBuilder& WithPayloadType(int payload_type) {
|
|
frame_->SetPayloadType(payload_type);
|
|
return *this;
|
|
}
|
|
|
|
FrameBuilder& WithPictureId(int picture_id) {
|
|
frame_->id.picture_id = picture_id;
|
|
return *this;
|
|
}
|
|
|
|
std::unique_ptr<FakeEncodedFrame> Build() { return std::move(frame_); }
|
|
|
|
private:
|
|
std::unique_ptr<FakeEncodedFrame> frame_;
|
|
};
|
|
|
|
class VideoStreamDecoderImplTest : public ::testing::Test {
|
|
public:
|
|
VideoStreamDecoderImplTest()
|
|
: time_controller_(Timestamp::Seconds(0)),
|
|
video_stream_decoder_(&callbacks_,
|
|
&decoder_factory_,
|
|
time_controller_.GetTaskQueueFactory(),
|
|
{{1, std::make_pair(SdpVideoFormat("VP8"), 1)},
|
|
{2, std::make_pair(SdpVideoFormat("AV1"), 1)}}) {
|
|
}
|
|
|
|
NiceMock<MockVideoStreamDecoderCallbacks> callbacks_;
|
|
FakeVideoDecoderFactory decoder_factory_;
|
|
GlobalSimulatedTimeController time_controller_;
|
|
VideoStreamDecoderImpl video_stream_decoder_;
|
|
};
|
|
|
|
TEST_F(VideoStreamDecoderImplTest, InsertAndDecodeFrame) {
|
|
video_stream_decoder_.OnFrame(FrameBuilder().WithPayloadType(1).Build());
|
|
EXPECT_CALL(callbacks_, OnDecodedFrame);
|
|
time_controller_.AdvanceTime(TimeDelta::Millis(1));
|
|
}
|
|
|
|
TEST_F(VideoStreamDecoderImplTest, NonDecodableStateWaitingForKeyframe) {
|
|
EXPECT_CALL(callbacks_, OnNonDecodableState);
|
|
time_controller_.AdvanceTime(TimeDelta::Millis(200));
|
|
}
|
|
|
|
TEST_F(VideoStreamDecoderImplTest, NonDecodableStateWaitingForDeltaFrame) {
|
|
video_stream_decoder_.OnFrame(FrameBuilder().WithPayloadType(1).Build());
|
|
EXPECT_CALL(callbacks_, OnDecodedFrame);
|
|
time_controller_.AdvanceTime(TimeDelta::Millis(1));
|
|
EXPECT_CALL(callbacks_, OnNonDecodableState);
|
|
time_controller_.AdvanceTime(TimeDelta::Millis(3000));
|
|
}
|
|
|
|
TEST_F(VideoStreamDecoderImplTest, InsertAndDecodeFrameWithKeyframeRequest) {
|
|
video_stream_decoder_.OnFrame(FrameBuilder().WithPayloadType(1).Build());
|
|
EXPECT_CALL(decoder_factory_.Vp8Decoder(), DecodeCall)
|
|
.WillOnce(Return(WEBRTC_VIDEO_CODEC_OK_REQUEST_KEYFRAME));
|
|
EXPECT_CALL(callbacks_, OnDecodedFrame);
|
|
EXPECT_CALL(callbacks_, OnNonDecodableState);
|
|
time_controller_.AdvanceTime(TimeDelta::Millis(1));
|
|
}
|
|
|
|
TEST_F(VideoStreamDecoderImplTest, FailToInitDecoder) {
|
|
video_stream_decoder_.OnFrame(FrameBuilder().WithPayloadType(1).Build());
|
|
ON_CALL(decoder_factory_.Vp8Decoder(), InitDecode)
|
|
.WillByDefault(Return(WEBRTC_VIDEO_CODEC_ERROR));
|
|
EXPECT_CALL(callbacks_, OnNonDecodableState);
|
|
time_controller_.AdvanceTime(TimeDelta::Millis(1));
|
|
}
|
|
|
|
TEST_F(VideoStreamDecoderImplTest, FailToDecodeFrame) {
|
|
video_stream_decoder_.OnFrame(FrameBuilder().WithPayloadType(1).Build());
|
|
ON_CALL(decoder_factory_.Vp8Decoder(), DecodeCall)
|
|
.WillByDefault(Return(WEBRTC_VIDEO_CODEC_ERROR));
|
|
EXPECT_CALL(callbacks_, OnNonDecodableState);
|
|
time_controller_.AdvanceTime(TimeDelta::Millis(1));
|
|
}
|
|
|
|
TEST_F(VideoStreamDecoderImplTest, ChangeFramePayloadType) {
|
|
video_stream_decoder_.OnFrame(
|
|
FrameBuilder().WithPayloadType(1).WithPictureId(0).Build());
|
|
EXPECT_CALL(decoder_factory_.Vp8Decoder(), DecodeCall);
|
|
EXPECT_CALL(callbacks_, OnDecodedFrame);
|
|
time_controller_.AdvanceTime(TimeDelta::Millis(1));
|
|
|
|
video_stream_decoder_.OnFrame(
|
|
FrameBuilder().WithPayloadType(2).WithPictureId(1).Build());
|
|
EXPECT_CALL(decoder_factory_.Av1Decoder(), DecodeCall);
|
|
EXPECT_CALL(callbacks_, OnDecodedFrame);
|
|
time_controller_.AdvanceTime(TimeDelta::Millis(1));
|
|
}
|
|
|
|
} // namespace
|
|
} // namespace webrtc
|