| // Copyright 2017 The Fuchsia Authors |
| // |
| // Use of this source code is governed by a MIT-style |
| // license that can be found in the LICENSE file or at |
| // https://opensource.org/licenses/MIT |
| |
| #include <dev/psci.h> |
| |
| #if WITH_DEV_PDEV |
| #include <pdev/driver.h> |
| #include <mdi/mdi.h> |
| #include <mdi/mdi-defs.h> |
| #endif |
| |
| static uint64_t shutdown_args[3] = { 0, 0, 0 }; |
| static uint64_t reboot_args[3] = { 0, 0, 0 }; |
| static uint64_t reboot_bootloader_args[3] = { 0, 0, 0 }; |
| |
| // in psci.S |
| extern uint64_t psci_smc_call(uint64_t arg0, uint64_t arg1, uint64_t arg2, uint64_t arg3); |
| extern uint64_t psci_hvc_call(uint64_t arg0, uint64_t arg1, uint64_t arg2, uint64_t arg3); |
| |
| #if PSCI_USE_HVC |
| psci_call_proc do_psci_call = psci_hvc_call; |
| #else |
| psci_call_proc do_psci_call = psci_smc_call; |
| #endif |
| |
| void psci_system_off(void) { |
| |
| do_psci_call(PSCI64_SYSTEM_OFF, shutdown_args[0], shutdown_args[1], shutdown_args[2]); |
| } |
| |
| void psci_system_reset(enum reboot_flags flags) { |
| uint64_t* args = reboot_args; |
| |
| if (flags == REBOOT_BOOTLOADER) |
| args = reboot_bootloader_args; |
| |
| do_psci_call(PSCI64_SYSTEM_RESET, args[0], args[1], args[2]); |
| } |
| |
| #if WITH_DEV_PDEV |
| static void arm_psci_read_arglist(mdi_node_ref_t* node, uint64_t* dest) { |
| if (mdi_array_length(node) != 3) |
| panic("bad array length in arm_psci_read_arglist for node %u\n", mdi_id(node)); |
| |
| for (int i = 0; i < 3; i++) |
| mdi_array_uint64(node, i, dest + i); |
| } |
| |
| static void arm_psci_init(mdi_node_ref_t* node, uint level) { |
| bool use_smc = false; |
| bool use_hvc = false; |
| |
| mdi_node_ref_t child; |
| mdi_each_child(node, &child) { |
| switch (mdi_id(&child)) { |
| case MDI_ARM_PSCI_USE_SMC: |
| mdi_node_boolean(&child, &use_smc); |
| break; |
| case MDI_ARM_PSCI_USE_HVC: |
| mdi_node_boolean(&child, &use_hvc); |
| break; |
| case MDI_ARM_PSCI_SHUTDOWN_ARGS: |
| arm_psci_read_arglist(&child, shutdown_args); |
| break; |
| case MDI_ARM_PSCI_REBOOT_ARGS: |
| arm_psci_read_arglist(&child, reboot_args); |
| break; |
| case MDI_ARM_PSCI_REBOOT_BOOTLOADER_ARGS: |
| arm_psci_read_arglist(&child, reboot_bootloader_args); |
| break; |
| } |
| } |
| |
| if (use_smc && use_hvc) { |
| panic("both use-smc and use-hvc set in arm_psci_init\n"); |
| } |
| if (!use_smc && !use_hvc) { |
| panic("neither use-smc and use-hvc set in arm_psci_init\n"); |
| } |
| do_psci_call = use_smc ? psci_smc_call : psci_hvc_call; |
| } |
| |
| LK_PDEV_INIT(arm_psci_init, MDI_ARM_PSCI, arm_psci_init, LK_INIT_LEVEL_PLATFORM_EARLY); |
| |
| #if WITH_LIB_CONSOLE |
| #include <lib/console.h> |
| |
| static int cmd_psci(int argc, const cmd_args *argv, uint32_t flags) { |
| uint64_t arg0, arg1 = 0, arg2 = 0, arg3 = 0; |
| |
| if (argc < 2) { |
| printf("not enough arguments\n"); |
| printf("%s arg0 [arg1] [arg2] [arg3]\n", argv[0].str); |
| return -1; |
| } |
| |
| arg0 = argv[1].u; |
| if (argc >= 3) { |
| arg1 = argv[2].u; |
| if (argc >= 4) { |
| arg2 = argv[3].u; |
| if (argc >= 5) { |
| arg3 = argv[4].u; |
| } |
| } |
| } |
| |
| uint32_t ret = do_psci_call(arg0, arg1, arg2, arg3); |
| printf("do_psci_call returned %u\n", ret); |
| return 0; |
| } |
| #endif // WITH_LIB_CONSOLE |
| |
| STATIC_COMMAND_START |
| STATIC_COMMAND_MASKED("psci", "execute PSCI command", &cmd_psci, CMD_AVAIL_ALWAYS) |
| STATIC_COMMAND_END(psci); |
| |
| #endif // WITH_DEV_PDEV |