blob: 1f0f71a543c13ec68a4192ff709ab6d395ef2269 [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"
21#include "rtc_base/fakeclock.h"
22#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);
54 return *ana_event->config_ == config;
55}
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(
minyue25f6a392016-09-22 22:23:20 -0700105 config,
106 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;
127 check.uplink_bandwidth_bps = rtc::Optional<int>(kBandwidth);
128 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;
137 check.uplink_packet_loss_fraction = rtc::Optional<float>(kPacketLoss);
138 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;
147 check.uplink_recoverable_packet_loss_fraction =
148 rtc::Optional<float>(kRecoverablePacketLoss);
149 SetExpectCallToUpdateNetworkMetrics(states.mock_controllers, check);
150 states.audio_network_adaptor->SetUplinkRecoverablePacketLossFraction(
151 kRecoverablePacketLoss);
152}
153
minyuea6a6d652017-01-30 10:50:00 -0800154TEST(AudioNetworkAdaptorImplTest, UpdateNetworkMetricsIsCalledOnSetRtt) {
155 auto states = CreateAudioNetworkAdaptor();
156 constexpr int kRtt = 100;
157 Controller::NetworkMetrics check;
158 check.rtt_ms = rtc::Optional<int>(kRtt);
159 SetExpectCallToUpdateNetworkMetrics(states.mock_controllers, check);
160 states.audio_network_adaptor->SetRtt(kRtt);
161}
162
163TEST(AudioNetworkAdaptorImplTest,
164 UpdateNetworkMetricsIsCalledOnSetTargetAudioBitrate) {
165 auto states = CreateAudioNetworkAdaptor();
166 constexpr int kTargetAudioBitrate = 15000;
167 Controller::NetworkMetrics check;
168 check.target_audio_bitrate_bps = rtc::Optional<int>(kTargetAudioBitrate);
169 SetExpectCallToUpdateNetworkMetrics(states.mock_controllers, check);
170 states.audio_network_adaptor->SetTargetAudioBitrate(kTargetAudioBitrate);
171}
172
173TEST(AudioNetworkAdaptorImplTest, UpdateNetworkMetricsIsCalledOnSetOverhead) {
174 auto states = CreateAudioNetworkAdaptor();
175 constexpr size_t kOverhead = 64;
176 Controller::NetworkMetrics check;
177 check.overhead_bytes_per_packet = rtc::Optional<size_t>(kOverhead);
178 SetExpectCallToUpdateNetworkMetrics(states.mock_controllers, check);
179 states.audio_network_adaptor->SetOverhead(kOverhead);
180}
181
182TEST(AudioNetworkAdaptorImplTest,
minyuecaa9cb22016-09-13 13:34:15 -0700183 MakeDecisionIsCalledOnGetEncoderRuntimeConfig) {
184 auto states = CreateAudioNetworkAdaptor();
minyuea6a6d652017-01-30 10:50:00 -0800185 for (auto& mock_controller : states.mock_controllers)
186 EXPECT_CALL(*mock_controller, MakeDecision(_));
minyuec9e80ee2016-11-29 13:00:28 -0800187 states.audio_network_adaptor->GetEncoderRuntimeConfig();
minyuecaa9cb22016-09-13 13:34:15 -0700188}
189
minyue25f6a392016-09-22 22:23:20 -0700190TEST(AudioNetworkAdaptorImplTest,
191 DumpEncoderRuntimeConfigIsCalledOnGetEncoderRuntimeConfig) {
ivoc17289092017-09-09 08:45:40 -0700192 test::ScopedFieldTrials override_field_trials(
193 "WebRTC-Audio-BitrateAdaptation/Enabled/WebRTC-Audio-FecAdaptation/"
194 "Enabled/");
michaelt92aef172017-04-18 00:11:48 -0700195 rtc::ScopedFakeClock fake_clock;
196 fake_clock.AdvanceTime(rtc::TimeDelta::FromMilliseconds(kClockInitialTimeMs));
minyue25f6a392016-09-22 22:23:20 -0700197 auto states = CreateAudioNetworkAdaptor();
michaeltcde46b72017-04-06 05:59:10 -0700198 AudioEncoderRuntimeConfig config;
minyue25f6a392016-09-22 22:23:20 -0700199 config.bitrate_bps = rtc::Optional<int>(32000);
200 config.enable_fec = rtc::Optional<bool>(true);
201
minyuea6a6d652017-01-30 10:50:00 -0800202 EXPECT_CALL(*states.mock_controllers[0], MakeDecision(_))
203 .WillOnce(SetArgPointee<0>(config));
minyue25f6a392016-09-22 22:23:20 -0700204
205 EXPECT_CALL(*states.mock_debug_dump_writer,
206 DumpEncoderRuntimeConfig(EncoderRuntimeConfigIs(config),
207 kClockInitialTimeMs));
208 states.audio_network_adaptor->GetEncoderRuntimeConfig();
209}
210
211TEST(AudioNetworkAdaptorImplTest,
212 DumpNetworkMetricsIsCalledOnSetNetworkMetrics) {
michaelt92aef172017-04-18 00:11:48 -0700213 rtc::ScopedFakeClock fake_clock;
214 fake_clock.AdvanceTime(rtc::TimeDelta::FromMilliseconds(kClockInitialTimeMs));
215
minyue25f6a392016-09-22 22:23:20 -0700216 auto states = CreateAudioNetworkAdaptor();
217
218 constexpr int kBandwidth = 16000;
219 constexpr float kPacketLoss = 0.7f;
elad.alondadb4dc2017-03-23 15:29:50 -0700220 const auto kRecoverablePacketLoss = 0.2f;
minyue25f6a392016-09-22 22:23:20 -0700221 constexpr int kRtt = 100;
minyuee5e632f2016-09-27 12:54:19 -0700222 constexpr int kTargetAudioBitrate = 15000;
minyuec9e80ee2016-11-29 13:00:28 -0800223 constexpr size_t kOverhead = 64;
minyue25f6a392016-09-22 22:23:20 -0700224
225 Controller::NetworkMetrics check;
226 check.uplink_bandwidth_bps = rtc::Optional<int>(kBandwidth);
227 int64_t timestamp_check = kClockInitialTimeMs;
228
229 EXPECT_CALL(*states.mock_debug_dump_writer,
230 DumpNetworkMetrics(NetworkMetricsIs(check), timestamp_check));
231 states.audio_network_adaptor->SetUplinkBandwidth(kBandwidth);
232
michaelt92aef172017-04-18 00:11:48 -0700233 fake_clock.AdvanceTime(rtc::TimeDelta::FromMilliseconds(100));
minyue25f6a392016-09-22 22:23:20 -0700234 timestamp_check += 100;
235 check.uplink_packet_loss_fraction = rtc::Optional<float>(kPacketLoss);
236 EXPECT_CALL(*states.mock_debug_dump_writer,
237 DumpNetworkMetrics(NetworkMetricsIs(check), timestamp_check));
238 states.audio_network_adaptor->SetUplinkPacketLossFraction(kPacketLoss);
239
michaelt92aef172017-04-18 00:11:48 -0700240 fake_clock.AdvanceTime(rtc::TimeDelta::FromMilliseconds(50));
elad.alondadb4dc2017-03-23 15:29:50 -0700241 timestamp_check += 50;
242 check.uplink_recoverable_packet_loss_fraction =
243 rtc::Optional<float>(kRecoverablePacketLoss);
244 EXPECT_CALL(*states.mock_debug_dump_writer,
245 DumpNetworkMetrics(NetworkMetricsIs(check), timestamp_check));
246 states.audio_network_adaptor->SetUplinkRecoverablePacketLossFraction(
247 kRecoverablePacketLoss);
248
michaelt92aef172017-04-18 00:11:48 -0700249 fake_clock.AdvanceTime(rtc::TimeDelta::FromMilliseconds(200));
minyue25f6a392016-09-22 22:23:20 -0700250 timestamp_check += 200;
251 check.rtt_ms = rtc::Optional<int>(kRtt);
252 EXPECT_CALL(*states.mock_debug_dump_writer,
253 DumpNetworkMetrics(NetworkMetricsIs(check), timestamp_check));
254 states.audio_network_adaptor->SetRtt(kRtt);
minyuee5e632f2016-09-27 12:54:19 -0700255
michaelt92aef172017-04-18 00:11:48 -0700256 fake_clock.AdvanceTime(rtc::TimeDelta::FromMilliseconds(150));
minyuee5e632f2016-09-27 12:54:19 -0700257 timestamp_check += 150;
258 check.target_audio_bitrate_bps = rtc::Optional<int>(kTargetAudioBitrate);
259 EXPECT_CALL(*states.mock_debug_dump_writer,
260 DumpNetworkMetrics(NetworkMetricsIs(check), timestamp_check));
261 states.audio_network_adaptor->SetTargetAudioBitrate(kTargetAudioBitrate);
minyuec9e80ee2016-11-29 13:00:28 -0800262
michaelt92aef172017-04-18 00:11:48 -0700263 fake_clock.AdvanceTime(rtc::TimeDelta::FromMilliseconds(50));
minyuec9e80ee2016-11-29 13:00:28 -0800264 timestamp_check += 50;
265 check.overhead_bytes_per_packet = rtc::Optional<size_t>(kOverhead);
266 EXPECT_CALL(*states.mock_debug_dump_writer,
267 DumpNetworkMetrics(NetworkMetricsIs(check), timestamp_check));
268 states.audio_network_adaptor->SetOverhead(kOverhead);
minyue25f6a392016-09-22 22:23:20 -0700269}
270
minyue4b7c9522017-01-24 04:54:59 -0800271TEST(AudioNetworkAdaptorImplTest, LogRuntimeConfigOnGetEncoderRuntimeConfig) {
ivoc17289092017-09-09 08:45:40 -0700272 test::ScopedFieldTrials override_field_trials(
273 "WebRTC-Audio-BitrateAdaptation/Enabled/WebRTC-Audio-FecAdaptation/"
274 "Enabled/");
minyue4b7c9522017-01-24 04:54:59 -0800275 auto states = CreateAudioNetworkAdaptor();
276
michaeltcde46b72017-04-06 05:59:10 -0700277 AudioEncoderRuntimeConfig config;
minyue4b7c9522017-01-24 04:54:59 -0800278 config.bitrate_bps = rtc::Optional<int>(32000);
279 config.enable_fec = rtc::Optional<bool>(true);
280
minyuea6a6d652017-01-30 10:50:00 -0800281 EXPECT_CALL(*states.mock_controllers[0], MakeDecision(_))
282 .WillOnce(SetArgPointee<0>(config));
minyue4b7c9522017-01-24 04:54:59 -0800283
Elad Alon4a87e1c2017-10-03 16:11:34 +0200284 EXPECT_CALL(*states.event_log, LogProxy(IsRtcEventAnaConfigEqualTo(config)))
minyue4b7c9522017-01-24 04:54:59 -0800285 .Times(1);
286 states.audio_network_adaptor->GetEncoderRuntimeConfig();
287}
288
ivoce1198e02017-09-08 08:13:19 -0700289TEST(AudioNetworkAdaptorImplTest, TestANAStats) {
290 auto states = CreateAudioNetworkAdaptor();
291
292 // Simulate some adaptation, otherwise the stats will not show anything.
293 AudioEncoderRuntimeConfig config1, config2;
294 config1.bitrate_bps = rtc::Optional<int>(32000);
ivoc17289092017-09-09 08:45:40 -0700295 config1.num_channels = rtc::Optional<size_t>(2);
ivoce1198e02017-09-08 08:13:19 -0700296 config1.enable_fec = rtc::Optional<bool>(true);
ivoc17289092017-09-09 08:45:40 -0700297 config1.enable_dtx = rtc::Optional<bool>(true);
298 config1.frame_length_ms = rtc::Optional<int>(120);
299 config1.uplink_packet_loss_fraction = rtc::Optional<float>(0.1f);
ivoce1198e02017-09-08 08:13:19 -0700300 config2.bitrate_bps = rtc::Optional<int>(16000);
ivoc17289092017-09-09 08:45:40 -0700301 config2.num_channels = rtc::Optional<size_t>(1);
ivoce1198e02017-09-08 08:13:19 -0700302 config2.enable_fec = rtc::Optional<bool>(false);
ivoc17289092017-09-09 08:45:40 -0700303 config2.enable_dtx = rtc::Optional<bool>(false);
304 config2.frame_length_ms = rtc::Optional<int>(60);
305 config1.uplink_packet_loss_fraction = rtc::Optional<float>(0.1f);
ivoce1198e02017-09-08 08:13:19 -0700306
307 EXPECT_CALL(*states.mock_controllers[0], MakeDecision(_))
308 .WillOnce(SetArgPointee<0>(config1));
309 states.audio_network_adaptor->GetEncoderRuntimeConfig();
310 EXPECT_CALL(*states.mock_controllers[0], MakeDecision(_))
311 .WillOnce(SetArgPointee<0>(config2));
312 states.audio_network_adaptor->GetEncoderRuntimeConfig();
ivoc17289092017-09-09 08:45:40 -0700313 EXPECT_CALL(*states.mock_controllers[0], MakeDecision(_))
314 .WillOnce(SetArgPointee<0>(config1));
315 states.audio_network_adaptor->GetEncoderRuntimeConfig();
ivoce1198e02017-09-08 08:13:19 -0700316
317 auto ana_stats = states.audio_network_adaptor->GetStats();
318
ivoc17289092017-09-09 08:45:40 -0700319 EXPECT_EQ(ana_stats.bitrate_action_counter, 2);
320 EXPECT_EQ(ana_stats.channel_action_counter, 2);
321 EXPECT_EQ(ana_stats.dtx_action_counter, 2);
322 EXPECT_EQ(ana_stats.fec_action_counter, 2);
323 EXPECT_EQ(ana_stats.frame_length_increase_counter, 1);
324 EXPECT_EQ(ana_stats.frame_length_decrease_counter, 1);
325 EXPECT_EQ(ana_stats.uplink_packet_loss_fraction, 0.1f);
ivoce1198e02017-09-08 08:13:19 -0700326}
327
minyuecaa9cb22016-09-13 13:34:15 -0700328} // namespace webrtc