/* * SPDX-License-Identifier: BSD-2-Clause * * Copyright (c) 2021 SiFive * Copyright (c) 2021 Western Digital Corporation or its affiliates. * * Authors: * Green Wan * Anup Patel */ #include #include #include #include #include #include #include #include #include struct gpio_reset { struct gpio_pin pin; u32 active_delay; u32 inactive_delay; }; static struct gpio_reset poweroff = { .active_delay = 100, .inactive_delay = 100 }; static struct gpio_reset restart = { .active_delay = 100, .inactive_delay = 100 }; static struct gpio_reset *gpio_reset_get(bool is_restart, u32 type) { struct gpio_reset *reset = NULL; switch (type) { case SBI_SRST_RESET_TYPE_SHUTDOWN: if (!is_restart) reset = &poweroff; break; case SBI_SRST_RESET_TYPE_COLD_REBOOT: case SBI_SRST_RESET_TYPE_WARM_REBOOT: if (is_restart) reset = &restart; break; } if (reset && !reset->pin.chip) reset = NULL; return reset; } static void gpio_reset_exec(struct gpio_reset *reset) { if (reset) { /* drive it active, also inactive->active edge */ gpio_direction_output(&reset->pin, 1); sbi_timer_mdelay(reset->active_delay); /* drive inactive, also active->inactive edge */ gpio_set(&reset->pin, 0); sbi_timer_mdelay(reset->inactive_delay); /* drive it active, also inactive->active edge */ gpio_set(&reset->pin, 1); } /* hang !!! */ sbi_hart_hang(); } static int gpio_system_poweroff_check(u32 type, u32 reason) { if (gpio_reset_get(FALSE, type)) return 128; return 0; } static void gpio_system_poweroff(u32 type, u32 reason) { gpio_reset_exec(gpio_reset_get(FALSE, type)); } static struct sbi_system_reset_device gpio_poweroff = { .name = "gpio-poweroff", .system_reset_check = gpio_system_poweroff_check, .system_reset = gpio_system_poweroff }; static int gpio_system_restart_check(u32 type, u32 reason) { if (gpio_reset_get(TRUE, type)) return 128; return 0; } static void gpio_system_restart(u32 type, u32 reason) { gpio_reset_exec(gpio_reset_get(TRUE, type)); } static struct sbi_system_reset_device gpio_restart = { .name = "gpio-restart", .system_reset_check = gpio_system_restart_check, .system_reset = gpio_system_restart }; static int gpio_reset_init(void *fdt, int nodeoff, const struct fdt_match *match) { int rc, len; const fdt32_t *val; bool is_restart = (ulong)match->data; const char *dir_prop = (is_restart) ? "open-source" : "input"; struct gpio_reset *reset = (is_restart) ? &restart : &poweroff; rc = fdt_gpio_pin_get(fdt, nodeoff, 0, &reset->pin); if (rc) return rc; if (fdt_getprop(fdt, nodeoff, dir_prop, &len)) { rc = gpio_direction_input(&reset->pin); if (rc) return rc; } val = fdt_getprop(fdt, nodeoff, "active-delay-ms", &len); if (len > 0) reset->active_delay = fdt32_to_cpu(*val); val = fdt_getprop(fdt, nodeoff, "inactive-delay-ms", &len); if (len > 0) reset->inactive_delay = fdt32_to_cpu(*val); if (is_restart) sbi_system_reset_add_device(&gpio_restart); else sbi_system_reset_add_device(&gpio_poweroff); return 0; } static const struct fdt_match gpio_poweroff_match[] = { { .compatible = "gpio-poweroff", .data = (void *)FALSE }, { }, }; struct fdt_reset fdt_poweroff_gpio = { .match_table = gpio_poweroff_match, .init = gpio_reset_init, }; static const struct fdt_match gpio_reset_match[] = { { .compatible = "gpio-restart", .data = (void *)TRUE }, { }, }; struct fdt_reset fdt_reset_gpio = { .match_table = gpio_reset_match, .init = gpio_reset_init, };