Skip to content

KULeuven-MICAS/snax-mlir

Repository files navigation

SNAX-MLIR

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]
Loading

Example input MLIR is available in the kernels folder, it contains:

  • Handwritten MLIR tests, mainly linalg on memrefs.
  • 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
Loading

Requirements

This repository has various requirements based on what you want to do:

  • Development with snax-opt on xdsl.
  • 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 need tensorflow-cpu (only required for importing MLPerf Tiny Benchmarks in MLIR, installed with pip install '-e.[nn]'), clang, mlir-opt, mlir-translate and make.
  • 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.

Setup with Docker Container

Getting the container remotely

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

Building the container locally (optional)

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

Running examples on SNAX

Run Snax Kernels

Inside the docker container:

cd /repo/kernels/simple_mult
make allrun

This will compile main.c two different kernels:

  1. baseline.c: A C implementation of the kernel
  2. linalg.mlir: An MLIR Linalg implementation of the kernel

The exact invocations of all the compilers can be seen in the terminal

Inspect traces for snax kernels

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.

Inspect disassembly for generated binaries

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.