am: add loongarch32r-nemu

This commit is contained in:
Zihao Yu 2022-11-15 15:24:49 +08:00
parent ff88f398bc
commit 67699be876
9 changed files with 201 additions and 0 deletions

View file

@ -0,0 +1,16 @@
#ifndef __ARCH_H__
#define __ARCH_H__
struct Context {
// TODO: fix the order of these members to match trap.S
uintptr_t gpr[32], era, estat, prmd;
void *pdir;
};
#define GPR1 gpr[11] // a7
#define GPR2 gpr[0]
#define GPR3 gpr[0]
#define GPR4 gpr[0]
#define GPRx gpr[0]
#endif

View file

@ -0,0 +1,24 @@
#ifndef LOONGARCH32R_H__
#define LOONGARCH32R_H__
#include <stdint.h>
static inline uint8_t inb(uintptr_t addr) { return *(volatile uint8_t *)addr; }
static inline uint16_t inw(uintptr_t addr) { return *(volatile uint16_t *)addr; }
static inline uint32_t inl(uintptr_t addr) { return *(volatile uint32_t *)addr; }
static inline void outb(uintptr_t addr, uint8_t data) { *(volatile uint8_t *)addr = data; }
static inline void outw(uintptr_t addr, uint16_t data) { *(volatile uint16_t *)addr = data; }
static inline void outl(uintptr_t addr, uint32_t data) { *(volatile uint32_t *)addr = data; }
#define PTE_V 0x1
#define PTE_D 0x2
// Page directory and page table constants
#define PTXSHFT 12 // Offset of PTX in a linear address
#define PDXSHFT 22 // Offset of PDX in a linear address
#define PDX(va) (((uint32_t)(va) >> PDXSHFT) & 0x3ff)
#define PTX(va) (((uint32_t)(va) >> PTXSHFT) & 0x3ff)
#endif

View file

@ -0,0 +1,47 @@
#include <am.h>
#include <loongarch/loongarch32r.h>
#include <klib.h>
static Context* (*user_handler)(Event, Context*) = NULL;
Context* __am_irq_handle(Context *c) {
if (user_handler) {
Event ev = {0};
uintptr_t ecode = 0;
switch (ccode) {
default: ev.event = EVENT_ERROR; break;
}
c = user_handler(ev, c);
assert(c != NULL);
}
return c;
}
extern void __am_asm_trap(void);
bool cte_init(Context*(*handler)(Event, Context*)) {
// initialize exception entry
asm volatile("csrwr %0, 0xc" : : "r"(__am_asm_trap)); // 0xc = eentry
// register event handler
user_handler = handler;
return true;
}
Context *kcontext(Area kstack, void (*entry)(void *), void *arg) {
return NULL;
}
void yield() {
asm volatile("li.w $a7, -1; syscall 0");
}
bool ienabled() {
return false;
}
void iset(bool enable) {
}

View file

@ -0,0 +1,8 @@
.section entry, "ax"
.globl _start
.type _start, @function
_start:
move $fp, $zero
la.local $sp, _stack_pointer
bl _trm_init

View file

@ -0,0 +1,50 @@
#define concat_temp(x, y) x ## y
#define concat(x, y) concat_temp(x, y)
#define MAP(c, f) c(f)
#define REGS(f) \
f( 1) f( 2) f( 4) f( 5) f( 6) f( 7) f( 8) f( 9) \
f(10) f(11) f(12) f(13) f(14) f(15) f(16) f(17) f(18) f(19) \
f(20) f(21) f(22) f(23) f(24) f(25) f(26) f(27) f(28) f(29) \
f(30) f(31)
#define PUSH(n) st.w $concat(r, n), $sp, (n * 4);
#define POP(n) ld.w $concat(r, n), $sp, (n * 4);
#define CONTEXT_SIZE ((32 + 3) * 4)
#define OFFSET_SP ( 3 * 4)
#define OFFSET_ESTAT (32 * 4)
#define OFFSET_PRMD (33 * 4)
#define OFFSET_ERA (34 * 4)
#define CSR_ESTAT 0x5
#define CSR_PRMD 0x1
#define CSR_ERA 0x6
.align 6
.globl __am_asm_trap
__am_asm_trap:
addi.w $sp, $sp, -CONTEXT_SIZE
MAP(REGS, PUSH)
csrrd $t0, CSR_ESTAT
csrrd $t1, CSR_PRMD
csrrd $t2, CSR_ERA
st.w $t0, $sp, OFFSET_ESTAT
st.w $t1, $sp, OFFSET_PRMD
st.w $t2, $sp, OFFSET_ERA
move $a0, $sp
bl __am_irq_handle
ld.w $t1, $sp, OFFSET_PRMD
ld.w $t2, $sp, OFFSET_ERA
csrwr $t1, CSR_PRMD
csrwr $t2, CSR_ERA
MAP(REGS, POP)
addi.w $sp, $sp, CONTEXT_SIZE
ertn

View file

@ -0,0 +1,42 @@
#include <am.h>
#include <nemu.h>
#define USER_SPACE RANGE(0x40000000, 0x80000000)
static void* (*pgalloc_usr)(int) = NULL;
static void (*pgfree_usr)(void*) = NULL;
static int vme_enable = 0;
static PTE *cur_pdir = NULL;
bool vme_init(void* (*pgalloc_f)(int), void (*pgfree_f)(void*)) {
pgalloc_usr = pgalloc_f;
pgfree_usr = pgfree_f;
vme_enable = 1;
return true;
}
void protect(AddrSpace *as) {
as->ptr = (PTE*)(pgalloc_usr(PGSIZE));
as->pgsize = PGSIZE;
as->area = USER_SPACE;
}
void unprotect(AddrSpace *as) {
}
void __am_get_cur_as(Context *c) {
c->pdir = (vme_enable ? cur_pdir : NULL);
}
void __am_switch(Context *c) {
if (vme_enable && c->pdir != NULL) {
cur_pdir = c->pdir;
}
}
void map(AddrSpace *as, void *va, void *pa, int prot) {
}
Context *ucontext(AddrSpace *as, Area kstack, void *entry) {
return NULL;
}

View file

@ -12,6 +12,8 @@
# define nemu_trap(code) asm volatile ("move $v0, %0; sdbbp" : :"r"(code))
#elif defined(__ISA_RISCV32__) || defined(__ISA_RISCV64__)
# define nemu_trap(code) asm volatile("mv a0, %0; ebreak" : :"r"(code))
#elif defined(__ISA_LOONGARCH32R__)
# define nemu_trap(code) asm volatile("move $a0, %0; break 0" : :"r"(code))
#elif
# error unsupported ISA __ISA__
#endif

View file

@ -0,0 +1,4 @@
CROSS_COMPILE := loongarch32r-linux-gnusf-
COMMON_FLAGS := -fno-pic
CFLAGS += $(COMMON_FLAGS) -static
ASFLAGS += $(COMMON_FLAGS) -O0

View file

@ -0,0 +1,8 @@
include $(AM_HOME)/scripts/isa/loongarch32r.mk
include $(AM_HOME)/scripts/platform/nemu.mk
CFLAGS += -DISA_H=\"loongarch/loongarch32r.h\"
AM_SRCS += loongarch/nemu/start.S \
loongarch/nemu/cte.c \
loongarch/nemu/trap.S \
loongarch/nemu/vme.c