mirror of
https://github.com/Atmosphere-NX/Atmosphere-libs.git
synced 2025-08-16 11:29:23 +02:00
191 lines
8.1 KiB
C++
191 lines
8.1 KiB
C++
/*
|
|
* Copyright (c) Atmosphère-NX
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms and conditions of the GNU General Public License,
|
|
* version 2, as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
* more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
#pragma once
|
|
#include <vapours.hpp>
|
|
#include <stratosphere/powctl/powctl_types.hpp>
|
|
#include <stratosphere/powctl/driver/impl/powctl_select_charger_parameters.hpp>
|
|
|
|
namespace ams::powctl::driver::impl {
|
|
|
|
#if defined(ATMOSPHERE_BOARD_NINTENDO_NX)
|
|
class ChargeArbiter {
|
|
private:
|
|
const ChargeParametersRule *m_rules;
|
|
size_t m_num_rules;
|
|
int m_charge_voltage_limit;
|
|
BatteryTemperatureLevel m_temperature_level;
|
|
int m_avg_v_cell;
|
|
float m_voltage_fuel_gauge_percentage;
|
|
bool m_has_battery_done_current;
|
|
int m_battery_done_current;
|
|
PowerState m_power_state;
|
|
const ChargeParametersRule *m_selected_rule;
|
|
bool m_check_battery_done_current;
|
|
private:
|
|
static constexpr bool IsInRange(int value, int min, int max) {
|
|
if (!(min <= value)) {
|
|
return false;
|
|
}
|
|
|
|
if (max == std::numeric_limits<int>::max()) {
|
|
return value <= max;
|
|
} else {
|
|
return value < max;
|
|
}
|
|
}
|
|
static constexpr bool IsInRangeFloat(float value, float min, float max) {
|
|
if (!(min <= value)) {
|
|
return false;
|
|
}
|
|
|
|
if (max == std::numeric_limits<float>::max()) {
|
|
return value <= max;
|
|
} else {
|
|
return value < max;
|
|
}
|
|
}
|
|
|
|
bool IsAcceptablePowerState(const PowerState *acceptable, size_t num_acceptable) const {
|
|
for (size_t i = 0; i < num_acceptable; ++i) {
|
|
if (m_power_state == acceptable[i]) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
public:
|
|
ChargeArbiter(const ChargeParametersRule *r, size_t nr, int cvl)
|
|
: m_rules(r), m_num_rules(nr), m_charge_voltage_limit(cvl), m_temperature_level(BatteryTemperatureLevel::Medium),
|
|
m_avg_v_cell(4080), m_voltage_fuel_gauge_percentage(25.0), m_has_battery_done_current(false), m_battery_done_current(0),
|
|
m_power_state(PowerState::FullAwake), m_selected_rule(nullptr), m_check_battery_done_current(false)
|
|
{
|
|
this->UpdateSelectedRule();
|
|
}
|
|
|
|
void SetBatteryTemperatureLevel(BatteryTemperatureLevel btl) {
|
|
m_temperature_level = btl;
|
|
this->UpdateSelectedRule();
|
|
}
|
|
|
|
void SetBatteryAverageVCell(int avg) {
|
|
m_avg_v_cell = avg;
|
|
this->UpdateSelectedRule();
|
|
}
|
|
|
|
void SetBatteryVoltageFuelGaugePercentage(float pct) {
|
|
m_voltage_fuel_gauge_percentage = pct;
|
|
this->UpdateSelectedRule();
|
|
}
|
|
|
|
void SetBatteryDoneCurrent(int current) {
|
|
m_battery_done_current = current;
|
|
m_has_battery_done_current = true;
|
|
this->UpdateSelectedRule();
|
|
}
|
|
|
|
void SetPowerState(PowerState ps) {
|
|
m_power_state = ps;
|
|
this->UpdateSelectedRule();
|
|
}
|
|
|
|
int GetChargeVoltageLimit() const {
|
|
return m_charge_voltage_limit;
|
|
}
|
|
|
|
bool IsBatteryDoneCurrentAcceptable(int current) const {
|
|
const auto *rule = this->GetSelectedRule();
|
|
AMS_ASSERT(rule != nullptr);
|
|
|
|
return IsInRange(current, rule->min_battery_done_current, rule->max_battery_done_current);
|
|
}
|
|
|
|
const ChargeParametersRule *GetSelectedRule() const {
|
|
return m_selected_rule;
|
|
}
|
|
|
|
void UpdateSelectedRule() {
|
|
/* Try to find an entry that fits our current requirements. */
|
|
const ChargeParametersRule *best_rule = nullptr;
|
|
|
|
for (size_t i = 0; i < m_num_rules; ++i) {
|
|
/* Get the current rule. */
|
|
const ChargeParametersRule &cur_rule = m_rules[i];
|
|
|
|
/* Check the temperature level. */
|
|
if (m_temperature_level != cur_rule.temperature_level) {
|
|
continue;
|
|
}
|
|
|
|
/* Check that average voltage is in range. */
|
|
if (!IsInRange(m_avg_v_cell, cur_rule.min_avg_v_cell, cur_rule.max_avg_v_cell)) {
|
|
continue;
|
|
}
|
|
|
|
/* Check that voltage fuel gauge percentage is in range. */
|
|
if (!IsInRangeFloat(m_voltage_fuel_gauge_percentage, cur_rule.min_voltage_fuel_gauge_percentage, cur_rule.max_voltage_fuel_gauge_percentage)) {
|
|
continue;
|
|
}
|
|
|
|
/* Check if our power state is acceptable. */
|
|
if (!this->IsAcceptablePowerState(cur_rule.acceptable_power_states, cur_rule.num_acceptable_power_states)) {
|
|
continue;
|
|
}
|
|
|
|
/* The limit is probably acceptable. */
|
|
if (m_selected_rule != std::addressof(cur_rule)) {
|
|
/* We're selecting a new rule. Check if our need to deal with battery current is acceptable. */
|
|
if (cur_rule.check_battery_current && m_check_battery_done_current) {
|
|
continue;
|
|
}
|
|
|
|
/* Determine whether we should check battery done current. */
|
|
bool check_battery_done_current;
|
|
|
|
if (m_selected_rule != nullptr && m_selected_rule->check_battery_current) {
|
|
if (m_selected_rule->temperature_level == m_temperature_level &&
|
|
IsInRange(m_avg_v_cell, m_selected_rule->min_avg_v_cell, m_selected_rule->max_avg_v_cell) &&
|
|
IsInRangeFloat(m_voltage_fuel_gauge_percentage, m_selected_rule->min_voltage_fuel_gauge_percentage, m_selected_rule->max_voltage_fuel_gauge_percentage))
|
|
{
|
|
check_battery_done_current = m_has_battery_done_current && !IsInRange(m_battery_done_current, m_selected_rule->min_battery_done_current, m_selected_rule->max_battery_done_current);
|
|
} else {
|
|
check_battery_done_current = true;
|
|
}
|
|
} else {
|
|
check_battery_done_current = false;
|
|
}
|
|
|
|
/* Set whether we need to check the battery done current. */
|
|
m_has_battery_done_current = false;
|
|
m_check_battery_done_current |= check_battery_done_current;
|
|
} else {
|
|
/* We're selecting the currently selected rule. Make sure the battery done current is acceptable if we have one. */
|
|
if (m_has_battery_done_current && !IsInRange(m_battery_done_current, cur_rule.min_battery_done_current, cur_rule.max_battery_done_current)) {
|
|
continue;
|
|
}
|
|
}
|
|
|
|
/* Select the current rule. */
|
|
best_rule = std::addressof(cur_rule);
|
|
break;
|
|
}
|
|
|
|
/* Update our selected rule. */
|
|
m_selected_rule = best_rule;
|
|
}
|
|
};
|
|
#endif
|
|
|
|
} |