2018-09-07 15:00:13 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2018 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/>.
|
|
|
|
*/
|
|
|
|
|
2018-03-15 15:14:41 +00:00
|
|
|
#include <stdbool.h>
|
2018-05-20 14:18:48 +00:00
|
|
|
#include <stdarg.h>
|
2018-03-15 15:14:41 +00:00
|
|
|
#include "utils.h"
|
|
|
|
#include "se.h"
|
|
|
|
#include "fuse.h"
|
|
|
|
#include "pmc.h"
|
2018-05-20 21:52:09 +00:00
|
|
|
#include "car.h"
|
2018-03-15 15:14:41 +00:00
|
|
|
#include "timers.h"
|
2018-08-18 16:59:33 +00:00
|
|
|
#include "btn.h"
|
2018-05-20 14:18:48 +00:00
|
|
|
#include "panic.h"
|
2018-09-26 00:13:20 +00:00
|
|
|
#include "lib/log.h"
|
2018-03-15 15:14:41 +00:00
|
|
|
|
2018-05-16 17:58:04 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <inttypes.h>
|
2018-03-15 15:14:41 +00:00
|
|
|
|
2018-07-29 16:17:51 +00:00
|
|
|
void wait(uint32_t microseconds) {
|
|
|
|
uint32_t old_time = TIMERUS_CNTR_1US_0;
|
|
|
|
while (TIMERUS_CNTR_1US_0 - old_time <= microseconds) {
|
|
|
|
/* Spin-lock. */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-20 12:11:46 +00:00
|
|
|
__attribute__((noreturn)) void watchdog_reboot(void) {
|
|
|
|
volatile watchdog_timers_t *wdt = GET_WDT(4);
|
|
|
|
wdt->PATTERN = WDT_REBOOT_PATTERN;
|
|
|
|
wdt->COMMAND = 2; /* Disable Counter. */
|
|
|
|
GET_WDT_REBOOT_CFG_REG(4) = 0xC0000000;
|
|
|
|
wdt->CONFIG = 0x8019; /* Full System Reset after Fourth Counter expires, using TIMER(9). */
|
|
|
|
wdt->COMMAND = 1; /* Enable Counter. */
|
|
|
|
while (true) {
|
|
|
|
/* Wait for reboot. */
|
2018-03-15 15:14:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-20 12:11:46 +00:00
|
|
|
__attribute__((noreturn)) void pmc_reboot(uint32_t scratch0) {
|
|
|
|
APBDEV_PMC_SCRATCH0_0 = scratch0;
|
|
|
|
|
|
|
|
/* Reset the processor. */
|
|
|
|
APBDEV_PMC_CONTROL = BIT(4);
|
|
|
|
while (true) {
|
|
|
|
/* Wait for reboot. */
|
|
|
|
}
|
2018-03-15 15:14:41 +00:00
|
|
|
}
|
|
|
|
|
2018-05-20 21:52:09 +00:00
|
|
|
__attribute__((noreturn)) void car_reboot(void) {
|
|
|
|
/* Reset the processor. */
|
|
|
|
car_get_regs()->rst_dev_l |= 1<<2;
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
/* Wait for reboot. */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
__attribute__((noreturn)) void wait_for_button_and_reboot(void) {
|
2018-05-20 12:11:46 +00:00
|
|
|
uint32_t button;
|
|
|
|
while (true) {
|
|
|
|
button = btn_read();
|
|
|
|
if (button & BTN_POWER) {
|
2018-05-20 21:52:09 +00:00
|
|
|
car_reboot();
|
2018-05-20 12:11:46 +00:00
|
|
|
}
|
|
|
|
}
|
2018-03-15 15:14:41 +00:00
|
|
|
}
|
|
|
|
|
2018-09-23 21:22:25 +00:00
|
|
|
void wait_for_button(void) {
|
|
|
|
uint32_t button;
|
|
|
|
while (true) {
|
|
|
|
button = btn_read();
|
|
|
|
if (button) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-20 12:11:46 +00:00
|
|
|
__attribute__ ((noreturn)) void generic_panic(void) {
|
2018-05-20 14:18:48 +00:00
|
|
|
panic(0xFF000006);
|
2018-05-20 12:11:46 +00:00
|
|
|
}
|
2018-05-20 14:18:48 +00:00
|
|
|
|
|
|
|
__attribute__((noreturn)) void fatal_error(const char *fmt, ...) {
|
|
|
|
va_list args;
|
2018-09-23 21:22:25 +00:00
|
|
|
print(SCREEN_LOG_LEVEL_ERROR, "Fatal error: ");
|
2018-05-20 14:18:48 +00:00
|
|
|
va_start(args, fmt);
|
2018-09-23 21:22:25 +00:00
|
|
|
vprint(SCREEN_LOG_LEVEL_ERROR, fmt, args);
|
2018-05-20 14:18:48 +00:00
|
|
|
va_end(args);
|
2018-09-25 21:43:52 +00:00
|
|
|
print(SCREEN_LOG_LEVEL_ERROR | SCREEN_LOG_LEVEL_NO_PREFIX, "\n Press POWER to reboot.\n");
|
2018-05-20 21:52:09 +00:00
|
|
|
wait_for_button_and_reboot();
|
2018-05-20 14:18:48 +00:00
|
|
|
}
|
|
|
|
|
2018-03-15 15:14:41 +00:00
|
|
|
__attribute__((noinline)) bool overlaps(uint64_t as, uint64_t ae, uint64_t bs, uint64_t be)
|
|
|
|
{
|
|
|
|
if(as <= bs && bs <= ae)
|
|
|
|
return true;
|
|
|
|
if(bs <= as && as <= be)
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
2018-05-16 17:58:04 +00:00
|
|
|
|
|
|
|
/* Adapted from https://gist.github.com/ccbrown/9722406 */
|
|
|
|
void hexdump(const void* data, size_t size, uintptr_t addrbase) {
|
|
|
|
const uint8_t *d = (const uint8_t *)data;
|
|
|
|
char ascii[17];
|
|
|
|
ascii[16] = '\0';
|
|
|
|
|
|
|
|
for (size_t i = 0; i < size; i++) {
|
|
|
|
if (i % 16 == 0) {
|
|
|
|
printf("%0*" PRIXPTR ": | ", 2 * sizeof(addrbase), addrbase + i);
|
|
|
|
}
|
|
|
|
printf("%02X ", d[i]);
|
|
|
|
if (d[i] >= ' ' && d[i] <= '~') {
|
|
|
|
ascii[i % 16] = d[i];
|
|
|
|
} else {
|
|
|
|
ascii[i % 16] = '.';
|
|
|
|
}
|
|
|
|
if ((i+1) % 8 == 0 || i+1 == size) {
|
|
|
|
printf(" ");
|
|
|
|
if ((i+1) % 16 == 0) {
|
|
|
|
printf("| %s \n", ascii);
|
|
|
|
} else if (i+1 == size) {
|
|
|
|
ascii[(i+1) % 16] = '\0';
|
|
|
|
if ((i+1) % 16 <= 8) {
|
|
|
|
printf(" ");
|
|
|
|
}
|
|
|
|
for (size_t j = (i+1) % 16; j < 16; j++) {
|
|
|
|
printf(" ");
|
|
|
|
}
|
|
|
|
printf("| %s \n", ascii);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|