> C without the standard library (called “libc”) is pretty rough
If you don't want to pull in musl or a traditional libc, there's a more Wasm-y solution known as the Web Assembly System Interface (WASI) [0] that delegates the libc functionality to the runtime.
A WASI Wasm module can be compiled using clang, as in the article. The only difference is to use the WASI sysroot [1].
> optimization
LTO and -O3 are great! I've also found the Twiggy [2] tool useful for more "manual" optimization.
One of the things I like about Rust as an embedded developer is that they make a distinction between core and std portions of the standard library. Core is everything that only depends on memset, memcpy, and memcmp,.which means you know you get it for free on new ports.
Author here :) I am very excited about WASI, but as I mentioned in a comment below, wanted to keep it to the fundamentals so you can appreciate what WASI does for you.
And definitely agree with the shout-out to Twiggy (which I mention in the previous post in the series)!
If size is really important, you probably want to try -Os (optimize for size) and -Oz (try harder to optimize for size, including at the expense of CPU) as well.
> LTO
If your project gets big enough that -flto results in unacceptable link times, try -flto=thin.
I just created a code snippet in C [1] which prints hello world by using the (bare-metal) WASI interface. No headers included. The compiled wasm file works with wasmer, lucet, wasmtime and an web implementation of WASI [2]
I've been thinking of writing a blogpost comparing the advantages and disadvantages of emscripten, wasi, and plain llvm (which is what is discussed here), and also how those interact with web vs server. This space has definitely gotten more interesting recently!
One of the pain points to using Wasm in the real world, is the lack of decent debugging.
eg no ability to run code in a debugger, set breakpoints, etc.
That being said, it's an area being worked on.
Wasm generated by LLVM can already have debugging info stored in it using Wasm "custom sections" (they're a thing) in DWARF format. eg .debug_info, .debug_str, (etc)
So, debuggers are at least possible.
Unfortunately, the way Wasm does variables doesn't map to the way DWARF currently does them. So they can't be encoded correctly. A Major problem. :(
Yury Delendik is working through a spec for fixing that (officially):
>AssemblyScript compiles strictly typed TypeScript (basically JavaScript with types) to WebAssembly using Binaryen. It generates lean and mean WebAssembly modules while being just an npm install away.
Since you can compile AssemblyScript into JavaScript with the TypeScript compiler as well as into WebAssembly, you can compare the speed of JavaScript -vs- WebAssembly on the same source code. Aaron did some interesting benchmarks using wasmboy, in the great tradition of using GameBoy emulators to benchmark JavaScript engines:
This is a great article that takes you pretty far with very little. I think it's much easier to tinker and experiment when the boilerplate and tooling is reduced to a minimum -- you get a much deeper understanding in what's actually going on behind the scenes.
Yes, much easier to tinker with a more barebones system. The downside of powerful toolchains is often their complexity. So there's a difference between one being better for shipping code and one better for learning.
Nice comprehensive writeup. The next step, beyond the basic allocator provided, would be to use wasi-sdk (https://github.com/cranestation/wasi-sdk) which provides a full musl-based libc, targeting the WASI interfaces. With this, you can invoke a C program with arguments, environment variables, and filesystem access.
The inNative WebAssembly Runtime ran into this problem as well, and includes wasm_malloc.c, which can be linked against your application to provide a simple malloc() implementation without having to write one yourself or depend on WASI.
Author here :D WASI is great and has me all kinds of excited, but I wanted to cover the fundamentals so people can appreciate what WASI really gives you.
Off-topic: the visual aesthetic of this blog really has its own unique charm. It's truly something else and nonetheless seems to do the trick pretty well.
Though I'm certain the security concerns were integral to the design of WASM, the ability to pass a pointer from JavaScript downstream into WASM terrifies the hell out of me.
I know that theoretically every WASM module is supposed to have a fully isolated memory block, but I can't help but wonder about the day where a bug allows WASM to deliver malware payloads to read other web browser tabs. Let's hope that WASM doesn't become everyday in advert networks.
[+] [-] _nhynes|6 years ago|reply
If you don't want to pull in musl or a traditional libc, there's a more Wasm-y solution known as the Web Assembly System Interface (WASI) [0] that delegates the libc functionality to the runtime.
A WASI Wasm module can be compiled using clang, as in the article. The only difference is to use the WASI sysroot [1].
> optimization
LTO and -O3 are great! I've also found the Twiggy [2] tool useful for more "manual" optimization.
[0] https://wasi.dev [1] https://github.com/CraneStation/wasi-sdk/releases [2] https://rustwasm.github.io/twiggy/
[+] [-] monocasa|6 years ago|reply
[+] [-] dassurma|6 years ago|reply
And definitely agree with the shout-out to Twiggy (which I mention in the previous post in the series)!
[+] [-] swolchok|6 years ago|reply
If size is really important, you probably want to try -Os (optimize for size) and -Oz (try harder to optimize for size, including at the expense of CPU) as well.
> LTO
If your project gets big enough that -flto results in unacceptable link times, try -flto=thin.
[+] [-] s-macke|6 years ago|reply
[1] https://gist.github.com/s-macke/6dd78c78be46214d418454abb667...
[2] https://wasi.dev/polyfill/
[+] [-] azakai|6 years ago|reply
[+] [-] pspeter3|6 years ago|reply
[+] [-] ngcc_hk|6 years ago|reply
[+] [-] ericb|6 years ago|reply
[+] [-] raboukhalil|6 years ago|reply
[+] [-] justinclift|6 years ago|reply
eg no ability to run code in a debugger, set breakpoints, etc.
That being said, it's an area being worked on.
Wasm generated by LLVM can already have debugging info stored in it using Wasm "custom sections" (they're a thing) in DWARF format. eg .debug_info, .debug_str, (etc)
So, debuggers are at least possible.
Unfortunately, the way Wasm does variables doesn't map to the way DWARF currently does them. So they can't be encoded correctly. A Major problem. :(
Yury Delendik is working through a spec for fixing that (officially):
https://yurydelendik.github.io/webassembly-dwarf/
His initial implementation, with patches (on an older) LLVM so it generates correct debug info according to the in-development spec, is here:
https://github.com/yurydelendik/llvm-project/tree/frame-poin...
Testing and feedback by a wider audience would be useful. :)
A more recent fork of LLVM (based on 8.0.1 dev ~3 days ago), with Yury's patches applied is here:
https://github.com/justinclift/llvm/commits/release_80-wasm_...
Personally, I'm still trying to get my head around generating DWARF debugging info. Hopefully work it out in a few days. :)
[+] [-] saagarjha|6 years ago|reply
[+] [-] DonHopkins|6 years ago|reply
https://github.com/AssemblyScript/assemblyscript
>AssemblyScript compiles strictly typed TypeScript (basically JavaScript with types) to WebAssembly using Binaryen. It generates lean and mean WebAssembly modules while being just an npm install away.
https://dev.to/jtenner/an-assemblyscript-primer-for-typescri...
Here's a great example of a project that uses it:
https://github.com/torch2424/wasmboy
>️Gameboy Emulator Library written in Web Assembly using AssemblyScript, Debugger/Shell in Preact ️
Here's an excellent talk about wasmboy by the author, Aaron Turner -- he's done some really outstanding work:
https://www.youtube.com/watch?v=ZlL1nduatZQ
Since you can compile AssemblyScript into JavaScript with the TypeScript compiler as well as into WebAssembly, you can compare the speed of JavaScript -vs- WebAssembly on the same source code. Aaron did some interesting benchmarks using wasmboy, in the great tradition of using GameBoy emulators to benchmark JavaScript engines:
https://medium.com/@torch2424/webassembly-is-fast-a-real-wor...
[+] [-] DonHopkins|6 years ago|reply
https://news.ycombinator.com/threads?id=torch2424
torch2424 7 months ago | unvote | parent [-] | on: Walt: JavaScript-like syntax for WebAssembly
I help out every once and a while on the AssemblyScript team with like issues, docs, and things. And made wasmboy, which uses AssemblyScript:
https://github.com/torch2424/wasmBoy
But that being said, usually when people are interested in the language, we usually direct them to the "n-body" example: https://github.com/AssemblyScript/assemblyscript/blob/master... . Which kind of looks more typescript-y :)
Also, just to stay on topic, I think walt is awesome. Stoked to see so many projects coming up with a "Wasm for JS devs" approach/story.
[+] [-] pjmlp|6 years ago|reply
[+] [-] torch2424|6 years ago|reply
Aaron Turner here, thank you for all the kind words! :) Yes I did all those things, and stoked to see people excited about it!
Definitely feel free to reach out anytime about AssemblyScript or WasmBoy. Would love to chat with you / anyone interested.
We also have a slack channel you can reach out and get invited to (see the wiki sidebar): https://github.com/AssemblyScript/assemblyscript/wiki
Thanks again!
[+] [-] lxe|6 years ago|reply
[+] [-] brighteyes|6 years ago|reply
[+] [-] phickey|6 years ago|reply
[+] [-] dassurma|6 years ago|reply
[+] [-] blackhole|6 years ago|reply
https://github.com/innative-sdk/innative/wiki/Compile-C---wi...
Of course, it'll be a lot easier to simply depend on WASI instead and re-implement standard libraries on top of it.
[+] [-] jedisct1|6 years ago|reply
[+] [-] dassurma|6 years ago|reply
[+] [-] wa1987|6 years ago|reply
[+] [-] dassurma|6 years ago|reply
[+] [-] cryptonector|6 years ago|reply
[+] [-] shakna|6 years ago|reply
I know that theoretically every WASM module is supposed to have a fully isolated memory block, but I can't help but wonder about the day where a bug allows WASM to deliver malware payloads to read other web browser tabs. Let's hope that WASM doesn't become everyday in advert networks.
[+] [-] maxmcd|6 years ago|reply
Getting isolation between tabs does seem to be an ongoing concern, but that's happening regardless of the presence of wasm: https://v8.dev/blog/spectre#site-isolation