blob: 4908b642585434865de3dd00e69d91daeac5e281 [file] [log] [blame] [edit]
// Copyright 2013 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_POWER_MONITOR_POWER_MONITOR_DEVICE_SOURCE_H_
#define BASE_POWER_MONITOR_POWER_MONITOR_DEVICE_SOURCE_H_
#include <memory>
#include <vector>
#include "base/base_export.h"
#include "base/power_monitor/power_monitor_source.h"
#include "base/power_monitor/power_observer.h"
#include "build/build_config.h"
#include "build/chromeos_buildflags.h"
#if BUILDFLAG(IS_WIN)
#include <windows.h>
#include "base/power_monitor/speed_limit_observer_win.h"
#include "base/threading/sequence_bound.h"
#endif // BUILDFLAG(IS_WIN)
#if BUILDFLAG(IS_MAC)
#include <IOKit/IOTypes.h>
#include "base/apple/scoped_cftyperef.h"
#include "base/mac/scoped_ionotificationportref.h"
#include "base/power_monitor/battery_level_provider.h"
#include "base/power_monitor/iopm_power_source_sampling_event_source.h"
#include "base/power_monitor/thermal_state_observer_mac.h"
#endif // BUILDFLAG(IS_MAC)
#if BUILDFLAG(IS_IOS)
#include <objc/runtime.h>
#endif // BUILDFLAG(IS_IOS)
namespace base {
// A class used to monitor the power state change and notify the observers about
// the change event.
class BASE_EXPORT PowerMonitorDeviceSource : public PowerMonitorSource {
public:
PowerMonitorDeviceSource();
PowerMonitorDeviceSource(const PowerMonitorDeviceSource&) = delete;
PowerMonitorDeviceSource& operator=(const PowerMonitorDeviceSource&) = delete;
~PowerMonitorDeviceSource() override;
#if BUILDFLAG(IS_CHROMEOS)
// On Chrome OS, Chrome receives power-related events from powerd, the system
// power daemon, via D-Bus signals received on the UI thread. base can't
// directly depend on that code, so this class instead exposes static methods
// so that events can be passed in.
static void SetPowerSource(
PowerStateObserver::BatteryPowerStatus battery_power_status);
static void HandleSystemSuspending();
static void HandleSystemResumed();
static void ThermalEventReceived(
PowerThermalObserver::DeviceThermalState state);
// These two methods is used for handling thermal state update requests, such
// as asking for initial state when starting lisitening to thermal change.
PowerThermalObserver::DeviceThermalState GetCurrentThermalState()
const override;
void SetCurrentThermalState(
PowerThermalObserver::DeviceThermalState state) override;
#endif
private:
friend class PowerMonitorDeviceSourceTest;
#if BUILDFLAG(IS_WIN)
// Represents a message-only window for power message handling on Windows.
// Only allow PowerMonitor to create it.
class PowerMessageWindow {
public:
PowerMessageWindow();
~PowerMessageWindow();
private:
static LRESULT CALLBACK WndProcThunk(HWND hwnd,
UINT message,
WPARAM wparam,
LPARAM lparam);
// Instance of the module containing the window procedure.
HMODULE instance_ = nullptr;
// A hidden message-only window.
HWND message_hwnd_ = nullptr;
// A handle, returned when we register for power setting notification
HPOWERNOTIFY power_notify_handle_ = nullptr;
};
#endif // BUILDFLAG(IS_WIN)
#if BUILDFLAG(IS_APPLE) || BUILDFLAG(IS_WIN)
void PlatformInit();
void PlatformDestroy();
#endif // BUILDFLAG(IS_APPLE) || BUILDFLAG(IS_WIN)
#if BUILDFLAG(IS_MAC)
// Callback from IORegisterForSystemPower(). |refcon| is the |this| pointer.
static void SystemPowerEventCallback(void* refcon,
io_service_t service,
natural_t message_type,
void* message_argument);
#endif // BUILDFLAG(IS_MAC)
// Platform-specific method to check whether the system is currently
// running on battery power. Returns kBatteryPower if running on battery,
// kExternalPower if running on external power or kUnknown if the power
// state is unknown (for example, during early process lifetime when the
// state hasn't been obtained yet).
PowerStateObserver::BatteryPowerStatus GetBatteryPowerStatus() const override;
#if BUILDFLAG(IS_ANDROID)
PowerThermalObserver::DeviceThermalState GetCurrentThermalState()
const override;
int GetRemainingBatteryCapacity() const override;
#endif // BUILDFLAG(IS_ANDROID)
#if BUILDFLAG(IS_WIN)
// PowerMonitorSource:
int GetInitialSpeedLimit() const override;
#endif // BUILDFLAG(IS_WIN)
#if BUILDFLAG(IS_MAC)
// PowerMonitorSource:
PowerThermalObserver::DeviceThermalState GetCurrentThermalState()
const override;
int GetInitialSpeedLimit() const override;
// Retrieves the current battery state to update `is_on_battery_`.
void GetBatteryState();
void OnBatteryStateReceived(
const std::optional<BatteryLevelProvider::BatteryState>& battery_state);
// Reference to the system IOPMrootDomain port.
io_connect_t power_manager_port_ = IO_OBJECT_NULL;
// Notification port that delivers power (sleep/wake) notifications.
mac::ScopedIONotificationPortRef notification_port_;
// Notifier reference for the |notification_port_|.
io_object_t notifier_ = IO_OBJECT_NULL;
// Generates power-source-change events.
IOPMPowerSourceSamplingEventSource power_source_event_source_;
std::unique_ptr<BatteryLevelProvider> battery_level_provider_;
// Observer of thermal state events: critical temperature etc.
std::unique_ptr<ThermalStateObserverMac> thermal_state_observer_;
PowerStateObserver::BatteryPowerStatus battery_power_status_ =
PowerStateObserver::BatteryPowerStatus::kUnknown;
#endif
#if BUILDFLAG(IS_IOS)
// Holds pointers to system event notification observers.
std::vector<id> notification_observers_;
#endif
#if BUILDFLAG(IS_WIN)
PowerMessageWindow power_message_window_;
// |speed_limit_observer_| is owned by the main/UI thread but the
// SpeedLimitObserverWin is bound to a different sequence.
std::unique_ptr<base::SequenceBound<SpeedLimitObserverWin>>
speed_limit_observer_;
#endif
#if BUILDFLAG(IS_CHROMEOS)
PowerThermalObserver::DeviceThermalState current_thermal_state_ =
PowerThermalObserver::DeviceThermalState::kUnknown;
#endif
};
} // namespace base
#endif // BASE_POWER_MONITOR_POWER_MONITOR_DEVICE_SOURCE_H_