blob: 0b32daaaa22f790aeabfb059fc17ebe6afece04e [file] [log] [blame]
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +00001/*
2 * Copyright (c) 2013 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
kwiberg3f55dea2016-02-29 05:51:59 -080011#include <memory>
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +000012#include <vector>
13
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020014#include "modules/video_coding/include/mock/mock_vcm_callbacks.h"
15#include "modules/video_coding/include/mock/mock_video_codec_interface.h"
16#include "modules/video_coding/include/video_coding.h"
17#include "modules/video_coding/test/test_util.h"
18#include "modules/video_coding/timing.h"
19#include "modules/video_coding/video_coding_impl.h"
20#include "system_wrappers/include/clock.h"
21#include "test/gtest.h"
22#include "test/video_codec_settings.h"
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +000023
24using ::testing::_;
tommid0a71ba2017-03-14 04:16:20 -070025using ::testing::AnyNumber;
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +000026using ::testing::NiceMock;
27
28namespace webrtc {
29namespace vcm {
30namespace {
31
32class TestVideoReceiver : public ::testing::Test {
33 protected:
34 static const int kUnusedPayloadType = 10;
35
36 TestVideoReceiver() : clock_(0) {}
37
38 virtual void SetUp() {
philipel721d4022016-12-15 07:10:57 -080039 timing_.reset(new VCMTiming(&clock_));
40 receiver_.reset(
41 new VideoReceiver(&clock_, &event_factory_, nullptr, timing_.get()));
perkj796cfaf2015-12-10 09:27:38 -080042 receiver_->RegisterExternalDecoder(&decoder_, kUnusedPayloadType);
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +000043 const size_t kMaxNackListSize = 250;
44 const int kMaxPacketAgeToNack = 450;
45 receiver_->SetNackSettings(kMaxNackListSize, kMaxPacketAgeToNack, 0);
46
mflodman351424e2017-08-10 02:43:14 -070047 webrtc::test::CodecSettings(kVideoCodecVP8, &settings_);
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +000048 settings_.plType = kUnusedPayloadType; // Use the mocked encoder.
49 EXPECT_EQ(0, receiver_->RegisterReceiveCodec(&settings_, 1, true));
tommid0a71ba2017-03-14 04:16:20 -070050
51 // Since we call Decode, we need to provide a valid receive callback.
52 // However, for the purposes of these tests, we ignore the callbacks.
53 EXPECT_CALL(receive_callback_, OnIncomingPayloadType(_)).Times(AnyNumber());
54 EXPECT_CALL(receive_callback_, OnDecoderImplementationName(_))
55 .Times(AnyNumber());
56 receiver_->RegisterReceiveCallback(&receive_callback_);
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +000057 }
58
59 void InsertAndVerifyPaddingFrame(const uint8_t* payload,
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +000060 WebRtcRTPHeader* header) {
61 ASSERT_TRUE(header != NULL);
62 for (int j = 0; j < 5; ++j) {
63 // Padding only packets are passed to the VCM with payload size 0.
64 EXPECT_EQ(0, receiver_->IncomingPacket(payload, 0, *header));
65 ++header->header.sequenceNumber;
66 }
pbosa26ac922016-02-25 04:50:01 -080067 receiver_->Process();
Niels Möller8df3a382018-05-07 16:10:01 +020068 EXPECT_CALL(decoder_, Decode(_, _, _, _)).Times(0);
perkj796cfaf2015-12-10 09:27:38 -080069 EXPECT_EQ(VCM_FRAME_NOT_READY, receiver_->Decode(100));
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +000070 }
71
72 void InsertAndVerifyDecodableFrame(const uint8_t* payload,
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +000073 size_t length,
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +000074 WebRtcRTPHeader* header) {
75 ASSERT_TRUE(header != NULL);
76 EXPECT_EQ(0, receiver_->IncomingPacket(payload, length, *header));
77 ++header->header.sequenceNumber;
78 EXPECT_CALL(packet_request_callback_, ResendPackets(_, _)).Times(0);
Yves Gerey665174f2018-06-19 15:03:05 +020079 receiver_->Process();
Niels Möller8df3a382018-05-07 16:10:01 +020080 EXPECT_CALL(decoder_, Decode(_, _, _, _)).Times(1);
perkj796cfaf2015-12-10 09:27:38 -080081 EXPECT_EQ(0, receiver_->Decode(100));
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +000082 }
83
84 SimulatedClock clock_;
85 NullEventFactory event_factory_;
86 VideoCodec settings_;
87 NiceMock<MockVideoDecoder> decoder_;
88 NiceMock<MockPacketRequestCallback> packet_request_callback_;
89
philipel721d4022016-12-15 07:10:57 -080090 std::unique_ptr<VCMTiming> timing_;
tommid0a71ba2017-03-14 04:16:20 -070091 MockVCMReceiveCallback receive_callback_;
kwiberg3f55dea2016-02-29 05:51:59 -080092 std::unique_ptr<VideoReceiver> receiver_;
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +000093};
94
95TEST_F(TestVideoReceiver, PaddingOnlyFrames) {
96 EXPECT_EQ(0, receiver_->SetVideoProtection(kProtectionNack, true));
97 EXPECT_EQ(
98 0, receiver_->RegisterPacketRequestCallback(&packet_request_callback_));
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +000099 const size_t kPaddingSize = 220;
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000100 const uint8_t payload[kPaddingSize] = {0};
philipel7d745e52018-08-02 14:03:53 +0200101 WebRtcRTPHeader header = {};
pbos22993e12015-10-19 02:39:06 -0700102 header.frameType = kEmptyFrame;
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000103 header.header.markerBit = false;
104 header.header.paddingLength = kPaddingSize;
105 header.header.payloadType = kUnusedPayloadType;
106 header.header.ssrc = 1;
107 header.header.headerLength = 12;
philipelcb96ad82018-07-02 14:41:58 +0200108 header.video_header().codec = kVideoCodecVP8;
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000109 for (int i = 0; i < 10; ++i) {
110 EXPECT_CALL(packet_request_callback_, ResendPackets(_, _)).Times(0);
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +0000111 InsertAndVerifyPaddingFrame(payload, &header);
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000112 clock_.AdvanceTimeMilliseconds(33);
113 header.header.timestamp += 3000;
114 }
115}
116
117TEST_F(TestVideoReceiver, PaddingOnlyFramesWithLosses) {
118 EXPECT_EQ(0, receiver_->SetVideoProtection(kProtectionNack, true));
119 EXPECT_EQ(
120 0, receiver_->RegisterPacketRequestCallback(&packet_request_callback_));
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +0000121 const size_t kFrameSize = 1200;
122 const size_t kPaddingSize = 220;
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000123 const uint8_t payload[kFrameSize] = {0};
philipel7d745e52018-08-02 14:03:53 +0200124 WebRtcRTPHeader header = {};
pbos22993e12015-10-19 02:39:06 -0700125 header.frameType = kEmptyFrame;
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000126 header.header.markerBit = false;
127 header.header.paddingLength = kPaddingSize;
128 header.header.payloadType = kUnusedPayloadType;
129 header.header.ssrc = 1;
130 header.header.headerLength = 12;
philipelcb96ad82018-07-02 14:41:58 +0200131 header.video_header().codec = kVideoCodecVP8;
philipelaf7afc62018-09-06 13:20:09 +0200132 header.video_header().video_type_header.emplace<RTPVideoHeaderVP8>();
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000133 // Insert one video frame to get one frame decoded.
134 header.frameType = kVideoFrameKey;
philipelcb96ad82018-07-02 14:41:58 +0200135 header.video_header().is_first_packet_in_frame = true;
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000136 header.header.markerBit = true;
137 InsertAndVerifyDecodableFrame(payload, kFrameSize, &header);
138 clock_.AdvanceTimeMilliseconds(33);
139 header.header.timestamp += 3000;
140
pbos22993e12015-10-19 02:39:06 -0700141 header.frameType = kEmptyFrame;
philipelcb96ad82018-07-02 14:41:58 +0200142 header.video_header().is_first_packet_in_frame = false;
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000143 header.header.markerBit = false;
144 // Insert padding frames.
145 for (int i = 0; i < 10; ++i) {
146 // Lose one packet from the 6th frame.
147 if (i == 5) {
148 ++header.header.sequenceNumber;
149 }
150 // Lose the 4th frame.
151 if (i == 3) {
152 header.header.sequenceNumber += 5;
153 } else {
154 if (i > 3 && i < 5) {
155 EXPECT_CALL(packet_request_callback_, ResendPackets(_, 5)).Times(1);
156 } else if (i >= 5) {
157 EXPECT_CALL(packet_request_callback_, ResendPackets(_, 6)).Times(1);
158 } else {
159 EXPECT_CALL(packet_request_callback_, ResendPackets(_, _)).Times(0);
160 }
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +0000161 InsertAndVerifyPaddingFrame(payload, &header);
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000162 }
163 clock_.AdvanceTimeMilliseconds(33);
164 header.header.timestamp += 3000;
165 }
166}
167
168TEST_F(TestVideoReceiver, PaddingOnlyAndVideo) {
169 EXPECT_EQ(0, receiver_->SetVideoProtection(kProtectionNack, true));
170 EXPECT_EQ(
171 0, receiver_->RegisterPacketRequestCallback(&packet_request_callback_));
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +0000172 const size_t kFrameSize = 1200;
173 const size_t kPaddingSize = 220;
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000174 const uint8_t payload[kFrameSize] = {0};
philipel7d745e52018-08-02 14:03:53 +0200175 WebRtcRTPHeader header = {};
pbos22993e12015-10-19 02:39:06 -0700176 header.frameType = kEmptyFrame;
philipelcb96ad82018-07-02 14:41:58 +0200177 header.video_header().is_first_packet_in_frame = false;
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000178 header.header.markerBit = false;
179 header.header.paddingLength = kPaddingSize;
180 header.header.payloadType = kUnusedPayloadType;
181 header.header.ssrc = 1;
182 header.header.headerLength = 12;
philipelcb96ad82018-07-02 14:41:58 +0200183 header.video_header().codec = kVideoCodecVP8;
philipelaf7afc62018-09-06 13:20:09 +0200184 auto& vp8_header =
185 header.video.video_type_header.emplace<RTPVideoHeaderVP8>();
186 vp8_header.pictureId = -1;
187 vp8_header.tl0PicIdx = -1;
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000188 for (int i = 0; i < 3; ++i) {
189 // Insert 2 video frames.
190 for (int j = 0; j < 2; ++j) {
191 if (i == 0 && j == 0) // First frame should be a key frame.
192 header.frameType = kVideoFrameKey;
193 else
194 header.frameType = kVideoFrameDelta;
philipelcb96ad82018-07-02 14:41:58 +0200195 header.video_header().is_first_packet_in_frame = true;
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000196 header.header.markerBit = true;
197 InsertAndVerifyDecodableFrame(payload, kFrameSize, &header);
198 clock_.AdvanceTimeMilliseconds(33);
199 header.header.timestamp += 3000;
200 }
201
202 // Insert 2 padding only frames.
pbos22993e12015-10-19 02:39:06 -0700203 header.frameType = kEmptyFrame;
philipelcb96ad82018-07-02 14:41:58 +0200204 header.video_header().is_first_packet_in_frame = false;
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000205 header.header.markerBit = false;
206 for (int j = 0; j < 2; ++j) {
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +0000207 // InsertAndVerifyPaddingFrame(payload, &header);
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000208 clock_.AdvanceTimeMilliseconds(33);
209 header.header.timestamp += 3000;
210 }
211 }
212}
213
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000214} // namespace
215} // namespace vcm
216} // namespace webrtc