Skip to content

Commit f2c4a29

Browse files
committed
Format markdown files
1 parent 3e18790 commit f2c4a29

File tree

6 files changed

+204
-167
lines changed

6 files changed

+204
-167
lines changed

Building.md

+54-49
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
# Building WebAssembly Smart Contracts
22

3-
The subdirectories are various examples of compiling smart contracts.
4-
Here are some tips useful for creating your own.
3+
The subdirectories are various examples of compiling smart contracts. Here are
4+
some tips useful for creating your own.
55

66
## Setup
77

@@ -13,27 +13,29 @@ via `cargo new --lib sample`. Then add the following to `Cargo.toml`:
1313
crate-type = ["cdylib", "rlib"]
1414
```
1515

16-
The `cdylib` is needed for the wasm target.
17-
The `rlib` is needed to compile artifacts for integration tests (and benchmarking).
16+
The `cdylib` is needed for the wasm target. The `rlib` is needed to compile
17+
artifacts for integration tests (and benchmarking).
1818

1919
**Note** throughout this demo I will use the name `hackatom` for the project.
20-
Please replace it with the real name of your crate. I intentionally didn't use `<name>`,
21-
so you can cut and paste for a quick demo. Just update this when making real code.
20+
Please replace it with the real name of your crate. I intentionally didn't use
21+
`<name>`, so you can cut and paste for a quick demo. Just update this when
22+
making real code.
2223

2324
## Requirements
2425

2526
You must support the rust target `wasm32-unknown-unknown`.
2627

27-
Check which ones you currently have installed via `rustup target list --installed`.
28-
If wasm32 is not on that list, install using `rustup target add wasm32-unknown-unknown`
29-
28+
Check which ones you currently have installed via
29+
`rustup target list --installed`. If wasm32 is not on that list, install using
30+
`rustup target add wasm32-unknown-unknown`
3031

3132
## Building
3233

3334
Go into the subdirectory, called `sample` from now on:
3435

35-
To compile the code, run `cargo build --release --target wasm32-unknown-unknown`.
36-
The output will be in `target/wasm32-unknown-unknown/release/hackatom.wasm`
36+
To compile the code, run
37+
`cargo build --release --target wasm32-unknown-unknown`. The output will be in
38+
`target/wasm32-unknown-unknown/release/hackatom.wasm`
3739

3840
You probably don't want to explicitly set the target every time, so you can just
3941
add the following to `.cargo/config`:
@@ -43,11 +45,14 @@ add the following to `.cargo/config`:
4345
wasm = "build --release --target wasm32-unknown-unknown"
4446
```
4547

46-
And you can now just call `cargo wasm` to build it, and `cargo test` to run tests.
48+
And you can now just call `cargo wasm` to build it, and `cargo test` to run
49+
tests.
4750

48-
**Note** Using `build.target` seems to force tests to use that target as well, remove this or find a work-around.
51+
**Note** Using `build.target` seems to force tests to use that target as well,
52+
remove this or find a work-around.
4953
[This discussion](https://internals.rust-lang.org/t/set-default-target-for-cargo-build-but-not-for-cargo-test/9777)
50-
and [closed PR](https://github.com/rust-lang/cargo/pull/6825) seem to suggest this will never be done.
54+
and [closed PR](https://github.com/rust-lang/cargo/pull/6825) seem to suggest
55+
this will never be done.
5156

5257
## Optimizations
5358

@@ -56,9 +61,9 @@ Here are some things to make it smaller.
5661

5762
### Smaller builds
5863

59-
If you want to request the compiler to make smaller binaries,
60-
you can hit a few flags (which raise compile time significantly).
61-
Try adding this custom profile to Cargo.toml:
64+
If you want to request the compiler to make smaller binaries, you can hit a few
65+
flags (which raise compile time significantly). Try adding this custom profile
66+
to Cargo.toml:
6267

6368
```yaml
6469
[profile.release]
@@ -73,19 +78,20 @@ incremental = false
7378
overflow-checks = true
7479
```
7580

76-
**IMPORTANT** it is essential that codegen-units is set to 1 for deterministic builds.
77-
Otherwise, you will have a different wasm output with each compile.
81+
**IMPORTANT** it is essential that codegen-units is set to 1 for deterministic
82+
builds. Otherwise, you will have a different wasm output with each compile.
7883

7984
## Shrinking the output
8085

8186
After compiling your contract, take a look at the size of the original output:
82-
`du -sh target/wasm32-unknown-unknown/release/hackatom.wasm`, it is likely around 1.5 MB.
83-
Most of that is unneeded and can easily be trimmed. The first approach is to use
84-
[`wasm-pack`](https://github.com/rustwasm/wasm-pack) to build it.
85-
This is designed for exporting small wasm builds and js bindings for the web, but is
86-
also the most actively maintained stack for trimmed wasm builds with rust.
87-
[`wasm-gc`](https://github.com/alexcrichton/wasm-gc), the older alternative, has
88-
been deprecated for this approach. (Note you must [install wasm-pack first](https://rustwasm.github.io/wasm-pack/installer/)):
87+
`du -sh target/wasm32-unknown-unknown/release/hackatom.wasm`, it is likely
88+
around 1.5 MB. Most of that is unneeded and can easily be trimmed. The first
89+
approach is to use [`wasm-pack`](https://github.com/rustwasm/wasm-pack) to build
90+
it. This is designed for exporting small wasm builds and js bindings for the
91+
web, but is also the most actively maintained stack for trimmed wasm builds with
92+
rust. [`wasm-gc`](https://github.com/alexcrichton/wasm-gc), the older
93+
alternative, has been deprecated for this approach. (Note you must
94+
[install wasm-pack first](https://rustwasm.github.io/wasm-pack/installer/)):
8995

9096
```sh
9197
cargo wasm
@@ -96,10 +102,9 @@ du -h pkg/hackatom_bg.wasm
96102
```
97103

98104
A bit smaller, huh? For the sample contract, this is around 64kB, but this
99-
varies a lot contract-by-contract.
100-
If you have plenty of dependencies and this is still too big,
101-
you can do a bit of investigation of where the size comes from, and maybe
102-
change your dependencies:
105+
varies a lot contract-by-contract. If you have plenty of dependencies and this
106+
is still too big, you can do a bit of investigation of where the size comes
107+
from, and maybe change your dependencies:
103108

104109
```sh
105110
cargo install twiggy
@@ -118,14 +123,15 @@ wasm-nm -i contract.wasm
118123

119124
## Ultra-Compression
120125

121-
You can still get a bit smaller. Note the symbol names that were used in twiggy. Well,
122-
those come from inside the wasm build. You can strip out these symbols and other debug
123-
info, that won't be usable when running anyway. For this we use `wasm-opt` from the
124-
[enscripten toolchain](). This is a bunch of C++ code that needs to be compiled, and to
125-
simplify the whole process, as well as create reproduceable builds, we have created
126-
[`cosmwasm-opt`](https://github.com/confio/cosmwasm-opt),
127-
which contains a `Dockerfile` that you can use to run both `wasm-pack` and `wasm-opt`.
128-
To make the build, just run the following in the project directory:
126+
You can still get a bit smaller. Note the symbol names that were used in twiggy.
127+
Well, those come from inside the wasm build. You can strip out these symbols and
128+
other debug info, that won't be usable when running anyway. For this we use
129+
`wasm-opt` from the [enscripten toolchain](). This is a bunch of C++ code that
130+
needs to be compiled, and to simplify the whole process, as well as create
131+
reproduceable builds, we have created
132+
[`cosmwasm-opt`](https://github.com/confio/cosmwasm-opt), which contains a
133+
`Dockerfile` that you can use to run both `wasm-pack` and `wasm-opt`. To make
134+
the build, just run the following in the project directory:
129135

130136
```sh
131137
docker run --rm -u $(id -u):$(id -g) -v $(pwd):/code confio/cosmwasm-opt:0.4.1
@@ -134,14 +140,15 @@ du -h contract.wasm
134140
```
135141

136142
Note that this always outputs the file as `contract.wasm`, not with the name of
137-
the project (yes, every tool chain has a different output location).
138-
For the hackatom sample, I now get down to 52kB, an 18% improvement.
139-
This is as far as you can minimize the input, without removing actual functionality.
143+
the project (yes, every tool chain has a different output location). For the
144+
hackatom sample, I now get down to 52kB, an 18% improvement. This is as far as
145+
you can minimize the input, without removing actual functionality.
140146

141-
While we cannot trim down the wasm code anymore, we can still reduce the size a bit
142-
for loading in blockchain transactions. We [soon plan](https://github.com/confio/go-cosmwasm/issues/20)
143-
to allow gzip-ed wasm in the transactions posted to the chain, which will further
144-
reduce gas cost of the code upload. Check out this final output:
147+
While we cannot trim down the wasm code anymore, we can still reduce the size a
148+
bit for loading in blockchain transactions. We
149+
[soon plan](https://github.com/confio/go-cosmwasm/issues/20) to allow gzip-ed
150+
wasm in the transactions posted to the chain, which will further reduce gas cost
151+
of the code upload. Check out this final output:
145152

146153
```sh
147154
$ gzip -k contract.wasm
@@ -152,7 +159,5 @@ $ du -h contract.wasm*
152159

153160
And there you have it. We have gone from 1.5MB for the naive build, to 72kB with
154161
the standard minification tooling, all the way down to 20kB with very aggressive
155-
trimming and compression. Less than 1.5% of the original size. This is indeed something
156-
you can easily fit inside a transaction.
157-
158-
162+
trimming and compression. Less than 1.5% of the original size. This is indeed
163+
something you can easily fit inside a transaction.

EntryPoints.md

+26-23
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
# Defining Entry Points to Wasm
22

3-
## Exports
3+
## Exports
44

55
`exports` are the functions that we export to the outside world. After
66
compilation these will be the only entry points that the application can call
@@ -14,18 +14,19 @@ To make an export in rust code, you can add the following lines:
1414
pub extern "C" fn double(n: i32) -> i32 {
1515
n * 2
1616
}
17-
```
17+
```
1818

1919
Note that you need the `#[no_mangle]` directive to keep the naming, and declare
2020
it as `pub extern "C"` to create a proper C ABI, which is the standard interface
2121
for Web Assembly, as well as FFI.
2222

2323
## Imports
2424

25-
If we want to interact with the outside world, the smart contract needs to define
26-
a set of `imports`. These are function signatures we expect to be implemented by
27-
the environment and provided when the VM is instantiated. If the proper imports
28-
are not provided, you will receive an error upon instantiating the contract.
25+
If we want to interact with the outside world, the smart contract needs to
26+
define a set of `imports`. These are function signatures we expect to be
27+
implemented by the environment and provided when the VM is instantiated. If the
28+
proper imports are not provided, you will receive an error upon instantiating
29+
the contract.
2930

3031
```rust
3132
extern "C" {
@@ -34,20 +35,22 @@ extern "C" {
3435
}
3536
```
3637

37-
The above expects the runtime to provide read/write access to some
38-
(persistent) singleton. Notably, the contract has no knowledge how it is stored,
39-
and no way to "jailbreak" or access other element of the database.
38+
The above expects the runtime to provide read/write access to some (persistent)
39+
singleton. Notably, the contract has no knowledge how it is stored, and no way
40+
to "jailbreak" or access other element of the database.
4041

4142
## Memory Management
4243

43-
If you look closely, you will see every function definition in `exports` accepts
44-
a fixed number of arguments of type `i32` and returns one result of type `i32` (or `void`).
45-
With such limitations, how can one pass in a `struct` to the contract, or even
46-
a serialized byte array (eg. json blob). And how can we return a string back?
44+
If you look closely, you will see every function definition in `exports` accepts
45+
a fixed number of arguments of type `i32` and returns one result of type `i32`
46+
(or `void`). With such limitations, how can one pass in a `struct` to the
47+
contract, or even a serialized byte array (eg. json blob). And how can we return
48+
a string back?
4749

4850
There is one more way in which the runtime can interact with a smart contract
4951
instance. It can directly read and write to the linear memory of the smart
50-
contract. In general, contracts are expected to export two well-defined functions:
52+
contract. In general, contracts are expected to export two well-defined
53+
functions:
5154

5255
```rust
5356
#[no_mangle]
@@ -66,15 +69,15 @@ pub extern "C" fn deallocate(pointer: *mut c_void, capacity: usize) {
6669
}
6770
```
6871

69-
`allocate` heap allocates `size` bytes and tells the wasm alloc library not to clear it
70-
(forget), after which it returns the pointer (integer offset) to the caller.
71-
The caller can now safely write up to `size` bytes to the given offset, eg.
72-
`copy(data, vm.Memory[offset:offset+size])`. Of course, this passes the responsibility
73-
of freeing the memory from the wasm code to the caller, so make sure to
74-
call `deallocate()` on the memory reference after the function call finishes.
72+
`allocate` heap allocates `size` bytes and tells the wasm alloc library not to
73+
clear it (forget), after which it returns the pointer (integer offset) to the
74+
caller. The caller can now safely write up to `size` bytes to the given offset,
75+
eg. `copy(data, vm.Memory[offset:offset+size])`. Of course, this passes the
76+
responsibility of freeing the memory from the wasm code to the caller, so make
77+
sure to call `deallocate()` on the memory reference after the function call
78+
finishes.
7579

7680
We can explore more complex and idiomatic ways of passing data between the
77-
environment and the wasm contract, but for now, just ensure you export these
78-
two functions and the runtime will make use of them to get `string` and `[]byte`
81+
environment and the wasm contract, but for now, just ensure you export these two
82+
functions and the runtime will make use of them to get `string` and `[]byte`
7983
into and out of the wasm contract.
80-

0 commit comments

Comments
 (0)