mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 1 | /* |
Yves Gerey | 665174f | 2018-06-19 15:03:05 +0200 | [diff] [blame] | 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 | */ |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 10 | |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 11 | #include <iterator> |
| 12 | #include <list> |
kwiberg | 84be511 | 2016-04-27 01:19:58 -0700 | [diff] [blame] | 13 | #include <memory> |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 14 | #include <set> |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 15 | |
Steve Anton | 91c2606 | 2019-03-28 10:56:11 -0700 | [diff] [blame] | 16 | #include "absl/algorithm/container.h" |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 17 | #include "api/call/transport.h" |
Niels Möller | 5fe9510 | 2019-03-04 16:49:25 +0100 | [diff] [blame] | 18 | #include "api/transport/field_trial_based_config.h" |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 19 | #include "call/rtp_stream_receiver_controller.h" |
| 20 | #include "call/rtx_receive_stream.h" |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 21 | #include "modules/rtp_rtcp/include/receive_statistics.h" |
| 22 | #include "modules/rtp_rtcp/include/rtp_rtcp.h" |
| 23 | #include "modules/rtp_rtcp/include/rtp_rtcp_defines.h" |
Niels Möller | 5fe9510 | 2019-03-04 16:49:25 +0100 | [diff] [blame] | 24 | #include "modules/rtp_rtcp/source/playout_delay_oracle.h" |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 25 | #include "modules/rtp_rtcp/source/rtp_packet_received.h" |
Niels Möller | 5fe9510 | 2019-03-04 16:49:25 +0100 | [diff] [blame] | 26 | #include "modules/rtp_rtcp/source/rtp_sender_video.h" |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 27 | #include "rtc_base/rate_limiter.h" |
| 28 | #include "test/gtest.h" |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 29 | |
danilchap | 6a6f089 | 2015-12-10 12:39:08 -0800 | [diff] [blame] | 30 | namespace webrtc { |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 31 | |
| 32 | const int kVideoNackListSize = 30; |
pbos@webrtc.org | 2f44673 | 2013-04-08 11:08:41 +0000 | [diff] [blame] | 33 | const uint32_t kTestSsrc = 3456; |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 34 | const uint32_t kTestRtxSsrc = kTestSsrc + 1; |
pbos@webrtc.org | 2f44673 | 2013-04-08 11:08:41 +0000 | [diff] [blame] | 35 | const uint16_t kTestSequenceNumber = 2345; |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 36 | const uint32_t kTestNumberOfPackets = 1350; |
| 37 | const int kTestNumberOfRtxPackets = 149; |
| 38 | const int kNumFrames = 30; |
Shao Changbin | e62202f | 2015-04-21 20:24:50 +0800 | [diff] [blame] | 39 | const int kPayloadType = 123; |
| 40 | const int kRtxPayloadType = 98; |
sprang | cd349d9 | 2016-07-13 09:11:28 -0700 | [diff] [blame] | 41 | const int64_t kMaxRttMs = 1000; |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 42 | |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 43 | class VerifyingMediaStream : public RtpPacketSinkInterface { |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 44 | public: |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 45 | VerifyingMediaStream() {} |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 46 | |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 47 | void OnRtpPacket(const RtpPacketReceived& packet) override { |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 48 | if (!sequence_numbers_.empty()) |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 49 | EXPECT_EQ(kTestSsrc, packet.Ssrc()); |
| 50 | |
| 51 | sequence_numbers_.push_back(packet.SequenceNumber()); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 52 | } |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 53 | std::list<uint16_t> sequence_numbers_; |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 54 | }; |
| 55 | |
| 56 | class RtxLoopBackTransport : public webrtc::Transport { |
| 57 | public: |
| 58 | explicit RtxLoopBackTransport(uint32_t rtx_ssrc) |
| 59 | : count_(0), |
| 60 | packet_loss_(0), |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 61 | consecutive_drop_start_(0), |
| 62 | consecutive_drop_end_(0), |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 63 | rtx_ssrc_(rtx_ssrc), |
| 64 | count_rtx_ssrc_(0), |
stefan@webrtc.org | 7bb8f02 | 2013-09-06 13:40:11 +0000 | [diff] [blame] | 65 | module_(NULL) {} |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 66 | |
Yves Gerey | 665174f | 2018-06-19 15:03:05 +0200 | [diff] [blame] | 67 | void SetSendModule(RtpRtcp* rtpRtcpModule) { module_ = rtpRtcpModule; } |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 68 | |
danilchap | 162abd3 | 2015-12-10 02:39:40 -0800 | [diff] [blame] | 69 | void DropEveryNthPacket(int n) { packet_loss_ = n; } |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 70 | |
| 71 | void DropConsecutivePackets(int start, int total) { |
| 72 | consecutive_drop_start_ = start; |
| 73 | consecutive_drop_end_ = start + total; |
| 74 | packet_loss_ = 0; |
| 75 | } |
| 76 | |
stefan | 1d8a506 | 2015-10-02 03:39:33 -0700 | [diff] [blame] | 77 | bool SendRtp(const uint8_t* data, |
| 78 | size_t len, |
| 79 | const PacketOptions& options) override { |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 80 | count_++; |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 81 | RtpPacketReceived packet; |
| 82 | if (!packet.Parse(data, len)) |
nisse | a646853 | 2017-09-08 05:00:54 -0700 | [diff] [blame] | 83 | return false; |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 84 | if (packet.Ssrc() == rtx_ssrc_) { |
| 85 | count_rtx_ssrc_++; |
| 86 | } else { |
| 87 | // For non-RTX packets only. |
terelius | e64fbce | 2015-09-17 03:19:45 -0700 | [diff] [blame] | 88 | expected_sequence_numbers_.insert(expected_sequence_numbers_.end(), |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 89 | packet.SequenceNumber()); |
terelius | e64fbce | 2015-09-17 03:19:45 -0700 | [diff] [blame] | 90 | } |
| 91 | if (packet_loss_ > 0) { |
| 92 | if ((count_ % packet_loss_) == 0) { |
pbos | 2d56668 | 2015-09-28 09:59:31 -0700 | [diff] [blame] | 93 | return true; |
terelius | e64fbce | 2015-09-17 03:19:45 -0700 | [diff] [blame] | 94 | } |
| 95 | } else if (count_ >= consecutive_drop_start_ && |
| 96 | count_ < consecutive_drop_end_) { |
pbos | 2d56668 | 2015-09-28 09:59:31 -0700 | [diff] [blame] | 97 | return true; |
terelius | e64fbce | 2015-09-17 03:19:45 -0700 | [diff] [blame] | 98 | } |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 99 | EXPECT_TRUE(stream_receiver_controller_.OnRtpPacket(packet)); |
pbos | 2d56668 | 2015-09-28 09:59:31 -0700 | [diff] [blame] | 100 | return true; |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 101 | } |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 102 | |
pbos | 2d56668 | 2015-09-28 09:59:31 -0700 | [diff] [blame] | 103 | bool SendRtcp(const uint8_t* data, size_t len) override { |
nisse | 479d3d7 | 2017-09-13 07:53:37 -0700 | [diff] [blame] | 104 | module_->IncomingRtcpPacket((const uint8_t*)data, len); |
| 105 | return true; |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 106 | } |
| 107 | int count_; |
| 108 | int packet_loss_; |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 109 | int consecutive_drop_start_; |
| 110 | int consecutive_drop_end_; |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 111 | uint32_t rtx_ssrc_; |
| 112 | int count_rtx_ssrc_; |
| 113 | RtpRtcp* module_; |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 114 | RtpStreamReceiverController stream_receiver_controller_; |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 115 | std::set<uint16_t> expected_sequence_numbers_; |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 116 | }; |
| 117 | |
stefan@webrtc.org | 4121146 | 2013-03-18 15:00:50 +0000 | [diff] [blame] | 118 | class RtpRtcpRtxNackTest : public ::testing::Test { |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 119 | protected: |
stefan@webrtc.org | 4121146 | 2013-03-18 15:00:50 +0000 | [diff] [blame] | 120 | RtpRtcpRtxNackTest() |
magjed | f3feeff | 2016-11-25 06:40:25 -0800 | [diff] [blame] | 121 | : rtp_rtcp_module_(nullptr), |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 122 | transport_(kTestRtxSsrc), |
Yves Gerey | 665174f | 2018-06-19 15:03:05 +0200 | [diff] [blame] | 123 | rtx_stream_(&media_stream_, rtx_associated_payload_types_, kTestSsrc), |
sprang | cd349d9 | 2016-07-13 09:11:28 -0700 | [diff] [blame] | 124 | fake_clock(123456), |
Erik Språng | 737336d | 2016-07-29 12:59:36 +0200 | [diff] [blame] | 125 | retransmission_rate_limiter_(&fake_clock, kMaxRttMs) {} |
Danil Chapovalov | dd7e284 | 2018-03-09 15:37:03 +0000 | [diff] [blame] | 126 | ~RtpRtcpRtxNackTest() override {} |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 127 | |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 +0000 | [diff] [blame] | 128 | void SetUp() override { |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 129 | RtpRtcp::Configuration configuration; |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 130 | configuration.audio = false; |
| 131 | configuration.clock = &fake_clock; |
Niels Möller | 0d210ee | 2019-08-07 16:16:45 +0200 | [diff] [blame] | 132 | receive_statistics_ = ReceiveStatistics::Create(&fake_clock); |
wu@webrtc.org | 822fbd8 | 2013-08-15 23:38:54 +0000 | [diff] [blame] | 133 | configuration.receive_statistics = receive_statistics_.get(); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 134 | configuration.outgoing_transport = &transport_; |
Erik Språng | 737336d | 2016-07-29 12:59:36 +0200 | [diff] [blame] | 135 | configuration.retransmission_rate_limiter = &retransmission_rate_limiter_; |
Erik Språng | 54d5d2c | 2019-08-20 17:22:36 +0200 | [diff] [blame] | 136 | configuration.local_media_ssrc = kTestSsrc; |
Erik Språng | 6841d25 | 2019-10-15 14:29:11 +0200 | [diff] [blame] | 137 | configuration.rtx_send_ssrc = kTestRtxSsrc; |
Danil Chapovalov | c44f6cc | 2019-03-06 11:31:09 +0100 | [diff] [blame] | 138 | rtp_rtcp_module_ = RtpRtcp::Create(configuration); |
Erik Språng | dc34a25 | 2019-10-04 15:17:29 +0200 | [diff] [blame] | 139 | FieldTrialBasedConfig field_trials; |
| 140 | RTPSenderVideo::Config video_config; |
| 141 | video_config.clock = &fake_clock; |
| 142 | video_config.rtp_sender = rtp_rtcp_module_->RtpSender(); |
| 143 | video_config.playout_delay_oracle = &playout_delay_oracle_; |
| 144 | video_config.field_trials = &field_trials; |
| 145 | rtp_sender_video_ = std::make_unique<RTPSenderVideo>(video_config); |
pbos | da903ea | 2015-10-02 02:36:56 -0700 | [diff] [blame] | 146 | rtp_rtcp_module_->SetRTCPStatus(RtcpMode::kCompound); |
pbos@webrtc.org | d16e839 | 2014-12-19 13:49:55 +0000 | [diff] [blame] | 147 | rtp_rtcp_module_->SetStorePacketsStatus(true, 600); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 148 | EXPECT_EQ(0, rtp_rtcp_module_->SetSendingStatus(true)); |
pbos@webrtc.org | d16e839 | 2014-12-19 13:49:55 +0000 | [diff] [blame] | 149 | rtp_rtcp_module_->SetSequenceNumber(kTestSequenceNumber); |
| 150 | rtp_rtcp_module_->SetStartTimestamp(111111); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 151 | |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 152 | // Used for NACK processing. |
| 153 | // TODO(nisse): Unclear on which side? It's confusing to use a |
| 154 | // single rtp_rtcp module for both send and receive side. |
| 155 | rtp_rtcp_module_->SetRemoteSSRC(kTestSsrc); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 156 | |
Shao Changbin | e62202f | 2015-04-21 20:24:50 +0800 | [diff] [blame] | 157 | rtp_rtcp_module_->SetRtxSendPayloadType(kRtxPayloadType, kPayloadType); |
Niels Möller | 5fe9510 | 2019-03-04 16:49:25 +0100 | [diff] [blame] | 158 | transport_.SetSendModule(rtp_rtcp_module_.get()); |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 159 | media_receiver_ = transport_.stream_receiver_controller_.CreateReceiver( |
| 160 | kTestSsrc, &media_stream_); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 161 | |
Danil Chapovalov | 51bf200 | 2019-10-11 10:53:27 +0200 | [diff] [blame] | 162 | for (size_t n = 0; n < sizeof(payload_data); n++) { |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 163 | payload_data[n] = n % 10; |
| 164 | } |
| 165 | } |
| 166 | |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 167 | int BuildNackList(uint16_t* nack_list) { |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 168 | media_stream_.sequence_numbers_.sort(); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 169 | std::list<uint16_t> missing_sequence_numbers; |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 170 | std::list<uint16_t>::iterator it = media_stream_.sequence_numbers_.begin(); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 171 | |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 172 | while (it != media_stream_.sequence_numbers_.end()) { |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 173 | uint16_t sequence_number_1 = *it; |
| 174 | ++it; |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 175 | if (it != media_stream_.sequence_numbers_.end()) { |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 176 | uint16_t sequence_number_2 = *it; |
| 177 | // Add all missing sequence numbers to list |
danilchap | 162abd3 | 2015-12-10 02:39:40 -0800 | [diff] [blame] | 178 | for (uint16_t i = sequence_number_1 + 1; i < sequence_number_2; ++i) { |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 179 | missing_sequence_numbers.push_back(i); |
| 180 | } |
| 181 | } |
| 182 | } |
| 183 | int n = 0; |
| 184 | for (it = missing_sequence_numbers.begin(); |
danilchap | 162abd3 | 2015-12-10 02:39:40 -0800 | [diff] [blame] | 185 | it != missing_sequence_numbers.end(); ++it) { |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 186 | nack_list[n++] = (*it); |
| 187 | } |
| 188 | return n; |
| 189 | } |
| 190 | |
| 191 | bool ExpectedPacketsReceived() { |
| 192 | std::list<uint16_t> received_sorted; |
Steve Anton | 91c2606 | 2019-03-28 10:56:11 -0700 | [diff] [blame] | 193 | absl::c_copy(media_stream_.sequence_numbers_, |
| 194 | std::back_inserter(received_sorted)); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 195 | received_sorted.sort(); |
Steve Anton | 91c2606 | 2019-03-28 10:56:11 -0700 | [diff] [blame] | 196 | return absl::c_equal(received_sorted, |
| 197 | transport_.expected_sequence_numbers_); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 198 | } |
| 199 | |
| 200 | void RunRtxTest(RtxMode rtx_method, int loss) { |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 201 | rtx_receiver_ = transport_.stream_receiver_controller_.CreateReceiver( |
| 202 | kTestRtxSsrc, &rtx_stream_); |
pbos@webrtc.org | 0b0c241 | 2015-01-13 14:15:15 +0000 | [diff] [blame] | 203 | rtp_rtcp_module_->SetRtxSendStatus(rtx_method); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 204 | transport_.DropEveryNthPacket(loss); |
| 205 | uint32_t timestamp = 3000; |
| 206 | uint16_t nack_list[kVideoNackListSize]; |
| 207 | for (int frame = 0; frame < kNumFrames; ++frame) { |
Sami Kalliomäki | 426a80c | 2018-08-08 11:37:59 +0200 | [diff] [blame] | 208 | RTPVideoHeader video_header; |
Niels Möller | 5fe9510 | 2019-03-04 16:49:25 +0100 | [diff] [blame] | 209 | EXPECT_TRUE(rtp_rtcp_module_->OnSendingRtpFrame(timestamp, timestamp / 90, |
| 210 | kPayloadType, false)); |
Danil Chapovalov | 51bf200 | 2019-10-11 10:53:27 +0200 | [diff] [blame] | 211 | video_header.frame_type = VideoFrameType::kVideoFrameDelta; |
Niels Möller | 5fe9510 | 2019-03-04 16:49:25 +0100 | [diff] [blame] | 212 | EXPECT_TRUE(rtp_sender_video_->SendVideo( |
Danil Chapovalov | 51bf200 | 2019-10-11 10:53:27 +0200 | [diff] [blame] | 213 | kPayloadType, VideoCodecType::kVideoCodecGeneric, timestamp, |
| 214 | timestamp / 90, payload_data, nullptr, video_header, 0)); |
sprang@webrtc.org | 43c8839 | 2015-01-29 09:09:17 +0000 | [diff] [blame] | 215 | // Min required delay until retransmit = 5 + RTT ms (RTT = 0). |
| 216 | fake_clock.AdvanceTimeMilliseconds(5); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 217 | int length = BuildNackList(nack_list); |
| 218 | if (length > 0) |
| 219 | rtp_rtcp_module_->SendNACK(nack_list, length); |
sprang@webrtc.org | 43c8839 | 2015-01-29 09:09:17 +0000 | [diff] [blame] | 220 | fake_clock.AdvanceTimeMilliseconds(28); // 33ms - 5ms delay. |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 221 | rtp_rtcp_module_->Process(); |
| 222 | // Prepare next frame. |
| 223 | timestamp += 3000; |
| 224 | } |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 225 | media_stream_.sequence_numbers_.sort(); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 226 | } |
| 227 | |
kwiberg | 84be511 | 2016-04-27 01:19:58 -0700 | [diff] [blame] | 228 | std::unique_ptr<ReceiveStatistics> receive_statistics_; |
Niels Möller | 5fe9510 | 2019-03-04 16:49:25 +0100 | [diff] [blame] | 229 | std::unique_ptr<RtpRtcp> rtp_rtcp_module_; |
| 230 | PlayoutDelayOracle playout_delay_oracle_; |
| 231 | std::unique_ptr<RTPSenderVideo> rtp_sender_video_; |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 232 | RtxLoopBackTransport transport_; |
Yves Gerey | 665174f | 2018-06-19 15:03:05 +0200 | [diff] [blame] | 233 | const std::map<int, int> rtx_associated_payload_types_ = { |
| 234 | {kRtxPayloadType, kPayloadType}}; |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 235 | VerifyingMediaStream media_stream_; |
| 236 | RtxReceiveStream rtx_stream_; |
danilchap | 162abd3 | 2015-12-10 02:39:40 -0800 | [diff] [blame] | 237 | uint8_t payload_data[65000]; |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 238 | SimulatedClock fake_clock; |
Erik Språng | 737336d | 2016-07-29 12:59:36 +0200 | [diff] [blame] | 239 | RateLimiter retransmission_rate_limiter_; |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 240 | std::unique_ptr<RtpStreamReceiverInterface> media_receiver_; |
| 241 | std::unique_ptr<RtpStreamReceiverInterface> rtx_receiver_; |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 242 | }; |
| 243 | |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 244 | TEST_F(RtpRtcpRtxNackTest, LongNackList) { |
| 245 | const int kNumPacketsToDrop = 900; |
| 246 | const int kNumRequiredRtcp = 4; |
pbos@webrtc.org | 2f44673 | 2013-04-08 11:08:41 +0000 | [diff] [blame] | 247 | uint32_t timestamp = 3000; |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 248 | uint16_t nack_list[kNumPacketsToDrop]; |
| 249 | // Disable StorePackets to be able to set a larger packet history. |
pbos@webrtc.org | d16e839 | 2014-12-19 13:49:55 +0000 | [diff] [blame] | 250 | rtp_rtcp_module_->SetStorePacketsStatus(false, 0); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 251 | // Enable StorePackets with a packet history of 2000 packets. |
pbos@webrtc.org | d16e839 | 2014-12-19 13:49:55 +0000 | [diff] [blame] | 252 | rtp_rtcp_module_->SetStorePacketsStatus(true, 2000); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 253 | // Drop 900 packets from the second one so that we get a NACK list which is |
| 254 | // big enough to require 4 RTCP packets to be fully transmitted to the sender. |
| 255 | transport_.DropConsecutivePackets(2, kNumPacketsToDrop); |
| 256 | // Send 30 frames which at the default size is roughly what we need to get |
| 257 | // enough packets. |
| 258 | for (int frame = 0; frame < kNumFrames; ++frame) { |
Sami Kalliomäki | 426a80c | 2018-08-08 11:37:59 +0200 | [diff] [blame] | 259 | RTPVideoHeader video_header; |
Niels Möller | 5fe9510 | 2019-03-04 16:49:25 +0100 | [diff] [blame] | 260 | EXPECT_TRUE(rtp_rtcp_module_->OnSendingRtpFrame(timestamp, timestamp / 90, |
| 261 | kPayloadType, false)); |
Danil Chapovalov | 51bf200 | 2019-10-11 10:53:27 +0200 | [diff] [blame] | 262 | video_header.frame_type = VideoFrameType::kVideoFrameDelta; |
Niels Möller | 5fe9510 | 2019-03-04 16:49:25 +0100 | [diff] [blame] | 263 | EXPECT_TRUE(rtp_sender_video_->SendVideo( |
Danil Chapovalov | 51bf200 | 2019-10-11 10:53:27 +0200 | [diff] [blame] | 264 | kPayloadType, VideoCodecType::kVideoCodecGeneric, timestamp, |
| 265 | timestamp / 90, payload_data, nullptr, video_header, 0)); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 266 | // Prepare next frame. |
| 267 | timestamp += 3000; |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 268 | fake_clock.AdvanceTimeMilliseconds(33); |
| 269 | rtp_rtcp_module_->Process(); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 270 | } |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 271 | EXPECT_FALSE(transport_.expected_sequence_numbers_.empty()); |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 272 | EXPECT_FALSE(media_stream_.sequence_numbers_.empty()); |
| 273 | size_t last_receive_count = media_stream_.sequence_numbers_.size(); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 274 | int length = BuildNackList(nack_list); |
| 275 | for (int i = 0; i < kNumRequiredRtcp - 1; ++i) { |
| 276 | rtp_rtcp_module_->SendNACK(nack_list, length); |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 277 | EXPECT_GT(media_stream_.sequence_numbers_.size(), last_receive_count); |
| 278 | last_receive_count = media_stream_.sequence_numbers_.size(); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 279 | EXPECT_FALSE(ExpectedPacketsReceived()); |
| 280 | } |
| 281 | rtp_rtcp_module_->SendNACK(nack_list, length); |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 282 | EXPECT_GT(media_stream_.sequence_numbers_.size(), last_receive_count); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 283 | EXPECT_TRUE(ExpectedPacketsReceived()); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 284 | } |
| 285 | |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 286 | TEST_F(RtpRtcpRtxNackTest, RtxNack) { |
| 287 | RunRtxTest(kRtxRetransmitted, 10); |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 288 | EXPECT_EQ(kTestSequenceNumber, *(media_stream_.sequence_numbers_.begin())); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 289 | EXPECT_EQ(kTestSequenceNumber + kTestNumberOfPackets - 1, |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 290 | *(media_stream_.sequence_numbers_.rbegin())); |
| 291 | EXPECT_EQ(kTestNumberOfPackets, media_stream_.sequence_numbers_.size()); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 292 | EXPECT_EQ(kTestNumberOfRtxPackets, transport_.count_rtx_ssrc_); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 293 | EXPECT_TRUE(ExpectedPacketsReceived()); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 294 | } |
danilchap | 6a6f089 | 2015-12-10 12:39:08 -0800 | [diff] [blame] | 295 | |
| 296 | } // namespace webrtc |