Wasmer uses the following components:
- Cranelift: for compiling WASM function binaries into Machine IR
- wabt: for transforming
.wast
files to.wasm
and also to run WebAssembly spectests - wasmparser: for parsing the
.wasm
files and translating them into WebAssembly Modules
The first time you run wasmer run myfile.wasm
, wasmer will:
- Check if is a
.wast
file. If so, transform it to.wasm
- Check that the provided binary is a valid WebAssembly one. That means, that its binary format starts with
\0asm
. - If it looks like a WebAssembly file, try to parse it with
wasmparser
and generate aModule
from it - Once a
Module
is generated, anInstance
is created with the properimport_object
(that means, if is detected as an emscripten file, it will add the emscripten expected imports) - Try to call the WebAssembly start function, or if unexistent try to search for the one that is exported as
main
.
Find below an more detailed explanation of the process:
As the WebAssembly file is being parsed, it will read the sections in the WebAssembly file (memory, table, function, global and element definitions) using the Module
(or ModuleEnvironment
) as the structure to hold this information.
However, the real IR initialization happens while a function body is being parsed/created. That means, when the parser reads the section (func ...)
.
While the function body is being parsed the corresponding FuncEnvironment
methods will be called.
So for example, if the function is using a table, the make_table
method within that FuncEnvironment
will be called.
Each of this methods will return the corresponding IR representation.
The Module
creation will be finished once the parsing is done, and will hold all the function IR as well as the imports/exports.
Now that we have a Module
(and all it's definitions living in ModuleInfo
) we should be ready to compile it's functions.
Right now, the Instance
is the one in charge of compiling this functions into machine code.
When creating the Instance
, each of the function bodies (IR) will be compiled into machine code that our architecture can understand.
Once we have the compiled values, we will push them to memory and mark them as executable, so we can call them from anywhere in our code.
Sometimes the functions that we generated will need to call other functions. However the generated code have no idea how to link this functions together.
For example, if a function A
is calling function B
(that means is having a (call b)
on it's body) while compiling A
we will have no idea where the function B
lives on memory (as B
is not yet compiled nor pushed into memory).
For that reason, we will start collecting all the calls that function A
will need to do under the hood, and save it's offsets.
We do that, so we can patch the function calls after compilation, to point to the correct memory address.
Note: Sometimes this functions rather than living in the same WebAssembly module, they will be provided as import values.
There will be other times where the function created will cause a trap (for example, if executing 0 / 0
).
When this happens, we will save the offset of the trap (while the function is being compiled).
Thanks to that when we execute a function, if it traps (that means a sigaction is called), we would be able to backtrack from a memory address to a specific trap case.
Once all the functions are compiled and patched with the proper relocations addresses, we will initialize the corresponding tables (where we save the pointers to all the exported functions), memories and globals that the instance need.
Once that's finished, we will have a Instance
function that will be ready to execute any function we need.
The Wasmer Emscripten integration tries to wrap (and emulate) all the different syscalls that Emscripten needs.
We provide this integration by filling the import_object
with the emscripten functions, while instantiating the WebAssembly Instance.