niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 1 | /* |
stefan@webrtc.org | 9354cc9 | 2012-06-07 08:10:14 +0000 | [diff] [blame] | 2 | * Copyright (c) 2012 The WebRTC project authors. All Rights Reserved. |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 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 Holmer | fb8fc53 | 2016-04-22 15:48:23 +0200 | [diff] [blame] | 11 | #include <algorithm> |
| 12 | |
pbos@webrtc.org | 47ce120 | 2013-05-27 12:41:33 +0000 | [diff] [blame] | 13 | #include "testing/gtest/include/gtest/gtest.h" |
tkchin | f75d008 | 2016-02-23 22:49:42 -0800 | [diff] [blame] | 14 | #include "webrtc/base/rate_statistics.h" |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 15 | |
| 16 | namespace { |
| 17 | |
sprang@webrtc.org | 37968a9 | 2013-12-03 10:31:59 +0000 | [diff] [blame] | 18 | using webrtc::RateStatistics; |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 19 | |
Stefan Holmer | fb8fc53 | 2016-04-22 15:48:23 +0200 | [diff] [blame] | 20 | const int64_t kWindowMs = 500; |
| 21 | |
sprang@webrtc.org | 37968a9 | 2013-12-03 10:31:59 +0000 | [diff] [blame] | 22 | class RateStatisticsTest : public ::testing::Test { |
solenberg@webrtc.org | d26457f | 2013-04-18 12:25:32 +0000 | [diff] [blame] | 23 | protected: |
Stefan Holmer | fb8fc53 | 2016-04-22 15:48:23 +0200 | [diff] [blame] | 24 | RateStatisticsTest() : stats_(kWindowMs, 8000) {} |
sprang@webrtc.org | 37968a9 | 2013-12-03 10:31:59 +0000 | [diff] [blame] | 25 | RateStatistics stats_; |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 26 | }; |
| 27 | |
sprang@webrtc.org | 37968a9 | 2013-12-03 10:31:59 +0000 | [diff] [blame] | 28 | TEST_F(RateStatisticsTest, TestStrictMode) { |
solenberg@webrtc.org | d26457f | 2013-04-18 12:25:32 +0000 | [diff] [blame] | 29 | int64_t now_ms = 0; |
| 30 | // Should be initialized to 0. |
sprang@webrtc.org | 37968a9 | 2013-12-03 10:31:59 +0000 | [diff] [blame] | 31 | EXPECT_EQ(0u, stats_.Rate(now_ms)); |
solenberg@webrtc.org | d26457f | 2013-04-18 12:25:32 +0000 | [diff] [blame] | 32 | stats_.Update(1500, now_ms); |
Stefan Holmer | fb8fc53 | 2016-04-22 15:48:23 +0200 | [diff] [blame] | 33 | // Expecting 1200 kbps since the window is initially kept small and grows as |
| 34 | // we have more data. |
| 35 | EXPECT_EQ(12000000u, stats_.Rate(now_ms)); |
sprang@webrtc.org | 37968a9 | 2013-12-03 10:31:59 +0000 | [diff] [blame] | 36 | stats_.Reset(); |
solenberg@webrtc.org | d26457f | 2013-04-18 12:25:32 +0000 | [diff] [blame] | 37 | // Expecting 0 after init. |
sprang@webrtc.org | 37968a9 | 2013-12-03 10:31:59 +0000 | [diff] [blame] | 38 | EXPECT_EQ(0u, stats_.Rate(now_ms)); |
solenberg@webrtc.org | d26457f | 2013-04-18 12:25:32 +0000 | [diff] [blame] | 39 | for (int i = 0; i < 100000; ++i) { |
| 40 | if (now_ms % 10 == 0) { |
| 41 | stats_.Update(1500, now_ms); |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 42 | } |
solenberg@webrtc.org | d26457f | 2013-04-18 12:25:32 +0000 | [diff] [blame] | 43 | // Approximately 1200 kbps expected. Not exact since when packets |
| 44 | // are removed we will jump 10 ms to the next packet. |
Stefan Holmer | fb8fc53 | 2016-04-22 15:48:23 +0200 | [diff] [blame] | 45 | if (now_ms > 0 && now_ms % kWindowMs == 0) { |
| 46 | EXPECT_NEAR(1200000u, stats_.Rate(now_ms), 22000u); |
solenberg@webrtc.org | d26457f | 2013-04-18 12:25:32 +0000 | [diff] [blame] | 47 | } |
| 48 | now_ms += 1; |
| 49 | } |
Stefan Holmer | fb8fc53 | 2016-04-22 15:48:23 +0200 | [diff] [blame] | 50 | now_ms += kWindowMs; |
solenberg@webrtc.org | d26457f | 2013-04-18 12:25:32 +0000 | [diff] [blame] | 51 | // The window is 2 seconds. If nothing has been received for that time |
| 52 | // the estimate should be 0. |
sprang@webrtc.org | 37968a9 | 2013-12-03 10:31:59 +0000 | [diff] [blame] | 53 | EXPECT_EQ(0u, stats_.Rate(now_ms)); |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 54 | } |
mikhal@webrtc.org | d89b52a | 2013-11-25 17:49:28 +0000 | [diff] [blame] | 55 | |
sprang@webrtc.org | 37968a9 | 2013-12-03 10:31:59 +0000 | [diff] [blame] | 56 | TEST_F(RateStatisticsTest, IncreasingThenDecreasingBitrate) { |
mikhal@webrtc.org | d89b52a | 2013-11-25 17:49:28 +0000 | [diff] [blame] | 57 | int64_t now_ms = 0; |
sprang@webrtc.org | 37968a9 | 2013-12-03 10:31:59 +0000 | [diff] [blame] | 58 | stats_.Reset(); |
mikhal@webrtc.org | d89b52a | 2013-11-25 17:49:28 +0000 | [diff] [blame] | 59 | // Expecting 0 after init. |
sprang@webrtc.org | 37968a9 | 2013-12-03 10:31:59 +0000 | [diff] [blame] | 60 | uint32_t bitrate = stats_.Rate(now_ms); |
mikhal@webrtc.org | d89b52a | 2013-11-25 17:49:28 +0000 | [diff] [blame] | 61 | EXPECT_EQ(0u, bitrate); |
Stefan Holmer | fb8fc53 | 2016-04-22 15:48:23 +0200 | [diff] [blame] | 62 | const uint32_t kExpectedBitrate = 8000000; |
mikhal@webrtc.org | d89b52a | 2013-11-25 17:49:28 +0000 | [diff] [blame] | 63 | // 1000 bytes per millisecond until plateau is reached. |
Stefan Holmer | fb8fc53 | 2016-04-22 15:48:23 +0200 | [diff] [blame] | 64 | int prev_error = kExpectedBitrate; |
mikhal@webrtc.org | d89b52a | 2013-11-25 17:49:28 +0000 | [diff] [blame] | 65 | while (++now_ms < 10000) { |
| 66 | stats_.Update(1000, now_ms); |
Stefan Holmer | fb8fc53 | 2016-04-22 15:48:23 +0200 | [diff] [blame] | 67 | bitrate = stats_.Rate(now_ms); |
| 68 | int error = kExpectedBitrate - bitrate; |
| 69 | error = std::abs(error); |
| 70 | // Expect the estimation error to decrease as the window is extended. |
| 71 | EXPECT_LE(error, prev_error + 1); |
| 72 | prev_error = error; |
mikhal@webrtc.org | d89b52a | 2013-11-25 17:49:28 +0000 | [diff] [blame] | 73 | } |
Stefan Holmer | fb8fc53 | 2016-04-22 15:48:23 +0200 | [diff] [blame] | 74 | // Window filled, expect to be close to 8000000. |
| 75 | EXPECT_EQ(kExpectedBitrate, bitrate); |
| 76 | |
mikhal@webrtc.org | d89b52a | 2013-11-25 17:49:28 +0000 | [diff] [blame] | 77 | // 1000 bytes per millisecond until 10-second mark, 8000 kbps expected. |
| 78 | while (++now_ms < 10000) { |
| 79 | stats_.Update(1000, now_ms); |
sprang@webrtc.org | 37968a9 | 2013-12-03 10:31:59 +0000 | [diff] [blame] | 80 | bitrate = stats_.Rate(now_ms); |
Stefan Holmer | fb8fc53 | 2016-04-22 15:48:23 +0200 | [diff] [blame] | 81 | EXPECT_EQ(kExpectedBitrate, bitrate); |
mikhal@webrtc.org | d89b52a | 2013-11-25 17:49:28 +0000 | [diff] [blame] | 82 | } |
| 83 | // Zero bytes per millisecond until 0 is reached. |
| 84 | while (++now_ms < 20000) { |
| 85 | stats_.Update(0, now_ms); |
sprang@webrtc.org | 37968a9 | 2013-12-03 10:31:59 +0000 | [diff] [blame] | 86 | uint32_t new_bitrate = stats_.Rate(now_ms); |
mikhal@webrtc.org | d89b52a | 2013-11-25 17:49:28 +0000 | [diff] [blame] | 87 | if (new_bitrate != bitrate) { |
| 88 | // New bitrate must be lower than previous one. |
| 89 | EXPECT_LT(new_bitrate, bitrate); |
| 90 | } else { |
| 91 | // 0 kbps expected. |
| 92 | EXPECT_EQ(0u, bitrate); |
| 93 | break; |
| 94 | } |
| 95 | bitrate = new_bitrate; |
| 96 | } |
| 97 | // Zero bytes per millisecond until 20-second mark, 0 kbps expected. |
| 98 | while (++now_ms < 20000) { |
| 99 | stats_.Update(0, now_ms); |
sprang@webrtc.org | 37968a9 | 2013-12-03 10:31:59 +0000 | [diff] [blame] | 100 | EXPECT_EQ(0u, stats_.Rate(now_ms)); |
mikhal@webrtc.org | d89b52a | 2013-11-25 17:49:28 +0000 | [diff] [blame] | 101 | } |
| 102 | } |
Stefan Holmer | fb8fc53 | 2016-04-22 15:48:23 +0200 | [diff] [blame] | 103 | |
| 104 | TEST_F(RateStatisticsTest, ResetAfterSilence) { |
| 105 | int64_t now_ms = 0; |
| 106 | stats_.Reset(); |
| 107 | // Expecting 0 after init. |
| 108 | uint32_t bitrate = stats_.Rate(now_ms); |
| 109 | EXPECT_EQ(0u, bitrate); |
| 110 | const uint32_t kExpectedBitrate = 8000000; |
| 111 | // 1000 bytes per millisecond until the window has been filled. |
| 112 | int prev_error = kExpectedBitrate; |
| 113 | while (++now_ms < 10000) { |
| 114 | stats_.Update(1000, now_ms); |
| 115 | bitrate = stats_.Rate(now_ms); |
| 116 | int error = kExpectedBitrate - bitrate; |
| 117 | error = std::abs(error); |
| 118 | // Expect the estimation error to decrease as the window is extended. |
| 119 | EXPECT_LE(error, prev_error + 1); |
| 120 | prev_error = error; |
| 121 | } |
| 122 | // Window filled, expect to be close to 8000000. |
| 123 | EXPECT_EQ(kExpectedBitrate, bitrate); |
| 124 | |
| 125 | now_ms += kWindowMs + 1; |
| 126 | EXPECT_EQ(0u, stats_.Rate(now_ms)); |
| 127 | stats_.Update(1000, now_ms); |
| 128 | // We expect one sample of 1000 bytes, and that the bitrate is measured over |
| 129 | // 1 ms, i.e., 8 * 1000 / 0.001 = 8000000. |
| 130 | EXPECT_EQ(kExpectedBitrate, stats_.Rate(now_ms)); |
| 131 | } |
solenberg@webrtc.org | d26457f | 2013-04-18 12:25:32 +0000 | [diff] [blame] | 132 | } // namespace |