blob: d8007613cc284301655fec7aabb065f7417476f9 [file] [log] [blame]
sprang@webrtc.orgccd42842014-01-07 09:54:34 +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
11// This file includes unit tests for SendStatisticsProxy.
12#include "webrtc/video/send_statistics_proxy.h"
13
14#include <map>
15#include <string>
16#include <vector>
17
18#include "testing/gtest/include/gtest/gtest.h"
19
20namespace webrtc {
21
sprang@webrtc.org09315702014-02-07 12:06:29 +000022class SendStatisticsProxyTest : public ::testing::Test,
23 protected SendStatisticsProxy::StatsProvider {
sprang@webrtc.orgccd42842014-01-07 09:54:34 +000024 public:
25 SendStatisticsProxyTest() : avg_delay_ms_(0), max_delay_ms_(0) {}
26 virtual ~SendStatisticsProxyTest() {}
27
28 protected:
29 virtual void SetUp() {
30 statistics_proxy_.reset(
31 new SendStatisticsProxy(GetTestConfig(), this));
32 config_ = GetTestConfig();
33 expected_ = VideoSendStream::Stats();
34 }
35
36 VideoSendStream::Config GetTestConfig() {
37 VideoSendStream::Config config;
38 config.rtp.ssrcs.push_back(17);
39 config.rtp.ssrcs.push_back(42);
40 return config;
41 }
42
43 virtual bool GetSendSideDelay(VideoSendStream::Stats* stats) OVERRIDE {
44 stats->avg_delay_ms = avg_delay_ms_;
45 stats->max_delay_ms = max_delay_ms_;
46 return true;
47 }
48
sprang@webrtc.org09315702014-02-07 12:06:29 +000049 void ExpectEqual(VideoSendStream::Stats one, VideoSendStream::Stats other) {
50 EXPECT_EQ(one.avg_delay_ms, other.avg_delay_ms);
51 EXPECT_EQ(one.input_frame_rate, other.input_frame_rate);
52 EXPECT_EQ(one.encode_frame_rate, other.encode_frame_rate);
53 EXPECT_EQ(one.avg_delay_ms, other.avg_delay_ms);
54 EXPECT_EQ(one.max_delay_ms, other.max_delay_ms);
henrik.lundin@webrtc.orgb10363f2014-03-13 13:31:21 +000055 EXPECT_EQ(one.suspended, other.suspended);
sprang@webrtc.org09315702014-02-07 12:06:29 +000056
57 EXPECT_EQ(one.substreams.size(), other.substreams.size());
58 for (std::map<uint32_t, StreamStats>::const_iterator it =
59 one.substreams.begin();
60 it != one.substreams.end();
61 ++it) {
62 std::map<uint32_t, StreamStats>::const_iterator corresponding_it =
63 other.substreams.find(it->first);
64 ASSERT_TRUE(corresponding_it != other.substreams.end());
65 const StreamStats& a = it->second;
66 const StreamStats& b = corresponding_it->second;
67
68 EXPECT_EQ(a.key_frames, b.key_frames);
69 EXPECT_EQ(a.delta_frames, b.delta_frames);
70 EXPECT_EQ(a.bitrate_bps, b.bitrate_bps);
71
72 EXPECT_EQ(a.rtp_stats.bytes, b.rtp_stats.bytes);
73 EXPECT_EQ(a.rtp_stats.header_bytes, b.rtp_stats.header_bytes);
74 EXPECT_EQ(a.rtp_stats.padding_bytes, b.rtp_stats.padding_bytes);
75 EXPECT_EQ(a.rtp_stats.packets, b.rtp_stats.packets);
76 EXPECT_EQ(a.rtp_stats.retransmitted_packets,
77 b.rtp_stats.retransmitted_packets);
78 EXPECT_EQ(a.rtp_stats.fec_packets, b.rtp_stats.fec_packets);
79
80 EXPECT_EQ(a.rtcp_stats.fraction_lost, b.rtcp_stats.fraction_lost);
81 EXPECT_EQ(a.rtcp_stats.cumulative_lost, b.rtcp_stats.cumulative_lost);
82 EXPECT_EQ(a.rtcp_stats.extended_max_sequence_number,
83 b.rtcp_stats.extended_max_sequence_number);
84 EXPECT_EQ(a.rtcp_stats.jitter, b.rtcp_stats.jitter);
85 }
86 }
87
sprang@webrtc.orgccd42842014-01-07 09:54:34 +000088 scoped_ptr<SendStatisticsProxy> statistics_proxy_;
89 VideoSendStream::Config config_;
90 int avg_delay_ms_;
91 int max_delay_ms_;
sprang@webrtc.orgccd42842014-01-07 09:54:34 +000092 VideoSendStream::Stats expected_;
93 typedef std::map<uint32_t, StreamStats>::const_iterator StreamIterator;
94};
95
96TEST_F(SendStatisticsProxyTest, RtcpStatistics) {
97 RtcpStatisticsCallback* callback = statistics_proxy_.get();
98 for (std::vector<uint32_t>::const_iterator it = config_.rtp.ssrcs.begin();
99 it != config_.rtp.ssrcs.end();
100 ++it) {
101 const uint32_t ssrc = *it;
102 StreamStats& ssrc_stats = expected_.substreams[ssrc];
103
104 // Add statistics with some arbitrary, but unique, numbers.
105 uint32_t offset = ssrc * sizeof(RtcpStatistics);
106 ssrc_stats.rtcp_stats.cumulative_lost = offset;
107 ssrc_stats.rtcp_stats.extended_max_sequence_number = offset + 1;
108 ssrc_stats.rtcp_stats.fraction_lost = offset + 2;
109 ssrc_stats.rtcp_stats.jitter = offset + 3;
110 callback->StatisticsUpdated(ssrc_stats.rtcp_stats, ssrc);
111 }
112
113 VideoSendStream::Stats stats = statistics_proxy_->GetStats();
sprang@webrtc.org09315702014-02-07 12:06:29 +0000114 ExpectEqual(expected_, stats);
sprang@webrtc.orgccd42842014-01-07 09:54:34 +0000115}
116
117TEST_F(SendStatisticsProxyTest, FrameRates) {
118 const int capture_fps = 31;
119 const int encode_fps = 29;
120
121 ViECaptureObserver* capture_observer = statistics_proxy_.get();
122 capture_observer->CapturedFrameRate(0, capture_fps);
123 ViEEncoderObserver* encoder_observer = statistics_proxy_.get();
124 encoder_observer->OutgoingRate(0, encode_fps, 0);
125
126 VideoSendStream::Stats stats = statistics_proxy_->GetStats();
127 EXPECT_EQ(capture_fps, stats.input_frame_rate);
128 EXPECT_EQ(encode_fps, stats.encode_frame_rate);
129}
130
henrik.lundin@webrtc.orgb10363f2014-03-13 13:31:21 +0000131TEST_F(SendStatisticsProxyTest, Suspended) {
132 // Verify that the value is false by default.
133 EXPECT_FALSE(statistics_proxy_->GetStats().suspended);
134
135 // Verify that we can set it to true.
136 ViEEncoderObserver* encoder_observer = statistics_proxy_.get();
137 encoder_observer->SuspendChange(0, true);
138 EXPECT_TRUE(statistics_proxy_->GetStats().suspended);
139
140 // Verify that we can set it back to false again.
141 encoder_observer->SuspendChange(0, false);
142 EXPECT_FALSE(statistics_proxy_->GetStats().suspended);
143}
144
sprang@webrtc.orgccd42842014-01-07 09:54:34 +0000145TEST_F(SendStatisticsProxyTest, FrameCounts) {
146 FrameCountObserver* observer = statistics_proxy_.get();
147 for (std::vector<uint32_t>::const_iterator it = config_.rtp.ssrcs.begin();
148 it != config_.rtp.ssrcs.end();
149 ++it) {
150 const uint32_t ssrc = *it;
151 // Add statistics with some arbitrary, but unique, numbers.
152 StreamStats& stats = expected_.substreams[ssrc];
153 uint32_t offset = ssrc * sizeof(StreamStats);
154 stats.key_frames = offset;
155 stats.delta_frames = offset + 1;
156 observer->FrameCountUpdated(kVideoFrameKey, stats.key_frames, ssrc);
157 observer->FrameCountUpdated(kVideoFrameDelta, stats.delta_frames, ssrc);
158 }
159
160 VideoSendStream::Stats stats = statistics_proxy_->GetStats();
sprang@webrtc.org09315702014-02-07 12:06:29 +0000161 ExpectEqual(expected_, stats);
sprang@webrtc.orgccd42842014-01-07 09:54:34 +0000162}
163
164TEST_F(SendStatisticsProxyTest, DataCounters) {
165 StreamDataCountersCallback* callback = statistics_proxy_.get();
166 for (std::vector<uint32_t>::const_iterator it = config_.rtp.ssrcs.begin();
167 it != config_.rtp.ssrcs.end();
168 ++it) {
169 const uint32_t ssrc = *it;
170 StreamDataCounters& counters = expected_.substreams[ssrc].rtp_stats;
171 // Add statistics with some arbitrary, but unique, numbers.
172 uint32_t offset = ssrc * sizeof(StreamDataCounters);
173 counters.bytes = offset;
174 counters.header_bytes = offset + 1;
175 counters.fec_packets = offset + 2;
176 counters.padding_bytes = offset + 3;
177 counters.retransmitted_packets = offset + 4;
178 counters.packets = offset + 5;
179 callback->DataCountersUpdated(counters, ssrc);
180 }
181
182 VideoSendStream::Stats stats = statistics_proxy_->GetStats();
sprang@webrtc.org09315702014-02-07 12:06:29 +0000183 ExpectEqual(expected_, stats);
sprang@webrtc.orgccd42842014-01-07 09:54:34 +0000184}
185
186TEST_F(SendStatisticsProxyTest, Bitrate) {
187 BitrateStatisticsObserver* observer = statistics_proxy_.get();
188 for (std::vector<uint32_t>::const_iterator it = config_.rtp.ssrcs.begin();
189 it != config_.rtp.ssrcs.end();
190 ++it) {
191 const uint32_t ssrc = *it;
192 BitrateStatistics bitrate;
193 bitrate.bitrate_bps = ssrc;
194 observer->Notify(bitrate, ssrc);
195 expected_.substreams[ssrc].bitrate_bps = ssrc;
196 }
197
198 VideoSendStream::Stats stats = statistics_proxy_->GetStats();
sprang@webrtc.org09315702014-02-07 12:06:29 +0000199 ExpectEqual(expected_, stats);
sprang@webrtc.orgccd42842014-01-07 09:54:34 +0000200}
201
202TEST_F(SendStatisticsProxyTest, StreamStats) {
203 avg_delay_ms_ = 1;
204 max_delay_ms_ = 2;
sprang@webrtc.orgccd42842014-01-07 09:54:34 +0000205
206 VideoSendStream::Stats stats = statistics_proxy_->GetStats();
207
208 EXPECT_EQ(avg_delay_ms_, stats.avg_delay_ms);
209 EXPECT_EQ(max_delay_ms_, stats.max_delay_ms);
sprang@webrtc.orgccd42842014-01-07 09:54:34 +0000210}
211
212TEST_F(SendStatisticsProxyTest, NoSubstreams) {
213 uint32_t exluded_ssrc =
214 *std::max_element(config_.rtp.ssrcs.begin(), config_.rtp.ssrcs.end()) + 1;
215 // From RtcpStatisticsCallback.
216 RtcpStatistics rtcp_stats;
217 RtcpStatisticsCallback* rtcp_callback = statistics_proxy_.get();
218 rtcp_callback->StatisticsUpdated(rtcp_stats, exluded_ssrc);
219
220 // From StreamDataCountersCallback.
221 StreamDataCounters rtp_stats;
222 StreamDataCountersCallback* rtp_callback = statistics_proxy_.get();
223 rtp_callback->DataCountersUpdated(rtp_stats, exluded_ssrc);
224
225 // From BitrateStatisticsObserver.
226 BitrateStatistics bitrate;
227 BitrateStatisticsObserver* bitrate_observer = statistics_proxy_.get();
228 bitrate_observer->Notify(bitrate, exluded_ssrc);
229
230 // From FrameCountObserver.
231 FrameCountObserver* fps_observer = statistics_proxy_.get();
232 fps_observer->FrameCountUpdated(kVideoFrameKey, 1, exluded_ssrc);
233
234 VideoSendStream::Stats stats = statistics_proxy_->GetStats();
235 EXPECT_TRUE(stats.substreams.empty());
236}
237
238} // namespace webrtc