xref: /aosp_15_r20/external/webrtc/audio/audio_receive_stream_unittest.cc (revision d9f758449e529ab9291ac668be2861e7a55c2422)
1*d9f75844SAndroid Build Coastguard Worker /*
2*d9f75844SAndroid Build Coastguard Worker  *  Copyright (c) 2015 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 "audio/audio_receive_stream.h"
12*d9f75844SAndroid Build Coastguard Worker 
13*d9f75844SAndroid Build Coastguard Worker #include <map>
14*d9f75844SAndroid Build Coastguard Worker #include <string>
15*d9f75844SAndroid Build Coastguard Worker #include <utility>
16*d9f75844SAndroid Build Coastguard Worker #include <vector>
17*d9f75844SAndroid Build Coastguard Worker 
18*d9f75844SAndroid Build Coastguard Worker #include "api/test/mock_audio_mixer.h"
19*d9f75844SAndroid Build Coastguard Worker #include "api/test/mock_frame_decryptor.h"
20*d9f75844SAndroid Build Coastguard Worker #include "audio/conversion.h"
21*d9f75844SAndroid Build Coastguard Worker #include "audio/mock_voe_channel_proxy.h"
22*d9f75844SAndroid Build Coastguard Worker #include "call/rtp_stream_receiver_controller.h"
23*d9f75844SAndroid Build Coastguard Worker #include "logging/rtc_event_log/mock/mock_rtc_event_log.h"
24*d9f75844SAndroid Build Coastguard Worker #include "modules/audio_device/include/mock_audio_device.h"
25*d9f75844SAndroid Build Coastguard Worker #include "modules/audio_processing/include/mock_audio_processing.h"
26*d9f75844SAndroid Build Coastguard Worker #include "modules/pacing/packet_router.h"
27*d9f75844SAndroid Build Coastguard Worker #include "modules/rtp_rtcp/source/byte_io.h"
28*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/time_utils.h"
29*d9f75844SAndroid Build Coastguard Worker #include "test/gtest.h"
30*d9f75844SAndroid Build Coastguard Worker #include "test/mock_audio_decoder_factory.h"
31*d9f75844SAndroid Build Coastguard Worker #include "test/mock_transport.h"
32*d9f75844SAndroid Build Coastguard Worker 
33*d9f75844SAndroid Build Coastguard Worker namespace webrtc {
34*d9f75844SAndroid Build Coastguard Worker namespace test {
35*d9f75844SAndroid Build Coastguard Worker namespace {
36*d9f75844SAndroid Build Coastguard Worker 
37*d9f75844SAndroid Build Coastguard Worker using ::testing::_;
38*d9f75844SAndroid Build Coastguard Worker using ::testing::FloatEq;
39*d9f75844SAndroid Build Coastguard Worker using ::testing::NiceMock;
40*d9f75844SAndroid Build Coastguard Worker using ::testing::Return;
41*d9f75844SAndroid Build Coastguard Worker 
MakeAudioDecodeStatsForTest()42*d9f75844SAndroid Build Coastguard Worker AudioDecodingCallStats MakeAudioDecodeStatsForTest() {
43*d9f75844SAndroid Build Coastguard Worker   AudioDecodingCallStats audio_decode_stats;
44*d9f75844SAndroid Build Coastguard Worker   audio_decode_stats.calls_to_silence_generator = 234;
45*d9f75844SAndroid Build Coastguard Worker   audio_decode_stats.calls_to_neteq = 567;
46*d9f75844SAndroid Build Coastguard Worker   audio_decode_stats.decoded_normal = 890;
47*d9f75844SAndroid Build Coastguard Worker   audio_decode_stats.decoded_neteq_plc = 123;
48*d9f75844SAndroid Build Coastguard Worker   audio_decode_stats.decoded_codec_plc = 124;
49*d9f75844SAndroid Build Coastguard Worker   audio_decode_stats.decoded_cng = 456;
50*d9f75844SAndroid Build Coastguard Worker   audio_decode_stats.decoded_plc_cng = 789;
51*d9f75844SAndroid Build Coastguard Worker   audio_decode_stats.decoded_muted_output = 987;
52*d9f75844SAndroid Build Coastguard Worker   return audio_decode_stats;
53*d9f75844SAndroid Build Coastguard Worker }
54*d9f75844SAndroid Build Coastguard Worker 
55*d9f75844SAndroid Build Coastguard Worker const uint32_t kRemoteSsrc = 1234;
56*d9f75844SAndroid Build Coastguard Worker const uint32_t kLocalSsrc = 5678;
57*d9f75844SAndroid Build Coastguard Worker const int kAudioLevelId = 3;
58*d9f75844SAndroid Build Coastguard Worker const int kTransportSequenceNumberId = 4;
59*d9f75844SAndroid Build Coastguard Worker const int kJitterBufferDelay = -7;
60*d9f75844SAndroid Build Coastguard Worker const int kPlayoutBufferDelay = 302;
61*d9f75844SAndroid Build Coastguard Worker const unsigned int kSpeechOutputLevel = 99;
62*d9f75844SAndroid Build Coastguard Worker const double kTotalOutputEnergy = 0.25;
63*d9f75844SAndroid Build Coastguard Worker const double kTotalOutputDuration = 0.5;
64*d9f75844SAndroid Build Coastguard Worker const int64_t kPlayoutNtpTimestampMs = 5678;
65*d9f75844SAndroid Build Coastguard Worker 
66*d9f75844SAndroid Build Coastguard Worker const CallReceiveStatistics kCallStats = {678, 234, -12, 567, 78, 890, 123};
67*d9f75844SAndroid Build Coastguard Worker const std::pair<int, SdpAudioFormat> kReceiveCodec = {
68*d9f75844SAndroid Build Coastguard Worker     123,
69*d9f75844SAndroid Build Coastguard Worker     {"codec_name_recv", 96000, 0}};
70*d9f75844SAndroid Build Coastguard Worker const NetworkStatistics kNetworkStats = {
71*d9f75844SAndroid Build Coastguard Worker     /*currentBufferSize=*/123,
72*d9f75844SAndroid Build Coastguard Worker     /*preferredBufferSize=*/456,
73*d9f75844SAndroid Build Coastguard Worker     /*jitterPeaksFound=*/false,
74*d9f75844SAndroid Build Coastguard Worker     /*totalSamplesReceived=*/789012,
75*d9f75844SAndroid Build Coastguard Worker     /*concealedSamples=*/3456,
76*d9f75844SAndroid Build Coastguard Worker     /*silentConcealedSamples=*/123,
77*d9f75844SAndroid Build Coastguard Worker     /*concealmentEvents=*/456,
78*d9f75844SAndroid Build Coastguard Worker     /*jitterBufferDelayMs=*/789,
79*d9f75844SAndroid Build Coastguard Worker     /*jitterBufferEmittedCount=*/543,
80*d9f75844SAndroid Build Coastguard Worker     /*jitterBufferTargetDelayMs=*/123,
81*d9f75844SAndroid Build Coastguard Worker     /*jitterBufferMinimumDelayMs=*/222,
82*d9f75844SAndroid Build Coastguard Worker     /*insertedSamplesForDeceleration=*/432,
83*d9f75844SAndroid Build Coastguard Worker     /*removedSamplesForAcceleration=*/321,
84*d9f75844SAndroid Build Coastguard Worker     /*fecPacketsReceived=*/123,
85*d9f75844SAndroid Build Coastguard Worker     /*fecPacketsDiscarded=*/101,
86*d9f75844SAndroid Build Coastguard Worker     /*packetsDiscarded=*/989,
87*d9f75844SAndroid Build Coastguard Worker     /*currentExpandRate=*/789,
88*d9f75844SAndroid Build Coastguard Worker     /*currentSpeechExpandRate=*/12,
89*d9f75844SAndroid Build Coastguard Worker     /*currentPreemptiveRate=*/345,
90*d9f75844SAndroid Build Coastguard Worker     /*currentAccelerateRate =*/678,
91*d9f75844SAndroid Build Coastguard Worker     /*currentSecondaryDecodedRate=*/901,
92*d9f75844SAndroid Build Coastguard Worker     /*currentSecondaryDiscardedRate=*/0,
93*d9f75844SAndroid Build Coastguard Worker     /*meanWaitingTimeMs=*/-1,
94*d9f75844SAndroid Build Coastguard Worker     /*maxWaitingTimeMs=*/-1,
95*d9f75844SAndroid Build Coastguard Worker     /*packetBufferFlushes=*/0,
96*d9f75844SAndroid Build Coastguard Worker     /*delayedPacketOutageSamples=*/0,
97*d9f75844SAndroid Build Coastguard Worker     /*relativePacketArrivalDelayMs=*/135,
98*d9f75844SAndroid Build Coastguard Worker     /*interruptionCount=*/-1,
99*d9f75844SAndroid Build Coastguard Worker     /*totalInterruptionDurationMs=*/-1};
100*d9f75844SAndroid Build Coastguard Worker const AudioDecodingCallStats kAudioDecodeStats = MakeAudioDecodeStatsForTest();
101*d9f75844SAndroid Build Coastguard Worker 
102*d9f75844SAndroid Build Coastguard Worker struct ConfigHelper {
ConfigHelperwebrtc::test::__anon15c4e4f00111::ConfigHelper103*d9f75844SAndroid Build Coastguard Worker   explicit ConfigHelper(bool use_null_audio_processing)
104*d9f75844SAndroid Build Coastguard Worker       : ConfigHelper(rtc::make_ref_counted<MockAudioMixer>(),
105*d9f75844SAndroid Build Coastguard Worker                      use_null_audio_processing) {}
106*d9f75844SAndroid Build Coastguard Worker 
ConfigHelperwebrtc::test::__anon15c4e4f00111::ConfigHelper107*d9f75844SAndroid Build Coastguard Worker   ConfigHelper(rtc::scoped_refptr<MockAudioMixer> audio_mixer,
108*d9f75844SAndroid Build Coastguard Worker                bool use_null_audio_processing)
109*d9f75844SAndroid Build Coastguard Worker       : audio_mixer_(audio_mixer) {
110*d9f75844SAndroid Build Coastguard Worker     using ::testing::Invoke;
111*d9f75844SAndroid Build Coastguard Worker 
112*d9f75844SAndroid Build Coastguard Worker     AudioState::Config config;
113*d9f75844SAndroid Build Coastguard Worker     config.audio_mixer = audio_mixer_;
114*d9f75844SAndroid Build Coastguard Worker     config.audio_processing =
115*d9f75844SAndroid Build Coastguard Worker         use_null_audio_processing
116*d9f75844SAndroid Build Coastguard Worker             ? nullptr
117*d9f75844SAndroid Build Coastguard Worker             : rtc::make_ref_counted<NiceMock<MockAudioProcessing>>();
118*d9f75844SAndroid Build Coastguard Worker     config.audio_device_module =
119*d9f75844SAndroid Build Coastguard Worker         rtc::make_ref_counted<testing::NiceMock<MockAudioDeviceModule>>();
120*d9f75844SAndroid Build Coastguard Worker     audio_state_ = AudioState::Create(config);
121*d9f75844SAndroid Build Coastguard Worker 
122*d9f75844SAndroid Build Coastguard Worker     channel_receive_ = new ::testing::StrictMock<MockChannelReceive>();
123*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*channel_receive_, SetNACKStatus(true, 15)).Times(1);
124*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*channel_receive_,
125*d9f75844SAndroid Build Coastguard Worker                 RegisterReceiverCongestionControlObjects(&packet_router_))
126*d9f75844SAndroid Build Coastguard Worker         .Times(1);
127*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*channel_receive_, ResetReceiverCongestionControlObjects())
128*d9f75844SAndroid Build Coastguard Worker         .Times(1);
129*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*channel_receive_, SetAssociatedSendChannel(nullptr)).Times(1);
130*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*channel_receive_, SetReceiveCodecs(_))
131*d9f75844SAndroid Build Coastguard Worker         .WillRepeatedly(Invoke([](const std::map<int, SdpAudioFormat>& codecs) {
132*d9f75844SAndroid Build Coastguard Worker           EXPECT_THAT(codecs, ::testing::IsEmpty());
133*d9f75844SAndroid Build Coastguard Worker         }));
134*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*channel_receive_, SetSourceTracker(_));
135*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*channel_receive_, GetLocalSsrc())
136*d9f75844SAndroid Build Coastguard Worker         .WillRepeatedly(Return(kLocalSsrc));
137*d9f75844SAndroid Build Coastguard Worker 
138*d9f75844SAndroid Build Coastguard Worker     stream_config_.rtp.local_ssrc = kLocalSsrc;
139*d9f75844SAndroid Build Coastguard Worker     stream_config_.rtp.remote_ssrc = kRemoteSsrc;
140*d9f75844SAndroid Build Coastguard Worker     stream_config_.rtp.nack.rtp_history_ms = 300;
141*d9f75844SAndroid Build Coastguard Worker     stream_config_.rtp.extensions.push_back(
142*d9f75844SAndroid Build Coastguard Worker         RtpExtension(RtpExtension::kAudioLevelUri, kAudioLevelId));
143*d9f75844SAndroid Build Coastguard Worker     stream_config_.rtp.extensions.push_back(RtpExtension(
144*d9f75844SAndroid Build Coastguard Worker         RtpExtension::kTransportSequenceNumberUri, kTransportSequenceNumberId));
145*d9f75844SAndroid Build Coastguard Worker     stream_config_.rtcp_send_transport = &rtcp_send_transport_;
146*d9f75844SAndroid Build Coastguard Worker     stream_config_.decoder_factory =
147*d9f75844SAndroid Build Coastguard Worker         rtc::make_ref_counted<MockAudioDecoderFactory>();
148*d9f75844SAndroid Build Coastguard Worker   }
149*d9f75844SAndroid Build Coastguard Worker 
CreateAudioReceiveStreamwebrtc::test::__anon15c4e4f00111::ConfigHelper150*d9f75844SAndroid Build Coastguard Worker   std::unique_ptr<AudioReceiveStreamImpl> CreateAudioReceiveStream() {
151*d9f75844SAndroid Build Coastguard Worker     auto ret = std::make_unique<AudioReceiveStreamImpl>(
152*d9f75844SAndroid Build Coastguard Worker         Clock::GetRealTimeClock(), &packet_router_, stream_config_,
153*d9f75844SAndroid Build Coastguard Worker         audio_state_, &event_log_,
154*d9f75844SAndroid Build Coastguard Worker         std::unique_ptr<voe::ChannelReceiveInterface>(channel_receive_));
155*d9f75844SAndroid Build Coastguard Worker     ret->RegisterWithTransport(&rtp_stream_receiver_controller_);
156*d9f75844SAndroid Build Coastguard Worker     return ret;
157*d9f75844SAndroid Build Coastguard Worker   }
158*d9f75844SAndroid Build Coastguard Worker 
configwebrtc::test::__anon15c4e4f00111::ConfigHelper159*d9f75844SAndroid Build Coastguard Worker   AudioReceiveStreamInterface::Config& config() { return stream_config_; }
audio_mixerwebrtc::test::__anon15c4e4f00111::ConfigHelper160*d9f75844SAndroid Build Coastguard Worker   rtc::scoped_refptr<MockAudioMixer> audio_mixer() { return audio_mixer_; }
channel_receivewebrtc::test::__anon15c4e4f00111::ConfigHelper161*d9f75844SAndroid Build Coastguard Worker   MockChannelReceive* channel_receive() { return channel_receive_; }
162*d9f75844SAndroid Build Coastguard Worker 
SetupMockForGetStatswebrtc::test::__anon15c4e4f00111::ConfigHelper163*d9f75844SAndroid Build Coastguard Worker   void SetupMockForGetStats() {
164*d9f75844SAndroid Build Coastguard Worker     using ::testing::DoAll;
165*d9f75844SAndroid Build Coastguard Worker     using ::testing::SetArgPointee;
166*d9f75844SAndroid Build Coastguard Worker 
167*d9f75844SAndroid Build Coastguard Worker     ASSERT_TRUE(channel_receive_);
168*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*channel_receive_, GetRTCPStatistics())
169*d9f75844SAndroid Build Coastguard Worker         .WillOnce(Return(kCallStats));
170*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*channel_receive_, GetDelayEstimate())
171*d9f75844SAndroid Build Coastguard Worker         .WillOnce(Return(kJitterBufferDelay + kPlayoutBufferDelay));
172*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*channel_receive_, GetSpeechOutputLevelFullRange())
173*d9f75844SAndroid Build Coastguard Worker         .WillOnce(Return(kSpeechOutputLevel));
174*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*channel_receive_, GetTotalOutputEnergy())
175*d9f75844SAndroid Build Coastguard Worker         .WillOnce(Return(kTotalOutputEnergy));
176*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*channel_receive_, GetTotalOutputDuration())
177*d9f75844SAndroid Build Coastguard Worker         .WillOnce(Return(kTotalOutputDuration));
178*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*channel_receive_, GetNetworkStatistics(_))
179*d9f75844SAndroid Build Coastguard Worker         .WillOnce(Return(kNetworkStats));
180*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*channel_receive_, GetDecodingCallStatistics())
181*d9f75844SAndroid Build Coastguard Worker         .WillOnce(Return(kAudioDecodeStats));
182*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*channel_receive_, GetReceiveCodec())
183*d9f75844SAndroid Build Coastguard Worker         .WillOnce(Return(kReceiveCodec));
184*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*channel_receive_, GetCurrentEstimatedPlayoutNtpTimestampMs(_))
185*d9f75844SAndroid Build Coastguard Worker         .WillOnce(Return(kPlayoutNtpTimestampMs));
186*d9f75844SAndroid Build Coastguard Worker   }
187*d9f75844SAndroid Build Coastguard Worker 
188*d9f75844SAndroid Build Coastguard Worker  private:
189*d9f75844SAndroid Build Coastguard Worker   PacketRouter packet_router_;
190*d9f75844SAndroid Build Coastguard Worker   MockRtcEventLog event_log_;
191*d9f75844SAndroid Build Coastguard Worker   rtc::scoped_refptr<AudioState> audio_state_;
192*d9f75844SAndroid Build Coastguard Worker   rtc::scoped_refptr<MockAudioMixer> audio_mixer_;
193*d9f75844SAndroid Build Coastguard Worker   AudioReceiveStreamInterface::Config stream_config_;
194*d9f75844SAndroid Build Coastguard Worker   ::testing::StrictMock<MockChannelReceive>* channel_receive_ = nullptr;
195*d9f75844SAndroid Build Coastguard Worker   RtpStreamReceiverController rtp_stream_receiver_controller_;
196*d9f75844SAndroid Build Coastguard Worker   MockTransport rtcp_send_transport_;
197*d9f75844SAndroid Build Coastguard Worker };
198*d9f75844SAndroid Build Coastguard Worker 
CreateRtcpSenderReport()199*d9f75844SAndroid Build Coastguard Worker const std::vector<uint8_t> CreateRtcpSenderReport() {
200*d9f75844SAndroid Build Coastguard Worker   std::vector<uint8_t> packet;
201*d9f75844SAndroid Build Coastguard Worker   const size_t kRtcpSrLength = 28;  // In bytes.
202*d9f75844SAndroid Build Coastguard Worker   packet.resize(kRtcpSrLength);
203*d9f75844SAndroid Build Coastguard Worker   packet[0] = 0x80;  // Version 2.
204*d9f75844SAndroid Build Coastguard Worker   packet[1] = 0xc8;  // PT = 200, SR.
205*d9f75844SAndroid Build Coastguard Worker   // Length in number of 32-bit words - 1.
206*d9f75844SAndroid Build Coastguard Worker   ByteWriter<uint16_t>::WriteBigEndian(&packet[2], 6);
207*d9f75844SAndroid Build Coastguard Worker   ByteWriter<uint32_t>::WriteBigEndian(&packet[4], kLocalSsrc);
208*d9f75844SAndroid Build Coastguard Worker   return packet;
209*d9f75844SAndroid Build Coastguard Worker }
210*d9f75844SAndroid Build Coastguard Worker }  // namespace
211*d9f75844SAndroid Build Coastguard Worker 
TEST(AudioReceiveStreamTest,ConfigToString)212*d9f75844SAndroid Build Coastguard Worker TEST(AudioReceiveStreamTest, ConfigToString) {
213*d9f75844SAndroid Build Coastguard Worker   AudioReceiveStreamInterface::Config config;
214*d9f75844SAndroid Build Coastguard Worker   config.rtp.remote_ssrc = kRemoteSsrc;
215*d9f75844SAndroid Build Coastguard Worker   config.rtp.local_ssrc = kLocalSsrc;
216*d9f75844SAndroid Build Coastguard Worker   config.rtp.extensions.push_back(
217*d9f75844SAndroid Build Coastguard Worker       RtpExtension(RtpExtension::kAudioLevelUri, kAudioLevelId));
218*d9f75844SAndroid Build Coastguard Worker   EXPECT_EQ(
219*d9f75844SAndroid Build Coastguard Worker       "{rtp: {remote_ssrc: 1234, local_ssrc: 5678, transport_cc: off, nack: "
220*d9f75844SAndroid Build Coastguard Worker       "{rtp_history_ms: 0}, extensions: [{uri: "
221*d9f75844SAndroid Build Coastguard Worker       "urn:ietf:params:rtp-hdrext:ssrc-audio-level, id: 3}]}, "
222*d9f75844SAndroid Build Coastguard Worker       "rtcp_send_transport: null}",
223*d9f75844SAndroid Build Coastguard Worker       config.ToString());
224*d9f75844SAndroid Build Coastguard Worker }
225*d9f75844SAndroid Build Coastguard Worker 
TEST(AudioReceiveStreamTest,ConstructDestruct)226*d9f75844SAndroid Build Coastguard Worker TEST(AudioReceiveStreamTest, ConstructDestruct) {
227*d9f75844SAndroid Build Coastguard Worker   for (bool use_null_audio_processing : {false, true}) {
228*d9f75844SAndroid Build Coastguard Worker     ConfigHelper helper(use_null_audio_processing);
229*d9f75844SAndroid Build Coastguard Worker     auto recv_stream = helper.CreateAudioReceiveStream();
230*d9f75844SAndroid Build Coastguard Worker     recv_stream->UnregisterFromTransport();
231*d9f75844SAndroid Build Coastguard Worker   }
232*d9f75844SAndroid Build Coastguard Worker }
233*d9f75844SAndroid Build Coastguard Worker 
TEST(AudioReceiveStreamTest,ReceiveRtcpPacket)234*d9f75844SAndroid Build Coastguard Worker TEST(AudioReceiveStreamTest, ReceiveRtcpPacket) {
235*d9f75844SAndroid Build Coastguard Worker   for (bool use_null_audio_processing : {false, true}) {
236*d9f75844SAndroid Build Coastguard Worker     ConfigHelper helper(use_null_audio_processing);
237*d9f75844SAndroid Build Coastguard Worker     helper.config().rtp.transport_cc = true;
238*d9f75844SAndroid Build Coastguard Worker     auto recv_stream = helper.CreateAudioReceiveStream();
239*d9f75844SAndroid Build Coastguard Worker     std::vector<uint8_t> rtcp_packet = CreateRtcpSenderReport();
240*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*helper.channel_receive(),
241*d9f75844SAndroid Build Coastguard Worker                 ReceivedRTCPPacket(&rtcp_packet[0], rtcp_packet.size()))
242*d9f75844SAndroid Build Coastguard Worker         .WillOnce(Return());
243*d9f75844SAndroid Build Coastguard Worker     recv_stream->DeliverRtcp(&rtcp_packet[0], rtcp_packet.size());
244*d9f75844SAndroid Build Coastguard Worker     recv_stream->UnregisterFromTransport();
245*d9f75844SAndroid Build Coastguard Worker   }
246*d9f75844SAndroid Build Coastguard Worker }
247*d9f75844SAndroid Build Coastguard Worker 
TEST(AudioReceiveStreamTest,GetStats)248*d9f75844SAndroid Build Coastguard Worker TEST(AudioReceiveStreamTest, GetStats) {
249*d9f75844SAndroid Build Coastguard Worker   for (bool use_null_audio_processing : {false, true}) {
250*d9f75844SAndroid Build Coastguard Worker     ConfigHelper helper(use_null_audio_processing);
251*d9f75844SAndroid Build Coastguard Worker     auto recv_stream = helper.CreateAudioReceiveStream();
252*d9f75844SAndroid Build Coastguard Worker     helper.SetupMockForGetStats();
253*d9f75844SAndroid Build Coastguard Worker     AudioReceiveStreamInterface::Stats stats =
254*d9f75844SAndroid Build Coastguard Worker         recv_stream->GetStats(/*get_and_clear_legacy_stats=*/true);
255*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kRemoteSsrc, stats.remote_ssrc);
256*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kCallStats.payload_bytes_rcvd, stats.payload_bytes_rcvd);
257*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kCallStats.header_and_padding_bytes_rcvd,
258*d9f75844SAndroid Build Coastguard Worker               stats.header_and_padding_bytes_rcvd);
259*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(static_cast<uint32_t>(kCallStats.packetsReceived),
260*d9f75844SAndroid Build Coastguard Worker               stats.packets_rcvd);
261*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kCallStats.cumulativeLost, stats.packets_lost);
262*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kReceiveCodec.second.name, stats.codec_name);
263*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(
264*d9f75844SAndroid Build Coastguard Worker         kCallStats.jitterSamples / (kReceiveCodec.second.clockrate_hz / 1000),
265*d9f75844SAndroid Build Coastguard Worker         stats.jitter_ms);
266*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kNetworkStats.currentBufferSize, stats.jitter_buffer_ms);
267*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kNetworkStats.preferredBufferSize,
268*d9f75844SAndroid Build Coastguard Worker               stats.jitter_buffer_preferred_ms);
269*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(static_cast<uint32_t>(kJitterBufferDelay + kPlayoutBufferDelay),
270*d9f75844SAndroid Build Coastguard Worker               stats.delay_estimate_ms);
271*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(static_cast<int32_t>(kSpeechOutputLevel), stats.audio_level);
272*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kTotalOutputEnergy, stats.total_output_energy);
273*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kNetworkStats.totalSamplesReceived, stats.total_samples_received);
274*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kTotalOutputDuration, stats.total_output_duration);
275*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kNetworkStats.concealedSamples, stats.concealed_samples);
276*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kNetworkStats.concealmentEvents, stats.concealment_events);
277*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(static_cast<double>(kNetworkStats.jitterBufferDelayMs) /
278*d9f75844SAndroid Build Coastguard Worker                   static_cast<double>(rtc::kNumMillisecsPerSec),
279*d9f75844SAndroid Build Coastguard Worker               stats.jitter_buffer_delay_seconds);
280*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kNetworkStats.jitterBufferEmittedCount,
281*d9f75844SAndroid Build Coastguard Worker               stats.jitter_buffer_emitted_count);
282*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(static_cast<double>(kNetworkStats.jitterBufferTargetDelayMs) /
283*d9f75844SAndroid Build Coastguard Worker                   static_cast<double>(rtc::kNumMillisecsPerSec),
284*d9f75844SAndroid Build Coastguard Worker               stats.jitter_buffer_target_delay_seconds);
285*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(static_cast<double>(kNetworkStats.jitterBufferMinimumDelayMs) /
286*d9f75844SAndroid Build Coastguard Worker                   static_cast<double>(rtc::kNumMillisecsPerSec),
287*d9f75844SAndroid Build Coastguard Worker               stats.jitter_buffer_minimum_delay_seconds);
288*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kNetworkStats.insertedSamplesForDeceleration,
289*d9f75844SAndroid Build Coastguard Worker               stats.inserted_samples_for_deceleration);
290*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kNetworkStats.removedSamplesForAcceleration,
291*d9f75844SAndroid Build Coastguard Worker               stats.removed_samples_for_acceleration);
292*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kNetworkStats.fecPacketsReceived, stats.fec_packets_received);
293*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kNetworkStats.fecPacketsDiscarded, stats.fec_packets_discarded);
294*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kNetworkStats.packetsDiscarded, stats.packets_discarded);
295*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(Q14ToFloat(kNetworkStats.currentExpandRate), stats.expand_rate);
296*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(Q14ToFloat(kNetworkStats.currentSpeechExpandRate),
297*d9f75844SAndroid Build Coastguard Worker               stats.speech_expand_rate);
298*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(Q14ToFloat(kNetworkStats.currentSecondaryDecodedRate),
299*d9f75844SAndroid Build Coastguard Worker               stats.secondary_decoded_rate);
300*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(Q14ToFloat(kNetworkStats.currentSecondaryDiscardedRate),
301*d9f75844SAndroid Build Coastguard Worker               stats.secondary_discarded_rate);
302*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(Q14ToFloat(kNetworkStats.currentAccelerateRate),
303*d9f75844SAndroid Build Coastguard Worker               stats.accelerate_rate);
304*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(Q14ToFloat(kNetworkStats.currentPreemptiveRate),
305*d9f75844SAndroid Build Coastguard Worker               stats.preemptive_expand_rate);
306*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kNetworkStats.packetBufferFlushes, stats.jitter_buffer_flushes);
307*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kNetworkStats.delayedPacketOutageSamples,
308*d9f75844SAndroid Build Coastguard Worker               stats.delayed_packet_outage_samples);
309*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(static_cast<double>(kNetworkStats.relativePacketArrivalDelayMs) /
310*d9f75844SAndroid Build Coastguard Worker                   static_cast<double>(rtc::kNumMillisecsPerSec),
311*d9f75844SAndroid Build Coastguard Worker               stats.relative_packet_arrival_delay_seconds);
312*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kNetworkStats.interruptionCount, stats.interruption_count);
313*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kNetworkStats.totalInterruptionDurationMs,
314*d9f75844SAndroid Build Coastguard Worker               stats.total_interruption_duration_ms);
315*d9f75844SAndroid Build Coastguard Worker 
316*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kAudioDecodeStats.calls_to_silence_generator,
317*d9f75844SAndroid Build Coastguard Worker               stats.decoding_calls_to_silence_generator);
318*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kAudioDecodeStats.calls_to_neteq, stats.decoding_calls_to_neteq);
319*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kAudioDecodeStats.decoded_normal, stats.decoding_normal);
320*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kAudioDecodeStats.decoded_neteq_plc, stats.decoding_plc);
321*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kAudioDecodeStats.decoded_codec_plc, stats.decoding_codec_plc);
322*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kAudioDecodeStats.decoded_cng, stats.decoding_cng);
323*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kAudioDecodeStats.decoded_plc_cng, stats.decoding_plc_cng);
324*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kAudioDecodeStats.decoded_muted_output,
325*d9f75844SAndroid Build Coastguard Worker               stats.decoding_muted_output);
326*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kCallStats.capture_start_ntp_time_ms_,
327*d9f75844SAndroid Build Coastguard Worker               stats.capture_start_ntp_time_ms);
328*d9f75844SAndroid Build Coastguard Worker     EXPECT_EQ(kPlayoutNtpTimestampMs, stats.estimated_playout_ntp_timestamp_ms);
329*d9f75844SAndroid Build Coastguard Worker     recv_stream->UnregisterFromTransport();
330*d9f75844SAndroid Build Coastguard Worker   }
331*d9f75844SAndroid Build Coastguard Worker }
332*d9f75844SAndroid Build Coastguard Worker 
TEST(AudioReceiveStreamTest,SetGain)333*d9f75844SAndroid Build Coastguard Worker TEST(AudioReceiveStreamTest, SetGain) {
334*d9f75844SAndroid Build Coastguard Worker   for (bool use_null_audio_processing : {false, true}) {
335*d9f75844SAndroid Build Coastguard Worker     ConfigHelper helper(use_null_audio_processing);
336*d9f75844SAndroid Build Coastguard Worker     auto recv_stream = helper.CreateAudioReceiveStream();
337*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*helper.channel_receive(),
338*d9f75844SAndroid Build Coastguard Worker                 SetChannelOutputVolumeScaling(FloatEq(0.765f)));
339*d9f75844SAndroid Build Coastguard Worker     recv_stream->SetGain(0.765f);
340*d9f75844SAndroid Build Coastguard Worker     recv_stream->UnregisterFromTransport();
341*d9f75844SAndroid Build Coastguard Worker   }
342*d9f75844SAndroid Build Coastguard Worker }
343*d9f75844SAndroid Build Coastguard Worker 
TEST(AudioReceiveStreamTest,StreamsShouldBeAddedToMixerOnceOnStart)344*d9f75844SAndroid Build Coastguard Worker TEST(AudioReceiveStreamTest, StreamsShouldBeAddedToMixerOnceOnStart) {
345*d9f75844SAndroid Build Coastguard Worker   for (bool use_null_audio_processing : {false, true}) {
346*d9f75844SAndroid Build Coastguard Worker     ConfigHelper helper1(use_null_audio_processing);
347*d9f75844SAndroid Build Coastguard Worker     ConfigHelper helper2(helper1.audio_mixer(), use_null_audio_processing);
348*d9f75844SAndroid Build Coastguard Worker     auto recv_stream1 = helper1.CreateAudioReceiveStream();
349*d9f75844SAndroid Build Coastguard Worker     auto recv_stream2 = helper2.CreateAudioReceiveStream();
350*d9f75844SAndroid Build Coastguard Worker 
351*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*helper1.channel_receive(), StartPlayout()).Times(1);
352*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*helper2.channel_receive(), StartPlayout()).Times(1);
353*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*helper1.channel_receive(), StopPlayout()).Times(1);
354*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*helper2.channel_receive(), StopPlayout()).Times(1);
355*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*helper1.audio_mixer(), AddSource(recv_stream1.get()))
356*d9f75844SAndroid Build Coastguard Worker         .WillOnce(Return(true));
357*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*helper1.audio_mixer(), AddSource(recv_stream2.get()))
358*d9f75844SAndroid Build Coastguard Worker         .WillOnce(Return(true));
359*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*helper1.audio_mixer(), RemoveSource(recv_stream1.get()))
360*d9f75844SAndroid Build Coastguard Worker         .Times(1);
361*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(*helper1.audio_mixer(), RemoveSource(recv_stream2.get()))
362*d9f75844SAndroid Build Coastguard Worker         .Times(1);
363*d9f75844SAndroid Build Coastguard Worker 
364*d9f75844SAndroid Build Coastguard Worker     recv_stream1->Start();
365*d9f75844SAndroid Build Coastguard Worker     recv_stream2->Start();
366*d9f75844SAndroid Build Coastguard Worker 
367*d9f75844SAndroid Build Coastguard Worker     // One more should not result in any more mixer sources added.
368*d9f75844SAndroid Build Coastguard Worker     recv_stream1->Start();
369*d9f75844SAndroid Build Coastguard Worker 
370*d9f75844SAndroid Build Coastguard Worker     // Stop stream before it is being destructed.
371*d9f75844SAndroid Build Coastguard Worker     recv_stream2->Stop();
372*d9f75844SAndroid Build Coastguard Worker 
373*d9f75844SAndroid Build Coastguard Worker     recv_stream1->UnregisterFromTransport();
374*d9f75844SAndroid Build Coastguard Worker     recv_stream2->UnregisterFromTransport();
375*d9f75844SAndroid Build Coastguard Worker   }
376*d9f75844SAndroid Build Coastguard Worker }
377*d9f75844SAndroid Build Coastguard Worker 
TEST(AudioReceiveStreamTest,ReconfigureWithUpdatedConfig)378*d9f75844SAndroid Build Coastguard Worker TEST(AudioReceiveStreamTest, ReconfigureWithUpdatedConfig) {
379*d9f75844SAndroid Build Coastguard Worker   for (bool use_null_audio_processing : {false, true}) {
380*d9f75844SAndroid Build Coastguard Worker     ConfigHelper helper(use_null_audio_processing);
381*d9f75844SAndroid Build Coastguard Worker     auto recv_stream = helper.CreateAudioReceiveStream();
382*d9f75844SAndroid Build Coastguard Worker 
383*d9f75844SAndroid Build Coastguard Worker     auto new_config = helper.config();
384*d9f75844SAndroid Build Coastguard Worker 
385*d9f75844SAndroid Build Coastguard Worker     new_config.rtp.extensions.clear();
386*d9f75844SAndroid Build Coastguard Worker     new_config.rtp.extensions.push_back(
387*d9f75844SAndroid Build Coastguard Worker         RtpExtension(RtpExtension::kAudioLevelUri, kAudioLevelId + 1));
388*d9f75844SAndroid Build Coastguard Worker     new_config.rtp.extensions.push_back(
389*d9f75844SAndroid Build Coastguard Worker         RtpExtension(RtpExtension::kTransportSequenceNumberUri,
390*d9f75844SAndroid Build Coastguard Worker                      kTransportSequenceNumberId + 1));
391*d9f75844SAndroid Build Coastguard Worker 
392*d9f75844SAndroid Build Coastguard Worker     MockChannelReceive& channel_receive = *helper.channel_receive();
393*d9f75844SAndroid Build Coastguard Worker 
394*d9f75844SAndroid Build Coastguard Worker     // TODO(tommi, nisse): This applies new extensions to the internal config,
395*d9f75844SAndroid Build Coastguard Worker     // but there's nothing that actually verifies that the changes take effect.
396*d9f75844SAndroid Build Coastguard Worker     // In fact Call manages the extensions separately in Call::ReceiveRtpConfig
397*d9f75844SAndroid Build Coastguard Worker     // and changing this config value (there seem to be a few copies), doesn't
398*d9f75844SAndroid Build Coastguard Worker     // affect that logic.
399*d9f75844SAndroid Build Coastguard Worker     recv_stream->ReconfigureForTesting(new_config);
400*d9f75844SAndroid Build Coastguard Worker 
401*d9f75844SAndroid Build Coastguard Worker     new_config.decoder_map.emplace(1, SdpAudioFormat("foo", 8000, 1));
402*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(channel_receive, SetReceiveCodecs(new_config.decoder_map));
403*d9f75844SAndroid Build Coastguard Worker     recv_stream->SetDecoderMap(new_config.decoder_map);
404*d9f75844SAndroid Build Coastguard Worker 
405*d9f75844SAndroid Build Coastguard Worker     EXPECT_CALL(channel_receive, SetNACKStatus(true, 15 + 1)).Times(1);
406*d9f75844SAndroid Build Coastguard Worker     recv_stream->SetTransportCc(new_config.rtp.transport_cc);
407*d9f75844SAndroid Build Coastguard Worker     recv_stream->SetNackHistory(300 + 20);
408*d9f75844SAndroid Build Coastguard Worker 
409*d9f75844SAndroid Build Coastguard Worker     recv_stream->UnregisterFromTransport();
410*d9f75844SAndroid Build Coastguard Worker   }
411*d9f75844SAndroid Build Coastguard Worker }
412*d9f75844SAndroid Build Coastguard Worker 
TEST(AudioReceiveStreamTest,ReconfigureWithFrameDecryptor)413*d9f75844SAndroid Build Coastguard Worker TEST(AudioReceiveStreamTest, ReconfigureWithFrameDecryptor) {
414*d9f75844SAndroid Build Coastguard Worker   for (bool use_null_audio_processing : {false, true}) {
415*d9f75844SAndroid Build Coastguard Worker     ConfigHelper helper(use_null_audio_processing);
416*d9f75844SAndroid Build Coastguard Worker     auto recv_stream = helper.CreateAudioReceiveStream();
417*d9f75844SAndroid Build Coastguard Worker 
418*d9f75844SAndroid Build Coastguard Worker     auto new_config_0 = helper.config();
419*d9f75844SAndroid Build Coastguard Worker     rtc::scoped_refptr<FrameDecryptorInterface> mock_frame_decryptor_0(
420*d9f75844SAndroid Build Coastguard Worker         rtc::make_ref_counted<MockFrameDecryptor>());
421*d9f75844SAndroid Build Coastguard Worker     new_config_0.frame_decryptor = mock_frame_decryptor_0;
422*d9f75844SAndroid Build Coastguard Worker 
423*d9f75844SAndroid Build Coastguard Worker     // TODO(tommi): While this changes the internal config value, it doesn't
424*d9f75844SAndroid Build Coastguard Worker     // actually change what frame_decryptor is used. WebRtcAudioReceiveStream
425*d9f75844SAndroid Build Coastguard Worker     // recreates the whole instance in order to change this value.
426*d9f75844SAndroid Build Coastguard Worker     // So, it's not clear if changing this post initialization needs to be
427*d9f75844SAndroid Build Coastguard Worker     // supported.
428*d9f75844SAndroid Build Coastguard Worker     recv_stream->ReconfigureForTesting(new_config_0);
429*d9f75844SAndroid Build Coastguard Worker 
430*d9f75844SAndroid Build Coastguard Worker     auto new_config_1 = helper.config();
431*d9f75844SAndroid Build Coastguard Worker     rtc::scoped_refptr<FrameDecryptorInterface> mock_frame_decryptor_1(
432*d9f75844SAndroid Build Coastguard Worker         rtc::make_ref_counted<MockFrameDecryptor>());
433*d9f75844SAndroid Build Coastguard Worker     new_config_1.frame_decryptor = mock_frame_decryptor_1;
434*d9f75844SAndroid Build Coastguard Worker     new_config_1.crypto_options.sframe.require_frame_encryption = true;
435*d9f75844SAndroid Build Coastguard Worker     recv_stream->ReconfigureForTesting(new_config_1);
436*d9f75844SAndroid Build Coastguard Worker     recv_stream->UnregisterFromTransport();
437*d9f75844SAndroid Build Coastguard Worker   }
438*d9f75844SAndroid Build Coastguard Worker }
439*d9f75844SAndroid Build Coastguard Worker 
440*d9f75844SAndroid Build Coastguard Worker }  // namespace test
441*d9f75844SAndroid Build Coastguard Worker }  // namespace webrtc
442