2018-12-01 18:46:59 +00:00
|
|
|
/*
|
2020-03-14 07:24:24 +00:00
|
|
|
* Copyright (c) 2018-2020 CTCaer
|
2019-04-14 01:24:37 +00:00
|
|
|
* Copyright (c) 2019 Atmosphère-NX
|
2018-12-01 18:46:59 +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 <string.h>
|
2019-04-14 01:24:37 +00:00
|
|
|
#include <stdlib.h>
|
2018-12-01 18:46:59 +00:00
|
|
|
|
|
|
|
#include "hos.h"
|
2020-06-14 13:45:45 +00:00
|
|
|
#include "../config.h"
|
|
|
|
#include <gfx/di.h>
|
|
|
|
#include <gfx_utils.h>
|
|
|
|
#include <libs/fatfs/ff.h>
|
|
|
|
#include <mem/heap.h>
|
|
|
|
#include <soc/fuse.h>
|
2019-06-30 00:24:58 +00:00
|
|
|
#include "../storage/emummc.h"
|
2020-06-14 10:49:59 +00:00
|
|
|
#include "../storage/nx_emmc.h"
|
2020-06-14 13:45:45 +00:00
|
|
|
#include <storage/nx_sd.h>
|
|
|
|
#include <storage/sdmmc.h>
|
|
|
|
#include <utils/btn.h>
|
|
|
|
#include <utils/util.h>
|
|
|
|
#include <utils/types.h>
|
2018-12-01 18:46:59 +00:00
|
|
|
|
2019-06-30 00:24:58 +00:00
|
|
|
extern hekate_config h_cfg;
|
|
|
|
|
|
|
|
enum emuMMC_Type
|
|
|
|
{
|
|
|
|
emuMMC_None = 0,
|
|
|
|
emuMMC_Partition,
|
|
|
|
emuMMC_File,
|
|
|
|
emuMMC_MAX
|
|
|
|
};
|
|
|
|
|
|
|
|
/* "EFS0" */
|
|
|
|
#define EMUMMC_MAGIC 0x30534645
|
|
|
|
#define EMUMMC_FILE_PATH_MAX 0x80
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
u32 magic;
|
|
|
|
u32 type;
|
|
|
|
u32 id;
|
|
|
|
u32 fs_ver;
|
|
|
|
} emummc_base_config_t;
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
u64 start_sector;
|
|
|
|
} emummc_partition_config_t;
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
char path[EMUMMC_FILE_PATH_MAX];
|
|
|
|
} emummc_file_config_t;
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
emummc_base_config_t base_cfg;
|
|
|
|
union
|
|
|
|
{
|
|
|
|
emummc_partition_config_t partition_cfg;
|
|
|
|
emummc_file_config_t file_cfg;
|
|
|
|
};
|
|
|
|
char nintendo_path[EMUMMC_FILE_PATH_MAX];
|
|
|
|
} exo_emummc_config_t;
|
|
|
|
|
2019-02-24 00:34:04 +00:00
|
|
|
typedef struct _exo_cfg_t
|
|
|
|
{
|
2019-06-30 00:24:58 +00:00
|
|
|
u32 magic;
|
|
|
|
u32 fwno;
|
|
|
|
u32 flags;
|
|
|
|
u32 reserved[5];
|
|
|
|
exo_emummc_config_t emummc_cfg;
|
2019-02-24 00:34:04 +00:00
|
|
|
} exo_cfg_t;
|
|
|
|
|
|
|
|
typedef struct _atm_meta_t
|
|
|
|
{
|
2019-06-30 00:24:58 +00:00
|
|
|
u32 magic;
|
|
|
|
u32 fwno;
|
2019-02-24 00:34:04 +00:00
|
|
|
} wb_cfg_t;
|
|
|
|
|
2019-04-14 01:24:37 +00:00
|
|
|
// Atmosphère reboot-to-fatal-error.
|
|
|
|
typedef struct _atm_fatal_error_ctx
|
|
|
|
{
|
|
|
|
u32 magic;
|
|
|
|
u32 error_desc;
|
|
|
|
u64 title_id;
|
|
|
|
union
|
|
|
|
{
|
|
|
|
u64 gprs[32];
|
|
|
|
struct
|
|
|
|
{
|
|
|
|
u64 _gprs[29];
|
|
|
|
u64 fp;
|
|
|
|
u64 lr;
|
|
|
|
u64 sp;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
u64 pc;
|
2019-06-30 00:55:52 +00:00
|
|
|
u64 module_base;
|
2019-04-14 01:24:37 +00:00
|
|
|
u32 pstate;
|
|
|
|
u32 afsr0;
|
|
|
|
u32 afsr1;
|
|
|
|
u32 esr;
|
|
|
|
u64 far;
|
|
|
|
u64 report_identifier; // Normally just system tick.
|
2019-06-30 00:55:52 +00:00
|
|
|
u64 stack_trace_size;
|
|
|
|
u64 stack_dump_size;
|
|
|
|
u64 stack_trace[0x20];
|
|
|
|
u8 stack_dump[0x100];
|
2020-03-09 07:09:25 +00:00
|
|
|
u8 tls[0x100];
|
2019-04-14 01:24:37 +00:00
|
|
|
} atm_fatal_error_ctx;
|
|
|
|
|
|
|
|
#define ATM_FATAL_ERR_CTX_ADDR 0x4003E000
|
2020-03-09 07:09:25 +00:00
|
|
|
#define ATM_FATAL_MAGIC 0x30454641 // AFE0
|
2019-04-14 01:24:37 +00:00
|
|
|
|
2019-02-24 00:34:04 +00:00
|
|
|
#define ATM_WB_HEADER_OFF 0x244
|
|
|
|
#define ATM_WB_MAGIC 0x30544257
|
|
|
|
|
2018-12-01 18:46:59 +00:00
|
|
|
// Exosphère mailbox defines.
|
2019-02-24 00:34:04 +00:00
|
|
|
#define EXO_CFG_ADDR 0x8000F000
|
2020-04-30 13:21:38 +00:00
|
|
|
#define EXO_MAGIC_VAL 0x304F5845
|
|
|
|
#define EXO_FLAG_DBG_PRIV (1 << 1)
|
|
|
|
#define EXO_FLAG_DBG_USER (1 << 2)
|
|
|
|
#define EXO_FLAG_NO_USER_EXC (1 << 3)
|
|
|
|
#define EXO_FLAG_USER_PMU (1 << 4)
|
|
|
|
#define EXO_FLAG_CAL0_BLANKING (1 << 5)
|
|
|
|
#define EXO_FLAG_CAL0_WRITES_SYS (1 << 6)
|
2019-02-24 00:34:04 +00:00
|
|
|
|
2020-06-14 10:49:03 +00:00
|
|
|
#define EXO_FW_VER(mj, mn, rv) (((mj) << 24) | ((mn) << 16) | ((rv) << 8))
|
|
|
|
|
2020-06-15 07:57:15 +00:00
|
|
|
void config_exosphere(launch_ctxt_t *ctxt, u32 warmboot_base, bool exo_new)
|
2018-12-01 18:46:59 +00:00
|
|
|
{
|
|
|
|
u32 exoFwNo = 0;
|
|
|
|
u32 exoFlags = 0;
|
2019-09-12 20:41:08 +00:00
|
|
|
u32 kb = ctxt->pkg1_id->kb;
|
2020-04-30 13:21:38 +00:00
|
|
|
bool user_debug = false;
|
|
|
|
bool cal0_blanking = false;
|
|
|
|
bool cal0_allow_writes_sys = false;
|
2018-12-01 18:46:59 +00:00
|
|
|
|
2019-06-30 00:40:37 +00:00
|
|
|
memset((exo_cfg_t *)EXO_CFG_ADDR, 0, sizeof(exo_cfg_t));
|
|
|
|
|
2019-02-24 00:34:04 +00:00
|
|
|
volatile exo_cfg_t *exo_cfg = (exo_cfg_t *)EXO_CFG_ADDR;
|
2018-12-01 18:46:59 +00:00
|
|
|
|
2020-06-14 10:49:03 +00:00
|
|
|
// Old exosphere target versioning.
|
2019-09-12 20:41:08 +00:00
|
|
|
switch (kb)
|
2018-12-01 18:46:59 +00:00
|
|
|
{
|
|
|
|
case KB_FIRMWARE_VERSION_100_200:
|
2019-09-09 14:00:13 +00:00
|
|
|
if (!strcmp(ctxt->pkg1_id->id, "20161121183008"))
|
2018-12-01 18:46:59 +00:00
|
|
|
exoFwNo = 1;
|
|
|
|
else
|
|
|
|
exoFwNo = 2;
|
|
|
|
break;
|
|
|
|
case KB_FIRMWARE_VERSION_300:
|
|
|
|
exoFwNo = 3;
|
|
|
|
break;
|
|
|
|
default:
|
2019-09-12 20:41:08 +00:00
|
|
|
exoFwNo = kb + 1;
|
|
|
|
if (!strcmp(ctxt->pkg1_id->id, "20190314172056") || (kb >= KB_FIRMWARE_VERSION_810))
|
2020-04-14 14:43:43 +00:00
|
|
|
exoFwNo++; // ATM_TARGET_FW_800/810/900/910.
|
|
|
|
if (!strcmp(ctxt->pkg1_id->id, "20200303104606"))
|
|
|
|
exoFwNo++; // ATM_TARGET_FW_1000.
|
2018-12-01 18:46:59 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-06-14 10:49:03 +00:00
|
|
|
// New exosphere target versioning.
|
2020-06-15 07:57:15 +00:00
|
|
|
if (exo_new)
|
2020-06-14 10:49:03 +00:00
|
|
|
{
|
|
|
|
// Feed old versioning.
|
|
|
|
switch (exoFwNo)
|
|
|
|
{
|
|
|
|
case 1:
|
|
|
|
case 2:
|
|
|
|
case 3:
|
|
|
|
case 4:
|
|
|
|
case 6:
|
|
|
|
exoFwNo = EXO_FW_VER(exoFwNo, 0, 0);
|
|
|
|
break;
|
|
|
|
case 5:
|
|
|
|
if (!ctxt->exo_ctx.fs_is_510)
|
|
|
|
exoFwNo = EXO_FW_VER(5, 0, 0);
|
|
|
|
else
|
|
|
|
exoFwNo = EXO_FW_VER(5, 1, 0);
|
|
|
|
break;
|
|
|
|
case 7:
|
|
|
|
exoFwNo = EXO_FW_VER(6, 2, 0);
|
|
|
|
break;
|
|
|
|
case 8:
|
|
|
|
exoFwNo = EXO_FW_VER(7, 0, 0);
|
|
|
|
break;
|
|
|
|
case 9:
|
|
|
|
exoFwNo = EXO_FW_VER(8, 0, 0);
|
|
|
|
break;
|
|
|
|
case 10:
|
|
|
|
exoFwNo = EXO_FW_VER(8, 1, 0);
|
|
|
|
break;
|
|
|
|
case 11:
|
|
|
|
exoFwNo = EXO_FW_VER(9, 0, 0);
|
|
|
|
break;
|
|
|
|
case 12:
|
|
|
|
exoFwNo = EXO_FW_VER(9, 1, 0);
|
|
|
|
break;
|
|
|
|
case 13:
|
|
|
|
exoFwNo = EXO_FW_VER(10, 0, 0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parse exosphere.ini.
|
2020-04-30 13:21:38 +00:00
|
|
|
if (!ctxt->stock)
|
|
|
|
{
|
|
|
|
LIST_INIT(ini_sections);
|
|
|
|
if (ini_parse(&ini_sections, "exosphere.ini", false))
|
|
|
|
{
|
|
|
|
LIST_FOREACH_ENTRY(ini_sec_t, ini_sec, &ini_sections, link)
|
|
|
|
{
|
|
|
|
// Only parse exosphere section.
|
|
|
|
if (!(ini_sec->type == INI_CHOICE) || strcmp(ini_sec->name, "exosphere"))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
LIST_FOREACH_ENTRY(ini_kv_t, kv, &ini_sec->kvs, link)
|
|
|
|
{
|
|
|
|
if (!strcmp("debugmode_user", kv->key))
|
|
|
|
user_debug = atoi(kv->val);
|
|
|
|
else if (emu_cfg.enabled && !h_cfg.emummc_force_disable)
|
|
|
|
{
|
|
|
|
if (!strcmp("blank_prodinfo_emummc", kv->key))
|
|
|
|
cal0_blanking = atoi(kv->val);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!strcmp("blank_prodinfo_sysmmc", kv->key))
|
|
|
|
cal0_blanking = atoi(kv->val);
|
|
|
|
else if (!strcmp("allow_writing_to_cal_sysmmc", kv->key))
|
|
|
|
cal0_allow_writes_sys = atoi(kv->val);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-09 18:49:00 +00:00
|
|
|
// To avoid problems, make private debug mode always on if not semi-stock.
|
2019-09-09 14:00:13 +00:00
|
|
|
if (!ctxt->stock || (emu_cfg.enabled && !h_cfg.emummc_force_disable))
|
2019-03-09 18:49:00 +00:00
|
|
|
exoFlags |= EXO_FLAG_DBG_PRIV;
|
2018-12-01 18:46:59 +00:00
|
|
|
|
2020-04-30 13:21:38 +00:00
|
|
|
// Enable user debug.
|
|
|
|
if (user_debug)
|
|
|
|
exoFlags |= EXO_FLAG_DBG_USER;
|
|
|
|
|
2019-09-09 14:00:13 +00:00
|
|
|
// Disable proper failure handling.
|
2020-06-13 22:29:41 +00:00
|
|
|
if (ctxt->exo_ctx.no_user_exceptions)
|
2019-09-09 14:00:13 +00:00
|
|
|
exoFlags |= EXO_FLAG_NO_USER_EXC;
|
|
|
|
|
2019-12-08 01:02:17 +00:00
|
|
|
// Enable user access to PMU.
|
2020-06-13 22:29:41 +00:00
|
|
|
if (ctxt->exo_ctx.user_pmu)
|
2019-12-08 01:02:17 +00:00
|
|
|
exoFlags |= EXO_FLAG_USER_PMU;
|
|
|
|
|
2020-05-05 15:58:53 +00:00
|
|
|
// Enable prodinfo blanking. Check if exo ini value is overridden. If not, check if enabled in exo ini.
|
2020-06-13 22:29:41 +00:00
|
|
|
if ((ctxt->exo_ctx.cal0_blank && *ctxt->exo_ctx.cal0_blank)
|
|
|
|
|| (!ctxt->exo_ctx.cal0_blank && cal0_blanking))
|
2020-04-30 13:21:38 +00:00
|
|
|
exoFlags |= EXO_FLAG_CAL0_BLANKING;
|
|
|
|
|
2020-05-05 15:58:53 +00:00
|
|
|
// Allow prodinfo writes. Check if exo ini value is overridden. If not, check if enabled in exo ini.
|
2020-06-13 22:29:41 +00:00
|
|
|
if ((ctxt->exo_ctx.cal0_allow_writes_sys && *ctxt->exo_ctx.cal0_allow_writes_sys)
|
|
|
|
|| (!ctxt->exo_ctx.cal0_allow_writes_sys && cal0_allow_writes_sys))
|
2020-04-30 13:21:38 +00:00
|
|
|
exoFlags |= EXO_FLAG_CAL0_WRITES_SYS;
|
|
|
|
|
2018-12-01 18:46:59 +00:00
|
|
|
// Set mailbox values.
|
2019-02-24 00:34:04 +00:00
|
|
|
exo_cfg->magic = EXO_MAGIC_VAL;
|
|
|
|
exo_cfg->fwno = exoFwNo;
|
|
|
|
exo_cfg->flags = exoFlags;
|
|
|
|
|
2019-03-07 22:19:04 +00:00
|
|
|
// If warmboot is lp0fw, add in RSA modulus.
|
2020-06-14 11:00:07 +00:00
|
|
|
volatile wb_cfg_t *wb_cfg = (wb_cfg_t *)(warmboot_base + ATM_WB_HEADER_OFF);
|
2019-02-24 00:34:04 +00:00
|
|
|
|
|
|
|
if (wb_cfg->magic == ATM_WB_MAGIC)
|
|
|
|
{
|
|
|
|
wb_cfg->fwno = exoFwNo;
|
|
|
|
|
|
|
|
// Set warmboot binary rsa modulus.
|
|
|
|
u8 *rsa_mod = (u8 *)malloc(512);
|
|
|
|
|
2020-06-14 10:49:59 +00:00
|
|
|
sdmmc_storage_set_mmc_partition(&emmc_storage, EMMC_BOOT0);
|
|
|
|
sdmmc_storage_read(&emmc_storage, 1, 1, rsa_mod);
|
2019-02-24 00:34:04 +00:00
|
|
|
|
2019-03-07 22:19:04 +00:00
|
|
|
// Patch AutoRCM out.
|
|
|
|
if ((fuse_read_odm(4) & 3) != 3)
|
|
|
|
rsa_mod[0x10] = 0xF7;
|
|
|
|
else
|
|
|
|
rsa_mod[0x10] = 0x37;
|
2019-02-24 00:34:04 +00:00
|
|
|
|
2020-06-14 11:00:07 +00:00
|
|
|
memcpy((void *)(warmboot_base + 0x10), rsa_mod + 0x10, 0x100);
|
2019-02-24 00:34:04 +00:00
|
|
|
}
|
2019-06-30 00:24:58 +00:00
|
|
|
|
|
|
|
if (emu_cfg.enabled && !h_cfg.emummc_force_disable)
|
|
|
|
{
|
|
|
|
exo_cfg->emummc_cfg.base_cfg.magic = EMUMMC_MAGIC;
|
|
|
|
exo_cfg->emummc_cfg.base_cfg.type = emu_cfg.sector ? emuMMC_Partition : emuMMC_File;
|
|
|
|
exo_cfg->emummc_cfg.base_cfg.fs_ver = emu_cfg.fs_ver;
|
|
|
|
exo_cfg->emummc_cfg.base_cfg.id = emu_cfg.id;
|
|
|
|
|
|
|
|
if (emu_cfg.sector)
|
|
|
|
exo_cfg->emummc_cfg.partition_cfg.start_sector = emu_cfg.sector;
|
|
|
|
else
|
|
|
|
strcpy((char *)exo_cfg->emummc_cfg.file_cfg.path, emu_cfg.path);
|
|
|
|
|
2019-09-09 14:00:13 +00:00
|
|
|
if (emu_cfg.nintendo_path && !ctxt->stock)
|
2019-06-30 00:24:58 +00:00
|
|
|
strcpy((char *)exo_cfg->emummc_cfg.nintendo_path, emu_cfg.nintendo_path);
|
2019-09-09 14:00:13 +00:00
|
|
|
else if (ctxt->stock)
|
2019-07-06 19:08:37 +00:00
|
|
|
strcpy((char *)exo_cfg->emummc_cfg.nintendo_path, "Nintendo");
|
2019-06-30 00:24:58 +00:00
|
|
|
else
|
|
|
|
exo_cfg->emummc_cfg.nintendo_path[0] = 0;
|
|
|
|
}
|
2018-12-01 18:46:59 +00:00
|
|
|
}
|
2019-04-14 01:24:37 +00:00
|
|
|
|
|
|
|
static const char *get_error_desc(u32 error_desc)
|
|
|
|
{
|
|
|
|
switch (error_desc)
|
|
|
|
{
|
|
|
|
case 0x100:
|
2020-07-18 09:53:05 +00:00
|
|
|
return "IABRT"; // Instruction Abort.
|
2019-04-14 01:24:37 +00:00
|
|
|
case 0x101:
|
2020-07-18 09:53:05 +00:00
|
|
|
return "DABRT"; // Data Abort.
|
2019-04-14 01:24:37 +00:00
|
|
|
case 0x102:
|
2020-07-18 09:53:05 +00:00
|
|
|
return "IUA"; // Instruction Unaligned Access.
|
2019-04-14 01:24:37 +00:00
|
|
|
case 0x103:
|
2020-07-18 09:53:05 +00:00
|
|
|
return "DUA"; // Data Unaligned Access.
|
2019-04-14 01:24:37 +00:00
|
|
|
case 0x104:
|
2020-07-18 09:53:05 +00:00
|
|
|
return "UDF"; // Undefined Instruction.
|
2019-04-14 01:24:37 +00:00
|
|
|
case 0x106:
|
2020-07-18 09:53:05 +00:00
|
|
|
return "SYS"; // System Error.
|
2019-04-14 01:24:37 +00:00
|
|
|
case 0x301:
|
2020-07-18 09:53:05 +00:00
|
|
|
return "SVC"; // Bad arguments or unimplemented SVC.
|
|
|
|
case 0xFFD:
|
|
|
|
return "SO"; // Stack Overflow.
|
2019-06-30 00:55:52 +00:00
|
|
|
case 0xFFE:
|
2020-07-14 19:41:48 +00:00
|
|
|
return "std::abort";
|
2019-04-14 01:24:37 +00:00
|
|
|
default:
|
2020-07-14 19:41:48 +00:00
|
|
|
return "UNK";
|
2019-04-14 01:24:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void secmon_exo_check_panic()
|
|
|
|
{
|
|
|
|
volatile atm_fatal_error_ctx *rpt = (atm_fatal_error_ctx *)ATM_FATAL_ERR_CTX_ADDR;
|
|
|
|
|
2020-03-09 07:09:25 +00:00
|
|
|
// Mask magic to maintain compatibility with any AFE version, thanks to additive struct members.
|
|
|
|
if ((rpt->magic & 0xF0FFFFFF) != ATM_FATAL_MAGIC)
|
2019-04-14 01:24:37 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
gfx_clear_grey(0x1B);
|
|
|
|
gfx_con_setpos(0, 0);
|
|
|
|
|
|
|
|
WPRINTF("Panic occurred while running Atmosphere.\n\n");
|
2020-07-14 19:41:48 +00:00
|
|
|
WPRINTFARGS("Title ID: %08X%08X", (u32)((u64)rpt->title_id >> 32), (u32)rpt->title_id);
|
|
|
|
WPRINTFARGS("Error: %s (0x%x)\n", get_error_desc(rpt->error_desc), rpt->error_desc);
|
2019-04-14 01:24:37 +00:00
|
|
|
|
2020-03-03 02:11:13 +00:00
|
|
|
// Save context to the SD card.
|
|
|
|
char filepath[0x40];
|
|
|
|
f_mkdir("atmosphere/fatal_errors");
|
|
|
|
strcpy(filepath, "/atmosphere/fatal_errors/report_");
|
|
|
|
itoa((u32)((u64)rpt->report_identifier >> 32), filepath + strlen(filepath), 16);
|
|
|
|
itoa((u32)(rpt->report_identifier), filepath + strlen(filepath), 16);
|
|
|
|
strcat(filepath, ".bin");
|
|
|
|
|
2020-07-14 19:41:48 +00:00
|
|
|
if (!sd_save_to_file((void *)rpt, sizeof(atm_fatal_error_ctx), filepath))
|
|
|
|
{
|
|
|
|
gfx_con.fntsz = 8;
|
|
|
|
WPRINTFARGS("Report saved to %s\n", filepath);
|
|
|
|
gfx_con.fntsz = 16;
|
|
|
|
}
|
2019-04-14 01:24:37 +00:00
|
|
|
|
2020-08-02 00:51:36 +00:00
|
|
|
// Change magic to invalid, to prevent double-display of error/bootlooping.
|
|
|
|
rpt->magic = 0;
|
|
|
|
|
2019-04-14 01:24:37 +00:00
|
|
|
gfx_printf("\n\nPress POWER to continue.\n");
|
|
|
|
|
|
|
|
display_backlight_brightness(100, 1000);
|
|
|
|
msleep(1000);
|
|
|
|
|
2020-07-14 19:41:48 +00:00
|
|
|
while (!(btn_wait() & BTN_POWER))
|
|
|
|
;
|
2019-04-14 01:24:37 +00:00
|
|
|
|
|
|
|
display_backlight_brightness(0, 1000);
|
|
|
|
gfx_con_setpos(0, 0);
|
|
|
|
}
|