chromium/chromeos/dbus/power/power_policy_controller.h

// 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_