mirror of
https://github.com/mollyim/webrtc.git
synced 2025-05-12 21:30:45 +01:00

describing video codecs with their parameters as static members of SdpVideoFormat: static const SdpVideoFormat VP8(); static const SdpVideoFormat H264(); static const SdpVideoFormat VP9Profile0(); static const SdpVideoFormat VP9Profile1(); static const SdpVideoFormat VP9Profile2(); static const SdpVideoFormat VP9Profile3(); static const SdpVideoFormat AV1Profile0(); static const SdpVideoFormat AV1Profile1(); This removes the need to craft instances of these by hand. BUG=webrtc:15703 Change-Id: I2171e08b48ec98f18424f53f3b5d6d148130532e Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/337441 Commit-Queue: Philipp Hancke <phancke@microsoft.com> Reviewed-by: Harald Alvestrand <hta@webrtc.org> Reviewed-by: Florent Castelli <orphis@webrtc.org> Cr-Commit-Position: refs/heads/main@{#41833}
222 lines
7.3 KiB
C++
222 lines
7.3 KiB
C++
/*
|
|
* 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 "api/video_codecs/video_decoder.h"
|
|
#include "test/fake_encoded_frame.h"
|
|
#include "test/gmock.h"
|
|
#include "test/gtest.h"
|
|
#include "test/scoped_key_value_config.h"
|
|
#include "test/time_controller/simulated_time_controller.h"
|
|
|
|
namespace webrtc {
|
|
namespace {
|
|
using ::testing::_;
|
|
using ::testing::NiceMock;
|
|
using ::testing::Return;
|
|
|
|
class MockVideoStreamDecoderCallbacks
|
|
: public VideoStreamDecoderInterface::Callbacks {
|
|
public:
|
|
MOCK_METHOD(void, OnNonDecodableState, (), (override));
|
|
MOCK_METHOD(void, OnContinuousUntil, (int64_t frame_id), (override));
|
|
MOCK_METHOD(
|
|
void,
|
|
OnDecodedFrame,
|
|
(VideoFrame frame,
|
|
const VideoStreamDecoderInterface::Callbacks::FrameInfo& frame_info),
|
|
(override));
|
|
};
|
|
|
|
class StubVideoDecoder : public VideoDecoder {
|
|
public:
|
|
StubVideoDecoder() { ON_CALL(*this, Configure).WillByDefault(Return(true)); }
|
|
|
|
MOCK_METHOD(bool, Configure, (const Settings&), (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) {}
|
|
|
|
bool Configure(const Settings& settings) override {
|
|
return decoder_->Configure(settings);
|
|
}
|
|
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 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::AV1Profile0(), 1)}},
|
|
&field_trials_) {
|
|
// Set the min playout delay to a value greater than zero to not activate
|
|
// the low-latency renderer.
|
|
video_stream_decoder_.SetMinPlayoutDelay(TimeDelta::Millis(10));
|
|
}
|
|
|
|
test::ScopedKeyValueConfig field_trials_;
|
|
NiceMock<MockVideoStreamDecoderCallbacks> callbacks_;
|
|
FakeVideoDecoderFactory decoder_factory_;
|
|
GlobalSimulatedTimeController time_controller_;
|
|
VideoStreamDecoderImpl video_stream_decoder_;
|
|
};
|
|
|
|
TEST_F(VideoStreamDecoderImplTest, InsertAndDecodeFrame) {
|
|
video_stream_decoder_.OnFrame(
|
|
test::FakeFrameBuilder().PayloadType(1).AsLast().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(
|
|
test::FakeFrameBuilder().PayloadType(1).AsLast().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(
|
|
test::FakeFrameBuilder().PayloadType(1).AsLast().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(
|
|
test::FakeFrameBuilder()
|
|
.ReceivedTime(time_controller_.GetClock()->CurrentTime())
|
|
.PayloadType(1)
|
|
.AsLast()
|
|
.Build());
|
|
ON_CALL(decoder_factory_.Vp8Decoder(), Configure)
|
|
.WillByDefault(Return(false));
|
|
EXPECT_CALL(callbacks_, OnNonDecodableState);
|
|
time_controller_.AdvanceTime(TimeDelta::Millis(1));
|
|
}
|
|
|
|
TEST_F(VideoStreamDecoderImplTest, FailToDecodeFrame) {
|
|
video_stream_decoder_.OnFrame(
|
|
test::FakeFrameBuilder().PayloadType(1).AsLast().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) {
|
|
constexpr TimeDelta kFrameInterval = TimeDelta::Millis(1000 / 60);
|
|
video_stream_decoder_.OnFrame(
|
|
test::FakeFrameBuilder().PayloadType(1).Id(0).AsLast().Build());
|
|
EXPECT_CALL(decoder_factory_.Vp8Decoder(), DecodeCall);
|
|
EXPECT_CALL(callbacks_, OnDecodedFrame);
|
|
time_controller_.AdvanceTime(kFrameInterval);
|
|
|
|
video_stream_decoder_.OnFrame(
|
|
test::FakeFrameBuilder().PayloadType(2).Id(1).AsLast().Build());
|
|
EXPECT_CALL(decoder_factory_.Av1Decoder(), DecodeCall);
|
|
EXPECT_CALL(callbacks_, OnDecodedFrame);
|
|
time_controller_.AdvanceTime(kFrameInterval);
|
|
}
|
|
|
|
} // namespace
|
|
} // namespace webrtc
|