2019-05-03 03:33:12 +01:00
|
|
|
/*
|
2020-01-24 10:10:40 +00:00
|
|
|
* Copyright (c) 2018-2020 Atmosphère-NX
|
2019-05-03 03:33:12 +01:00
|
|
|
*
|
|
|
|
* 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/>.
|
|
|
|
*/
|
2020-05-11 23:02:10 +01:00
|
|
|
#include <stratosphere.hpp>
|
2019-06-22 08:10:21 +01:00
|
|
|
#include "boot_i2c_utils.hpp"
|
2019-05-03 03:33:12 +01:00
|
|
|
#include "boot_pmic_driver.hpp"
|
|
|
|
|
2019-10-24 10:30:10 +01:00
|
|
|
namespace ams::boot {
|
2019-05-03 03:33:12 +01:00
|
|
|
|
2019-06-22 08:10:21 +01:00
|
|
|
void PmicDriver::ShutdownSystem() {
|
2020-02-23 07:05:14 +00:00
|
|
|
R_ABORT_UNLESS(this->ShutdownSystem(false));
|
2019-05-03 03:33:12 +01:00
|
|
|
}
|
|
|
|
|
2019-06-22 08:10:21 +01:00
|
|
|
void PmicDriver::RebootSystem() {
|
2020-02-23 07:05:14 +00:00
|
|
|
R_ABORT_UNLESS(this->ShutdownSystem(true));
|
2019-05-09 10:45:31 +01:00
|
|
|
}
|
|
|
|
|
2019-06-22 08:10:21 +01:00
|
|
|
Result PmicDriver::GetAcOk(bool *out) {
|
|
|
|
u8 power_status;
|
|
|
|
R_TRY(this->GetPowerStatus(&power_status));
|
|
|
|
*out = (power_status & 0x02) != 0;
|
2019-10-24 09:40:44 +01:00
|
|
|
return ResultSuccess();
|
2019-05-09 10:45:31 +01:00
|
|
|
}
|
|
|
|
|
2019-06-22 08:10:21 +01:00
|
|
|
Result PmicDriver::GetPowerIntr(u8 *out) {
|
|
|
|
const u8 addr = 0x0B;
|
|
|
|
return ReadI2cRegister(this->i2c_session, out, sizeof(*out), &addr, sizeof(addr));
|
2019-05-09 10:45:31 +01:00
|
|
|
}
|
|
|
|
|
2019-06-22 08:10:21 +01:00
|
|
|
Result PmicDriver::GetPowerStatus(u8 *out) {
|
|
|
|
const u8 addr = 0x15;
|
|
|
|
return ReadI2cRegister(this->i2c_session, out, sizeof(*out), &addr, sizeof(addr));
|
2019-05-09 10:45:31 +01:00
|
|
|
}
|
|
|
|
|
2019-06-22 08:10:21 +01:00
|
|
|
Result PmicDriver::GetNvErc(u8 *out) {
|
|
|
|
const u8 addr = 0x0C;
|
|
|
|
return ReadI2cRegister(this->i2c_session, out, sizeof(*out), &addr, sizeof(addr));
|
|
|
|
}
|
2019-05-09 10:45:31 +01:00
|
|
|
|
2019-06-22 08:10:21 +01:00
|
|
|
Result PmicDriver::GetPowerButtonPressed(bool *out) {
|
|
|
|
u8 power_intr;
|
|
|
|
R_TRY(this->GetPowerIntr(&power_intr));
|
|
|
|
*out = (power_intr & 0x08) != 0;
|
2019-10-24 09:40:44 +01:00
|
|
|
return ResultSuccess();
|
2019-05-09 10:45:31 +01:00
|
|
|
}
|
|
|
|
|
2019-06-22 08:10:21 +01:00
|
|
|
Result PmicDriver::ShutdownSystem(bool reboot) {
|
|
|
|
const u8 on_off_1_addr = 0x41;
|
|
|
|
const u8 on_off_2_addr = 0x42;
|
|
|
|
|
|
|
|
/* Get value, set or clear software reset mask. */
|
|
|
|
u8 on_off_2_val = 0;
|
2020-02-23 07:05:14 +00:00
|
|
|
R_ABORT_UNLESS(ReadI2cRegister(this->i2c_session, &on_off_2_val, sizeof(on_off_2_val), &on_off_2_addr, sizeof(on_off_2_addr)));
|
2019-06-22 08:10:21 +01:00
|
|
|
if (reboot) {
|
|
|
|
on_off_2_val |= 0x80;
|
|
|
|
} else {
|
|
|
|
on_off_2_val &= ~0x80;
|
|
|
|
}
|
2020-02-23 07:05:14 +00:00
|
|
|
R_ABORT_UNLESS(WriteI2cRegister(this->i2c_session, &on_off_2_val, sizeof(on_off_2_val), &on_off_2_addr, sizeof(on_off_2_addr)));
|
2019-06-22 08:10:21 +01:00
|
|
|
|
|
|
|
/* Get value, set software reset mask. */
|
|
|
|
u8 on_off_1_val = 0;
|
2020-02-23 07:05:14 +00:00
|
|
|
R_ABORT_UNLESS(ReadI2cRegister(this->i2c_session, &on_off_1_val, sizeof(on_off_1_val), &on_off_1_addr, sizeof(on_off_1_addr)));
|
2019-06-22 08:10:21 +01:00
|
|
|
on_off_1_val |= 0x80;
|
|
|
|
|
2020-02-24 16:21:31 +00:00
|
|
|
/* Finalize the battery on non-Calcio. */
|
|
|
|
if (spl::GetHardwareType() != spl::HardwareType::Calcio) {
|
2019-06-22 08:10:21 +01:00
|
|
|
BatteryDriver battery_driver;
|
|
|
|
this->FinalizeBattery(&battery_driver);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Actually write the value to trigger shutdown/reset. */
|
2020-02-23 07:05:14 +00:00
|
|
|
R_ABORT_UNLESS(WriteI2cRegister(this->i2c_session, &on_off_1_val, sizeof(on_off_1_val), &on_off_1_addr, sizeof(on_off_1_addr)));
|
2019-06-22 08:10:21 +01:00
|
|
|
|
|
|
|
/* Allow up to 5 seconds for shutdown/reboot to take place. */
|
|
|
|
svcSleepThread(5'000'000'000ul);
|
2020-02-23 07:05:14 +00:00
|
|
|
AMS_ABORT_UNLESS(false);
|
2019-05-09 10:45:31 +01:00
|
|
|
}
|
|
|
|
|
2019-06-22 08:10:21 +01:00
|
|
|
void PmicDriver::FinalizeBattery(BatteryDriver *battery_driver) {
|
|
|
|
/* Set shutdown timer. */
|
|
|
|
battery_driver->SetShutdownTimer();
|
|
|
|
|
|
|
|
/* Get whether shutdown is enabled. */
|
|
|
|
bool shutdown_enabled;
|
|
|
|
if (R_FAILED(battery_driver->GetShutdownEnabled(&shutdown_enabled))) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-02-24 16:21:31 +00:00
|
|
|
/* On Hoag, we don't want to use the desired shutdown value when battery charged. */
|
|
|
|
bool use_desired_shutdown = true;
|
|
|
|
if (spl::GetHardwareType() == spl::HardwareType::Hoag) {
|
|
|
|
double battery_charge;
|
|
|
|
if (R_FAILED(battery_driver->GetSocRep(&battery_charge)) || battery_charge >= 80.0) {
|
|
|
|
use_desired_shutdown = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-22 08:10:21 +01:00
|
|
|
bool ac_ok;
|
|
|
|
bool desired_shutdown_enabled;
|
|
|
|
if (R_FAILED(this->GetAcOk(&ac_ok)) || ac_ok) {
|
|
|
|
desired_shutdown_enabled = false;
|
|
|
|
} else {
|
|
|
|
desired_shutdown_enabled = true;
|
|
|
|
}
|
|
|
|
|
2020-02-24 16:21:31 +00:00
|
|
|
desired_shutdown_enabled &= use_desired_shutdown;
|
|
|
|
|
2019-06-22 08:10:21 +01:00
|
|
|
if (shutdown_enabled != desired_shutdown_enabled) {
|
|
|
|
battery_driver->SetShutdownEnabled(desired_shutdown_enabled);
|
|
|
|
}
|
2019-05-09 10:45:31 +01:00
|
|
|
}
|
2019-06-22 08:10:21 +01:00
|
|
|
|
2019-05-09 10:45:31 +01:00
|
|
|
}
|