blob: 2a757069e81611629ac2fd663c76553aa42cda30 [file] [log] [blame]
turaj@webrtc.orge46c8d32013-05-22 20:39:43 +00001/*
2 * Copyright (c) 2013 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
kwiberg37478382016-02-14 20:40:57 -080011#include <memory>
12
Mirko Bonadei71207422017-09-15 13:58:09 +020013#include "common_types.h" // NOLINT(build/include)
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020014#include "modules/audio_coding/codecs/pcm16b/pcm16b.h"
15#include "modules/audio_coding/include/audio_coding_module.h"
16#include "modules/audio_coding/test/utility.h"
17#include "modules/include/module_common_types.h"
18#include "test/gtest.h"
19#include "test/testsupport/fileutils.h"
turaj@webrtc.orge46c8d32013-05-22 20:39:43 +000020
21namespace webrtc {
turaj@webrtc.orge46c8d32013-05-22 20:39:43 +000022
henrik.lundin@webrtc.orgadaf8092014-04-17 08:29:10 +000023class TargetDelayTest : public ::testing::Test {
24 protected:
25 TargetDelayTest() : acm_(AudioCodingModule::Create(0)) {}
turaj@webrtc.org6ea3d1c2013-10-02 21:44:33 +000026
27 ~TargetDelayTest() {}
turaj@webrtc.orge46c8d32013-05-22 20:39:43 +000028
29 void SetUp() {
andrew@webrtc.org89df0922013-09-12 01:27:43 +000030 EXPECT_TRUE(acm_.get() != NULL);
turaj@webrtc.orge46c8d32013-05-22 20:39:43 +000031
turaj@webrtc.orge46c8d32013-05-22 20:39:43 +000032 ASSERT_EQ(0, acm_->InitializeReceiver());
kwibergda2bf4e2016-10-24 13:47:09 -070033 constexpr int pltype = 108;
34 ASSERT_EQ(true,
35 acm_->RegisterReceiveCodec(pltype, {"L16", kSampleRateHz, 1}));
turaj@webrtc.orge46c8d32013-05-22 20:39:43 +000036
kwibergda2bf4e2016-10-24 13:47:09 -070037 rtp_info_.header.payloadType = pltype;
turaj@webrtc.orge46c8d32013-05-22 20:39:43 +000038 rtp_info_.header.timestamp = 0;
39 rtp_info_.header.ssrc = 0x12345678;
40 rtp_info_.header.markerBit = false;
41 rtp_info_.header.sequenceNumber = 0;
42 rtp_info_.type.Audio.channel = 1;
43 rtp_info_.type.Audio.isCNG = false;
44 rtp_info_.frameType = kAudioFrameSpeech;
turaj@webrtc.org6ea3d1c2013-10-02 21:44:33 +000045
46 int16_t audio[kFrameSizeSamples];
47 const int kRange = 0x7FF; // 2047, easy for masking.
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +000048 for (size_t n = 0; n < kFrameSizeSamples; ++n)
turaj@webrtc.org6ea3d1c2013-10-02 21:44:33 +000049 audio[n] = (rand() & kRange) - kRange / 2;
50 WebRtcPcm16b_Encode(audio, kFrameSizeSamples, payload_);
turaj@webrtc.orge46c8d32013-05-22 20:39:43 +000051 }
52
turaj@webrtc.org6ea3d1c2013-10-02 21:44:33 +000053 void OutOfRangeInput() {
54 EXPECT_EQ(-1, SetMinimumDelay(-1));
55 EXPECT_EQ(-1, SetMinimumDelay(10001));
56 }
57
58 void NoTargetDelayBufferSizeChanges() {
59 for (int n = 0; n < 30; ++n) // Run enough iterations.
60 Run(true);
61 int clean_optimal_delay = GetCurrentOptimalDelayMs();
62 Run(false); // Run with jitter.
63 int jittery_optimal_delay = GetCurrentOptimalDelayMs();
64 EXPECT_GT(jittery_optimal_delay, clean_optimal_delay);
65 int required_delay = RequiredDelay();
66 EXPECT_GT(required_delay, 0);
67 EXPECT_NEAR(required_delay, jittery_optimal_delay, 1);
68 }
69
70 void WithTargetDelayBufferNotChanging() {
71 // A target delay that is one packet larger than jitter.
72 const int kTargetDelayMs = (kInterarrivalJitterPacket + 1) *
73 kNum10msPerFrame * 10;
74 ASSERT_EQ(0, SetMinimumDelay(kTargetDelayMs));
75 for (int n = 0; n < 30; ++n) // Run enough iterations to fill the buffer.
76 Run(true);
77 int clean_optimal_delay = GetCurrentOptimalDelayMs();
78 EXPECT_EQ(kTargetDelayMs, clean_optimal_delay);
79 Run(false); // Run with jitter.
80 int jittery_optimal_delay = GetCurrentOptimalDelayMs();
81 EXPECT_EQ(jittery_optimal_delay, clean_optimal_delay);
82 }
83
84 void RequiredDelayAtCorrectRange() {
85 for (int n = 0; n < 30; ++n) // Run clean and store delay.
86 Run(true);
87 int clean_optimal_delay = GetCurrentOptimalDelayMs();
88
89 // A relatively large delay.
90 const int kTargetDelayMs = (kInterarrivalJitterPacket + 10) *
91 kNum10msPerFrame * 10;
92 ASSERT_EQ(0, SetMinimumDelay(kTargetDelayMs));
93 for (int n = 0; n < 300; ++n) // Run enough iterations to fill the buffer.
94 Run(true);
95 Run(false); // Run with jitter.
96
97 int jittery_optimal_delay = GetCurrentOptimalDelayMs();
98 EXPECT_EQ(kTargetDelayMs, jittery_optimal_delay);
99
100 int required_delay = RequiredDelay();
101
102 // Checking |required_delay| is in correct range.
103 EXPECT_GT(required_delay, 0);
104 EXPECT_GT(jittery_optimal_delay, required_delay);
105 EXPECT_GT(required_delay, clean_optimal_delay);
106
107 // A tighter check for the value of |required_delay|.
108 // The jitter forces a delay of
109 // |kInterarrivalJitterPacket * kNum10msPerFrame * 10| milliseconds. So we
110 // expect |required_delay| be close to that.
111 EXPECT_NEAR(kInterarrivalJitterPacket * kNum10msPerFrame * 10,
112 required_delay, 1);
113 }
114
115 void TargetDelayBufferMinMax() {
116 const int kTargetMinDelayMs = kNum10msPerFrame * 10;
117 ASSERT_EQ(0, SetMinimumDelay(kTargetMinDelayMs));
118 for (int m = 0; m < 30; ++m) // Run enough iterations to fill the buffer.
119 Run(true);
120 int clean_optimal_delay = GetCurrentOptimalDelayMs();
121 EXPECT_EQ(kTargetMinDelayMs, clean_optimal_delay);
122
123 const int kTargetMaxDelayMs = 2 * (kNum10msPerFrame * 10);
124 ASSERT_EQ(0, SetMaximumDelay(kTargetMaxDelayMs));
125 for (int n = 0; n < 30; ++n) // Run enough iterations to fill the buffer.
126 Run(false);
127
128 int capped_optimal_delay = GetCurrentOptimalDelayMs();
129 EXPECT_EQ(kTargetMaxDelayMs, capped_optimal_delay);
130 }
131
132 private:
133 static const int kSampleRateHz = 16000;
134 static const int kNum10msPerFrame = 2;
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +0000135 static const size_t kFrameSizeSamples = 320; // 20 ms @ 16 kHz.
turaj@webrtc.org6ea3d1c2013-10-02 21:44:33 +0000136 // payload-len = frame-samples * 2 bytes/sample.
137 static const int kPayloadLenBytes = 320 * 2;
138 // Inter-arrival time in number of packets in a jittery channel. One is no
139 // jitter.
140 static const int kInterarrivalJitterPacket = 2;
141
turaj@webrtc.orge46c8d32013-05-22 20:39:43 +0000142 void Push() {
143 rtp_info_.header.timestamp += kFrameSizeSamples;
144 rtp_info_.header.sequenceNumber++;
turaj@webrtc.org6ea3d1c2013-10-02 21:44:33 +0000145 ASSERT_EQ(0, acm_->IncomingPacket(payload_, kFrameSizeSamples * 2,
turaj@webrtc.orge46c8d32013-05-22 20:39:43 +0000146 rtp_info_));
147 }
148
149 // Pull audio equivalent to the amount of audio in one RTP packet.
150 void Pull() {
151 AudioFrame frame;
henrik.lundind4ccb002016-05-17 12:21:55 -0700152 bool muted;
turaj@webrtc.orge46c8d32013-05-22 20:39:43 +0000153 for (int k = 0; k < kNum10msPerFrame; ++k) { // Pull one frame.
henrik.lundind4ccb002016-05-17 12:21:55 -0700154 ASSERT_EQ(0, acm_->PlayoutData10Ms(-1, &frame, &muted));
155 ASSERT_FALSE(muted);
turaj@webrtc.orge46c8d32013-05-22 20:39:43 +0000156 // Had to use ASSERT_TRUE, ASSERT_EQ generated error.
157 ASSERT_TRUE(kSampleRateHz == frame.sample_rate_hz_);
Peter Kasting69558702016-01-12 16:26:35 -0800158 ASSERT_EQ(1u, frame.num_channels_);
turaj@webrtc.orge46c8d32013-05-22 20:39:43 +0000159 ASSERT_TRUE(kSampleRateHz / 100 == frame.samples_per_channel_);
160 }
161 }
162
163 void Run(bool clean) {
164 for (int n = 0; n < 10; ++n) {
165 for (int m = 0; m < 5; ++m) {
166 Push();
167 Pull();
168 }
169
170 if (!clean) {
171 for (int m = 0; m < 10; ++m) { // Long enough to trigger delay change.
172 Push();
173 for (int n = 0; n < kInterarrivalJitterPacket; ++n)
174 Pull();
175 }
176 }
177 }
178 }
179
180 int SetMinimumDelay(int delay_ms) {
181 return acm_->SetMinimumPlayoutDelay(delay_ms);
182 }
183
pwestin@webrtc.org401ef362013-08-06 21:01:36 +0000184 int SetMaximumDelay(int delay_ms) {
185 return acm_->SetMaximumPlayoutDelay(delay_ms);
186 }
187
turaj@webrtc.orge46c8d32013-05-22 20:39:43 +0000188 int GetCurrentOptimalDelayMs() {
minyue@webrtc.orgc0bd7be2015-02-18 15:24:13 +0000189 NetworkStatistics stats;
190 acm_->GetNetworkStatistics(&stats);
turaj@webrtc.orge46c8d32013-05-22 20:39:43 +0000191 return stats.preferredBufferSize;
192 }
193
194 int RequiredDelay() {
195 return acm_->LeastRequiredDelayMs();
196 }
197
kwiberg37478382016-02-14 20:40:57 -0800198 std::unique_ptr<AudioCodingModule> acm_;
turaj@webrtc.orge46c8d32013-05-22 20:39:43 +0000199 WebRtcRTPHeader rtp_info_;
turaj@webrtc.org6ea3d1c2013-10-02 21:44:33 +0000200 uint8_t payload_[kPayloadLenBytes];
turaj@webrtc.orge46c8d32013-05-22 20:39:43 +0000201};
202
kjellanderb7d24f62017-02-26 22:10:14 -0800203// Flaky on iOS: webrtc:7057.
204#if defined(WEBRTC_ANDROID) || defined(WEBRTC_IOS)
Peter Boströme2976c82016-01-04 22:44:05 +0100205#define MAYBE_OutOfRangeInput DISABLED_OutOfRangeInput
206#else
207#define MAYBE_OutOfRangeInput OutOfRangeInput
208#endif
209TEST_F(TargetDelayTest, MAYBE_OutOfRangeInput) {
henrik.lundin@webrtc.orgadaf8092014-04-17 08:29:10 +0000210 OutOfRangeInput();
turaj@webrtc.orge46c8d32013-05-22 20:39:43 +0000211}
212
kjellanderb7d24f62017-02-26 22:10:14 -0800213// Flaky on iOS: webrtc:7057.
214#if defined(WEBRTC_ANDROID) || defined(WEBRTC_IOS)
Peter Boströme2976c82016-01-04 22:44:05 +0100215#define MAYBE_NoTargetDelayBufferSizeChanges \
216 DISABLED_NoTargetDelayBufferSizeChanges
217#else
218#define MAYBE_NoTargetDelayBufferSizeChanges NoTargetDelayBufferSizeChanges
219#endif
220TEST_F(TargetDelayTest, MAYBE_NoTargetDelayBufferSizeChanges) {
henrik.lundin@webrtc.orgadaf8092014-04-17 08:29:10 +0000221 NoTargetDelayBufferSizeChanges();
turaj@webrtc.orge46c8d32013-05-22 20:39:43 +0000222}
223
kjellanderb7d24f62017-02-26 22:10:14 -0800224// Flaky on iOS: webrtc:7057.
225#if defined(WEBRTC_ANDROID) || defined(WEBRTC_IOS)
Peter Boströme2976c82016-01-04 22:44:05 +0100226#define MAYBE_WithTargetDelayBufferNotChanging \
227 DISABLED_WithTargetDelayBufferNotChanging
228#else
229#define MAYBE_WithTargetDelayBufferNotChanging WithTargetDelayBufferNotChanging
230#endif
231TEST_F(TargetDelayTest, MAYBE_WithTargetDelayBufferNotChanging) {
henrik.lundin@webrtc.orgadaf8092014-04-17 08:29:10 +0000232 WithTargetDelayBufferNotChanging();
turaj@webrtc.orge46c8d32013-05-22 20:39:43 +0000233}
234
kjellanderb7d24f62017-02-26 22:10:14 -0800235// Flaky on iOS: webrtc:7057.
236#if defined(WEBRTC_ANDROID) || defined(WEBRTC_IOS)
Peter Boströme2976c82016-01-04 22:44:05 +0100237#define MAYBE_RequiredDelayAtCorrectRange DISABLED_RequiredDelayAtCorrectRange
238#else
239#define MAYBE_RequiredDelayAtCorrectRange RequiredDelayAtCorrectRange
240#endif
241TEST_F(TargetDelayTest, MAYBE_RequiredDelayAtCorrectRange) {
henrik.lundin@webrtc.orgadaf8092014-04-17 08:29:10 +0000242 RequiredDelayAtCorrectRange();
turaj@webrtc.orge46c8d32013-05-22 20:39:43 +0000243}
244
kjellanderb7d24f62017-02-26 22:10:14 -0800245// Flaky on iOS: webrtc:7057.
246#if defined(WEBRTC_ANDROID) || defined(WEBRTC_IOS)
Peter Boströme2976c82016-01-04 22:44:05 +0100247#define MAYBE_TargetDelayBufferMinMax DISABLED_TargetDelayBufferMinMax
248#else
249#define MAYBE_TargetDelayBufferMinMax TargetDelayBufferMinMax
250#endif
251TEST_F(TargetDelayTest, MAYBE_TargetDelayBufferMinMax) {
henrik.lundin@webrtc.orgadaf8092014-04-17 08:29:10 +0000252 TargetDelayBufferMinMax();
turaj@webrtc.org6ea3d1c2013-10-02 21:44:33 +0000253}
254
255} // namespace webrtc