1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96
|
// 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 "base/metrics/user_metrics.h"
#include <stddef.h>
#include <algorithm>
#include <vector>
#include "base/functional/bind.h"
#include "base/lazy_instance.h"
#include "base/location.h"
#include "base/threading/thread_checker.h"
#include "base/time/time.h"
#include "base/trace_event/user_action.h"
namespace base {
namespace {
LazyInstance<std::vector<ActionCallback>>::DestructorAtExit g_callbacks =
LAZY_INSTANCE_INITIALIZER;
LazyInstance<scoped_refptr<SingleThreadTaskRunner>>::DestructorAtExit
g_task_runner = LAZY_INSTANCE_INITIALIZER;
void DispatchAction(const std::string& action, TimeTicks action_time) {
for (const ActionCallback& callback : g_callbacks.Get()) {
callback.Run(action, action_time);
}
}
} // namespace
void RecordAction(const UserMetricsAction& action) {
RecordComputedAction(action.str_);
}
void RecordComputedAction(const std::string& action) {
RecordComputedActionAt(action, TimeTicks::Now());
}
void RecordComputedActionSince(const std::string& action,
TimeDelta time_since) {
RecordComputedActionAt(action, TimeTicks::Now() - time_since);
}
void RecordComputedActionAt(const std::string& action, TimeTicks action_time) {
trace_event::EmitUserActionEvent(action, action_time);
if (!g_task_runner.Get()) {
DCHECK(g_callbacks.Get().empty());
return;
}
if (!g_task_runner.Get()->BelongsToCurrentThread()) {
g_task_runner.Get()->PostTask(
FROM_HERE, BindOnce(&DispatchAction, action, action_time));
return;
}
DispatchAction(action, action_time);
}
void AddActionCallback(const ActionCallback& callback) {
// Only allow adding a callback if the task runner is set.
DCHECK(g_task_runner.Get());
DCHECK(g_task_runner.Get()->BelongsToCurrentThread());
g_callbacks.Get().push_back(callback);
}
void RemoveActionCallback(const ActionCallback& callback) {
DCHECK(g_task_runner.Get());
DCHECK(g_task_runner.Get()->BelongsToCurrentThread());
std::vector<ActionCallback>* callbacks = g_callbacks.Pointer();
const auto i = std::ranges::find(*callbacks, callback);
if (i != callbacks->end()) {
callbacks->erase(i);
}
}
void SetRecordActionTaskRunner(
scoped_refptr<SingleThreadTaskRunner> task_runner) {
DCHECK(task_runner->BelongsToCurrentThread());
DCHECK(!g_task_runner.Get() || g_task_runner.Get()->BelongsToCurrentThread());
g_task_runner.Get() = task_runner;
}
scoped_refptr<SingleThreadTaskRunner> GetRecordActionTaskRunner() {
if (g_task_runner.IsCreated()) {
return g_task_runner.Get();
}
return nullptr;
}
} // namespace base
|