blob: affe70081b5b8b37fa27180e6f44612ba56801c4 [file] [log] [blame]
Bin Meng055700e2018-09-26 06:55:14 -07001// SPDX-License-Identifier: GPL-2.0+
2/*
3 * Copyright (C) 2018, Bin Meng <bmeng.cn@gmail.com>
4 */
5
Simon Glass34ee3ed2023-12-15 20:14:09 -07006#include <command.h>
Bin Meng7a3bbfb2018-12-12 06:12:34 -08007#include <cpu.h>
Simon Glass34ee3ed2023-12-15 20:14:09 -07008#include <cpu_func.h>
Bin Mengedfe9a92018-12-12 06:12:38 -08009#include <dm.h>
Heinrich Schuchardtcc382ff2021-09-12 21:11:46 +020010#include <dm/lists.h>
Simon Glassfc557362022-03-04 08:43:05 -070011#include <event.h>
Simon Glass34ee3ed2023-12-15 20:14:09 -070012#include <hang.h>
Simon Glass97589732020-05-10 11:40:02 -060013#include <init.h>
Bin Meng7a3bbfb2018-12-12 06:12:34 -080014#include <log.h>
Bin Menga7544ed2018-12-12 06:12:40 -080015#include <asm/encoding.h>
Simon Glassfc557362022-03-04 08:43:05 -070016#include <asm/system.h>
Bin Mengedfe9a92018-12-12 06:12:38 -080017#include <dm/uclass-internal.h>
Simon Glass4dcacfc2020-05-10 11:40:13 -060018#include <linux/bitops.h>
Bin Meng055700e2018-09-26 06:55:14 -070019
Lukas Auer39a652b2018-11-22 11:26:29 +010020/*
Lukas Auera3596652019-03-17 19:28:37 +010021 * The variables here must be stored in the data section since they are used
Lukas Auer39a652b2018-11-22 11:26:29 +010022 * before the bss section is available.
23 */
Nikita Shubin7e5e0292022-09-02 11:47:39 +030024#if !CONFIG_IS_ENABLED(XIP)
Marek BehĂșn4bebdd32021-05-20 13:23:52 +020025u32 hart_lottery __section(".data") = 0;
Lukas Auera3596652019-03-17 19:28:37 +010026
Rick Chen9c4d5c12022-09-21 14:34:54 +080027#ifdef CONFIG_AVAILABLE_HARTS
Lukas Auera3596652019-03-17 19:28:37 +010028/*
29 * The main hart running U-Boot has acquired available_harts_lock until it has
30 * finished initialization of global data.
31 */
32u32 available_harts_lock = 1;
Rick Chene5e6c362019-04-30 13:49:33 +080033#endif
Rick Chen9c4d5c12022-09-21 14:34:54 +080034#endif
Lukas Auer39a652b2018-11-22 11:26:29 +010035
Bin Meng055700e2018-09-26 06:55:14 -070036static inline bool supports_extension(char ext)
37{
Nikita Shubinc9382b12022-12-14 08:58:43 +030038#if CONFIG_IS_ENABLED(RISCV_MMODE)
39 return csr_read(CSR_MISA) & (1 << (ext - 'a'));
40#elif CONFIG_CPU
Conor Dooley801bbf92024-03-18 15:16:03 +000041 char sext[2] = {ext};
Bin Mengedfe9a92018-12-12 06:12:38 -080042 struct udevice *dev;
Conor Dooley41981552024-03-18 15:16:02 +000043 const char *isa;
Conor Dooley801bbf92024-03-18 15:16:03 +000044 int ret, i;
Bin Mengedfe9a92018-12-12 06:12:38 -080045
46 uclass_find_first_device(UCLASS_CPU, &dev);
47 if (!dev) {
48 debug("unable to find the RISC-V cpu device\n");
49 return false;
50 }
Conor Dooley41981552024-03-18 15:16:02 +000051
Conor Dooley801bbf92024-03-18 15:16:03 +000052 ret = dev_read_stringlist_search(dev, "riscv,isa-extensions", sext);
53 if (ret >= 0)
54 return true;
55
56 /*
57 * Only if the property is not found (ENODATA) is the fallback to
58 * riscv,isa used, otherwise the extension is not present in this
59 * CPU.
60 */
61 if (ret != -ENODATA)
62 return false;
63
Conor Dooley41981552024-03-18 15:16:02 +000064 isa = dev_read_string(dev, "riscv,isa");
Conor Dooley801bbf92024-03-18 15:16:03 +000065 if (!isa)
66 return false;
67
68 /*
69 * Skip the first 4 characters (rv32|rv64).
70 */
71 for (i = 4; i < sizeof(isa); i++) {
72 switch (isa[i]) {
73 case 's':
74 case 'x':
75 case 'z':
76 case '_':
77 case '\0':
78 /*
79 * Any of these characters mean the single
80 * letter extensions have all been consumed.
81 */
82 return false;
83 default:
84 if (isa[i] == ext)
85 return true;
Yu Chien Peter Lina35afb82022-11-05 14:02:14 +080086 }
Bin Mengedfe9a92018-12-12 06:12:38 -080087 }
88
89 return false;
90#else /* !CONFIG_CPU */
Bin Mengedfe9a92018-12-12 06:12:38 -080091#warning "There is no way to determine the available extensions in S-mode."
92#warning "Please convert your board to use the RISC-V CPU driver."
93 return false;
Bin Mengedfe9a92018-12-12 06:12:38 -080094#endif /* CONFIG_CPU */
Bin Meng055700e2018-09-26 06:55:14 -070095}
96
Tom Rinif4d52f62023-09-04 15:06:34 -040097static int riscv_cpu_probe(void)
Bin Meng7a3bbfb2018-12-12 06:12:34 -080098{
99#ifdef CONFIG_CPU
100 int ret;
101
102 /* probe cpus so that RISC-V timer can be bound */
103 ret = cpu_probe_all();
104 if (ret)
105 return log_msg_ret("RISC-V cpus probe failed\n", ret);
106#endif
107
108 return 0;
109}
Tom Rinif4d52f62023-09-04 15:06:34 -0400110EVENT_SPY_SIMPLE(EVT_DM_POST_INIT_R, riscv_cpu_probe);
Bin Meng7a3bbfb2018-12-12 06:12:34 -0800111
Sean Andersondd1cd702020-09-21 07:51:38 -0400112/*
113 * This is called on secondary harts just after the IPI is init'd. Currently
114 * there's nothing to do, since we just need to clear any existing IPIs, and
115 * that is handled by the sending of an ipi itself.
116 */
117#if CONFIG_IS_ENABLED(SMP)
118static void dummy_pending_ipi_clear(ulong hart, ulong arg0, ulong arg1)
119{
120}
121#endif
122
Simon Glassb8357c12023-08-21 21:16:56 -0600123int riscv_cpu_setup(void)
Bin Meng7a3bbfb2018-12-12 06:12:34 -0800124{
Tom Rinic32177d2023-09-04 15:06:35 -0400125 int __maybe_unused ret;
Bin Menga7544ed2018-12-12 06:12:40 -0800126
127 /* Enable FPU */
128 if (supports_extension('d') || supports_extension('f')) {
129 csr_set(MODE_PREFIX(status), MSTATUS_FS);
Bin Mengf9426362019-07-10 23:43:13 -0700130 csr_write(CSR_FCSR, 0);
Bin Menga7544ed2018-12-12 06:12:40 -0800131 }
132
133 if (CONFIG_IS_ENABLED(RISCV_MMODE)) {
134 /*
135 * Enable perf counters for cycle, time,
136 * and instret counters only
137 */
Nikita Shubinc9382b12022-12-14 08:58:43 +0300138 if (supports_extension('u')) {
Sean Anderson7f4b6662020-06-24 06:41:19 -0400139#ifdef CONFIG_RISCV_PRIV_1_9
Nikita Shubinc9382b12022-12-14 08:58:43 +0300140 csr_write(CSR_MSCOUNTEREN, GENMASK(2, 0));
141 csr_write(CSR_MUCOUNTEREN, GENMASK(2, 0));
Sean Anderson7f4b6662020-06-24 06:41:19 -0400142#else
Nikita Shubinc9382b12022-12-14 08:58:43 +0300143 csr_write(CSR_MCOUNTEREN, GENMASK(2, 0));
Sean Anderson7f4b6662020-06-24 06:41:19 -0400144#endif
Nikita Shubinc9382b12022-12-14 08:58:43 +0300145 }
Bin Menga7544ed2018-12-12 06:12:40 -0800146
147 /* Disable paging */
148 if (supports_extension('s'))
Sean Anderson7f4b6662020-06-24 06:41:19 -0400149#ifdef CONFIG_RISCV_PRIV_1_9
150 csr_read_clear(CSR_MSTATUS, SR_VM);
151#else
Bin Mengf9426362019-07-10 23:43:13 -0700152 csr_write(CSR_SATP, 0);
Sean Anderson7f4b6662020-06-24 06:41:19 -0400153#endif
Bin Menga7544ed2018-12-12 06:12:40 -0800154 }
155
Bin Meng257875d2020-07-19 23:17:07 -0700156#if CONFIG_IS_ENABLED(SMP)
Sean Andersonb1d0cb32020-06-24 06:41:18 -0400157 ret = riscv_init_ipi();
158 if (ret)
159 return ret;
Sean Andersondd1cd702020-09-21 07:51:38 -0400160
161 /*
162 * Clear all pending IPIs on secondary harts. We don't do anything on
163 * the boot hart, since we never send an IPI to ourselves, and no
164 * interrupts are enabled
165 */
166 ret = smp_call_function((ulong)dummy_pending_ipi_clear, 0, 0, 0);
167 if (ret)
168 return ret;
Sean Andersonb1d0cb32020-06-24 06:41:18 -0400169#endif
170
Bin Menga7544ed2018-12-12 06:12:40 -0800171 return 0;
Bin Meng7a3bbfb2018-12-12 06:12:34 -0800172}
Simon Glassb8357c12023-08-21 21:16:56 -0600173EVENT_SPY_SIMPLE(EVT_DM_POST_INIT_F, riscv_cpu_setup);
Bin Meng7a3bbfb2018-12-12 06:12:34 -0800174
175int arch_early_init_r(void)
176{
Heinrich Schuchardtcc382ff2021-09-12 21:11:46 +0200177 if (IS_ENABLED(CONFIG_SYSRESET_SBI))
178 device_bind_driver(gd->dm_root, "sbi-sysreset",
179 "sbi-sysreset", NULL);
180
181 return 0;
Bin Meng7a3bbfb2018-12-12 06:12:34 -0800182}
Green Wan26120802021-05-02 23:23:04 -0700183
184/**
185 * harts_early_init() - A callback function called by start.S to configure
186 * feature settings of each hart.
187 *
188 * In a multi-core system, memory access shall be careful here, it shall
189 * take care of race conditions.
190 */
191__weak void harts_early_init(void)
192{
193}
Simon Glass34ee3ed2023-12-15 20:14:09 -0700194
195#if !CONFIG_IS_ENABLED(SYSRESET)
196void reset_cpu(void)
197{
198 printf("resetting ...\n");
199
200 printf("reset not supported yet\n");
201 hang();
202}
203#endif