// 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 CHROMEOS_DBUS_POWER_POWER_POLICY_CONTROLLER_H_
#define CHROMEOS_DBUS_POWER_POWER_POLICY_CONTROLLER_H_
#include <map>
#include <optional>
#include <string>
#include <vector>
#include "base/component_export.h"
#include "base/memory/raw_ptr.h"
#include "base/values.h"
#include "chromeos/dbus/power/power_manager_client.h"
#include "chromeos/dbus/power_manager/policy.pb.h"
namespace chromeos {
// PowerPolicyController is responsible for sending Chrome's assorted power
// management preferences to the Chrome OS power manager.
class COMPONENT_EXPORT(DBUS_POWER) PowerPolicyController
: public PowerManagerClient::Observer {
public:
using PeakShiftDayConfig =
power_manager::PowerManagementPolicy::PeakShiftDayConfig;
using AdvancedBatteryChargeModeDayConfig =
power_manager::PowerManagementPolicy::AdvancedBatteryChargeModeDayConfig;
// Sets the global instance. Must be called before any calls to Get().
static void Initialize(PowerManagerClient* power_manager_client);
// Returns true if the global instance has been initialized.
static bool IsInitialized();
// Destroys the global instance.
static void Shutdown();
// Returns the global instance. Initialize() must be called first.
static PowerPolicyController* Get();
PowerPolicyController(const PowerPolicyController&) = delete;
PowerPolicyController& operator=(const PowerPolicyController&) = delete;
// Reasons why a wake lock may be added.
// TODO(derat): Remove this enum in favor of device::mojom::WakeLockReason
// once this class has been moved to the device service:
// https://crbug.com/702449
enum WakeLockReason {
REASON_AUDIO_PLAYBACK,
REASON_VIDEO_PLAYBACK,
REASON_OTHER,
};
// Note: Do not change these values; they are used by preferences.
enum Action {
ACTION_SUSPEND = 0,
ACTION_STOP_SESSION = 1,
ACTION_SHUT_DOWN = 2,
ACTION_DO_NOTHING = 3,
};
// Values of various power-management-related preferences.
struct PrefValues {
PrefValues();
~PrefValues();
// -1 is interpreted as "unset" by powerd, resulting in powerd's default
// delays being used instead. There are no similarly-interpreted values for
// the other fields, unfortunately (but the default values would only reach
// powerd if Chrome failed to override them with the pref-assigned values).
int ac_screen_dim_delay_ms = -1;
int ac_quick_dim_delay_ms = -1;
int ac_screen_off_delay_ms = -1;
int ac_screen_lock_delay_ms = -1;
int ac_quick_lock_delay_ms = -1;
int ac_idle_warning_delay_ms = -1;
int ac_idle_delay_ms = -1;
int battery_screen_dim_delay_ms = -1;
int battery_quick_dim_delay_ms = -1;
int battery_screen_off_delay_ms = -1;
int battery_screen_lock_delay_ms = -1;
int battery_quick_lock_delay_ms = -1;
int battery_idle_warning_delay_ms = -1;
int battery_idle_delay_ms = -1;
Action ac_idle_action = ACTION_SUSPEND;
Action battery_idle_action = ACTION_SUSPEND;
Action lid_closed_action = ACTION_SUSPEND;
bool use_audio_activity = true;
bool use_video_activity = true;
double ac_brightness_percent = -1.0;
double battery_brightness_percent = -1.0;
bool allow_wake_locks = true;
bool allow_screen_wake_locks = true;
bool enable_auto_screen_lock = false;
double presentation_screen_dim_delay_factor = 1.0;
double user_activity_screen_dim_delay_factor = 1.0;
bool wait_for_initial_user_activity = false;
bool force_nonzero_brightness_for_user_activity = true;
bool fast_suspend_when_backlights_forced_off = true;
bool peak_shift_enabled = false;
int peak_shift_battery_threshold = -1;
std::vector<PeakShiftDayConfig> peak_shift_day_configs;
bool advanced_battery_charge_mode_enabled = false;
std::vector<AdvancedBatteryChargeModeDayConfig>
advanced_battery_charge_mode_day_configs;
bool boot_on_ac = false;
bool usb_power_share = true;
power_manager::PowerManagementPolicy::BatteryChargeMode::Mode
battery_charge_mode =
power_manager::PowerManagementPolicy::BatteryChargeMode::ADAPTIVE;
int custom_charge_start = -1;
int custom_charge_stop = -1;
// Only set send_feedback_if_undimmed in policy proto if this field is set.
std::optional<bool> send_feedback_if_undimmed;
// Only set adaptive_charging_enabled in policy proto if this field is set.
std::optional<bool> adaptive_charging_enabled;
// Adaptive charging configs, only set when adaptive_charging_enabled.
// Configurable via base::FeatureParam.
double adaptive_charging_min_probability = -1.0;
int adaptive_charging_hold_percent = -1;
double adaptive_charging_max_delay_percentile = -1.0;
int adaptive_charging_min_days_history = -1;
double adaptive_charging_min_full_on_ac_ratio = -1.0;
};
// Converts |base::Value::Dict| to |std::vector<PeakShiftDayConfig>| and
// returns true if there are no missing fields and errors.
static bool GetPeakShiftDayConfigs(
const base::Value::Dict& value,
std::vector<PeakShiftDayConfig>* configs_out);
// Converts |base::Value::Dict| to
// |std::vector<AdvancedBatteryChargeModeDayConfig>| and returns true if there
// are no missing fields and errors.
static bool GetAdvancedBatteryChargeModeDayConfigs(
const base::Value::Dict& value,
std::vector<AdvancedBatteryChargeModeDayConfig>* configs_out);
// Saves appropriate value to |mode_out| and returns true if there is mapping
// between battery charge mode int and enum value.
static bool GetBatteryChargeModeFromInteger(
int mode,
power_manager::PowerManagementPolicy::BatteryChargeMode::Mode* mode_out);
// Returns a string describing |policy|. Useful for comparisons in tests.
static std::string GetPolicyDebugString(
const power_manager::PowerManagementPolicy& policy);
// Returns a string describing |PeakShift| part of |policy|. Useful for
// comparisons in tests.
static std::string GetPeakShiftPolicyDebugString(
const power_manager::PowerManagementPolicy& policy);
// Returns a string describing |AdvancedBatteryChargeMode| part of |policy|.
// Useful for comparisons in tests.
static std::string GetAdvancedBatteryChargeModePolicyDebugString(
const power_manager::PowerManagementPolicy& policy);
// Delay in milliseconds between the screen being turned off and the screen
// being locked. Used if the |enable_auto_screen_lock| pref is set but
// |*_screen_lock_delay_ms| are unset or set to higher values than what this
// constant would imply.
static const int kScreenLockAfterOffDelayMs;
// String added to a PowerManagementPolicy |reason| field if the policy has
// been modified by preferences.
static const char kPrefsReason[];
bool honor_screen_wake_locks_for_test() const {
return honor_screen_wake_locks_;
}
// Updates |prefs_policy_| with |values| and sends an updated policy.
void ApplyPrefs(const PrefValues& values);
// Registers a request to temporarily prevent the screen from getting dimmed
// or turned off or the system from suspending in response to user inactivity
// and sends an updated policy. |description| is a human-readable description
// of the reason the lock was created. Returns a unique ID that can be passed
// to RemoveWakeLock() later.
// See the comment above WakeLock::Type for descriptions of the lock types.
int AddScreenWakeLock(WakeLockReason reason, const std::string& description);
int AddDimWakeLock(WakeLockReason reason, const std::string& description);
int AddSystemWakeLock(WakeLockReason reason, const std::string& description);
// Unregisters a request previously created via an Add*WakeLock() call
// and sends an updated policy.
void RemoveWakeLock(int id);
// Adjusts policy while Chrome is exiting. The lid-closed action
// is overridden to ensure that the system doesn't suspend or shut
// down.
void NotifyChromeIsExiting();
// Adjusts policy when the display is forced off in response to the
// user tapping the power button, or when it's no longer forced off.
void HandleBacklightsForcedOffForPowerButton(bool forced_off);
// Adjusts policy when the migration of a user homedir to a new
// encryption format starts or stops. While migration is active,
// the lid-closed action is overridden to ensure the system
// doesn't shut down.
void SetEncryptionMigrationActive(bool active);
// PowerManagerClient::Observer implementation:
void PowerManagerRestarted() override;
void ScreenBrightnessChanged(
const power_manager::BacklightBrightnessChange& change) override;
// Returns the maximum time set by policy for the screen to lock when idle
// between AC and battery power sources. Returns zero if screen autolock is
// disabled by policy.
// If delay is set to zero, Google Chrome OS does not lock the screen when
// the user becomes idle.
// Note: The actual screen lock delay on the OS may differ as it takes other
// factors into account, like wake locks.
base::TimeDelta GetMaxPolicyAutoScreenLockDelay();
private:
explicit PowerPolicyController(PowerManagerClient* client);
~PowerPolicyController() override;
// Details about a wake lock added via Add*WakeLock().
// SCREEN and DIM will keep the screen on and prevent it from locking.
// SCREEN will also prevent it from dimming. SYSTEM will prevent idle
// suspends, but the screen will turn off and lock normally.
struct WakeLock {
// TODO(derat): Remove this enum in favor of device::mojom::WakeLockType
// once this class has been moved to the device service:
// https://crbug.com/702449
enum Type {
TYPE_SCREEN,
TYPE_DIM,
TYPE_SYSTEM,
};
WakeLock(Type type, WakeLockReason reason, const std::string& description);
~WakeLock();
const Type type;
const WakeLockReason reason;
const std::string description;
};
using WakeLockMap = std::map<int, WakeLock>;
// Helper method for AddScreenWakeLock() and AddSystemWakeLock().
int AddWakeLockInternal(WakeLock::Type type,
WakeLockReason reason,
const std::string& description);
// Sends a policy based on |prefs_policy_| to the power manager.
void SendCurrentPolicy();
raw_ptr<PowerManagerClient> client_; // weak
// Policy derived from values passed to ApplyPrefs().
power_manager::PowerManagementPolicy prefs_policy_;
// Was ApplyPrefs() called?
bool prefs_were_set_ = false;
// Maps from an ID representing a request to prevent the screen from
// getting dimmed or turned off or to prevent the system from suspending
// to details about the request.
WakeLockMap wake_locks_;
// Should |wake_locks_| be honored?
bool honor_wake_locks_ = true;
// If wake locks are honored, should TYPE_SCREEN or TYPE_DIM entries in
// |wake_locks_| be honored?
// If false, screen wake locks are just treated as TYPE_SYSTEM instead.
bool honor_screen_wake_locks_ = true;
// Next ID to be used by an Add*WakeLock() request.
int next_wake_lock_id_ = 1;
// True if Chrome is in the process of exiting.
bool chrome_is_exiting_ = false;
// True if the screen is currently forced off due to the user having tapped
// the power button.
bool backlights_forced_off_for_power_button_ = false;
// True if suspend delays should be shortened when
// |backlights_forced_off_for_power_button_| is true. Set from prefs.
bool fast_suspend_when_backlights_forced_off_ = true;
// True if a user homedir is in the process of migrating encryption formats.
bool encryption_migration_active_ = false;
// Whether brightness policy value was overridden by a user adjustment in the
// current user session.
bool per_session_brightness_override_ = false;
// Indicates if screen autolock is enabled or not by policy.
bool auto_screen_lock_enabled_ = false;
};
} // namespace chromeos
#endif // CHROMEOS_DBUS_POWER_POWER_POLICY_CONTROLLER_H_