Skip to content

Commit 1461f56

Browse files
authored
Implement the GC in Rust (#1750)
This implements the current garbage collector in Rust. No changes were made to the GC design -- it's just ports the one implemented in code generator to Rust. The goals are: - Evaluate Rust for Motoko's RTS implementation - Make the collector easier to read, understand, modify, and extend.
1 parent 5a67f48 commit 1461f56

14 files changed

+732
-511
lines changed

rts/Makefile

+1
Original file line numberDiff line numberDiff line change
@@ -177,6 +177,7 @@ mo-rts.wasm: $(RTS_RUST_WASM_O) $(RTS_WASM_O) $(TOMMATH_WASM_O) $(MUSL_WASM_O)
177177
$(WASM_LD) -o $@ \
178178
--import-memory --shared --no-entry --gc-sections \
179179
--export=__wasm_call_ctors \
180+
--export=memcpy \
180181
--whole-archive \
181182
$+
182183

rts/bigint.c

+1-1
Original file line numberDiff line numberDiff line change
@@ -50,7 +50,7 @@ export void* mp_realloc(void *ptr, size_t old_size, size_t new_size) {
5050
if (new_size > FIELD(r, 1)) {
5151
void *newptr = mp_alloc(new_size);
5252
if (old_size != FIELD(r, 1)) bigint_trap();
53-
as_memcpy(newptr, ptr, old_size);
53+
memcpy(newptr, ptr, old_size);
5454
return newptr;
5555
} else if (new_size == FIELD(r, 1)) {
5656
// No need to grow

rts/motoko-rts/Cargo.toml

+11-1
Original file line numberDiff line numberDiff line change
@@ -7,9 +7,19 @@ edition = "2018"
77
[dependencies.libc]
88
version = "0.2.73"
99

10-
# added here so that it ends up in Cargo.lock, so that nix will pre-fetch it
10+
# Added here so that it ends up in Cargo.lock, so that nix will pre-fetch it
1111
[dependencies.compiler_builtins]
1212
version = "0.1.32"
13+
# Without this feature we get dozens of duplicate symbol errors when generating
14+
# the final shared .wasm file:
15+
#
16+
# wasm-ld: error: duplicate symbol: __multi3
17+
# >>> defined in _build/wasm/libmotoko_rts.a(compiler_builtins-d709bd899857aa61.compiler_builtins.3abndchk-cgu.0.rcgu.o)
18+
# >>> defined in _build/wasm/libmotoko_rts.a(compiler_builtins-06d1ead628e1f468.compiler_builtins.6moz1ltd-cgu.0.rcgu.o)
19+
#
20+
# It seems like we're linking multiple versions of compiler_builtins in the same
21+
# shared library, which we should fix at some point. TODO
22+
features = ["mangled-names"]
1323

1424
[lib]
1525
crate-type = ["staticlib"]

rts/motoko-rts/Xargo.toml

-1
Original file line numberDiff line numberDiff line change
@@ -3,5 +3,4 @@ stage = 0
33

44
[dependencies.compiler_builtins]
55
stage = 1
6-
features = [ "mem" ]
76
version = "0.1.32"

rts/motoko-rts/src/alloc.rs

+41
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
//! Implements allocation routines used by the generated code and the GC.
2+
3+
use core::arch::wasm32;
4+
5+
use crate::gc;
6+
use crate::rts_trap_with;
7+
use crate::types::{skew, Bytes, SkewedPtr, Words};
8+
9+
#[no_mangle]
10+
unsafe extern "C" fn alloc_bytes(n: Bytes<u32>) -> SkewedPtr {
11+
alloc_words(n.to_words())
12+
}
13+
14+
#[no_mangle]
15+
unsafe extern "C" fn alloc_words(n: Words<u32>) -> SkewedPtr {
16+
let bytes = n.to_bytes();
17+
// Update ALLOCATED
18+
gc::ALLOCATED += Bytes(bytes.0 as u64);
19+
20+
// Update heap pointer
21+
let old_hp = gc::HP;
22+
let new_hp = old_hp + bytes.0;
23+
gc::HP = new_hp;
24+
25+
// Grow memory if needed
26+
grow_memory(new_hp as usize);
27+
28+
skew(old_hp as usize)
29+
}
30+
31+
/// Page allocation. Ensures that the memory up to the given pointer is allocated.
32+
pub(crate) unsafe fn grow_memory(ptr: usize) {
33+
let total_pages_needed = ((ptr / 65536) + 1) as i32;
34+
let current_pages = wasm32::memory_size(0) as i32;
35+
let new_pages_needed = total_pages_needed - current_pages;
36+
if new_pages_needed > 0 {
37+
if wasm32::memory_grow(0, new_pages_needed as usize) == core::usize::MAX {
38+
rts_trap_with("Cannot grow memory\0".as_ptr());
39+
}
40+
}
41+
}

0 commit comments

Comments
 (0)