mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 1 | /* |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [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 | |
| 11 | #include <algorithm> |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 12 | #include <iterator> |
| 13 | #include <list> |
kwiberg | 84be511 | 2016-04-27 01:19:58 -0700 | [diff] [blame] | 14 | #include <memory> |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 15 | #include <set> |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 16 | |
aleloi | a8eb756 | 2016-11-28 07:02:13 -0800 | [diff] [blame] | 17 | #include "webrtc/api/call/transport.h" |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 18 | #include "webrtc/call/rtp_stream_receiver_controller.h" |
| 19 | #include "webrtc/call/rtx_receive_stream.h" |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 20 | #include "webrtc/common_types.h" |
Henrik Kjellander | ff761fb | 2015-11-04 08:31:52 +0100 | [diff] [blame] | 21 | #include "webrtc/modules/rtp_rtcp/include/receive_statistics.h" |
Henrik Kjellander | ff761fb | 2015-11-04 08:31:52 +0100 | [diff] [blame] | 22 | #include "webrtc/modules/rtp_rtcp/include/rtp_rtcp.h" |
| 23 | #include "webrtc/modules/rtp_rtcp/include/rtp_rtcp_defines.h" |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 24 | #include "webrtc/modules/rtp_rtcp/source/rtp_packet_received.h" |
Edward Lemur | c20978e | 2017-07-06 19:44:34 +0200 | [diff] [blame] | 25 | #include "webrtc/rtc_base/rate_limiter.h" |
kwiberg | ac9f876 | 2016-09-30 22:29:43 -0700 | [diff] [blame] | 26 | #include "webrtc/test/gtest.h" |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 27 | |
danilchap | 6a6f089 | 2015-12-10 12:39:08 -0800 | [diff] [blame] | 28 | namespace webrtc { |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 29 | |
| 30 | const int kVideoNackListSize = 30; |
pbos@webrtc.org | 2f44673 | 2013-04-08 11:08:41 +0000 | [diff] [blame] | 31 | const uint32_t kTestSsrc = 3456; |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 32 | const uint32_t kTestRtxSsrc = kTestSsrc + 1; |
pbos@webrtc.org | 2f44673 | 2013-04-08 11:08:41 +0000 | [diff] [blame] | 33 | const uint16_t kTestSequenceNumber = 2345; |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 34 | const uint32_t kTestNumberOfPackets = 1350; |
| 35 | const int kTestNumberOfRtxPackets = 149; |
| 36 | const int kNumFrames = 30; |
Shao Changbin | e62202f | 2015-04-21 20:24:50 +0800 | [diff] [blame] | 37 | const int kPayloadType = 123; |
| 38 | const int kRtxPayloadType = 98; |
sprang | cd349d9 | 2016-07-13 09:11:28 -0700 | [diff] [blame] | 39 | const int64_t kMaxRttMs = 1000; |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 40 | |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 41 | class VerifyingMediaStream : public RtpPacketSinkInterface { |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 42 | public: |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 43 | VerifyingMediaStream() {} |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 44 | |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 45 | void OnRtpPacket(const RtpPacketReceived& packet) override { |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 46 | if (!sequence_numbers_.empty()) |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 47 | EXPECT_EQ(kTestSsrc, packet.Ssrc()); |
| 48 | |
| 49 | sequence_numbers_.push_back(packet.SequenceNumber()); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 50 | } |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 51 | std::list<uint16_t> sequence_numbers_; |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 52 | }; |
| 53 | |
| 54 | class RtxLoopBackTransport : public webrtc::Transport { |
| 55 | public: |
| 56 | explicit RtxLoopBackTransport(uint32_t rtx_ssrc) |
| 57 | : count_(0), |
| 58 | packet_loss_(0), |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 59 | consecutive_drop_start_(0), |
| 60 | consecutive_drop_end_(0), |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 61 | rtx_ssrc_(rtx_ssrc), |
| 62 | count_rtx_ssrc_(0), |
stefan@webrtc.org | 7bb8f02 | 2013-09-06 13:40:11 +0000 | [diff] [blame] | 63 | module_(NULL) {} |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 64 | |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 65 | void SetSendModule(RtpRtcp* rtpRtcpModule) { |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 66 | module_ = rtpRtcpModule; |
| 67 | } |
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), |
| 123 | rtx_stream_(&media_stream_, |
| 124 | rtx_associated_payload_types_, |
| 125 | kTestSsrc), |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 126 | payload_data_length(sizeof(payload_data)), |
sprang | cd349d9 | 2016-07-13 09:11:28 -0700 | [diff] [blame] | 127 | fake_clock(123456), |
Erik Språng | 737336d | 2016-07-29 12:59:36 +0200 | [diff] [blame] | 128 | retransmission_rate_limiter_(&fake_clock, kMaxRttMs) {} |
stefan@webrtc.org | 4121146 | 2013-03-18 15:00:50 +0000 | [diff] [blame] | 129 | ~RtpRtcpRtxNackTest() {} |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 130 | |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 +0000 | [diff] [blame] | 131 | void SetUp() override { |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 132 | RtpRtcp::Configuration configuration; |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 133 | configuration.audio = false; |
| 134 | configuration.clock = &fake_clock; |
wu@webrtc.org | 822fbd8 | 2013-08-15 23:38:54 +0000 | [diff] [blame] | 135 | receive_statistics_.reset(ReceiveStatistics::Create(&fake_clock)); |
| 136 | configuration.receive_statistics = receive_statistics_.get(); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 137 | configuration.outgoing_transport = &transport_; |
Erik Språng | 737336d | 2016-07-29 12:59:36 +0200 | [diff] [blame] | 138 | configuration.retransmission_rate_limiter = &retransmission_rate_limiter_; |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 139 | rtp_rtcp_module_ = RtpRtcp::CreateRtpRtcp(configuration); |
| 140 | |
stefan@webrtc.org | ef92755 | 2014-06-05 08:25:29 +0000 | [diff] [blame] | 141 | rtp_rtcp_module_->SetSSRC(kTestSsrc); |
pbos | da903ea | 2015-10-02 02:36:56 -0700 | [diff] [blame] | 142 | rtp_rtcp_module_->SetRTCPStatus(RtcpMode::kCompound); |
pbos@webrtc.org | d16e839 | 2014-12-19 13:49:55 +0000 | [diff] [blame] | 143 | rtp_rtcp_module_->SetStorePacketsStatus(true, 600); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 144 | EXPECT_EQ(0, rtp_rtcp_module_->SetSendingStatus(true)); |
pbos@webrtc.org | d16e839 | 2014-12-19 13:49:55 +0000 | [diff] [blame] | 145 | rtp_rtcp_module_->SetSequenceNumber(kTestSequenceNumber); |
| 146 | rtp_rtcp_module_->SetStartTimestamp(111111); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 147 | |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 148 | // Used for NACK processing. |
| 149 | // TODO(nisse): Unclear on which side? It's confusing to use a |
| 150 | // single rtp_rtcp module for both send and receive side. |
| 151 | rtp_rtcp_module_->SetRemoteSSRC(kTestSsrc); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 152 | |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 153 | rtp_rtcp_module_->RegisterVideoSendPayload(kPayloadType, "video"); |
Shao Changbin | e62202f | 2015-04-21 20:24:50 +0800 | [diff] [blame] | 154 | rtp_rtcp_module_->SetRtxSendPayloadType(kRtxPayloadType, kPayloadType); |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 155 | transport_.SetSendModule(rtp_rtcp_module_); |
| 156 | media_receiver_ = transport_.stream_receiver_controller_.CreateReceiver( |
| 157 | kTestSsrc, &media_stream_); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 158 | |
pkasting@chromium.org | 4591fbd | 2014-11-20 22:28:14 +0000 | [diff] [blame] | 159 | for (size_t n = 0; n < payload_data_length; n++) { |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 160 | payload_data[n] = n % 10; |
| 161 | } |
| 162 | } |
| 163 | |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 164 | int BuildNackList(uint16_t* nack_list) { |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 165 | media_stream_.sequence_numbers_.sort(); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 166 | std::list<uint16_t> missing_sequence_numbers; |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 167 | std::list<uint16_t>::iterator it = media_stream_.sequence_numbers_.begin(); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 168 | |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 169 | while (it != media_stream_.sequence_numbers_.end()) { |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 170 | uint16_t sequence_number_1 = *it; |
| 171 | ++it; |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 172 | if (it != media_stream_.sequence_numbers_.end()) { |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 173 | uint16_t sequence_number_2 = *it; |
| 174 | // Add all missing sequence numbers to list |
danilchap | 162abd3 | 2015-12-10 02:39:40 -0800 | [diff] [blame] | 175 | 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] | 176 | missing_sequence_numbers.push_back(i); |
| 177 | } |
| 178 | } |
| 179 | } |
| 180 | int n = 0; |
| 181 | for (it = missing_sequence_numbers.begin(); |
danilchap | 162abd3 | 2015-12-10 02:39:40 -0800 | [diff] [blame] | 182 | it != missing_sequence_numbers.end(); ++it) { |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 183 | nack_list[n++] = (*it); |
| 184 | } |
| 185 | return n; |
| 186 | } |
| 187 | |
| 188 | bool ExpectedPacketsReceived() { |
| 189 | std::list<uint16_t> received_sorted; |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 190 | std::copy(media_stream_.sequence_numbers_.begin(), |
| 191 | media_stream_.sequence_numbers_.end(), |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 192 | std::back_inserter(received_sorted)); |
| 193 | received_sorted.sort(); |
terelius | e64fbce | 2015-09-17 03:19:45 -0700 | [diff] [blame] | 194 | return received_sorted.size() == |
| 195 | transport_.expected_sequence_numbers_.size() && |
| 196 | std::equal(received_sorted.begin(), received_sorted.end(), |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 197 | transport_.expected_sequence_numbers_.begin()); |
| 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); |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 204 | rtp_rtcp_module_->SetRtxSsrc(kTestRtxSsrc); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 205 | transport_.DropEveryNthPacket(loss); |
| 206 | uint32_t timestamp = 3000; |
| 207 | uint16_t nack_list[kVideoNackListSize]; |
| 208 | for (int frame = 0; frame < kNumFrames; ++frame) { |
Sergey Ulanov | 525df3f | 2016-08-02 17:46:41 -0700 | [diff] [blame] | 209 | EXPECT_TRUE(rtp_rtcp_module_->SendOutgoingData( |
| 210 | webrtc::kVideoFrameDelta, kPayloadType, timestamp, timestamp / 90, |
| 211 | payload_data, payload_data_length, nullptr, nullptr, nullptr)); |
sprang@webrtc.org | 43c8839 | 2015-01-29 09:09:17 +0000 | [diff] [blame] | 212 | // Min required delay until retransmit = 5 + RTT ms (RTT = 0). |
| 213 | fake_clock.AdvanceTimeMilliseconds(5); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 214 | int length = BuildNackList(nack_list); |
| 215 | if (length > 0) |
| 216 | rtp_rtcp_module_->SendNACK(nack_list, length); |
sprang@webrtc.org | 43c8839 | 2015-01-29 09:09:17 +0000 | [diff] [blame] | 217 | fake_clock.AdvanceTimeMilliseconds(28); // 33ms - 5ms delay. |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 218 | rtp_rtcp_module_->Process(); |
| 219 | // Prepare next frame. |
| 220 | timestamp += 3000; |
| 221 | } |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 222 | media_stream_.sequence_numbers_.sort(); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 223 | } |
| 224 | |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 +0000 | [diff] [blame] | 225 | void TearDown() override { delete rtp_rtcp_module_; } |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 226 | |
kwiberg | 84be511 | 2016-04-27 01:19:58 -0700 | [diff] [blame] | 227 | std::unique_ptr<ReceiveStatistics> receive_statistics_; |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 228 | RtpRtcp* rtp_rtcp_module_; |
| 229 | RtxLoopBackTransport transport_; |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 230 | const std::map<int, int> rtx_associated_payload_types_ = |
| 231 | {{kRtxPayloadType, kPayloadType}}; |
| 232 | VerifyingMediaStream media_stream_; |
| 233 | RtxReceiveStream rtx_stream_; |
danilchap | 162abd3 | 2015-12-10 02:39:40 -0800 | [diff] [blame] | 234 | uint8_t payload_data[65000]; |
pkasting@chromium.org | 4591fbd | 2014-11-20 22:28:14 +0000 | [diff] [blame] | 235 | size_t payload_data_length; |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 236 | SimulatedClock fake_clock; |
Erik Språng | 737336d | 2016-07-29 12:59:36 +0200 | [diff] [blame] | 237 | RateLimiter retransmission_rate_limiter_; |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 238 | std::unique_ptr<RtpStreamReceiverInterface> media_receiver_; |
| 239 | std::unique_ptr<RtpStreamReceiverInterface> rtx_receiver_; |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 240 | }; |
| 241 | |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 242 | TEST_F(RtpRtcpRtxNackTest, LongNackList) { |
| 243 | const int kNumPacketsToDrop = 900; |
| 244 | const int kNumRequiredRtcp = 4; |
pbos@webrtc.org | 2f44673 | 2013-04-08 11:08:41 +0000 | [diff] [blame] | 245 | uint32_t timestamp = 3000; |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 246 | uint16_t nack_list[kNumPacketsToDrop]; |
| 247 | // Disable StorePackets to be able to set a larger packet history. |
pbos@webrtc.org | d16e839 | 2014-12-19 13:49:55 +0000 | [diff] [blame] | 248 | rtp_rtcp_module_->SetStorePacketsStatus(false, 0); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 249 | // Enable StorePackets with a packet history of 2000 packets. |
pbos@webrtc.org | d16e839 | 2014-12-19 13:49:55 +0000 | [diff] [blame] | 250 | rtp_rtcp_module_->SetStorePacketsStatus(true, 2000); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 251 | // Drop 900 packets from the second one so that we get a NACK list which is |
| 252 | // big enough to require 4 RTCP packets to be fully transmitted to the sender. |
| 253 | transport_.DropConsecutivePackets(2, kNumPacketsToDrop); |
| 254 | // Send 30 frames which at the default size is roughly what we need to get |
| 255 | // enough packets. |
| 256 | for (int frame = 0; frame < kNumFrames; ++frame) { |
Sergey Ulanov | 525df3f | 2016-08-02 17:46:41 -0700 | [diff] [blame] | 257 | EXPECT_TRUE(rtp_rtcp_module_->SendOutgoingData( |
| 258 | webrtc::kVideoFrameDelta, kPayloadType, timestamp, timestamp / 90, |
| 259 | payload_data, payload_data_length, nullptr, nullptr, nullptr)); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 260 | // Prepare next frame. |
| 261 | timestamp += 3000; |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 262 | fake_clock.AdvanceTimeMilliseconds(33); |
| 263 | rtp_rtcp_module_->Process(); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 264 | } |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 265 | EXPECT_FALSE(transport_.expected_sequence_numbers_.empty()); |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 266 | EXPECT_FALSE(media_stream_.sequence_numbers_.empty()); |
| 267 | size_t last_receive_count = media_stream_.sequence_numbers_.size(); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 268 | int length = BuildNackList(nack_list); |
| 269 | for (int i = 0; i < kNumRequiredRtcp - 1; ++i) { |
| 270 | rtp_rtcp_module_->SendNACK(nack_list, length); |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 271 | EXPECT_GT(media_stream_.sequence_numbers_.size(), last_receive_count); |
| 272 | last_receive_count = media_stream_.sequence_numbers_.size(); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 273 | EXPECT_FALSE(ExpectedPacketsReceived()); |
| 274 | } |
| 275 | rtp_rtcp_module_->SendNACK(nack_list, length); |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 276 | EXPECT_GT(media_stream_.sequence_numbers_.size(), last_receive_count); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 277 | EXPECT_TRUE(ExpectedPacketsReceived()); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 278 | } |
| 279 | |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 280 | TEST_F(RtpRtcpRtxNackTest, RtxNack) { |
| 281 | RunRtxTest(kRtxRetransmitted, 10); |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 282 | EXPECT_EQ(kTestSequenceNumber, *(media_stream_.sequence_numbers_.begin())); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 283 | EXPECT_EQ(kTestSequenceNumber + kTestNumberOfPackets - 1, |
nisse | f54573b | 2017-09-13 07:13:57 -0700 | [diff] [blame] | 284 | *(media_stream_.sequence_numbers_.rbegin())); |
| 285 | EXPECT_EQ(kTestNumberOfPackets, media_stream_.sequence_numbers_.size()); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 286 | EXPECT_EQ(kTestNumberOfRtxPackets, transport_.count_rtx_ssrc_); |
mikhal@webrtc.org | 6cfa390 | 2013-05-15 20:17:43 +0000 | [diff] [blame] | 287 | EXPECT_TRUE(ExpectedPacketsReceived()); |
mikhal@webrtc.org | bda7f30 | 2013-03-15 23:21:52 +0000 | [diff] [blame] | 288 | } |
danilchap | 6a6f089 | 2015-12-10 12:39:08 -0800 | [diff] [blame] | 289 | |
| 290 | } // namespace webrtc |