| // 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/atomicops.h" |
| #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()) { |
| DCHECK(PowerMonitor::IsInitialized()); |
| PowerMonitor::AddPowerSuspendObserver(this); |
| const bool on_battery = |
| PowerMonitor::AddPowerStateObserverAndReturnOnBatteryState(this); |
| UseHiResClock(!on_battery); |
| |
| // Start polling the high resolution timer usage. |
| Time::ResetHighResolutionTimerUsage(); |
| timer_.Start(FROM_HERE, kUsageSampleInterval, |
| BindRepeating(&ReportHighResolutionTimerUsage)); |
| } |
| } |
| |
| HighResolutionTimerManager::~HighResolutionTimerManager() { |
| if (HighResolutionTimerAllowed()) { |
| PowerMonitor::RemovePowerSuspendObserver(this); |
| PowerMonitor::RemovePowerStateObserver(this); |
| UseHiResClock(false); |
| } |
| } |
| |
| void HighResolutionTimerManager::OnPowerStateChange(bool on_battery_power) { |
| DCHECK(HighResolutionTimerAllowed()); |
| UseHiResClock(!on_battery_power); |
| } |
| |
| 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 |