-
Notifications
You must be signed in to change notification settings - Fork 165
/
Copy pathwasmer.go
124 lines (124 loc) · 4.13 KB
/
wasmer.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
// Package wasmer is a complete and mature WebAssembly runtime for Go
// based on Wasmer (https://github.com/wasmerio/wasmer).
//
// Features
//
// • Easy to use: The wasmer API mimics the standard WebAssembly API,
//
// • Fast: wasmer executes the WebAssembly modules as fast as possible, close to native speed,
//
// • Safe: All calls to WebAssembly will be fast, but more importantly, complete safe and sandboxed.
//
// Quick Introduction
//
// The wasmer Go package brings the required API to execute WebAssembly
// modules. In a nutshell, wasmer compiles the WebAssembly module into
// compiled code, and then executes it. wasmer is designed to work in
// various environments and platforms: From nano single-board
// computers to large and powerful servers, including more exotic
// ones. To address those requirements, Wasmer provides 2 engines and
// 3 compilers.
//
// Succinctly, an engine is responsible to drive the compilation and
// the execution of a WebAssembly module. By extension, a headless
// engine can only execute a WebAssembly module, i.e. a module that
// has previously been compiled, or compiled, serialized and
// deserialized. By default, the wasmer package comes with 2 headless
// engines:
//
// • Universal, the compiled machine code lives in memory,
//
// • Dylib, the compiled machine code lives in a shared object file
// (.so, .dylib, or .dll), and is natively executed.
//
// The wasmer Go packages comes with 3 compilers:
//
// • Singlepass: Super fast compilation times, slower execution
// times. Not prone to JIT-bombs. Ideal for blockchains.
//
// • Cranelift: Fast compilation times, fast execution times. Ideal
// for development.
//
// • LLVM: Slow compilation times, very fast execution times (close to
// native). Ideal for production.
//
// WebAssembly C API standard
//
// Wasmer —the runtime— is written in Rust; C and C++ bindings
// exist. This Go package relies on the so-called wasm_c_api,
// https://github.com/WebAssembly/wasm-c-api, which is the new
// standard C API, implemented inside Wasmer as the Wasmer C API,
// https://wasmerio.github.io/wasmer/crates/wasmer_c_api/. This
// standard is characterized as a living standard. The API is not yet
// stable, even though it shows maturity overtime. However, the Wasmer
// C API provides some extensions, like the wasi_* or wasmer_* types
// and functions, which aren't yet defined by the standard. The Go
// package commits to keep a semantic versioning over the API,
// regardless what happens with the C API.
//
// Examples
//
// The very basic example is the following
//
// // Let's assume we don't have WebAssembly bytes at hand. We
// // will write WebAssembly manually.
// wasmBytes := []byte(`
// (module
// (type (func (param i32 i32) (result i32)))
// (func (type 0)
// local.get 0
// local.get 1
// i32.add)
// (export "sum" (func 0)))
// `)
//
// // Create an Engine
// engine := wasmer.NewEngine()
//
// // Create a Store
// store := wasmer.NewStore(engine)
//
// // Let's compile the module.
// module, err := wasmer.NewModule(store, wasmBytes)
//
// if err != nil {
// fmt.Println("Failed to compile module:", err)
// }
//
// // Create an empty import object.
// importObject := wasmer.NewImportObject()
//
// // Let's instantiate the WebAssembly module.
// instance, err := wasmer.NewInstance(module, importObject)
//
// if err != nil {
// panic(fmt.Sprintln("Failed to instantiate the module:", err))
// }
//
// // Now let's execute the `sum` function.
// sum, err := instance.Exports.GetFunction("sum")
//
// if err != nil {
// panic(fmt.Sprintln("Failed to get the `add_one` function:", err))
// }
//
// result, err := sum(1, 2)
//
// if err != nil {
// panic(fmt.Sprintln("Failed to call the `add_one` function:", err))
// }
//
// fmt.Println("Results of `sum`:", result)
//
// // Output:
// // Results of `sum`: 3
//
// That's it. Now explore the API! Some pointers for the adventurers:
//
// • The basic elements are Module and Instance,
//
// • Exports of an instance are represented by the Exports type,
//
// • Maybe your module needs to import Function, Memory, Global or
// Table? Well, there is the ImportObject for that!
package wasmer