blob: 76531d02e972294b28d72d00c961da8e43693fdb [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
11#include <utility>
12#include <vector>
13
Elad Alon4a87e1c2017-10-03 16:11:34 +020014#include "logging/rtc_event_log/events/rtc_event.h"
15#include "logging/rtc_event_log/events/rtc_event_audio_network_adaptation.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020016#include "logging/rtc_event_log/mock/mock_rtc_event_log.h"
17#include "modules/audio_coding/audio_network_adaptor/audio_network_adaptor_impl.h"
18#include "modules/audio_coding/audio_network_adaptor/mock/mock_controller.h"
19#include "modules/audio_coding/audio_network_adaptor/mock/mock_controller_manager.h"
20#include "modules/audio_coding/audio_network_adaptor/mock/mock_debug_dump_writer.h"
Steve Anton10542f22019-01-11 09:11:00 -080021#include "rtc_base/fake_clock.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020022#include "test/field_trial.h"
23#include "test/gtest.h"
minyuecaa9cb22016-09-13 13:34:15 -070024
25namespace webrtc {
26
27using ::testing::_;
28using ::testing::NiceMock;
29using ::testing::Return;
minyue25f6a392016-09-22 22:23:20 -070030using ::testing::SetArgPointee;
minyuecaa9cb22016-09-13 13:34:15 -070031
32namespace {
33
34constexpr size_t kNumControllers = 2;
35
minyue25f6a392016-09-22 22:23:20 -070036constexpr int64_t kClockInitialTimeMs = 12345678;
37
minyuecaa9cb22016-09-13 13:34:15 -070038MATCHER_P(NetworkMetricsIs, metric, "") {
39 return arg.uplink_bandwidth_bps == metric.uplink_bandwidth_bps &&
minyue4aec1d42016-09-21 23:01:26 -070040 arg.target_audio_bitrate_bps == metric.target_audio_bitrate_bps &&
minyue25f6a392016-09-22 22:23:20 -070041 arg.rtt_ms == metric.rtt_ms &&
minyuec9e80ee2016-11-29 13:00:28 -080042 arg.overhead_bytes_per_packet == metric.overhead_bytes_per_packet &&
elad.alondadb4dc2017-03-23 15:29:50 -070043 arg.uplink_packet_loss_fraction ==
44 metric.uplink_packet_loss_fraction &&
45 arg.uplink_recoverable_packet_loss_fraction ==
46 metric.uplink_recoverable_packet_loss_fraction;
minyuecaa9cb22016-09-13 13:34:15 -070047}
48
Elad Alon4a87e1c2017-10-03 16:11:34 +020049MATCHER_P(IsRtcEventAnaConfigEqualTo, config, "") {
50 if (arg->GetType() != RtcEvent::Type::AudioNetworkAdaptation) {
51 return false;
52 }
53 auto ana_event = static_cast<RtcEventAudioNetworkAdaptation*>(arg);
Elad Alon0b1b5c12018-11-09 21:50:14 +010054 return ana_event->config() == config;
Elad Alon4a87e1c2017-10-03 16:11:34 +020055}
56
minyue25f6a392016-09-22 22:23:20 -070057MATCHER_P(EncoderRuntimeConfigIs, config, "") {
58 return arg.bitrate_bps == config.bitrate_bps &&
59 arg.frame_length_ms == config.frame_length_ms &&
60 arg.uplink_packet_loss_fraction ==
61 config.uplink_packet_loss_fraction &&
62 arg.enable_fec == config.enable_fec &&
63 arg.enable_dtx == config.enable_dtx &&
64 arg.num_channels == config.num_channels;
65}
66
minyuecaa9cb22016-09-13 13:34:15 -070067struct AudioNetworkAdaptorStates {
68 std::unique_ptr<AudioNetworkAdaptorImpl> audio_network_adaptor;
69 std::vector<std::unique_ptr<MockController>> mock_controllers;
minyue4b7c9522017-01-24 04:54:59 -080070 std::unique_ptr<MockRtcEventLog> event_log;
minyue25f6a392016-09-22 22:23:20 -070071 MockDebugDumpWriter* mock_debug_dump_writer;
minyuecaa9cb22016-09-13 13:34:15 -070072};
73
74AudioNetworkAdaptorStates CreateAudioNetworkAdaptor() {
75 AudioNetworkAdaptorStates states;
76 std::vector<Controller*> controllers;
77 for (size_t i = 0; i < kNumControllers; ++i) {
78 auto controller =
79 std::unique_ptr<MockController>(new NiceMock<MockController>());
80 EXPECT_CALL(*controller, Die());
81 controllers.push_back(controller.get());
82 states.mock_controllers.push_back(std::move(controller));
83 }
84
85 auto controller_manager = std::unique_ptr<MockControllerManager>(
86 new NiceMock<MockControllerManager>());
87
88 EXPECT_CALL(*controller_manager, Die());
89 EXPECT_CALL(*controller_manager, GetControllers())
90 .WillRepeatedly(Return(controllers));
91 EXPECT_CALL(*controller_manager, GetSortedControllers(_))
92 .WillRepeatedly(Return(controllers));
93
minyue4b7c9522017-01-24 04:54:59 -080094 states.event_log.reset(new NiceMock<MockRtcEventLog>());
minyue25f6a392016-09-22 22:23:20 -070095
96 auto debug_dump_writer =
97 std::unique_ptr<MockDebugDumpWriter>(new NiceMock<MockDebugDumpWriter>());
98 EXPECT_CALL(*debug_dump_writer, Die());
99 states.mock_debug_dump_writer = debug_dump_writer.get();
100
101 AudioNetworkAdaptorImpl::Config config;
minyue4b7c9522017-01-24 04:54:59 -0800102 config.event_log = states.event_log.get();
minyuecaa9cb22016-09-13 13:34:15 -0700103 // AudioNetworkAdaptorImpl governs the lifetime of controller manager.
104 states.audio_network_adaptor.reset(new AudioNetworkAdaptorImpl(
Yves Gerey665174f2018-06-19 15:03:05 +0200105 config, std::move(controller_manager), std::move(debug_dump_writer)));
minyuecaa9cb22016-09-13 13:34:15 -0700106
107 return states;
108}
109
minyuea6a6d652017-01-30 10:50:00 -0800110void SetExpectCallToUpdateNetworkMetrics(
111 const std::vector<std::unique_ptr<MockController>>& controllers,
112 const Controller::NetworkMetrics& check) {
113 for (auto& mock_controller : controllers) {
114 EXPECT_CALL(*mock_controller,
115 UpdateNetworkMetrics(NetworkMetricsIs(check)));
116 }
117}
118
minyuecaa9cb22016-09-13 13:34:15 -0700119} // namespace
120
121TEST(AudioNetworkAdaptorImplTest,
minyuea6a6d652017-01-30 10:50:00 -0800122 UpdateNetworkMetricsIsCalledOnSetUplinkBandwidth) {
123 auto states = CreateAudioNetworkAdaptor();
124 constexpr int kBandwidth = 16000;
125 Controller::NetworkMetrics check;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100126 check.uplink_bandwidth_bps = kBandwidth;
minyuea6a6d652017-01-30 10:50:00 -0800127 SetExpectCallToUpdateNetworkMetrics(states.mock_controllers, check);
128 states.audio_network_adaptor->SetUplinkBandwidth(kBandwidth);
129}
130
131TEST(AudioNetworkAdaptorImplTest,
132 UpdateNetworkMetricsIsCalledOnSetUplinkPacketLossFraction) {
133 auto states = CreateAudioNetworkAdaptor();
134 constexpr float kPacketLoss = 0.7f;
135 Controller::NetworkMetrics check;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100136 check.uplink_packet_loss_fraction = kPacketLoss;
minyuea6a6d652017-01-30 10:50:00 -0800137 SetExpectCallToUpdateNetworkMetrics(states.mock_controllers, check);
138 states.audio_network_adaptor->SetUplinkPacketLossFraction(kPacketLoss);
139}
140
elad.alondadb4dc2017-03-23 15:29:50 -0700141TEST(AudioNetworkAdaptorImplTest,
142 UpdateNetworkMetricsIsCalledOnSetUplinkRecoverablePacketLossFraction) {
143 auto states = CreateAudioNetworkAdaptor();
144 constexpr float kRecoverablePacketLoss = 0.1f;
145 Controller::NetworkMetrics check;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100146 check.uplink_recoverable_packet_loss_fraction = kRecoverablePacketLoss;
elad.alondadb4dc2017-03-23 15:29:50 -0700147 SetExpectCallToUpdateNetworkMetrics(states.mock_controllers, check);
148 states.audio_network_adaptor->SetUplinkRecoverablePacketLossFraction(
149 kRecoverablePacketLoss);
150}
151
minyuea6a6d652017-01-30 10:50:00 -0800152TEST(AudioNetworkAdaptorImplTest, UpdateNetworkMetricsIsCalledOnSetRtt) {
153 auto states = CreateAudioNetworkAdaptor();
154 constexpr int kRtt = 100;
155 Controller::NetworkMetrics check;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100156 check.rtt_ms = kRtt;
minyuea6a6d652017-01-30 10:50:00 -0800157 SetExpectCallToUpdateNetworkMetrics(states.mock_controllers, check);
158 states.audio_network_adaptor->SetRtt(kRtt);
159}
160
161TEST(AudioNetworkAdaptorImplTest,
162 UpdateNetworkMetricsIsCalledOnSetTargetAudioBitrate) {
163 auto states = CreateAudioNetworkAdaptor();
164 constexpr int kTargetAudioBitrate = 15000;
165 Controller::NetworkMetrics check;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100166 check.target_audio_bitrate_bps = kTargetAudioBitrate;
minyuea6a6d652017-01-30 10:50:00 -0800167 SetExpectCallToUpdateNetworkMetrics(states.mock_controllers, check);
168 states.audio_network_adaptor->SetTargetAudioBitrate(kTargetAudioBitrate);
169}
170
171TEST(AudioNetworkAdaptorImplTest, UpdateNetworkMetricsIsCalledOnSetOverhead) {
172 auto states = CreateAudioNetworkAdaptor();
173 constexpr size_t kOverhead = 64;
174 Controller::NetworkMetrics check;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100175 check.overhead_bytes_per_packet = kOverhead;
minyuea6a6d652017-01-30 10:50:00 -0800176 SetExpectCallToUpdateNetworkMetrics(states.mock_controllers, check);
177 states.audio_network_adaptor->SetOverhead(kOverhead);
178}
179
180TEST(AudioNetworkAdaptorImplTest,
minyuecaa9cb22016-09-13 13:34:15 -0700181 MakeDecisionIsCalledOnGetEncoderRuntimeConfig) {
182 auto states = CreateAudioNetworkAdaptor();
minyuea6a6d652017-01-30 10:50:00 -0800183 for (auto& mock_controller : states.mock_controllers)
184 EXPECT_CALL(*mock_controller, MakeDecision(_));
minyuec9e80ee2016-11-29 13:00:28 -0800185 states.audio_network_adaptor->GetEncoderRuntimeConfig();
minyuecaa9cb22016-09-13 13:34:15 -0700186}
187
minyue25f6a392016-09-22 22:23:20 -0700188TEST(AudioNetworkAdaptorImplTest,
189 DumpEncoderRuntimeConfigIsCalledOnGetEncoderRuntimeConfig) {
ivoc17289092017-09-09 08:45:40 -0700190 test::ScopedFieldTrials override_field_trials(
191 "WebRTC-Audio-BitrateAdaptation/Enabled/WebRTC-Audio-FecAdaptation/"
192 "Enabled/");
michaelt92aef172017-04-18 00:11:48 -0700193 rtc::ScopedFakeClock fake_clock;
Sebastian Jansson5f83cf02018-05-08 14:52:22 +0200194 fake_clock.AdvanceTime(TimeDelta::ms(kClockInitialTimeMs));
minyue25f6a392016-09-22 22:23:20 -0700195 auto states = CreateAudioNetworkAdaptor();
michaeltcde46b72017-04-06 05:59:10 -0700196 AudioEncoderRuntimeConfig config;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100197 config.bitrate_bps = 32000;
198 config.enable_fec = true;
minyue25f6a392016-09-22 22:23:20 -0700199
minyuea6a6d652017-01-30 10:50:00 -0800200 EXPECT_CALL(*states.mock_controllers[0], MakeDecision(_))
201 .WillOnce(SetArgPointee<0>(config));
minyue25f6a392016-09-22 22:23:20 -0700202
203 EXPECT_CALL(*states.mock_debug_dump_writer,
204 DumpEncoderRuntimeConfig(EncoderRuntimeConfigIs(config),
205 kClockInitialTimeMs));
206 states.audio_network_adaptor->GetEncoderRuntimeConfig();
207}
208
209TEST(AudioNetworkAdaptorImplTest,
210 DumpNetworkMetricsIsCalledOnSetNetworkMetrics) {
michaelt92aef172017-04-18 00:11:48 -0700211 rtc::ScopedFakeClock fake_clock;
Sebastian Jansson5f83cf02018-05-08 14:52:22 +0200212 fake_clock.AdvanceTime(TimeDelta::ms(kClockInitialTimeMs));
michaelt92aef172017-04-18 00:11:48 -0700213
minyue25f6a392016-09-22 22:23:20 -0700214 auto states = CreateAudioNetworkAdaptor();
215
216 constexpr int kBandwidth = 16000;
217 constexpr float kPacketLoss = 0.7f;
elad.alondadb4dc2017-03-23 15:29:50 -0700218 const auto kRecoverablePacketLoss = 0.2f;
minyue25f6a392016-09-22 22:23:20 -0700219 constexpr int kRtt = 100;
minyuee5e632f2016-09-27 12:54:19 -0700220 constexpr int kTargetAudioBitrate = 15000;
minyuec9e80ee2016-11-29 13:00:28 -0800221 constexpr size_t kOverhead = 64;
minyue25f6a392016-09-22 22:23:20 -0700222
223 Controller::NetworkMetrics check;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100224 check.uplink_bandwidth_bps = kBandwidth;
minyue25f6a392016-09-22 22:23:20 -0700225 int64_t timestamp_check = kClockInitialTimeMs;
226
227 EXPECT_CALL(*states.mock_debug_dump_writer,
228 DumpNetworkMetrics(NetworkMetricsIs(check), timestamp_check));
229 states.audio_network_adaptor->SetUplinkBandwidth(kBandwidth);
230
Sebastian Jansson5f83cf02018-05-08 14:52:22 +0200231 fake_clock.AdvanceTime(TimeDelta::ms(100));
minyue25f6a392016-09-22 22:23:20 -0700232 timestamp_check += 100;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100233 check.uplink_packet_loss_fraction = kPacketLoss;
minyue25f6a392016-09-22 22:23:20 -0700234 EXPECT_CALL(*states.mock_debug_dump_writer,
235 DumpNetworkMetrics(NetworkMetricsIs(check), timestamp_check));
236 states.audio_network_adaptor->SetUplinkPacketLossFraction(kPacketLoss);
237
Sebastian Jansson5f83cf02018-05-08 14:52:22 +0200238 fake_clock.AdvanceTime(TimeDelta::ms(50));
elad.alondadb4dc2017-03-23 15:29:50 -0700239 timestamp_check += 50;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100240 check.uplink_recoverable_packet_loss_fraction = kRecoverablePacketLoss;
elad.alondadb4dc2017-03-23 15:29:50 -0700241 EXPECT_CALL(*states.mock_debug_dump_writer,
242 DumpNetworkMetrics(NetworkMetricsIs(check), timestamp_check));
243 states.audio_network_adaptor->SetUplinkRecoverablePacketLossFraction(
244 kRecoverablePacketLoss);
245
Sebastian Jansson5f83cf02018-05-08 14:52:22 +0200246 fake_clock.AdvanceTime(TimeDelta::ms(200));
minyue25f6a392016-09-22 22:23:20 -0700247 timestamp_check += 200;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100248 check.rtt_ms = kRtt;
minyue25f6a392016-09-22 22:23:20 -0700249 EXPECT_CALL(*states.mock_debug_dump_writer,
250 DumpNetworkMetrics(NetworkMetricsIs(check), timestamp_check));
251 states.audio_network_adaptor->SetRtt(kRtt);
minyuee5e632f2016-09-27 12:54:19 -0700252
Sebastian Jansson5f83cf02018-05-08 14:52:22 +0200253 fake_clock.AdvanceTime(TimeDelta::ms(150));
minyuee5e632f2016-09-27 12:54:19 -0700254 timestamp_check += 150;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100255 check.target_audio_bitrate_bps = kTargetAudioBitrate;
minyuee5e632f2016-09-27 12:54:19 -0700256 EXPECT_CALL(*states.mock_debug_dump_writer,
257 DumpNetworkMetrics(NetworkMetricsIs(check), timestamp_check));
258 states.audio_network_adaptor->SetTargetAudioBitrate(kTargetAudioBitrate);
minyuec9e80ee2016-11-29 13:00:28 -0800259
Sebastian Jansson5f83cf02018-05-08 14:52:22 +0200260 fake_clock.AdvanceTime(TimeDelta::ms(50));
minyuec9e80ee2016-11-29 13:00:28 -0800261 timestamp_check += 50;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100262 check.overhead_bytes_per_packet = kOverhead;
minyuec9e80ee2016-11-29 13:00:28 -0800263 EXPECT_CALL(*states.mock_debug_dump_writer,
264 DumpNetworkMetrics(NetworkMetricsIs(check), timestamp_check));
265 states.audio_network_adaptor->SetOverhead(kOverhead);
minyue25f6a392016-09-22 22:23:20 -0700266}
267
minyue4b7c9522017-01-24 04:54:59 -0800268TEST(AudioNetworkAdaptorImplTest, LogRuntimeConfigOnGetEncoderRuntimeConfig) {
ivoc17289092017-09-09 08:45:40 -0700269 test::ScopedFieldTrials override_field_trials(
270 "WebRTC-Audio-BitrateAdaptation/Enabled/WebRTC-Audio-FecAdaptation/"
271 "Enabled/");
minyue4b7c9522017-01-24 04:54:59 -0800272 auto states = CreateAudioNetworkAdaptor();
273
michaeltcde46b72017-04-06 05:59:10 -0700274 AudioEncoderRuntimeConfig config;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100275 config.bitrate_bps = 32000;
276 config.enable_fec = true;
minyue4b7c9522017-01-24 04:54:59 -0800277
minyuea6a6d652017-01-30 10:50:00 -0800278 EXPECT_CALL(*states.mock_controllers[0], MakeDecision(_))
279 .WillOnce(SetArgPointee<0>(config));
minyue4b7c9522017-01-24 04:54:59 -0800280
Elad Alon4a87e1c2017-10-03 16:11:34 +0200281 EXPECT_CALL(*states.event_log, LogProxy(IsRtcEventAnaConfigEqualTo(config)))
minyue4b7c9522017-01-24 04:54:59 -0800282 .Times(1);
283 states.audio_network_adaptor->GetEncoderRuntimeConfig();
284}
285
ivoce1198e02017-09-08 08:13:19 -0700286TEST(AudioNetworkAdaptorImplTest, TestANAStats) {
287 auto states = CreateAudioNetworkAdaptor();
288
289 // Simulate some adaptation, otherwise the stats will not show anything.
290 AudioEncoderRuntimeConfig config1, config2;
Oskar Sundbom12ab00b2017-11-16 15:31:38 +0100291 config1.bitrate_bps = 32000;
292 config1.num_channels = 2;
293 config1.enable_fec = true;
294 config1.enable_dtx = true;
295 config1.frame_length_ms = 120;
296 config1.uplink_packet_loss_fraction = 0.1f;
297 config2.bitrate_bps = 16000;
298 config2.num_channels = 1;
299 config2.enable_fec = false;
300 config2.enable_dtx = false;
301 config2.frame_length_ms = 60;
302 config1.uplink_packet_loss_fraction = 0.1f;
ivoce1198e02017-09-08 08:13:19 -0700303
304 EXPECT_CALL(*states.mock_controllers[0], MakeDecision(_))
305 .WillOnce(SetArgPointee<0>(config1));
306 states.audio_network_adaptor->GetEncoderRuntimeConfig();
307 EXPECT_CALL(*states.mock_controllers[0], MakeDecision(_))
308 .WillOnce(SetArgPointee<0>(config2));
309 states.audio_network_adaptor->GetEncoderRuntimeConfig();
ivoc17289092017-09-09 08:45:40 -0700310 EXPECT_CALL(*states.mock_controllers[0], MakeDecision(_))
311 .WillOnce(SetArgPointee<0>(config1));
312 states.audio_network_adaptor->GetEncoderRuntimeConfig();
ivoce1198e02017-09-08 08:13:19 -0700313
314 auto ana_stats = states.audio_network_adaptor->GetStats();
315
Danil Chapovalov4da18e82018-04-06 18:03:46 +0200316 EXPECT_EQ(ana_stats.bitrate_action_counter, 2u);
317 EXPECT_EQ(ana_stats.channel_action_counter, 2u);
318 EXPECT_EQ(ana_stats.dtx_action_counter, 2u);
319 EXPECT_EQ(ana_stats.fec_action_counter, 2u);
320 EXPECT_EQ(ana_stats.frame_length_increase_counter, 1u);
321 EXPECT_EQ(ana_stats.frame_length_decrease_counter, 1u);
ivoc17289092017-09-09 08:45:40 -0700322 EXPECT_EQ(ana_stats.uplink_packet_loss_fraction, 0.1f);
ivoce1198e02017-09-08 08:13:19 -0700323}
324
minyuecaa9cb22016-09-13 13:34:15 -0700325} // namespace webrtc