blob: 490294dc7b34e8e67f83ba0a34a14c92c29e8dae [file] [log] [blame]
// Copyright 2014 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "media/cast/logging/stats_event_subscriber.h"
#include <algorithm>
#include <cmath>
#include <memory>
#include <utility>
#include "base/check_op.h"
#include "base/format_macros.h"
#include "base/notreached.h"
#include "base/strings/stringprintf.h"
#include "base/values.h"
#define STAT_ENUM_TO_STRING(enum) \
case enum: \
return #enum
namespace media {
namespace cast {
namespace {
using media::cast::CastLoggingEvent;
using media::cast::EventMediaType;
const size_t kMaxPacketEventTimeMapSize = 1000;
bool IsReceiverEvent(CastLoggingEvent event) {
return event == FRAME_DECODED
|| event == FRAME_PLAYOUT
|| event == FRAME_ACK_SENT
|| event == PACKET_RECEIVED;
}
} // namespace
StatsEventSubscriber::SimpleHistogram::SimpleHistogram(int64_t min,
int64_t max,
int64_t width)
: min_(min), max_(max), width_(width), buckets_((max - min) / width + 2) {
CHECK_GT(buckets_.size(), 2u);
CHECK_EQ(0, (max_ - min_) % width_);
}
StatsEventSubscriber::SimpleHistogram::~SimpleHistogram() = default;
void StatsEventSubscriber::SimpleHistogram::Add(int64_t sample) {
if (sample < min_) {
++buckets_.front();
} else if (sample >= max_) {
++buckets_.back();
} else {
size_t index = 1 + (sample - min_) / width_;
DCHECK_LT(index, buckets_.size());
++buckets_[index];
}
}
void StatsEventSubscriber::SimpleHistogram::Reset() {
buckets_.assign(buckets_.size(), 0);
}
base::Value::List StatsEventSubscriber::SimpleHistogram::GetHistogram() const {
base::Value::List histo;
if (buckets_.front()) {
base::Value::Dict bucket;
bucket.Set(base::StringPrintf("<%" PRId64, min_), buckets_.front());
histo.Append(std::move(bucket));
}
for (size_t i = 1; i < buckets_.size() - 1; i++) {
if (!buckets_[i])
continue;
base::Value::Dict bucket;
int64_t lower = min_ + (i - 1) * width_;
int64_t upper = lower + width_ - 1;
bucket.Set(base::StringPrintf("%" PRId64 "-%" PRId64, lower, upper),
buckets_[i]);
histo.Append(std::move(bucket));
}
if (buckets_.back()) {
base::Value::Dict bucket;
bucket.Set(base::StringPrintf(">=%" PRId64, max_), buckets_.back());
histo.Append(std::move(bucket));
}
return histo;
}
StatsEventSubscriber::StatsEventSubscriber(
EventMediaType event_media_type,
const base::TickClock* clock,
ReceiverTimeOffsetEstimator* offset_estimator)
: event_media_type_(event_media_type),
clock_(clock),
offset_estimator_(offset_estimator),
capture_latency_datapoints_(0),
encode_time_datapoints_(0),
queueing_latency_datapoints_(0),
network_latency_datapoints_(0),
packet_latency_datapoints_(0),
frame_latency_datapoints_(0),
e2e_latency_datapoints_(0),
num_frames_dropped_by_encoder_(0),
num_frames_late_(0),
start_time_(clock_->NowTicks()) {
DCHECK(event_media_type == AUDIO_EVENT || event_media_type == VIDEO_EVENT);
InitHistograms();
}
StatsEventSubscriber::~StatsEventSubscriber() {
DCHECK(thread_checker_.CalledOnValidThread());
}
void StatsEventSubscriber::OnReceiveFrameEvent(const FrameEvent& frame_event) {
DCHECK(thread_checker_.CalledOnValidThread());
CastLoggingEvent type = frame_event.type;
if (frame_event.media_type != event_media_type_)
return;
auto it = frame_stats_.find(type);
if (it == frame_stats_.end()) {
FrameLogStats stats;
stats.event_counter = 1;
stats.sum_size = frame_event.size;
stats.sum_delay = frame_event.delay_delta;
frame_stats_.insert(std::make_pair(type, stats));
} else {
++(it->second.event_counter);
it->second.sum_size += frame_event.size;
it->second.sum_delay += frame_event.delay_delta;
}
bool is_receiver_event = IsReceiverEvent(type);
UpdateFirstLastEventTime(frame_event.timestamp, is_receiver_event);
if (type == FRAME_CAPTURE_BEGIN) {
RecordFrameCaptureTime(frame_event);
} else if (type == FRAME_CAPTURE_END) {
RecordCaptureLatency(frame_event);
} else if (type == FRAME_ENCODED) {
RecordEncodeLatency(frame_event);
} else if (type == FRAME_ACK_SENT) {
RecordFrameTxLatency(frame_event);
} else if (type == FRAME_PLAYOUT) {
RecordE2ELatency(frame_event);
base::TimeDelta delay_delta = frame_event.delay_delta;
// Positive delay_delta means the frame is late.
if (delay_delta.is_positive()) {
num_frames_late_++;
histograms_[LATE_FRAME_MS_HISTO]->Add(delay_delta.InMillisecondsF());
}
}
if (is_receiver_event)
UpdateLastResponseTime(frame_event.timestamp);
}
void StatsEventSubscriber::OnReceivePacketEvent(
const PacketEvent& packet_event) {
DCHECK(thread_checker_.CalledOnValidThread());
CastLoggingEvent type = packet_event.type;
if (packet_event.media_type != event_media_type_)
return;
auto it = packet_stats_.find(type);
if (it == packet_stats_.end()) {
PacketLogStats stats;
stats.event_counter = 1;
stats.sum_size = packet_event.size;
packet_stats_.insert(std::make_pair(type, stats));
} else {
++(it->second.event_counter);
it->second.sum_size += packet_event.size;
}
bool is_receiver_event = IsReceiverEvent(type);
UpdateFirstLastEventTime(packet_event.timestamp, is_receiver_event);
if (type == PACKET_SENT_TO_NETWORK ||
type == PACKET_RECEIVED) {
RecordPacketRelatedLatencies(packet_event);
} else if (type == PACKET_RETRANSMITTED) {
// We only measure network latency using packets that doesn't have to be
// retransmitted as there is precisely one sent-receive timestamp pairs.
ErasePacketSentTime(packet_event);
}
if (is_receiver_event)
UpdateLastResponseTime(packet_event.timestamp);
}
void StatsEventSubscriber::UpdateFirstLastEventTime(base::TimeTicks timestamp,
bool is_receiver_event) {
if (is_receiver_event) {
base::TimeDelta receiver_offset;
if (!GetReceiverOffset(&receiver_offset))
return;
timestamp -= receiver_offset;
}
if (first_event_time_.is_null()) {
first_event_time_ = timestamp;
} else {
first_event_time_ = std::min(first_event_time_, timestamp);
}
if (last_event_time_.is_null()) {
last_event_time_ = timestamp;
} else {
last_event_time_ = std::max(last_event_time_, timestamp);
}
}
base::Value::Dict StatsEventSubscriber::GetStats() const {
StatsMap stats_map;
GetStatsInternal(&stats_map);
base::Value::Dict ret;
base::Value::Dict stats;
for (StatsMap::const_iterator it = stats_map.begin(); it != stats_map.end();
++it) {
// Round to 3 digits after the decimal point.
stats.Set(CastStatToString(it->first), round(it->second * 1000.0) / 1000.0);
}
// Populate all histograms.
for (auto it = histograms_.begin(); it != histograms_.end(); ++it) {
stats.Set(CastStatToString(it->first), it->second->GetHistogram());
}
ret.Set(event_media_type_ == AUDIO_EVENT
? StatsEventSubscriber::kAudioStatsDictKey
: StatsEventSubscriber::kVideoStatsDictKey,
std::move(stats));
return ret;
}
void StatsEventSubscriber::Reset() {
DCHECK(thread_checker_.CalledOnValidThread());
frame_stats_.clear();
packet_stats_.clear();
total_capture_latency_ = base::TimeDelta();
capture_latency_datapoints_ = 0;
total_encode_time_ = base::TimeDelta();
encode_time_datapoints_ = 0;
total_queueing_latency_ = base::TimeDelta();
queueing_latency_datapoints_ = 0;
total_network_latency_ = base::TimeDelta();
network_latency_datapoints_ = 0;
total_packet_latency_ = base::TimeDelta();
packet_latency_datapoints_ = 0;
total_frame_latency_ = base::TimeDelta();
frame_latency_datapoints_ = 0;
total_e2e_latency_ = base::TimeDelta();
e2e_latency_datapoints_ = 0;
num_frames_dropped_by_encoder_ = 0;
num_frames_late_ = 0;
recent_frame_infos_.clear();
packet_sent_times_.clear();
start_time_ = clock_->NowTicks();
last_response_received_time_ = base::TimeTicks();
for (auto it = histograms_.begin(); it != histograms_.end(); ++it) {
it->second->Reset();
}
first_event_time_ = base::TimeTicks();
last_event_time_ = base::TimeTicks();
}
// static
const char* StatsEventSubscriber::CastStatToString(CastStat stat) {
switch (stat) {
STAT_ENUM_TO_STRING(CAPTURE_FPS);
STAT_ENUM_TO_STRING(ENCODE_FPS);
STAT_ENUM_TO_STRING(DECODE_FPS);
STAT_ENUM_TO_STRING(AVG_CAPTURE_LATENCY_MS);
STAT_ENUM_TO_STRING(AVG_ENCODE_TIME_MS);
STAT_ENUM_TO_STRING(AVG_QUEUEING_LATENCY_MS);
STAT_ENUM_TO_STRING(AVG_NETWORK_LATENCY_MS);
STAT_ENUM_TO_STRING(AVG_PACKET_LATENCY_MS);
STAT_ENUM_TO_STRING(AVG_FRAME_LATENCY_MS);
STAT_ENUM_TO_STRING(AVG_E2E_LATENCY_MS);
STAT_ENUM_TO_STRING(ENCODE_KBPS);
STAT_ENUM_TO_STRING(TRANSMISSION_KBPS);
STAT_ENUM_TO_STRING(RETRANSMISSION_KBPS);
STAT_ENUM_TO_STRING(MS_SINCE_LAST_RECEIVER_RESPONSE);
STAT_ENUM_TO_STRING(NUM_FRAMES_CAPTURED);
STAT_ENUM_TO_STRING(NUM_FRAMES_DROPPED_BY_ENCODER);
STAT_ENUM_TO_STRING(NUM_FRAMES_LATE);
STAT_ENUM_TO_STRING(NUM_PACKETS_SENT);
STAT_ENUM_TO_STRING(NUM_PACKETS_RETRANSMITTED);
STAT_ENUM_TO_STRING(NUM_PACKETS_RECEIVED);
STAT_ENUM_TO_STRING(NUM_PACKETS_RTX_REJECTED);
STAT_ENUM_TO_STRING(FIRST_EVENT_TIME_MS);
STAT_ENUM_TO_STRING(LAST_EVENT_TIME_MS);
STAT_ENUM_TO_STRING(CAPTURE_LATENCY_MS_HISTO);
STAT_ENUM_TO_STRING(ENCODE_TIME_MS_HISTO);
STAT_ENUM_TO_STRING(QUEUEING_LATENCY_MS_HISTO);
STAT_ENUM_TO_STRING(NETWORK_LATENCY_MS_HISTO);
STAT_ENUM_TO_STRING(PACKET_LATENCY_MS_HISTO);
STAT_ENUM_TO_STRING(FRAME_LATENCY_MS_HISTO);
STAT_ENUM_TO_STRING(E2E_LATENCY_MS_HISTO);
STAT_ENUM_TO_STRING(LATE_FRAME_MS_HISTO);
}
NOTREACHED();
return "";
}
const int kDefaultMaxLatencyBucketMs = 800;
const int kDefaultBucketWidthMs = 20;
// For small latency values.
const int kSmallMaxLatencyBucketMs = 100;
const int kSmallBucketWidthMs = 5;
// For large latency values.
const int kLargeMaxLatencyBucketMs = 1200;
const int kLargeBucketWidthMs = 50;
void StatsEventSubscriber::InitHistograms() {
histograms_[E2E_LATENCY_MS_HISTO] = std::make_unique<SimpleHistogram>(
0, kLargeMaxLatencyBucketMs, kLargeBucketWidthMs);
histograms_[QUEUEING_LATENCY_MS_HISTO] = std::make_unique<SimpleHistogram>(
0, kDefaultMaxLatencyBucketMs, kDefaultBucketWidthMs);
histograms_[NETWORK_LATENCY_MS_HISTO] = std::make_unique<SimpleHistogram>(
0, kDefaultMaxLatencyBucketMs, kDefaultBucketWidthMs);
histograms_[PACKET_LATENCY_MS_HISTO] = std::make_unique<SimpleHistogram>(
0, kDefaultMaxLatencyBucketMs, kDefaultBucketWidthMs);
histograms_[FRAME_LATENCY_MS_HISTO] = std::make_unique<SimpleHistogram>(
0, kDefaultMaxLatencyBucketMs, kDefaultBucketWidthMs);
histograms_[LATE_FRAME_MS_HISTO] = std::make_unique<SimpleHistogram>(
0, kDefaultMaxLatencyBucketMs, kDefaultBucketWidthMs);
histograms_[CAPTURE_LATENCY_MS_HISTO] = std::make_unique<SimpleHistogram>(
0, kSmallMaxLatencyBucketMs, kSmallBucketWidthMs);
histograms_[ENCODE_TIME_MS_HISTO] = std::make_unique<SimpleHistogram>(
0, kSmallMaxLatencyBucketMs, kSmallBucketWidthMs);
}
void StatsEventSubscriber::GetStatsInternal(StatsMap* stats_map) const {
DCHECK(thread_checker_.CalledOnValidThread());
stats_map->clear();
base::TimeTicks end_time = clock_->NowTicks();
PopulateFpsStat(
end_time, FRAME_CAPTURE_BEGIN, CAPTURE_FPS, stats_map);
PopulateFpsStat(
end_time, FRAME_ENCODED, ENCODE_FPS, stats_map);
PopulateFpsStat(
end_time, FRAME_DECODED, DECODE_FPS, stats_map);
PopulateFrameBitrateStat(end_time, stats_map);
PopulatePacketBitrateStat(end_time,
PACKET_SENT_TO_NETWORK,
TRANSMISSION_KBPS,
stats_map);
PopulatePacketBitrateStat(end_time,
PACKET_RETRANSMITTED,
RETRANSMISSION_KBPS,
stats_map);
PopulateFrameCountStat(FRAME_CAPTURE_END, NUM_FRAMES_CAPTURED, stats_map);
PopulatePacketCountStat(PACKET_SENT_TO_NETWORK, NUM_PACKETS_SENT, stats_map);
PopulatePacketCountStat(
PACKET_RETRANSMITTED, NUM_PACKETS_RETRANSMITTED, stats_map);
PopulatePacketCountStat(PACKET_RECEIVED, NUM_PACKETS_RECEIVED, stats_map);
PopulatePacketCountStat(
PACKET_RTX_REJECTED, NUM_PACKETS_RTX_REJECTED, stats_map);
if (capture_latency_datapoints_ > 0) {
double avg_capture_latency_ms =
total_capture_latency_.InMillisecondsF() / capture_latency_datapoints_;
stats_map->insert(
std::make_pair(AVG_CAPTURE_LATENCY_MS, avg_capture_latency_ms));
}
if (encode_time_datapoints_ > 0) {
double avg_encode_time_ms =
total_encode_time_.InMillisecondsF() / encode_time_datapoints_;
stats_map->insert(
std::make_pair(AVG_ENCODE_TIME_MS, avg_encode_time_ms));
}
if (queueing_latency_datapoints_ > 0) {
double avg_queueing_latency_ms =
total_queueing_latency_.InMillisecondsF() /
queueing_latency_datapoints_;
stats_map->insert(
std::make_pair(AVG_QUEUEING_LATENCY_MS, avg_queueing_latency_ms));
}
if (network_latency_datapoints_ > 0) {
double avg_network_latency_ms =
total_network_latency_.InMillisecondsF() / network_latency_datapoints_;
stats_map->insert(
std::make_pair(AVG_NETWORK_LATENCY_MS, avg_network_latency_ms));
}
if (packet_latency_datapoints_ > 0) {
double avg_packet_latency_ms =
total_packet_latency_.InMillisecondsF() / packet_latency_datapoints_;
stats_map->insert(
std::make_pair(AVG_PACKET_LATENCY_MS, avg_packet_latency_ms));
}
if (frame_latency_datapoints_ > 0) {
double avg_frame_latency_ms =
total_frame_latency_.InMillisecondsF() / frame_latency_datapoints_;
stats_map->insert(
std::make_pair(AVG_FRAME_LATENCY_MS, avg_frame_latency_ms));
}
if (e2e_latency_datapoints_ > 0) {
double avg_e2e_latency_ms =
total_e2e_latency_.InMillisecondsF() / e2e_latency_datapoints_;
stats_map->insert(std::make_pair(AVG_E2E_LATENCY_MS, avg_e2e_latency_ms));
}
if (!last_response_received_time_.is_null()) {
stats_map->insert(
std::make_pair(MS_SINCE_LAST_RECEIVER_RESPONSE,
(end_time - last_response_received_time_).InMillisecondsF()));
}
stats_map->insert(std::make_pair(NUM_FRAMES_DROPPED_BY_ENCODER,
num_frames_dropped_by_encoder_));
stats_map->insert(std::make_pair(NUM_FRAMES_LATE, num_frames_late_));
if (!first_event_time_.is_null()) {
stats_map->insert(std::make_pair(
FIRST_EVENT_TIME_MS,
(first_event_time_ - base::TimeTicks::UnixEpoch()).InMillisecondsF()));
}
if (!last_event_time_.is_null()) {
stats_map->insert(std::make_pair(
LAST_EVENT_TIME_MS,
(last_event_time_ - base::TimeTicks::UnixEpoch()).InMillisecondsF()));
}
}
StatsEventSubscriber::SimpleHistogram*
StatsEventSubscriber::GetHistogramForTesting(
CastStat stats) const {
DCHECK(histograms_.find(stats) != histograms_.end());
return histograms_.find(stats)->second.get();
}
bool StatsEventSubscriber::GetReceiverOffset(base::TimeDelta* offset) {
base::TimeDelta receiver_offset_lower_bound;
base::TimeDelta receiver_offset_upper_bound;
if (!offset_estimator_->GetReceiverOffsetBounds(
&receiver_offset_lower_bound, &receiver_offset_upper_bound)) {
return false;
}
*offset = (receiver_offset_lower_bound + receiver_offset_upper_bound) / 2;
return true;
}
void StatsEventSubscriber::MaybeInsertFrameInfo(RtpTimeTicks rtp_timestamp,
const FrameInfo& frame_info) {
// No need to insert if |rtp_timestamp| is the smaller than every key in the
// map as it is just going to get erased anyway.
if (recent_frame_infos_.size() == kMaxFrameInfoMapSize &&
rtp_timestamp < recent_frame_infos_.begin()->first) {
return;
}
recent_frame_infos_.insert(std::make_pair(rtp_timestamp, frame_info));
if (recent_frame_infos_.size() >= kMaxFrameInfoMapSize) {
auto erase_it = recent_frame_infos_.begin();
if (erase_it->second.encode_end_time.is_null())
num_frames_dropped_by_encoder_++;
recent_frame_infos_.erase(erase_it);
}
}
void StatsEventSubscriber::RecordFrameCaptureTime(
const FrameEvent& frame_event) {
FrameInfo frame_info;
frame_info.capture_time = frame_event.timestamp;
MaybeInsertFrameInfo(frame_event.rtp_timestamp, frame_info);
}
void StatsEventSubscriber::RecordCaptureLatency(const FrameEvent& frame_event) {
auto it = recent_frame_infos_.find(frame_event.rtp_timestamp);
if (it == recent_frame_infos_.end()) {
return;
}
if (!it->second.capture_time.is_null()) {
base::TimeDelta latency = frame_event.timestamp - it->second.capture_time;
total_capture_latency_ += latency;
capture_latency_datapoints_++;
histograms_[CAPTURE_LATENCY_MS_HISTO]->Add(latency.InMillisecondsF());
}
it->second.capture_end_time = frame_event.timestamp;
}
void StatsEventSubscriber::RecordEncodeLatency(const FrameEvent& frame_event) {
auto it = recent_frame_infos_.find(frame_event.rtp_timestamp);
if (it == recent_frame_infos_.end()) {
FrameInfo frame_info;
frame_info.encode_end_time = frame_event.timestamp;
MaybeInsertFrameInfo(frame_event.rtp_timestamp, frame_info);
return;
}
if (!it->second.capture_end_time.is_null()) {
base::TimeDelta latency =
frame_event.timestamp - it->second.capture_end_time;
total_encode_time_ += latency;
encode_time_datapoints_++;
histograms_[ENCODE_TIME_MS_HISTO]->Add(latency.InMillisecondsF());
}
it->second.encode_end_time = frame_event.timestamp;
}
void StatsEventSubscriber::RecordFrameTxLatency(const FrameEvent& frame_event) {
auto it = recent_frame_infos_.find(frame_event.rtp_timestamp);
if (it == recent_frame_infos_.end())
return;
if (it->second.encode_end_time.is_null())
return;
base::TimeDelta receiver_offset;
if (!GetReceiverOffset(&receiver_offset))
return;
base::TimeTicks sender_time = frame_event.timestamp - receiver_offset;
base::TimeDelta latency = sender_time - it->second.encode_end_time;
total_frame_latency_ += latency;
frame_latency_datapoints_++;
histograms_[FRAME_LATENCY_MS_HISTO]->Add(latency.InMillisecondsF());
}
void StatsEventSubscriber::RecordE2ELatency(const FrameEvent& frame_event) {
base::TimeDelta receiver_offset;
if (!GetReceiverOffset(&receiver_offset))
return;
auto it = recent_frame_infos_.find(frame_event.rtp_timestamp);
if (it == recent_frame_infos_.end())
return;
base::TimeTicks playout_time = frame_event.timestamp - receiver_offset;
base::TimeDelta latency = playout_time - it->second.capture_time;
total_e2e_latency_ += latency;
e2e_latency_datapoints_++;
histograms_[E2E_LATENCY_MS_HISTO]->Add(latency.InMillisecondsF());
}
void StatsEventSubscriber::UpdateLastResponseTime(
base::TimeTicks receiver_time) {
base::TimeDelta receiver_offset;
if (!GetReceiverOffset(&receiver_offset))
return;
base::TimeTicks sender_time = receiver_time - receiver_offset;
last_response_received_time_ = sender_time;
}
void StatsEventSubscriber::ErasePacketSentTime(
const PacketEvent& packet_event) {
std::pair<RtpTimeTicks, uint16_t> key(
std::make_pair(packet_event.rtp_timestamp, packet_event.packet_id));
packet_sent_times_.erase(key);
}
void StatsEventSubscriber::RecordPacketRelatedLatencies(
const PacketEvent& packet_event) {
// Log queueing latency.
if (packet_event.type == PACKET_SENT_TO_NETWORK) {
auto it = recent_frame_infos_.find(packet_event.rtp_timestamp);
if (it != recent_frame_infos_.end()) {
base::TimeDelta latency =
packet_event.timestamp - it->second.encode_end_time;
total_queueing_latency_ += latency;
queueing_latency_datapoints_++;
histograms_[QUEUEING_LATENCY_MS_HISTO]->Add(
latency.InMillisecondsF());
}
}
// Log network latency and total packet latency;
base::TimeDelta receiver_offset;
if (!GetReceiverOffset(&receiver_offset))
return;
std::pair<RtpTimeTicks, uint16_t> key(
std::make_pair(packet_event.rtp_timestamp, packet_event.packet_id));
auto it = packet_sent_times_.find(key);
if (it == packet_sent_times_.end()) {
std::pair<base::TimeTicks, CastLoggingEvent> value =
std::make_pair(packet_event.timestamp, packet_event.type);
packet_sent_times_.insert(std::make_pair(key, value));
if (packet_sent_times_.size() > kMaxPacketEventTimeMapSize)
packet_sent_times_.erase(packet_sent_times_.begin());
} else {
std::pair<base::TimeTicks, CastLoggingEvent> value = it->second;
CastLoggingEvent recorded_type = value.second;
bool match = false;
base::TimeTicks packet_sent_time;
base::TimeTicks packet_received_time;
if (recorded_type == PACKET_SENT_TO_NETWORK &&
packet_event.type == PACKET_RECEIVED) {
packet_sent_time = value.first;
packet_received_time = packet_event.timestamp;
match = true;
} else if (recorded_type == PACKET_RECEIVED &&
packet_event.type == PACKET_SENT_TO_NETWORK) {
packet_sent_time = packet_event.timestamp;
packet_received_time = value.first;
match = true;
}
if (match) {
packet_sent_times_.erase(it);
// Subtract by offset.
packet_received_time -= receiver_offset;
base::TimeDelta latency_delta = packet_received_time - packet_sent_time;
total_network_latency_ += latency_delta;
network_latency_datapoints_++;
histograms_[NETWORK_LATENCY_MS_HISTO]->Add(
latency_delta.InMillisecondsF());
// Log total network latency.
auto frame_it = recent_frame_infos_.find(packet_event.rtp_timestamp);
if (frame_it != recent_frame_infos_.end()) {
base::TimeDelta latency =
packet_received_time - frame_it->second.encode_end_time;
total_packet_latency_ += latency;
packet_latency_datapoints_++;
histograms_[PACKET_LATENCY_MS_HISTO]->Add(
latency.InMillisecondsF());
}
}
}
}
void StatsEventSubscriber::PopulateFpsStat(base::TimeTicks end_time,
CastLoggingEvent event,
CastStat stat,
StatsMap* stats_map) const {
auto it = frame_stats_.find(event);
if (it != frame_stats_.end()) {
double fps = 0.0;
base::TimeDelta duration = (end_time - start_time_);
int count = it->second.event_counter;
if (duration.is_positive())
fps = count / duration.InSecondsF();
stats_map->insert(std::make_pair(stat, fps));
}
}
void StatsEventSubscriber::PopulateFrameCountStat(CastLoggingEvent event,
CastStat stat,
StatsMap* stats_map) const {
auto it = frame_stats_.find(event);
if (it != frame_stats_.end()) {
stats_map->insert(std::make_pair(stat, it->second.event_counter));
}
}
void StatsEventSubscriber::PopulatePacketCountStat(CastLoggingEvent event,
CastStat stat,
StatsMap* stats_map) const {
auto it = packet_stats_.find(event);
if (it != packet_stats_.end()) {
stats_map->insert(std::make_pair(stat, it->second.event_counter));
}
}
void StatsEventSubscriber::PopulateFrameBitrateStat(base::TimeTicks end_time,
StatsMap* stats_map) const {
auto it = frame_stats_.find(FRAME_ENCODED);
if (it != frame_stats_.end()) {
double kbps = 0.0;
base::TimeDelta duration = end_time - start_time_;
if (duration.is_positive()) {
kbps = it->second.sum_size / duration.InMillisecondsF() * 8;
}
stats_map->insert(std::make_pair(ENCODE_KBPS, kbps));
}
}
void StatsEventSubscriber::PopulatePacketBitrateStat(
base::TimeTicks end_time,
CastLoggingEvent event,
CastStat stat,
StatsMap* stats_map) const {
auto it = packet_stats_.find(event);
if (it != packet_stats_.end()) {
double kbps = 0;
base::TimeDelta duration = end_time - start_time_;
if (duration.is_positive()) {
kbps = it->second.sum_size / duration.InMillisecondsF() * 8;
}
stats_map->insert(std::make_pair(stat, kbps));
}
}
StatsEventSubscriber::FrameLogStats::FrameLogStats()
: event_counter(0), sum_size(0) {}
StatsEventSubscriber::FrameLogStats::~FrameLogStats() = default;
StatsEventSubscriber::PacketLogStats::PacketLogStats()
: event_counter(0), sum_size(0) {}
StatsEventSubscriber::PacketLogStats::~PacketLogStats() = default;
StatsEventSubscriber::FrameInfo::FrameInfo() : encoded(false) {
}
StatsEventSubscriber::FrameInfo::~FrameInfo() = default;
} // namespace cast
} // namespace media