blob: a4778e8e884416dc0b9fa360d6097be209672990 [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};
philipeldfbced62018-07-12 15:09:06 +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};
philipeldfbced62018-07-12 15:09:06 +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;
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000132 // Insert one video frame to get one frame decoded.
133 header.frameType = kVideoFrameKey;
philipelcb96ad82018-07-02 14:41:58 +0200134 header.video_header().is_first_packet_in_frame = true;
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000135 header.header.markerBit = true;
136 InsertAndVerifyDecodableFrame(payload, kFrameSize, &header);
137 clock_.AdvanceTimeMilliseconds(33);
138 header.header.timestamp += 3000;
139
pbos22993e12015-10-19 02:39:06 -0700140 header.frameType = kEmptyFrame;
philipelcb96ad82018-07-02 14:41:58 +0200141 header.video_header().is_first_packet_in_frame = false;
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000142 header.header.markerBit = false;
143 // Insert padding frames.
144 for (int i = 0; i < 10; ++i) {
145 // Lose one packet from the 6th frame.
146 if (i == 5) {
147 ++header.header.sequenceNumber;
148 }
149 // Lose the 4th frame.
150 if (i == 3) {
151 header.header.sequenceNumber += 5;
152 } else {
153 if (i > 3 && i < 5) {
154 EXPECT_CALL(packet_request_callback_, ResendPackets(_, 5)).Times(1);
155 } else if (i >= 5) {
156 EXPECT_CALL(packet_request_callback_, ResendPackets(_, 6)).Times(1);
157 } else {
158 EXPECT_CALL(packet_request_callback_, ResendPackets(_, _)).Times(0);
159 }
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +0000160 InsertAndVerifyPaddingFrame(payload, &header);
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000161 }
162 clock_.AdvanceTimeMilliseconds(33);
163 header.header.timestamp += 3000;
164 }
165}
166
167TEST_F(TestVideoReceiver, PaddingOnlyAndVideo) {
168 EXPECT_EQ(0, receiver_->SetVideoProtection(kProtectionNack, true));
169 EXPECT_EQ(
170 0, receiver_->RegisterPacketRequestCallback(&packet_request_callback_));
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +0000171 const size_t kFrameSize = 1200;
172 const size_t kPaddingSize = 220;
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000173 const uint8_t payload[kFrameSize] = {0};
philipeldfbced62018-07-12 15:09:06 +0200174 WebRtcRTPHeader header = {};
pbos22993e12015-10-19 02:39:06 -0700175 header.frameType = kEmptyFrame;
philipelcb96ad82018-07-02 14:41:58 +0200176 header.video_header().is_first_packet_in_frame = false;
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000177 header.header.markerBit = false;
178 header.header.paddingLength = kPaddingSize;
179 header.header.payloadType = kUnusedPayloadType;
180 header.header.ssrc = 1;
181 header.header.headerLength = 12;
philipelcb96ad82018-07-02 14:41:58 +0200182 header.video_header().codec = kVideoCodecVP8;
philipel5ab67a52018-07-05 12:27:04 +0200183 header.video_header().vp8().pictureId = -1;
184 header.video_header().vp8().tl0PicIdx = -1;
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000185 for (int i = 0; i < 3; ++i) {
186 // Insert 2 video frames.
187 for (int j = 0; j < 2; ++j) {
188 if (i == 0 && j == 0) // First frame should be a key frame.
189 header.frameType = kVideoFrameKey;
190 else
191 header.frameType = kVideoFrameDelta;
philipelcb96ad82018-07-02 14:41:58 +0200192 header.video_header().is_first_packet_in_frame = true;
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000193 header.header.markerBit = true;
194 InsertAndVerifyDecodableFrame(payload, kFrameSize, &header);
195 clock_.AdvanceTimeMilliseconds(33);
196 header.header.timestamp += 3000;
197 }
198
199 // Insert 2 padding only frames.
pbos22993e12015-10-19 02:39:06 -0700200 header.frameType = kEmptyFrame;
philipelcb96ad82018-07-02 14:41:58 +0200201 header.video_header().is_first_packet_in_frame = false;
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000202 header.header.markerBit = false;
203 for (int j = 0; j < 2; ++j) {
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +0000204 // InsertAndVerifyPaddingFrame(payload, &header);
andresp@webrtc.orge401c2e2013-09-16 20:29:13 +0000205 clock_.AdvanceTimeMilliseconds(33);
206 header.header.timestamp += 3000;
207 }
208 }
209}
210
211TEST_F(TestVideoReceiver, ReceiverDelay) {
212 EXPECT_EQ(0, receiver_->SetMinReceiverDelay(0));
213 EXPECT_EQ(0, receiver_->SetMinReceiverDelay(5000));
214 EXPECT_EQ(-1, receiver_->SetMinReceiverDelay(-100));
215 EXPECT_EQ(-1, receiver_->SetMinReceiverDelay(10010));
216}
217
218} // namespace
219} // namespace vcm
220} // namespace webrtc