Have you ever wanted to run C/C++/Rust/WebAssembly applications in Minecraft? No? Well, now you can!
Wasmcraft is a compiler from WebAssembly to Minecraft Java Edition datapacks. Since WebAssembly is a well-supported target for many languages, this means that you can run code written in e.g. C in Minecraft.
This was inspired by Sethbling's Atari 2600 Datapack and is the (much, much improved) spiritual successor to Langcraft.
Here is a short overview video that includes demonstrations of four different games compiled using Wasmcraft:
- Doom 1993
- Celeste classic
- Minecraft 4k
- NES Emulator (running Super Mario Bros)
Additionally, there is another video of the CHIP-8 emulator from here running Pong.
- Minimal porting required
- Most non-hosted/freestanding code that doesn't use floats is compatible
- Compatible with newlib to provide libc features like
printf
andmalloc
- Common, well-supported, and stable source language
- Can be targeted from C, C++, and Rust
- All WASM integer and control flow operations supported (and tested against the WebAssembly test suite)
- SSA-based optimizations
- Strength reduction
- Constant folding
- Dead code elimination
- Register allocation
- No game modification required
- Compatible with vanilla Java Edition Minecraft 1.16+ (tested on 1.19+)
- Simulation and debugging tools for datapack developers
Let's say we want to run the following program:
/* foo.c */
#include "mcinterface.h"
// Since there is no standard library, we use `_start` instead of `main`
int _start() {
for (int i = 0; i < 10; ++i) {
print(i);
}
return 0;
}
First, compile it to an object file:
clang foo.c -target wasm32 -nostdlib -c -o foo.o
wasm-ld foo.o --lto-O3 --gc-sections --import-undefined -o foo.wasm
wasm2wat foo.wasm -o foo.wat # Optional, useful for debugging
Ensure you have Rust version >= 1.58 installed, which is very recent. To update Rust, run rustup update
.
Then, simply navigate to the wasmcraft2 directory and run:
cargo run --release -- ../foo.wasm -O1 -o ../nameofdatapack`
This will create a datapack in the folder nameofdatapack
, which can be directly placed in the datapacks folder
of any Minecraft Java Edition world (this has only been tested on 1.18/1.19, but should work on older and newer versions as well).
Warning: all of the contents of the folder ../nameofdatapack
will be deleted.
Do NOT point it to a folder with important files!
Warning: due to limitations with Minecraft commands, this needs to fill a few chunks near 0, 0 with jukeboxes, so do NOT run this in a world with builds you don't want destroyed!
Run these commands:
/gamerule maxCommandChainLength 100000000
(This only needs to be run once per world)
/reload
(This only needs to be run when the datapack is changed while the world is open)
/datapack enable "name/of/my/datapack"
(If it is not already enabled)
/function wasmrunner:init
/function wasmrunner:_start
And you should see the numbers 0 to 9 printed out in the chat.
To adjust the number of commands run per tick, you can set a value any time while the datapack is running:
scoreboard players set %%max_commands reg 30000
This number can be adjusted higher or lower depending on your system's performance.
Currently, programs compiled under Wasmcraft have to have sleep calls manually inserted into them under certain circumstances. This is because Minecraft tries to execute all commands in a datapack function call within a single game tick, so in the worst case a very long-running function will freeze the game world indefinitely.
Wasmcraft currently inserts sleep checks before functions and inside of loops, but long stretches of code that don't have any loops or function calls can end up having too many commands to work properly, and some instructions are not command-counted properly (like memset).
In these cases, the mc_sleep()
function provided in mcinterface.h
will pause execution and resume it on the next game tick.
If sleep calls are inserted too frequently, the datapack will run very slowly.
However, if sleep calls are not inserted frequently enough, the game will lag or command execution will be canceled by the game entirely.
In order to determine if some code has sleep calls inserted frequently enough, Wasmcraft can simulate the code using the following command:
cargo run --release -- ../foo.wasm -O1 -o ../nameofdatapack --no-persist-output --run-output
If MaxTickCommandsRun
appears in the output,
too many commands were executed in a single tick and mc_sleep()
must be inserted somewhere.
The provided stacktrace, combined with print
statements or the .wat
file, can be used to find the problematic code.
(Wasmcraft function IDs always match the ID of the corresponding function in the WebAssembly file).
When using the datapack in-game, press Ctrl-Alt-F3 to get a tick time graph (on the right side). This can be used to find ticks that are running too slowly or are lagging the game.
Manual sleep calls are planned to be fixed in a future update.
In order to use C library functions like printf
, malloc
, open
, etc.,
projects can be compiled with the Newlib C stdlib implementation.
An example of how to do that can be found here
Minecraft will silently ignore most unintended datapack behaviors (e.g. accessing undefined variables, division by zero, etc), which makes debugging difficult. In addition, Wasmcraft's compile times can make iterating and bugfixing more tedious.
For the first stage of development, the Wasmcraft Preview Simulator is useful for prototyping and quickly testing programs before actually providing them to the Wasmcraft compiler. See its repository for details on how to use it.
Once programs work under the preview simulator, they can also be tested using Wasmcraft itself.
This interprets the actual generated commands, so it should behave almost identically to the program
in a real Minecraft world. The Wasmcraft simulator can be used by passing the --run-output
flag:
# --no-persist-output doesn't save the datapack to disk, so it makes testing a bit faster
cargo run --release -- /my/input/file.wasm --no-persist-output -o /ignored --run-output
By default, the Wasmcraft simulator runs in text mode.
Setting the gui
feature flag will cause it to also display a window that shows placed blocks.
The GUI mode supports most of the same flags as the Preview Simulator, which can be passed using --sim-flags
:
cargo run --release --features gui -- <same as above> --sim-flags="--z-plane=-5 --frame-sleep=100"
- Floating point operations are not supported (yet). Use fixed point operations instead, e.g. libfixmath
- Bitwise operations and 64-bit divisions have to be emulated and are very slow.
- 8-bit and 16-bit accesses are fairly slow, and all memory accesses have not-insignificant overhead.
- Manual calls to
mc_sleep()
have to be inserted in some cases. - Only a limited subset of Minecraft commands are available in the interface.
Licensed under either of
- MIT License
- Apache License, Version 2.0