# Rust bindings for LLZK.
> [!WARNING]
> These crates are under active development and things may change unexpectedly.
Rust bindings for [LLZK](https://project-llzk.github.io/llzk-lib/) over its C API.
The bindings' API is meant to be more user friendly than the raw C API and more idiomatic. Its design is heavily inspired
by [melior](https://github.com/mlir-rs/melior) and depends on it for handling the MLIR parts that are not
specific to LLZK.
The primary supported use case of these bindings is creating IR and running passes on it. Support for other things,
such as writing custom passes, is limited and not as ergonomic as it is in C++.
## Usage
Run `cargo doc -p llzk` to generate the API documentation of the rust API and you can visit
[LLZK's documentation](https://project-llzk.github.io/llzk-lib/) for more information about the IR itself.
For the high-level usage of the bindings you can check the examples in `llzk/examples`.
### Optional features
We include some optional functionality guarded by feature flags. We currently have the following features:
- `bigint`: Allows creating constant values from [`num-bigint`'s Big integers](https://docs.rs/num-bigint/latest/num_bigint/struct.BigUint.html).
## Manual installation
### Prerequisites
Install LLVM 20 and note the installation path. While building your project the build scripts will look for LLVM using `llvm-config`.
If you don't have that tool in your `PATH` or it doesn't point to an LLVM 20 installation set the following environment variables
to the path where LLVM is installed and the build scripts will then use `$MLIR_SYS_200_PREFIX/bin/llvm-config` instead.
```text
export MLIR_SYS_200_PREFIX=/path/to/llvm/20/
export TABLEGEN_200_PREFIX=/path/to/llvm/20/
```
### Building PCL
Clone and build the `pcl-mlir` component at a known-good commit:
```sh
git clone https://github.com/Veridise/pcl-mlir.git
cd pcl-mlir && git checkout 55cf619b032314198aacafc305871fb66b12b70e && cd ..
```
Build with CMake:
```sh
cmake -S pcl-mlir -B pcl-mlir/build \
-DCMAKE_BUILD_TYPE=Debug \
-DBUILD_TESTING=OFF \
-DCMAKE_PREFIX_PATH=$MLIR_SYS_200_PREFIX \
-DCMAKE_INSTALL_PREFIX=$(pwd)/pcl-mlir/build
cmake --build pcl-mlir/build
cmake --install pcl-mlir/build
```
Then set the following environment variables to point to the source and build directories:
```text
export LLZK_PCL_ROOT=/path/to/pcl-mlir
export LLZK_PCL_PREFIX=/path/to/pcl-mlir/build
```
### Building LLZK
Clone the [LLZK library](https://github.com/project-llzk/llzk-lib) and build it:
If you want to run LLZK's tests when building, you also need [lit](https://pypi.org/project/lit/) (LLVM's test runner):
```text
pip install lit
```
```sh
git clone https://github.com/project-llzk/llzk-lib.git
cmake -B llzk-lib/out/build -S llzk-lib \
-DCMAKE_INSTALL_PREFIX=$(pwd)/llzk-lib/out \
-DCMAKE_PREFIX_PATH="$MLIR_SYS_200_PREFIX;$LLZK_PCL_PREFIX"
cmake --build llzk-lib/out/build
cmake --install llzk-lib/out/build
```
Then set the `LLZK_SYS_10_PREFIX` environment variable to point to the install location:
```text
export LLZK_SYS_10_PREFIX=/path/to/llzk-lib/out
```
### Adding the crates to your project
In your rust project, add the crates to your Cargo.toml:
```text
llzk-sys = { git = "https://github.com/project-llzk/llzk-rs" }
llzk = { git = "https://github.com/project-llzk/llzk-rs" }
```
### Building tips
If you are using homebrew in macos you can access MLIR 20 by installing `llvm@20` with homebrew.
Setting the following environment variables configures the build system with the correct versions of MLIR and its dependencies.
Depending on the version of your default C++ compiler you may need to set `CXX` and `CC` to a compiler that supports C++ 20.
```text
export MLIR_SYS_200_PREFIX=$(brew --prefix llvm@20)
export TABLEGEN_200_PREFIX=$(brew --prefix llvm@20)
export LIBCLANG_PATH=$(brew --prefix llvm@20)/lib
export CXX=clang++
export CC=clang
export LLZK_PCL_ROOT=/path/to/pcl-mlir
export LLZK_PCL_PREFIX=/path/to/pcl-mlir/build
export RUSTFLAGS='-L /opt/homebrew/lib/'
```
See [`llzk-sys`'s README](llzk-sys/README.md) for more details on setting up the build environment.
## Nix installation
We also include a nix flake that creates an environment with the right versions of LLVM, MLIR, and PCL.
All dependencies, including the correct `pcl-mlir` commit, are pinned in `flake.lock` and set up automatically.
If you are already using nix, this may be your preferred method.
You can use this flake for configuring your development environment.
For example, to work within a nix developer shell you can use the following command.
```text
nix develop 'github:project-llzk/llzk-rs#llzk-rs'
```
Another alternative is to use [direnv](https://direnv.net/) with the following `.envrc` to automatically enter
the developer environment when you enter your project's directory.
```text
fi
use flake 'github:project-llzk/llzk-rs#llzk-rs'
```
## Updating LLZK
### Mmanual installation
If you need to update the llzk-lib dependency, pull the latest changes from the [llzk-lib repository](https://github.com/project-llzk/llzk-lib) and rebuild it.
### Nix installation
To update the llzk-lib dependency run `nix flake update llzk-lib`