This repo contains a build script to compile Z3 to WebAssembly using emscripten. To make things more reproducible, the build script may be run in a Vagrant VM.
Loading Z3 is fairly slow (~15 seconds on Chrome, though less than 1 second on Firefox), but verification typically is within a factor 2 to 5 of native performance.
Pre-build archives are available at https://github.com/cpitclaudel/z3.wasm/releases. Demo at https://people.csail.mit.edu/cpitcla/z3.wasm/z3.html.
Install Vagrant, then run this:
vagrant up vagrant ssh VAGRANT=true /vagrant/z3.sh
The first build can take up to two hours (emscripten may require a custom build of LLVM, Z3 is large, and all of this is running in a VM). The output (z3w.js
, z3w.wasm
, z3smt2w.js
, and z3smt2w.wasm
) is written to build/z3/
.
Some Vagrant configurations can cause clang to crash or Vagrant to hang. Building on a physical machine is the most reliable workaround.
On Debian/Ubuntu systems, you may prefer to build natively, which will be much faster. In that case, just run ./z3.sh
, which will place all outputs in build/
.
The build script produces two artefacts: a copy of the full Z3 command line application (z3w.js
, z3w.wasm
), and a Javascript version of a small SMT2 REPL-style API, built by linking z3smt2.c
with libz3
.
There is a small example script using nodejs for demonstration purposes in the repo:
$ nodejs z3test.js /nodefs/test.smt2 sat (model (define-fun y () Int 3) (define-fun x () Int 2) (define-fun z () Int 1) (define-fun g ((x!0 Int) (x!1 Int)) Int (ite (and (= x!0 2) (= x!1 2)) 0 0)) (define-fun f ((x!0 Int)) Int (ite (= x!0 2) (- 1) (ite (= x!0 3) 1 (- 1)))) ) unsat (error "line 14 column 10: model is not available") unsat
From a webpage the process is roughly the same: write Z3's input to a file using emscripten's virtual file system, and use callMain(args)
to run Z3. To capture Z3's output, you can pass a custom print
function to the emscripten module upon initialization.
This is all demoed in html/z3.html
(that example also uses a WebWorker to run Z3, keeping the page responsive while it runs). Try it like this:
cp build/z3/z3w.js build/z3/z3w.wasm ./html python3 -m http.server # Open your browser to http://localhost:8000/html/z3.html
This script exposes a small API useful for verifiers (like F* or Dafny) that interact with Z3 on through pipes. It exposes the following functions:
// Create a fresh SMT2 context Z3_context smt2Init() // Globally set `option` to `value`. void smt2SetParam(const char* option, const char* value) // Send `query` to `ctx` and fetch the response as a string. const char* smt2Ask(Z3_context ctx, const char* query) // Release the resources used by `ctx`. void smt2Destroy(Z3_context ctx)
There is a small example script using nodejs for demonstration purposes in the repo:
$ nodejs ./z3smt2test.js sat (model (define-fun y () Int 4) (define-fun x () Int 2) (define-fun z () Int 0) (define-fun g ((x!0 Int) (x!1 Int)) Int (ite (and (= x!0 2) (= x!1 2)) 0 0)) (define-fun f ((x!0 Int)) Int (ite (= x!0 2) (- 1) (ite (= x!0 4) 1 (- 1)))) ) unsat (error "line 1 column 11: model is not available") unsat
Check the source code of F*.js for an example of how to use this in a larger application.
Chrome precompiles WebAssembly programs before running them — this makes startup slow, though verification after that is fast. The recommendation is to cache compiled modules, but Chrome doesn't (2018-03) allow that yet.
Firefox is much better at this, though the code eventually does run a slower.
The limited z3smt2.wasm
is a bit smaller than the full Z3: use that if you can.
On Apache servers, use the following to compress WASM files (gzipping saves about 75%):
AddType application/wasm .wasm AddOutputFilterByType DEFLATE application/wasm