kernel-mode #1

Manually merged
anton merged 69 commits from kernel-mode into master 2021-11-16 08:02:40 +01:00
Showing only changes of commit 60a2a8d546 - Show all commits

View File

@ -7,6 +7,7 @@ from ..colors import FMT_CSR, FMT_NONE
from .CSRConsts import CSR_NAME_TO_ADDR, MSTATUS_LEN_2, MSTATUS_OFFSETS from .CSRConsts import CSR_NAME_TO_ADDR, MSTATUS_LEN_2, MSTATUS_OFFSETS
class CSR: class CSR:
""" """
This holds all Control and Status Registers (CSR) This holds all Control and Status Registers (CSR)
@ -21,13 +22,16 @@ class CSR:
list of virtual CSR registers, with values computed on read list of virtual CSR registers, with values computed on read
""" """
listeners: Dict[int, Callable[[int, int], None]] listeners: Dict[int, Callable[[int, int], None]]
mstatus_cache: Dict[str, int]
mstatus_cache_dirty = True
def __init__(self): def __init__(self):
self.regs = defaultdict(lambda: 0) self.regs = defaultdict(lambda: 0)
self.listeners = defaultdict(lambda: (lambda x, y: None)) self.listeners = defaultdict(lambda: (lambda x, y: None))
self.virtual_regs = dict() self.virtual_regs = dict()
self.mstatus_cache = dict()
# TODO: implement write masks (bitmasks which control writeable bits in registers # TODO: implement write masks (bitmasks which control writeable bits in registers
def set(self, addr: Union[str, int], val: int): def set(self, addr: Union[str, int], val: int):
@ -36,6 +40,8 @@ class CSR:
return return
val = to_unsigned(val) val = to_unsigned(val)
self.listeners[addr](self.regs[addr], val) self.listeners[addr](self.regs[addr], val)
if addr == 0x300:
self.mstatus_cache_dirty = True
self.regs[addr] = val self.regs[addr] = val
def get(self, addr: Union[str, int]) -> int: def get(self, addr: Union[str, int]) -> int:
@ -74,15 +80,24 @@ class CSR:
self.set('mstatus', new_val) self.set('mstatus', new_val)
def get_mstatus(self, name) -> int: def get_mstatus(self, name) -> int:
if not self.mstatus_cache_dirty and name in self.mstatus_cache:
return self.mstatus_cache[name]
size = 2 if name in MSTATUS_LEN_2 else 1 size = 2 if name in MSTATUS_LEN_2 else 1
off = MSTATUS_OFFSETS[name] off = MSTATUS_OFFSETS[name]
mask = (2 ** size - 1) << off mask = (2 ** size - 1) << off
return (self.get('mstatus') & mask) >> off val = (self.get('mstatus') & mask) >> off
if self.mstatus_cache_dirty:
self.mstatus_cache = dict(name=val)
else:
self.mstatus_cache[name] = val
return val
def callback(self, addr: Union[str, int]): def callback(self, addr: Union[str, int]):
def inner(func: Callable[[int, int], None]): def inner(func: Callable[[int, int], None]):
self.set_listener(addr, func) self.set_listener(addr, func)
return func return func
return inner return inner
def assert_can_read(self, mode: PrivModes, addr: int): def assert_can_read(self, mode: PrivModes, addr: int):