blob: 0d099863e65e20464e7bf53ec59667d92c8ee938 [file] [log] [blame] [edit]
// 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
// in psci.S
extern uint64_t psci_smc_call(ulong arg0, ulong arg1, ulong arg2, ulong arg3);
extern uint64_t psci_hvc_call(ulong arg0, ulong arg1, ulong arg2, ulong 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
#if WITH_DEV_PDEV
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;
}
}
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) {
ulong 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