2019-03-09 02:04:08 +01:00
|
|
|
import "allocator/tlsf";
|
2019-03-09 02:37:05 +01:00
|
|
|
|
|
|
|
var register_ref: usize = 0;
|
|
|
|
|
2019-03-10 21:38:15 +01:00
|
|
|
@global namespace gc {
|
|
|
|
export function register(ref: usize): void {
|
|
|
|
register_ref = ref;
|
|
|
|
}
|
|
|
|
export function link(ref: usize, parentRef: usize): void {
|
|
|
|
}
|
2019-03-09 02:37:05 +01:00
|
|
|
}
|
|
|
|
|
2019-03-09 02:04:08 +01:00
|
|
|
import {
|
|
|
|
HEADER,
|
|
|
|
HEADER_SIZE,
|
|
|
|
HEADER_MAGIC,
|
2019-03-10 21:38:15 +01:00
|
|
|
ADJUST,
|
2019-03-09 02:04:08 +01:00
|
|
|
ALLOC,
|
|
|
|
REALLOC,
|
|
|
|
FREE,
|
2019-03-09 02:44:46 +01:00
|
|
|
REGISTER,
|
|
|
|
ArrayBufferBase,
|
|
|
|
StringBase
|
2019-03-09 02:37:05 +01:00
|
|
|
} from "runtime";
|
2019-03-09 02:04:08 +01:00
|
|
|
|
|
|
|
class A {}
|
|
|
|
class B {}
|
2019-03-10 21:38:15 +01:00
|
|
|
assert(gc.classId<A>() != gc.classId<B>());
|
2019-03-09 02:04:08 +01:00
|
|
|
|
|
|
|
function isPowerOf2(x: i32): bool {
|
|
|
|
return x != 0 && (x & (x - 1)) == 0;
|
|
|
|
}
|
|
|
|
|
2019-03-10 21:38:15 +01:00
|
|
|
assert(ADJUST(0) > 0);
|
2019-03-09 02:04:08 +01:00
|
|
|
for (let i = 0; i < 9000; ++i) {
|
2019-03-10 21:38:15 +01:00
|
|
|
assert(isPowerOf2(ADJUST(i)));
|
2019-03-09 02:04:08 +01:00
|
|
|
}
|
|
|
|
|
2019-03-10 21:38:15 +01:00
|
|
|
var barrier1 = ADJUST(0);
|
2019-03-09 02:04:08 +01:00
|
|
|
var barrier2 = barrier1 + 1;
|
2019-03-10 21:38:15 +01:00
|
|
|
while (ADJUST(barrier2 + 1) == ADJUST(barrier2)) ++barrier2;
|
2019-03-09 02:04:08 +01:00
|
|
|
var barrier3 = barrier2 + 1;
|
2019-03-10 21:38:15 +01:00
|
|
|
while (ADJUST(barrier3 + 1) == ADJUST(barrier3)) ++barrier3;
|
2019-03-09 02:04:08 +01:00
|
|
|
|
|
|
|
trace("barrier1", 1, barrier1);
|
|
|
|
trace("barrier2", 1, barrier2);
|
|
|
|
trace("barrier3", 1, barrier3);
|
|
|
|
|
|
|
|
var ref1 = ALLOC(1);
|
|
|
|
var header1 = changetype<HEADER>(ref1 - HEADER_SIZE);
|
|
|
|
assert(header1.classId == HEADER_MAGIC);
|
|
|
|
assert(header1.payloadSize == 1);
|
|
|
|
assert(ref1 == REALLOC(ref1, barrier1)); // same segment
|
|
|
|
assert(header1.payloadSize == barrier1);
|
|
|
|
var ref2 = REALLOC(ref1, barrier2);
|
|
|
|
assert(ref1 != ref2); // moves
|
|
|
|
var header2 = changetype<HEADER>(ref2 - HEADER_SIZE);
|
|
|
|
assert(header2.payloadSize == barrier2);
|
|
|
|
FREE(ref2);
|
|
|
|
var ref3 = ALLOC(barrier2);
|
|
|
|
assert(ref1 == ref3); // reuses space of ref1 (free'd in realloc), ref2 (explicitly free'd)
|
|
|
|
|
|
|
|
var ref4 = ALLOC(barrier1);
|
2019-03-10 02:57:05 +01:00
|
|
|
REGISTER<A>(ref4); // should call __REGISTER_IMPL
|
2019-03-09 02:37:05 +01:00
|
|
|
assert(register_ref == ref4);
|
|
|
|
var header4 = changetype<HEADER>(register_ref - HEADER_SIZE);
|
2019-03-10 21:38:15 +01:00
|
|
|
assert(header4.classId == gc.classId<A>());
|
2019-03-09 02:37:05 +01:00
|
|
|
assert(header4.payloadSize == barrier1);
|
2019-03-09 02:44:46 +01:00
|
|
|
|
|
|
|
var ref5 = ALLOC(10);
|
|
|
|
assert(changetype<ArrayBufferBase>(ref5).byteLength == 10);
|
|
|
|
assert(changetype<StringBase>(ref5).length == 5);
|