1
- use super :: Scratch ;
1
+ use super :: HeapManager ;
2
2
use super :: WASM_PAGE_BITS ;
3
3
use super :: WASM_PAGE_SIZE ;
4
- use super :: HeapManager ;
4
+ use wasmgen :: Scratch ;
5
5
6
6
use crate :: var_conv:: * ;
7
7
@@ -10,6 +10,9 @@ mod copy_funcs;
10
10
mod copy_indirect_elements;
11
11
mod do_cheney;
12
12
13
+ #[ cfg( feature = "wasmtest" ) ]
14
+ pub mod wasmtest;
15
+
13
16
/**
14
17
* Cheney is a GC implementation that uses Cheney's algorithm.
15
18
* At every alternate run of the GC, it will ensure that it has as much free space as used space (not including the swap space).
@@ -130,7 +133,7 @@ impl<'a, 'b, 'c> Cheney<'a, 'b, 'c> {
130
133
(*ptr) = I32_MIN | (new_ptr >> 1); // say that we already copied it.
131
134
return new_ptr;
132
135
}
133
- // `bytes_required`: min number of bytes we want
136
+ // `bytes_required`: min number of bytes we want (including the tag!)
134
137
// returns nonzero if successfully made enough space, otherwise zero.
135
138
fn do_cheney(bytes_required: u32) -> i32 {
136
139
if (end_mem_ptr != gc_roots_stack_base_ptr) {
@@ -274,7 +277,9 @@ impl<'a, 'b, 'c> Cheney<'a, 'b, 'c> {
274
277
}
275
278
}
276
279
277
- fn filter_roots ( local_roots : & [ ( ir:: VarType , wasmgen:: LocalIdx ) ] ) -> Box < [ ( ir:: VarType , wasmgen:: LocalIdx ) ] > {
280
+ fn filter_roots (
281
+ local_roots : & [ ( ir:: VarType , wasmgen:: LocalIdx ) ] ,
282
+ ) -> Box < [ ( ir:: VarType , wasmgen:: LocalIdx ) ] > {
278
283
local_roots
279
284
. iter ( )
280
285
. cloned ( )
@@ -286,10 +291,10 @@ impl<'a, 'b, 'c> Cheney<'a, 'b, 'c> {
286
291
_ => true ,
287
292
} )
288
293
. collect ( )
289
- }
294
+ }
290
295
291
296
// Helper function used to encode heap allocation.
292
- // `f` should be a function that has net wasm stack [] -> [i32(size)]
297
+ // `f` should be a function that has net wasm stack [] -> [i32(size)], it pushes the bytes required (including tag) on the stack.
293
298
// net wasm stack: [] -> [i32(ptr)]
294
299
fn encode_allocation < F : Fn ( & mut wasmgen:: ExprBuilder ) -> ( ) > (
295
300
& self ,
@@ -322,6 +327,7 @@ impl<'a, 'b, 'c> Cheney<'a, 'b, 'c> {
322
327
label:
323
328
ret = free_mem_ptr; // ret is the value that is left on the stack
324
329
free_mem_ptr += size;
330
+ ret += 4;
325
331
*/
326
332
327
333
// (end_mem_ptr - free_mem_ptr < size)
@@ -369,6 +375,8 @@ impl<'a, 'b, 'c> Cheney<'a, 'b, 'c> {
369
375
encode_size ( expr_builder) ;
370
376
expr_builder. i32_add ( ) ;
371
377
expr_builder. global_set ( self . free_mem_ptr ) ;
378
+ expr_builder. i32_const ( 4 ) ;
379
+ expr_builder. i32_add ( ) ;
372
380
}
373
381
}
374
382
@@ -396,7 +404,7 @@ impl<'a, 'b, 'c> HeapManager for Cheney<'a, 'b, 'c> {
396
404
self . encode_allocation (
397
405
|expr_builder| {
398
406
// net wasm stack: [] -> [i32(size)]
399
- expr_builder. i32_const ( size as i32 ) ;
407
+ expr_builder. i32_const ( ( size + 4 ) as i32 ) ;
400
408
} ,
401
409
local_roots,
402
410
scratch,
@@ -452,9 +460,9 @@ impl<'a, 'b, 'c> HeapManager for Cheney<'a, 'b, 'c> {
452
460
match ir_vartype {
453
461
ir:: VarType :: String => {
454
462
let localidx_mem_size: wasmgen:: LocalIdx = scratch. push_i32 ( ) ;
455
- // Algorithm: mem_size = ((num_bytes + 7 ) & (~3)) // equivalent to (4 + round_up_to_multiple_of_4(num_bytes))
463
+ // Algorithm: mem_size = ((num_bytes + 11 ) & (~3)) // equivalent to (4 + round_up_to_multiple_of_4(num_bytes))
456
464
// net wasm stack: [i32(num_bytes)] -> []
457
- expr_builder. i32_const ( 7 ) ;
465
+ expr_builder. i32_const ( 11 ) ;
458
466
expr_builder. i32_add ( ) ;
459
467
expr_builder. i32_const ( -4 ) ; // equivalent to (~3) in two's complement
460
468
expr_builder. i32_and ( ) ;
@@ -501,8 +509,9 @@ impl<'a, 'b, 'c> HeapManager for Cheney<'a, 'b, 'c> {
501
509
scratch : & mut Scratch ,
502
510
expr_builder : & mut wasmgen:: ExprBuilder ,
503
511
) -> Self :: RootsStackHandle {
504
- let filtered_roots: Box < [ ( ir:: VarType , wasmgen:: LocalIdx ) ] > = Self :: filter_roots ( local_roots) ;
505
-
512
+ let filtered_roots: Box < [ ( ir:: VarType , wasmgen:: LocalIdx ) ] > =
513
+ Self :: filter_roots ( local_roots) ;
514
+
506
515
// if there are no roots to add, then we don't need to load the gc_roots_stack_ptr.
507
516
// net wasm stack: [] -> []
508
517
if !filtered_roots. is_empty ( ) {
@@ -540,8 +549,9 @@ impl<'a, 'b, 'c> HeapManager for Cheney<'a, 'b, 'c> {
540
549
scratch : & mut Scratch ,
541
550
expr_builder : & mut wasmgen:: ExprBuilder ,
542
551
) {
543
- let filtered_roots: Box < [ ( ir:: VarType , wasmgen:: LocalIdx ) ] > = Self :: filter_roots ( local_roots) ;
544
-
552
+ let filtered_roots: Box < [ ( ir:: VarType , wasmgen:: LocalIdx ) ] > =
553
+ Self :: filter_roots ( local_roots) ;
554
+
545
555
if !filtered_roots. is_empty ( ) {
546
556
let localidx_gc_roots_stack_ptr = scratch. push_i32 ( ) ;
547
557
0 commit comments