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