| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183 |
- // SPDX-License-Identifier: GPL-2.0-only
- /*
- * Copyright (c) 2021 Western Digital Corporation or its affiliates.
- * Copyright (c) 2022 Ventana Micro Systems Inc.
- */
- #define pr_fmt(fmt) "suspend: " fmt
- #include <linux/ftrace.h>
- #include <linux/suspend.h>
- #include <asm/csr.h>
- #include <asm/sbi.h>
- #include <asm/suspend.h>
- void suspend_save_csrs(struct suspend_context *context)
- {
- if (riscv_cpu_has_extension_unlikely(smp_processor_id(), RISCV_ISA_EXT_XLINUXENVCFG))
- context->envcfg = csr_read(CSR_ENVCFG);
- context->tvec = csr_read(CSR_TVEC);
- context->ie = csr_read(CSR_IE);
- /*
- * No need to save/restore IP CSR (i.e. MIP or SIP) because:
- *
- * 1. For no-MMU (M-mode) kernel, the bits in MIP are set by
- * external devices (such as interrupt controller, timer, etc).
- * 2. For MMU (S-mode) kernel, the bits in SIP are set by
- * M-mode firmware and external devices (such as interrupt
- * controller, etc).
- */
- #ifdef CONFIG_MMU
- context->satp = csr_read(CSR_SATP);
- #endif
- }
- void suspend_restore_csrs(struct suspend_context *context)
- {
- csr_write(CSR_SCRATCH, 0);
- if (riscv_cpu_has_extension_unlikely(smp_processor_id(), RISCV_ISA_EXT_XLINUXENVCFG))
- csr_write(CSR_ENVCFG, context->envcfg);
- csr_write(CSR_TVEC, context->tvec);
- csr_write(CSR_IE, context->ie);
- #ifdef CONFIG_MMU
- csr_write(CSR_SATP, context->satp);
- #endif
- }
- int cpu_suspend(unsigned long arg,
- int (*finish)(unsigned long arg,
- unsigned long entry,
- unsigned long context))
- {
- int rc = 0;
- struct suspend_context context = { 0 };
- /* Finisher should be non-NULL */
- if (!finish)
- return -EINVAL;
- /* Save additional CSRs*/
- suspend_save_csrs(&context);
- /*
- * Function graph tracer state gets incosistent when the kernel
- * calls functions that never return (aka finishers) hence disable
- * graph tracing during their execution.
- */
- pause_graph_tracing();
- /* Save context on stack */
- if (__cpu_suspend_enter(&context)) {
- /* Call the finisher */
- rc = finish(arg, __pa_symbol(__cpu_resume_enter),
- (ulong)&context);
- /*
- * Should never reach here, unless the suspend finisher
- * fails. Successful cpu_suspend() should return from
- * __cpu_resume_entry()
- */
- if (!rc)
- rc = -EOPNOTSUPP;
- }
- /* Enable function graph tracer */
- unpause_graph_tracing();
- /* Restore additional CSRs */
- suspend_restore_csrs(&context);
- return rc;
- }
- #ifdef CONFIG_RISCV_SBI
- static int sbi_system_suspend(unsigned long sleep_type,
- unsigned long resume_addr,
- unsigned long opaque)
- {
- struct sbiret ret;
- ret = sbi_ecall(SBI_EXT_SUSP, SBI_EXT_SUSP_SYSTEM_SUSPEND,
- sleep_type, resume_addr, opaque, 0, 0, 0);
- if (ret.error)
- return sbi_err_map_linux_errno(ret.error);
- return ret.value;
- }
- static int sbi_system_suspend_enter(suspend_state_t state)
- {
- return cpu_suspend(SBI_SUSP_SLEEP_TYPE_SUSPEND_TO_RAM, sbi_system_suspend);
- }
- static const struct platform_suspend_ops sbi_system_suspend_ops = {
- .valid = suspend_valid_only_mem,
- .enter = sbi_system_suspend_enter,
- };
- static int __init sbi_system_suspend_init(void)
- {
- if (sbi_spec_version >= sbi_mk_version(2, 0) &&
- sbi_probe_extension(SBI_EXT_SUSP) > 0) {
- pr_info("SBI SUSP extension detected\n");
- if (IS_ENABLED(CONFIG_SUSPEND))
- suspend_set_ops(&sbi_system_suspend_ops);
- }
- return 0;
- }
- arch_initcall(sbi_system_suspend_init);
- static int sbi_suspend_finisher(unsigned long suspend_type,
- unsigned long resume_addr,
- unsigned long opaque)
- {
- struct sbiret ret;
- ret = sbi_ecall(SBI_EXT_HSM, SBI_EXT_HSM_HART_SUSPEND,
- suspend_type, resume_addr, opaque, 0, 0, 0);
- return (ret.error) ? sbi_err_map_linux_errno(ret.error) : 0;
- }
- int riscv_sbi_hart_suspend(u32 state)
- {
- if (state & SBI_HSM_SUSP_NON_RET_BIT)
- return cpu_suspend(state, sbi_suspend_finisher);
- else
- return sbi_suspend_finisher(state, 0, 0);
- }
- bool riscv_sbi_suspend_state_is_valid(u32 state)
- {
- if (state > SBI_HSM_SUSPEND_RET_DEFAULT &&
- state < SBI_HSM_SUSPEND_RET_PLATFORM)
- return false;
- if (state > SBI_HSM_SUSPEND_NON_RET_DEFAULT &&
- state < SBI_HSM_SUSPEND_NON_RET_PLATFORM)
- return false;
- return true;
- }
- bool riscv_sbi_hsm_is_supported(void)
- {
- /*
- * The SBI HSM suspend function is only available when:
- * 1) SBI version is 0.3 or higher
- * 2) SBI HSM extension is available
- */
- if (sbi_spec_version < sbi_mk_version(0, 3) ||
- !sbi_probe_extension(SBI_EXT_HSM)) {
- pr_info("HSM suspend not available\n");
- return false;
- }
- return true;
- }
- #endif /* CONFIG_RISCV_SBI */
|