OpenCores
URL https://opencores.org/ocsvn/neorv32/neorv32/trunk

Subversion Repositories neorv32

[/] [neorv32/] [trunk/] [sw/] [lib/] [include/] [neorv32_cpu.h] - Blame information for rev 62

Go to most recent revision | Details | Compare with Previous | View Log

Line No. Rev Author Line
1 2 zero_gravi
// #################################################################################################
2
// # << NEORV32: neorv32_cpu.h - CPU Core Functions HW Driver >>                                   #
3
// # ********************************************************************************************* #
4
// # BSD 3-Clause License                                                                          #
5
// #                                                                                               #
6 42 zero_gravi
// # Copyright (c) 2021, Stephan Nolting. All rights reserved.                                     #
7 2 zero_gravi
// #                                                                                               #
8
// # Redistribution and use in source and binary forms, with or without modification, are          #
9
// # permitted provided that the following conditions are met:                                     #
10
// #                                                                                               #
11
// # 1. Redistributions of source code must retain the above copyright notice, this list of        #
12
// #    conditions and the following disclaimer.                                                   #
13
// #                                                                                               #
14
// # 2. Redistributions in binary form must reproduce the above copyright notice, this list of     #
15
// #    conditions and the following disclaimer in the documentation and/or other materials        #
16
// #    provided with the distribution.                                                            #
17
// #                                                                                               #
18
// # 3. Neither the name of the copyright holder nor the names of its contributors may be used to  #
19
// #    endorse or promote products derived from this software without specific prior written      #
20
// #    permission.                                                                                #
21
// #                                                                                               #
22
// # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS   #
23
// # OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF               #
24
// # MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE    #
25
// # COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,     #
26
// # EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE #
27
// # GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED    #
28
// # AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING     #
29
// # NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED  #
30
// # OF THE POSSIBILITY OF SUCH DAMAGE.                                                            #
31
// # ********************************************************************************************* #
32
// # The NEORV32 Processor - https://github.com/stnolting/neorv32              (c) Stephan Nolting #
33
// #################################################################################################
34
 
35
 
36
/**********************************************************************//**
37
 * @file neorv32_cpu.h
38
 * @author Stephan Nolting
39
 * @brief CPU Core Functions HW driver header file.
40
 **************************************************************************/
41
 
42
#ifndef neorv32_cpu_h
43
#define neorv32_cpu_h
44
 
45
// prototypes
46
int neorv32_cpu_irq_enable(uint8_t irq_sel);
47
int neorv32_cpu_irq_disable(uint8_t irq_sel);
48 12 zero_gravi
uint64_t neorv32_cpu_get_cycle(void);
49
void neorv32_cpu_set_mcycle(uint64_t value);
50
uint64_t neorv32_cpu_get_instret(void);
51
void neorv32_cpu_set_minstret(uint64_t value);
52
uint64_t neorv32_cpu_get_systime(void);
53 56 zero_gravi
void neorv32_cpu_delay_ms(int16_t time_ms);
54 15 zero_gravi
void __attribute__((naked)) neorv32_cpu_goto_user_mode(void);
55 42 zero_gravi
uint32_t neorv32_cpu_pmp_get_num_regions(void);
56 40 zero_gravi
uint32_t neorv32_cpu_pmp_get_granularity(void);
57
int neorv32_cpu_pmp_configure_region(uint32_t index, uint32_t base, uint32_t size, uint8_t config);
58 42 zero_gravi
uint32_t neorv32_cpu_hpm_get_counters(void);
59 56 zero_gravi
uint32_t neorv32_cpu_hpm_get_size(void);
60 60 zero_gravi
int neorv32_cpu_check_zext(uint8_t flag_id);
61 2 zero_gravi
 
62
 
63
/**********************************************************************//**
64 61 zero_gravi
 * Prototype for "after-main handler". This function is called if main() returns.
65
 *
66
 * @param[in] return_code Return value of main() function.
67
 * @return Return value is irrelevant (there is no one left to check for it...).
68
 **************************************************************************/
69 62 zero_gravi
extern int __neorv32_crt0_after_main(int32_t return_code) __attribute__ ((weak));
70 61 zero_gravi
 
71
 
72
/**********************************************************************//**
73 56 zero_gravi
 * Store unsigned word to address space.
74
 *
75
 * @note An unaligned access address will raise an alignment exception.
76
 *
77
 * @param[in] addr Address (32-bit).
78
 * @param[in] wdata Data word (32-bit) to store.
79 57 zero_gravi
 * @return Operation status (32-bit).
80 56 zero_gravi
 **************************************************************************/
81 57 zero_gravi
inline uint32_t __attribute__ ((always_inline)) neorv32_cpu_store_conditional(uint32_t addr, uint32_t wdata) {
82
 
83
#if defined __riscv_atomic || defined __riscv_a
84
  register uint32_t reg_addr = addr;
85
  register uint32_t reg_data = wdata;
86
  register uint32_t reg_status;
87
 
88
  asm volatile ("sc.w %[status], %[da], (%[ad])" : [status] "=r" (reg_status) : [da] "r" (reg_data), [ad] "r" (reg_addr));
89
 
90
  return reg_status;
91
#else
92
  return 1; // always failing
93
#endif
94
}
95
 
96
 
97
/**********************************************************************//**
98
 * Conditional store unsigned word to address space if atomic access reservation is valid.
99
 *
100
 * @note An unaligned access address will raise an alignment exception.
101
 *
102
 * @param[in] addr Address (32-bit).
103
 * @param[in] wdata Data word (32-bit) to store.
104
 **************************************************************************/
105 56 zero_gravi
inline void __attribute__ ((always_inline)) neorv32_cpu_store_unsigned_word(uint32_t addr, uint32_t wdata) {
106
 
107
  register uint32_t reg_addr = addr;
108
  register uint32_t reg_data = wdata;
109
 
110
  asm volatile ("sw %[da], 0(%[ad])" : : [da] "r" (reg_data), [ad] "r" (reg_addr));
111
}
112
 
113
 
114
/**********************************************************************//**
115
 * Store unsigned half-word to address space.
116
 *
117
 * @note An unaligned access address will raise an alignment exception.
118
 *
119
 * @param[in] addr Address (32-bit).
120
 * @param[in] wdata Data half-word (16-bit) to store.
121
 **************************************************************************/
122
inline void __attribute__ ((always_inline)) neorv32_cpu_store_unsigned_half(uint32_t addr, uint16_t wdata) {
123
 
124
  register uint32_t reg_addr = addr;
125
  register uint32_t reg_data = (uint32_t)wdata;
126
 
127
  asm volatile ("sh %[da], 0(%[ad])" : : [da] "r" (reg_data), [ad] "r" (reg_addr));
128
}
129
 
130
 
131
/**********************************************************************//**
132
 * Store unsigned byte to address space.
133
 *
134
 * @param[in] addr Address (32-bit).
135
 * @param[in] wdata Data byte (8-bit) to store.
136
 **************************************************************************/
137
inline void __attribute__ ((always_inline)) neorv32_cpu_store_unsigned_byte(uint32_t addr, uint8_t wdata) {
138
 
139
  register uint32_t reg_addr = addr;
140
  register uint32_t reg_data = (uint32_t)wdata;
141
 
142
  asm volatile ("sb %[da], 0(%[ad])" : : [da] "r" (reg_data), [ad] "r" (reg_addr));
143
}
144
 
145
 
146
/**********************************************************************//**
147 57 zero_gravi
 * Load unsigned word from address space and make reservation for atomic access.
148
 *
149
 * @note An unaligned access address will raise an alignment exception.
150
 *
151
 * @param[in] addr Address (32-bit).
152
 * @return Read data word (32-bit).
153
 **************************************************************************/
154
inline uint32_t __attribute__ ((always_inline)) neorv32_cpu_load_reservate_word(uint32_t addr) {
155
 
156
  register uint32_t reg_addr = addr;
157
  register uint32_t reg_data;
158
 
159
#if defined __riscv_atomic || defined __riscv_a
160
  asm volatile ("lr.w %[da], 0(%[ad])" : [da] "=r" (reg_data) : [ad] "r" (reg_addr));
161
#else
162
  asm volatile ("lw %[da], 0(%[ad])" : [da] "=r" (reg_data) : [ad] "r" (reg_addr));
163
#endif
164
 
165
  return (uint32_t)reg_data;
166
}
167
 
168
 
169
 
170
/**********************************************************************//**
171 56 zero_gravi
 * Load unsigned word from address space.
172
 *
173
 * @note An unaligned access address will raise an alignment exception.
174
 *
175
 * @param[in] addr Address (32-bit).
176
 * @return Read data word (32-bit).
177
 **************************************************************************/
178
inline uint32_t __attribute__ ((always_inline)) neorv32_cpu_load_unsigned_word(uint32_t addr) {
179
 
180
  register uint32_t reg_addr = addr;
181
  register uint32_t reg_data;
182
 
183
  asm volatile ("lw %[da], 0(%[ad])" : [da] "=r" (reg_data) : [ad] "r" (reg_addr));
184
 
185
  return (uint32_t)reg_data;
186
}
187
 
188
 
189
/**********************************************************************//**
190
 * Load unsigned half-word from address space.
191
 *
192
 * @note An unaligned access address will raise an alignment exception.
193
 *
194
 * @param[in] addr Address (32-bit).
195
 * @return Read data half-word (16-bit).
196
 **************************************************************************/
197
inline uint16_t __attribute__ ((always_inline)) neorv32_cpu_load_unsigned_half(uint32_t addr) {
198
 
199
  register uint32_t reg_addr = addr;
200
  register uint32_t reg_data;
201
 
202
  asm volatile ("lhu %[da], 0(%[ad])" : [da] "=r" (reg_data) : [ad] "r" (reg_addr));
203
 
204
  return (uint16_t)reg_data;
205
}
206
 
207
 
208
/**********************************************************************//**
209
 * Load unsigned byte from address space.
210
 *
211
 * @param[in] addr Address (32-bit).
212
 * @return Read data byte (8-bit).
213
 **************************************************************************/
214
inline uint8_t __attribute__ ((always_inline)) neorv32_cpu_load_unsigned_byte(uint32_t addr) {
215
 
216
  register uint32_t reg_addr = addr;
217
  register uint32_t reg_data;
218
 
219
  asm volatile ("lbu %[da], 0(%[ad])" : [da] "=r" (reg_data) : [ad] "r" (reg_addr));
220
 
221
  return (uint8_t)reg_data;
222
}
223
 
224
 
225
/**********************************************************************//**
226 2 zero_gravi
 * Read data from CPU configuration and status register (CSR).
227
 *
228 42 zero_gravi
 * @param[in] csr_id ID of CSR to read. See #NEORV32_CSR_enum.
229 2 zero_gravi
 * @return Read data (uint32_t).
230
 **************************************************************************/
231
inline uint32_t __attribute__ ((always_inline)) neorv32_cpu_csr_read(const int csr_id) {
232
 
233
  register uint32_t csr_data;
234
 
235 6 zero_gravi
  asm volatile ("csrr %[result], %[input_i]" : [result] "=r" (csr_data) : [input_i] "i" (csr_id));
236 2 zero_gravi
 
237
  return csr_data;
238
}
239
 
240
 
241
/**********************************************************************//**
242
 * Write data to CPU configuration and status register (CSR).
243
 *
244 42 zero_gravi
 * @param[in] csr_id ID of CSR to write. See #NEORV32_CSR_enum.
245 2 zero_gravi
 * @param[in] data Data to write (uint32_t).
246
 **************************************************************************/
247
inline void __attribute__ ((always_inline)) neorv32_cpu_csr_write(const int csr_id, uint32_t data) {
248
 
249
  register uint32_t csr_data = data;
250
 
251 6 zero_gravi
  asm volatile ("csrw %[input_i], %[input_j]" :  : [input_i] "i" (csr_id), [input_j] "r" (csr_data));
252 2 zero_gravi
}
253
 
254 9 zero_gravi
 
255
/**********************************************************************//**
256
 * Put CPU into "sleep" mode.
257
 *
258
 * @note This function executes the WFI insstruction.
259
 * The WFI (wait for interrupt) instruction will make the CPU stall until
260
 * an interupt request is detected. Interrupts have to be globally enabled
261 56 zero_gravi
 * and at least one external source must be enabled (like the MTI machine
262
 * timer interrupt) to allow the CPU to wake up again. If 'Zicsr' CPU extension is disabled,
263 9 zero_gravi
 * this will permanently stall the CPU.
264
 **************************************************************************/
265
inline void __attribute__ ((always_inline)) neorv32_cpu_sleep(void) {
266
 
267
  asm volatile ("wfi");
268
}
269
 
270
 
271
/**********************************************************************//**
272
 * Enable global CPU interrupts (via MIE flag in mstatus CSR).
273
 **************************************************************************/
274
inline void __attribute__ ((always_inline)) neorv32_cpu_eint(void) {
275
 
276 42 zero_gravi
  asm volatile ("csrrsi zero, mstatus, %0" : : "i" (1 << CSR_MSTATUS_MIE));
277 40 zero_gravi
  asm volatile ("nop");
278
  asm volatile ("nop");
279 9 zero_gravi
}
280
 
281
 
282
/**********************************************************************//**
283
 * Disable global CPU interrupts (via MIE flag in mstatus CSR).
284
 **************************************************************************/
285
inline void __attribute__ ((always_inline)) neorv32_cpu_dint(void) {
286
 
287 42 zero_gravi
  asm volatile ("csrrci zero, mstatus, %0" : : "i" (1 << CSR_MSTATUS_MIE));
288 40 zero_gravi
  asm volatile ("nop");
289
  asm volatile ("nop");
290 9 zero_gravi
}
291
 
292
 
293
/**********************************************************************//**
294
 * Trigger breakpoint exception (via EBREAK instruction).
295
 **************************************************************************/
296
inline void __attribute__ ((always_inline)) neorv32_cpu_breakpoint(void) {
297
 
298
  asm volatile ("ebreak");
299
}
300
 
301
 
302
/**********************************************************************//**
303
 * Trigger "environment call" exception (via ECALL instruction).
304
 **************************************************************************/
305
inline void __attribute__ ((always_inline)) neorv32_cpu_env_call(void) {
306
 
307
  asm volatile ("ecall");
308
}
309
 
310
 
311 2 zero_gravi
#endif // neorv32_cpu_h

powered by: WebSVN 2.1.0

© copyright 1999-2024 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.