123 lines
2.4 KiB
C
123 lines
2.4 KiB
C
|
//See LICENSE for license details.
|
||
|
#include <stdint.h>
|
||
|
#include <stdio.h>
|
||
|
#include <unistd.h>
|
||
|
|
||
|
#include "platform.h"
|
||
|
#include "encoding.h"
|
||
|
|
||
|
#define CPU_FREQ 65000000
|
||
|
#define XSTR(x) #x
|
||
|
#define STR(x) XSTR(x)
|
||
|
|
||
|
#ifndef VECT_IRQ
|
||
|
#define TRAP_ENTRY trap_entry
|
||
|
#else
|
||
|
#define TRAP_ENTRY vtrap_entry
|
||
|
#endif
|
||
|
|
||
|
extern int main(int argc, char** argv);
|
||
|
extern void TRAP_ENTRY();
|
||
|
|
||
|
unsigned long get_cpu_freq()
|
||
|
{
|
||
|
return CPU_FREQ;
|
||
|
}
|
||
|
|
||
|
unsigned long get_timer_freq()
|
||
|
{
|
||
|
return get_cpu_freq();
|
||
|
}
|
||
|
|
||
|
uint64_t get_timer_value()
|
||
|
{
|
||
|
#if __riscv_xlen == 32
|
||
|
while (1) {
|
||
|
uint32_t hi = read_csr(mcycleh);
|
||
|
uint32_t lo = read_csr(mcycle);
|
||
|
if (hi == read_csr(mcycleh))
|
||
|
return ((uint64_t)hi << 32) | lo;
|
||
|
}
|
||
|
#else
|
||
|
return read_csr(mcycle);
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
static void uart_init(size_t baud_rate)
|
||
|
{
|
||
|
UART0_REG(UART_REG_DIV) = (get_cpu_freq() / 2) / baud_rate - 1;
|
||
|
UART0_REG(UART_REG_TXCTRL) |= UART_TXEN;
|
||
|
}
|
||
|
|
||
|
|
||
|
#ifdef USE_PLIC
|
||
|
extern void handle_m_ext_interrupt();
|
||
|
#endif
|
||
|
|
||
|
#ifdef USE_M_TIME
|
||
|
extern void handle_m_time_interrupt();
|
||
|
#endif
|
||
|
|
||
|
#ifdef USE_LOCAL_ISR
|
||
|
typedef void (*my_interrupt_function_ptr_t) (void);
|
||
|
extern my_interrupt_function_ptr_t localISR[];
|
||
|
#endif
|
||
|
|
||
|
#ifndef VECT_IRQ
|
||
|
uintptr_t handle_trap(uintptr_t mcause, uintptr_t epc) __attribute__((noinline));
|
||
|
uintptr_t handle_trap(uintptr_t mcause, uintptr_t epc)
|
||
|
{
|
||
|
if (0){
|
||
|
#ifdef USE_PLIC
|
||
|
// External Machine-Level interrupt from PLIC
|
||
|
} else if ((mcause & MCAUSE_INT) && ((mcause & MCAUSE_CAUSE) == IRQ_M_EXT)) {
|
||
|
handle_m_ext_interrupt();
|
||
|
#endif
|
||
|
#ifdef USE_M_TIME
|
||
|
// External Machine-Level interrupt from PLIC
|
||
|
} else if ((mcause & MCAUSE_INT) && ((mcause & MCAUSE_CAUSE) == IRQ_M_TIMER)){
|
||
|
handle_m_time_interrupt();
|
||
|
#endif
|
||
|
#ifdef USE_LOCAL_ISR
|
||
|
} else if (mcause & MCAUSE_INT) {
|
||
|
localISR[mcause & MCAUSE_CAUSE] ();
|
||
|
#endif
|
||
|
}
|
||
|
else {
|
||
|
write(1, "Unhandled Trap:\n", 16);
|
||
|
_exit(1 + mcause);
|
||
|
}
|
||
|
return epc;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
#ifdef USE_CLIC
|
||
|
void trap_entry(void) __attribute__((interrupt("SiFive-CLIC-preemptible"), aligned(64)));
|
||
|
void trap_entry(void)
|
||
|
{
|
||
|
unsigned long mcause = read_csr(mcause);
|
||
|
unsigned long mepc = read_csr(mepc);
|
||
|
handle_trap(mcause, mepc);
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
void _init()
|
||
|
{
|
||
|
#ifndef NO_INIT
|
||
|
uart_init(115200);
|
||
|
|
||
|
puts("core freq at " STR(CPU_FREQ) " Hz\n");
|
||
|
|
||
|
#ifdef USE_CLIC
|
||
|
write_csr(mtvec, ((unsigned long)&trap_entry | MTVEC_CLIC));
|
||
|
#else
|
||
|
write_csr(mtvec, ((unsigned long)&TRAP_ENTRY | MTVEC_VECTORED));
|
||
|
#endif
|
||
|
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
void _fini()
|
||
|
{
|
||
|
}
|