mirror of
https://github.com/mollyim/webrtc.git
synced 2025-05-12 21:30:45 +01:00

Bug: webrtc:11566 Change-Id: I9013298ad31861b356b377013bb3172d1a39a1e8 Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/325262 Reviewed-by: Mirko Bonadei <mbonadei@webrtc.org> Commit-Queue: Björn Terelius <terelius@webrtc.org> Cr-Commit-Position: refs/heads/main@{#41035}
199 lines
7.2 KiB
C++
199 lines
7.2 KiB
C++
/*
|
|
* Copyright (c) 2020 The WebRTC project authors. All Rights Reserved.
|
|
*
|
|
* Use of this source code is governed by a BSD-style license
|
|
* that can be found in the LICENSE file in the root of the source
|
|
* tree. An additional intellectual property rights grant can be found
|
|
* in the file PATENTS. All contributing project authors may
|
|
* be found in the AUTHORS file in the root of the source tree.
|
|
*/
|
|
|
|
#ifndef RTC_TOOLS_RTC_EVENT_LOG_VISUALIZER_ANALYZER_COMMON_H_
|
|
#define RTC_TOOLS_RTC_EVENT_LOG_VISUALIZER_ANALYZER_COMMON_H_
|
|
|
|
#include <cstddef>
|
|
#include <cstdint>
|
|
#include <string>
|
|
|
|
#include "absl/types/optional.h"
|
|
#include "api/function_view.h"
|
|
#include "api/units/time_delta.h"
|
|
#include "api/units/timestamp.h"
|
|
#include "logging/rtc_event_log/rtc_event_log_parser.h"
|
|
#include "rtc_tools/rtc_event_log_visualizer/plot_base.h"
|
|
|
|
namespace webrtc {
|
|
|
|
constexpr int kNumMicrosecsPerSec = 1000000;
|
|
constexpr int kNumMillisecsPerSec = 1000;
|
|
constexpr float kLeftMargin = 0.01f;
|
|
constexpr float kRightMargin = 0.02f;
|
|
constexpr float kBottomMargin = 0.02f;
|
|
constexpr float kTopMargin = 0.05f;
|
|
|
|
class AnalyzerConfig {
|
|
public:
|
|
float GetCallTimeSec(Timestamp timestamp) const {
|
|
Timestamp offset = normalize_time_ ? begin_time_ : Timestamp::Zero();
|
|
return static_cast<float>((timestamp - offset).us()) / 1000000;
|
|
}
|
|
|
|
float GetCallTimeSecFromMs(int64_t timestamp_ms) const {
|
|
return GetCallTimeSec(Timestamp::Millis(timestamp_ms));
|
|
}
|
|
|
|
float CallBeginTimeSec() const { return GetCallTimeSec(begin_time_); }
|
|
|
|
float CallEndTimeSec() const { return GetCallTimeSec(end_time_); }
|
|
|
|
int64_t CallTimeToUtcOffsetMs() {
|
|
if (normalize_time_) {
|
|
Timestamp utc_begin_time_ = begin_time_ + rtc_to_utc_offset_;
|
|
return utc_begin_time_.ms();
|
|
}
|
|
return rtc_to_utc_offset_.ms();
|
|
}
|
|
|
|
// Window and step size used for calculating moving averages, e.g. bitrate.
|
|
// The generated data points will be `step_.ms()` milliseconds apart.
|
|
// Only events occurring at most `window_duration_.ms()` milliseconds before
|
|
// the current data point will be part of the average.
|
|
TimeDelta window_duration_ = TimeDelta::Millis(250);
|
|
TimeDelta step_ = TimeDelta::Millis(10);
|
|
|
|
// First and last events of the log.
|
|
Timestamp begin_time_ = Timestamp::MinusInfinity();
|
|
Timestamp end_time_ = Timestamp::MinusInfinity();
|
|
TimeDelta rtc_to_utc_offset_ = TimeDelta::Zero();
|
|
bool normalize_time_;
|
|
};
|
|
|
|
struct LayerDescription {
|
|
LayerDescription(uint32_t ssrc, uint8_t spatial_layer, uint8_t temporal_layer)
|
|
: ssrc(ssrc),
|
|
spatial_layer(spatial_layer),
|
|
temporal_layer(temporal_layer) {}
|
|
bool operator<(const LayerDescription& other) const {
|
|
if (ssrc != other.ssrc)
|
|
return ssrc < other.ssrc;
|
|
if (spatial_layer != other.spatial_layer)
|
|
return spatial_layer < other.spatial_layer;
|
|
return temporal_layer < other.temporal_layer;
|
|
}
|
|
uint32_t ssrc;
|
|
uint8_t spatial_layer;
|
|
uint8_t temporal_layer;
|
|
};
|
|
|
|
bool IsRtxSsrc(const ParsedRtcEventLog& parsed_log,
|
|
PacketDirection direction,
|
|
uint32_t ssrc);
|
|
bool IsVideoSsrc(const ParsedRtcEventLog& parsed_log,
|
|
PacketDirection direction,
|
|
uint32_t ssrc);
|
|
bool IsAudioSsrc(const ParsedRtcEventLog& parsed_log,
|
|
PacketDirection direction,
|
|
uint32_t ssrc);
|
|
|
|
std::string GetStreamName(const ParsedRtcEventLog& parsed_log,
|
|
PacketDirection direction,
|
|
uint32_t ssrc);
|
|
std::string GetLayerName(LayerDescription layer);
|
|
|
|
// For each element in data_view, use `f()` to extract a y-coordinate and
|
|
// store the result in a TimeSeries.
|
|
template <typename DataType, typename IterableType>
|
|
void ProcessPoints(rtc::FunctionView<float(const DataType&)> fx,
|
|
rtc::FunctionView<absl::optional<float>(const DataType&)> fy,
|
|
const IterableType& data_view,
|
|
TimeSeries* result) {
|
|
for (size_t i = 0; i < data_view.size(); i++) {
|
|
const DataType& elem = data_view[i];
|
|
float x = fx(elem);
|
|
absl::optional<float> y = fy(elem);
|
|
if (y)
|
|
result->points.emplace_back(x, *y);
|
|
}
|
|
}
|
|
|
|
// For each pair of adjacent elements in `data`, use `f()` to extract a
|
|
// y-coordinate and store the result in a TimeSeries. Note that the x-coordinate
|
|
// will be the time of the second element in the pair.
|
|
template <typename DataType, typename ResultType, typename IterableType>
|
|
void ProcessPairs(
|
|
rtc::FunctionView<float(const DataType&)> fx,
|
|
rtc::FunctionView<absl::optional<ResultType>(const DataType&,
|
|
const DataType&)> fy,
|
|
const IterableType& data,
|
|
TimeSeries* result) {
|
|
for (size_t i = 1; i < data.size(); i++) {
|
|
float x = fx(data[i]);
|
|
absl::optional<ResultType> y = fy(data[i - 1], data[i]);
|
|
if (y)
|
|
result->points.emplace_back(x, static_cast<float>(*y));
|
|
}
|
|
}
|
|
|
|
// For each pair of adjacent elements in `data`, use `f()` to extract a
|
|
// y-coordinate and store the result in a TimeSeries. Note that the x-coordinate
|
|
// will be the time of the second element in the pair.
|
|
template <typename DataType, typename ResultType, typename IterableType>
|
|
void AccumulatePairs(
|
|
rtc::FunctionView<float(const DataType&)> fx,
|
|
rtc::FunctionView<absl::optional<ResultType>(const DataType&,
|
|
const DataType&)> fy,
|
|
const IterableType& data,
|
|
TimeSeries* result) {
|
|
ResultType sum = 0;
|
|
for (size_t i = 1; i < data.size(); i++) {
|
|
float x = fx(data[i]);
|
|
absl::optional<ResultType> y = fy(data[i - 1], data[i]);
|
|
if (y) {
|
|
sum += *y;
|
|
result->points.emplace_back(x, static_cast<float>(sum));
|
|
}
|
|
}
|
|
}
|
|
|
|
// Calculates a moving average of `data` and stores the result in a TimeSeries.
|
|
// A data point is generated every `step` microseconds from `begin_time`
|
|
// to `end_time`. The value of each data point is the average of the data
|
|
// during the preceding `window_duration_us` microseconds.
|
|
template <typename DataType, typename ResultType, typename IterableType>
|
|
void MovingAverage(
|
|
rtc::FunctionView<absl::optional<ResultType>(const DataType&)> fy,
|
|
const IterableType& data_view,
|
|
AnalyzerConfig config,
|
|
TimeSeries* result) {
|
|
size_t window_index_begin = 0;
|
|
size_t window_index_end = 0;
|
|
ResultType sum_in_window = 0;
|
|
|
|
for (Timestamp t = config.begin_time_; t < config.end_time_ + config.step_;
|
|
t += config.step_) {
|
|
while (window_index_end < data_view.size() &&
|
|
data_view[window_index_end].log_time() < t) {
|
|
absl::optional<ResultType> value = fy(data_view[window_index_end]);
|
|
if (value)
|
|
sum_in_window += *value;
|
|
++window_index_end;
|
|
}
|
|
while (window_index_begin < data_view.size() &&
|
|
data_view[window_index_begin].log_time() <
|
|
t - config.window_duration_) {
|
|
absl::optional<ResultType> value = fy(data_view[window_index_begin]);
|
|
if (value)
|
|
sum_in_window -= *value;
|
|
++window_index_begin;
|
|
}
|
|
float window_duration_s =
|
|
static_cast<float>(config.window_duration_.us()) / kNumMicrosecsPerSec;
|
|
float x = config.GetCallTimeSec(t);
|
|
float y = sum_in_window / window_duration_s;
|
|
result->points.emplace_back(x, y);
|
|
}
|
|
}
|
|
|
|
} // namespace webrtc
|
|
|
|
#endif // RTC_TOOLS_RTC_EVENT_LOG_VISUALIZER_ANALYZER_COMMON_H_
|