sprang@webrtc.org | ccd4284 | 2014-01-07 09:54:34 | [diff] [blame] | 1 | /* |
| 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 | #ifndef WEBRTC_VIDEO_SEND_STATISTICS_PROXY_H_ |
| 12 | #define WEBRTC_VIDEO_SEND_STATISTICS_PROXY_H_ |
| 13 | |
mflodman | d1590b2 | 2015-12-09 15:07:59 | [diff] [blame^] | 14 | #include <map> |
sprang@webrtc.org | ccd4284 | 2014-01-07 09:54:34 | [diff] [blame] | 15 | #include <string> |
| 16 | |
Peter Boström | f2f8283 | 2015-05-01 11:00:41 | [diff] [blame] | 17 | #include "webrtc/base/criticalsection.h" |
asapersson | 1aa420b | 2015-12-07 11:12:22 | [diff] [blame] | 18 | #include "webrtc/base/exp_filter.h" |
perkj@webrtc.org | af612d5 | 2015-03-18 09:51:05 | [diff] [blame] | 19 | #include "webrtc/base/ratetracker.h" |
kwiberg@webrtc.org | 00b8f6b | 2015-02-26 14:34:55 | [diff] [blame] | 20 | #include "webrtc/base/scoped_ptr.h" |
pbos@webrtc.org | 38344ed | 2014-09-24 06:05:00 | [diff] [blame] | 21 | #include "webrtc/base/thread_annotations.h" |
sprang@webrtc.org | ccd4284 | 2014-01-07 09:54:34 | [diff] [blame] | 22 | #include "webrtc/common_types.h" |
Henrik Kjellander | 2557b86 | 2015-11-18 21:00:21 | [diff] [blame] | 23 | #include "webrtc/modules/video_coding/include/video_codec_interface.h" |
| 24 | #include "webrtc/modules/video_coding/include/video_coding_defines.h" |
Henrik Kjellander | 98f5351 | 2015-10-28 17:17:40 | [diff] [blame] | 25 | #include "webrtc/system_wrappers/include/clock.h" |
Peter Boström | 7623ce4 | 2015-12-09 11:13:30 | [diff] [blame] | 26 | #include "webrtc/video/overuse_frame_detector.h" |
| 27 | #include "webrtc/video/vie_encoder.h" |
pbos@webrtc.org | 273a414 | 2014-12-01 15:23:21 | [diff] [blame] | 28 | #include "webrtc/video_send_stream.h" |
sprang@webrtc.org | ccd4284 | 2014-01-07 09:54:34 | [diff] [blame] | 29 | |
| 30 | namespace webrtc { |
| 31 | |
pbos@webrtc.org | 3e6e271 | 2015-02-26 12:19:31 | [diff] [blame] | 32 | class SendStatisticsProxy : public CpuOveruseMetricsObserver, |
| 33 | public RtcpStatisticsCallback, |
pbos@webrtc.org | 1d0fa5d | 2015-02-19 12:47:00 | [diff] [blame] | 34 | public RtcpPacketTypeCounterObserver, |
sprang@webrtc.org | ccd4284 | 2014-01-07 09:54:34 | [diff] [blame] | 35 | public StreamDataCountersCallback, |
| 36 | public BitrateStatisticsObserver, |
| 37 | public FrameCountObserver, |
pbos@webrtc.org | 891d483 | 2015-02-26 13:15:22 | [diff] [blame] | 38 | public VideoEncoderRateObserver, |
stefan@webrtc.org | 168f23f | 2014-07-11 13:44:02 | [diff] [blame] | 39 | public SendSideDelayObserver { |
sprang@webrtc.org | ccd4284 | 2014-01-07 09:54:34 | [diff] [blame] | 40 | public: |
pbos@webrtc.org | 273a414 | 2014-12-01 15:23:21 | [diff] [blame] | 41 | static const int kStatsTimeoutMs; |
| 42 | |
sprang | b4a1ae5 | 2015-12-03 16:10:08 | [diff] [blame] | 43 | SendStatisticsProxy(Clock* clock, |
| 44 | const VideoSendStream::Config& config, |
| 45 | VideoEncoderConfig::ContentType content_type); |
sprang@webrtc.org | ccd4284 | 2014-01-07 09:54:34 | [diff] [blame] | 46 | virtual ~SendStatisticsProxy(); |
| 47 | |
pbos@webrtc.org | 273a414 | 2014-12-01 15:23:21 | [diff] [blame] | 48 | VideoSendStream::Stats GetStats(); |
| 49 | |
| 50 | virtual void OnSendEncodedImage(const EncodedImage& encoded_image, |
| 51 | const RTPVideoHeader* rtp_video_header); |
perkj@webrtc.org | af612d5 | 2015-03-18 09:51:05 | [diff] [blame] | 52 | // Used to update incoming frame rate. |
asapersson | d89920b | 2015-07-22 13:52:00 | [diff] [blame] | 53 | void OnIncomingFrame(int width, int height); |
sprang@webrtc.org | ccd4284 | 2014-01-07 09:54:34 | [diff] [blame] | 54 | |
asapersson | 6718e97 | 2015-07-24 07:20:58 | [diff] [blame] | 55 | // Used to update encode time of frames. |
| 56 | void OnEncodedFrame(int encode_time_ms); |
| 57 | |
pbos@webrtc.org | 891d483 | 2015-02-26 13:15:22 | [diff] [blame] | 58 | // From VideoEncoderRateObserver. |
| 59 | void OnSetRates(uint32_t bitrate_bps, int framerate) override; |
| 60 | |
Peter Boström | 7083e11 | 2015-09-22 14:28:51 | [diff] [blame] | 61 | void OnOutgoingRate(uint32_t framerate, uint32_t bitrate); |
| 62 | void OnSuspendChange(bool is_suspended); |
Peter Boström | 20f3f94 | 2015-05-15 09:33:39 | [diff] [blame] | 63 | void OnInactiveSsrc(uint32_t ssrc); |
| 64 | |
sprang | b4a1ae5 | 2015-12-03 16:10:08 | [diff] [blame] | 65 | // Used to indicate change in content type, which may require a change in |
| 66 | // how stats are collected. |
| 67 | void SetContentType(VideoEncoderConfig::ContentType content_type); |
| 68 | |
sprang@webrtc.org | ccd4284 | 2014-01-07 09:54:34 | [diff] [blame] | 69 | protected: |
pbos@webrtc.org | 3e6e271 | 2015-02-26 12:19:31 | [diff] [blame] | 70 | // From CpuOveruseMetricsObserver. |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 | [diff] [blame] | 71 | void CpuOveruseMetricsUpdated(const CpuOveruseMetrics& metrics) override; |
sprang@webrtc.org | ccd4284 | 2014-01-07 09:54:34 | [diff] [blame] | 72 | // From RtcpStatisticsCallback. |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 | [diff] [blame] | 73 | void StatisticsUpdated(const RtcpStatistics& statistics, |
| 74 | uint32_t ssrc) override; |
| 75 | void CNameChanged(const char* cname, uint32_t ssrc) override; |
asapersson | d89920b | 2015-07-22 13:52:00 | [diff] [blame] | 76 | // From RtcpPacketTypeCounterObserver. |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 | [diff] [blame] | 77 | void RtcpPacketTypesCounterUpdated( |
pbos@webrtc.org | 1d0fa5d | 2015-02-19 12:47:00 | [diff] [blame] | 78 | uint32_t ssrc, |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 | [diff] [blame] | 79 | const RtcpPacketTypeCounter& packet_counter) override; |
sprang@webrtc.org | ccd4284 | 2014-01-07 09:54:34 | [diff] [blame] | 80 | // From StreamDataCountersCallback. |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 | [diff] [blame] | 81 | void DataCountersUpdated(const StreamDataCounters& counters, |
| 82 | uint32_t ssrc) override; |
sprang@webrtc.org | ccd4284 | 2014-01-07 09:54:34 | [diff] [blame] | 83 | |
| 84 | // From BitrateStatisticsObserver. |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 | [diff] [blame] | 85 | void Notify(const BitrateStatistics& total_stats, |
| 86 | const BitrateStatistics& retransmit_stats, |
| 87 | uint32_t ssrc) override; |
sprang@webrtc.org | ccd4284 | 2014-01-07 09:54:34 | [diff] [blame] | 88 | |
| 89 | // From FrameCountObserver. |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 | [diff] [blame] | 90 | void FrameCountUpdated(const FrameCounts& frame_counts, |
| 91 | uint32_t ssrc) override; |
sprang@webrtc.org | ccd4284 | 2014-01-07 09:54:34 | [diff] [blame] | 92 | |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 | [diff] [blame] | 93 | void SendSideDelayUpdated(int avg_delay_ms, |
| 94 | int max_delay_ms, |
| 95 | uint32_t ssrc) override; |
stefan@webrtc.org | 168f23f | 2014-07-11 13:44:02 | [diff] [blame] | 96 | |
sprang@webrtc.org | ccd4284 | 2014-01-07 09:54:34 | [diff] [blame] | 97 | private: |
asapersson | da535c4 | 2015-10-20 06:32:41 | [diff] [blame] | 98 | class SampleCounter { |
| 99 | public: |
asapersson | d89920b | 2015-07-22 13:52:00 | [diff] [blame] | 100 | SampleCounter() : sum(0), num_samples(0) {} |
asapersson | da535c4 | 2015-10-20 06:32:41 | [diff] [blame] | 101 | ~SampleCounter() {} |
asapersson | d89920b | 2015-07-22 13:52:00 | [diff] [blame] | 102 | void Add(int sample); |
| 103 | int Avg(int min_required_samples) const; |
| 104 | |
| 105 | private: |
| 106 | int sum; |
| 107 | int num_samples; |
| 108 | }; |
asapersson | da535c4 | 2015-10-20 06:32:41 | [diff] [blame] | 109 | class BoolSampleCounter { |
| 110 | public: |
asapersson | dec5ebf | 2015-10-05 09:36:17 | [diff] [blame] | 111 | BoolSampleCounter() : sum(0), num_samples(0) {} |
asapersson | da535c4 | 2015-10-20 06:32:41 | [diff] [blame] | 112 | ~BoolSampleCounter() {} |
asapersson | dec5ebf | 2015-10-05 09:36:17 | [diff] [blame] | 113 | void Add(bool sample); |
| 114 | int Percent(int min_required_samples) const; |
| 115 | int Permille(int min_required_samples) const; |
| 116 | |
| 117 | private: |
| 118 | int Fraction(int min_required_samples, float multiplier) const; |
| 119 | int sum; |
| 120 | int num_samples; |
| 121 | }; |
pbos@webrtc.org | 273a414 | 2014-12-01 15:23:21 | [diff] [blame] | 122 | struct StatsUpdateTimes { |
sprang | b4a1ae5 | 2015-12-03 16:10:08 | [diff] [blame] | 123 | StatsUpdateTimes() : resolution_update_ms(0), bitrate_update_ms(0) {} |
pbos@webrtc.org | 273a414 | 2014-12-01 15:23:21 | [diff] [blame] | 124 | int64_t resolution_update_ms; |
Peter Boström | 20f3f94 | 2015-05-15 09:33:39 | [diff] [blame] | 125 | int64_t bitrate_update_ms; |
pbos@webrtc.org | 273a414 | 2014-12-01 15:23:21 | [diff] [blame] | 126 | }; |
| 127 | void PurgeOldStats() EXCLUSIVE_LOCKS_REQUIRED(crit_); |
pbos@webrtc.org | 09c77b9 | 2015-02-25 10:42:16 | [diff] [blame] | 128 | VideoSendStream::StreamStats* GetStatsEntry(uint32_t ssrc) |
| 129 | EXCLUSIVE_LOCKS_REQUIRED(crit_); |
sprang@webrtc.org | ccd4284 | 2014-01-07 09:54:34 | [diff] [blame] | 130 | |
pbos@webrtc.org | 273a414 | 2014-12-01 15:23:21 | [diff] [blame] | 131 | Clock* const clock_; |
sprang@webrtc.org | ccd4284 | 2014-01-07 09:54:34 | [diff] [blame] | 132 | const VideoSendStream::Config config_; |
Peter Boström | f2f8283 | 2015-05-01 11:00:41 | [diff] [blame] | 133 | mutable rtc::CriticalSection crit_; |
sprang | b4a1ae5 | 2015-12-03 16:10:08 | [diff] [blame] | 134 | VideoEncoderConfig::ContentType content_type_ GUARDED_BY(crit_); |
pbos@webrtc.org | de1429e | 2014-04-28 13:00:21 | [diff] [blame] | 135 | VideoSendStream::Stats stats_ GUARDED_BY(crit_); |
Åsa Persson | 24b4eda | 2015-06-16 08:17:01 | [diff] [blame] | 136 | uint32_t last_sent_frame_timestamp_ GUARDED_BY(crit_); |
pbos@webrtc.org | 273a414 | 2014-12-01 15:23:21 | [diff] [blame] | 137 | std::map<uint32_t, StatsUpdateTimes> update_times_ GUARDED_BY(crit_); |
asapersson | 1aa420b | 2015-12-07 11:12:22 | [diff] [blame] | 138 | rtc::ExpFilter encode_time_ GUARDED_BY(crit_); |
asapersson | d89920b | 2015-07-22 13:52:00 | [diff] [blame] | 139 | |
sprang | b4a1ae5 | 2015-12-03 16:10:08 | [diff] [blame] | 140 | // Contains stats used for UMA histograms. These stats will be reset if |
| 141 | // content type changes between real-time video and screenshare, since these |
| 142 | // will be reported separately. |
| 143 | struct UmaSamplesContainer { |
| 144 | explicit UmaSamplesContainer(const char* prefix); |
| 145 | ~UmaSamplesContainer(); |
| 146 | |
| 147 | void UpdateHistograms(); |
| 148 | |
| 149 | const std::string uma_prefix_; |
| 150 | int max_sent_width_per_timestamp_; |
| 151 | int max_sent_height_per_timestamp_; |
| 152 | SampleCounter input_width_counter_; |
| 153 | SampleCounter input_height_counter_; |
| 154 | SampleCounter sent_width_counter_; |
| 155 | SampleCounter sent_height_counter_; |
| 156 | SampleCounter encode_time_counter_; |
| 157 | BoolSampleCounter key_frame_counter_; |
| 158 | BoolSampleCounter quality_limited_frame_counter_; |
| 159 | SampleCounter quality_downscales_counter_; |
| 160 | BoolSampleCounter bw_limited_frame_counter_; |
| 161 | SampleCounter bw_resolutions_disabled_counter_; |
| 162 | SampleCounter delay_counter_; |
| 163 | SampleCounter max_delay_counter_; |
| 164 | rtc::RateTracker input_frame_rate_tracker_; |
| 165 | rtc::RateTracker sent_frame_rate_tracker_; |
| 166 | }; |
| 167 | |
| 168 | rtc::scoped_ptr<UmaSamplesContainer> uma_container_ GUARDED_BY(crit_); |
sprang@webrtc.org | ccd4284 | 2014-01-07 09:54:34 | [diff] [blame] | 169 | }; |
| 170 | |
| 171 | } // namespace webrtc |
| 172 | #endif // WEBRTC_VIDEO_SEND_STATISTICS_PROXY_H_ |