xref: /aosp_15_r20/external/webrtc/modules/audio_coding/test/PacketLossTest.cc (revision d9f758449e529ab9291ac668be2861e7a55c2422)
1*d9f75844SAndroid Build Coastguard Worker /*
2*d9f75844SAndroid Build Coastguard Worker  *  Copyright (c) 2014 The WebRTC project authors. All Rights Reserved.
3*d9f75844SAndroid Build Coastguard Worker  *
4*d9f75844SAndroid Build Coastguard Worker  *  Use of this source code is governed by a BSD-style license
5*d9f75844SAndroid Build Coastguard Worker  *  that can be found in the LICENSE file in the root of the source
6*d9f75844SAndroid Build Coastguard Worker  *  tree. An additional intellectual property rights grant can be found
7*d9f75844SAndroid Build Coastguard Worker  *  in the file PATENTS.  All contributing project authors may
8*d9f75844SAndroid Build Coastguard Worker  *  be found in the AUTHORS file in the root of the source tree.
9*d9f75844SAndroid Build Coastguard Worker  */
10*d9f75844SAndroid Build Coastguard Worker 
11*d9f75844SAndroid Build Coastguard Worker #include "modules/audio_coding/test/PacketLossTest.h"
12*d9f75844SAndroid Build Coastguard Worker 
13*d9f75844SAndroid Build Coastguard Worker #include <memory>
14*d9f75844SAndroid Build Coastguard Worker 
15*d9f75844SAndroid Build Coastguard Worker #include "absl/strings/string_view.h"
16*d9f75844SAndroid Build Coastguard Worker #include "api/audio_codecs/builtin_audio_decoder_factory.h"
17*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/strings/string_builder.h"
18*d9f75844SAndroid Build Coastguard Worker #include "test/gtest.h"
19*d9f75844SAndroid Build Coastguard Worker #include "test/testsupport/file_utils.h"
20*d9f75844SAndroid Build Coastguard Worker 
21*d9f75844SAndroid Build Coastguard Worker namespace webrtc {
22*d9f75844SAndroid Build Coastguard Worker 
ReceiverWithPacketLoss()23*d9f75844SAndroid Build Coastguard Worker ReceiverWithPacketLoss::ReceiverWithPacketLoss()
24*d9f75844SAndroid Build Coastguard Worker     : loss_rate_(0),
25*d9f75844SAndroid Build Coastguard Worker       burst_length_(1),
26*d9f75844SAndroid Build Coastguard Worker       packet_counter_(0),
27*d9f75844SAndroid Build Coastguard Worker       lost_packet_counter_(0),
28*d9f75844SAndroid Build Coastguard Worker       burst_lost_counter_(burst_length_) {}
29*d9f75844SAndroid Build Coastguard Worker 
Setup(AudioCodingModule * acm,RTPStream * rtpStream,absl::string_view out_file_name,int channels,int file_num,int loss_rate,int burst_length)30*d9f75844SAndroid Build Coastguard Worker void ReceiverWithPacketLoss::Setup(AudioCodingModule* acm,
31*d9f75844SAndroid Build Coastguard Worker                                    RTPStream* rtpStream,
32*d9f75844SAndroid Build Coastguard Worker                                    absl::string_view out_file_name,
33*d9f75844SAndroid Build Coastguard Worker                                    int channels,
34*d9f75844SAndroid Build Coastguard Worker                                    int file_num,
35*d9f75844SAndroid Build Coastguard Worker                                    int loss_rate,
36*d9f75844SAndroid Build Coastguard Worker                                    int burst_length) {
37*d9f75844SAndroid Build Coastguard Worker   loss_rate_ = loss_rate;
38*d9f75844SAndroid Build Coastguard Worker   burst_length_ = burst_length;
39*d9f75844SAndroid Build Coastguard Worker   burst_lost_counter_ = burst_length_;  // To prevent first packet gets lost.
40*d9f75844SAndroid Build Coastguard Worker   rtc::StringBuilder ss;
41*d9f75844SAndroid Build Coastguard Worker   ss << out_file_name << "_" << loss_rate_ << "_" << burst_length_ << "_";
42*d9f75844SAndroid Build Coastguard Worker   Receiver::Setup(acm, rtpStream, ss.str(), channels, file_num);
43*d9f75844SAndroid Build Coastguard Worker }
44*d9f75844SAndroid Build Coastguard Worker 
IncomingPacket()45*d9f75844SAndroid Build Coastguard Worker bool ReceiverWithPacketLoss::IncomingPacket() {
46*d9f75844SAndroid Build Coastguard Worker   if (!_rtpStream->EndOfFile()) {
47*d9f75844SAndroid Build Coastguard Worker     if (packet_counter_ == 0) {
48*d9f75844SAndroid Build Coastguard Worker       _realPayloadSizeBytes = _rtpStream->Read(&_rtpHeader, _incomingPayload,
49*d9f75844SAndroid Build Coastguard Worker                                                _payloadSizeBytes, &_nextTime);
50*d9f75844SAndroid Build Coastguard Worker       if (_realPayloadSizeBytes == 0) {
51*d9f75844SAndroid Build Coastguard Worker         if (_rtpStream->EndOfFile()) {
52*d9f75844SAndroid Build Coastguard Worker           packet_counter_ = 0;
53*d9f75844SAndroid Build Coastguard Worker           return true;
54*d9f75844SAndroid Build Coastguard Worker         } else {
55*d9f75844SAndroid Build Coastguard Worker           return false;
56*d9f75844SAndroid Build Coastguard Worker         }
57*d9f75844SAndroid Build Coastguard Worker       }
58*d9f75844SAndroid Build Coastguard Worker     }
59*d9f75844SAndroid Build Coastguard Worker 
60*d9f75844SAndroid Build Coastguard Worker     if (!PacketLost()) {
61*d9f75844SAndroid Build Coastguard Worker       _acm->IncomingPacket(_incomingPayload, _realPayloadSizeBytes, _rtpHeader);
62*d9f75844SAndroid Build Coastguard Worker     }
63*d9f75844SAndroid Build Coastguard Worker     packet_counter_++;
64*d9f75844SAndroid Build Coastguard Worker     _realPayloadSizeBytes = _rtpStream->Read(&_rtpHeader, _incomingPayload,
65*d9f75844SAndroid Build Coastguard Worker                                              _payloadSizeBytes, &_nextTime);
66*d9f75844SAndroid Build Coastguard Worker     if (_realPayloadSizeBytes == 0 && _rtpStream->EndOfFile()) {
67*d9f75844SAndroid Build Coastguard Worker       packet_counter_ = 0;
68*d9f75844SAndroid Build Coastguard Worker       lost_packet_counter_ = 0;
69*d9f75844SAndroid Build Coastguard Worker     }
70*d9f75844SAndroid Build Coastguard Worker   }
71*d9f75844SAndroid Build Coastguard Worker   return true;
72*d9f75844SAndroid Build Coastguard Worker }
73*d9f75844SAndroid Build Coastguard Worker 
PacketLost()74*d9f75844SAndroid Build Coastguard Worker bool ReceiverWithPacketLoss::PacketLost() {
75*d9f75844SAndroid Build Coastguard Worker   if (burst_lost_counter_ < burst_length_) {
76*d9f75844SAndroid Build Coastguard Worker     lost_packet_counter_++;
77*d9f75844SAndroid Build Coastguard Worker     burst_lost_counter_++;
78*d9f75844SAndroid Build Coastguard Worker     return true;
79*d9f75844SAndroid Build Coastguard Worker   }
80*d9f75844SAndroid Build Coastguard Worker 
81*d9f75844SAndroid Build Coastguard Worker   if (lost_packet_counter_ * 100 < loss_rate_ * packet_counter_) {
82*d9f75844SAndroid Build Coastguard Worker     lost_packet_counter_++;
83*d9f75844SAndroid Build Coastguard Worker     burst_lost_counter_ = 1;
84*d9f75844SAndroid Build Coastguard Worker     return true;
85*d9f75844SAndroid Build Coastguard Worker   }
86*d9f75844SAndroid Build Coastguard Worker   return false;
87*d9f75844SAndroid Build Coastguard Worker }
88*d9f75844SAndroid Build Coastguard Worker 
SenderWithFEC()89*d9f75844SAndroid Build Coastguard Worker SenderWithFEC::SenderWithFEC() : expected_loss_rate_(0) {}
90*d9f75844SAndroid Build Coastguard Worker 
Setup(AudioCodingModule * acm,RTPStream * rtpStream,absl::string_view in_file_name,int payload_type,SdpAudioFormat format,int expected_loss_rate)91*d9f75844SAndroid Build Coastguard Worker void SenderWithFEC::Setup(AudioCodingModule* acm,
92*d9f75844SAndroid Build Coastguard Worker                           RTPStream* rtpStream,
93*d9f75844SAndroid Build Coastguard Worker                           absl::string_view in_file_name,
94*d9f75844SAndroid Build Coastguard Worker                           int payload_type,
95*d9f75844SAndroid Build Coastguard Worker                           SdpAudioFormat format,
96*d9f75844SAndroid Build Coastguard Worker                           int expected_loss_rate) {
97*d9f75844SAndroid Build Coastguard Worker   Sender::Setup(acm, rtpStream, in_file_name, format.clockrate_hz, payload_type,
98*d9f75844SAndroid Build Coastguard Worker                 format);
99*d9f75844SAndroid Build Coastguard Worker   EXPECT_TRUE(SetFEC(true));
100*d9f75844SAndroid Build Coastguard Worker   EXPECT_TRUE(SetPacketLossRate(expected_loss_rate));
101*d9f75844SAndroid Build Coastguard Worker }
102*d9f75844SAndroid Build Coastguard Worker 
SetFEC(bool enable_fec)103*d9f75844SAndroid Build Coastguard Worker bool SenderWithFEC::SetFEC(bool enable_fec) {
104*d9f75844SAndroid Build Coastguard Worker   bool success = false;
105*d9f75844SAndroid Build Coastguard Worker   _acm->ModifyEncoder([&](std::unique_ptr<AudioEncoder>* enc) {
106*d9f75844SAndroid Build Coastguard Worker     if (*enc && (*enc)->SetFec(enable_fec)) {
107*d9f75844SAndroid Build Coastguard Worker       success = true;
108*d9f75844SAndroid Build Coastguard Worker     }
109*d9f75844SAndroid Build Coastguard Worker   });
110*d9f75844SAndroid Build Coastguard Worker   return success;
111*d9f75844SAndroid Build Coastguard Worker }
112*d9f75844SAndroid Build Coastguard Worker 
SetPacketLossRate(int expected_loss_rate)113*d9f75844SAndroid Build Coastguard Worker bool SenderWithFEC::SetPacketLossRate(int expected_loss_rate) {
114*d9f75844SAndroid Build Coastguard Worker   if (_acm->SetPacketLossRate(expected_loss_rate) == 0) {
115*d9f75844SAndroid Build Coastguard Worker     expected_loss_rate_ = expected_loss_rate;
116*d9f75844SAndroid Build Coastguard Worker     return true;
117*d9f75844SAndroid Build Coastguard Worker   }
118*d9f75844SAndroid Build Coastguard Worker   return false;
119*d9f75844SAndroid Build Coastguard Worker }
120*d9f75844SAndroid Build Coastguard Worker 
PacketLossTest(int channels,int expected_loss_rate,int actual_loss_rate,int burst_length)121*d9f75844SAndroid Build Coastguard Worker PacketLossTest::PacketLossTest(int channels,
122*d9f75844SAndroid Build Coastguard Worker                                int expected_loss_rate,
123*d9f75844SAndroid Build Coastguard Worker                                int actual_loss_rate,
124*d9f75844SAndroid Build Coastguard Worker                                int burst_length)
125*d9f75844SAndroid Build Coastguard Worker     : channels_(channels),
126*d9f75844SAndroid Build Coastguard Worker       in_file_name_(channels_ == 1 ? "audio_coding/testfile32kHz"
127*d9f75844SAndroid Build Coastguard Worker                                    : "audio_coding/teststereo32kHz"),
128*d9f75844SAndroid Build Coastguard Worker       sample_rate_hz_(32000),
129*d9f75844SAndroid Build Coastguard Worker       expected_loss_rate_(expected_loss_rate),
130*d9f75844SAndroid Build Coastguard Worker       actual_loss_rate_(actual_loss_rate),
131*d9f75844SAndroid Build Coastguard Worker       burst_length_(burst_length) {}
132*d9f75844SAndroid Build Coastguard Worker 
Perform()133*d9f75844SAndroid Build Coastguard Worker void PacketLossTest::Perform() {
134*d9f75844SAndroid Build Coastguard Worker #ifndef WEBRTC_CODEC_OPUS
135*d9f75844SAndroid Build Coastguard Worker   return;
136*d9f75844SAndroid Build Coastguard Worker #else
137*d9f75844SAndroid Build Coastguard Worker   RTPFile rtpFile;
138*d9f75844SAndroid Build Coastguard Worker   std::unique_ptr<AudioCodingModule> acm(AudioCodingModule::Create(
139*d9f75844SAndroid Build Coastguard Worker       AudioCodingModule::Config(CreateBuiltinAudioDecoderFactory())));
140*d9f75844SAndroid Build Coastguard Worker   SdpAudioFormat send_format = SdpAudioFormat("opus", 48000, 2);
141*d9f75844SAndroid Build Coastguard Worker   if (channels_ == 2) {
142*d9f75844SAndroid Build Coastguard Worker     send_format.parameters = {{"stereo", "1"}};
143*d9f75844SAndroid Build Coastguard Worker   }
144*d9f75844SAndroid Build Coastguard Worker 
145*d9f75844SAndroid Build Coastguard Worker   std::string fileName = webrtc::test::TempFilename(webrtc::test::OutputPath(),
146*d9f75844SAndroid Build Coastguard Worker                                                     "packet_loss_test");
147*d9f75844SAndroid Build Coastguard Worker   rtpFile.Open(fileName.c_str(), "wb+");
148*d9f75844SAndroid Build Coastguard Worker   rtpFile.WriteHeader();
149*d9f75844SAndroid Build Coastguard Worker   SenderWithFEC sender;
150*d9f75844SAndroid Build Coastguard Worker   sender.Setup(acm.get(), &rtpFile, in_file_name_, 120, send_format,
151*d9f75844SAndroid Build Coastguard Worker                expected_loss_rate_);
152*d9f75844SAndroid Build Coastguard Worker   sender.Run();
153*d9f75844SAndroid Build Coastguard Worker   sender.Teardown();
154*d9f75844SAndroid Build Coastguard Worker   rtpFile.Close();
155*d9f75844SAndroid Build Coastguard Worker 
156*d9f75844SAndroid Build Coastguard Worker   rtpFile.Open(fileName.c_str(), "rb");
157*d9f75844SAndroid Build Coastguard Worker   rtpFile.ReadHeader();
158*d9f75844SAndroid Build Coastguard Worker   ReceiverWithPacketLoss receiver;
159*d9f75844SAndroid Build Coastguard Worker   receiver.Setup(acm.get(), &rtpFile, "packetLoss_out", channels_, 15,
160*d9f75844SAndroid Build Coastguard Worker                  actual_loss_rate_, burst_length_);
161*d9f75844SAndroid Build Coastguard Worker   receiver.Run();
162*d9f75844SAndroid Build Coastguard Worker   receiver.Teardown();
163*d9f75844SAndroid Build Coastguard Worker   rtpFile.Close();
164*d9f75844SAndroid Build Coastguard Worker #endif
165*d9f75844SAndroid Build Coastguard Worker }
166*d9f75844SAndroid Build Coastguard Worker 
167*d9f75844SAndroid Build Coastguard Worker }  // namespace webrtc
168