Repository for research on an MLIR-based compiler for open-source RISC-V based heterogeneous accelerators platforms developed by KULeuven and friends.
Here snax-opt
is developed, an MLIR compiler in python, based on xdsl.
The open-source hardware that snax-opt
is targetting is available over at KULeuven-MICAS/snax_cluster
Please note that this repository is currently under heavy development and not ready for general consumption. We very happily welcome active collaborations and pull requests though!
General flow:
graph LR
A[Input MLIR] --> B(mlir-opt) --> C(snax-opt) --> D(mlir-opt)--> E(mlir-translate)
B --> D
E --> F(clang) --> G[RISC-V SNAX Executable]
Example input MLIR is available in the kernels
folder, it contains:
- Handwritten MLIR tests, mainly
linalg
onmemref
s. - Scripts to download and build MLPerf Tiny Benchmark tests.
The output binary can be run on a SNAX Cluster RTL simulator through verilator. These simulator builds are developed and maintained over at KULeuven-MICAS/snax_cluster, and some prebuilt systems are shipped with the docker container.
graph LR
A[SNAX platform 1\n executable A] --> B[RISC-V SNAX platform 1\nVerilated model]
C[SNAX platform 1\n executable B] --> B
D[SNAX platform 2\n executable C] --> E[RISC-V SNAX platform 2\nVerilated model]
F[SNAX platform 2\n executable D] --> E
This repository has various requirements based on what you want to do:
- Development with
snax-opt
onxdsl
. - For testing, the required packages can be installed with `pip install '-e.[dev]'. Currently all tests are performed with python 3.10.
- End-to-end compilation of kernels requires all the above packages, and a per-platform customized version of the
snitch_runtime
(libSnRuntime.a
). Furthermore you will needtensorflow-cpu
(only required for importing MLPerf Tiny Benchmarks in MLIR, installed withpip install '-e.[nn]'
),clang
,mlir-opt
,mlir-translate
andmake
. - Running RTL simulations of compiled kernels requires all above dependences, and a build of a verilated model.
The requirements in this case are specified at KULeuven-MICAS/snax_cluster.
Tracing requires a custom version of
spike-dasm
.
Unless you are an expert user, we recommend you to use our monolithic docker setup outlined below. This setup contains all dependencies in the right versions to perform both hardware and software development on SNAX.
You can run tests/experiments in this repository with docker.
For <docker-version>
we recommend taking note of the docker version currently used in all continuous integration tests (see .github/workflows/build-and-run-kernels.yml
).
docker run -itv `pwd`:/repo:z ghcr.io/kuleuven-micas/snax:<docker-version>
This will download the image if it is not present on your system yet.
The repository will be available under /repo
inside the container.
To use the snax-opt
command you need to additionaly install it with pip:
pip install -e /repo
Make sure all python dependencies are up-to-date with:
cd /repo
pip install -r requirements.txt
To build the container locally, you can get the container from the hardware repository with the following commands:
git clone git@github.com:kuleuven-micas/snax_cluster
cd snax_cluster
docker build -t ghcr.io/kuleuven-micas/snax:main -f util/container/Dockerfile .
Then you can run the experiments with the above docker run
command
Inside the docker container:
cd /repo/kernels/simple_mult
make allrun
This will compile main.c
two different kernel
s:
baseline.c
: A C implementation of the kernellinalg.mlir
: An MLIR Linalg implementation of the kernel
The exact invocations of all the compilers can be seen in the terminal
Tracing tracks individual instructions as they are executed by the RISC-V cores in the snax-cluster.
Therefore tracing requires running a program with a tracer.
The default allrun
recipe in the makefile runs all examples with a tracer.
To convert the machine-readable traces to human-readable format, use
Inside the docker container:
cd /kernels/simple_mult
make allrun # If you haven't ran the kernels before
make traces
Human readable traces are put in a .logs
directory with the same name as the kernel binary.
Statistics are computed for each section of the program execution.
Note: In this context a section means a part of the trace as delimited by an mcycle
instruction.
E.g. calling mcycle
in a program once will yield two sections, one before the mcycle instruction and one after.
Disassembly is the conversion of the compiled binary to human-readable form. In this way you can inspect the program the way it is put into the memory.
cd /kernels/simple_mult
make baseline.o # Make an object file
llvm-objdump-17 -d baseline.o
As you can see, disassembly does not require running the program.
Note: The dissassembly might show multiple "sections". In this context, a section is a unit of information in an ELF-file.
E.g. the .text
section will container your program and the .data
section will contain your static data.