ZKsync Era is a layer 2 rollup that uses zero-knowledge proofs to scale Ethereum without compromising on security or decentralization. As it's EVM-compatible (with Solidity/Vyper), 99% of Ethereum projects can redeploy without needing to refactor or re-audit any code. ZKsync Era also uses an LLVM-based compiler that will eventually enable developers to write smart contracts in popular languages such as C++ and Rust.
The era-compiler-tester
integration test framework runs tests for Matter Labs compilers which target the EraVM,
for supported languages listed below. It compiles source code via external API calls,
e.g. to Inkwell. In software quality assurance jargon,
this makes it a whitebox testing framework.
The era-compiler-tester
repository includes the Compiler Tests Collection repository as a submodule.
By default, the Tester SHOULD run the entire Collection in all possible combinations of compiler versions and settings, but it MAY omit some subset of the combinations for the sake of saving time, e.g. when only front-end changes have been made, and there is no point in running tests in all LLVM optimization modes.
1. Install the system prerequisites.
-
Linux (Debian):
Install the following packages:
apt install cmake ninja-build curl git libssl-dev pkg-config clang lld
-
Linux (Arch):
Install the following packages:
pacman -Syu which cmake ninja curl git pkg-config clang lld
-
MacOS:
-
Install the HomeBrew package manager.
-
Install the following packages:
brew install cmake ninja coreutils
-
Install your choice of a recent LLVM/Clang compiler, e.g. via Xcode, Apple’s Command Line Tools, or your preferred package manager.
-
2. Install Rust.
-
Follow the latest [official instructions](https://www.rust-lang.org/tools/install:
curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh . ${HOME}/.cargo/env
Currently we are not pinned to any specific version of Rust, so just install the latest stable build for your platform.
3. Checkout or clone the repository.
-
If you have not cloned this repository yet:
git clone https://github.com/matter-labs/era-compiler-tester.git --recursive
-
If you have already cloned this repository:
git submodule update --init --recursive --remote
4. Build ZKsync LLVM framework.
-
Install the builder using
cargo
:cargo install compiler-llvm-builder
The builder is not the ZKsync LLVM framework itself, but a tool that clones its repository and runs a sequence of build commands. By default it is installed in
~/.cargo/bin/
, which is recommended to be added to your$PATH
. -
Clone and build the ZKsync LLVM framework using the
zksync-llvm
tool:zksync-llvm clone zksync-llvm build
-
If you have already cloned the LLVM repository:
zksync-llvm checkout zksync-llvm build
-
If you would like to use your local LLVM build:
export LLVM_SYS_170_PREFIX='<ABSOLUTE_PATH_TO_YOUR_LOCAL_LLVM_BUILD>'
5. Build compiler executables.
6. Build the main application.
- Build era-compiler-tester with
cargo
:cargo build --release
When the build succeeds, you can run the tests using the examples below.
The era-compiler-tester
is integrated into the GitHub Actions workflows of the following projects:
To allow testing custom FE and VM changes in Pull Requests (PRs) of these repositories, two additional tags are supported:
era-compiler-llvm-test
era-solidity-test
If these tags exist, the tester from these tags will be used by the workflows instead of the default main
branch.
When testing is done, these tags should be removed.
- Solidity
- Yul
- Vyper
- LLVM IR
- EraVM assembly
- LLVM middle-end optimizer (levels 0 to 3, s, z, e.g.
M0
,Mz
etc.) - LLVM back-end optimizer (levels 0 and 3, i.e.
B0
andB3
) solc
optimizer (-
or+
)vyper
optimizer (-
or+
)
- Yul pure (
Y
) - EVM assembly from Yul (
I
) - EVM assembly pure (
E
) - Vyper LLL (
V
)
>=0.8
for compiling Solidity via Yul>=0.8.13
for compiling Solidity via EVM assembly from Yul- [0.4.10; latest] for compiling Solidity via EVM assembly
- [0.3.3, 0.3.9] for compiling Vyper via LLL IR
Currently only relevant for the Solidity compiler, where you can choose the IR:
- Yul (preferred for Solidity ≥0.8)
- EVM (supports Solidity ≥0.4)
Most of the specifiers support wildcards *
(any), ^
('3' and 'z').
With no mode argument, iterates over all option combinations (approximately 800).
Each command assumes you are at the root of the compiler-tester
repository.
cargo run --release --bin compiler-tester -- [-v] [-D] [-T[T]] \
[--path="${PATH}"]* \
[--mode="${MODE}"]*
There are more rarely used options, which you may check out with ./target/release/compiler-tester --help
.
Run a simple Solidity test, dumping Yul, unoptimized and optimized LLVM IR, and EraVM assembly to the specified directory.
Use:
- Yul as the Solidity IR (
Y
) - Yul optimizations enabled (
+
) - level 3 optimizations in LLVM middle-end (
M3
) - level 3 optimizations in LLVM back-end (
B3
) - Solidity compiler version (
0.8.26
)
Output:
- failed and invalid tests only (absence of
-v
) - the compiler debug data to the
./debug/
directory (-D
) - the VM trace data to the
./trace/
directory (-T
)
cargo run --release --bin compiler-tester -- -DT \
--path='tests/solidity/simple/default.sol' \
--mode='Y+M3B3 0.8.26' \
--zksolc '../era-compiler-solidity/target/release/zksolc'
Modes are insensitive to spaces, therefore options such as 'Y+M3B3 0.8.26'
and 'Y + M3B3 0.8.26'
are equivalent.
Run all simple Yul tests. This currently runs about three hundred tests and takes about eight minutes.
Use:
- level 1 optimizations in LLVM middle-end (
M1
) - level 2 optimizations in LLVM back-end (
B2
)
Output:
- all tests, passed and failed (
-v
) - the VM trace data to the
./trace/
directory (-T
)
cargo run --release --bin compiler-tester -- -vT \
--path='tests/yul/' \
--mode='M1B2'
Run all tests (currently about three million) in all modes. This takes a few hours on the CI server, and probably much longer on your personal machine.
cargo run --release --bin compiler-tester -- \
--zksolc '../era-compiler-solidity/target/release/zksolc' \
--zkvyper '../era-compiler-vyper/target/release/zkvyper'
- Change the LLVM branch to the base in the
LLVM.lock
file at the repository root, checkout and build it:
zksync-llvm checkout && zksync-llvm build
- Run the Tester with the desired filters and the output JSON path:
./target/release/compiler-tester \
--path='tests/solidity/simple/default.sol' \
--mode='Y+M^B3 0.8.26' \
--benchmark='reference.json'
- Change the LLVM branch to your patch in the
LLVM.lock
file at the repository root, checkout and build it:
zksync-llvm checkout && zksync-llvm build
- Run the Tester with the desired filters and the output JSON path:
./target/release/compiler-tester \
--path='tests/solidity/simple/default.sol' \
--mode='Y+M^B3 0.8.26' \
--benchmark='candidate.json'
- Run the benchmark analyzer on the two JSONs:
cargo run --release --bin benchmark-analyzer -- --reference reference.json --candidate candidate.json
After you make any changes in LLVM, you only need to repeat steps 2-3 to update the working branch benchmark data.
By default, benchmark analyzer compares tests from groups with the same name, which means that every test should be compiled with the same codegen and optimizations.
To compare two groups with different names, use the options --query-reference
and --query-candidate
. Then, use benchmark analyzer:
cargo run --release --bin benchmark-analyzer -- --reference reference.json --candidate candidate.json --query-reference "M0B0" --query-candidate "M3B3"
The queries are regular expressions, and the group name, codegen, and optimization options are matched against it.
Use the parameter --benchmark-format
of compiler tester to select the output format: json
(default), csv
, or json-lnt
.
If json-lnt
format is selected:
- The benchmark report will consist of multiple files. They will be placed in the directory provided via the
--output
argument. - It is mandatory to pass a JSON file with additional information using
--benchmark-context
. Here is a minimal example:
{
"machine": "some_machine",
"target": "some_target",
"toolchain": "some_solc_type"
}
- Unset any LLVM-related environment variables you may have set, especially
LLVM_SYS_<version>_PREFIX
(see e.g. https://crates.io/crates/llvm-sys and https://llvm.org/docs/GettingStarted.html#local-llvm-configuration). To make sure:set | grep LLVM
.
The Era Compiler Tester is distributed under the terms of either
- Apache License, Version 2.0, (LICENSE-APACHE or http://www.apache.org/licenses/LICENSE-2.0)
- MIT license (LICENSE-MIT or http://opensource.org/licenses/MIT)
at your option.
ZKsync Era compiler toolchain documentation
ZKsync Era has been through extensive testing and audits, and although it is live, it is still in alpha state and will undergo further audits and bug bounty programs. We would love to hear our community's thoughts and suggestions about it! It's important to note that forking it now could potentially lead to missing important security updates, critical features, and performance improvements.