blob: c29350e80b5d76c1b4b80f4113a05248959ecfe1 [file] [log] [blame]
henrik.lundin@webrtc.orgd94659d2013-01-29 12:09:21 +00001/*
2 * Copyright (c) 2012 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// Unit tests for DecisionLogic class and derived classes.
12
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020013#include "modules/audio_coding/neteq/decision_logic.h"
Jonas Olssona4d87372019-07-05 19:08:33 +020014
Ivo Creusen3ce44a32019-10-31 14:38:11 +010015#include "api/neteq/neteq_controller.h"
16#include "api/neteq/tick_timer.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020017#include "modules/audio_coding/neteq/buffer_level_filter.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020018#include "modules/audio_coding/neteq/delay_manager.h"
Jakob Ivarsson609b0472020-10-19 09:19:34 +020019#include "modules/audio_coding/neteq/mock/mock_buffer_level_filter.h"
20#include "modules/audio_coding/neteq/mock/mock_delay_manager.h"
21#include "test/field_trial.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020022#include "test/gtest.h"
henrik.lundin@webrtc.orgd94659d2013-01-29 12:09:21 +000023
24namespace webrtc {
25
Jakob Ivarsson609b0472020-10-19 09:19:34 +020026namespace {
27
28constexpr int kSampleRate = 8000;
29constexpr int kSamplesPerMs = kSampleRate / 1000;
30constexpr int kOutputSizeSamples = kSamplesPerMs * 10;
31constexpr int kMinTimescaleInterval = 5;
32
33NetEqController::NetEqStatus CreateNetEqStatus(NetEq::Mode last_mode,
34 int current_delay_ms) {
35 NetEqController::NetEqStatus status;
36 status.play_dtmf = false;
37 status.last_mode = last_mode;
38 status.target_timestamp = 1234;
39 status.generated_noise_samples = 0;
40 status.expand_mutefactor = 0;
41 status.packet_buffer_info.num_samples = current_delay_ms * kSamplesPerMs;
42 status.packet_buffer_info.span_samples = current_delay_ms * kSamplesPerMs;
43 status.packet_buffer_info.span_samples_no_dtx =
44 current_delay_ms * kSamplesPerMs;
45 status.packet_buffer_info.dtx_or_cng = false;
46 status.next_packet = {status.target_timestamp, false, false};
47 return status;
henrik.lundin@webrtc.orgd94659d2013-01-29 12:09:21 +000048}
49
Jakob Ivarsson609b0472020-10-19 09:19:34 +020050using ::testing::Return;
51
52} // namespace
53
54class DecisionLogicTest : public ::testing::Test {
55 protected:
56 DecisionLogicTest() {
57 test::ScopedFieldTrials field_trial(
58 "WebRTC-Audio-NetEqDecisionLogicSettings/"
59 "estimate_dtx_delay:true,time_stretch_cn:true/");
60
61 NetEqController::Config config;
62 config.tick_timer = &tick_timer_;
63 config.allow_time_stretching = true;
64 std::unique_ptr<Histogram> histogram =
65 std::make_unique<Histogram>(200, 12345, 2);
66 auto delay_manager = std::make_unique<MockDelayManager>(
67 200, 0, 12300, config.tick_timer, std::move(histogram));
68 mock_delay_manager_ = delay_manager.get();
69 auto buffer_level_filter = std::make_unique<MockBufferLevelFilter>();
70 mock_buffer_level_filter_ = buffer_level_filter.get();
71 decision_logic_ = std::make_unique<DecisionLogic>(
72 config, std::move(delay_manager), std::move(buffer_level_filter));
73 decision_logic_->SetSampleRate(kSampleRate, kOutputSizeSamples);
74 }
75
76 TickTimer tick_timer_;
77 std::unique_ptr<DecisionLogic> decision_logic_;
78 MockDelayManager* mock_delay_manager_;
79 MockBufferLevelFilter* mock_buffer_level_filter_;
80};
81
82TEST_F(DecisionLogicTest, NormalOperation) {
83 EXPECT_CALL(*mock_delay_manager_, TargetDelayMs())
84 .WillRepeatedly(Return(100));
85 EXPECT_CALL(*mock_buffer_level_filter_, filtered_current_level())
86 .WillRepeatedly(Return(90 * kSamplesPerMs));
87
88 bool reset_decoder = false;
89 tick_timer_.Increment(kMinTimescaleInterval + 1);
90 EXPECT_EQ(decision_logic_->GetDecision(
91 CreateNetEqStatus(NetEq::Mode::kNormal, 100), &reset_decoder),
92 NetEq::Operation::kNormal);
93 EXPECT_FALSE(reset_decoder);
94}
95
96TEST_F(DecisionLogicTest, Accelerate) {
97 EXPECT_CALL(*mock_delay_manager_, TargetDelayMs())
98 .WillRepeatedly(Return(100));
99 EXPECT_CALL(*mock_buffer_level_filter_, filtered_current_level())
100 .WillRepeatedly(Return(110 * kSamplesPerMs));
101
102 bool reset_decoder = false;
103 tick_timer_.Increment(kMinTimescaleInterval + 1);
104 EXPECT_EQ(decision_logic_->GetDecision(
105 CreateNetEqStatus(NetEq::Mode::kNormal, 100), &reset_decoder),
106 NetEq::Operation::kAccelerate);
107 EXPECT_FALSE(reset_decoder);
108}
109
110TEST_F(DecisionLogicTest, FastAccelerate) {
111 EXPECT_CALL(*mock_delay_manager_, TargetDelayMs())
112 .WillRepeatedly(Return(100));
113 EXPECT_CALL(*mock_buffer_level_filter_, filtered_current_level())
114 .WillRepeatedly(Return(400 * kSamplesPerMs));
115
116 bool reset_decoder = false;
117 tick_timer_.Increment(kMinTimescaleInterval + 1);
118 EXPECT_EQ(decision_logic_->GetDecision(
119 CreateNetEqStatus(NetEq::Mode::kNormal, 100), &reset_decoder),
120 NetEq::Operation::kFastAccelerate);
121 EXPECT_FALSE(reset_decoder);
122}
123
124TEST_F(DecisionLogicTest, PreemptiveExpand) {
125 EXPECT_CALL(*mock_delay_manager_, TargetDelayMs())
126 .WillRepeatedly(Return(100));
127 EXPECT_CALL(*mock_buffer_level_filter_, filtered_current_level())
128 .WillRepeatedly(Return(50 * kSamplesPerMs));
129
130 bool reset_decoder = false;
131 tick_timer_.Increment(kMinTimescaleInterval + 1);
132 EXPECT_EQ(decision_logic_->GetDecision(
133 CreateNetEqStatus(NetEq::Mode::kNormal, 100), &reset_decoder),
134 NetEq::Operation::kPreemptiveExpand);
135 EXPECT_FALSE(reset_decoder);
136}
137
138TEST_F(DecisionLogicTest, DecelerationTargetLevelOffset) {
139 EXPECT_CALL(*mock_delay_manager_, TargetDelayMs())
140 .WillRepeatedly(Return(500));
141 EXPECT_CALL(*mock_buffer_level_filter_, filtered_current_level())
142 .WillRepeatedly(Return(400 * kSamplesPerMs));
143
144 bool reset_decoder = false;
145 tick_timer_.Increment(kMinTimescaleInterval + 1);
146 EXPECT_EQ(decision_logic_->GetDecision(
147 CreateNetEqStatus(NetEq::Mode::kNormal, 400), &reset_decoder),
148 NetEq::Operation::kPreemptiveExpand);
149 EXPECT_FALSE(reset_decoder);
150}
151
152TEST_F(DecisionLogicTest, PostponeDecodeAfterExpand) {
153 EXPECT_CALL(*mock_delay_manager_, TargetDelayMs())
154 .WillRepeatedly(Return(500));
155
156 // Below 50% target delay threshold.
157 bool reset_decoder = false;
158 EXPECT_EQ(decision_logic_->GetDecision(
159 CreateNetEqStatus(NetEq::Mode::kExpand, 200), &reset_decoder),
160 NetEq::Operation::kExpand);
161 EXPECT_FALSE(reset_decoder);
162
163 // Above 50% target delay threshold.
164 EXPECT_EQ(decision_logic_->GetDecision(
165 CreateNetEqStatus(NetEq::Mode::kExpand, 250), &reset_decoder),
166 NetEq::Operation::kNormal);
167 EXPECT_FALSE(reset_decoder);
168}
169
170TEST_F(DecisionLogicTest, TimeStrechComfortNoise) {
171 EXPECT_CALL(*mock_delay_manager_, TargetDelayMs())
172 .WillRepeatedly(Return(500));
173
174 {
175 bool reset_decoder = false;
176 // Below target window.
177 auto status = CreateNetEqStatus(NetEq::Mode::kCodecInternalCng, 400);
178 status.generated_noise_samples = 400 * kSamplesPerMs;
179 status.next_packet->timestamp =
180 status.target_timestamp + 400 * kSamplesPerMs;
181 EXPECT_EQ(decision_logic_->GetDecision(status, &reset_decoder),
182 NetEq::Operation::kCodecInternalCng);
183 EXPECT_FALSE(reset_decoder);
184 }
185
186 {
187 bool reset_decoder = false;
188 // Above target window.
189 auto status = CreateNetEqStatus(NetEq::Mode::kCodecInternalCng, 600);
190 status.generated_noise_samples = 200 * kSamplesPerMs;
191 status.next_packet->timestamp =
192 status.target_timestamp + 400 * kSamplesPerMs;
193 EXPECT_EQ(decision_logic_->GetDecision(status, &reset_decoder),
194 NetEq::Operation::kNormal);
195 EXPECT_FALSE(reset_decoder);
196
197 // The buffer level filter should be adjusted with the number of samples
198 // that was skipped.
199 int timestamp_leap = status.next_packet->timestamp -
200 status.target_timestamp -
201 status.generated_noise_samples;
202 EXPECT_CALL(*mock_buffer_level_filter_,
203 Update(400 * kSamplesPerMs, timestamp_leap));
204 EXPECT_EQ(decision_logic_->GetDecision(
205 CreateNetEqStatus(NetEq::Mode::kNormal, 400), &reset_decoder),
206 NetEq::Operation::kNormal);
207 EXPECT_FALSE(reset_decoder);
208 }
209}
henrik.lundin@webrtc.orgd94659d2013-01-29 12:09:21 +0000210
211} // namespace webrtc