blob: 7af4427a8f89ff0da249a2b2f4c2fa6adc7cea0e [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.
#ifndef BASE_TASK_COMMON_TASK_ANNOTATOR_H_
#define BASE_TASK_COMMON_TASK_ANNOTATOR_H_
#include <stdint.h>
#include <string_view>
#include "base/auto_reset.h"
#include "base/base_export.h"
#include "base/memory/raw_ptr_exclusion.h"
#include "base/memory/safety_checks.h"
#include "base/pending_task.h"
#include "base/time/tick_clock.h"
#include "base/trace_event/trace_event.h"
#include "base/types/pass_key.h"
namespace base {
// Enables task-controlled purge for the scheduler loop quarantine in
// TaskAnnotator.
BASE_EXPORT void EnableSchedulerLoopQuarantineTaskControlledPurge();
// Disables task-controlled purge for the scheduler loop quarantine in
// TaskAnnotator. For use in tests.
BASE_EXPORT void DisableSchedulerLoopQuarantineTaskControlledPurgeForTesting();
namespace sequence_manager::internal {
class WorkQueue;
}
// Constant used to measure which long-running tasks should be traced.
constexpr TimeDelta kMaxTaskDurationTimeDelta = Milliseconds(4);
// Implements common debug annotations for posted tasks. This includes data
// such as task origins, IPC message contexts, queueing durations and memory
// usage.
class BASE_EXPORT TaskAnnotator {
public:
class ObserverForTesting {
public:
// Invoked just before RunTask() in the scope in which the task is about to
// be executed.
virtual void BeforeRunTask(const PendingTask* pending_task) = 0;
};
// This is used to set the |ipc_hash| field for PendingTasks. It is intended
// to be used only from within generated IPC handler dispatch code.
class ScopedSetIpcHash;
// This is used to track long-running browser-UI tasks. It is intended to
// be used for low-overhead logging to produce longer traces, particularly to
// help the scroll jank reduction effort.
class LongTaskTracker;
static const PendingTask* CurrentTaskForThread();
static void SetCurrentTaskForThread(
PassKey<sequence_manager::internal::WorkQueue>,
const PendingTask* pending_task);
static void OnIPCReceived(const char* interface_name,
uint32_t (*method_info)(),
bool is_response);
static void MarkCurrentTaskAsInterestingForTracing();
// TRACE_EVENT argument helper, writing the task start time into
// EventContext.
// NOTE: Should only be used with TRACE_EVENT or TRACE_EVENT_BEGIN since the
// function records the timestamp for event start at call time.
static void EmitTaskTimingDetails(perfetto::EventContext& ctx);
TaskAnnotator();
TaskAnnotator(const TaskAnnotator&) = delete;
TaskAnnotator& operator=(const TaskAnnotator&) = delete;
~TaskAnnotator();
// Called to indicate that a task is about to be queued to run in the future,
// giving one last chance for this TaskAnnotator to add metadata to
// |pending_task| before it is moved into the queue.
void WillQueueTask(perfetto::StaticString trace_event_name,
TaskMetadata* pending_task);
// Creates a process-wide unique ID to represent this task in trace events.
// This will be mangled with a Process ID hash to reduce the likelyhood of
// colliding with TaskAnnotator pointers on other processes. Callers may use
// this when generating their own flow events (i.e. when passing
// |queue_function == nullptr| in above methods).
uint64_t GetTaskTraceID(const TaskMetadata& task) const;
// Run the given task, emitting the toplevel trace event and additional
// trace event arguments. Like for TRACE_EVENT macros, all of the arguments
// are used (i.e. lambdas are invoked) before this function exits, so it's
// safe to pass reference-capturing lambdas here.
template <typename... Args>
void RunTask(perfetto::StaticString event_name,
PendingTask& pending_task,
Args&&... args) {
TRACE_EVENT(
"toplevel", event_name,
[&](perfetto::EventContext& ctx) {
EmitTaskLocation(ctx, pending_task);
MaybeEmitDelayAndPolicy(ctx, pending_task);
MaybeEmitIncomingTaskFlow(ctx, pending_task);
MaybeEmitIPCHash(ctx, pending_task);
},
std::forward<Args>(args)...);
RunTaskImpl(pending_task);
}
private:
friend class TaskAnnotatorBacktraceIntegrationTest;
// Run a previously queued task.
NOT_TAIL_CALLED void RunTaskImpl(PendingTask& pending_task);
// Registers an ObserverForTesting that will be invoked by all TaskAnnotators'
// RunTask(). This registration and the implementation of BeforeRunTask() are
// responsible to ensure thread-safety.
static void RegisterObserverForTesting(ObserverForTesting* observer);
static void ClearObserverForTesting();
// TRACE_EVENT argument helper, writing the task location data into
// EventContext.
static void EmitTaskLocation(perfetto::EventContext& ctx,
const PendingTask& task);
static void MaybeEmitDelayAndPolicy(perfetto::EventContext& ctx,
const PendingTask& task);
// TRACE_EVENT argument helper, writing the incoming task flow information
// into EventContext if toplevel.flow category is enabled.
void MaybeEmitIncomingTaskFlow(perfetto::EventContext& ctx,
const PendingTask& task) const;
void MaybeEmitIPCHash(perfetto::EventContext& ctx,
const PendingTask& task) const;
};
class BASE_EXPORT [[maybe_unused, nodiscard]] TaskAnnotator::ScopedSetIpcHash {
public:
explicit ScopedSetIpcHash(uint32_t ipc_hash);
// Compile-time-const string identifying the current IPC context. Not always
// available due to binary size constraints, so IPC hash might be set instead.
explicit ScopedSetIpcHash(const char* ipc_interface_name);
ScopedSetIpcHash(const ScopedSetIpcHash&) = delete;
ScopedSetIpcHash& operator=(const ScopedSetIpcHash&) = delete;
~ScopedSetIpcHash();
uint32_t GetIpcHash() const { return ipc_hash_; }
const char* GetIpcInterfaceName() const { return ipc_interface_name_; }
static uint32_t MD5HashMetricName(std::string_view name);
private:
ScopedSetIpcHash(uint32_t ipc_hash, const char* ipc_interface_name);
const AutoReset<ScopedSetIpcHash*> resetter_;
uint32_t ipc_hash_;
const char* ipc_interface_name_;
};
class BASE_EXPORT [[maybe_unused, nodiscard]] TaskAnnotator::LongTaskTracker {
public:
LongTaskTracker(const TickClock* tick_clock,
PendingTask& pending_task,
TaskAnnotator* task_annotator,
TimeTicks task_start_time);
LongTaskTracker(const LongTaskTracker&) = delete;
~LongTaskTracker();
void SetIpcDetails(const char* interface_name,
uint32_t (*method_info)(),
bool is_response);
void MaybeTraceInterestingTaskDetails();
// In long-task tracking, not every task (including its queue time) will be
// recorded in a trace. If a particular task + queue time needs to be
// recorded, flag it explicitly. For example, input tasks are required for
// calculating scroll jank metrics.
bool is_interesting_task = false;
TimeTicks GetTaskStartTime() const { return task_start_time_; }
private:
void EmitReceivedIPCDetails(perfetto::EventContext& ctx);
const AutoReset<LongTaskTracker*> resetter_;
// For tracking task duration.
//
// RAW_PTR_EXCLUSION: Performance reasons: based on analysis of sampling
// profiler data (TaskAnnotator::LongTaskTracker::~LongTaskTracker).
RAW_PTR_EXCLUSION const TickClock* tick_clock_; // Not owned.
// Task start time, sampled before the LongTaskTracker instance
// is created.
TimeTicks task_start_time_;
TimeTicks task_end_time_;
// Tracing variables.
const char* ipc_interface_name_ = nullptr;
uint32_t ipc_hash_ = 0;
// IPC method info to retrieve IPC hash and method address from trace, if
// known. Note that this will not compile in the Native client.
uint32_t (*ipc_method_info_)();
bool is_response_ = false;
// RAW_PTR_EXCLUSION: Performance reasons: based on analysis of sampling
// profiler data (TaskAnnotator::LongTaskTracker::~LongTaskTracker).
[[maybe_unused]] RAW_PTR_EXCLUSION PendingTask& pending_task_;
[[maybe_unused]] RAW_PTR_EXCLUSION TaskAnnotator* task_annotator_;
};
} // namespace base
#endif // BASE_TASK_COMMON_TASK_ANNOTATOR_H_