diff options
Diffstat (limited to 'chip/ish/system.c')
-rw-r--r-- | chip/ish/system.c | 35 |
1 files changed, 34 insertions, 1 deletions
diff --git a/chip/ish/system.c b/chip/ish/system.c index 0d778eaa5e..f634981817 100644 --- a/chip/ish/system.c +++ b/chip/ish/system.c @@ -10,6 +10,7 @@ #include "gpio.h" #include "hooks.h" #include "host_command.h" +#include "interrupts.h" #include "ish_fwst.h" #include "ish_persistent_data.h" #include "power_mgt.h" @@ -21,6 +22,10 @@ #include "timer.h" #include "util.h" +#define CPUTS(outstr) cputs(CC_SYSTEM, outstr) +#define CPRINTF(format, args...) cprintf(CC_SYSTEM, format, ## args) +#define CPRINTS(format, args...) cprints(CC_SYSTEM, format, ## args) + int system_is_reboot_warm(void) { return !(system_get_reset_flags() & @@ -45,6 +50,27 @@ uint32_t chip_read_reset_flags(void) return ish_persistent_data.reset_flags; } +/* + * Kill the Minute-IA core and don't come back alive. + * + * Used when the watchdog timer exceeds max retries and we want to + * disable ISH completely. + */ +__attribute__((noreturn)) +static void system_halt(void) +{ + cflush(); + + while (1) { + disable_all_interrupts(); + WDT_CONTROL = 0; + CCU_TCG_EN = 1; + __asm__ volatile ( + "cli\n" + "hlt\n"); + } +} + void system_reset(int flags) { uint32_t save_flags; @@ -60,8 +86,15 @@ void system_reset(int flags) system_encode_save_flags(flags, &save_flags); - if (flags & SYSTEM_RESET_AP_WATCHDOG) + if (flags & SYSTEM_RESET_AP_WATCHDOG) { save_flags |= RESET_FLAG_WATCHDOG; + ish_persistent_data.watchdog_counter += 1; + if (ish_persistent_data.watchdog_counter + >= CONFIG_WATCHDOG_MAX_RETRIES) { + CPRINTS("Halting ISH due to max watchdog resets"); + system_halt(); + } + } chip_save_reset_flags(save_flags); |