blob: ca8f5423689e95b6221f8e2882244373b6158512 [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
Niels Möller213618e2018-07-24 09:29:58 +020019#include "api/video/video_stream_encoder_observer.h"
Yves Gerey665174f2018-06-19 15:03:05 +020020#include "call/video_send_stream.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020021#include "modules/video_coding/include/video_codec_interface.h"
22#include "modules/video_coding/include/video_coding_defines.h"
Steve Anton10542f22019-01-11 09:11:00 -080023#include "rtc_base/critical_section.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020024#include "rtc_base/numerics/exp_filter.h"
Steve Anton10542f22019-01-11 09:11:00 -080025#include "rtc_base/rate_tracker.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020026#include "rtc_base/thread_annotations.h"
27#include "system_wrappers/include/clock.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020028#include "video/report_block_stats.h"
29#include "video/stats_counter.h"
sprang@webrtc.orgccd42842014-01-07 09:54:34 +000030
31namespace webrtc {
32
Niels Möller213618e2018-07-24 09:29:58 +020033class SendStatisticsProxy : public VideoStreamEncoderObserver,
pbos@webrtc.org3e6e2712015-02-26 12:19:31 +000034 public RtcpStatisticsCallback,
pbos@webrtc.org1d0fa5d2015-02-19 12:47:00 +000035 public RtcpPacketTypeCounterObserver,
sprang@webrtc.orgccd42842014-01-07 09:54:34 +000036 public StreamDataCountersCallback,
37 public BitrateStatisticsObserver,
38 public FrameCountObserver,
stefan@webrtc.org168f23f2014-07-11 13:44:02 +000039 public SendSideDelayObserver {
sprang@webrtc.orgccd42842014-01-07 09:54:34 +000040 public:
pbos@webrtc.org273a4142014-12-01 15:23:21 +000041 static const int kStatsTimeoutMs;
perkj803d97f2016-11-01 11:45:46 -070042 // Number of required samples to be collected before a metric is added
43 // to a rtc histogram.
44 static const int kMinRequiredMetricsSamples = 200;
pbos@webrtc.org273a4142014-12-01 15:23:21 +000045
sprangb4a1ae52015-12-03 08:10:08 -080046 SendStatisticsProxy(Clock* clock,
47 const VideoSendStream::Config& config,
48 VideoEncoderConfig::ContentType content_type);
Stefan Holmerdbdb3a02018-07-17 16:03:46 +020049 ~SendStatisticsProxy() override;
sprang@webrtc.orgccd42842014-01-07 09:54:34 +000050
sprangc5d62e22017-04-02 23:53:04 -070051 virtual VideoSendStream::Stats GetStats();
pbos@webrtc.org273a4142014-12-01 15:23:21 +000052
Stefan Holmerdbdb3a02018-07-17 16:03:46 +020053 void OnSendEncodedImage(const EncodedImage& encoded_image,
Niels Möller213618e2018-07-24 09:29:58 +020054 const CodecSpecificInfo* codec_info) override;
Erik Språnge2fd86a2018-10-24 11:32:39 +020055
56 void OnEncoderImplementationChanged(
57 const std::string& implementation_name) override;
58
perkj@webrtc.orgaf612d52015-03-18 09:51:05 +000059 // Used to update incoming frame rate.
Niels Möller213618e2018-07-24 09:29:58 +020060 void OnIncomingFrame(int width, int height) override;
sprang@webrtc.orgccd42842014-01-07 09:54:34 +000061
Ilya Nikolaevskiyd79314f2017-10-23 10:45:37 +020062 // Dropped frame stats.
Niels Möller213618e2018-07-24 09:29:58 +020063 void OnFrameDropped(DropReason) override;
Ilya Nikolaevskiyd79314f2017-10-23 10:45:37 +020064
asapersson09f05612017-05-15 23:40:18 -070065 // Adaptation stats.
Niels Möller213618e2018-07-24 09:29:58 +020066 void OnAdaptationChanged(AdaptationReason reason,
67 const AdaptationSteps& cpu_counts,
68 const AdaptationSteps& quality_counts) override;
perkj803d97f2016-11-01 11:45:46 -070069
Niels Möller213618e2018-07-24 09:29:58 +020070 void OnMinPixelLimitReached() override;
71 void OnInitialQualityResolutionAdaptDown() override;
72
73 void OnSuspendChange(bool is_suspended) override;
Peter Boström20f3f942015-05-15 11:33:39 +020074 void OnInactiveSsrc(uint32_t ssrc);
75
sprangb4a1ae52015-12-03 08:10:08 -080076 // Used to indicate change in content type, which may require a change in
Niels Möller97e04882018-05-25 09:43:26 +020077 // how stats are collected.
Pera48ddb72016-09-29 11:48:50 +020078 void OnEncoderReconfigured(const VideoEncoderConfig& encoder_config,
Niels Möller213618e2018-07-24 09:29:58 +020079 const std::vector<VideoStream>& streams) override;
sprangb4a1ae52015-12-03 08:10:08 -080080
perkjf5b2e512016-07-05 08:34:04 -070081 // Used to update the encoder target rate.
82 void OnSetEncoderTargetRate(uint32_t bitrate_bps);
Peter Boströme4499152016-02-05 11:13:28 +010083
84 // Implements CpuOveruseMetricsObserver.
85 void OnEncodedFrameTimeMeasured(int encode_time_ms,
Niels Möller213618e2018-07-24 09:29:58 +020086 int encode_usage_percent) override;
Peter Boströme4499152016-02-05 11:13:28 +010087
Niels Möller213618e2018-07-24 09:29:58 +020088 int GetInputFrameRate() const override;
Per69b332d2016-06-02 15:45:42 +020089 int GetSendFrameRate() const;
90
sprang@webrtc.orgccd42842014-01-07 09:54:34 +000091 protected:
92 // From RtcpStatisticsCallback.
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +000093 void StatisticsUpdated(const RtcpStatistics& statistics,
94 uint32_t ssrc) override;
95 void CNameChanged(const char* cname, uint32_t ssrc) override;
asaperssond89920b2015-07-22 06:52:00 -070096 // From RtcpPacketTypeCounterObserver.
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +000097 void RtcpPacketTypesCounterUpdated(
pbos@webrtc.org1d0fa5d2015-02-19 12:47:00 +000098 uint32_t ssrc,
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +000099 const RtcpPacketTypeCounter& packet_counter) override;
sprang@webrtc.orgccd42842014-01-07 09:54:34 +0000100 // From StreamDataCountersCallback.
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000101 void DataCountersUpdated(const StreamDataCounters& counters,
102 uint32_t ssrc) override;
sprang@webrtc.orgccd42842014-01-07 09:54:34 +0000103
104 // From BitrateStatisticsObserver.
sprangcd349d92016-07-13 09:11:28 -0700105 void Notify(uint32_t total_bitrate_bps,
106 uint32_t retransmit_bitrate_bps,
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000107 uint32_t ssrc) override;
sprang@webrtc.orgccd42842014-01-07 09:54:34 +0000108
109 // From FrameCountObserver.
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000110 void FrameCountUpdated(const FrameCounts& frame_counts,
111 uint32_t ssrc) override;
sprang@webrtc.orgccd42842014-01-07 09:54:34 +0000112
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000113 void SendSideDelayUpdated(int avg_delay_ms,
114 int max_delay_ms,
115 uint32_t ssrc) override;
stefan@webrtc.org168f23f2014-07-11 13:44:02 +0000116
sprang@webrtc.orgccd42842014-01-07 09:54:34 +0000117 private:
asaperssonda535c42015-10-19 23:32:41 -0700118 class SampleCounter {
119 public:
asaperssond89920b2015-07-22 06:52:00 -0700120 SampleCounter() : sum(0), num_samples(0) {}
asaperssonda535c42015-10-19 23:32:41 -0700121 ~SampleCounter() {}
asaperssond89920b2015-07-22 06:52:00 -0700122 void Add(int sample);
asapersson66d4b372016-12-19 06:50:53 -0800123 int Avg(int64_t min_required_samples) const;
asaperssond89920b2015-07-22 06:52:00 -0700124
125 private:
asapersson66d4b372016-12-19 06:50:53 -0800126 int64_t sum;
127 int64_t num_samples;
asaperssond89920b2015-07-22 06:52:00 -0700128 };
asaperssonda535c42015-10-19 23:32:41 -0700129 class BoolSampleCounter {
130 public:
asaperssondec5ebf2015-10-05 02:36:17 -0700131 BoolSampleCounter() : sum(0), num_samples(0) {}
asaperssonda535c42015-10-19 23:32:41 -0700132 ~BoolSampleCounter() {}
asaperssondec5ebf2015-10-05 02:36:17 -0700133 void Add(bool sample);
asapersson66d4b372016-12-19 06:50:53 -0800134 void Add(bool sample, int64_t count);
135 int Percent(int64_t min_required_samples) const;
136 int Permille(int64_t min_required_samples) const;
asaperssondec5ebf2015-10-05 02:36:17 -0700137
138 private:
asapersson66d4b372016-12-19 06:50:53 -0800139 int Fraction(int64_t min_required_samples, float multiplier) const;
140 int64_t sum;
141 int64_t num_samples;
asaperssondec5ebf2015-10-05 02:36:17 -0700142 };
pbos@webrtc.org273a4142014-12-01 15:23:21 +0000143 struct StatsUpdateTimes {
sprangb4a1ae52015-12-03 08:10:08 -0800144 StatsUpdateTimes() : resolution_update_ms(0), bitrate_update_ms(0) {}
pbos@webrtc.org273a4142014-12-01 15:23:21 +0000145 int64_t resolution_update_ms;
Peter Boström20f3f942015-05-15 11:33:39 +0200146 int64_t bitrate_update_ms;
pbos@webrtc.org273a4142014-12-01 15:23:21 +0000147 };
asapersson66d4b372016-12-19 06:50:53 -0800148 struct TargetRateUpdates {
149 TargetRateUpdates()
150 : pause_resume_events(0), last_paused_or_resumed(false), last_ms(-1) {}
151 int pause_resume_events;
152 bool last_paused_or_resumed;
153 int64_t last_ms;
154 };
asapersson8d75ac72017-09-15 06:41:15 -0700155 struct FallbackEncoderInfo {
Mirko Bonadei8fdcac32018-08-28 16:30:18 +0200156 FallbackEncoderInfo();
asapersson8d75ac72017-09-15 06:41:15 -0700157 bool is_possible = true;
158 bool is_active = false;
159 int on_off_events = 0;
160 int64_t elapsed_ms = 0;
Danil Chapovalovb9b146c2018-06-15 12:28:07 +0200161 absl::optional<int64_t> last_update_ms;
asapersson8d75ac72017-09-15 06:41:15 -0700162 const int max_frame_diff_ms = 2000;
163 };
Åsa Perssonc3ed6302017-11-16 14:04:52 +0100164 struct FallbackEncoderInfoDisabled {
165 bool is_possible = true;
166 bool min_pixel_limit_reached = false;
167 };
asapersson6eca98b2017-04-04 23:40:50 -0700168 struct StatsTimer {
169 void Start(int64_t now_ms);
170 void Stop(int64_t now_ms);
171 void Restart(int64_t now_ms);
172 int64_t start_ms = -1;
173 int64_t total_ms = 0;
174 };
asapersson118ef002016-03-31 00:00:19 -0700175 struct QpCounters {
asapersson6eca98b2017-04-04 23:40:50 -0700176 SampleCounter vp8; // QP range: 0-127.
177 SampleCounter vp9; // QP range: 0-255.
178 SampleCounter h264; // QP range: 0-51.
asapersson118ef002016-03-31 00:00:19 -0700179 };
Åsa Persson875841d2018-01-08 08:49:53 +0100180 struct AdaptChanges {
181 int down = 0;
182 int up = 0;
183 };
Åsa Persson0122e842017-10-16 12:19:23 +0200184
185 // Map holding encoded frames (mapped by timestamp).
186 // If simulcast layers are encoded on different threads, there is no guarantee
187 // that one frame of all layers are encoded before the next start.
188 struct TimestampOlderThan {
189 bool operator()(uint32_t ts1, uint32_t ts2) const {
190 return IsNewerTimestamp(ts2, ts1);
191 }
192 };
193 struct Frame {
Niels Möllerd3b8c632018-08-27 15:33:42 +0200194 Frame(int64_t send_ms, uint32_t width, uint32_t height, int simulcast_idx)
Åsa Perssonaa329e72017-12-15 15:54:44 +0100195 : send_ms(send_ms),
196 max_width(width),
197 max_height(height),
198 max_simulcast_idx(simulcast_idx) {}
Åsa Persson0122e842017-10-16 12:19:23 +0200199 const int64_t
200 send_ms; // Time when first frame with this timestamp is sent.
201 uint32_t max_width; // Max width with this timestamp.
202 uint32_t max_height; // Max height with this timestamp.
Niels Möllerd3b8c632018-08-27 15:33:42 +0200203 int max_simulcast_idx; // Max simulcast index with this timestamp.
Åsa Persson0122e842017-10-16 12:19:23 +0200204 };
205 typedef std::map<uint32_t, Frame, TimestampOlderThan> EncodedFrameMap;
206
danilchapa37de392017-09-09 04:17:22 -0700207 void PurgeOldStats() RTC_EXCLUSIVE_LOCKS_REQUIRED(crit_);
pbos@webrtc.org09c77b92015-02-25 10:42:16 +0000208 VideoSendStream::StreamStats* GetStatsEntry(uint32_t ssrc)
danilchapa37de392017-09-09 04:17:22 -0700209 RTC_EXCLUSIVE_LOCKS_REQUIRED(crit_);
sprang@webrtc.orgccd42842014-01-07 09:54:34 +0000210
Niels Möller213618e2018-07-24 09:29:58 +0200211 void SetAdaptTimer(const AdaptationSteps& counts, StatsTimer* timer)
212 RTC_EXCLUSIVE_LOCKS_REQUIRED(crit_);
213 void UpdateAdaptationStats(const AdaptationSteps& cpu_counts,
214 const AdaptationSteps& quality_counts)
danilchapa37de392017-09-09 04:17:22 -0700215 RTC_EXCLUSIVE_LOCKS_REQUIRED(crit_);
Åsa Persson875841d2018-01-08 08:49:53 +0100216 void TryUpdateInitialQualityResolutionAdaptUp(
Niels Möller213618e2018-07-24 09:29:58 +0200217 const AdaptationSteps& quality_counts)
Åsa Persson875841d2018-01-08 08:49:53 +0100218 RTC_EXCLUSIVE_LOCKS_REQUIRED(crit_);
asapersson09f05612017-05-15 23:40:18 -0700219
Åsa Persson45bbc8a2017-11-13 10:16:47 +0100220 void UpdateEncoderFallbackStats(const CodecSpecificInfo* codec_info,
Niels Möllerd3b8c632018-08-27 15:33:42 +0200221 int pixels,
222 int simulcast_index)
asapersson8d75ac72017-09-15 06:41:15 -0700223 RTC_EXCLUSIVE_LOCKS_REQUIRED(crit_);
Åsa Perssonc3ed6302017-11-16 14:04:52 +0100224 void UpdateFallbackDisabledStats(const CodecSpecificInfo* codec_info,
Niels Möllerd3b8c632018-08-27 15:33:42 +0200225 int pixels,
226 int simulcast_index)
Åsa Perssonc3ed6302017-11-16 14:04:52 +0100227 RTC_EXCLUSIVE_LOCKS_REQUIRED(crit_);
asapersson8d75ac72017-09-15 06:41:15 -0700228
pbos@webrtc.org273a4142014-12-01 15:23:21 +0000229 Clock* const clock_;
perkj26091b12016-09-01 01:17:40 -0700230 const std::string payload_name_;
Stefan Holmerdbdb3a02018-07-17 16:03:46 +0200231 const RtpConfig rtp_config_;
Danil Chapovalovb9b146c2018-06-15 12:28:07 +0200232 const absl::optional<int> fallback_max_pixels_;
233 const absl::optional<int> fallback_max_pixels_disabled_;
pbos5ad935c2016-01-25 03:52:44 -0800234 rtc::CriticalSection crit_;
danilchapa37de392017-09-09 04:17:22 -0700235 VideoEncoderConfig::ContentType content_type_ RTC_GUARDED_BY(crit_);
asapersson4374a092016-07-27 00:39:09 -0700236 const int64_t start_ms_;
danilchapa37de392017-09-09 04:17:22 -0700237 VideoSendStream::Stats stats_ RTC_GUARDED_BY(crit_);
danilchapa37de392017-09-09 04:17:22 -0700238 std::map<uint32_t, StatsUpdateTimes> update_times_ RTC_GUARDED_BY(crit_);
239 rtc::ExpFilter encode_time_ RTC_GUARDED_BY(crit_);
240 int quality_downscales_ RTC_GUARDED_BY(crit_);
241 int cpu_downscales_ RTC_GUARDED_BY(crit_);
Åsa Persson0122e842017-10-16 12:19:23 +0200242 rtc::RateTracker media_byte_rate_tracker_ RTC_GUARDED_BY(crit_);
243 rtc::RateTracker encoded_frame_rate_tracker_ RTC_GUARDED_BY(crit_);
asaperssond89920b2015-07-22 06:52:00 -0700244
Danil Chapovalovb9b146c2018-06-15 12:28:07 +0200245 absl::optional<int64_t> last_outlier_timestamp_ RTC_GUARDED_BY(crit_);
Ilya Nikolaevskiy70473fc2018-02-28 16:35:03 +0100246
Erik Språnge2fd86a2018-10-24 11:32:39 +0200247 struct EncoderChangeEvent {
248 std::string previous_encoder_implementation;
249 std::string new_encoder_implementation;
250 };
251 // Stores the last change in encoder implementation in an optional, so that
252 // the event can be consumed.
253 absl::optional<EncoderChangeEvent> encoder_changed_;
254
sprangb4a1ae52015-12-03 08:10:08 -0800255 // Contains stats used for UMA histograms. These stats will be reset if
256 // content type changes between real-time video and screenshare, since these
257 // will be reported separately.
258 struct UmaSamplesContainer {
sprang07fb9be2016-02-24 07:55:00 -0800259 UmaSamplesContainer(const char* prefix,
260 const VideoSendStream::Stats& start_stats,
261 Clock* clock);
sprangb4a1ae52015-12-03 08:10:08 -0800262 ~UmaSamplesContainer();
263
Stefan Holmerdbdb3a02018-07-17 16:03:46 +0200264 void UpdateHistograms(const RtpConfig& rtp_config,
sprang07fb9be2016-02-24 07:55:00 -0800265 const VideoSendStream::Stats& current_stats);
sprangb4a1ae52015-12-03 08:10:08 -0800266
asapersson93e1e232017-02-06 05:18:35 -0800267 void InitializeBitrateCounters(const VideoSendStream::Stats& stats);
268
Åsa Perssonaa329e72017-12-15 15:54:44 +0100269 bool InsertEncodedFrame(const EncodedImage& encoded_frame,
Niels Möllerd3b8c632018-08-27 15:33:42 +0200270 int simulcast_idx,
Åsa Perssonaa329e72017-12-15 15:54:44 +0100271 bool* is_limited_in_resolution);
272 void RemoveOld(int64_t now_ms, bool* is_limited_in_resolution);
Åsa Persson0122e842017-10-16 12:19:23 +0200273
sprangb4a1ae52015-12-03 08:10:08 -0800274 const std::string uma_prefix_;
sprang07fb9be2016-02-24 07:55:00 -0800275 Clock* const clock_;
sprangb4a1ae52015-12-03 08:10:08 -0800276 SampleCounter input_width_counter_;
277 SampleCounter input_height_counter_;
278 SampleCounter sent_width_counter_;
279 SampleCounter sent_height_counter_;
280 SampleCounter encode_time_counter_;
281 BoolSampleCounter key_frame_counter_;
282 BoolSampleCounter quality_limited_frame_counter_;
283 SampleCounter quality_downscales_counter_;
perkj803d97f2016-11-01 11:45:46 -0700284 BoolSampleCounter cpu_limited_frame_counter_;
sprangb4a1ae52015-12-03 08:10:08 -0800285 BoolSampleCounter bw_limited_frame_counter_;
286 SampleCounter bw_resolutions_disabled_counter_;
287 SampleCounter delay_counter_;
288 SampleCounter max_delay_counter_;
289 rtc::RateTracker input_frame_rate_tracker_;
asapersson320e45a2016-11-29 01:40:35 -0800290 RateCounter input_fps_counter_;
291 RateCounter sent_fps_counter_;
asapersson93e1e232017-02-06 05:18:35 -0800292 RateAccCounter total_byte_counter_;
293 RateAccCounter media_byte_counter_;
294 RateAccCounter rtx_byte_counter_;
295 RateAccCounter padding_byte_counter_;
296 RateAccCounter retransmit_byte_counter_;
297 RateAccCounter fec_byte_counter_;
sprange2d83d62016-02-19 09:03:26 -0800298 int64_t first_rtcp_stats_time_ms_;
Erik Språng22c2b482016-03-01 09:40:42 +0100299 int64_t first_rtp_stats_time_ms_;
asapersson09f05612017-05-15 23:40:18 -0700300 StatsTimer cpu_adapt_timer_;
301 StatsTimer quality_adapt_timer_;
asapersson66d4b372016-12-19 06:50:53 -0800302 BoolSampleCounter paused_time_counter_;
303 TargetRateUpdates target_rate_updates_;
asapersson8d75ac72017-09-15 06:41:15 -0700304 BoolSampleCounter fallback_active_counter_;
305 FallbackEncoderInfo fallback_info_;
Åsa Perssonc3ed6302017-11-16 14:04:52 +0100306 FallbackEncoderInfoDisabled fallback_info_disabled_;
sprange2d83d62016-02-19 09:03:26 -0800307 ReportBlockStats report_block_stats_;
sprang07fb9be2016-02-24 07:55:00 -0800308 const VideoSendStream::Stats start_stats_;
Åsa Perssonaa329e72017-12-15 15:54:44 +0100309 size_t num_streams_; // Number of configured streams to encoder.
310 size_t num_pixels_highest_stream_;
Åsa Persson0122e842017-10-16 12:19:23 +0200311 EncodedFrameMap encoded_frames_;
Åsa Persson875841d2018-01-08 08:49:53 +0100312 AdaptChanges initial_quality_changes_;
sprangc5d62e22017-04-02 23:53:04 -0700313
asapersson118ef002016-03-31 00:00:19 -0700314 std::map<int, QpCounters>
315 qp_counters_; // QP counters mapped by spatial idx.
sprangb4a1ae52015-12-03 08:10:08 -0800316 };
317
danilchapa37de392017-09-09 04:17:22 -0700318 std::unique_ptr<UmaSamplesContainer> uma_container_ RTC_GUARDED_BY(crit_);
sprang@webrtc.orgccd42842014-01-07 09:54:34 +0000319};
320
321} // namespace webrtc
Mirko Bonadei92ea95e2017-09-15 06:47:31 +0200322#endif // VIDEO_SEND_STATISTICS_PROXY_H_