blob: c5765208ecbfe1bfe4345aac098e2a6db55f86f0 [file] [log] [blame]
// Copyright 2025 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "base/threading/platform_thread_metrics.h"
#include <memory>
#include <mutex>
#include <optional>
#include "base/check.h"
#include "base/functional/bind.h"
#include "base/functional/callback_forward.h"
#include "base/memory/ptr_util.h"
#include "base/metrics/histogram_base.h"
#include "base/metrics/sparse_histogram.h"
#include "base/no_destructor.h"
#include "base/process/current_process.h"
#include "base/strings/strcat.h"
#include "base/strings/string_util.h"
#include "base/synchronization/lock.h"
#include "base/task/thread_pool.h"
#include "base/threading/platform_thread.h"
#include "base/time/time.h"
#include "build/build_config.h"
#if BUILDFLAG(IS_ANDROID)
#include "base/threading/platform_thread_internal_posix.h"
#endif // BUILDFLAG(IS_ANDROID)
namespace base {
namespace {
#if BUILDFLAG(IS_ANDROID)
std::string GetCurrentProcessName(CurrentProcess::NameKey key) {
std::string process_name = CurrentProcess::GetInstance().GetName(key);
// The process name may have whitespaces (e.g. 'GPU Process') and that won't
// work as a histogram name.
RemoveChars(process_name, " ", &process_name);
return process_name;
}
#endif // BUILDFLAG(IS_ANDROID)
} // namespace
// Apple and Windows have platform-specific create methods.
#if BUILDFLAG(IS_APPLE) || BUILDFLAG(IS_WIN)
// static
std::unique_ptr<PlatformThreadMetrics>
PlatformThreadMetrics::CreateForCurrentThread() {
return CreateFromHandle(PlatformThread::CurrentHandle());
}
#elif BUILDFLAG(IS_ANDROID) || BUILDFLAG(IS_CHROMEOS) || \
BUILDFLAG(IS_FUCHSIA) || BUILDFLAG(IS_LINUX)
// static
std::unique_ptr<PlatformThreadMetrics>
PlatformThreadMetrics::CreateForCurrentThread() {
return CreateFromId(PlatformThread::CurrentId());
}
// static
std::unique_ptr<PlatformThreadMetrics> PlatformThreadMetrics::CreateFromId(
PlatformThreadId tid) {
if (tid == kInvalidThreadId) {
return nullptr;
}
return WrapUnique(new PlatformThreadMetrics(tid));
}
#endif // BUILDFLAG(IS_ANDROID) || BUILDFLAG(IS_CHROMEOS) ||
// BUILDFLAG(IS_FUCHSIA) || BUILDFLAG(IS_LINUX)
PlatformThreadMetrics::~PlatformThreadMetrics() = default;
double PlatformThreadMetrics::GetCPUUsageProportion(TimeDelta cumulative_cpu) {
TimeTicks time = TimeTicks::Now();
if (!last_cpu_time_.has_value()) {
// First call, just set the last values.
CHECK(last_cumulative_cpu_.is_zero());
last_cumulative_cpu_ = cumulative_cpu;
last_cpu_time_ = time;
return 0;
}
TimeDelta cpu_time_delta = cumulative_cpu - last_cumulative_cpu_;
TimeDelta time_delta = time - last_cpu_time_.value();
if (time_delta.is_zero()) {
return 0;
}
last_cumulative_cpu_ = cumulative_cpu;
last_cpu_time_ = time;
return 100.0 * cpu_time_delta / time_delta;
}
std::optional<double> PlatformThreadMetrics::GetCPUUsageProportion() {
if (const std::optional<TimeDelta> cpu_usage = GetCumulativeCPUUsage()) {
return GetCPUUsageProportion(cpu_usage.value());
}
return std::nullopt;
}
#if BUILDFLAG(IS_ANDROID)
PlatformThreadPriorityMonitor::PlatformThreadPriorityMonitor()
: process_name_(GetCurrentProcessName({})) {}
PlatformThreadPriorityMonitor::~PlatformThreadPriorityMonitor() = default;
// static
PlatformThreadPriorityMonitor& PlatformThreadPriorityMonitor::Get() {
static base::NoDestructor<PlatformThreadPriorityMonitor> instance;
return *instance;
}
std::string PlatformThreadPriorityMonitor::GetHistogramNameForSuffix(
const std::string_view suffix) {
return StrCat({"Scheduling.ThreadPriority.", process_name_, ".", suffix});
}
void PlatformThreadPriorityMonitor::RegisterCurrentThread(
const std::string_view suffix) {
PlatformThreadId current_thread_id = PlatformThread::CurrentId();
HistogramBase* histogram =
SparseHistogram::FactoryGet(GetHistogramNameForSuffix(suffix),
HistogramBase::kUmaTargetedHistogramFlag);
AutoLock auto_lock(lock_);
thread_id_to_histogram_[current_thread_id] = histogram;
}
void PlatformThreadPriorityMonitor::UnregisterCurrentThread() {
PlatformThreadId current_thread_id = PlatformThread::CurrentId();
AutoLock auto_lock(lock_);
thread_id_to_histogram_.erase(current_thread_id);
}
void PlatformThreadPriorityMonitor::RecordThreadPriorities() {
AutoLock auto_lock(lock_);
for (auto& [thread_id, histogram] : thread_id_to_histogram_) {
histogram->Add(internal::GetThreadNiceValue(thread_id));
}
}
// static
void PlatformThreadPriorityMonitor::ScheduleRecordingTask() {
static NoDestructor<RepeatingClosure> closure(BindRepeating([] {
auto& monitor = PlatformThreadPriorityMonitor::Get();
monitor.RecordThreadPriorities();
monitor.ScheduleRecordingTask();
}));
ThreadPool::PostDelayedTask(
FROM_HERE,
{TaskPriority::BEST_EFFORT, TaskShutdownBehavior::SKIP_ON_SHUTDOWN},
*closure, kMinSamplingInterval);
}
void PlatformThreadPriorityMonitor::Start() {
std::call_once(once_flag_,
&PlatformThreadPriorityMonitor::ScheduleRecordingTask);
}
#endif // BUILDFLAG(IS_ANDROID)
} // namespace base