blob: c4f665677f609bd377927effbc7fad08469c2bec [file] [log] [blame]
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +00001/*
2 * Copyright (c) 2014 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
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020011#include "modules/audio_coding/test/PacketLossTest.h"
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +000012
kwiberg37478382016-02-14 20:40:57 -080013#include <memory>
14
Ali Tofigh714e3cb2022-07-20 12:53:07 +020015#include "absl/strings/string_view.h"
Karl Wiberg5817d3d2018-04-06 10:06:42 +020016#include "api/audio_codecs/builtin_audio_decoder_factory.h"
Jonas Olsson366a50c2018-09-06 13:41:30 +020017#include "rtc_base/strings/string_builder.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020018#include "test/gtest.h"
Steve Anton10542f22019-01-11 09:11:00 -080019#include "test/testsupport/file_utils.h"
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +000020
21namespace webrtc {
22
23ReceiverWithPacketLoss::ReceiverWithPacketLoss()
24 : loss_rate_(0),
25 burst_length_(1),
26 packet_counter_(0),
27 lost_packet_counter_(0),
Yves Gerey665174f2018-06-19 15:03:05 +020028 burst_lost_counter_(burst_length_) {}
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +000029
Henrik Lundin84f75692023-02-01 12:07:10 +000030void ReceiverWithPacketLoss::Setup(acm2::AcmReceiver* acm_receiver,
Yves Gerey665174f2018-06-19 15:03:05 +020031 RTPStream* rtpStream,
Ali Tofigh714e3cb2022-07-20 12:53:07 +020032 absl::string_view out_file_name,
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +000033 int channels,
Fredrik Solenberg657b2962018-12-05 10:30:25 +010034 int file_num,
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +000035 int loss_rate,
36 int burst_length) {
37 loss_rate_ = loss_rate;
38 burst_length_ = burst_length;
39 burst_lost_counter_ = burst_length_; // To prevent first packet gets lost.
Jonas Olsson366a50c2018-09-06 13:41:30 +020040 rtc::StringBuilder ss;
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +000041 ss << out_file_name << "_" << loss_rate_ << "_" << burst_length_ << "_";
Henrik Lundin84f75692023-02-01 12:07:10 +000042 Receiver::Setup(acm_receiver, rtpStream, ss.str(), channels, file_num);
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +000043}
44
45bool ReceiverWithPacketLoss::IncomingPacket() {
46 if (!_rtpStream->EndOfFile()) {
47 if (packet_counter_ == 0) {
Niels Möllerbf474952019-02-18 12:00:06 +010048 _realPayloadSizeBytes = _rtpStream->Read(&_rtpHeader, _incomingPayload,
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +000049 _payloadSizeBytes, &_nextTime);
50 if (_realPayloadSizeBytes == 0) {
51 if (_rtpStream->EndOfFile()) {
52 packet_counter_ = 0;
53 return true;
54 } else {
55 return false;
56 }
57 }
58 }
59
60 if (!PacketLost()) {
Henrik Lundin84f75692023-02-01 12:07:10 +000061 _acm_receiver->InsertPacket(
62 _rtpHeader, rtc::ArrayView<const uint8_t>(_incomingPayload,
63 _realPayloadSizeBytes));
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +000064 }
65 packet_counter_++;
Niels Möllerbf474952019-02-18 12:00:06 +010066 _realPayloadSizeBytes = _rtpStream->Read(&_rtpHeader, _incomingPayload,
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +000067 _payloadSizeBytes, &_nextTime);
68 if (_realPayloadSizeBytes == 0 && _rtpStream->EndOfFile()) {
69 packet_counter_ = 0;
70 lost_packet_counter_ = 0;
71 }
72 }
73 return true;
74}
75
76bool ReceiverWithPacketLoss::PacketLost() {
77 if (burst_lost_counter_ < burst_length_) {
78 lost_packet_counter_++;
79 burst_lost_counter_++;
80 return true;
81 }
82
83 if (lost_packet_counter_ * 100 < loss_rate_ * packet_counter_) {
84 lost_packet_counter_++;
85 burst_lost_counter_ = 1;
86 return true;
87 }
88 return false;
89}
90
Yves Gerey665174f2018-06-19 15:03:05 +020091SenderWithFEC::SenderWithFEC() : expected_loss_rate_(0) {}
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +000092
Yves Gerey665174f2018-06-19 15:03:05 +020093void SenderWithFEC::Setup(AudioCodingModule* acm,
94 RTPStream* rtpStream,
Ali Tofigh714e3cb2022-07-20 12:53:07 +020095 absl::string_view in_file_name,
Fredrik Solenberg657b2962018-12-05 10:30:25 +010096 int payload_type,
97 SdpAudioFormat format,
Yves Gerey665174f2018-06-19 15:03:05 +020098 int expected_loss_rate) {
Fredrik Solenberg657b2962018-12-05 10:30:25 +010099 Sender::Setup(acm, rtpStream, in_file_name, format.clockrate_hz, payload_type,
100 format);
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000101 EXPECT_TRUE(SetFEC(true));
102 EXPECT_TRUE(SetPacketLossRate(expected_loss_rate));
103}
104
105bool SenderWithFEC::SetFEC(bool enable_fec) {
Karl Wiberg658a5522018-08-15 15:20:49 +0200106 bool success = false;
107 _acm->ModifyEncoder([&](std::unique_ptr<AudioEncoder>* enc) {
108 if (*enc && (*enc)->SetFec(enable_fec)) {
109 success = true;
110 }
111 });
112 return success;
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000113}
114
115bool SenderWithFEC::SetPacketLossRate(int expected_loss_rate) {
116 if (_acm->SetPacketLossRate(expected_loss_rate) == 0) {
117 expected_loss_rate_ = expected_loss_rate;
118 return true;
119 }
120 return false;
121}
122
Yves Gerey665174f2018-06-19 15:03:05 +0200123PacketLossTest::PacketLossTest(int channels,
124 int expected_loss_rate,
125 int actual_loss_rate,
126 int burst_length)
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000127 : channels_(channels),
Yves Gerey665174f2018-06-19 15:03:05 +0200128 in_file_name_(channels_ == 1 ? "audio_coding/testfile32kHz"
129 : "audio_coding/teststereo32kHz"),
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000130 sample_rate_hz_(32000),
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000131 expected_loss_rate_(expected_loss_rate),
132 actual_loss_rate_(actual_loss_rate),
Yves Gerey665174f2018-06-19 15:03:05 +0200133 burst_length_(burst_length) {}
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000134
135void PacketLossTest::Perform() {
136#ifndef WEBRTC_CODEC_OPUS
137 return;
138#else
Fredrik Solenbergec0f45b2018-12-03 15:50:44 +0000139 RTPFile rtpFile;
Henrik Lundin84f75692023-02-01 12:07:10 +0000140 std::unique_ptr<AudioCodingModule> acm(AudioCodingModule::Create());
Fredrik Solenberg657b2962018-12-05 10:30:25 +0100141 SdpAudioFormat send_format = SdpAudioFormat("opus", 48000, 2);
142 if (channels_ == 2) {
143 send_format.parameters = {{"stereo", "1"}};
144 }
145
pbos@webrtc.orgc86e45d2014-10-01 10:05:40 +0000146 std::string fileName = webrtc::test::TempFilename(webrtc::test::OutputPath(),
147 "packet_loss_test");
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000148 rtpFile.Open(fileName.c_str(), "wb+");
149 rtpFile.WriteHeader();
Fredrik Solenberg657b2962018-12-05 10:30:25 +0100150 SenderWithFEC sender;
151 sender.Setup(acm.get(), &rtpFile, in_file_name_, 120, send_format,
Jonas Olssona4d87372019-07-05 19:08:33 +0200152 expected_loss_rate_);
Fredrik Solenberg657b2962018-12-05 10:30:25 +0100153 sender.Run();
154 sender.Teardown();
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000155 rtpFile.Close();
156
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000157 rtpFile.Open(fileName.c_str(), "rb");
158 rtpFile.ReadHeader();
Henrik Lundin84f75692023-02-01 12:07:10 +0000159 std::unique_ptr<acm2::AcmReceiver> acm_receiver(
160 std::make_unique<acm2::AcmReceiver>(
161 acm2::AcmReceiver::Config(CreateBuiltinAudioDecoderFactory())));
Fredrik Solenberg657b2962018-12-05 10:30:25 +0100162 ReceiverWithPacketLoss receiver;
Henrik Lundin84f75692023-02-01 12:07:10 +0000163 receiver.Setup(acm_receiver.get(), &rtpFile, "packetLoss_out", channels_, 15,
Jonas Olssona4d87372019-07-05 19:08:33 +0200164 actual_loss_rate_, burst_length_);
Fredrik Solenberg657b2962018-12-05 10:30:25 +0100165 receiver.Run();
166 receiver.Teardown();
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000167 rtpFile.Close();
168#endif
169}
170
171} // namespace webrtc