blob: 01e57a44525b042fd74c7cc4e4650ef6c5789cdb [file] [log] [blame]
// Copyright 2017 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "media/mojo/services/media_metrics_provider.h"
#include <memory>
#include <utility>
#include "base/logging.h"
#include "base/metrics/histogram_functions.h"
#include "base/metrics/histogram_macros.h"
#include "build/build_config.h"
#include "media/learning/mojo/mojo_learning_task_controller_service.h"
#include "media/mojo/services/video_decode_stats_recorder.h"
#include "media/mojo/services/watch_time_recorder.h"
#include "mojo/public/cpp/bindings/self_owned_receiver.h"
#include "services/metrics/public/cpp/ukm_builders.h"
#include "services/metrics/public/cpp/ukm_recorder.h"
#if !defined(OS_ANDROID)
#include "media/filters/decrypting_video_decoder.h"
#endif
namespace media {
constexpr char kInvalidInitialize[] = "Initialize() was not called correctly.";
static uint64_t g_player_id = 0;
MediaMetricsProvider::PipelineInfo::PipelineInfo(bool is_incognito)
: is_incognito(is_incognito) {}
MediaMetricsProvider::PipelineInfo::~PipelineInfo() = default;
MediaMetricsProvider::MediaMetricsProvider(
BrowsingMode is_incognito,
FrameStatus is_top_frame,
ukm::SourceId source_id,
learning::FeatureValue origin,
VideoDecodePerfHistory::SaveCallback save_cb,
GetLearningSessionCallback learning_session_cb,
RecordAggregateWatchTimeCallback record_playback_cb)
: player_id_(g_player_id++),
is_top_frame_(is_top_frame == FrameStatus::kTopFrame),
source_id_(source_id),
origin_(origin),
save_cb_(std::move(save_cb)),
learning_session_cb_(learning_session_cb),
record_playback_cb_(std::move(record_playback_cb)),
uma_info_(is_incognito == BrowsingMode::kIncognito) {}
MediaMetricsProvider::~MediaMetricsProvider() {
// UKM may be unavailable in content_shell or other non-chrome/ builds; it
// may also be unavailable if browser shutdown has started; so this may be a
// nullptr. If it's unavailable, UKM reporting will be skipped.
ukm::UkmRecorder* ukm_recorder = ukm::UkmRecorder::Get();
if (!ukm_recorder || !initialized_)
return;
ukm::builders::Media_WebMediaPlayerState builder(source_id_);
builder.SetPlayerID(player_id_);
builder.SetIsTopFrame(is_top_frame_);
builder.SetIsEME(uma_info_.is_eme);
builder.SetIsMSE(is_mse_);
builder.SetFinalPipelineStatus(uma_info_.last_pipeline_status);
if (!is_mse_) {
builder.SetURLScheme(static_cast<int64_t>(url_scheme_));
if (container_name_)
builder.SetContainerName(*container_name_);
}
if (time_to_metadata_ != kNoTimestamp)
builder.SetTimeToMetadata(time_to_metadata_.InMilliseconds());
if (time_to_first_frame_ != kNoTimestamp)
builder.SetTimeToFirstFrame(time_to_first_frame_.InMilliseconds());
if (time_to_play_ready_ != kNoTimestamp)
builder.SetTimeToPlayReady(time_to_play_ready_.InMilliseconds());
builder.Record(ukm_recorder);
ReportPipelineUMA();
}
std::string MediaMetricsProvider::GetUMANameForAVStream(
const PipelineInfo& player_info) {
constexpr char kPipelineUmaPrefix[] = "Media.PipelineStatus.AudioVideo.";
std::string uma_name = kPipelineUmaPrefix;
if (player_info.video_codec == kCodecVP8) {
uma_name += "VP8.";
} else if (player_info.video_codec == kCodecVP9) {
uma_name += "VP9.";
} else if (player_info.video_codec == kCodecH264) {
uma_name += "H264.";
} else if (player_info.video_codec == kCodecAV1) {
uma_name += "AV1.";
} else {
return uma_name + "Other";
}
#if !defined(OS_ANDROID)
if (player_info.video_pipeline_info.decoder_name ==
media::DecryptingVideoDecoder::kDecoderName) {
return uma_name + "DVD";
}
#endif
if (player_info.video_pipeline_info.has_decrypting_demuxer_stream) {
uma_name += "DDS.";
}
// Note that HW essentially means 'platform' anyway. MediaCodec has been
// reported as HW forever, regardless of the underlying platform
// implementation.
if (player_info.video_pipeline_info.is_platform_decoder) {
uma_name += "HW";
} else {
uma_name += "SW";
}
return uma_name;
}
void MediaMetricsProvider::ReportPipelineUMA() {
if (uma_info_.has_video && uma_info_.has_audio) {
base::UmaHistogramExactLinear(GetUMANameForAVStream(uma_info_),
uma_info_.last_pipeline_status,
media::PIPELINE_STATUS_MAX + 1);
} else if (uma_info_.has_audio) {
base::UmaHistogramExactLinear("Media.PipelineStatus.AudioOnly",
uma_info_.last_pipeline_status,
media::PIPELINE_STATUS_MAX + 1);
} else if (uma_info_.has_video) {
base::UmaHistogramExactLinear("Media.PipelineStatus.VideoOnly",
uma_info_.last_pipeline_status,
media::PIPELINE_STATUS_MAX + 1);
} else {
// Note: This metric can be recorded as a result of normal operation with
// Media Source Extensions. If a site creates a MediaSource object but never
// creates a source buffer or appends data, PIPELINE_OK will be recorded.
base::UmaHistogramExactLinear("Media.PipelineStatus.Unsupported",
uma_info_.last_pipeline_status,
media::PIPELINE_STATUS_MAX + 1);
}
// Report whether video decoder fallback happened, but only if a video decoder
// was reported.
if (!uma_info_.video_pipeline_info.decoder_name.empty()) {
base::UmaHistogramBoolean("Media.VideoDecoderFallback",
uma_info_.video_decoder_changed);
}
// Report whether this player ever saw a playback event. Used to measure the
// effectiveness of efforts to reduce loaded-but-never-used players.
if (uma_info_.has_reached_have_enough) {
base::UmaHistogramBoolean("Media.HasEverPlayed", uma_info_.has_ever_played);
}
// Report whether an encrypted playback is in incognito window, excluding
// never-used players.
if (uma_info_.is_eme && uma_info_.has_ever_played) {
base::UmaHistogramBoolean("Media.EME.IsIncognito", uma_info_.is_incognito);
}
}
// static
void MediaMetricsProvider::Create(
BrowsingMode is_incognito,
FrameStatus is_top_frame,
GetSourceIdCallback get_source_id_cb,
GetOriginCallback get_origin_cb,
VideoDecodePerfHistory::SaveCallback save_cb,
GetLearningSessionCallback learning_session_cb,
GetRecordAggregateWatchTimeCallback get_record_playback_cb,
mojo::PendingReceiver<mojom::MediaMetricsProvider> receiver) {
mojo::MakeSelfOwnedReceiver(
std::make_unique<MediaMetricsProvider>(
is_incognito, is_top_frame, get_source_id_cb.Run(),
get_origin_cb.Run(), std::move(save_cb), learning_session_cb,
std::move(get_record_playback_cb).Run()),
std::move(receiver));
}
void MediaMetricsProvider::SetHasPlayed() {
uma_info_.has_ever_played = true;
}
void MediaMetricsProvider::SetHasAudio(AudioCodec audio_codec) {
uma_info_.audio_codec = audio_codec;
uma_info_.has_audio = true;
}
void MediaMetricsProvider::SetHasVideo(VideoCodec video_codec) {
uma_info_.video_codec = video_codec;
uma_info_.has_video = true;
}
void MediaMetricsProvider::SetHaveEnough() {
uma_info_.has_reached_have_enough = true;
}
void MediaMetricsProvider::SetVideoPipelineInfo(
const PipelineDecoderInfo& info) {
auto old_name = uma_info_.video_pipeline_info.decoder_name;
if (!old_name.empty() && old_name != info.decoder_name)
uma_info_.video_decoder_changed = true;
uma_info_.video_pipeline_info = info;
}
void MediaMetricsProvider::SetAudioPipelineInfo(
const PipelineDecoderInfo& info) {
uma_info_.audio_pipeline_info = info;
}
void MediaMetricsProvider::Initialize(bool is_mse,
mojom::MediaURLScheme url_scheme) {
if (initialized_) {
mojo::ReportBadMessage(kInvalidInitialize);
return;
}
is_mse_ = is_mse;
initialized_ = true;
url_scheme_ = url_scheme;
}
void MediaMetricsProvider::OnError(PipelineStatus status) {
DCHECK(initialized_);
uma_info_.last_pipeline_status = status;
}
void MediaMetricsProvider::SetIsEME() {
// This may be called before Initialize().
uma_info_.is_eme = true;
}
void MediaMetricsProvider::SetTimeToMetadata(base::TimeDelta elapsed) {
DCHECK(initialized_);
DCHECK_EQ(time_to_metadata_, kNoTimestamp);
time_to_metadata_ = elapsed;
}
void MediaMetricsProvider::SetTimeToFirstFrame(base::TimeDelta elapsed) {
DCHECK(initialized_);
DCHECK_EQ(time_to_first_frame_, kNoTimestamp);
time_to_first_frame_ = elapsed;
}
void MediaMetricsProvider::SetTimeToPlayReady(base::TimeDelta elapsed) {
DCHECK(initialized_);
DCHECK_EQ(time_to_play_ready_, kNoTimestamp);
time_to_play_ready_ = elapsed;
}
void MediaMetricsProvider::SetContainerName(
container_names::MediaContainerName container_name) {
DCHECK(initialized_);
DCHECK(!container_name_.has_value());
container_name_ = container_name;
}
void MediaMetricsProvider::AcquireWatchTimeRecorder(
mojom::PlaybackPropertiesPtr properties,
mojo::PendingReceiver<mojom::WatchTimeRecorder> receiver) {
if (!initialized_) {
mojo::ReportBadMessage(kInvalidInitialize);
return;
}
mojo::MakeSelfOwnedReceiver(
std::make_unique<WatchTimeRecorder>(std::move(properties), source_id_,
is_top_frame_, player_id_,
record_playback_cb_),
std::move(receiver));
}
void MediaMetricsProvider::AcquireVideoDecodeStatsRecorder(
mojo::PendingReceiver<mojom::VideoDecodeStatsRecorder> receiver) {
if (!initialized_) {
mojo::ReportBadMessage(kInvalidInitialize);
return;
}
if (!save_cb_) {
DVLOG(3) << __func__ << " Ignoring request, SaveCallback is null";
return;
}
mojo::MakeSelfOwnedReceiver(
std::make_unique<VideoDecodeStatsRecorder>(save_cb_, source_id_, origin_,
is_top_frame_, player_id_),
std::move(receiver));
}
void MediaMetricsProvider::AcquireLearningTaskController(
const std::string& taskName,
mojo::PendingReceiver<media::learning::mojom::LearningTaskController>
receiver) {
learning::LearningSession* session = learning_session_cb_.Run();
if (!session) {
DVLOG(3) << __func__ << " Ignoring request, unable to get LearningSession.";
return;
}
auto controller = session->GetController(taskName);
if (!controller) {
DVLOG(3) << __func__ << " Ignoring request, no controller found for task: '"
<< taskName << "'.";
return;
}
mojo::MakeSelfOwnedReceiver(
std::make_unique<learning::MojoLearningTaskControllerService>(
controller->GetLearningTask(), std::move(controller)),
std::move(receiver));
}
} // namespace media