blob: 8445c5823e178cc61e786373434c29d807ee88ca [file] [log] [blame]
Bin Meng2fab2e92018-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 Glassc35bfd02023-12-15 20:14:09 -07006#include <command.h>
Bin Meng39cad5b2018-12-12 06:12:34 -08007#include <cpu.h>
Simon Glassc35bfd02023-12-15 20:14:09 -07008#include <cpu_func.h>
Bin Mengaef59e52018-12-12 06:12:38 -08009#include <dm.h>
Heinrich Schuchardt24ed5312021-09-12 21:11:46 +020010#include <dm/lists.h>
Simon Glass7fe32b32022-03-04 08:43:05 -070011#include <event.h>
Simon Glassc35bfd02023-12-15 20:14:09 -070012#include <hang.h>
Simon Glass691d7192020-05-10 11:40:02 -060013#include <init.h>
Bin Meng39cad5b2018-12-12 06:12:34 -080014#include <log.h>
Bin Meng485e8222018-12-12 06:12:40 -080015#include <asm/encoding.h>
Simon Glass7fe32b32022-03-04 08:43:05 -070016#include <asm/system.h>
Bin Mengaef59e52018-12-12 06:12:38 -080017#include <dm/uclass-internal.h>
Simon Glasscd93d622020-05-10 11:40:13 -060018#include <linux/bitops.h>
Bin Meng2fab2e92018-09-26 06:55:14 -070019
Lukas Auer5d8b2e72018-11-22 11:26:29 +010020/*
Lukas Auer3dea63c2019-03-17 19:28:37 +010021 * The variables here must be stored in the data section since they are used
Lukas Auer5d8b2e72018-11-22 11:26:29 +010022 * before the bss section is available.
23 */
Nikita Shubinc2bdf022022-09-02 11:47:39 +030024#if !CONFIG_IS_ENABLED(XIP)
Marek BehĂșn236f2ec2021-05-20 13:23:52 +020025u32 hart_lottery __section(".data") = 0;
Lukas Auer3dea63c2019-03-17 19:28:37 +010026
Rick Chene0465f82022-09-21 14:34:54 +080027#ifdef CONFIG_AVAILABLE_HARTS
Lukas Auer3dea63c2019-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 Chenbdce3892019-04-30 13:49:33 +080033#endif
Rick Chene0465f82022-09-21 14:34:54 +080034#endif
Lukas Auer5d8b2e72018-11-22 11:26:29 +010035
Bin Meng2fab2e92018-09-26 06:55:14 -070036static inline bool supports_extension(char ext)
37{
Nikita Shubin81b56a52022-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
Bin Mengaef59e52018-12-12 06:12:38 -080041 struct udevice *dev;
42 char desc[32];
Yu Chien Peter Linc277c782022-11-05 14:02:14 +080043 int i;
Bin Mengaef59e52018-12-12 06:12:38 -080044
45 uclass_find_first_device(UCLASS_CPU, &dev);
46 if (!dev) {
47 debug("unable to find the RISC-V cpu device\n");
48 return false;
49 }
50 if (!cpu_get_desc(dev, desc, sizeof(desc))) {
Yu Chien Peter Linc277c782022-11-05 14:02:14 +080051 /*
52 * skip the first 4 characters (rv32|rv64) and
53 * check until underscore
54 */
55 for (i = 4; i < sizeof(desc); i++) {
56 if (desc[i] == '_' || desc[i] == '\0')
57 break;
58 if (desc[i] == ext)
59 return true;
60 }
Bin Mengaef59e52018-12-12 06:12:38 -080061 }
62
63 return false;
64#else /* !CONFIG_CPU */
Bin Mengaef59e52018-12-12 06:12:38 -080065#warning "There is no way to determine the available extensions in S-mode."
66#warning "Please convert your board to use the RISC-V CPU driver."
67 return false;
Bin Mengaef59e52018-12-12 06:12:38 -080068#endif /* CONFIG_CPU */
Bin Meng2fab2e92018-09-26 06:55:14 -070069}
70
Tom Rini68f446f2023-09-04 15:06:34 -040071static int riscv_cpu_probe(void)
Bin Meng39cad5b2018-12-12 06:12:34 -080072{
73#ifdef CONFIG_CPU
74 int ret;
75
76 /* probe cpus so that RISC-V timer can be bound */
77 ret = cpu_probe_all();
78 if (ret)
79 return log_msg_ret("RISC-V cpus probe failed\n", ret);
80#endif
81
82 return 0;
83}
Tom Rini68f446f2023-09-04 15:06:34 -040084EVENT_SPY_SIMPLE(EVT_DM_POST_INIT_R, riscv_cpu_probe);
Bin Meng39cad5b2018-12-12 06:12:34 -080085
Sean Anderson768502e2020-09-21 07:51:38 -040086/*
87 * This is called on secondary harts just after the IPI is init'd. Currently
88 * there's nothing to do, since we just need to clear any existing IPIs, and
89 * that is handled by the sending of an ipi itself.
90 */
91#if CONFIG_IS_ENABLED(SMP)
92static void dummy_pending_ipi_clear(ulong hart, ulong arg0, ulong arg1)
93{
94}
95#endif
96
Simon Glassf72d0d42023-08-21 21:16:56 -060097int riscv_cpu_setup(void)
Bin Meng39cad5b2018-12-12 06:12:34 -080098{
Tom Rini59d2a7d2023-09-04 15:06:35 -040099 int __maybe_unused ret;
Bin Meng485e8222018-12-12 06:12:40 -0800100
101 /* Enable FPU */
102 if (supports_extension('d') || supports_extension('f')) {
103 csr_set(MODE_PREFIX(status), MSTATUS_FS);
Bin Meng4d2583d2019-07-10 23:43:13 -0700104 csr_write(CSR_FCSR, 0);
Bin Meng485e8222018-12-12 06:12:40 -0800105 }
106
107 if (CONFIG_IS_ENABLED(RISCV_MMODE)) {
108 /*
109 * Enable perf counters for cycle, time,
110 * and instret counters only
111 */
Nikita Shubin81b56a52022-12-14 08:58:43 +0300112 if (supports_extension('u')) {
Sean Andersonb8bc1202020-06-24 06:41:19 -0400113#ifdef CONFIG_RISCV_PRIV_1_9
Nikita Shubin81b56a52022-12-14 08:58:43 +0300114 csr_write(CSR_MSCOUNTEREN, GENMASK(2, 0));
115 csr_write(CSR_MUCOUNTEREN, GENMASK(2, 0));
Sean Andersonb8bc1202020-06-24 06:41:19 -0400116#else
Nikita Shubin81b56a52022-12-14 08:58:43 +0300117 csr_write(CSR_MCOUNTEREN, GENMASK(2, 0));
Sean Andersonb8bc1202020-06-24 06:41:19 -0400118#endif
Nikita Shubin81b56a52022-12-14 08:58:43 +0300119 }
Bin Meng485e8222018-12-12 06:12:40 -0800120
121 /* Disable paging */
122 if (supports_extension('s'))
Sean Andersonb8bc1202020-06-24 06:41:19 -0400123#ifdef CONFIG_RISCV_PRIV_1_9
124 csr_read_clear(CSR_MSTATUS, SR_VM);
125#else
Bin Meng4d2583d2019-07-10 23:43:13 -0700126 csr_write(CSR_SATP, 0);
Sean Andersonb8bc1202020-06-24 06:41:19 -0400127#endif
Bin Meng485e8222018-12-12 06:12:40 -0800128 }
129
Bin Menga0018fc2020-07-19 23:17:07 -0700130#if CONFIG_IS_ENABLED(SMP)
Sean Anderson40686c32020-06-24 06:41:18 -0400131 ret = riscv_init_ipi();
132 if (ret)
133 return ret;
Sean Anderson768502e2020-09-21 07:51:38 -0400134
135 /*
136 * Clear all pending IPIs on secondary harts. We don't do anything on
137 * the boot hart, since we never send an IPI to ourselves, and no
138 * interrupts are enabled
139 */
140 ret = smp_call_function((ulong)dummy_pending_ipi_clear, 0, 0, 0);
141 if (ret)
142 return ret;
Sean Anderson40686c32020-06-24 06:41:18 -0400143#endif
144
Bin Meng485e8222018-12-12 06:12:40 -0800145 return 0;
Bin Meng39cad5b2018-12-12 06:12:34 -0800146}
Simon Glassf72d0d42023-08-21 21:16:56 -0600147EVENT_SPY_SIMPLE(EVT_DM_POST_INIT_F, riscv_cpu_setup);
Bin Meng39cad5b2018-12-12 06:12:34 -0800148
149int arch_early_init_r(void)
150{
Heinrich Schuchardt24ed5312021-09-12 21:11:46 +0200151 if (IS_ENABLED(CONFIG_SYSRESET_SBI))
152 device_bind_driver(gd->dm_root, "sbi-sysreset",
153 "sbi-sysreset", NULL);
154
155 return 0;
Bin Meng39cad5b2018-12-12 06:12:34 -0800156}
Green Wanedd9ad82021-05-02 23:23:04 -0700157
158/**
159 * harts_early_init() - A callback function called by start.S to configure
160 * feature settings of each hart.
161 *
162 * In a multi-core system, memory access shall be careful here, it shall
163 * take care of race conditions.
164 */
165__weak void harts_early_init(void)
166{
167}
Simon Glassc35bfd02023-12-15 20:14:09 -0700168
169#if !CONFIG_IS_ENABLED(SYSRESET)
170void reset_cpu(void)
171{
172 printf("resetting ...\n");
173
174 printf("reset not supported yet\n");
175 hang();
176}
177#endif