2018-11-10 20:38:24 +00:00
|
|
|
/*
|
2020-01-24 10:10:40 +00:00
|
|
|
* Copyright (c) 2018-2020 Atmosphère-NX
|
2018-11-10 20:38:24 +00: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/>.
|
|
|
|
*/
|
|
|
|
#include "fatal_config.hpp"
|
|
|
|
|
2019-10-24 10:30:10 +01:00
|
|
|
namespace ams::fatal::srv {
|
2018-11-10 20:38:24 +00:00
|
|
|
|
2019-07-19 03:09:35 +01:00
|
|
|
namespace {
|
2018-11-10 20:38:24 +00:00
|
|
|
|
2019-07-19 03:09:35 +01:00
|
|
|
/* Global config. */
|
|
|
|
FatalConfig g_config;
|
2018-11-10 20:38:24 +00:00
|
|
|
|
2019-07-19 03:09:35 +01:00
|
|
|
/* Event creator. */
|
2019-09-28 02:04:58 +01:00
|
|
|
Handle GetFatalDirtyEventReadableHandle() {
|
2019-07-19 03:09:35 +01:00
|
|
|
Event evt;
|
2020-03-18 07:14:26 +00:00
|
|
|
R_ABORT_UNLESS(setsysAcquireFatalDirtyFlagEventHandle(&evt));
|
2019-09-28 02:04:58 +01:00
|
|
|
return evt.revent;
|
2018-11-10 20:38:24 +00:00
|
|
|
}
|
2019-06-18 00:41:03 +01:00
|
|
|
|
2019-07-19 03:09:35 +01:00
|
|
|
/* Global event. */
|
2020-04-08 10:21:35 +01:00
|
|
|
os::SystemEventType g_fatal_dirty_event;
|
|
|
|
os::WaitableHolderType g_fatal_dirty_waitable_holder;
|
|
|
|
bool g_initialized;
|
2019-09-28 02:04:58 +01:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-04-08 10:21:35 +01:00
|
|
|
os::WaitableHolderType *GetFatalDirtyWaitableHolder() {
|
|
|
|
if (AMS_UNLIKELY(!g_initialized)) {
|
|
|
|
os::AttachReadableHandleToSystemEvent(std::addressof(g_fatal_dirty_event), GetFatalDirtyEventReadableHandle(), true, os::EventClearMode_ManualClear);
|
|
|
|
os::InitializeWaitableHolder(std::addressof(g_fatal_dirty_waitable_holder), std::addressof(g_fatal_dirty_event));
|
|
|
|
os::SetWaitableHolderUserData(std::addressof(g_fatal_dirty_waitable_holder), reinterpret_cast<uintptr_t>(std::addressof(g_fatal_dirty_waitable_holder)));
|
|
|
|
g_initialized = true;
|
|
|
|
}
|
|
|
|
return std::addressof(g_fatal_dirty_waitable_holder);
|
2019-09-28 02:04:58 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void OnFatalDirtyEvent() {
|
2020-04-08 10:21:35 +01:00
|
|
|
os::ClearSystemEvent(std::addressof(g_fatal_dirty_event));
|
2019-09-28 02:04:58 +01:00
|
|
|
|
|
|
|
u64 flags_0, flags_1;
|
|
|
|
if (R_SUCCEEDED(setsysGetFatalDirtyFlags(&flags_0, &flags_1)) && (flags_0 & 1)) {
|
|
|
|
g_config.UpdateLanguageCode();
|
|
|
|
}
|
2018-11-14 01:53:26 +00:00
|
|
|
}
|
2019-06-18 00:41:03 +01:00
|
|
|
|
2019-07-19 03:09:35 +01:00
|
|
|
FatalConfig::FatalConfig() {
|
|
|
|
/* Clear this. */
|
|
|
|
std::memset(this, 0, sizeof(*this));
|
2018-11-14 01:53:26 +00:00
|
|
|
|
2019-07-19 03:09:35 +01:00
|
|
|
/* Get information from set. */
|
|
|
|
setsysGetSerialNumber(this->serial_number);
|
|
|
|
setsysGetFirmwareVersion(&this->firmware_version);
|
2019-10-18 18:37:03 +01:00
|
|
|
setsysGetQuestFlag(&this->quest_flag);
|
2019-07-19 03:09:35 +01:00
|
|
|
this->UpdateLanguageCode();
|
2019-06-18 00:41:03 +01:00
|
|
|
|
2019-07-19 03:09:35 +01:00
|
|
|
/* Read information from settings. */
|
2019-11-27 04:58:39 +00:00
|
|
|
settings::fwdbg::GetSettingsItemValue(&this->transition_to_fatal, sizeof(this->transition_to_fatal), "fatal", "transition_to_fatal");
|
|
|
|
settings::fwdbg::GetSettingsItemValue(&this->show_extra_info, sizeof(this->show_extra_info), "fatal", "show_extra_info");
|
|
|
|
settings::fwdbg::GetSettingsItemValue(&this->quest_reboot_interval_second, sizeof(this->quest_reboot_interval_second), "fatal", "quest_reboot_interval_second");
|
2019-06-18 00:41:03 +01:00
|
|
|
|
2019-07-19 03:09:35 +01:00
|
|
|
/* Atmosphere extension for automatic reboot. */
|
2019-11-27 04:58:39 +00:00
|
|
|
if (settings::fwdbg::GetSettingsItemValue(&this->fatal_auto_reboot_interval, sizeof(this->fatal_auto_reboot_interval), "atmosphere", "fatal_auto_reboot_interval") == sizeof(this->fatal_auto_reboot_interval)) {
|
2019-07-19 03:09:35 +01:00
|
|
|
this->fatal_auto_reboot_enabled = this->fatal_auto_reboot_interval != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Setup messages. */
|
|
|
|
{
|
|
|
|
this->error_msg = u8"Error Code: 2%03d-%04d (0x%x)\n";
|
|
|
|
|
|
|
|
this->error_desc = u8"An error has occured.\n\n"
|
|
|
|
u8"Please press the POWER Button to restart the console normally, or a VOL button\n"
|
|
|
|
u8"to reboot to a payload (or RCM, if none is present). If you are unable to\n"
|
|
|
|
u8"restart the console, hold the POWER Button for 12 seconds to turn the console off.\n\n"
|
|
|
|
u8"If the problem persists, refer to the Nintendo Support Website.\n"
|
|
|
|
u8"support.nintendo.com/switch/error\n";
|
|
|
|
|
|
|
|
/* If you're running Atmosphere on a quest unit for some reason, talk to me on discord. */
|
|
|
|
this->quest_desc = u8"Please call 1-800-875-1852 for service.\n\n"
|
|
|
|
u8"Also, please be aware that running Atmosphere on a Quest device is not fully\n"
|
|
|
|
u8"supported. Perhaps try booting your device without Atmosphere before calling\n"
|
|
|
|
u8"an official Nintendo service hotline. If you encounter further issues, please\n"
|
|
|
|
u8"contact SciresM#0524 on Discord, or via some other means.\n";
|
|
|
|
|
|
|
|
/* TODO: Try to load dynamically? */
|
|
|
|
/* FsStorage message_storage; */
|
|
|
|
/* TODO: if (R_SUCCEEDED(fsOpenDataStorageByDataId(0x010000000000081D, "fatal_msg"))) { ... } */
|
|
|
|
}
|
|
|
|
}
|
2019-06-18 00:41:03 +01:00
|
|
|
|
2019-07-19 03:09:35 +01:00
|
|
|
const FatalConfig &GetFatalConfig() {
|
|
|
|
return g_config;
|
|
|
|
}
|
2019-04-22 08:18:01 +01:00
|
|
|
|
2018-11-10 20:38:24 +00:00
|
|
|
}
|