Skip to content

Integration testing framework for ZKsync compilers

License

Apache-2.0, MIT licenses found

Licenses found

Apache-2.0
LICENSE-APACHE
MIT
LICENSE-MIT
Notifications You must be signed in to change notification settings

matter-labs/era-compiler-tester

ZKsync Era: The EraVM Compiler Integration Test Framework

Logo

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.

Building

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:

    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 zksolc and zkvyper compilers.
  • Build zksolc and zkvyper compilers and add the binaries to $PATH, or use the --zksolc or --zkvyper options to specify their paths.
6. Build era-compiler-tester.
  • Build the Tester with cargo:
    cargo build --release

When the build succeeds, you can run the tests using the examples below.

What is supported

Languages

  • Solidity
  • Yul
  • Vyper
  • LLVM IR
  • EraVM assembly

Optimizers

  • 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 and B3)
  • solc optimizer (- or +)
  • vyper optimizer (- or +)

Solidity codegens

  • Yul pure (Y)
  • EVM assembly from Yul (I)
  • EVM assembly pure (E)
  • Vyper LLL (V)

Compiler versions

  • >=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

Compiler pipelines

Currently only relevant for the Solidity compiler, where you can choose the IR:

  • Yul (preferred for Solidity ≥0.8)
  • EVM (supports Solidity ≥0.4)

Wildcards

Most of the specifiers support wildcards * (any), ^ ('3' and 'z'). With no mode argument, iterates over all option combinations (approximately 800).

Usage

Each command assumes you are at the root of the compiler-tester repository.

Generic command

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.

Example 1

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'

Example 2

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'

Example 3

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'

Tracing

If you run the tester with -T flag, JSON trace files will be written to the ./trace/ directory. The trace files can be used with our custom ZKsync EraVM assembly tracer for debugging and research purposes.

Benchmarking

  1. 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
  1. 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'
  1. 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
  1. 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'
  1. 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.

Troubleshooting

License

The Era Compiler Tester is distributed under the terms of either

at your option.

Resources

ZKsync Era compiler toolchain documentation

Official Links

Disclaimer

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.

About

Integration testing framework for ZKsync compilers

Topics

Resources

License

Apache-2.0, MIT licenses found

Licenses found

Apache-2.0
LICENSE-APACHE
MIT
LICENSE-MIT

Security policy

Stars

Watchers

Forks