xref: /haiku/src/system/kernel/arch/riscv64/arch_cpu.cpp (revision 125b262675217084e0c59014b4a98f724f1c4fb3)
1 /*
2  * Copyright 2019, Adrien Destugues, pulkomandy@pulkomandy.tk.
3  * Distributed under the terms of the MIT License.
4  */
5 
6 
7 #include <KernelExport.h>
8 
9 #include <arch/cpu.h>
10 #include <boot/kernel_args.h>
11 #include <vm/VMAddressSpace.h>
12 #include <commpage.h>
13 #include <elf.h>
14 #include <Htif.h>
15 #include <platform/sbi/sbi_syscalls.h>
16 
17 
18 extern "C" void SVec();
19 
20 extern uint32 gPlatform;
21 
22 
23 status_t
24 arch_cpu_preboot_init_percpu(kernel_args *args, int curr_cpu)
25 {
26 	// dprintf("arch_cpu_preboot_init_percpu(%" B_PRId32 ")\n", curr_cpu);
27 	return B_OK;
28 }
29 
30 
31 status_t
32 arch_cpu_init_percpu(kernel_args *args, int curr_cpu)
33 {
34 	SetStvec((uint64)SVec);
35 	SstatusReg sstatus{.val = Sstatus()};
36 	sstatus.ie = 0;
37 	sstatus.fs = extStatusInitial; // enable FPU
38 	sstatus.xs = extStatusOff;
39 	SetSstatus(sstatus.val);
40 	SetBitsSie((1 << sTimerInt) | (1 << sSoftInt) | (1 << sExternInt));
41 
42 	return B_OK;
43 }
44 
45 
46 status_t
47 arch_cpu_init(kernel_args *args)
48 {
49 	for (uint32 curCpu = 0; curCpu < args->num_cpus; curCpu++) {
50 		cpu_ent* cpu = &gCPU[curCpu];
51 
52 		cpu->arch.hartId = args->arch_args.hartIds[curCpu];
53 
54 		cpu->topology_id[CPU_TOPOLOGY_PACKAGE] = 0;
55 		cpu->topology_id[CPU_TOPOLOGY_CORE] = curCpu;
56 		cpu->topology_id[CPU_TOPOLOGY_SMT] = 0;
57 
58 		for (unsigned int i = 0; i < CPU_MAX_CACHE_LEVEL; i++)
59 			cpu->cache_id[i] = -1;
60 	}
61 
62 	uint64 conversionFactor
63 		= (1LL << 32) * 1000000LL / args->arch_args.timerFrequency;
64 
65 	__riscv64_setup_system_time(conversionFactor);
66 
67 	return B_OK;
68 }
69 
70 
71 status_t
72 arch_cpu_init_post_vm(kernel_args *args)
73 {
74 	// Set address space ownership to currently running threads
75 	for (uint32 i = 0; i < args->num_cpus; i++) {
76 		VMAddressSpace::Kernel()->Get();
77 	}
78 
79 	return B_OK;
80 }
81 
82 
83 status_t
84 arch_cpu_init_post_modules(kernel_args *args)
85 {
86 	return B_OK;
87 }
88 
89 
90 void
91 arch_cpu_sync_icache(void *address, size_t len)
92 {
93 }
94 
95 
96 void
97 arch_cpu_memory_read_barrier(void)
98 {
99 }
100 
101 
102 void
103 arch_cpu_memory_write_barrier(void)
104 {
105 }
106 
107 
108 void
109 arch_cpu_invalidate_TLB_range(addr_t start, addr_t end)
110 {
111 	int32 numPages = end / B_PAGE_SIZE - start / B_PAGE_SIZE;
112 	while (numPages-- >= 0) {
113 		FlushTlbPage(start);
114 		start += B_PAGE_SIZE;
115 	}
116 }
117 
118 
119 void
120 arch_cpu_invalidate_TLB_list(addr_t pages[], int num_pages)
121 {
122 	for (int i = 0; i < num_pages; i++)
123 		FlushTlbPage(pages[i]);
124 }
125 
126 
127 void
128 arch_cpu_global_TLB_invalidate(void)
129 {
130 	FlushTlbAll();
131 }
132 
133 
134 void
135 arch_cpu_user_TLB_invalidate(void)
136 {
137 	FlushTlbAll();
138 }
139 
140 
141 status_t
142 arch_cpu_shutdown(bool reboot)
143 {
144 	if (gPlatform == kPlatformSbi) {
145 		sbi_system_reset(
146 			reboot ? SBI_RESET_TYPE_COLD_REBOOT : SBI_RESET_TYPE_SHUTDOWN,
147 			SBI_RESET_REASON_NONE);
148 	}
149 
150 	HtifShutdown();
151 	return B_ERROR;
152 }
153