blob: 9612996505e4d7931a6a7dd09c0c521740142d8a [file] [log] [blame]
minyuecaa9cb22016-09-13 13:34:15 -07001/*
2 * Copyright (c) 2016 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
Jonas Olssona4d87372019-07-05 19:08:33 +020011#include "modules/audio_coding/audio_network_adaptor/audio_network_adaptor_impl.h"
12
minyuecaa9cb22016-09-13 13:34:15 -070013#include <utility>
14#include <vector>
15
Elad Alon4a87e1c2017-10-03 16:11:34 +020016#include "logging/rtc_event_log/events/rtc_event.h"
17#include "logging/rtc_event_log/events/rtc_event_audio_network_adaptation.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020018#include "logging/rtc_event_log/mock/mock_rtc_event_log.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020019#include "modules/audio_coding/audio_network_adaptor/mock/mock_controller.h"
20#include "modules/audio_coding/audio_network_adaptor/mock/mock_controller_manager.h"
21#include "modules/audio_coding/audio_network_adaptor/mock/mock_debug_dump_writer.h"
Steve Anton10542f22019-01-11 09:11:00 -080022#include "rtc_base/fake_clock.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020023#include "test/field_trial.h"
24#include "test/gtest.h"
minyuecaa9cb22016-09-13 13:34:15 -070025
26namespace webrtc {
27
28using ::testing::_;
29using ::testing::NiceMock;
30using ::testing::Return;
minyue25f6a392016-09-22 22:23:20 -070031using ::testing::SetArgPointee;
minyuecaa9cb22016-09-13 13:34:15 -070032
33namespace {
34
35constexpr size_t kNumControllers = 2;
36
minyue25f6a392016-09-22 22:23:20 -070037constexpr int64_t kClockInitialTimeMs = 12345678;
38
minyuecaa9cb22016-09-13 13:34:15 -070039MATCHER_P(NetworkMetricsIs, metric, "") {
40 return arg.uplink_bandwidth_bps == metric.uplink_bandwidth_bps &&
minyue4aec1d42016-09-21 23:01:26 -070041 arg.target_audio_bitrate_bps == metric.target_audio_bitrate_bps &&
minyue25f6a392016-09-22 22:23:20 -070042 arg.rtt_ms == metric.rtt_ms &&
minyuec9e80ee2016-11-29 13:00:28 -080043 arg.overhead_bytes_per_packet == metric.overhead_bytes_per_packet &&
elad.alondadb4dc2017-03-23 15:29:50 -070044 arg.uplink_packet_loss_fraction ==
45 metric.uplink_packet_loss_fraction &&
46 arg.uplink_recoverable_packet_loss_fraction ==
47 metric.uplink_recoverable_packet_loss_fraction;
minyuecaa9cb22016-09-13 13:34:15 -070048}
49
Elad Alon4a87e1c2017-10-03 16:11:34 +020050MATCHER_P(IsRtcEventAnaConfigEqualTo, config, "") {
51 if (arg->GetType() != RtcEvent::Type::AudioNetworkAdaptation) {
52 return false;
53 }
54 auto ana_event = static_cast<RtcEventAudioNetworkAdaptation*>(arg);
Elad Alon0b1b5c12018-11-09 21:50:14 +010055 return ana_event->config() == config;
Elad Alon4a87e1c2017-10-03 16:11:34 +020056}
57
minyue25f6a392016-09-22 22:23:20 -070058MATCHER_P(EncoderRuntimeConfigIs, config, "") {
59 return arg.bitrate_bps == config.bitrate_bps &&
60 arg.frame_length_ms == config.frame_length_ms &&
61 arg.uplink_packet_loss_fraction ==
62 config.uplink_packet_loss_fraction &&
63 arg.enable_fec == config.enable_fec &&
64 arg.enable_dtx == config.enable_dtx &&
65 arg.num_channels == config.num_channels;
66}
67
minyuecaa9cb22016-09-13 13:34:15 -070068struct AudioNetworkAdaptorStates {
69 std::unique_ptr<AudioNetworkAdaptorImpl> audio_network_adaptor;
70 std::vector<std::unique_ptr<MockController>> mock_controllers;
minyue4b7c9522017-01-24 04:54:59 -080071 std::unique_ptr<MockRtcEventLog> event_log;
minyue25f6a392016-09-22 22:23:20 -070072 MockDebugDumpWriter* mock_debug_dump_writer;
minyuecaa9cb22016-09-13 13:34:15 -070073};
74
75AudioNetworkAdaptorStates CreateAudioNetworkAdaptor() {
76 AudioNetworkAdaptorStates states;
77 std::vector<Controller*> controllers;
78 for (size_t i = 0; i < kNumControllers; ++i) {
79 auto controller =
80 std::unique_ptr<MockController>(new NiceMock<MockController>());
81 EXPECT_CALL(*controller, Die());
82 controllers.push_back(controller.get());
83 states.mock_controllers.push_back(std::move(controller));
84 }
85
86 auto controller_manager = std::unique_ptr<MockControllerManager>(
87 new NiceMock<MockControllerManager>());
88
89 EXPECT_CALL(*controller_manager, Die());
90 EXPECT_CALL(*controller_manager, GetControllers())
91 .WillRepeatedly(Return(controllers));
92 EXPECT_CALL(*controller_manager, GetSortedControllers(_))
93 .WillRepeatedly(Return(controllers));
94
minyue4b7c9522017-01-24 04:54:59 -080095 states.event_log.reset(new NiceMock<MockRtcEventLog>());
minyue25f6a392016-09-22 22:23:20 -070096
97 auto debug_dump_writer =
98 std::unique_ptr<MockDebugDumpWriter>(new NiceMock<MockDebugDumpWriter>());
99 EXPECT_CALL(*debug_dump_writer, Die());
100 states.mock_debug_dump_writer = debug_dump_writer.get();
101
102 AudioNetworkAdaptorImpl::Config config;
minyue4b7c9522017-01-24 04:54:59 -0800103 config.event_log = states.event_log.get();
minyuecaa9cb22016-09-13 13:34:15 -0700104 // AudioNetworkAdaptorImpl governs the lifetime of controller manager.
105 states.audio_network_adaptor.reset(new AudioNetworkAdaptorImpl(
Yves Gerey665174f2018-06-19 15:03:05 +0200106 config, std::move(controller_manager), std::move(debug_dump_writer)));
minyuecaa9cb22016-09-13 13:34:15 -0700107
108 return states;
109}
110
minyuea6a6d652017-01-30 10:50:00 -0800111void SetExpectCallToUpdateNetworkMetrics(
112 const std::vector<std::unique_ptr<MockController>>& controllers,
113 const Controller::NetworkMetrics& check) {
114 for (auto& mock_controller : controllers) {
115 EXPECT_CALL(*mock_controller,
116 UpdateNetworkMetrics(NetworkMetricsIs(check)));
117 }
118}
119
minyuecaa9cb22016-09-13 13:34:15 -0700120} // namespace
121
122TEST(AudioNetworkAdaptorImplTest,
minyuea6a6d652017-01-30 10:50:00 -0800123 UpdateNetworkMetricsIsCalledOnSetUplinkBandwidth) {
124 auto states = CreateAudioNetworkAdaptor();
125 constexpr int kBandwidth = 16000;
126 Controller::NetworkMetrics check;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100127 check.uplink_bandwidth_bps = kBandwidth;
minyuea6a6d652017-01-30 10:50:00 -0800128 SetExpectCallToUpdateNetworkMetrics(states.mock_controllers, check);
129 states.audio_network_adaptor->SetUplinkBandwidth(kBandwidth);
130}
131
132TEST(AudioNetworkAdaptorImplTest,
133 UpdateNetworkMetricsIsCalledOnSetUplinkPacketLossFraction) {
134 auto states = CreateAudioNetworkAdaptor();
135 constexpr float kPacketLoss = 0.7f;
136 Controller::NetworkMetrics check;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100137 check.uplink_packet_loss_fraction = kPacketLoss;
minyuea6a6d652017-01-30 10:50:00 -0800138 SetExpectCallToUpdateNetworkMetrics(states.mock_controllers, check);
139 states.audio_network_adaptor->SetUplinkPacketLossFraction(kPacketLoss);
140}
141
elad.alondadb4dc2017-03-23 15:29:50 -0700142TEST(AudioNetworkAdaptorImplTest,
143 UpdateNetworkMetricsIsCalledOnSetUplinkRecoverablePacketLossFraction) {
144 auto states = CreateAudioNetworkAdaptor();
145 constexpr float kRecoverablePacketLoss = 0.1f;
146 Controller::NetworkMetrics check;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100147 check.uplink_recoverable_packet_loss_fraction = kRecoverablePacketLoss;
elad.alondadb4dc2017-03-23 15:29:50 -0700148 SetExpectCallToUpdateNetworkMetrics(states.mock_controllers, check);
149 states.audio_network_adaptor->SetUplinkRecoverablePacketLossFraction(
150 kRecoverablePacketLoss);
151}
152
minyuea6a6d652017-01-30 10:50:00 -0800153TEST(AudioNetworkAdaptorImplTest, UpdateNetworkMetricsIsCalledOnSetRtt) {
154 auto states = CreateAudioNetworkAdaptor();
155 constexpr int kRtt = 100;
156 Controller::NetworkMetrics check;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100157 check.rtt_ms = kRtt;
minyuea6a6d652017-01-30 10:50:00 -0800158 SetExpectCallToUpdateNetworkMetrics(states.mock_controllers, check);
159 states.audio_network_adaptor->SetRtt(kRtt);
160}
161
162TEST(AudioNetworkAdaptorImplTest,
163 UpdateNetworkMetricsIsCalledOnSetTargetAudioBitrate) {
164 auto states = CreateAudioNetworkAdaptor();
165 constexpr int kTargetAudioBitrate = 15000;
166 Controller::NetworkMetrics check;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100167 check.target_audio_bitrate_bps = kTargetAudioBitrate;
minyuea6a6d652017-01-30 10:50:00 -0800168 SetExpectCallToUpdateNetworkMetrics(states.mock_controllers, check);
169 states.audio_network_adaptor->SetTargetAudioBitrate(kTargetAudioBitrate);
170}
171
172TEST(AudioNetworkAdaptorImplTest, UpdateNetworkMetricsIsCalledOnSetOverhead) {
173 auto states = CreateAudioNetworkAdaptor();
174 constexpr size_t kOverhead = 64;
175 Controller::NetworkMetrics check;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100176 check.overhead_bytes_per_packet = kOverhead;
minyuea6a6d652017-01-30 10:50:00 -0800177 SetExpectCallToUpdateNetworkMetrics(states.mock_controllers, check);
178 states.audio_network_adaptor->SetOverhead(kOverhead);
179}
180
181TEST(AudioNetworkAdaptorImplTest,
minyuecaa9cb22016-09-13 13:34:15 -0700182 MakeDecisionIsCalledOnGetEncoderRuntimeConfig) {
183 auto states = CreateAudioNetworkAdaptor();
minyuea6a6d652017-01-30 10:50:00 -0800184 for (auto& mock_controller : states.mock_controllers)
185 EXPECT_CALL(*mock_controller, MakeDecision(_));
minyuec9e80ee2016-11-29 13:00:28 -0800186 states.audio_network_adaptor->GetEncoderRuntimeConfig();
minyuecaa9cb22016-09-13 13:34:15 -0700187}
188
minyue25f6a392016-09-22 22:23:20 -0700189TEST(AudioNetworkAdaptorImplTest,
190 DumpEncoderRuntimeConfigIsCalledOnGetEncoderRuntimeConfig) {
ivoc17289092017-09-09 08:45:40 -0700191 test::ScopedFieldTrials override_field_trials(
192 "WebRTC-Audio-BitrateAdaptation/Enabled/WebRTC-Audio-FecAdaptation/"
193 "Enabled/");
michaelt92aef172017-04-18 00:11:48 -0700194 rtc::ScopedFakeClock fake_clock;
Sebastian Jansson5f83cf02018-05-08 14:52:22 +0200195 fake_clock.AdvanceTime(TimeDelta::ms(kClockInitialTimeMs));
minyue25f6a392016-09-22 22:23:20 -0700196 auto states = CreateAudioNetworkAdaptor();
michaeltcde46b72017-04-06 05:59:10 -0700197 AudioEncoderRuntimeConfig config;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100198 config.bitrate_bps = 32000;
199 config.enable_fec = true;
minyue25f6a392016-09-22 22:23:20 -0700200
minyuea6a6d652017-01-30 10:50:00 -0800201 EXPECT_CALL(*states.mock_controllers[0], MakeDecision(_))
202 .WillOnce(SetArgPointee<0>(config));
minyue25f6a392016-09-22 22:23:20 -0700203
204 EXPECT_CALL(*states.mock_debug_dump_writer,
205 DumpEncoderRuntimeConfig(EncoderRuntimeConfigIs(config),
206 kClockInitialTimeMs));
207 states.audio_network_adaptor->GetEncoderRuntimeConfig();
208}
209
210TEST(AudioNetworkAdaptorImplTest,
211 DumpNetworkMetricsIsCalledOnSetNetworkMetrics) {
michaelt92aef172017-04-18 00:11:48 -0700212 rtc::ScopedFakeClock fake_clock;
Sebastian Jansson5f83cf02018-05-08 14:52:22 +0200213 fake_clock.AdvanceTime(TimeDelta::ms(kClockInitialTimeMs));
michaelt92aef172017-04-18 00:11:48 -0700214
minyue25f6a392016-09-22 22:23:20 -0700215 auto states = CreateAudioNetworkAdaptor();
216
217 constexpr int kBandwidth = 16000;
218 constexpr float kPacketLoss = 0.7f;
elad.alondadb4dc2017-03-23 15:29:50 -0700219 const auto kRecoverablePacketLoss = 0.2f;
minyue25f6a392016-09-22 22:23:20 -0700220 constexpr int kRtt = 100;
minyuee5e632f2016-09-27 12:54:19 -0700221 constexpr int kTargetAudioBitrate = 15000;
minyuec9e80ee2016-11-29 13:00:28 -0800222 constexpr size_t kOverhead = 64;
minyue25f6a392016-09-22 22:23:20 -0700223
224 Controller::NetworkMetrics check;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100225 check.uplink_bandwidth_bps = kBandwidth;
minyue25f6a392016-09-22 22:23:20 -0700226 int64_t timestamp_check = kClockInitialTimeMs;
227
228 EXPECT_CALL(*states.mock_debug_dump_writer,
229 DumpNetworkMetrics(NetworkMetricsIs(check), timestamp_check));
230 states.audio_network_adaptor->SetUplinkBandwidth(kBandwidth);
231
Sebastian Jansson5f83cf02018-05-08 14:52:22 +0200232 fake_clock.AdvanceTime(TimeDelta::ms(100));
minyue25f6a392016-09-22 22:23:20 -0700233 timestamp_check += 100;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100234 check.uplink_packet_loss_fraction = kPacketLoss;
minyue25f6a392016-09-22 22:23:20 -0700235 EXPECT_CALL(*states.mock_debug_dump_writer,
236 DumpNetworkMetrics(NetworkMetricsIs(check), timestamp_check));
237 states.audio_network_adaptor->SetUplinkPacketLossFraction(kPacketLoss);
238
Sebastian Jansson5f83cf02018-05-08 14:52:22 +0200239 fake_clock.AdvanceTime(TimeDelta::ms(50));
elad.alondadb4dc2017-03-23 15:29:50 -0700240 timestamp_check += 50;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100241 check.uplink_recoverable_packet_loss_fraction = kRecoverablePacketLoss;
elad.alondadb4dc2017-03-23 15:29:50 -0700242 EXPECT_CALL(*states.mock_debug_dump_writer,
243 DumpNetworkMetrics(NetworkMetricsIs(check), timestamp_check));
244 states.audio_network_adaptor->SetUplinkRecoverablePacketLossFraction(
245 kRecoverablePacketLoss);
246
Sebastian Jansson5f83cf02018-05-08 14:52:22 +0200247 fake_clock.AdvanceTime(TimeDelta::ms(200));
minyue25f6a392016-09-22 22:23:20 -0700248 timestamp_check += 200;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100249 check.rtt_ms = kRtt;
minyue25f6a392016-09-22 22:23:20 -0700250 EXPECT_CALL(*states.mock_debug_dump_writer,
251 DumpNetworkMetrics(NetworkMetricsIs(check), timestamp_check));
252 states.audio_network_adaptor->SetRtt(kRtt);
minyuee5e632f2016-09-27 12:54:19 -0700253
Sebastian Jansson5f83cf02018-05-08 14:52:22 +0200254 fake_clock.AdvanceTime(TimeDelta::ms(150));
minyuee5e632f2016-09-27 12:54:19 -0700255 timestamp_check += 150;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100256 check.target_audio_bitrate_bps = kTargetAudioBitrate;
minyuee5e632f2016-09-27 12:54:19 -0700257 EXPECT_CALL(*states.mock_debug_dump_writer,
258 DumpNetworkMetrics(NetworkMetricsIs(check), timestamp_check));
259 states.audio_network_adaptor->SetTargetAudioBitrate(kTargetAudioBitrate);
minyuec9e80ee2016-11-29 13:00:28 -0800260
Sebastian Jansson5f83cf02018-05-08 14:52:22 +0200261 fake_clock.AdvanceTime(TimeDelta::ms(50));
minyuec9e80ee2016-11-29 13:00:28 -0800262 timestamp_check += 50;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100263 check.overhead_bytes_per_packet = kOverhead;
minyuec9e80ee2016-11-29 13:00:28 -0800264 EXPECT_CALL(*states.mock_debug_dump_writer,
265 DumpNetworkMetrics(NetworkMetricsIs(check), timestamp_check));
266 states.audio_network_adaptor->SetOverhead(kOverhead);
minyue25f6a392016-09-22 22:23:20 -0700267}
268
minyue4b7c9522017-01-24 04:54:59 -0800269TEST(AudioNetworkAdaptorImplTest, LogRuntimeConfigOnGetEncoderRuntimeConfig) {
ivoc17289092017-09-09 08:45:40 -0700270 test::ScopedFieldTrials override_field_trials(
271 "WebRTC-Audio-BitrateAdaptation/Enabled/WebRTC-Audio-FecAdaptation/"
272 "Enabled/");
minyue4b7c9522017-01-24 04:54:59 -0800273 auto states = CreateAudioNetworkAdaptor();
274
michaeltcde46b72017-04-06 05:59:10 -0700275 AudioEncoderRuntimeConfig config;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100276 config.bitrate_bps = 32000;
277 config.enable_fec = true;
minyue4b7c9522017-01-24 04:54:59 -0800278
minyuea6a6d652017-01-30 10:50:00 -0800279 EXPECT_CALL(*states.mock_controllers[0], MakeDecision(_))
280 .WillOnce(SetArgPointee<0>(config));
minyue4b7c9522017-01-24 04:54:59 -0800281
Elad Alon4a87e1c2017-10-03 16:11:34 +0200282 EXPECT_CALL(*states.event_log, LogProxy(IsRtcEventAnaConfigEqualTo(config)))
minyue4b7c9522017-01-24 04:54:59 -0800283 .Times(1);
284 states.audio_network_adaptor->GetEncoderRuntimeConfig();
285}
286
ivoce1198e02017-09-08 08:13:19 -0700287TEST(AudioNetworkAdaptorImplTest, TestANAStats) {
288 auto states = CreateAudioNetworkAdaptor();
289
290 // Simulate some adaptation, otherwise the stats will not show anything.
291 AudioEncoderRuntimeConfig config1, config2;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100292 config1.bitrate_bps = 32000;
293 config1.num_channels = 2;
294 config1.enable_fec = true;
295 config1.enable_dtx = true;
296 config1.frame_length_ms = 120;
297 config1.uplink_packet_loss_fraction = 0.1f;
298 config2.bitrate_bps = 16000;
299 config2.num_channels = 1;
300 config2.enable_fec = false;
301 config2.enable_dtx = false;
302 config2.frame_length_ms = 60;
303 config1.uplink_packet_loss_fraction = 0.1f;
ivoce1198e02017-09-08 08:13:19 -0700304
305 EXPECT_CALL(*states.mock_controllers[0], MakeDecision(_))
306 .WillOnce(SetArgPointee<0>(config1));
307 states.audio_network_adaptor->GetEncoderRuntimeConfig();
308 EXPECT_CALL(*states.mock_controllers[0], MakeDecision(_))
309 .WillOnce(SetArgPointee<0>(config2));
310 states.audio_network_adaptor->GetEncoderRuntimeConfig();
ivoc17289092017-09-09 08:45:40 -0700311 EXPECT_CALL(*states.mock_controllers[0], MakeDecision(_))
312 .WillOnce(SetArgPointee<0>(config1));
313 states.audio_network_adaptor->GetEncoderRuntimeConfig();
ivoce1198e02017-09-08 08:13:19 -0700314
315 auto ana_stats = states.audio_network_adaptor->GetStats();
316
Danil Chapovalov4da18e82018-04-06 18:03:46 +0200317 EXPECT_EQ(ana_stats.bitrate_action_counter, 2u);
318 EXPECT_EQ(ana_stats.channel_action_counter, 2u);
319 EXPECT_EQ(ana_stats.dtx_action_counter, 2u);
320 EXPECT_EQ(ana_stats.fec_action_counter, 2u);
321 EXPECT_EQ(ana_stats.frame_length_increase_counter, 1u);
322 EXPECT_EQ(ana_stats.frame_length_decrease_counter, 1u);
ivoc17289092017-09-09 08:45:40 -0700323 EXPECT_EQ(ana_stats.uplink_packet_loss_fraction, 0.1f);
ivoce1198e02017-09-08 08:13:19 -0700324}
325
minyuecaa9cb22016-09-13 13:34:15 -0700326} // namespace webrtc