LIU Yu <[email protected]>
2024/08/29 (v0.0.4)
This project provides Python bindings for the Spike RISC-V ISA Simulator. The Pythonic Spike (or PySpike) opens up Spike's C++ internals (such as RISC-V disassembler, processors, controllers, peripherals, etc.) for interoperation with Python scripts. It enables users to write ISA / RoCC extensions and MMIO device models in Python, and plug them into vanilla Spike for (co-)simulating complex hardware systems. Through integrating Spike more seamlessly into the Python ecosystem, PySpike aims to boost the agility of Python-based hardware verification tools and workflows.
PyPI package: N/A (candidate name spike
pending PEP 541)
PySpike requires: Python 3.8+ and vanilla Spike (commit 37b0dc0 or later).
- Build from source code, preferrably in a virtual environment.
(.venv) $ python -m build --no-isolation
...
Successfully built spike-0.0.5.dev2.tar.gz and spike-0.0.5.dev2-cp38-cp38-linux_x86_64.whl
- Install the wheel package with
pip
.
(.venv) $ pip install spike-0.0.5.dev2-cp38-cp38-linux_x86_64.whl
...
Successfully installed spike-0.0.5.dev2
- Check that you installed the correct version
(.venv) $ pyspike --help
Spike RISC-V ISA Simulator 1.1.1-dev
...
An ISA extension in PySpike is a class that inherits riscv.isa.ISA
. It should implement a minimum of two methods: get_instructions
and get_disasms
. The former provides functional models of one or more RISC-V instructions, while the latter provides their disassemblers. A special decorator @isa.register("myisa")
is used to register the extension under the name myisa
.
from typing import List
from riscv import isa
from riscv.disasm import disasm_insn_t
from riscv.processor import insn_desc_t
@isa.register("myisa")
class MyISA(isa.ISA):
def __init__(self): ...
def get_instructions(self) -> List[insn_desc_t]: ...
def get_disasms(self) -> List[disasm_insn_t]: ...
def reset(self) -> None: ...
Likewise to the ISA extension, an MMIO model in PySpike is a class that inherits riscv.dev.MMIO
. It should implement a minimum of three methods: __init__
, load
, and store
. The former initializes the model, the latter two handle memory read and write operations. A special decorator @dev.register("mydev")
is used to register the model under the name mydev
.
from typing import Optional
from riscv import dev
from riscv.sim import sim_t
@dev.register("mydev")
class MyDEV(dev.MMIO):
def __init__(self, sim: sim_t, args: Optional[str] = None): ...
def load(self, addr: int, size: int) -> bytes: ...
def store(self, addr: int, data: bytes) -> None: ...
def tick(self, rtc_ticks: int) -> None:
PySpike provides a command-line wrapper called pyspike
. It is 100%-compatible to the command-line interface of vanilla Spike, with additional support for Python-based MMIO / RoCC extensions via --extlib=<name>
.
$ pyspike \
--isa=rv32imc_xmyisa --priv=m \
--pc=0x90000000 \
-m0x90000000:0x4000000 \
--extlib=myisa.py \
--extlib=mydev.py \
--device=mydev,0x20000000 \
tests/data/libc-printf_hello.elf
Hello, World!
$ python -m venv .venv
$ source .venv/bin/activate
(.venv) $ pip install -r requirements.txt
(.venv) $ python setup.py build_ext --inplace
(.venv) $ pytest -v
(.venv) $ python -m build