move fixed object (mark and sweep) support into heap.cpp and refine algorithms for determining when and how much to GC

This commit is contained in:
Joel Dice 2007-10-28 13:14:53 -06:00
parent 60072b9fdc
commit 7f1837fecd
9 changed files with 463 additions and 500 deletions

View File

@ -28,7 +28,7 @@ src = src
classpath = classpath classpath = classpath
test = test test = test
input = $(test-build)/Enums.class input = $(test-build)/GC.class
build-cxx = g++ build-cxx = g++
build-cc = gcc build-cc = gcc

View File

@ -232,10 +232,6 @@ bitsToFloat(uint32_t bits)
return f; return f;
} }
// an object must survive TenureThreshold + 2 garbage collections
// before being copied to gen2 (muat be at least 1):
const unsigned TenureThreshold = 3;
class Machine; class Machine;
class Thread; class Thread;

View File

@ -674,7 +674,7 @@ class StackMapper {
MyProtector(StackMapper* mapper): Protector(mapper->t), mapper(mapper) { } MyProtector(StackMapper* mapper): Protector(mapper->t), mapper(mapper) { }
virtual void visit(Heap::Visitor* v) { virtual void visit(Heap::Visitor* v) {
vm::visit(t, v, &(mapper->method)); v->visit(&(mapper->method));
} }
StackMapper* mapper; StackMapper* mapper;
@ -1073,14 +1073,14 @@ visitParameters(MyThread* t, Heap::Visitor* v, void* frame)
unsigned index = 0; unsigned index = 0;
if ((methodFlags(t, method) & ACC_STATIC) == 0) { if ((methodFlags(t, method) & ACC_STATIC) == 0) {
visit(t, v, frameLocalObject(t, frame, index++)); v->visit(frameLocalObject(t, frame, index++));
} }
for (MethodSpecIterator it(t, spec); it.hasNext();) { for (MethodSpecIterator it(t, spec); it.hasNext();) {
switch (*it.next()) { switch (*it.next()) {
case 'L': case 'L':
case '[': case '[':
visit(t, v, frameLocalObject(t, frame, index++)); v->visit(frameLocalObject(t, frame, index++));
break; break;
case 'J': case 'J':
@ -1122,7 +1122,7 @@ visitStackAndLocals(MyThread* t, Heap::Visitor* v, void* frame)
for (unsigned i = 0; i < count; ++i) { for (unsigned i = 0; i < count; ++i) {
if (getBit(stackMap, i)) { if (getBit(stackMap, i)) {
visit(t, v, frameLocalObject(t, frame, i + parameterFootprint)); v->visit(frameLocalObject(t, frame, i + parameterFootprint));
} }
} }
} }
@ -1134,7 +1134,7 @@ visitStack(MyThread* t, Heap::Visitor* v)
void* frame = frameStart(t); void* frame = frameStart(t);
if (frameValid(frame)) { if (frameValid(frame)) {
visit(t, v, &frameMethod(frame)); v->visit(&frameMethod(frame));
} }
for (; frameValid(frame); frame = frameNext(frame)) { for (; frameValid(frame); frame = frameNext(frame)) {
@ -1142,7 +1142,7 @@ visitStack(MyThread* t, Heap::Visitor* v)
// caller is native. Otherwise, the caller owns them. // caller is native. Otherwise, the caller owns them.
void* next = frameNext(frame); void* next = frameNext(frame);
if (frameValid(next)) { if (frameValid(next)) {
visit(t, v, &frameMethod(next)); v->visit(&frameMethod(next));
if (methodFlags(t, frameMethod(next)) & ACC_NATIVE) { if (methodFlags(t, frameMethod(next)) & ACC_NATIVE) {
visitParameters(t, v, frame); visitParameters(t, v, frame);
@ -4573,10 +4573,10 @@ class JavaCompiler: public Compiler {
MyProtector(JavaCompiler* c): Protector(c->t), c(c) { } MyProtector(JavaCompiler* c): Protector(c->t), c(c) { }
virtual void visit(Heap::Visitor* v) { virtual void visit(Heap::Visitor* v) {
vm::visit(t, v, &(c->method)); v->visit(&(c->method));
for (unsigned i = 0; i < c->pool.length(); i += BytesPerWord) { for (unsigned i = 0; i < c->pool.length(); i += BytesPerWord) {
vm::visit(t, v, reinterpret_cast<object*>(&(c->pool.getAddress(i)))); v->visit(reinterpret_cast<object*>(&(c->pool.getAddress(i))));
} }
} }
@ -4799,7 +4799,7 @@ class ArgumentList {
virtual void visit(Heap::Visitor* v) { virtual void visit(Heap::Visitor* v) {
for (unsigned i = 0; i < list->position; ++i) { for (unsigned i = 0; i < list->position; ++i) {
if (list->objectMask[i]) { if (list->objectMask[i]) {
vm::visit(t, v, reinterpret_cast<object*>(list->array + i)); v->visit(reinterpret_cast<object*>(list->array + i));
} }
} }
} }
@ -4984,7 +4984,7 @@ class MyProcessor: public Processor {
if (t->m->active) { if (t->m->active) {
for (Reference* r = t->reference; r; r = r->next) { for (Reference* r = t->reference; r; r = r->next) {
visit(t, v, &(r->target)); v->visit(&(r->target));
} }
visitStack(t, v); visitStack(t, v);

View File

@ -6,6 +6,12 @@ using namespace vm;
namespace { namespace {
// an object must survive TenureThreshold + 2 garbage collections
// before being copied to gen2 (muat be at least 1):
const unsigned TenureThreshold = 3;
const unsigned FixieTenureThreshold = TenureThreshold + 2;
const unsigned Top = ~static_cast<unsigned>(0); const unsigned Top = ~static_cast<unsigned>(0);
const unsigned InitialGen2CapacityInBytes = 4 * 1024 * 1024; const unsigned InitialGen2CapacityInBytes = 4 * 1024 * 1024;
@ -381,11 +387,77 @@ class Segment {
} }
}; };
class Fixie {
public:
Fixie(unsigned size, bool hasMask, Fixie** handle):
age(0),
hasMask(hasMask),
marked(false),
dirty(false)
{
memset(mask(size), 0, maskSize(size, hasMask));
add(handle);
if (Debug) {
fprintf(stderr, "make %p\n", this);
}
}
void add(Fixie** handle) {
this->handle = handle;
next = *handle;
if (next) next->handle = &next;
*handle = this;
}
void remove() {
*handle = next;
if (next) next->handle = handle;
}
void move(Fixie** handle) {
remove();
add(handle);
}
uintptr_t* mask(unsigned size) {
return body + size;
}
static unsigned maskSize(unsigned size, bool hasMask) {
return hasMask * ceiling(size, BytesPerWord) * BytesPerWord;
}
static unsigned totalSize(unsigned size, bool hasMask) {
return sizeof(Fixie) + (size * BytesPerWord) + maskSize(size, hasMask);
}
unsigned totalSize(unsigned size) {
return totalSize(size, hasMask);
}
uint8_t age;
bool hasMask;
bool marked;
bool dirty;
Fixie* next;
Fixie** handle;
uintptr_t body[0];
};
Fixie*
fixie(void* body)
{
return static_cast<Fixie*>(body) - 1;
}
void
free(Context* c, Fixie** fixies);
class Context { class Context {
public: public:
Context(System* system): Context(System* system, Heap::Client* client):
system(system), system(system),
client(0), client(client),
ageMap(&gen1, log(TenureThreshold), 1, 0, false), ageMap(&gen1, log(TenureThreshold), 1, 0, false),
gen1(this, &ageMap, 0, 0), gen1(this, &ageMap, 0, 0),
@ -406,10 +478,17 @@ class Context {
gen2Base(0), gen2Base(0),
tenureFootprint(0), tenureFootprint(0),
fixieTenureFootprint(0),
gen1padding(0), gen1padding(0),
gen2padding(0), gen2padding(0),
mode(Heap::MinorCollection), mode(Heap::MinorCollection),
fixies(0),
tenuredFixies(0),
dirtyFixies(0),
markedFixies(0),
visitedFixies(0),
lastCollectionTime(system->now()), lastCollectionTime(system->now()),
totalCollectionTime(0), totalCollectionTime(0),
totalTime(0) totalTime(0)
@ -420,6 +499,10 @@ class Context {
nextGen1.dispose(); nextGen1.dispose();
gen2.dispose(); gen2.dispose();
nextGen2.dispose(); nextGen2.dispose();
free(this, &tenuredFixies);
free(this, &dirtyFixies);
free(this, &fixies);
client->dispose();
} }
System* system; System* system;
@ -444,11 +527,18 @@ class Context {
unsigned gen2Base; unsigned gen2Base;
unsigned tenureFootprint; unsigned tenureFootprint;
unsigned fixieTenureFootprint;
unsigned gen1padding; unsigned gen1padding;
unsigned gen2padding; unsigned gen2padding;
Heap::CollectionType mode; Heap::CollectionType mode;
Fixie* fixies;
Fixie* tenuredFixies;
Fixie* dirtyFixies;
Fixie* markedFixies;
Fixie* visitedFixies;
int64_t lastCollectionTime; int64_t lastCollectionTime;
int64_t totalCollectionTime; int64_t totalCollectionTime;
int64_t totalTime; int64_t totalTime;
@ -568,6 +658,62 @@ bitset(Context* c UNUSED, void* o)
return &cast<uintptr_t>(o, BytesPerWord * 2); return &cast<uintptr_t>(o, BytesPerWord * 2);
} }
void
free(Context* c, Fixie** fixies)
{
for (Fixie** p = fixies; *p;) {
Fixie* f = *p;
*p = f->next;
if (Debug) {
fprintf(stderr, "free %p\n", f);
}
c->system->free(f);
}
}
void
sweepFixies(Context* c)
{
assert(c, c->markedFixies == 0);
if (c->mode == Heap::MajorCollection) {
free(c, &(c->tenuredFixies));
free(c, &(c->dirtyFixies));
}
free(c, &(c->fixies));
for (Fixie** p = &(c->visitedFixies); *p;) {
Fixie* f = *p;
*p = f->next;
unsigned size = c->client->sizeInWords(f->body);
++ f->age;
if (f->age > FixieTenureThreshold) {
f->age = FixieTenureThreshold;
} else if (static_cast<unsigned>(f->age + 1) == FixieTenureThreshold) {
c->fixieTenureFootprint += f->totalSize(size);
}
if (f->age == FixieTenureThreshold) {
if (Debug) {
fprintf(stderr, "tenure %p\n", f);
}
f->move(&(c->tenuredFixies));
if (f->dirty) {
uintptr_t* mask = f->mask(size);
memset(mask, 0, ceiling(size, BytesPerWord) * BytesPerWord);
f->dirty = false;
}
} else {
f->move(&(c->fixies));
}
f->marked = false;
}
}
inline void* inline void*
copyTo(Context* c, Segment* s, void* o, unsigned size) copyTo(Context* c, Segment* s, void* o, unsigned size)
{ {
@ -644,7 +790,18 @@ update3(Context* c, void* o, bool* needsVisit)
if (wasCollected(c, o)) { if (wasCollected(c, o)) {
*needsVisit = false; *needsVisit = false;
return follow(c, o); return follow(c, o);
} else if (c->client->checkFixed(o)) { } else if (c->client->isFixed(o)) {
Fixie* f = fixie(o);
if ((not f->marked)
and (c->mode == Heap::MajorCollection
or f->age < FixieTenureThreshold))
{
if (Debug) {
fprintf(stderr, "mark %p\n", f);
}
f->marked = true;
f->move(&(c->markedFixies));
}
*needsVisit = false; *needsVisit = false;
return o; return o;
} else { } else {
@ -992,6 +1149,64 @@ collect(Context* c, void** p)
} }
} }
void
visitDirtyFixies(Context* c)
{
for (Fixie** p = &(c->dirtyFixies); *p;) {
Fixie* f = *p;
*p = f->next;
unsigned size = c->client->sizeInWords(f->body);
uintptr_t* mask = f->mask(size);
for (unsigned word = 0; word < wordOf(size); ++ word) {
if (mask[word]) {
for (unsigned bit = 0; bit < bitOf(size); ++ bit) {
unsigned index = indexOf(word, bit);
if (getBit(mask, index)) {
collect(c, reinterpret_cast<void**>(f->body) + index);
}
}
}
}
memset(mask, 0, ceiling(size, BytesPerWord) * BytesPerWord);
f->dirty = false;
f->move(&(c->tenuredFixies));
}
}
void
visitMarkedFixies(Context* c)
{
for (Fixie** p = &(c->markedFixies); *p;) {
Fixie* f = *p;
*p = f->next;
if (Debug) {
fprintf(stderr, "visit %p\n", f);
}
class Walker: public Heap::Walker {
public:
Walker(Context* c, void** p):
c(c), p(p)
{ }
virtual bool visit(unsigned offset) {
collect(c, p + offset);
return true;
}
Context* c;
void** p;
} w(c, reinterpret_cast<void**>(f->body));
c->client->walk(reinterpret_cast<void**>(f->body), &w);
f->move(&(c->visitedFixies));
}
}
void void
collect(Context* c, Segment::Map* map, unsigned start, unsigned end, collect(Context* c, Segment::Map* map, unsigned start, unsigned end,
bool* dirty, bool expectDirty UNUSED) bool* dirty, bool expectDirty UNUSED)
@ -1038,6 +1253,7 @@ collect2(Context* c)
{ {
c->gen2Base = Top; c->gen2Base = Top;
c->tenureFootprint = 0; c->tenureFootprint = 0;
c->fixieTenureFootprint = 0;
c->gen1padding = 0; c->gen1padding = 0;
c->gen2padding = 0; c->gen2padding = 0;
@ -1048,12 +1264,17 @@ collect2(Context* c)
collect(c, &(c->heapMap), start, end, &dirty, false); collect(c, &(c->heapMap), start, end, &dirty, false);
} }
if (c->mode == Heap::MinorCollection) {
visitDirtyFixies(c);
}
class Visitor : public Heap::Visitor { class Visitor : public Heap::Visitor {
public: public:
Visitor(Context* c): c(c) { } Visitor(Context* c): c(c) { }
virtual void visit(void* p) { virtual void visit(void* p) {
collect(c, static_cast<void**>(p)); collect(c, static_cast<void**>(p));
visitMarkedFixies(c);
} }
Context* c; Context* c;
@ -1065,7 +1286,9 @@ collect2(Context* c)
void void
collect(Context* c, unsigned footprint) collect(Context* c, unsigned footprint)
{ {
if (c->tenureFootprint > c->gen2.remaining()) { if (c->tenureFootprint > c->gen2.remaining()
or c->fixieTenureFootprint)
{
c->mode = Heap::MajorCollection; c->mode = Heap::MajorCollection;
} }
@ -1092,6 +1315,8 @@ collect(Context* c, unsigned footprint)
c->gen2.replaceWith(&(c->nextGen2)); c->gen2.replaceWith(&(c->nextGen2));
} }
sweepFixies(c);
if (Verbose) { if (Verbose) {
int64_t now = c->system->now(); int64_t now = c->system->now();
int64_t collection = now - then; int64_t collection = now - then;
@ -1114,32 +1339,61 @@ collect(Context* c, unsigned footprint)
class MyHeap: public Heap { class MyHeap: public Heap {
public: public:
MyHeap(System* system): c(system) { } MyHeap(System* system, Heap::Client* client): c(system, client) { }
virtual void collect(CollectionType type, Client* client, unsigned footprint) virtual void collect(CollectionType type, unsigned footprint) {
{
c.mode = type; c.mode = type;
c.client = client;
::collect(&c, footprint); ::collect(&c, footprint);
} }
virtual void* allocateFixed(unsigned sizeInWords, bool objectMask,
unsigned* totalInBytes)
{
*totalInBytes = Fixie::totalSize(sizeInWords, objectMask);
return (new (c.system->allocate(*totalInBytes))
Fixie(sizeInWords, objectMask, &(c.fixies)))->body;
}
virtual bool needsMark(void* p) { virtual bool needsMark(void* p) {
return *static_cast<void**>(p) if (c.client->isFixed(p)) {
and c.gen2.contains(p) return fixie(p)->age == FixieTenureThreshold;
and not c.gen2.contains(*static_cast<void**>(p)); } else {
return c.gen2.contains(p);
}
} }
virtual void mark(void* p) { bool targetNeedsMark(void* target) {
if (Debug) { return target
fprintf(stderr, "mark %p (%s) at %p (%s)\n", and not c.gen2.contains(target)
*static_cast<void**>(p), and not (c.client->isFixed(target)
segment(&c, *static_cast<void**>(p)), and fixie(target)->age == FixieTenureThreshold);
p,
segment(&c, p));
} }
c.heapMap.set(p); virtual void mark(void* p, unsigned offset, unsigned count) {
if (c.client->isFixed(p)) {
Fixie* f = fixie(p);
unsigned size = c.client->sizeInWords(p);
for (unsigned i = 0; i < count; ++i) {
void** target = static_cast<void**>(p) + offset + i;
if (targetNeedsMark(*target)) {
f->dirty = true;
markBit(f->mask(size), offset + i);
}
}
if (f->dirty) {
f->move(&(c.dirtyFixies));
}
} else {
for (unsigned i = 0; i < count; ++i) {
void** target = static_cast<void**>(p) + offset + i;
if (targetNeedsMark(*target)) {
c.heapMap.set(target);
}
}
}
} }
virtual void pad(void* p, unsigned extra) { virtual void pad(void* p, unsigned extra) {
@ -1207,9 +1461,9 @@ class MyHeap: public Heap {
namespace vm { namespace vm {
Heap* Heap*
makeHeap(System* system) makeHeap(System* system, Heap::Client* client)
{ {
return new (system->allocate(sizeof(MyHeap))) MyHeap(system); return new (system->allocate(sizeof(MyHeap))) MyHeap(system, client);
} }
} // namespace vm } // namespace vm

View File

@ -35,17 +35,20 @@ class Heap {
public: public:
virtual ~Client() { } virtual ~Client() { }
virtual void visitRoots(Visitor*) = 0; virtual void visitRoots(Visitor*) = 0;
virtual bool checkFixed(void*) = 0; virtual bool isFixed(void*) = 0;
virtual unsigned sizeInWords(void*) = 0;
virtual unsigned copiedSizeInWords(void*) = 0; virtual unsigned copiedSizeInWords(void*) = 0;
virtual void copy(void*, void*) = 0; virtual void copy(void*, void*) = 0;
virtual void walk(void*, Walker*) = 0; virtual void walk(void*, Walker*) = 0;
virtual void dispose() = 0;
}; };
virtual ~Heap() { } virtual ~Heap() { }
virtual void collect(CollectionType type, Client* client, unsigned footprint) virtual void collect(CollectionType type, unsigned footprint) = 0;
= 0; virtual void* allocateFixed(unsigned sizeInWords, bool objectMask,
unsigned* totalInBytes) = 0;
virtual bool needsMark(void* p) = 0; virtual bool needsMark(void* p) = 0;
virtual void mark(void* p) = 0; virtual void mark(void* p, unsigned offset, unsigned count) = 0;
virtual void pad(void* p, unsigned extra) = 0; virtual void pad(void* p, unsigned extra) = 0;
virtual void* follow(void* p) = 0; virtual void* follow(void* p) = 0;
virtual Status status(void* p) = 0; virtual Status status(void* p) = 0;
@ -53,7 +56,7 @@ class Heap {
virtual void dispose() = 0; virtual void dispose() = 0;
}; };
Heap* makeHeap(System* system); Heap* makeHeap(System* system, Heap::Client* client);
} // namespace vm } // namespace vm

View File

@ -2955,11 +2955,11 @@ class MyProcessor: public Processor {
{ {
Thread* t = static_cast<Thread*>(vmt); Thread* t = static_cast<Thread*>(vmt);
visit(t, v, &(t->code)); v->visit(&(t->code));
for (unsigned i = 0; i < t->sp; ++i) { for (unsigned i = 0; i < t->sp; ++i) {
if (t->stack[i * 2] == ObjectTag) { if (t->stack[i * 2] == ObjectTag) {
visit(t, v, reinterpret_cast<object*>(t->stack + (i * 2) + 1)); v->visit(reinterpret_cast<object*>(t->stack + (i * 2) + 1));
} }
} }
} }

View File

@ -22,7 +22,6 @@ DestroyJavaVM(Machine* m)
{ {
System* s = m->system; System* s = m->system;
Processor* p = m->processor; Processor* p = m->processor;
Heap* h = m->heap;
Finder* f = m->finder; Finder* f = m->finder;
Thread* t = m->rootThread; Thread* t = m->rootThread;
@ -32,7 +31,6 @@ DestroyJavaVM(Machine* m)
m->dispose(); m->dispose();
p->dispose(); p->dispose();
h->dispose();
f->dispose(); f->dispose();
s->dispose(); s->dispose();
@ -1884,10 +1882,9 @@ JNI_CreateJavaVM(Machine** m, Thread** t, void* args)
BUILTIN_CLASSPATH, s->pathSeparator(), a->classpath); BUILTIN_CLASSPATH, s->pathSeparator(), a->classpath);
Finder* f = makeFinder(s, classpath); Finder* f = makeFinder(s, classpath);
Heap* h = makeHeap(s);
Processor* p = makeProcessor(s); Processor* p = makeProcessor(s);
*m = new (s->allocate(sizeof(Machine))) Machine(s, h, f, p); *m = new (s->allocate(sizeof(Machine))) Machine(s, f, p);
if (a->properties) { if (a->properties) {
for (const char** p = a->properties; *p; ++p) { for (const char** p = a->properties; *p; ++p) {

View File

@ -126,8 +126,8 @@ void
visitRoots(Thread* t, Heap::Visitor* v) visitRoots(Thread* t, Heap::Visitor* v)
{ {
if (t->state != Thread::ZombieState) { if (t->state != Thread::ZombieState) {
visit(t, v, &(t->javaThread)); v->visit(&(t->javaThread));
visit(t, v, &(t->exception)); v->visit(&(t->exception));
t->m->processor->visitObjects(t, v); t->m->processor->visitObjects(t, v);
@ -210,7 +210,7 @@ walk(Thread* t, object o, Heap::Walker* w)
void void
finalizerTargetUnreachable(Thread* t, Heap::Visitor* v, object* p) finalizerTargetUnreachable(Thread* t, Heap::Visitor* v, object* p)
{ {
visit(t, v, &finalizerTarget(t, *p)); v->visit(&finalizerTarget(t, *p));
object finalizer = *p; object finalizer = *p;
*p = finalizerNext(t, finalizer); *p = finalizerNext(t, finalizer);
@ -226,7 +226,7 @@ referenceTargetUnreachable(Thread* t, Heap::Visitor* v, object* p)
jreferenceTarget(t, *p), *p); jreferenceTarget(t, *p), *p);
} }
visit(t, v, p); v->visit(p);
jreferenceTarget(t, *p) = 0; jreferenceTarget(t, *p) = 0;
if (jreferenceQueue(t, *p) if (jreferenceQueue(t, *p)
@ -234,7 +234,7 @@ referenceTargetUnreachable(Thread* t, Heap::Visitor* v, object* p)
{ {
// queue is reachable - add the reference // queue is reachable - add the reference
visit(t, v, &jreferenceQueue(t, *p)); v->visit(&jreferenceQueue(t, *p));
object q = jreferenceQueue(t, *p); object q = jreferenceQueue(t, *p);
@ -278,89 +278,13 @@ referenceTargetReachable(Thread* t, Heap::Visitor* v, object* p)
jreferenceTarget(t, *p), *p); jreferenceTarget(t, *p), *p);
} }
visit(t, v, p); v->visit(p);
visit(t, v, &jreferenceTarget(t, *p)); v->visit(&jreferenceTarget(t, *p));
if (t->m->heap->status(jreferenceQueue(t, *p)) == Heap::Unreachable) { if (t->m->heap->status(jreferenceQueue(t, *p)) == Heap::Unreachable) {
jreferenceQueue(t, *p) = 0; jreferenceQueue(t, *p) = 0;
} else { } else {
visit(t, v, &jreferenceQueue(t, *p)); v->visit(&jreferenceQueue(t, *p));
}
}
void
freeFixies(Thread* t, object* fixies)
{
for (object* p = fixies; *p;) {
object o = *p;
*p = fixedNext(t, o);
t->m->system->free(fixedStart(t, o));
}
}
void
sweepFixies(Thread* t)
{
Machine* m = t->m;
assert(t, m->markedFixies == 0);
if (m->heap->collectionType() == Heap::MajorCollection) {
freeFixies(t, &(m->tenuredFixies));
freeFixies(t, &(m->dirtyFixies));
}
freeFixies(t, &(m->fixies));
for (object* p = &(m->visitedFixies); *p;) {
object o = *p;
*p = fixedNext(t, o);
fixedAge(t, o) = (fixedAge(t, o) + 1);
if (fixedAge(t, o) > TenureThreshold) {
fixedAge(t, o) = TenureThreshold;
}
if (fixedAge(t, o) == TenureThreshold) {
fixedMove(t, o, &(m->tenuredFixies));
if (fixedDirty(t, o)) {
unsigned size = baseSize(t, o, objectClass(t, o));
uintptr_t* mask = fixedMask(t, o, size);
memset(mask, 0, ceiling(size, BytesPerWord) * BytesPerWord);
fixedDirty(t, o) = 0;
}
} else {
fixedMove(t, o, &(m->fixies));
}
fixedMarked(t, o) = 0;
}
}
void
visitDirtyFixies(Thread* t, Heap::Visitor* v)
{
for (object* p = &(t->m->dirtyFixies); *p;) {
object o = *p;
*p = fixedNext(t, o);
unsigned size = baseSize(t, o, objectClass(t, o));
uintptr_t* mask = fixedMask(t, o, size);
for (unsigned word = 0; word < wordOf(size); ++ word) {
if (mask[word]) {
for (unsigned bit = 0; bit < bitOf(size); ++ bit) {
unsigned index = indexOf(word, bit);
if (getBit(mask, index)) {
visit(t, v, &cast<object>(o, index));
}
}
}
}
memset(mask, 0, ceiling(size, BytesPerWord) * BytesPerWord);
fixedDirty(t, o) = 0;
fixedMove(t, o, &(t->m->tenuredFixies));
} }
} }
@ -371,27 +295,27 @@ postVisit(Thread* t, Heap::Visitor* v)
bool major = m->heap->collectionType() == Heap::MajorCollection; bool major = m->heap->collectionType() == Heap::MajorCollection;
for (object* p = &(m->finalizeQueue); *p; p = &(finalizerNext(t, *p))) { for (object* p = &(m->finalizeQueue); *p; p = &(finalizerNext(t, *p))) {
visit(t, v, p); v->visit(p);
visit(t, v, &finalizerTarget(t, *p)); v->visit(&finalizerTarget(t, *p));
} }
for (object* p = &(m->finalizeQueue); *p; p = &(finalizerNext(t, *p))) { for (object* p = &(m->finalizeQueue); *p; p = &(finalizerNext(t, *p))) {
visit(t, v, p); v->visit(p);
visit(t, v, &finalizerTarget(t, *p)); v->visit(&finalizerTarget(t, *p));
} }
object firstNewTenuredFinalizer = 0; object firstNewTenuredFinalizer = 0;
object lastNewTenuredFinalizer = 0; object lastNewTenuredFinalizer = 0;
for (object* p = &(m->finalizers); *p;) { for (object* p = &(m->finalizers); *p;) {
visit(t, v, p); v->visit(p);
if (m->heap->status(finalizerTarget(t, *p)) == Heap::Unreachable) { if (m->heap->status(finalizerTarget(t, *p)) == Heap::Unreachable) {
// target is unreachable - queue it up for finalization // target is unreachable - queue it up for finalization
finalizerTargetUnreachable(t, v, p); finalizerTargetUnreachable(t, v, p);
} else { } else {
// target is reachable // target is reachable
visit(t, v, &finalizerTarget(t, *p)); v->visit(&finalizerTarget(t, *p));
if (m->heap->status(*p) == Heap::Tenured) { if (m->heap->status(*p) == Heap::Tenured) {
// the finalizer is tenured, so we remove it from // the finalizer is tenured, so we remove it from
@ -448,14 +372,14 @@ postVisit(Thread* t, Heap::Visitor* v)
if (major) { if (major) {
for (object* p = &(m->tenuredFinalizers); *p;) { for (object* p = &(m->tenuredFinalizers); *p;) {
visit(t, v, p); v->visit(p);
if (m->heap->status(finalizerTarget(t, *p)) == Heap::Unreachable) { if (m->heap->status(finalizerTarget(t, *p)) == Heap::Unreachable) {
// target is unreachable - queue it up for finalization // target is unreachable - queue it up for finalization
finalizerTargetUnreachable(t, v, p); finalizerTargetUnreachable(t, v, p);
} else { } else {
// target is reachable // target is reachable
visit(t, v, &finalizerTarget(t, *p)); v->visit(&finalizerTarget(t, *p));
p = &finalizerNext(t, *p); p = &finalizerNext(t, *p);
} }
} }
@ -500,7 +424,6 @@ postCollect(Thread* t)
t->heap = t->defaultHeap; t->heap = t->defaultHeap;
t->heapOffset = 0; t->heapOffset = 0;
t->heapIndex = 0; t->heapIndex = 0;
t->allocatedLarge = false;
for (Thread* c = t->child; c; c = c->peer) { for (Thread* c = t->child; c; c = c->peer) {
postCollect(c); postCollect(c);
@ -1505,15 +1428,109 @@ invoke(Thread* t, const char* className, int argc, const char** argv)
(t, className, "main", "([Ljava/lang/String;)V", 0, args); (t, className, "main", "([Ljava/lang/String;)V", 0, args);
} }
class HeapClient: public Heap::Client {
public:
HeapClient(Machine* m): m(m) { }
virtual void visitRoots(Heap::Visitor* v) {
v->visit(&(m->loader));
v->visit(&(m->bootstrapClassMap));
v->visit(&(m->monitorMap));
v->visit(&(m->stringMap));
v->visit(&(m->types));
v->visit(&(m->jniInterfaceTable));
for (Reference* r = m->jniReferences; r; r = r->next) {
v->visit(&(r->target));
}
for (Thread* t = m->rootThread; t; t = t->peer) {
::visitRoots(t, v);
}
postVisit(m->rootThread, v);
}
virtual bool isFixed(void* p) {
return objectFixed(m->rootThread, static_cast<object>(p));
}
virtual unsigned sizeInWords(void* p) {
Thread* t = m->rootThread;
object o = static_cast<object>(m->heap->follow(mask(p)));
unsigned n = baseSize(t, o, static_cast<object>
(m->heap->follow(objectClass(t, o))));
if (objectExtended(t, o)) {
++ n;
}
return n;
}
virtual unsigned copiedSizeInWords(void* p) {
Thread* t = m->rootThread;
object o = static_cast<object>(m->heap->follow(mask(p)));
assert(t, not objectFixed(t, o));
unsigned n = baseSize(t, o, static_cast<object>
(m->heap->follow(objectClass(t, o))));
if (objectExtended(t, o) or hashTaken(t, o)) {
++ n;
}
return n;
}
virtual void copy(void* srcp, void* dstp) {
Thread* t = m->rootThread;
object src = static_cast<object>(m->heap->follow(mask(srcp)));
assert(t, not objectFixed(t, src));
object class_ = static_cast<object>
(m->heap->follow(objectClass(t, src)));
unsigned base = baseSize(t, src, class_);
unsigned n = extendedSize(t, src, base);
object dst = static_cast<object>(dstp);
memcpy(dst, src, n * BytesPerWord);
if (hashTaken(t, src)) {
cast<uintptr_t>(dst, 0) &= PointerMask;
cast<uintptr_t>(dst, 0) |= ExtendedMark;
extendedWord(t, dst, base) = takeHash(t, src);
}
}
virtual void walk(void* p, Heap::Walker* w) {
object o = static_cast<object>(m->heap->follow(mask(p)));
::walk(m->rootThread, o, w);
}
virtual void dispose() {
m->system->free(this);
}
private:
Machine* m;
};
} // namespace } // namespace
namespace vm { namespace vm {
Machine::Machine(System* system, Heap* heap, Finder* finder, Machine::Machine(System* system, Finder* finder, Processor* processor):
Processor* processor):
vtable(&javaVMVTable), vtable(&javaVMVTable),
system(system), system(system),
heap(heap), heap(makeHeap(system, new (system->allocate(sizeof(HeapClient)))
HeapClient(this))),
finder(finder), finder(finder),
processor(processor), processor(processor),
rootThread(0), rootThread(0),
@ -1522,6 +1539,7 @@ Machine::Machine(System* system, Heap* heap, Finder* finder,
builtins(0), builtins(0),
activeCount(0), activeCount(0),
liveCount(0), liveCount(0),
fixedFootprint(0),
localThread(0), localThread(0),
stateLock(0), stateLock(0),
heapLock(0), heapLock(0),
@ -1539,11 +1557,6 @@ Machine::Machine(System* system, Heap* heap, Finder* finder,
finalizeQueue(0), finalizeQueue(0),
weakReferences(0), weakReferences(0),
tenuredWeakReferences(0), tenuredWeakReferences(0),
fixies(0),
tenuredFixies(0),
dirtyFixies(0),
markedFixies(0),
visitedFixies(0),
unsafe(false), unsafe(false),
heapPoolIndex(0) heapPoolIndex(0)
{ {
@ -1583,6 +1596,8 @@ Machine::dispose()
system->free(heapPool[i]); system->free(heapPool[i]);
} }
heap->dispose();
system->free(this); system->free(this);
} }
@ -1593,7 +1608,6 @@ Thread::Thread(Machine* m, object javaThread, Thread* parent):
peer((parent ? parent->child : 0)), peer((parent ? parent->child : 0)),
child(0), child(0),
state(NoState), state(NoState),
allocatedLarge(false),
criticalLevel(0), criticalLevel(0),
systemThread(0), systemThread(0),
javaThread(javaThread), javaThread(javaThread),
@ -1769,10 +1783,6 @@ exit(Thread* t)
function(t, finalizerTarget(t, f)); function(t, finalizerTarget(t, f));
} }
freeFixies(t, &(t->m->tenuredFixies));
freeFixies(t, &(t->m->dirtyFixies));
freeFixies(t, &(t->m->fixies));
disposeAll(t, t->m->rootThread); disposeAll(t, t->m->rootThread);
} }
@ -1887,44 +1897,7 @@ enter(Thread* t, Thread::State s)
} }
object object
allocateFixed(Thread* t, unsigned sizeInBytes, bool objectMask) allocate2(Thread* t, unsigned sizeInBytes, bool objectMask, bool fixed)
{
ENTER(t, Thread::ExclusiveState);
unsigned mask = objectMask
* ceiling(sizeInBytes / BytesPerWord, BytesPerWord)
* BytesPerWord;
unsigned total = sizeInBytes + FixedFootprint + mask;
uint8_t* p = static_cast<uint8_t*>(t->m->system->tryAllocate(total));
if (p == 0) {
collect(t, Heap::MajorCollection);
p = static_cast<uint8_t*>(t->m->system->allocate(total));
}
memset(p + FixedFootprint + sizeInBytes, 0, mask);
object o = reinterpret_cast<object>(p + FixedFootprint);
cast<uintptr_t>(o, 0) = FixedMark;
fixedAge(t, o) = 0;
fixedMarked(t, o) = 0;
fixedDirty(t, o) = 0;
fixedAdd(t, o, &(t->m->fixies));
return o;
}
object
allocateLarge(Thread* t, unsigned sizeInBytes, bool objectMask)
{
t->allocatedLarge = true;
return allocateFixed(t, sizeInBytes, objectMask);
}
object
allocate2(Thread* t, unsigned sizeInBytes, bool objectMask)
{ {
ACQUIRE_RAW(t, t->m->stateLock); ACQUIRE_RAW(t, t->m->stateLock);
@ -1934,14 +1907,17 @@ allocate2(Thread* t, unsigned sizeInBytes, bool objectMask)
ENTER(t, Thread::IdleState); ENTER(t, Thread::IdleState);
} }
if (sizeInBytes <= Thread::HeapSizeInBytes if (fixed) {
and t->heapIndex + ceiling(sizeInBytes, BytesPerWord) if (t->m->fixedFootprint + sizeInBytes
> Machine::FixedFootprintThresholdInBytes)
{
t->heap = 0;
}
} else if (t->heapIndex + ceiling(sizeInBytes, BytesPerWord)
>= Thread::HeapSizeInWords) >= Thread::HeapSizeInWords)
{ {
t->heap = 0; t->heap = 0;
if ((not t->allocatedLarge) if (t->m->heapPoolIndex < Machine::HeapPoolSize) {
and t->m->heapPoolIndex < Machine::HeapPoolSize)
{
t->heap = static_cast<uintptr_t*> t->heap = static_cast<uintptr_t*>
(t->m->system->tryAllocate(Thread::HeapSizeInBytes)); (t->m->system->tryAllocate(Thread::HeapSizeInBytes));
if (t->heap) { if (t->heap) {
@ -1950,15 +1926,24 @@ allocate2(Thread* t, unsigned sizeInBytes, bool objectMask)
t->heapIndex = 0; t->heapIndex = 0;
} }
} }
}
if (t->heap == 0) { if (t->heap == 0) {
ENTER(t, Thread::ExclusiveState); ENTER(t, Thread::ExclusiveState);
collect(t, Heap::MinorCollection); collect(t, Heap::MinorCollection);
} }
}
if (sizeInBytes > Thread::HeapSizeInBytes) { if (fixed) {
return allocateLarge(t, sizeInBytes, objectMask); unsigned total;
object o = static_cast<object>
(t->m->heap->allocateFixed
(ceiling(sizeInBytes, BytesPerWord), objectMask, &total));
cast<uintptr_t>(o, 0) = FixedMark;
t->m->fixedFootprint += total;
return o;
} else { } else {
return allocateSmall(t, sizeInBytes); return allocateSmall(t, sizeInBytes);
} }
@ -2705,107 +2690,8 @@ collect(Thread* t, Heap::CollectionType type)
{ {
Machine* m = t->m; Machine* m = t->m;
class Client: public Heap::Client {
public:
Client(Machine* m): m(m) { }
virtual void visitRoots(Heap::Visitor* v) {
Thread* t = m->rootThread;
visit(t, v, &(m->loader));
visit(t, v, &(m->bootstrapClassMap));
visit(t, v, &(m->monitorMap));
visit(t, v, &(m->stringMap));
visit(t, v, &(m->types));
visit(t, v, &(m->jniInterfaceTable));
for (Reference* r = m->jniReferences; r; r = r->next) {
visit(t, v, &(r->target));
}
for (Thread* t = m->rootThread; t; t = t->peer) {
::visitRoots(t, v);
}
if (m->heap->collectionType() == Heap::MinorCollection) {
visitDirtyFixies(t, v);
}
postVisit(m->rootThread, v);
}
virtual bool checkFixed(void* p) {
Thread* t = m->rootThread;
object o = static_cast<object>(p);
if (objectFixed(t, o)) {
if ((not fixedMarked(t, o))
and m->heap->collectionType() == Heap::MajorCollection
or fixedAge(t, o) < TenureThreshold)
{
fixedMarked(t, o) = 1;
fixedMove(t, o, &(m->markedFixies));
}
return true;
} else {
return false;
}
}
virtual unsigned copiedSizeInWords(void* p) {
Thread* t = m->rootThread;
object o = static_cast<object>(m->heap->follow(mask(p)));
assert(t, not objectFixed(t, o));
unsigned n = baseSize(t, o, static_cast<object>
(m->heap->follow(objectClass(t, o))));
if (objectExtended(t, o) or hashTaken(t, o)) {
++ n;
}
return n;
}
virtual void copy(void* srcp, void* dstp) {
Thread* t = m->rootThread;
object src = static_cast<object>(m->heap->follow(mask(srcp)));
assert(t, not objectFixed(t, src));
object class_ = static_cast<object>
(m->heap->follow(objectClass(t, src)));
unsigned base = baseSize(t, src, class_);
unsigned n = extendedSize(t, src, base);
object dst = static_cast<object>(dstp);
memcpy(dst, src, n * BytesPerWord);
if (hashTaken(t, src)) {
cast<uintptr_t>(dst, 0) &= PointerMask;
cast<uintptr_t>(dst, 0) |= ExtendedMark;
extendedWord(t, dst, base) = takeHash(t, src);
}
}
virtual void walk(void* p, Heap::Walker* w) {
Thread* t = m->rootThread;
object o = static_cast<object>(m->heap->follow(mask(p)));
assert(t, not objectFixed(t, o));
::walk(m->rootThread, o, w);
}
private:
Machine* m;
} it(m);
m->unsafe = true; m->unsafe = true;
m->heap->collect(type, &it, footprint(m->rootThread)); m->heap->collect(type, footprint(m->rootThread));
m->unsafe = false; m->unsafe = false;
postCollect(m->rootThread); postCollect(m->rootThread);
@ -2824,7 +2710,7 @@ collect(Thread* t, Heap::CollectionType type)
} }
m->heapPoolIndex = 0; m->heapPoolIndex = 0;
sweepFixies(t); m->fixedFootprint = 0;
} }
void void
@ -2907,84 +2793,6 @@ makeTrace(Thread* t, uintptr_t start)
return trace; return trace;
} }
void
mark(Thread* t, object o, unsigned offset)
{
if (objectFixed(t, o)) {
if (fixedAge(t, o) == TenureThreshold) {
ACQUIRE_RAW(t, t->m->heapLock);
markBit(fixedMask(t, o), offset / BytesPerWord);
fixedDirty(t, o) = 1;
fixedMove(t, o, &(t->m->dirtyFixies));
}
} else if (t->m->heap->needsMark(&cast<void*>(o, offset))) {
ACQUIRE_RAW(t, t->m->heapLock);
t->m->heap->mark(&cast<void*>(o, offset));
}
}
void
mark(Thread* t, object o, unsigned offset, unsigned count)
{
if (objectFixed(t, o)) {
if (fixedAge(t, o) == TenureThreshold) {
ACQUIRE_RAW(t, t->m->heapLock);
uintptr_t* mask = fixedMask(t, o);
for (unsigned i = 0; i < count; ++i) {
markBit(mask, (offset / BytesPerWord) + i);
}
fixedDirty(t, o) = 1;
fixedMove(t, o, &(t->m->dirtyFixies));
}
} else {
ACQUIRE_RAW(t, t->m->heapLock);
for (unsigned i = 0; i < count; ++i) {
unsigned j = offset + (i * BytesPerWord);
if (t->m->heap->needsMark(&cast<void*>(o, j))) {
t->m->heap->mark(&cast<void*>(o, j));
}
}
}
}
void
visit(Thread* t, Heap::Visitor* v, object* p)
{
v->visit(p);
for (object* p = &(t->m->markedFixies); *p;) {
object o = *p;
*p = fixedNext(t, o);
class Walker: public Heap::Walker {
public:
Walker(Thread* t, Heap::Visitor* v, object o):
t(t), v(v), o(o)
{ }
virtual bool visit(unsigned offset) {
::visit(t, v, &cast<object>(o, offset * BytesPerWord));
return true;
}
Thread* t;
Heap::Visitor* v;
object o;
} w(t, v, o);
walk(t, o, &w);
fixedMove(t, o, &(t->m->visitedFixies));
}
}
void void
noop() noop()
{ } { }

View File

@ -34,13 +34,6 @@ const uintptr_t HashTakenMark = 1;
const uintptr_t ExtendedMark = 2; const uintptr_t ExtendedMark = 2;
const uintptr_t FixedMark = 3; const uintptr_t FixedMark = 3;
const unsigned FixedAge = 0;
const unsigned FixedMarked = 1;
const unsigned FixedDirty = 2;
const unsigned FixedHandle = BytesPerWord;
const unsigned FixedNext = BytesPerWord * 2;
const unsigned FixedFootprint = BytesPerWord * 3;
enum FieldCode { enum FieldCode {
VoidField, VoidField,
ByteField, ByteField,
@ -1116,7 +1109,7 @@ class Machine {
ActiveState ActiveState
}; };
Machine(System* system, Heap* heap, Finder* finder, Processor* processor); Machine(System* system, Finder* finder, Processor* processor);
~Machine() { ~Machine() {
dispose(); dispose();
@ -1124,6 +1117,8 @@ class Machine {
static const unsigned HeapPoolSize = 8; static const unsigned HeapPoolSize = 8;
static const unsigned FixedFootprintThresholdInBytes = 256 * 1024;
void dispose(); void dispose();
JavaVMVTable* vtable; JavaVMVTable* vtable;
@ -1137,6 +1132,7 @@ class Machine {
const char* builtins; const char* builtins;
unsigned activeCount; unsigned activeCount;
unsigned liveCount; unsigned liveCount;
unsigned fixedFootprint;
System::Local* localThread; System::Local* localThread;
System::Monitor* stateLock; System::Monitor* stateLock;
System::Monitor* heapLock; System::Monitor* heapLock;
@ -1154,11 +1150,6 @@ class Machine {
object finalizeQueue; object finalizeQueue;
object weakReferences; object weakReferences;
object tenuredWeakReferences; object tenuredWeakReferences;
object fixies;
object tenuredFixies;
object dirtyFixies;
object markedFixies;
object visitedFixies;
bool unsafe; bool unsafe;
JavaVMVTable javaVMVTable; JavaVMVTable javaVMVTable;
JNIEnvVTable jniEnvVTable; JNIEnvVTable jniEnvVTable;
@ -1175,9 +1166,6 @@ threadInterrupted(Thread* t, object thread);
void void
enterActiveState(Thread* t); enterActiveState(Thread* t);
void
visit(Thread* t, Heap::Visitor* v, object* p);
class Thread { class Thread {
public: public:
enum State { enum State {
@ -1211,7 +1199,7 @@ class Thread {
SingleProtector(Thread* t, object* p): Protector(t), p(p) { } SingleProtector(Thread* t, object* p): Protector(t), p(p) { }
virtual void visit(Heap::Visitor* v) { virtual void visit(Heap::Visitor* v) {
vm::visit(t, v, p); v->visit(p);
} }
object* p; object* p;
@ -1266,7 +1254,6 @@ class Thread {
Thread* peer; Thread* peer;
Thread* child; Thread* child;
State state; State state;
bool allocatedLarge;
unsigned criticalLevel; unsigned criticalLevel;
System::Thread* systemThread; System::Thread* systemThread;
object javaThread; object javaThread;
@ -1422,10 +1409,7 @@ expect(Thread* t, bool v)
} }
object object
allocateFixed(Thread* t, unsigned sizeInBytes, bool objectMask); allocate2(Thread* t, unsigned sizeInBytes, bool objectMask, bool fixed);
object
allocate2(Thread* t, unsigned sizeInBytes, bool objectMask);
inline object inline object
allocateSmall(Thread* t, unsigned sizeInBytes) allocateSmall(Thread* t, unsigned sizeInBytes)
@ -1445,17 +1429,27 @@ allocate(Thread* t, unsigned sizeInBytes, bool objectMask)
>= Thread::HeapSizeInWords >= Thread::HeapSizeInWords
or t->m->exclusive)) or t->m->exclusive))
{ {
return allocate2(t, sizeInBytes, objectMask); return allocate2(t, sizeInBytes, objectMask,
sizeInBytes > Thread::HeapSizeInBytes);
} else { } else {
return allocateSmall(t, sizeInBytes); return allocateSmall(t, sizeInBytes);
} }
} }
void inline void
mark(Thread* t, object target, unsigned offset); mark(Thread* t, object o, unsigned offset, unsigned count)
{
if (t->m->heap->needsMark(o)) {
ACQUIRE_RAW(t, t->m->heapLock);
t->m->heap->mark(o, offset / BytesPerWord, count);
}
}
void inline void
mark(Thread* t, object target, unsigned offset, unsigned count); mark(Thread* t, object o, unsigned offset)
{
mark(t, o, offset, 1);
}
inline void inline void
set(Thread* t, object target, unsigned offset, object value) set(Thread* t, object target, unsigned offset, object value)
@ -1505,95 +1499,6 @@ baseSize(Thread* t, object o, object class_)
BytesPerWord); BytesPerWord);
} }
inline void*
fixedStart(Thread* t UNUSED, object o)
{
assert(t, objectFixed(t, o));
return &cast<object>(o, - FixedFootprint);
}
inline uint8_t&
fixedAge(Thread* t UNUSED, object o)
{
assert(t, objectFixed(t, o));
return cast<uint8_t>(o, - (FixedFootprint - FixedAge));
}
inline uint8_t&
fixedMarked(Thread* t UNUSED, object o)
{
assert(t, objectFixed(t, o));
return cast<uint8_t>(o, - (FixedFootprint - FixedMarked));
}
inline uint8_t&
fixedDirty(Thread* t UNUSED, object o)
{
assert(t, objectFixed(t, o));
return cast<uint8_t>(o, - (FixedFootprint - FixedDirty));
}
inline object*&
fixedHandle(Thread* t UNUSED, object o)
{
assert(t, objectFixed(t, o));
return cast<object*>(o, - (FixedFootprint - FixedHandle));
}
inline object&
fixedNext(Thread* t UNUSED, object o)
{
assert(t, objectFixed(t, o));
return cast<object>(o, - (FixedFootprint - FixedNext));
}
inline void
fixedAdd(Thread* t, object o, object* handle)
{
// fprintf(stderr, "add %p to %s\n", o,
// handle == &(t->m->fixies) ? "fixies" :
// handle == &(t->m->tenuredFixies) ? "tenured" :
// handle == &(t->m->dirtyFixies) ? "dirty" :
// handle == &(t->m->markedFixies) ? "marked" :
// handle == &(t->m->visitedFixies) ? "visited" : "unknown");
fixedHandle(t, o) = handle;
fixedNext(t, o) = *handle;
if (*handle) {
fixedHandle(t, *handle) = &fixedNext(t, o);
}
*handle = o;
}
inline void
fixedRemove(Thread* t, object o)
{
*fixedHandle(t, o) = fixedNext(t, o);
if (fixedNext(t, o)) {
fixedHandle(t, fixedNext(t, o)) = fixedHandle(t, o);
}
}
inline void
fixedMove(Thread* t, object o, object* handle)
{
fixedRemove(t, o);
fixedAdd(t, o, handle);
}
inline uintptr_t*
fixedMask(Thread* t UNUSED, object o, unsigned size)
{
assert(t, objectFixed(t, o));
return &cast<uintptr_t>(o, size * BytesPerWord);
}
inline uintptr_t*
fixedMask(Thread* t, object o)
{
return fixedMask(t, o, baseSize(t, o, objectClass(t, o)));
}
object object
makeTrace(Thread* t, uintptr_t start); makeTrace(Thread* t, uintptr_t start);