2019-03-17 18:28:32 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0+
|
|
|
|
/*
|
|
|
|
* Copyright (C) 2019 Fraunhofer AISEC,
|
|
|
|
* Lukas Auer <lukas.auer@aisec.fraunhofer.de>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <common.h>
|
2019-11-14 19:57:39 +00:00
|
|
|
#include <cpu_func.h>
|
2019-03-17 18:28:32 +00:00
|
|
|
#include <dm.h>
|
|
|
|
#include <asm/barrier.h>
|
2020-10-31 03:38:53 +00:00
|
|
|
#include <asm/global_data.h>
|
2019-03-17 18:28:32 +00:00
|
|
|
#include <asm/smp.h>
|
|
|
|
|
|
|
|
DECLARE_GLOBAL_DATA_PTR;
|
|
|
|
|
2019-12-08 22:28:51 +00:00
|
|
|
static int send_ipi_many(struct ipi_data *ipi, int wait)
|
2019-03-17 18:28:32 +00:00
|
|
|
{
|
|
|
|
ofnode node, cpus;
|
|
|
|
u32 reg;
|
2019-12-08 22:28:51 +00:00
|
|
|
int ret, pending;
|
2019-03-17 18:28:32 +00:00
|
|
|
|
|
|
|
cpus = ofnode_path("/cpus");
|
|
|
|
if (!ofnode_valid(cpus)) {
|
|
|
|
pr_err("Can't find cpus node!\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
ofnode_for_each_subnode(node, cpus) {
|
|
|
|
/* skip if hart is marked as not available in the device tree */
|
|
|
|
if (!ofnode_is_available(node))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* read hart ID of CPU */
|
|
|
|
ret = ofnode_read_u32(node, "reg", ®);
|
|
|
|
if (ret)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* skip if it is the hart we are running on */
|
|
|
|
if (reg == gd->arch.boot_hart)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (reg >= CONFIG_NR_CPUS) {
|
|
|
|
pr_err("Hart ID %d is out of range, increase CONFIG_NR_CPUS\n",
|
|
|
|
reg);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2022-09-02 08:47:39 +00:00
|
|
|
#if !CONFIG_IS_ENABLED(XIP)
|
2019-03-17 18:28:32 +00:00
|
|
|
/* skip if hart is not available */
|
|
|
|
if (!(gd->arch.available_harts & (1 << reg)))
|
|
|
|
continue;
|
2019-04-30 05:49:33 +00:00
|
|
|
#endif
|
2019-03-17 18:28:32 +00:00
|
|
|
|
|
|
|
gd->arch.ipi[reg].addr = ipi->addr;
|
|
|
|
gd->arch.ipi[reg].arg0 = ipi->arg0;
|
|
|
|
gd->arch.ipi[reg].arg1 = ipi->arg1;
|
|
|
|
|
2020-09-21 11:51:37 +00:00
|
|
|
/*
|
|
|
|
* Ensure valid only becomes set when the IPI parameters are
|
|
|
|
* set. An IPI may already be pending on other harts, so we
|
|
|
|
* need a way to signal that the IPI device has been
|
|
|
|
* initialized, and that it is ok to call the function.
|
|
|
|
*/
|
|
|
|
__smp_store_release(&gd->arch.ipi[reg].valid, 1);
|
2020-09-21 11:51:36 +00:00
|
|
|
|
2019-03-17 18:28:32 +00:00
|
|
|
ret = riscv_send_ipi(reg);
|
|
|
|
if (ret) {
|
|
|
|
pr_err("Cannot send IPI to hart %d\n", reg);
|
|
|
|
return ret;
|
|
|
|
}
|
2019-12-08 22:28:51 +00:00
|
|
|
|
|
|
|
if (wait) {
|
|
|
|
pending = 1;
|
|
|
|
while (pending) {
|
|
|
|
ret = riscv_get_ipi(reg, &pending);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
2019-03-17 18:28:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void handle_ipi(ulong hart)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
void (*smp_function)(ulong hart, ulong arg0, ulong arg1);
|
|
|
|
|
|
|
|
if (hart >= CONFIG_NR_CPUS)
|
|
|
|
return;
|
|
|
|
|
2020-09-21 11:51:37 +00:00
|
|
|
/*
|
|
|
|
* If valid is not set, then U-Boot has not requested the IPI. The
|
|
|
|
* IPI device may not be initialized, so all we can do is wait for
|
|
|
|
* U-Boot to initialize it and send an IPI
|
|
|
|
*/
|
|
|
|
if (!__smp_load_acquire(&gd->arch.ipi[hart].valid))
|
|
|
|
return;
|
2019-12-08 22:28:51 +00:00
|
|
|
|
|
|
|
smp_function = (void (*)(ulong, ulong, ulong))gd->arch.ipi[hart].addr;
|
|
|
|
invalidate_icache_all();
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Clear the IPI to acknowledge the request before jumping to the
|
|
|
|
* requested function.
|
|
|
|
*/
|
2019-03-17 18:28:32 +00:00
|
|
|
ret = riscv_clear_ipi(hart);
|
|
|
|
if (ret) {
|
2020-06-24 10:41:18 +00:00
|
|
|
pr_err("Cannot clear IPI of hart %ld (error %d)\n", hart, ret);
|
2019-03-17 18:28:32 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
smp_function(hart, gd->arch.ipi[hart].arg0, gd->arch.ipi[hart].arg1);
|
|
|
|
}
|
|
|
|
|
2019-12-08 22:28:51 +00:00
|
|
|
int smp_call_function(ulong addr, ulong arg0, ulong arg1, int wait)
|
2019-03-17 18:28:32 +00:00
|
|
|
{
|
2020-06-24 10:41:18 +00:00
|
|
|
struct ipi_data ipi = {
|
|
|
|
.addr = addr,
|
|
|
|
.arg0 = arg0,
|
|
|
|
.arg1 = arg1,
|
|
|
|
};
|
2019-03-17 18:28:32 +00:00
|
|
|
|
2020-06-24 10:41:18 +00:00
|
|
|
return send_ipi_many(&ipi, wait);
|
2019-03-17 18:28:32 +00:00
|
|
|
}
|