xref: /aosp_15_r20/external/webrtc/test/scenario/stats_collection.cc (revision d9f758449e529ab9291ac668be2861e7a55c2422)
1*d9f75844SAndroid Build Coastguard Worker /*
2*d9f75844SAndroid Build Coastguard Worker  *  Copyright 2019 The WebRTC project authors. All Rights Reserved.
3*d9f75844SAndroid Build Coastguard Worker  *
4*d9f75844SAndroid Build Coastguard Worker  *  Use of this source code is governed by a BSD-style license
5*d9f75844SAndroid Build Coastguard Worker  *  that can be found in the LICENSE file in the root of the source
6*d9f75844SAndroid Build Coastguard Worker  *  tree. An additional intellectual property rights grant can be found
7*d9f75844SAndroid Build Coastguard Worker  *  in the file PATENTS.  All contributing project authors may
8*d9f75844SAndroid Build Coastguard Worker  *  be found in the AUTHORS file in the root of the source tree.
9*d9f75844SAndroid Build Coastguard Worker  */
10*d9f75844SAndroid Build Coastguard Worker 
11*d9f75844SAndroid Build Coastguard Worker #include "test/scenario/stats_collection.h"
12*d9f75844SAndroid Build Coastguard Worker 
13*d9f75844SAndroid Build Coastguard Worker #include "common_video/libyuv/include/webrtc_libyuv.h"
14*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/memory_usage.h"
15*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/thread.h"
16*d9f75844SAndroid Build Coastguard Worker 
17*d9f75844SAndroid Build Coastguard Worker namespace webrtc {
18*d9f75844SAndroid Build Coastguard Worker namespace test {
19*d9f75844SAndroid Build Coastguard Worker 
VideoQualityAnalyzer(VideoQualityAnalyzerConfig config,std::unique_ptr<RtcEventLogOutput> writer)20*d9f75844SAndroid Build Coastguard Worker VideoQualityAnalyzer::VideoQualityAnalyzer(
21*d9f75844SAndroid Build Coastguard Worker     VideoQualityAnalyzerConfig config,
22*d9f75844SAndroid Build Coastguard Worker     std::unique_ptr<RtcEventLogOutput> writer)
23*d9f75844SAndroid Build Coastguard Worker     : config_(config), writer_(std::move(writer)) {
24*d9f75844SAndroid Build Coastguard Worker   if (writer_) {
25*d9f75844SAndroid Build Coastguard Worker     PrintHeaders();
26*d9f75844SAndroid Build Coastguard Worker   }
27*d9f75844SAndroid Build Coastguard Worker }
28*d9f75844SAndroid Build Coastguard Worker 
29*d9f75844SAndroid Build Coastguard Worker VideoQualityAnalyzer::~VideoQualityAnalyzer() = default;
30*d9f75844SAndroid Build Coastguard Worker 
PrintHeaders()31*d9f75844SAndroid Build Coastguard Worker void VideoQualityAnalyzer::PrintHeaders() {
32*d9f75844SAndroid Build Coastguard Worker   writer_->Write(
33*d9f75844SAndroid Build Coastguard Worker       "capture_time render_time capture_width capture_height render_width "
34*d9f75844SAndroid Build Coastguard Worker       "render_height psnr\n");
35*d9f75844SAndroid Build Coastguard Worker }
36*d9f75844SAndroid Build Coastguard Worker 
Handler()37*d9f75844SAndroid Build Coastguard Worker std::function<void(const VideoFramePair&)> VideoQualityAnalyzer::Handler() {
38*d9f75844SAndroid Build Coastguard Worker   return [this](VideoFramePair pair) { HandleFramePair(pair); };
39*d9f75844SAndroid Build Coastguard Worker }
40*d9f75844SAndroid Build Coastguard Worker 
HandleFramePair(VideoFramePair sample,double psnr)41*d9f75844SAndroid Build Coastguard Worker void VideoQualityAnalyzer::HandleFramePair(VideoFramePair sample, double psnr) {
42*d9f75844SAndroid Build Coastguard Worker   layer_analyzers_[sample.layer_id].HandleFramePair(sample, psnr,
43*d9f75844SAndroid Build Coastguard Worker                                                     writer_.get());
44*d9f75844SAndroid Build Coastguard Worker   cached_.reset();
45*d9f75844SAndroid Build Coastguard Worker }
46*d9f75844SAndroid Build Coastguard Worker 
HandleFramePair(VideoFramePair sample)47*d9f75844SAndroid Build Coastguard Worker void VideoQualityAnalyzer::HandleFramePair(VideoFramePair sample) {
48*d9f75844SAndroid Build Coastguard Worker   double psnr = NAN;
49*d9f75844SAndroid Build Coastguard Worker   if (sample.decoded)
50*d9f75844SAndroid Build Coastguard Worker     psnr = I420PSNR(*sample.captured->ToI420(), *sample.decoded->ToI420());
51*d9f75844SAndroid Build Coastguard Worker 
52*d9f75844SAndroid Build Coastguard Worker   if (config_.thread) {
53*d9f75844SAndroid Build Coastguard Worker     config_.thread->PostTask(
54*d9f75844SAndroid Build Coastguard Worker         [this, sample, psnr] { HandleFramePair(std::move(sample), psnr); });
55*d9f75844SAndroid Build Coastguard Worker   } else {
56*d9f75844SAndroid Build Coastguard Worker     HandleFramePair(std::move(sample), psnr);
57*d9f75844SAndroid Build Coastguard Worker   }
58*d9f75844SAndroid Build Coastguard Worker }
59*d9f75844SAndroid Build Coastguard Worker 
layer_stats() const60*d9f75844SAndroid Build Coastguard Worker std::vector<VideoQualityStats> VideoQualityAnalyzer::layer_stats() const {
61*d9f75844SAndroid Build Coastguard Worker   std::vector<VideoQualityStats> res;
62*d9f75844SAndroid Build Coastguard Worker   for (auto& layer : layer_analyzers_)
63*d9f75844SAndroid Build Coastguard Worker     res.push_back(layer.second.stats_);
64*d9f75844SAndroid Build Coastguard Worker   return res;
65*d9f75844SAndroid Build Coastguard Worker }
66*d9f75844SAndroid Build Coastguard Worker 
stats()67*d9f75844SAndroid Build Coastguard Worker VideoQualityStats& VideoQualityAnalyzer::stats() {
68*d9f75844SAndroid Build Coastguard Worker   if (!cached_) {
69*d9f75844SAndroid Build Coastguard Worker     cached_ = VideoQualityStats();
70*d9f75844SAndroid Build Coastguard Worker     for (auto& layer : layer_analyzers_)
71*d9f75844SAndroid Build Coastguard Worker       cached_->AddStats(layer.second.stats_);
72*d9f75844SAndroid Build Coastguard Worker   }
73*d9f75844SAndroid Build Coastguard Worker   return *cached_;
74*d9f75844SAndroid Build Coastguard Worker }
75*d9f75844SAndroid Build Coastguard Worker 
HandleFramePair(VideoFramePair sample,double psnr,RtcEventLogOutput * writer)76*d9f75844SAndroid Build Coastguard Worker void VideoLayerAnalyzer::HandleFramePair(VideoFramePair sample,
77*d9f75844SAndroid Build Coastguard Worker                                          double psnr,
78*d9f75844SAndroid Build Coastguard Worker                                          RtcEventLogOutput* writer) {
79*d9f75844SAndroid Build Coastguard Worker   RTC_CHECK(sample.captured);
80*d9f75844SAndroid Build Coastguard Worker   HandleCapturedFrame(sample);
81*d9f75844SAndroid Build Coastguard Worker   if (!sample.decoded) {
82*d9f75844SAndroid Build Coastguard Worker     // Can only happen in the beginning of a call or if the resolution is
83*d9f75844SAndroid Build Coastguard Worker     // reduced. Otherwise we will detect a freeze.
84*d9f75844SAndroid Build Coastguard Worker     ++stats_.lost_count;
85*d9f75844SAndroid Build Coastguard Worker     ++skip_count_;
86*d9f75844SAndroid Build Coastguard Worker   } else {
87*d9f75844SAndroid Build Coastguard Worker     stats_.psnr_with_freeze.AddSample(psnr);
88*d9f75844SAndroid Build Coastguard Worker     if (sample.repeated) {
89*d9f75844SAndroid Build Coastguard Worker       ++stats_.freeze_count;
90*d9f75844SAndroid Build Coastguard Worker       ++skip_count_;
91*d9f75844SAndroid Build Coastguard Worker     } else {
92*d9f75844SAndroid Build Coastguard Worker       stats_.psnr.AddSample(psnr);
93*d9f75844SAndroid Build Coastguard Worker       HandleRenderedFrame(sample);
94*d9f75844SAndroid Build Coastguard Worker     }
95*d9f75844SAndroid Build Coastguard Worker   }
96*d9f75844SAndroid Build Coastguard Worker   if (writer) {
97*d9f75844SAndroid Build Coastguard Worker     LogWriteFormat(writer, "%.3f %.3f %.3f %i %i %i %i %.3f\n",
98*d9f75844SAndroid Build Coastguard Worker                    sample.capture_time.seconds<double>(),
99*d9f75844SAndroid Build Coastguard Worker                    sample.render_time.seconds<double>(),
100*d9f75844SAndroid Build Coastguard Worker                    sample.captured->width(), sample.captured->height(),
101*d9f75844SAndroid Build Coastguard Worker                    sample.decoded ? sample.decoded->width() : 0,
102*d9f75844SAndroid Build Coastguard Worker                    sample.decoded ? sample.decoded->height() : 0, psnr);
103*d9f75844SAndroid Build Coastguard Worker   }
104*d9f75844SAndroid Build Coastguard Worker }
105*d9f75844SAndroid Build Coastguard Worker 
HandleCapturedFrame(const VideoFramePair & sample)106*d9f75844SAndroid Build Coastguard Worker void VideoLayerAnalyzer::HandleCapturedFrame(const VideoFramePair& sample) {
107*d9f75844SAndroid Build Coastguard Worker   stats_.capture.AddFrameInfo(*sample.captured, sample.capture_time);
108*d9f75844SAndroid Build Coastguard Worker   if (last_freeze_time_.IsInfinite())
109*d9f75844SAndroid Build Coastguard Worker     last_freeze_time_ = sample.capture_time;
110*d9f75844SAndroid Build Coastguard Worker }
111*d9f75844SAndroid Build Coastguard Worker 
HandleRenderedFrame(const VideoFramePair & sample)112*d9f75844SAndroid Build Coastguard Worker void VideoLayerAnalyzer::HandleRenderedFrame(const VideoFramePair& sample) {
113*d9f75844SAndroid Build Coastguard Worker   stats_.capture_to_decoded_delay.AddSample(sample.decoded_time -
114*d9f75844SAndroid Build Coastguard Worker                                             sample.capture_time);
115*d9f75844SAndroid Build Coastguard Worker   stats_.end_to_end_delay.AddSample(sample.render_time - sample.capture_time);
116*d9f75844SAndroid Build Coastguard Worker   stats_.render.AddFrameInfo(*sample.decoded, sample.render_time);
117*d9f75844SAndroid Build Coastguard Worker   stats_.skipped_between_rendered.AddSample(skip_count_);
118*d9f75844SAndroid Build Coastguard Worker   skip_count_ = 0;
119*d9f75844SAndroid Build Coastguard Worker 
120*d9f75844SAndroid Build Coastguard Worker   if (last_render_time_.IsFinite()) {
121*d9f75844SAndroid Build Coastguard Worker     RTC_DCHECK(sample.render_time.IsFinite());
122*d9f75844SAndroid Build Coastguard Worker     TimeDelta render_interval = sample.render_time - last_render_time_;
123*d9f75844SAndroid Build Coastguard Worker     TimeDelta mean_interval = stats_.render.frames.interval().Mean();
124*d9f75844SAndroid Build Coastguard Worker     if (render_interval > TimeDelta::Millis(150) + mean_interval ||
125*d9f75844SAndroid Build Coastguard Worker         render_interval > 3 * mean_interval) {
126*d9f75844SAndroid Build Coastguard Worker       stats_.freeze_duration.AddSample(render_interval);
127*d9f75844SAndroid Build Coastguard Worker       stats_.time_between_freezes.AddSample(last_render_time_ -
128*d9f75844SAndroid Build Coastguard Worker                                             last_freeze_time_);
129*d9f75844SAndroid Build Coastguard Worker       last_freeze_time_ = sample.render_time;
130*d9f75844SAndroid Build Coastguard Worker     }
131*d9f75844SAndroid Build Coastguard Worker   }
132*d9f75844SAndroid Build Coastguard Worker   last_render_time_ = sample.render_time;
133*d9f75844SAndroid Build Coastguard Worker }
134*d9f75844SAndroid Build Coastguard Worker 
AddStats(Call::Stats sample)135*d9f75844SAndroid Build Coastguard Worker void CallStatsCollector::AddStats(Call::Stats sample) {
136*d9f75844SAndroid Build Coastguard Worker   if (sample.send_bandwidth_bps > 0)
137*d9f75844SAndroid Build Coastguard Worker     stats_.target_rate.AddSampleBps(sample.send_bandwidth_bps);
138*d9f75844SAndroid Build Coastguard Worker   if (sample.pacer_delay_ms > 0)
139*d9f75844SAndroid Build Coastguard Worker     stats_.pacer_delay.AddSample(TimeDelta::Millis(sample.pacer_delay_ms));
140*d9f75844SAndroid Build Coastguard Worker   if (sample.rtt_ms > 0)
141*d9f75844SAndroid Build Coastguard Worker     stats_.round_trip_time.AddSample(TimeDelta::Millis(sample.rtt_ms));
142*d9f75844SAndroid Build Coastguard Worker   stats_.memory_usage.AddSample(rtc::GetProcessResidentSizeBytes());
143*d9f75844SAndroid Build Coastguard Worker }
144*d9f75844SAndroid Build Coastguard Worker 
AddStats(AudioReceiveStreamInterface::Stats sample)145*d9f75844SAndroid Build Coastguard Worker void AudioReceiveStatsCollector::AddStats(
146*d9f75844SAndroid Build Coastguard Worker     AudioReceiveStreamInterface::Stats sample) {
147*d9f75844SAndroid Build Coastguard Worker   stats_.expand_rate.AddSample(sample.expand_rate);
148*d9f75844SAndroid Build Coastguard Worker   stats_.accelerate_rate.AddSample(sample.accelerate_rate);
149*d9f75844SAndroid Build Coastguard Worker   stats_.jitter_buffer.AddSampleMs(sample.jitter_buffer_ms);
150*d9f75844SAndroid Build Coastguard Worker }
151*d9f75844SAndroid Build Coastguard Worker 
AddStats(VideoSendStream::Stats sample,Timestamp at_time)152*d9f75844SAndroid Build Coastguard Worker void VideoSendStatsCollector::AddStats(VideoSendStream::Stats sample,
153*d9f75844SAndroid Build Coastguard Worker                                        Timestamp at_time) {
154*d9f75844SAndroid Build Coastguard Worker   // It's not certain that we yet have estimates for any of these stats.
155*d9f75844SAndroid Build Coastguard Worker   // Check that they are positive before mixing them in.
156*d9f75844SAndroid Build Coastguard Worker   if (sample.encode_frame_rate <= 0)
157*d9f75844SAndroid Build Coastguard Worker     return;
158*d9f75844SAndroid Build Coastguard Worker 
159*d9f75844SAndroid Build Coastguard Worker   stats_.encode_frame_rate.AddSample(sample.encode_frame_rate);
160*d9f75844SAndroid Build Coastguard Worker   stats_.encode_time.AddSampleMs(sample.avg_encode_time_ms);
161*d9f75844SAndroid Build Coastguard Worker   stats_.encode_usage.AddSample(sample.encode_usage_percent / 100.0);
162*d9f75844SAndroid Build Coastguard Worker   stats_.media_bitrate.AddSampleBps(sample.media_bitrate_bps);
163*d9f75844SAndroid Build Coastguard Worker 
164*d9f75844SAndroid Build Coastguard Worker   size_t fec_bytes = 0;
165*d9f75844SAndroid Build Coastguard Worker   for (const auto& kv : sample.substreams) {
166*d9f75844SAndroid Build Coastguard Worker     fec_bytes += kv.second.rtp_stats.fec.payload_bytes +
167*d9f75844SAndroid Build Coastguard Worker                  kv.second.rtp_stats.fec.padding_bytes;
168*d9f75844SAndroid Build Coastguard Worker   }
169*d9f75844SAndroid Build Coastguard Worker   if (last_update_.IsFinite()) {
170*d9f75844SAndroid Build Coastguard Worker     auto fec_delta = DataSize::Bytes(fec_bytes - last_fec_bytes_);
171*d9f75844SAndroid Build Coastguard Worker     auto time_delta = at_time - last_update_;
172*d9f75844SAndroid Build Coastguard Worker     stats_.fec_bitrate.AddSample(fec_delta / time_delta);
173*d9f75844SAndroid Build Coastguard Worker   }
174*d9f75844SAndroid Build Coastguard Worker   last_fec_bytes_ = fec_bytes;
175*d9f75844SAndroid Build Coastguard Worker   last_update_ = at_time;
176*d9f75844SAndroid Build Coastguard Worker }
177*d9f75844SAndroid Build Coastguard Worker 
AddStats(VideoReceiveStreamInterface::Stats sample)178*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStatsCollector::AddStats(
179*d9f75844SAndroid Build Coastguard Worker     VideoReceiveStreamInterface::Stats sample) {
180*d9f75844SAndroid Build Coastguard Worker   if (sample.decode_ms > 0)
181*d9f75844SAndroid Build Coastguard Worker     stats_.decode_time.AddSampleMs(sample.decode_ms);
182*d9f75844SAndroid Build Coastguard Worker   if (sample.max_decode_ms > 0)
183*d9f75844SAndroid Build Coastguard Worker     stats_.decode_time_max.AddSampleMs(sample.max_decode_ms);
184*d9f75844SAndroid Build Coastguard Worker   if (sample.width > 0 && sample.height > 0) {
185*d9f75844SAndroid Build Coastguard Worker     stats_.decode_pixels.AddSample(sample.width * sample.height);
186*d9f75844SAndroid Build Coastguard Worker     stats_.resolution.AddSample(sample.height);
187*d9f75844SAndroid Build Coastguard Worker   }
188*d9f75844SAndroid Build Coastguard Worker }
189*d9f75844SAndroid Build Coastguard Worker }  // namespace test
190*d9f75844SAndroid Build Coastguard Worker }  // namespace webrtc
191