blob: 094e78fa71c7010a8d77619b57b90942e0fd526c [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
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020011#ifndef VIDEO_SEND_STATISTICS_PROXY_H_
12#define VIDEO_SEND_STATISTICS_PROXY_H_
sprang@webrtc.orgccd42842014-01-07 09:54:34 +000013
mflodmand1590b22015-12-09 07:07:59 -080014#include <map>
kwiberg27f982b2016-03-01 11:52:33 -080015#include <memory>
sprang@webrtc.orgccd42842014-01-07 09:54:34 +000016#include <string>
Pera48ddb72016-09-29 11:48:50 +020017#include <vector>
sprang@webrtc.orgccd42842014-01-07 09:54:34 +000018
Mirko Bonadei71207422017-09-15 13:58:09 +020019#include "common_types.h" // NOLINT(build/include)
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020020#include "modules/video_coding/include/video_codec_interface.h"
21#include "modules/video_coding/include/video_coding_defines.h"
22#include "rtc_base/criticalsection.h"
23#include "rtc_base/numerics/exp_filter.h"
24#include "rtc_base/ratetracker.h"
25#include "rtc_base/thread_annotations.h"
26#include "system_wrappers/include/clock.h"
27#include "video/overuse_frame_detector.h"
28#include "video/report_block_stats.h"
29#include "video/stats_counter.h"
30#include "video/video_stream_encoder.h"
31#include "call/video_send_stream.h"
sprang@webrtc.orgccd42842014-01-07 09:54:34 +000032
33namespace webrtc {
34
pbos@webrtc.org3e6e2712015-02-26 12:19:31 +000035class SendStatisticsProxy : public CpuOveruseMetricsObserver,
36 public RtcpStatisticsCallback,
pbos@webrtc.org1d0fa5d2015-02-19 12:47:00 +000037 public RtcpPacketTypeCounterObserver,
sprang@webrtc.orgccd42842014-01-07 09:54:34 +000038 public StreamDataCountersCallback,
39 public BitrateStatisticsObserver,
40 public FrameCountObserver,
stefan@webrtc.org168f23f2014-07-11 13:44:02 +000041 public SendSideDelayObserver {
sprang@webrtc.orgccd42842014-01-07 09:54:34 +000042 public:
pbos@webrtc.org273a4142014-12-01 15:23:21 +000043 static const int kStatsTimeoutMs;
perkj803d97f2016-11-01 11:45:46 -070044 // Number of required samples to be collected before a metric is added
45 // to a rtc histogram.
46 static const int kMinRequiredMetricsSamples = 200;
pbos@webrtc.org273a4142014-12-01 15:23:21 +000047
sprangb4a1ae52015-12-03 08:10:08 -080048 SendStatisticsProxy(Clock* clock,
49 const VideoSendStream::Config& config,
50 VideoEncoderConfig::ContentType content_type);
sprang@webrtc.orgccd42842014-01-07 09:54:34 +000051 virtual ~SendStatisticsProxy();
52
sprangc5d62e22017-04-02 23:53:04 -070053 virtual VideoSendStream::Stats GetStats();
pbos@webrtc.org273a4142014-12-01 15:23:21 +000054
55 virtual void OnSendEncodedImage(const EncodedImage& encoded_image,
kjellander02b3d272016-04-20 05:05:54 -070056 const CodecSpecificInfo* codec_info);
perkj@webrtc.orgaf612d52015-03-18 09:51:05 +000057 // Used to update incoming frame rate.
asaperssond89920b2015-07-22 06:52:00 -070058 void OnIncomingFrame(int width, int height);
sprang@webrtc.orgccd42842014-01-07 09:54:34 +000059
Ilya Nikolaevskiyd79314f2017-10-23 10:45:37 +020060 // Dropped frame stats.
61 void OnFrameDroppedBySource();
62 void OnFrameDroppedInEncoderQueue();
63 void OnFrameDroppedByEncoder();
64 void OnFrameDroppedByMediaOptimizations();
65
asapersson09f05612017-05-15 23:40:18 -070066 // Adaptation stats.
mflodmancc3d4422017-08-03 08:27:51 -070067 void SetAdaptationStats(
68 const VideoStreamEncoder::AdaptCounts& cpu_counts,
69 const VideoStreamEncoder::AdaptCounts& quality_counts);
70 void OnCpuAdaptationChanged(
71 const VideoStreamEncoder::AdaptCounts& cpu_counts,
72 const VideoStreamEncoder::AdaptCounts& quality_counts);
asapersson09f05612017-05-15 23:40:18 -070073 void OnQualityAdaptationChanged(
mflodmancc3d4422017-08-03 08:27:51 -070074 const VideoStreamEncoder::AdaptCounts& cpu_counts,
75 const VideoStreamEncoder::AdaptCounts& quality_counts);
Åsa Perssonc3ed6302017-11-16 14:04:52 +010076 void OnMinPixelLimitReached();
perkj803d97f2016-11-01 11:45:46 -070077
Peter Boström7083e112015-09-22 16:28:51 +020078 void OnSuspendChange(bool is_suspended);
Peter Boström20f3f942015-05-15 11:33:39 +020079 void OnInactiveSsrc(uint32_t ssrc);
80
sprangb4a1ae52015-12-03 08:10:08 -080081 // Used to indicate change in content type, which may require a change in
Pera48ddb72016-09-29 11:48:50 +020082 // how stats are collected and set the configured preferred media bitrate.
83 void OnEncoderReconfigured(const VideoEncoderConfig& encoder_config,
84 uint32_t preferred_bitrate_bps);
sprangb4a1ae52015-12-03 08:10:08 -080085
perkjf5b2e512016-07-05 08:34:04 -070086 // Used to update the encoder target rate.
87 void OnSetEncoderTargetRate(uint32_t bitrate_bps);
Peter Boströme4499152016-02-05 11:13:28 +010088
89 // Implements CpuOveruseMetricsObserver.
90 void OnEncodedFrameTimeMeasured(int encode_time_ms,
91 const CpuOveruseMetrics& metrics) override;
92
Per69b332d2016-06-02 15:45:42 +020093 int GetSendFrameRate() const;
94
sprang@webrtc.orgccd42842014-01-07 09:54:34 +000095 protected:
96 // From RtcpStatisticsCallback.
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +000097 void StatisticsUpdated(const RtcpStatistics& statistics,
98 uint32_t ssrc) override;
99 void CNameChanged(const char* cname, uint32_t ssrc) override;
asaperssond89920b2015-07-22 06:52:00 -0700100 // From RtcpPacketTypeCounterObserver.
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000101 void RtcpPacketTypesCounterUpdated(
pbos@webrtc.org1d0fa5d2015-02-19 12:47:00 +0000102 uint32_t ssrc,
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000103 const RtcpPacketTypeCounter& packet_counter) override;
sprang@webrtc.orgccd42842014-01-07 09:54:34 +0000104 // From StreamDataCountersCallback.
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000105 void DataCountersUpdated(const StreamDataCounters& counters,
106 uint32_t ssrc) override;
sprang@webrtc.orgccd42842014-01-07 09:54:34 +0000107
108 // From BitrateStatisticsObserver.
sprangcd349d92016-07-13 09:11:28 -0700109 void Notify(uint32_t total_bitrate_bps,
110 uint32_t retransmit_bitrate_bps,
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000111 uint32_t ssrc) override;
sprang@webrtc.orgccd42842014-01-07 09:54:34 +0000112
113 // From FrameCountObserver.
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000114 void FrameCountUpdated(const FrameCounts& frame_counts,
115 uint32_t ssrc) override;
sprang@webrtc.orgccd42842014-01-07 09:54:34 +0000116
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000117 void SendSideDelayUpdated(int avg_delay_ms,
118 int max_delay_ms,
119 uint32_t ssrc) override;
stefan@webrtc.org168f23f2014-07-11 13:44:02 +0000120
sprang@webrtc.orgccd42842014-01-07 09:54:34 +0000121 private:
asaperssonda535c42015-10-19 23:32:41 -0700122 class SampleCounter {
123 public:
asaperssond89920b2015-07-22 06:52:00 -0700124 SampleCounter() : sum(0), num_samples(0) {}
asaperssonda535c42015-10-19 23:32:41 -0700125 ~SampleCounter() {}
asaperssond89920b2015-07-22 06:52:00 -0700126 void Add(int sample);
asapersson66d4b372016-12-19 06:50:53 -0800127 int Avg(int64_t min_required_samples) const;
asaperssond89920b2015-07-22 06:52:00 -0700128
129 private:
asapersson66d4b372016-12-19 06:50:53 -0800130 int64_t sum;
131 int64_t num_samples;
asaperssond89920b2015-07-22 06:52:00 -0700132 };
asaperssonda535c42015-10-19 23:32:41 -0700133 class BoolSampleCounter {
134 public:
asaperssondec5ebf2015-10-05 02:36:17 -0700135 BoolSampleCounter() : sum(0), num_samples(0) {}
asaperssonda535c42015-10-19 23:32:41 -0700136 ~BoolSampleCounter() {}
asaperssondec5ebf2015-10-05 02:36:17 -0700137 void Add(bool sample);
asapersson66d4b372016-12-19 06:50:53 -0800138 void Add(bool sample, int64_t count);
139 int Percent(int64_t min_required_samples) const;
140 int Permille(int64_t min_required_samples) const;
asaperssondec5ebf2015-10-05 02:36:17 -0700141
142 private:
asapersson66d4b372016-12-19 06:50:53 -0800143 int Fraction(int64_t min_required_samples, float multiplier) const;
144 int64_t sum;
145 int64_t num_samples;
asaperssondec5ebf2015-10-05 02:36:17 -0700146 };
pbos@webrtc.org273a4142014-12-01 15:23:21 +0000147 struct StatsUpdateTimes {
sprangb4a1ae52015-12-03 08:10:08 -0800148 StatsUpdateTimes() : resolution_update_ms(0), bitrate_update_ms(0) {}
pbos@webrtc.org273a4142014-12-01 15:23:21 +0000149 int64_t resolution_update_ms;
Peter Boström20f3f942015-05-15 11:33:39 +0200150 int64_t bitrate_update_ms;
pbos@webrtc.org273a4142014-12-01 15:23:21 +0000151 };
asapersson66d4b372016-12-19 06:50:53 -0800152 struct TargetRateUpdates {
153 TargetRateUpdates()
154 : pause_resume_events(0), last_paused_or_resumed(false), last_ms(-1) {}
155 int pause_resume_events;
156 bool last_paused_or_resumed;
157 int64_t last_ms;
158 };
asapersson8d75ac72017-09-15 06:41:15 -0700159 struct FallbackEncoderInfo {
160 bool is_possible = true;
161 bool is_active = false;
162 int on_off_events = 0;
163 int64_t elapsed_ms = 0;
164 rtc::Optional<int64_t> last_update_ms;
165 const int max_frame_diff_ms = 2000;
166 };
Åsa Perssonc3ed6302017-11-16 14:04:52 +0100167 struct FallbackEncoderInfoDisabled {
168 bool is_possible = true;
169 bool min_pixel_limit_reached = false;
170 };
asapersson6eca98b2017-04-04 23:40:50 -0700171 struct StatsTimer {
172 void Start(int64_t now_ms);
173 void Stop(int64_t now_ms);
174 void Restart(int64_t now_ms);
175 int64_t start_ms = -1;
176 int64_t total_ms = 0;
177 };
asapersson118ef002016-03-31 00:00:19 -0700178 struct QpCounters {
asapersson6eca98b2017-04-04 23:40:50 -0700179 SampleCounter vp8; // QP range: 0-127.
180 SampleCounter vp9; // QP range: 0-255.
181 SampleCounter h264; // QP range: 0-51.
asapersson118ef002016-03-31 00:00:19 -0700182 };
Åsa Persson0122e842017-10-16 12:19:23 +0200183
184 // Map holding encoded frames (mapped by timestamp).
185 // If simulcast layers are encoded on different threads, there is no guarantee
186 // that one frame of all layers are encoded before the next start.
187 struct TimestampOlderThan {
188 bool operator()(uint32_t ts1, uint32_t ts2) const {
189 return IsNewerTimestamp(ts2, ts1);
190 }
191 };
192 struct Frame {
193 Frame(int64_t send_ms, uint32_t width, uint32_t height)
194 : send_ms(send_ms), max_width(width), max_height(height) {}
195 const int64_t
196 send_ms; // Time when first frame with this timestamp is sent.
197 uint32_t max_width; // Max width with this timestamp.
198 uint32_t max_height; // Max height with this timestamp.
199 };
200 typedef std::map<uint32_t, Frame, TimestampOlderThan> EncodedFrameMap;
201
danilchapa37de392017-09-09 04:17:22 -0700202 void PurgeOldStats() RTC_EXCLUSIVE_LOCKS_REQUIRED(crit_);
pbos@webrtc.org09c77b92015-02-25 10:42:16 +0000203 VideoSendStream::StreamStats* GetStatsEntry(uint32_t ssrc)
danilchapa37de392017-09-09 04:17:22 -0700204 RTC_EXCLUSIVE_LOCKS_REQUIRED(crit_);
sprang@webrtc.orgccd42842014-01-07 09:54:34 +0000205
mflodmancc3d4422017-08-03 08:27:51 -0700206 void SetAdaptTimer(const VideoStreamEncoder::AdaptCounts& counts,
danilchapa37de392017-09-09 04:17:22 -0700207 StatsTimer* timer) RTC_EXCLUSIVE_LOCKS_REQUIRED(crit_);
mflodmancc3d4422017-08-03 08:27:51 -0700208 void UpdateAdaptationStats(
209 const VideoStreamEncoder::AdaptCounts& cpu_counts,
210 const VideoStreamEncoder::AdaptCounts& quality_counts)
danilchapa37de392017-09-09 04:17:22 -0700211 RTC_EXCLUSIVE_LOCKS_REQUIRED(crit_);
asapersson09f05612017-05-15 23:40:18 -0700212
Åsa Persson45bbc8a2017-11-13 10:16:47 +0100213 void UpdateEncoderFallbackStats(const CodecSpecificInfo* codec_info,
214 int pixels)
asapersson8d75ac72017-09-15 06:41:15 -0700215 RTC_EXCLUSIVE_LOCKS_REQUIRED(crit_);
Åsa Perssonc3ed6302017-11-16 14:04:52 +0100216 void UpdateFallbackDisabledStats(const CodecSpecificInfo* codec_info,
217 int pixels)
218 RTC_EXCLUSIVE_LOCKS_REQUIRED(crit_);
asapersson8d75ac72017-09-15 06:41:15 -0700219
pbos@webrtc.org273a4142014-12-01 15:23:21 +0000220 Clock* const clock_;
perkj26091b12016-09-01 01:17:40 -0700221 const std::string payload_name_;
222 const VideoSendStream::Config::Rtp rtp_config_;
Åsa Persson45bbc8a2017-11-13 10:16:47 +0100223 const rtc::Optional<int> fallback_max_pixels_;
Åsa Perssonc3ed6302017-11-16 14:04:52 +0100224 const rtc::Optional<int> fallback_max_pixels_disabled_;
pbos5ad935c2016-01-25 03:52:44 -0800225 rtc::CriticalSection crit_;
danilchapa37de392017-09-09 04:17:22 -0700226 VideoEncoderConfig::ContentType content_type_ RTC_GUARDED_BY(crit_);
asapersson4374a092016-07-27 00:39:09 -0700227 const int64_t start_ms_;
danilchapa37de392017-09-09 04:17:22 -0700228 VideoSendStream::Stats stats_ RTC_GUARDED_BY(crit_);
danilchapa37de392017-09-09 04:17:22 -0700229 std::map<uint32_t, StatsUpdateTimes> update_times_ RTC_GUARDED_BY(crit_);
230 rtc::ExpFilter encode_time_ RTC_GUARDED_BY(crit_);
231 int quality_downscales_ RTC_GUARDED_BY(crit_);
232 int cpu_downscales_ RTC_GUARDED_BY(crit_);
Åsa Persson0122e842017-10-16 12:19:23 +0200233 rtc::RateTracker media_byte_rate_tracker_ RTC_GUARDED_BY(crit_);
234 rtc::RateTracker encoded_frame_rate_tracker_ RTC_GUARDED_BY(crit_);
asaperssond89920b2015-07-22 06:52:00 -0700235
sprangb4a1ae52015-12-03 08:10:08 -0800236 // Contains stats used for UMA histograms. These stats will be reset if
237 // content type changes between real-time video and screenshare, since these
238 // will be reported separately.
239 struct UmaSamplesContainer {
sprang07fb9be2016-02-24 07:55:00 -0800240 UmaSamplesContainer(const char* prefix,
241 const VideoSendStream::Stats& start_stats,
242 Clock* clock);
sprangb4a1ae52015-12-03 08:10:08 -0800243 ~UmaSamplesContainer();
244
perkj26091b12016-09-01 01:17:40 -0700245 void UpdateHistograms(const VideoSendStream::Config::Rtp& rtp_config,
sprang07fb9be2016-02-24 07:55:00 -0800246 const VideoSendStream::Stats& current_stats);
sprangb4a1ae52015-12-03 08:10:08 -0800247
asapersson93e1e232017-02-06 05:18:35 -0800248 void InitializeBitrateCounters(const VideoSendStream::Stats& stats);
249
Åsa Persson0122e842017-10-16 12:19:23 +0200250 bool InsertEncodedFrame(const EncodedImage& encoded_frame);
251 void RemoveOld(int64_t now_ms);
252
sprangb4a1ae52015-12-03 08:10:08 -0800253 const std::string uma_prefix_;
sprang07fb9be2016-02-24 07:55:00 -0800254 Clock* const clock_;
sprangb4a1ae52015-12-03 08:10:08 -0800255 SampleCounter input_width_counter_;
256 SampleCounter input_height_counter_;
257 SampleCounter sent_width_counter_;
258 SampleCounter sent_height_counter_;
259 SampleCounter encode_time_counter_;
260 BoolSampleCounter key_frame_counter_;
261 BoolSampleCounter quality_limited_frame_counter_;
262 SampleCounter quality_downscales_counter_;
perkj803d97f2016-11-01 11:45:46 -0700263 BoolSampleCounter cpu_limited_frame_counter_;
sprangb4a1ae52015-12-03 08:10:08 -0800264 BoolSampleCounter bw_limited_frame_counter_;
265 SampleCounter bw_resolutions_disabled_counter_;
266 SampleCounter delay_counter_;
267 SampleCounter max_delay_counter_;
268 rtc::RateTracker input_frame_rate_tracker_;
asapersson320e45a2016-11-29 01:40:35 -0800269 RateCounter input_fps_counter_;
270 RateCounter sent_fps_counter_;
asapersson93e1e232017-02-06 05:18:35 -0800271 RateAccCounter total_byte_counter_;
272 RateAccCounter media_byte_counter_;
273 RateAccCounter rtx_byte_counter_;
274 RateAccCounter padding_byte_counter_;
275 RateAccCounter retransmit_byte_counter_;
276 RateAccCounter fec_byte_counter_;
sprange2d83d62016-02-19 09:03:26 -0800277 int64_t first_rtcp_stats_time_ms_;
Erik Språng22c2b482016-03-01 09:40:42 +0100278 int64_t first_rtp_stats_time_ms_;
asapersson09f05612017-05-15 23:40:18 -0700279 StatsTimer cpu_adapt_timer_;
280 StatsTimer quality_adapt_timer_;
asapersson66d4b372016-12-19 06:50:53 -0800281 BoolSampleCounter paused_time_counter_;
282 TargetRateUpdates target_rate_updates_;
asapersson8d75ac72017-09-15 06:41:15 -0700283 BoolSampleCounter fallback_active_counter_;
284 FallbackEncoderInfo fallback_info_;
Åsa Perssonc3ed6302017-11-16 14:04:52 +0100285 FallbackEncoderInfoDisabled fallback_info_disabled_;
sprange2d83d62016-02-19 09:03:26 -0800286 ReportBlockStats report_block_stats_;
sprang07fb9be2016-02-24 07:55:00 -0800287 const VideoSendStream::Stats start_stats_;
Åsa Persson0122e842017-10-16 12:19:23 +0200288 EncodedFrameMap encoded_frames_;
sprangc5d62e22017-04-02 23:53:04 -0700289
asapersson118ef002016-03-31 00:00:19 -0700290 std::map<int, QpCounters>
291 qp_counters_; // QP counters mapped by spatial idx.
sprangb4a1ae52015-12-03 08:10:08 -0800292 };
293
danilchapa37de392017-09-09 04:17:22 -0700294 std::unique_ptr<UmaSamplesContainer> uma_container_ RTC_GUARDED_BY(crit_);
sprang@webrtc.orgccd42842014-01-07 09:54:34 +0000295};
296
297} // namespace webrtc
Mirko Bonadei92ea95e2017-09-15 06:47:31 +0200298#endif // VIDEO_SEND_STATISTICS_PROXY_H_