Binaryen
Optimizer and compiler/toolchain library for WebAssembly
Install / Use
/learn @WebAssembly/BinaryenREADME
Binaryen
Binaryen is a compiler and toolchain infrastructure library for WebAssembly, written in C++. It aims to make [compiling to WebAssembly] easy, fast, and effective:
-
Easy: Binaryen has a simple [C API] in a single header, and can also be [used from JavaScript][JS_API]. It accepts input in [WebAssembly-like form][compile_to_wasm] but also accepts a general [control flow graph] for compilers that prefer that.
-
Fast: Binaryen's internal IR uses compact data structures and is designed for completely parallel codegen and optimization, using all available CPU cores. Binaryen's IR also compiles down to WebAssembly extremely easily and quickly because it is essentially a subset of WebAssembly.
-
Effective: Binaryen's optimizer has many passes (see an overview later down) that can improve code size and speed. These optimizations aim to make Binaryen powerful enough to be used as a [compiler backend][backend] by itself. One specific area of focus is on WebAssembly-specific optimizations (that general-purpose compilers might not do), which you can think of as wasm [minification], similar to minification for JavaScript, CSS, etc., all of which are language-specific.
Toolchains using Binaryen as a component (typically running wasm-opt) include:
Emscripten(C/C++)wasm-pack(Rust)J2CL(Java;J2Wasm)Kotlin(Kotlin/Wasm)Dart(Flutter)wasm_of_ocaml(OCaml)
For more on how some of those work, see the toolchain architecture parts of the V8 WasmGC porting blogpost.
Compilers using Binaryen as a library include:
AssemblyScriptwhich compiles a variant of TypeScript to WebAssemblywasm2jswhich compiles WebAssembly to JSAsteriuswhich compiles Haskell to WebAssemblyGrainwhich compiles Grain to WebAssembly
Binaryen also provides a set of toolchain utilities that can
- Parse and emit WebAssembly. In particular this lets you load WebAssembly, optimize it using Binaryen, and re-emit it, thus implementing a wasm-to-wasm optimizer in a single command.
- Interpret WebAssembly as well as run the WebAssembly spec tests.
- Integrate with Emscripten in order to provide a complete compiler toolchain from C and C++ to WebAssembly.
- Polyfill WebAssembly by running it in the interpreter compiled to JavaScript, if the browser does not yet have native support (useful for testing).
Consult the contributing instructions if you're interested in participating.
Binaryen IR
Binaryen's internal IR is designed to be
- Flexible and fast for optimization.
- As close as possible to WebAssembly so it is simple and fast to convert it to and from WebAssembly.
There are a few differences between Binaryen IR and the WebAssembly language:
-
Tree structure
-
Binaryen IR [is a tree][binaryen_ir], i.e., it has hierarchical structure, for convenience of optimization. This differs from the WebAssembly binary format which is a stack machine.
-
Binaryen uses Stack IR to optimize "stacky" code (that can't be represented in structured form).
-
When stacky code must be represented in Binaryen IR, such as with multivalue instructions and blocks, it is represented with tuple types that do not exist in the WebAssembly language. In addition to multivalue instructions, locals and globals can also have tuple types in Binaryen IR but not in WebAssembly. Experiments show that better support for multivalue could enable useful but small code size savings of 1-3%, so it has not been worth changing the core IR structure to support it better.
-
Block input values (currently only supported in
catchblocks in the exception handling feature) are represented aspopsubexpressions.
-
-
Types and unreachable code
-
WebAssembly limits block/if/loop types to none and the concrete value types (i32, i64, f32, f64). Binaryen IR has an unreachable type, and it allows block/if/loop to take it, allowing [local transforms that don't need to know the global context][unreachable]. As a result, Binaryen's default text output is not necessarily valid wasm text. (To get valid wasm text, you can do
--generate-stack-ir --print-stack-ir, which prints Stack IR, this is guaranteed to be valid for wasm parsers.) -
Binaryen supports a
stringreftype. This is similar to the currently- inactive [stringref proposal], with the difference that the string type is a subtype ofexternrefrather thananyref. Doing so allows toolchains to emit code in a form that uses [js string builtins] which Binaryen can then "lift" into stringref in its internal IR, optimize (for example, a concatenation of "a" and "b" can be optimized at compile time to "ab"), and then "lower" that into js string builtins once more.
-
-
Blocks
-
Binaryen IR has only one control flow structure that contains a variable-length list of children: the block. WebAssembly on the other hand allows all control flow structures, such as loops, if arms, and function bodies, to have multiple children. In Binaryen IR, these other control flow structures have a single child. This child may of course be a block. The motivation for this property is that many passes need special code for iterating on lists of instructions, so having a single IR node with a list simplifies them.
-
As in the Wasm text format, blocks and loops may have names. Branch targets in the IR are resolved by name (as opposed to nesting depth). This has 2 consequences:
-
Blocks without names may not be branch targets.
-
Names are required to be unique. (Reading .wat files with duplicate names is supported; the names are modified when the IR is constructed).
-
-
As an optimization, a block with no name, which can never be a branch target, will not be emitted when generating wasm. Instead its list of children will be directly used in the containing control flow structure. Such a block is sometimes called an "implicit block".
-
-
Reference Types
-
The wasm text and binary formats require that a function whose address is taken by
ref.funcmust be either in the table, or declared via an(elem declare func $..). Binaryen will emit that data when necessary, but it does not represent it in IR. That is, IR can be worked on without needing to think about declaring function references. -
Binaryen IR allows non-nullable locals in the form that the Wasm spec does, in which a
local.getmust be structurally dominated by alocal.setin order to validate (that ensures we do not read the default value of null). Despite being aligned with the Wasm spec, there are some minor details that you may notice:-
A nameless
Blockin Binaryen IR does not interfere with validation. Nameless blocks are never emitted into the binary format (we just emit their contents), so we ignore them for purposes of validating non-nullable locals. As a result, if you read wasm text emitted by Binaryen then you may see what seems to be code that should not validate per the spec (and may not validate in Wasm text parsers), but that difference will not exist in the binary format (binaries emitted by Binaryen will always work everywhere, aside from bugs of course). -
The Binaryen pass runner will automatically fix up validation after each pass (finding things that do not validate and fixing them up, usually by demoting a local to be nullable). As a result you do not need to worry much about this when writing Binaryen passes. For more details see the
requiresNonNullableLocalFixups()hook inpass.hand theLocalStructuralDominanceclass.
-
-
Binaryen IR uses the most refined types possible for references, specifically:
-
The IR type of a
ref.funcis always an exact, non-nullable reference to a defined function type, and not plainfuncref, even if no features beyond basic reference types are enabled. -
The IR type of allocation instructions such as
struct.neworarray.newis always an exact reference, even if Custom Descriptors are not enabled. -
Non-nullable types are also used for the type that
try_tablesends on branches (if we branch, a null is never sent), that is, it sends (ref exn) and not (ref null exn). -
As a result, non-nullable and exact references are generally allowed in the IR even when GC or Custom Descriptors is not enabled. When reading a binary, the more refined types will be applied as we build the IR.
In all cases the binary writer will generalize the type as necessary for the enabled feature set. For example, if only Reference Types is enabled, all function reference types will be emitted as
funcref. -
-
br_ifoutput types are more refined in Binary
-
Related Skills
node-connect
337.7kDiagnose OpenClaw node connection and pairing failures for Android, iOS, and macOS companion apps
frontend-design
83.3kCreate distinctive, production-grade frontend interfaces with high design quality. Use this skill when the user asks to build web components, pages, or applications. Generates creative, polished code that avoids generic AI aesthetics.
openai-whisper-api
337.7kTranscribe audio via OpenAI Audio Transcriptions API (Whisper).
commit-push-pr
83.3kCommit, push, and open a PR
