blob: ac22136314f647b0a7a10a27199ce39d385ea9f9 [file] [log] [blame]
Lukas Auerfa33f082019-03-17 19:28:32 +01001// SPDX-License-Identifier: GPL-2.0+
2/*
3 * Copyright (C) 2019 Fraunhofer AISEC,
4 * Lukas Auer <lukas.auer@aisec.fraunhofer.de>
5 */
6
7#include <common.h>
Simon Glass1eb69ae2019-11-14 12:57:39 -07008#include <cpu_func.h>
Lukas Auerfa33f082019-03-17 19:28:32 +01009#include <dm.h>
10#include <asm/barrier.h>
11#include <asm/smp.h>
12
13DECLARE_GLOBAL_DATA_PTR;
14
Lukas Auer90ae2812019-12-08 23:28:51 +010015static int send_ipi_many(struct ipi_data *ipi, int wait)
Lukas Auerfa33f082019-03-17 19:28:32 +010016{
17 ofnode node, cpus;
18 u32 reg;
Lukas Auer90ae2812019-12-08 23:28:51 +010019 int ret, pending;
Lukas Auerfa33f082019-03-17 19:28:32 +010020
21 cpus = ofnode_path("/cpus");
22 if (!ofnode_valid(cpus)) {
23 pr_err("Can't find cpus node!\n");
24 return -EINVAL;
25 }
26
27 ofnode_for_each_subnode(node, cpus) {
28 /* skip if hart is marked as not available in the device tree */
29 if (!ofnode_is_available(node))
30 continue;
31
32 /* read hart ID of CPU */
33 ret = ofnode_read_u32(node, "reg", &reg);
34 if (ret)
35 continue;
36
37 /* skip if it is the hart we are running on */
38 if (reg == gd->arch.boot_hart)
39 continue;
40
41 if (reg >= CONFIG_NR_CPUS) {
42 pr_err("Hart ID %d is out of range, increase CONFIG_NR_CPUS\n",
43 reg);
44 continue;
45 }
46
Rick Chenbdce3892019-04-30 13:49:33 +080047#ifndef CONFIG_XIP
Lukas Auerfa33f082019-03-17 19:28:32 +010048 /* skip if hart is not available */
49 if (!(gd->arch.available_harts & (1 << reg)))
50 continue;
Rick Chenbdce3892019-04-30 13:49:33 +080051#endif
Lukas Auerfa33f082019-03-17 19:28:32 +010052
53 gd->arch.ipi[reg].addr = ipi->addr;
54 gd->arch.ipi[reg].arg0 = ipi->arg0;
55 gd->arch.ipi[reg].arg1 = ipi->arg1;
56
57 ret = riscv_send_ipi(reg);
58 if (ret) {
59 pr_err("Cannot send IPI to hart %d\n", reg);
60 return ret;
61 }
Lukas Auer90ae2812019-12-08 23:28:51 +010062
63 if (wait) {
64 pending = 1;
65 while (pending) {
66 ret = riscv_get_ipi(reg, &pending);
67 if (ret)
68 return ret;
69 }
70 }
Lukas Auerfa33f082019-03-17 19:28:32 +010071 }
72
73 return 0;
74}
75
76void handle_ipi(ulong hart)
77{
78 int ret;
79 void (*smp_function)(ulong hart, ulong arg0, ulong arg1);
80
81 if (hart >= CONFIG_NR_CPUS)
82 return;
83
Lukas Auer90ae2812019-12-08 23:28:51 +010084 __smp_mb();
85
86 smp_function = (void (*)(ulong, ulong, ulong))gd->arch.ipi[hart].addr;
87 invalidate_icache_all();
88
89 /*
90 * Clear the IPI to acknowledge the request before jumping to the
91 * requested function.
92 */
Lukas Auerfa33f082019-03-17 19:28:32 +010093 ret = riscv_clear_ipi(hart);
94 if (ret) {
Sean Anderson40686c32020-06-24 06:41:18 -040095 pr_err("Cannot clear IPI of hart %ld (error %d)\n", hart, ret);
Lukas Auerfa33f082019-03-17 19:28:32 +010096 return;
97 }
98
Lukas Auerfa33f082019-03-17 19:28:32 +010099 smp_function(hart, gd->arch.ipi[hart].arg0, gd->arch.ipi[hart].arg1);
100}
101
Lukas Auer90ae2812019-12-08 23:28:51 +0100102int smp_call_function(ulong addr, ulong arg0, ulong arg1, int wait)
Lukas Auerfa33f082019-03-17 19:28:32 +0100103{
Sean Anderson40686c32020-06-24 06:41:18 -0400104 struct ipi_data ipi = {
105 .addr = addr,
106 .arg0 = arg0,
107 .arg1 = arg1,
108 };
Lukas Auerfa33f082019-03-17 19:28:32 +0100109
Sean Anderson40686c32020-06-24 06:41:18 -0400110 return send_ipi_many(&ipi, wait);
Lukas Auerfa33f082019-03-17 19:28:32 +0100111}