#include "heap.h" #include "system.h" #include "common.h" #define CHAIN_HEADER_SIZE divide(sizeof(Segment::Chain), BytesPerWord) using namespace vm; namespace { // an object must survive TenureThreshold + 2 garbage collections // before being copied to gen2: const unsigned TenureThreshold = 3; const unsigned MinimumGen1Size = 64 * 1024; const unsigned MinimumGen2Size = 128 * 1024; const unsigned Top = ~static_cast(0); const bool Verbose = true; const bool Debug = true; class Context; System* system(Context*); void NO_RETURN abort(Context*); void assert(Context*, bool); class Segment { public: class Map { public: class Chain; class Iterator { public: Map* map; unsigned index; unsigned limit; Iterator(Map* map, unsigned start, unsigned end): map(map) { assert(map->segment->context, map); assert(map->segment->context, map->bitsPerRecord == 1); assert(map->segment->context, map->segment); assert(map->segment->context, start <= map->segment->position()); if (end > map->segment->position()) end = map->segment->position(); index = map->indexOf(start); limit = map->indexOf(end); if ((end - start) % map->scale) ++ limit; } bool hasMore() { assert(map->segment->context, map); unsigned word = wordOf(index); unsigned bit = bitOf(index); unsigned wordLimit = wordOf(limit); unsigned bitLimit = bitOf(limit); for (; word <= wordLimit and (word < wordLimit or bit < bitLimit); ++word) { uintptr_t* p = map->data() + word; if (*p) { for (; bit < BitsPerWord and (word < wordLimit or bit < bitLimit); ++bit) { if (map->data()[word] & (static_cast(1) << bit)) { index = ::indexOf(word, bit); // printf("hit at index %d\n", index); return true; } else { // printf("miss at index %d\n", indexOf(word, bit)); } } } bit = 0; } index = limit; return false; } unsigned next() { assert(map->segment->context, hasMore()); assert(map->segment->context, map->segment); return (index++) * map->scale; } }; Segment* segment; unsigned bitsPerRecord; unsigned scale; Map* child; Map(Segment* segment = 0, unsigned bitsPerRecord = 1, unsigned scale = 1, Map* child = 0): segment(segment), bitsPerRecord(bitsPerRecord), scale(scale), child(child) { if (segment) { assert(segment->context, bitsPerRecord); assert(segment->context, scale); assert(segment->context, powerOfTwo(scale)); } } void replaceWith(Map* m) { assert(segment->context, m); assert(segment->context, bitsPerRecord == m->bitsPerRecord); assert(segment->context, scale == m->scale); m->segment = 0; if (child) child->replaceWith(m->child); } unsigned offset(unsigned capacity) { unsigned n = 0; if (child) n += child->footprint(capacity); return n; } unsigned offset() { return offset(segment->capacity()); } uintptr_t* data() { return segment->rear->data() + segment->rear->capacity + offset(); } unsigned size(unsigned capacity) { unsigned result = divide(divide(capacity, scale) * bitsPerRecord, BitsPerWord); assert(segment->context, result); return result; } unsigned size() { return size(max(segment->capacity(), 1)); } unsigned indexOf(unsigned segmentIndex) { return (segmentIndex / scale) * bitsPerRecord; } unsigned indexOf(void* p) { assert(segment->context, segment); assert(segment->context, segment->almostContains(p)); assert(segment->context, segment->capacity()); return indexOf(segment->indexOf(p)); } void update(uintptr_t* newData, unsigned capacity) { assert(segment->context, capacity >= segment->capacity()); uintptr_t* p = newData + offset(capacity); memcpy(p, data(), size(segment->position()) * BytesPerWord); if (child) { child->update(newData, capacity); } } void clear() { memset(data(), 0, size() * BytesPerWord); if (child) child->clear(); } void clearBit(unsigned i) { assert(segment->context, wordOf(i) * BytesPerWord < size()); data()[wordOf(i)] &= ~(static_cast(1) << bitOf(i)); } void setBit(unsigned i) { assert(segment->context, wordOf(i) * BytesPerWord < size()); data()[wordOf(i)] |= static_cast(1) << bitOf(i); } void clearOnlyIndex(unsigned index) { for (unsigned i = index, limit = index + bitsPerRecord; i < limit; ++i) { clearBit(i); } } void clearOnly(unsigned segmentIndex) { clearOnlyIndex(indexOf(segmentIndex)); } void clearOnly(void* p) { clearOnlyIndex(indexOf(p)); } void clear(void* p) { clearOnly(p); if (child) child->clear(p); } void setOnlyIndex(unsigned index, unsigned v = 1) { unsigned i = index + bitsPerRecord - 1; while (true) { if (v & 1) setBit(i); else clearBit(i); v >>= 1; if (i == index) break; --i; } } void setOnly(unsigned segmentIndex, unsigned v = 1) { setOnlyIndex(indexOf(segmentIndex), v); } void setOnly(void* p, unsigned v = 1) { setOnlyIndex(indexOf(p), v); } void set(void* p, unsigned v = 1) { setOnly(p, v); assert(segment->context, get(p) == v); if (child) child->set(p, v); } unsigned get(void* p) { unsigned index = indexOf(p); unsigned v = 0; for (unsigned i = index, limit = index + bitsPerRecord; i < limit; ++i) { unsigned wi = bitOf(i); v <<= 1; v |= ((data()[wordOf(i)]) & (static_cast(1) << wi)) >> wi; } return v; } unsigned footprint(unsigned capacity) { unsigned n = size(capacity); if (child) n += child->footprint(capacity); return n; } void setSegment(Segment* s) { segment = s; if (child) child->setSegment(s); } }; class Chain { public: Segment* segment; unsigned offset; unsigned position; unsigned capacity; Chain* next; Chain* previous; Chain(Segment* segment, unsigned offset, unsigned capacity, Chain* previous): segment(segment), offset(offset), position(0), capacity(capacity), next(0), previous(previous) { assert(segment->context, sizeof(Chain) % BytesPerWord == 0); } static void dispose(Chain* c) { if (c) { if (c->next) dispose(c->next); system(c->segment->context)->free(c); } } uintptr_t* data() { return reinterpret_cast(this) + CHAIN_HEADER_SIZE; } static unsigned footprint(unsigned capacity, Map* map) { unsigned n = CHAIN_HEADER_SIZE + capacity; if (map) { n += map->footprint(capacity); } return n; } unsigned footprint() { return footprint(capacity, segment->map); } }; Context* context; Chain* front; Chain* rear; Map* map; Segment(Context* context, unsigned capacity, Map* map = 0, bool clearMap = true): context(context), front(0), rear(0), map(map) { front = rear = 0; this->map = map; if (capacity) { front = rear = new (system(context)->allocate (Chain::footprint(capacity, map) * BytesPerWord)) Chain(this, 0, capacity, 0); if (map) { map->setSegment(this); if (clearMap) { map->clear(); } } } } unsigned capacity() { return (rear? rear->offset + rear->capacity : 0); } unsigned position() { return (rear? rear->offset + rear->position : 0); } void truncate(unsigned offset) { assert(context, offset <= position()); for (Chain* c = front; c; c = c->next) { if (offset >= c->offset and offset <= c->offset + c->position) { c->position = offset - c->offset; Chain::dispose(c->next); return; } } abort(context); } void copyTo(void* p) { for (Chain* c = front; c; c = c->next) { if (c->position) { memcpy(static_cast(p) + c->offset, c->data(), c->position * BytesPerWord); } else { return; } } } unsigned footprint() { unsigned n = 0; for (Chain* c = front; c; c = c->next) n += c->footprint(); return n; } unsigned remaining() { return capacity() - position(); } void replaceWith(Segment* s) { Chain::dispose(front); front = s->front; rear = s->rear; s->front = s->rear = 0; if (s->map) { if (map) { map->replaceWith(s->map); } else { map = s->map; map->setSegment(this); } s->map = 0; } else { map = 0; } } bool contains(void* p) { for (Chain* c = front; c; c = c->next) { if (c->position and p >= c->data() and p < c->data() + c->position) { return true; } } return false; } bool almostContains(void* p) { return contains(p) or p == rear->data() + rear->position; } void* get(unsigned offset) { for (Chain* c = front; c; c = c->next) { if (c->position and offset >= c->offset and offset < c->offset + c->position) { return c->data() + (offset - c->offset); } } if (offset == rear->offset + rear->position) { return rear->data() + (offset - rear->offset); } abort(context); } unsigned indexOf(void* p) { for (Chain* c = front; c; c = c->next) { if (c->position and p >= c->data() and p < c->data() + c->position) { return (static_cast(p) - c->data()) + c->offset; } } if (p == rear->data() + rear->position) { return (static_cast(p) - rear->data()) + rear->offset; } abort(context); } void* allocate(unsigned sizeInBytes) { unsigned size = divide(sizeInBytes, BytesPerWord); assert(context, size); assert(context, rear->position + size <= rear->capacity); void* p = reinterpret_cast(rear->data()) + rear->position; rear->position += size; return p; } void* add(void* p, unsigned size) { void* target = allocate(size); memcpy(target, p, size * BytesPerWord); return target; } void ensure(unsigned minimum) { if (remaining() < minimum) { if (Verbose) { fprintf(stderr, "grow\n"); } unsigned desired = (position() + minimum) * 2; void* p = 0; unsigned newCapacity = desired; while (p == 0) { p = system(context)->tryAllocate (Chain::footprint(newCapacity, map) * BytesPerWord); if (p == 0) { if (newCapacity > minimum) { newCapacity = minimum + ((newCapacity - minimum) / 2); } else { abort(context); } } } Chain* c = new (p) Chain (this, rear->offset + rear->position, newCapacity, rear); if (map) { map->update(c->data() + c->capacity, newCapacity); } rear->previous = c; rear = c; } } void dispose() { Chain::dispose(front); front = rear = 0; map = 0; } }; enum CollectionMode { MinorCollection, MajorCollection, OverflowCollection, Gen2Collection }; class Context { public: Context(System* system): system(system), client(0), gen1(this, 0), nextGen1(this, 0), gen2(this, 0), nextGen2(this, 0) { } void dispose() { gen1.dispose(); nextGen1.dispose(); gen2.dispose(); nextGen2.dispose(); } System* system; Heap::Client* client; Segment gen1; Segment nextGen1; Segment gen2; Segment nextGen2; unsigned gen2Base; Segment::Map ageMap; Segment::Map nextAgeMap; Segment::Map pointerMap; Segment::Map pageMap; Segment::Map heapMap; CollectionMode mode; }; inline System* system(Context* c) { return c->system; } inline void NO_RETURN abort(Context* c) { c->system->abort(); // this should not return ::abort(); } inline void assert(Context* c, bool v) { if (UNLIKELY(not v)) abort(c); } void initGen1(Context* c) { new (&(c->ageMap)) Segment::Map(&(c->gen1), log(TenureThreshold)); new (&(c->gen1)) Segment (c, MinimumGen1Size / BytesPerWord, &(c->ageMap), false); } void initGen2(Context* c) { new (&(c->pointerMap)) Segment::Map(&(c->gen2)); new (&(c->pageMap)) Segment::Map (&(c->gen2), 1, LikelyPageSize / BytesPerWord, &(c->pointerMap)); new (&(c->heapMap)) Segment::Map (&(c->gen2), 1, c->pageMap.scale * 1024, &(c->pageMap)); new (&(c->gen2)) Segment(c, MinimumGen2Size / BytesPerWord, &(c->heapMap)); } void initNextGen1(Context* c) { unsigned size = max(MinimumGen1Size / BytesPerWord, nextPowerOfTwo(c->gen1.position())); new (&(c->nextAgeMap)) Segment::Map(&(c->nextGen1), log(TenureThreshold)); new (&(c->nextGen1)) Segment(c, size, &(c->nextAgeMap), false); } void initNextGen2(Context* c) { unsigned size = max(MinimumGen2Size / BytesPerWord, nextPowerOfTwo(c->gen2.position())); new (&(c->pointerMap)) Segment::Map(&(c->nextGen2)); new (&(c->pageMap)) Segment::Map (&(c->nextGen2), 1, LikelyPageSize / BytesPerWord, &(c->pointerMap)); new (&(c->heapMap)) Segment::Map (&(c->nextGen2), 1, c->pageMap.scale * 1024, &(c->pageMap)); new (&(c->nextGen2)) Segment(c, size, &(c->heapMap)); c->gen2.map = 0; } inline bool fresh(Context* c, object o) { return c->nextGen1.contains(o) or c->nextGen2.contains(o) or (c->gen2.contains(o) and c->gen2.indexOf(o) > c->gen2Base); } inline bool wasCollected(Context* c, object o) { return o and (not fresh(c, o)) and fresh(c, cast(o, 0)); } inline object follow(Context* c, object o) { assert(c, wasCollected(c, o)); return cast(o, 0); } inline object& parent(Context* c, object o) { assert(c, wasCollected(c, o)); return cast(o, BytesPerWord); } inline uintptr_t* bitset(Context* c, object o) { assert(c, wasCollected(c, o)); return &cast(o, BytesPerWord * 2); } inline object copyTo(Context*, Segment* s, object o, unsigned size) { s->ensure(size); return static_cast(s->add(o, size)); } object copy2(Context* c, object o) { unsigned size = c->client->sizeInWords(o); if (c->gen2.contains(o)) { assert(c, c->mode == MajorCollection or c->mode == Gen2Collection); return copyTo(c, &(c->nextGen2), o, size); } else if (c->gen1.contains(o)) { unsigned age = c->ageMap.get(o); if (age == TenureThreshold) { if (c->mode == MinorCollection) { if (c->gen2.front == 0) initGen2(c); if (c->gen2.remaining() >= size) { if (c->gen2Base == Top) { c->gen2Base = c->gen2.position(); } return copyTo(c, &(c->gen2), o, size); } else { if (Verbose) { fprintf(stderr, "overflow collection\n"); } c->mode = OverflowCollection; initNextGen2(c); return copyTo(c, &(c->nextGen2), o, size); } } else { return copyTo(c, &(c->nextGen2), o, size); } } else { o = copyTo(c, &(c->nextGen1), o, size); c->nextAgeMap.setOnly(o, age + 1); return o; } } else { assert(c, not c->nextGen1.contains(o)); assert(c, not c->nextGen2.contains(o)); o = copyTo(c, &(c->nextGen1), o, size); c->nextAgeMap.clear(o); return o; } } object copy(Context* c, object o) { object r = copy2(c, o); if (Debug) { fprintf(stderr, "copy %p to %p\n", o, r); } // leave a pointer to the copy in the original cast(o, 0) = r; return r; } object update3(Context* c, object* p, bool* needsVisit) { if (wasCollected(c, *p)) { *needsVisit = false; return follow(c, *p); } else { *needsVisit = true; return copy(c, *p); } } object update2(Context* c, object* p, bool* needsVisit) { switch (c->mode) { case MinorCollection: case OverflowCollection: if (c->gen2.contains(*p)) { *needsVisit = false; return *p; } break; case Gen2Collection: if (c->gen2.contains(*p)) { return update3(c, p, needsVisit); } else { *needsVisit = false; return *p; } break; default: break; } return update3(c, p, needsVisit); } object update(Context* c, object* p, bool* needsVisit) { if (*p == 0) { *needsVisit = false; return *p; } object r = update2(c, p, needsVisit); // update heap map. if (r) { if (c->mode == MinorCollection) { if (c->gen2.contains(p) and not c->gen2.contains(r)) { c->heapMap.set(p); } } else { if (c->nextGen2.contains(p) and not c->nextGen2.contains(r)) { c->heapMap.set(p); } } } return r; } const uintptr_t BitsetExtensionBit = (static_cast(1) << (BitsPerWord - 1)); void bitsetInit(uintptr_t* p) { memset(p, 0, BytesPerWord); } void bitsetClear(uintptr_t* p, unsigned start, unsigned end) { if (end < BitsPerWord - 1) { // do nothing } else if (start < BitsPerWord - 1) { memset(p + 1, 0, (wordOf(end + (BitsPerWord * 2) + 1)) * BytesPerWord); } else { unsigned startWord = wordOf(start + (BitsPerWord * 2) + 1); unsigned endWord = wordOf(end + (BitsPerWord * 2) + 1); if (endWord > startWord) { memset(p + startWord + 1, 0, (endWord - startWord) * BytesPerWord); } } } void bitsetSet(uintptr_t* p, unsigned i, bool v) { if (i >= BitsPerWord - 1) { i += (BitsPerWord * 2) + 1; if (v) { p[0] |= BitsetExtensionBit; if (p[2] <= wordOf(i) - 3) p[2] = wordOf(i) - 2; } } if (v) { p[wordOf(i)] |= static_cast(1) << bitOf(i); } else { p[wordOf(i)] &= ~(static_cast(1) << bitOf(i)); } } unsigned bitsetHasMore(uintptr_t* p) { switch (*p) { case 0: return false; case BitsetExtensionBit: { uintptr_t length = p[2]; uintptr_t word = wordOf(p[1]); for (; word < length; ++word) { if (p[word + 3]) { p[1] = indexOf(word, 0); return true; } } p[1] = indexOf(word, 0); return false; } default: return true; } } unsigned bitsetNext(Context* c, uintptr_t* p) { assert(c, bitsetHasMore(p)); switch (*p) { case 0: abort(c); case BitsetExtensionBit: { uintptr_t i = p[1]; uintptr_t word = wordOf(i); assert(c, word < p[2]); for (uintptr_t bit = bitOf(i); bit < BitsPerWord; ++bit) { if (p[word + 3] & (static_cast(1) << bit)) { p[1] = indexOf(word, bit) + 1; bitsetSet(p, p[1] + BitsPerWord - 2, false); return p[1] + BitsPerWord - 2; } } abort(c); } default: { for (unsigned i = 0; i < BitsPerWord - 1; ++i) { if (*p & (static_cast(1) << i)) { bitsetSet(p, i, false); return i; } } abort(c); } } } void collect(Context* c, void** p) { object original = *p; object parent = 0; if (Debug) { fprintf(stderr, "update %p at %p\n", *p, p); } bool needsVisit; *p = update(c, p, &needsVisit); if (Debug) { fprintf(stderr, " result: %p (visit? %d)\n", *p, needsVisit); } if (not needsVisit) return; visit: { object copy = follow(c, original); class Walker : public Heap::Walker { public: Walker(Context* c, object copy, uintptr_t* bitset): c(c), copy(copy), bitset(bitset), first(0), second(0), last(0), visits(0), total(0) { } virtual bool visit(unsigned offset) { if (Debug) { fprintf(stderr, " update %p at %p - offset %d from %p\n", cast(copy, offset * BytesPerWord), &cast(copy, offset * BytesPerWord), offset, copy); } bool needsVisit; object childCopy = update (c, &cast(copy, offset * BytesPerWord), &needsVisit); if (Debug) { fprintf(stderr, " result: %p (visit? %d)\n", childCopy, needsVisit); } ++ total; if (total == 3) { bitsetInit(bitset); } if (needsVisit) { ++ visits; if (visits == 1) { first = offset; } else if (visits == 2) { second = offset; } } else { cast(copy, offset * BytesPerWord) = childCopy; } if (visits > 1 and total > 2 and (second or needsVisit)) { bitsetClear(bitset, last, offset); last = offset; if (second) { bitsetSet(bitset, second, true); second = 0; } if (needsVisit) { bitsetSet(bitset, offset, true); } } return true; } Context* c; object copy; uintptr_t* bitset; unsigned first; unsigned second; unsigned last; unsigned visits; unsigned total; } walker(c, copy, bitset(c, original)); if (Debug) { fprintf(stderr, "walk %p\n", copy); } c->client->walk(copy, &walker); if (walker.visits) { // descend if (walker.visits > 1) { ::parent(c, original) = parent; parent = original; } original = cast(copy, walker.first * BytesPerWord); cast(copy, walker.first * BytesPerWord) = follow(c, original); goto visit; } else { // ascend original = parent; } } if (original) { object copy = follow(c, original); class Walker : public Heap::Walker { public: Walker(Context* c, uintptr_t* bitset): c(c), bitset(bitset), next(0), total(0) { } virtual bool visit(unsigned offset) { switch (++ total) { case 1: return true; case 2: next = offset; return true; case 3: next = bitsetNext(c, bitset); return false; default: abort(c); } } Context* c; uintptr_t* bitset; unsigned next; unsigned total; } walker(c, bitset(c, original)); if (Debug) { fprintf(stderr, "scan %p\n", copy); } c->client->walk(copy, &walker); assert(c, walker.total > 1); if (walker.total == 3 and bitsetHasMore(bitset(c, original))) { parent = original; } else { parent = ::parent(c, original); } if (Debug) { fprintf(stderr, " next is %p at %p - offset %d from %p\n", cast(copy, walker.next * BytesPerWord), &cast(copy, walker.next * BytesPerWord), walker.next, copy); } original = cast(copy, walker.next * BytesPerWord); cast(copy, walker.next * BytesPerWord) = follow(c, original); goto visit; } else { return; } } void collect(Context* c, Segment::Map* map, unsigned start, unsigned end, bool* dirty, bool expectDirty) { bool wasDirty = false; for (Segment::Map::Iterator it(map, start, end); it.hasMore();) { wasDirty = true; if (map->child) { assert(c, map->scale > 1); unsigned s = it.next(); unsigned e = s + map->scale; map->clearOnly(s); bool childDirty = false; collect(c, map->child, s, e, &childDirty, true); if (c->mode == OverflowCollection) { return; } else if (childDirty) { map->setOnly(s); *dirty = true; } } else { assert(c, map->scale == 1); object* p = reinterpret_cast(map->segment->get(it.next())); map->clearOnly(p); if (c->nextGen1.contains(*p)) { map->setOnly(p); *dirty = true; } else { collect(c, p); if (c->mode == OverflowCollection) { return; } else if (c->gen2.contains(*p)) { // done } else { map->setOnly(p); *dirty = true; } } } } assert(c, wasDirty or not expectDirty); } class ObjectSegmentIterator { public: Context* context; Segment::Chain* chain; unsigned index; unsigned end; bool dirty; ObjectSegmentIterator(Segment* segment, unsigned end): context(segment->context), chain(segment->front), index(0), end(end), dirty(false) { } bool hasNext() { if (dirty) { dirty = false; uintptr_t* p = chain->data() + index; index += context->client->sizeInWords(p); } if (chain and index == chain->position) { chain = chain->next; index = 0; } return chain and index + chain->offset < end; } object next() { dirty = true; return chain->data() + index; } }; void collect(Context* c, Segment* s, unsigned limit) { for (ObjectSegmentIterator it(s, limit); it.hasNext();) { object p = it.next(); class Walker : public Heap::Walker { public: Walker(Context* c, object p): c(c), p(p) { } virtual bool visit(unsigned offset) { collect(c, &cast(p, offset * BytesPerWord)); return true; } Context* c; object p; } walker(c, p); c->client->walk(p, &walker); } } void collect2(Context* c) { if (c->mode == MinorCollection and c->gen2.position()) { unsigned start = 0; unsigned end = start + c->gen2.position(); bool dirty; collect(c, &(c->heapMap), start, end, &dirty, false); } else if (c->mode == Gen2Collection) { unsigned ng2Position = c->nextGen2.position(); collect(c, &(c->nextGen1), c->nextGen1.position()); collect(c, &(c->nextGen2), ng2Position); } class Visitor : public Heap::Visitor { public: Visitor(Context* c): c(c) { } virtual void visit(void** p) { collect(c, p); } Context* c; } v(c); c->client->visitRoots(&v); } void collect(Context* c) { if (c->gen1.front == 0) initGen1(c); c->gen2Base = Top; switch (c->mode) { case MinorCollection: { initNextGen1(c); if (Verbose) { fprintf(stderr, "minor collection\n"); } collect2(c); if (c->mode == OverflowCollection) { c->mode = Gen2Collection; if (Verbose) { fprintf(stderr, "gen2 collection\n"); } c->gen2Base = Top; collect2(c); c->gen2.replaceWith(&(c->nextGen2)); } c->gen1.replaceWith(&(c->nextGen1)); } break; case MajorCollection: { initNextGen1(c); initNextGen2(c); c->heapMap.clear(); if (Verbose) { fprintf(stderr, "major collection\n"); } collect2(c); c->gen1.replaceWith(&(c->nextGen1)); c->gen2.replaceWith(&(c->nextGen2)); } break; default: abort(c); } } } // namespace namespace vm { Heap* makeHeap(System* system) { class Heap: public vm::Heap { public: Heap(System* system): c(system) { } virtual void collect(CollectionType type, Client* client) { switch (type) { case MinorCollection: c.mode = ::MinorCollection; break; case MajorCollection: c.mode = ::MajorCollection; break; default: abort(&c); } c.client = client; ::collect(&c); } virtual bool needsMark(void** p) { return *p and c.gen2.contains(p) and not c.gen2.contains(*p); } virtual void mark(void** p) { c.heapMap.set(p); } virtual void dispose() { c.dispose(); c.system->free(this); } virtual void* follow(void* p) { if (wasCollected(&c, p)) { if (Debug) { fprintf(stderr, "follow: %p: %p\n", p, ::follow(&c, p)); } return ::follow(&c, p); } else { return p; } } Context c; }; return new (system->allocate(sizeof(Heap))) Heap(system); } } // namespace vm