blob: ba992100adf1c569152d852f1c843c794a6c9d07 [file] [log] [blame]
Lukas Auer83d573d2019-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 Glass63334482019-11-14 12:57:39 -07008#include <cpu_func.h>
Lukas Auer83d573d2019-03-17 19:28:32 +01009#include <dm.h>
10#include <asm/barrier.h>
Simon Glass3ba929a2020-10-30 21:38:53 -060011#include <asm/global_data.h>
Lukas Auer83d573d2019-03-17 19:28:32 +010012#include <asm/smp.h>
13
14DECLARE_GLOBAL_DATA_PTR;
15
Lukas Auerc308e012019-12-08 23:28:51 +010016static int send_ipi_many(struct ipi_data *ipi, int wait)
Lukas Auer83d573d2019-03-17 19:28:32 +010017{
18 ofnode node, cpus;
19 u32 reg;
Lukas Auerc308e012019-12-08 23:28:51 +010020 int ret, pending;
Lukas Auer83d573d2019-03-17 19:28:32 +010021
22 cpus = ofnode_path("/cpus");
23 if (!ofnode_valid(cpus)) {
24 pr_err("Can't find cpus node!\n");
25 return -EINVAL;
26 }
27
28 ofnode_for_each_subnode(node, cpus) {
29 /* skip if hart is marked as not available in the device tree */
30 if (!ofnode_is_available(node))
31 continue;
32
33 /* read hart ID of CPU */
34 ret = ofnode_read_u32(node, "reg", &reg);
35 if (ret)
36 continue;
37
38 /* skip if it is the hart we are running on */
39 if (reg == gd->arch.boot_hart)
40 continue;
41
42 if (reg >= CONFIG_NR_CPUS) {
43 pr_err("Hart ID %d is out of range, increase CONFIG_NR_CPUS\n",
44 reg);
45 continue;
46 }
47
Rick Chene5e6c362019-04-30 13:49:33 +080048#ifndef CONFIG_XIP
Lukas Auer83d573d2019-03-17 19:28:32 +010049 /* skip if hart is not available */
50 if (!(gd->arch.available_harts & (1 << reg)))
51 continue;
Rick Chene5e6c362019-04-30 13:49:33 +080052#endif
Lukas Auer83d573d2019-03-17 19:28:32 +010053
54 gd->arch.ipi[reg].addr = ipi->addr;
55 gd->arch.ipi[reg].arg0 = ipi->arg0;
56 gd->arch.ipi[reg].arg1 = ipi->arg1;
57
Sean Andersonff184fe2020-09-21 07:51:37 -040058 /*
59 * Ensure valid only becomes set when the IPI parameters are
60 * set. An IPI may already be pending on other harts, so we
61 * need a way to signal that the IPI device has been
62 * initialized, and that it is ok to call the function.
63 */
64 __smp_store_release(&gd->arch.ipi[reg].valid, 1);
Sean Andersoncfb08092020-09-21 07:51:36 -040065
Lukas Auer83d573d2019-03-17 19:28:32 +010066 ret = riscv_send_ipi(reg);
67 if (ret) {
68 pr_err("Cannot send IPI to hart %d\n", reg);
69 return ret;
70 }
Lukas Auerc308e012019-12-08 23:28:51 +010071
72 if (wait) {
73 pending = 1;
74 while (pending) {
75 ret = riscv_get_ipi(reg, &pending);
76 if (ret)
77 return ret;
78 }
79 }
Lukas Auer83d573d2019-03-17 19:28:32 +010080 }
81
82 return 0;
83}
84
85void handle_ipi(ulong hart)
86{
87 int ret;
88 void (*smp_function)(ulong hart, ulong arg0, ulong arg1);
89
90 if (hart >= CONFIG_NR_CPUS)
91 return;
92
Sean Andersonff184fe2020-09-21 07:51:37 -040093 /*
94 * If valid is not set, then U-Boot has not requested the IPI. The
95 * IPI device may not be initialized, so all we can do is wait for
96 * U-Boot to initialize it and send an IPI
97 */
98 if (!__smp_load_acquire(&gd->arch.ipi[hart].valid))
99 return;
Lukas Auerc308e012019-12-08 23:28:51 +0100100
101 smp_function = (void (*)(ulong, ulong, ulong))gd->arch.ipi[hart].addr;
102 invalidate_icache_all();
103
104 /*
105 * Clear the IPI to acknowledge the request before jumping to the
106 * requested function.
107 */
Lukas Auer83d573d2019-03-17 19:28:32 +0100108 ret = riscv_clear_ipi(hart);
109 if (ret) {
Sean Andersonb1d0cb32020-06-24 06:41:18 -0400110 pr_err("Cannot clear IPI of hart %ld (error %d)\n", hart, ret);
Lukas Auer83d573d2019-03-17 19:28:32 +0100111 return;
112 }
113
Lukas Auer83d573d2019-03-17 19:28:32 +0100114 smp_function(hart, gd->arch.ipi[hart].arg0, gd->arch.ipi[hart].arg1);
115}
116
Lukas Auerc308e012019-12-08 23:28:51 +0100117int smp_call_function(ulong addr, ulong arg0, ulong arg1, int wait)
Lukas Auer83d573d2019-03-17 19:28:32 +0100118{
Sean Andersonb1d0cb32020-06-24 06:41:18 -0400119 struct ipi_data ipi = {
120 .addr = addr,
121 .arg0 = arg0,
122 .arg1 = arg1,
123 };
Lukas Auer83d573d2019-03-17 19:28:32 +0100124
Sean Andersonb1d0cb32020-06-24 06:41:18 -0400125 return send_ipi_many(&ipi, wait);
Lukas Auer83d573d2019-03-17 19:28:32 +0100126}