| // SPDX-License-Identifier: GPL-2.0 |
| /* |
| * Copyright (C) 2020-2022 Loongson Technology Corporation Limited |
| */ |
| #include <linux/kernel.h> |
| #include <linux/acpi.h> |
| #include <linux/efi.h> |
| #include <linux/export.h> |
| #include <linux/pm.h> |
| #include <linux/types.h> |
| #include <linux/reboot.h> |
| #include <linux/delay.h> |
| #include <linux/console.h> |
| |
| #include <acpi/reboot.h> |
| #include <asm/compiler.h> |
| #include <asm/idle.h> |
| #include <asm/loongarch.h> |
| #include <asm/reboot.h> |
| |
| static void default_halt(void) |
| { |
| local_irq_disable(); |
| clear_csr_ecfg(ECFG0_IM); |
| |
| pr_notice("\n\n** You can safely turn off the power now **\n\n"); |
| console_flush_on_panic(CONSOLE_FLUSH_PENDING); |
| |
| while (true) { |
| __arch_cpu_idle(); |
| } |
| } |
| |
| static void default_poweroff(void) |
| { |
| #ifdef CONFIG_EFI |
| efi.reset_system(EFI_RESET_SHUTDOWN, EFI_SUCCESS, 0, NULL); |
| #endif |
| while (true) { |
| __arch_cpu_idle(); |
| } |
| } |
| |
| static void default_restart(void) |
| { |
| #ifdef CONFIG_EFI |
| if (efi_capsule_pending(NULL)) |
| efi_reboot(REBOOT_WARM, NULL); |
| else |
| efi_reboot(REBOOT_COLD, NULL); |
| #endif |
| if (!acpi_disabled) |
| acpi_reboot(); |
| |
| while (true) { |
| __arch_cpu_idle(); |
| } |
| } |
| |
| void (*pm_restart)(void); |
| EXPORT_SYMBOL(pm_restart); |
| |
| void (*pm_power_off)(void); |
| EXPORT_SYMBOL(pm_power_off); |
| |
| void machine_halt(void) |
| { |
| #ifdef CONFIG_SMP |
| preempt_disable(); |
| smp_send_stop(); |
| #endif |
| default_halt(); |
| } |
| |
| void machine_power_off(void) |
| { |
| #ifdef CONFIG_SMP |
| preempt_disable(); |
| smp_send_stop(); |
| #endif |
| pm_power_off(); |
| } |
| |
| void machine_restart(char *command) |
| { |
| #ifdef CONFIG_SMP |
| preempt_disable(); |
| smp_send_stop(); |
| #endif |
| do_kernel_restart(command); |
| pm_restart(); |
| } |
| |
| static int __init loongarch_reboot_setup(void) |
| { |
| pm_restart = default_restart; |
| pm_power_off = default_poweroff; |
| |
| return 0; |
| } |
| |
| arch_initcall(loongarch_reboot_setup); |