blob: 23d1a3c4075236e11fb76553b05bd5cf58610e3c [file] [log] [blame]
andrew@webrtc.orgb015cbe2012-10-22 18:19:231/*
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
Stefan Holmer3eb722b2016-04-22 13:48:2311#include <algorithm>
12
kjellander19796962017-06-30 17:45:2113#include "webrtc/rtc_base/rate_statistics.h"
kwiberg36a24792016-10-01 05:29:4314#include "webrtc/test/gtest.h"
andrew@webrtc.orgb015cbe2012-10-22 18:19:2315
16namespace {
17
sprang@webrtc.org7374da32013-12-03 10:31:5918using webrtc::RateStatistics;
andrew@webrtc.orgb015cbe2012-10-22 18:19:2319
Stefan Holmer3eb722b2016-04-22 13:48:2320const int64_t kWindowMs = 500;
21
sprang@webrtc.org7374da32013-12-03 10:31:5922class RateStatisticsTest : public ::testing::Test {
solenberg@webrtc.orge5117e72013-04-18 12:25:3223 protected:
Stefan Holmer3eb722b2016-04-22 13:48:2324 RateStatisticsTest() : stats_(kWindowMs, 8000) {}
sprang@webrtc.org7374da32013-12-03 10:31:5925 RateStatistics stats_;
andrew@webrtc.orgb015cbe2012-10-22 18:19:2326};
27
sprang@webrtc.org7374da32013-12-03 10:31:5928TEST_F(RateStatisticsTest, TestStrictMode) {
solenberg@webrtc.orge5117e72013-04-18 12:25:3229 int64_t now_ms = 0;
Erik Språng475c05f2016-06-10 20:13:2130 EXPECT_FALSE(static_cast<bool>(stats_.Rate(now_ms)));
31
32 const uint32_t kPacketSize = 1500u;
33 const uint32_t kExpectedRateBps = kPacketSize * 1000 * 8;
34
35 // Single data point is not enough for valid estimate.
36 stats_.Update(kPacketSize, now_ms++);
37 EXPECT_FALSE(static_cast<bool>(stats_.Rate(now_ms)));
38
Stefan Holmer3eb722b2016-04-22 13:48:2339 // Expecting 1200 kbps since the window is initially kept small and grows as
40 // we have more data.
Erik Språng475c05f2016-06-10 20:13:2141 stats_.Update(kPacketSize, now_ms);
42 EXPECT_EQ(kExpectedRateBps, *stats_.Rate(now_ms));
43
sprang@webrtc.org7374da32013-12-03 10:31:5944 stats_.Reset();
solenberg@webrtc.orge5117e72013-04-18 12:25:3245 // Expecting 0 after init.
Erik Språng475c05f2016-06-10 20:13:2146 EXPECT_FALSE(static_cast<bool>(stats_.Rate(now_ms)));
47
48 const int kInterval = 10;
solenberg@webrtc.orge5117e72013-04-18 12:25:3249 for (int i = 0; i < 100000; ++i) {
Erik Språng475c05f2016-06-10 20:13:2150 if (i % kInterval == 0)
51 stats_.Update(kPacketSize, now_ms);
52
solenberg@webrtc.orge5117e72013-04-18 12:25:3253 // Approximately 1200 kbps expected. Not exact since when packets
54 // are removed we will jump 10 ms to the next packet.
Erik Språng475c05f2016-06-10 20:13:2155 if (i > kInterval) {
56 rtc::Optional<uint32_t> rate = stats_.Rate(now_ms);
57 EXPECT_TRUE(static_cast<bool>(rate));
58 uint32_t samples = i / kInterval + 1;
59 uint64_t total_bits = samples * kPacketSize * 8;
60 uint32_t rate_bps = static_cast<uint32_t>((1000 * total_bits) / (i + 1));
61 EXPECT_NEAR(rate_bps, *rate, 22000u);
solenberg@webrtc.orge5117e72013-04-18 12:25:3262 }
63 now_ms += 1;
64 }
Stefan Holmer3eb722b2016-04-22 13:48:2365 now_ms += kWindowMs;
solenberg@webrtc.orge5117e72013-04-18 12:25:3266 // The window is 2 seconds. If nothing has been received for that time
67 // the estimate should be 0.
Erik Språng475c05f2016-06-10 20:13:2168 EXPECT_FALSE(static_cast<bool>(stats_.Rate(now_ms)));
andrew@webrtc.orgb015cbe2012-10-22 18:19:2369}
mikhal@webrtc.org66fba2b2013-11-25 17:49:2870
sprang@webrtc.org7374da32013-12-03 10:31:5971TEST_F(RateStatisticsTest, IncreasingThenDecreasingBitrate) {
mikhal@webrtc.org66fba2b2013-11-25 17:49:2872 int64_t now_ms = 0;
sprang@webrtc.org7374da32013-12-03 10:31:5973 stats_.Reset();
mikhal@webrtc.org66fba2b2013-11-25 17:49:2874 // Expecting 0 after init.
Erik Språng475c05f2016-06-10 20:13:2175 EXPECT_FALSE(static_cast<bool>(stats_.Rate(now_ms)));
76
77 stats_.Update(1000, ++now_ms);
Stefan Holmer3eb722b2016-04-22 13:48:2378 const uint32_t kExpectedBitrate = 8000000;
mikhal@webrtc.org66fba2b2013-11-25 17:49:2879 // 1000 bytes per millisecond until plateau is reached.
Stefan Holmer3eb722b2016-04-22 13:48:2380 int prev_error = kExpectedBitrate;
Erik Språng475c05f2016-06-10 20:13:2181 rtc::Optional<uint32_t> bitrate;
mikhal@webrtc.org66fba2b2013-11-25 17:49:2882 while (++now_ms < 10000) {
83 stats_.Update(1000, now_ms);
Stefan Holmer3eb722b2016-04-22 13:48:2384 bitrate = stats_.Rate(now_ms);
Erik Språng475c05f2016-06-10 20:13:2185 EXPECT_TRUE(static_cast<bool>(bitrate));
86 int error = kExpectedBitrate - *bitrate;
Stefan Holmer3eb722b2016-04-22 13:48:2387 error = std::abs(error);
88 // Expect the estimation error to decrease as the window is extended.
89 EXPECT_LE(error, prev_error + 1);
90 prev_error = error;
mikhal@webrtc.org66fba2b2013-11-25 17:49:2891 }
Stefan Holmer3eb722b2016-04-22 13:48:2392 // Window filled, expect to be close to 8000000.
Erik Språng475c05f2016-06-10 20:13:2193 EXPECT_EQ(kExpectedBitrate, *bitrate);
Stefan Holmer3eb722b2016-04-22 13:48:2394
mikhal@webrtc.org66fba2b2013-11-25 17:49:2895 // 1000 bytes per millisecond until 10-second mark, 8000 kbps expected.
96 while (++now_ms < 10000) {
97 stats_.Update(1000, now_ms);
sprang@webrtc.org7374da32013-12-03 10:31:5998 bitrate = stats_.Rate(now_ms);
Erik Språng475c05f2016-06-10 20:13:2199 EXPECT_EQ(kExpectedBitrate, *bitrate);
mikhal@webrtc.org66fba2b2013-11-25 17:49:28100 }
Erik Språng475c05f2016-06-10 20:13:21101
mikhal@webrtc.org66fba2b2013-11-25 17:49:28102 // Zero bytes per millisecond until 0 is reached.
103 while (++now_ms < 20000) {
104 stats_.Update(0, now_ms);
Erik Språng475c05f2016-06-10 20:13:21105 rtc::Optional<uint32_t> new_bitrate = stats_.Rate(now_ms);
106 if (static_cast<bool>(new_bitrate) && *new_bitrate != *bitrate) {
mikhal@webrtc.org66fba2b2013-11-25 17:49:28107 // New bitrate must be lower than previous one.
Erik Språng475c05f2016-06-10 20:13:21108 EXPECT_LT(*new_bitrate, *bitrate);
mikhal@webrtc.org66fba2b2013-11-25 17:49:28109 } else {
110 // 0 kbps expected.
Erik Språng475c05f2016-06-10 20:13:21111 EXPECT_EQ(0u, *new_bitrate);
mikhal@webrtc.org66fba2b2013-11-25 17:49:28112 break;
113 }
114 bitrate = new_bitrate;
115 }
Erik Språng475c05f2016-06-10 20:13:21116
mikhal@webrtc.org66fba2b2013-11-25 17:49:28117 // Zero bytes per millisecond until 20-second mark, 0 kbps expected.
118 while (++now_ms < 20000) {
119 stats_.Update(0, now_ms);
Erik Språng475c05f2016-06-10 20:13:21120 EXPECT_EQ(0u, *stats_.Rate(now_ms));
mikhal@webrtc.org66fba2b2013-11-25 17:49:28121 }
122}
Stefan Holmer3eb722b2016-04-22 13:48:23123
124TEST_F(RateStatisticsTest, ResetAfterSilence) {
125 int64_t now_ms = 0;
126 stats_.Reset();
127 // Expecting 0 after init.
Erik Språng475c05f2016-06-10 20:13:21128 EXPECT_FALSE(static_cast<bool>(stats_.Rate(now_ms)));
129
Stefan Holmer3eb722b2016-04-22 13:48:23130 const uint32_t kExpectedBitrate = 8000000;
131 // 1000 bytes per millisecond until the window has been filled.
132 int prev_error = kExpectedBitrate;
Erik Språng475c05f2016-06-10 20:13:21133 rtc::Optional<uint32_t> bitrate;
Stefan Holmer3eb722b2016-04-22 13:48:23134 while (++now_ms < 10000) {
135 stats_.Update(1000, now_ms);
136 bitrate = stats_.Rate(now_ms);
Erik Språng475c05f2016-06-10 20:13:21137 if (bitrate) {
138 int error = kExpectedBitrate - *bitrate;
139 error = std::abs(error);
140 // Expect the estimation error to decrease as the window is extended.
141 EXPECT_LE(error, prev_error + 1);
142 prev_error = error;
143 }
Stefan Holmer3eb722b2016-04-22 13:48:23144 }
145 // Window filled, expect to be close to 8000000.
Erik Språng475c05f2016-06-10 20:13:21146 EXPECT_EQ(kExpectedBitrate, *bitrate);
Stefan Holmer3eb722b2016-04-22 13:48:23147
148 now_ms += kWindowMs + 1;
Erik Språng475c05f2016-06-10 20:13:21149 EXPECT_FALSE(static_cast<bool>(stats_.Rate(now_ms)));
Stefan Holmer3eb722b2016-04-22 13:48:23150 stats_.Update(1000, now_ms);
Erik Språng475c05f2016-06-10 20:13:21151 ++now_ms;
152 stats_.Update(1000, now_ms);
153 // We expect two samples of 1000 bytes, and that the bitrate is measured over
154 // 500 ms, i.e. 2 * 8 * 1000 / 0.500 = 32000.
155 EXPECT_EQ(32000u, *stats_.Rate(now_ms));
156
157 // Reset, add the same samples again.
158 stats_.Reset();
159 EXPECT_FALSE(static_cast<bool>(stats_.Rate(now_ms)));
160 stats_.Update(1000, now_ms);
161 ++now_ms;
162 stats_.Update(1000, now_ms);
163 // We expect two samples of 1000 bytes, and that the bitrate is measured over
164 // 2 ms (window size has been reset) i.e. 2 * 8 * 1000 / 0.002 = 8000000.
165 EXPECT_EQ(kExpectedBitrate, *stats_.Rate(now_ms));
166}
167
168TEST_F(RateStatisticsTest, HandlesChangingWindowSize) {
169 int64_t now_ms = 0;
170 stats_.Reset();
171
172 // Sanity test window size.
173 EXPECT_TRUE(stats_.SetWindowSize(kWindowMs, now_ms));
174 EXPECT_FALSE(stats_.SetWindowSize(kWindowMs + 1, now_ms));
175 EXPECT_FALSE(stats_.SetWindowSize(0, now_ms));
176 EXPECT_TRUE(stats_.SetWindowSize(1, now_ms));
177 EXPECT_TRUE(stats_.SetWindowSize(kWindowMs, now_ms));
178
179 // Fill the buffer at a rate of 1 byte / millisecond (8 kbps).
180 const int kBatchSize = 10;
181 for (int i = 0; i <= kWindowMs; i += kBatchSize)
182 stats_.Update(kBatchSize, now_ms += kBatchSize);
183 EXPECT_EQ(static_cast<uint32_t>(8000), *stats_.Rate(now_ms));
184
185 // Halve the window size, rate should stay the same.
186 EXPECT_TRUE(stats_.SetWindowSize(kWindowMs / 2, now_ms));
187 EXPECT_EQ(static_cast<uint32_t>(8000), *stats_.Rate(now_ms));
188
189 // Double the window size again, rate should stay the same. (As the window
190 // won't actually expand until new bit and bobs fall into it.
191 EXPECT_TRUE(stats_.SetWindowSize(kWindowMs, now_ms));
192 EXPECT_EQ(static_cast<uint32_t>(8000), *stats_.Rate(now_ms));
193
194 // Fill the now empty half with bits it twice the rate.
195 for (int i = 0; i < kWindowMs / 2; i += kBatchSize)
196 stats_.Update(kBatchSize * 2, now_ms += kBatchSize);
197
198 // Rate should have increase be 50%.
199 EXPECT_EQ(static_cast<uint32_t>((8000 * 3) / 2), *stats_.Rate(now_ms));
200}
201
202TEST_F(RateStatisticsTest, RespectsWindowSizeEdges) {
203 int64_t now_ms = 0;
204 stats_.Reset();
205 // Expecting 0 after init.
206 EXPECT_FALSE(static_cast<bool>(stats_.Rate(now_ms)));
207
208 // One byte per ms, using one big sample.
209 stats_.Update(kWindowMs, now_ms);
210 now_ms += kWindowMs - 2;
211 // Shouldn't work! (Only one sample, not full window size.)
212 EXPECT_FALSE(static_cast<bool>(stats_.Rate(now_ms)));
213
214 // Window size should be full, and the single data point should be accepted.
215 ++now_ms;
216 rtc::Optional<uint32_t> bitrate = stats_.Rate(now_ms);
217 EXPECT_TRUE(static_cast<bool>(bitrate));
218 EXPECT_EQ(1000 * 8u, *bitrate);
219
220 // Add another, now we have twice the bitrate.
221 stats_.Update(kWindowMs, now_ms);
222 bitrate = stats_.Rate(now_ms);
223 EXPECT_TRUE(static_cast<bool>(bitrate));
224 EXPECT_EQ(2 * 1000 * 8u, *bitrate);
225
226 // Now that first sample should drop out...
227 now_ms += 1;
228 bitrate = stats_.Rate(now_ms);
229 EXPECT_TRUE(static_cast<bool>(bitrate));
230 EXPECT_EQ(1000 * 8u, *bitrate);
231}
232
233TEST_F(RateStatisticsTest, HandlesZeroCounts) {
234 int64_t now_ms = 0;
235 stats_.Reset();
236 // Expecting 0 after init.
237 EXPECT_FALSE(static_cast<bool>(stats_.Rate(now_ms)));
238
239 stats_.Update(kWindowMs, now_ms);
240 now_ms += kWindowMs - 1;
241 stats_.Update(0, now_ms);
242 rtc::Optional<uint32_t> bitrate = stats_.Rate(now_ms);
243 EXPECT_TRUE(static_cast<bool>(bitrate));
244 EXPECT_EQ(1000 * 8u, *bitrate);
245
246 // Move window along so first data point falls out.
247 ++now_ms;
248 bitrate = stats_.Rate(now_ms);
249 EXPECT_TRUE(static_cast<bool>(bitrate));
250 EXPECT_EQ(0u, *bitrate);
251
252 // Move window so last data point falls out.
253 now_ms += kWindowMs;
254 EXPECT_FALSE(static_cast<bool>(stats_.Rate(now_ms)));
255}
256
257TEST_F(RateStatisticsTest, HandlesQuietPeriods) {
258 int64_t now_ms = 0;
259 stats_.Reset();
260 // Expecting 0 after init.
261 EXPECT_FALSE(static_cast<bool>(stats_.Rate(now_ms)));
262
263 stats_.Update(0, now_ms);
264 now_ms += kWindowMs - 1;
265 rtc::Optional<uint32_t> bitrate = stats_.Rate(now_ms);
266 EXPECT_TRUE(static_cast<bool>(bitrate));
267 EXPECT_EQ(0u, *bitrate);
268
269 // Move window along so first data point falls out.
270 ++now_ms;
271 EXPECT_FALSE(static_cast<bool>(stats_.Rate(now_ms)));
272
273 // Move window a long way out.
274 now_ms += 2 * kWindowMs;
275 stats_.Update(0, now_ms);
276 bitrate = stats_.Rate(now_ms);
277 EXPECT_TRUE(static_cast<bool>(bitrate));
278 EXPECT_EQ(0u, *bitrate);
Stefan Holmer3eb722b2016-04-22 13:48:23279}
solenberg@webrtc.orge5117e72013-04-18 12:25:32280} // namespace