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
base / timer / hi_res_timer_manager_win.cc [blame]
// Copyright 2011 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/timer/hi_res_timer_manager.h"
#include <algorithm>
#include "base/base_switches.h"
#include "base/command_line.h"
#include "base/metrics/histogram_macros.h"
#include "base/power_monitor/power_monitor.h"
#include "base/time/time.h"
namespace base {
namespace {
constexpr TimeDelta kUsageSampleInterval = Minutes(10);
void ReportHighResolutionTimerUsage() {
UMA_HISTOGRAM_PERCENTAGE("Windows.HighResolutionTimerUsage",
Time::GetHighResolutionTimerUsage());
// Reset usage for the next interval.
Time::ResetHighResolutionTimerUsage();
}
bool HighResolutionTimerAllowed() {
return !CommandLine::ForCurrentProcess()->HasSwitch(
switches::kDisableHighResTimer);
}
} // namespace
HighResolutionTimerManager::HighResolutionTimerManager()
: hi_res_clock_available_(false) {
// Register for PowerMonitor callbacks only if high-resolution
// timers are allowed. If high-resolution timers are disabled
// we won't receive power state change callbacks and
// hi_res_clock_available_ will remain at its initial value.
if (HighResolutionTimerAllowed()) {
auto* power_monitor = base::PowerMonitor::GetInstance();
DCHECK(power_monitor->IsInitialized());
power_monitor->AddPowerSuspendObserver(this);
const bool on_battery =
power_monitor->AddPowerStateObserverAndReturnBatteryPowerStatus(this) ==
PowerStateObserver::BatteryPowerStatus::kBatteryPower;
UseHiResClock(!on_battery);
// Start polling the high resolution timer usage.
Time::ResetHighResolutionTimerUsage();
timer_.Start(FROM_HERE, kUsageSampleInterval,
BindRepeating(&ReportHighResolutionTimerUsage));
}
}
HighResolutionTimerManager::~HighResolutionTimerManager() {
if (HighResolutionTimerAllowed()) {
auto* power_monitor = base::PowerMonitor::GetInstance();
power_monitor->RemovePowerSuspendObserver(this);
power_monitor->RemovePowerStateObserver(this);
UseHiResClock(false);
}
}
void HighResolutionTimerManager::OnBatteryPowerStatusChange(
base::PowerStateObserver::BatteryPowerStatus battery_power_status) {
DCHECK(HighResolutionTimerAllowed());
UseHiResClock(battery_power_status !=
PowerStateObserver::BatteryPowerStatus::kBatteryPower);
}
void HighResolutionTimerManager::OnSuspend() {
DCHECK(HighResolutionTimerAllowed());
// Stop polling the usage to avoid including the standby time.
timer_.Stop();
}
void HighResolutionTimerManager::OnResume() {
DCHECK(HighResolutionTimerAllowed());
// Resume polling the usage.
Time::ResetHighResolutionTimerUsage();
timer_.Reset();
}
void HighResolutionTimerManager::UseHiResClock(bool use) {
DCHECK(HighResolutionTimerAllowed());
if (use == hi_res_clock_available_)
return;
hi_res_clock_available_ = use;
Time::EnableHighResolutionTimer(use);
}
} // namespace base