Merge branch 'master' into compiler

Conflicts:

	src/compile.cpp
	src/compiler.cpp
	src/compiler.h
This commit is contained in:
Joel Dice 2008-04-27 16:40:53 -06:00
commit 9f8d77a2a9
13 changed files with 309 additions and 229 deletions

View File

@ -27,6 +27,34 @@ public class StackTraceElement {
this.line = line;
}
public int hashCode() {
return class_.hashCode() ^ method.hashCode() ^ line;
}
public boolean equals(Object o) {
if (o instanceof StackTraceElement) {
StackTraceElement e = (StackTraceElement) o;
return class_.equals(e.class_)
&& method.equals(e.method)
&& line == e.line;
} else {
return false;
}
}
public String toString() {
StringBuilder sb = new StringBuilder();
sb.append(class_).append(".").append(method);
if (line == NativeLine) {
sb.append(" (native)");
} else if (line >= 0) {
sb.append(" (line ").append(line).append(")");
}
return sb.toString();
}
public String getClassName() {
return class_;
}

View File

@ -101,21 +101,7 @@ public class Throwable {
StackTraceElement[] trace = resolveTrace();
for (int i = 0; i < trace.length; ++i) {
sb.append(" at ")
.append(trace[i].getClassName())
.append(".")
.append(trace[i].getMethodName());
if (trace[i].isNativeMethod()) {
sb.append(" (native)");
} else {
int line = trace[i].getLineNumber();
if (line >= 0) {
sb.append(" (line ").append(line).append(")");
}
}
sb.append(nl);
sb.append(" at ").append(trace[i].toString()).append(nl);
}
if (cause != null) {

View File

@ -53,6 +53,7 @@ enumerateThreads(Thread* t, Thread* x, object array, unsigned* index,
{
if (*index < limit) {
set(t, array, ArrayBody + (*index * BytesPerWord), x->javaThread);
++ (*index);
if (x->peer) enumerateThreads(t, x->peer, array, index, limit);
@ -711,7 +712,7 @@ Java_java_lang_Thread_interrupt(Thread* t, jclass, jlong peer)
}
extern "C" JNIEXPORT jobject JNICALL
Java_java_lang_Thread_getTrace(Thread* t, jclass, jlong peer)
Java_java_lang_Thread_getStackTrace(Thread* t, jclass, jlong peer)
{
ENTER(t, Thread::ActiveState);

View File

@ -29,11 +29,12 @@
#ifdef __i386__
# define LD "d"
# define LX "x"
# define LLD "lld"
#ifdef __APPLE__
# define ULD "lu"
# define LX "lx"
#else
# define LX "x"
# define ULD "u"
#endif
#elif defined __x86_64__

View File

@ -41,14 +41,12 @@ class MyThread: public Thread {
public:
CallTrace(MyThread* t):
t(t),
ip(t->ip),
base(t->base),
stack(t->stack),
nativeMethod(0),
next(t->trace)
{
t->trace = this;
t->ip = 0;
t->base = 0;
t->stack = 0;
}
@ -56,7 +54,6 @@ class MyThread: public Thread {
~CallTrace() {
t->stack = stack;
t->base = base;
t->ip = ip;
t->trace = next;
}
@ -153,6 +150,14 @@ methodForIp(MyThread* t, void* ip)
class MyStackWalker: public Processor::StackWalker {
public:
enum State {
Start,
Next,
Method,
NativeMethod,
Finish
};
class MyProtector: public Thread::Protector {
public:
MyProtector(MyStackWalker* walker):
@ -161,7 +166,6 @@ class MyStackWalker: public Processor::StackWalker {
virtual void visit(Heap::Visitor* v) {
v->visit(&(walker->method_));
v->visit(&(walker->nativeMethod));
}
MyStackWalker* walker;
@ -169,109 +173,156 @@ class MyStackWalker: public Processor::StackWalker {
MyStackWalker(MyThread* t):
t(t),
ip_(t->ip ? t->ip : (t->stack ? *static_cast<void**>(t->stack) : 0)),
state(Start),
ip_(t->ip),
base(t->base),
stack(t->stack),
trace(t->trace),
nativeMethod(trace->nativeMethod),
method_(ip_ ? methodForIp(t, ip_) : 0),
method_(0),
protector(this)
{ }
MyStackWalker(MyStackWalker* w):
t(w->t),
state(w->state),
ip_(w->ip_),
base(w->base),
stack(w->stack),
trace(w->trace),
nativeMethod(w->nativeMethod),
method_(w->method_),
protector(this)
{ }
virtual void walk(Processor::StackVisitor* v) {
if (stack == 0) {
return;
}
if (not v->visit(this)) {
return;
}
for (MyStackWalker it(this); it.next();) {
for (MyStackWalker it(this); it.valid();) {
MyStackWalker walker(it);
if (not v->visit(&walker)) {
break;
}
it.next();
}
}
bool next() {
if (nativeMethod) {
nativeMethod = 0;
} else {
stack = static_cast<void**>(base) + 1;
base = *static_cast<void**>(base);
bool valid() {
while (true) {
// fprintf(stderr, "state: %d\n", state);
switch (state) {
case Start:
if (ip_ == 0 and stack) {
ip_ = *static_cast<void**>(stack);
method_ = methodForIp(t, *static_cast<void**>(stack));
if (method_ == 0) {
if (trace and trace->stack) {
}
if (trace and trace->nativeMethod) {
method_ = trace->nativeMethod;
state = NativeMethod;
} else if (ip_) {
state = Next;
} else {
state = Finish;
}
break;
case Next:
if (stack) {
method_ = methodForIp(t, ip_);
if (method_) {
state = Method;
} else if (trace) {
base = trace->base;
stack = static_cast<void**>(trace->stack);
ip_ = *static_cast<void**>(stack);
method_ = methodForIp(t, *static_cast<void**>(stack));
nativeMethod = trace->nativeMethod;
trace = trace->next;
ip_ = (stack ? *static_cast<void**>(stack) : 0);
if (trace->nativeMethod) {
method_ = trace->nativeMethod;
state = NativeMethod;
} else {
return false;
trace = trace->next;
state = Next;
}
} else {
state = Finish;
}
} else {
state = Finish;
}
break;
case Method:
case NativeMethod:
return true;
case Finish:
return false;
default:
abort(t);
}
}
}
void next() {
switch (state) {
case Method:
stack = static_cast<void**>(base) + 1;
ip_ = (stack ? *static_cast<void**>(stack) : 0);
base = *static_cast<void**>(base);
state = Next;
break;
case NativeMethod:
trace = trace->next;
state = Next;
break;
default:
abort(t);
}
}
virtual object method() {
if (nativeMethod) {
return nativeMethod;
} else {
switch (state) {
case Method:
return method_;
case NativeMethod:
return trace->nativeMethod;
default:
abort(t);
}
}
virtual int ip() {
if (nativeMethod) {
return 0;
} else {
switch (state) {
case Method:
return reinterpret_cast<intptr_t>(ip_) - reinterpret_cast<intptr_t>
(&singletonValue(t, methodCompiled(t, method_), 0));
case NativeMethod:
return 0;
default:
abort(t);
}
}
virtual unsigned count() {
class Visitor: public Processor::StackVisitor {
public:
Visitor(): count(0) { }
unsigned count = 0;
virtual bool visit(Processor::StackWalker*) {
for (MyStackWalker walker(this); walker.valid();) {
walker.next();
++ count;
return true;
}
unsigned count;
} v;
MyStackWalker walker(this);
walker.walk(&v);
return v.count;
return count;
}
MyThread* t;
State state;
void* ip_;
void* base;
void* stack;
MyThread::CallTrace* trace;
object nativeMethod;
object method_;
MyProtector protector;
};
@ -1012,14 +1063,9 @@ findCallNode(MyThread* t, void* address);
void
insertCallNode(MyThread* t, object node);
void
removeCallNode(MyThread* t, object node);
void*
findExceptionHandler(Thread* t, object method, void* ip)
{
PROTECT(t, method);
object table = codeExceptionHandlerTable(t, methodCode(t, method));
if (table) {
object index = arrayBody(t, table, 0);
@ -1033,25 +1079,7 @@ findExceptionHandler(Thread* t, object method, void* ip)
unsigned key = difference(ip, compiled) - 1;
if (key >= start and key < end) {
object catchType = 0;
if (arrayBody(t, table, i + 1)) {
object e = t->exception;
t->exception = 0;
PROTECT(t, e);
PROTECT(t, table);
PROTECT(t, index);
catchType = resolveClassInObject
(t, table, ArrayBody + ((i + 1) * BytesPerWord));
if (catchType) {
t->exception = e;
} else {
// can't find what we're supposed to catch - move on.
continue;
}
}
object catchType = arrayBody(t, table, i + 1);
if (catchType == 0 or instanceOf(t, catchType, t->exception)) {
return compiled + intArrayBody(t, index, (i * 3) + 2);
@ -1070,9 +1098,7 @@ findUnwindTarget(MyThread* t, void** targetIp, void** targetBase,
void* ip = t->ip;
void* base = t->base;
void** stack = static_cast<void**>(t->stack);
if (ip) {
t->ip = 0;
} else {
if (ip == 0) {
ip = *stack;
}
@ -2091,21 +2117,21 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
frame->pushInt
(c->call
(c->constant(reinterpret_cast<intptr_t>(doubleToFloat)),
0, 0, 0, 4, 1, frame->popLong()));
context->indirection, 0, 0, 4, 1, frame->popLong()));
} break;
case d2i: {
frame->pushInt
(c->call
(c->constant(reinterpret_cast<intptr_t>(doubleToInt)),
0, 0, 0, 4, 1, frame->popLong()));
context->indirection, 0, 0, 4, 1, frame->popLong()));
} break;
case d2l: {
frame->pushLong
(c->call
(c->constant(reinterpret_cast<intptr_t>(doubleToLong)),
0, 0, 0, 8, 1, frame->popLong()));
context->indirection, 0, 0, 8, 1, frame->popLong()));
} break;
case dadd: {
@ -2115,7 +2141,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
frame->pushLong
(c->call
(c->constant(reinterpret_cast<intptr_t>(addDouble)),
0, 0, 0, 8, 2, a, b));
context->indirection, 0, 0, 8, 2, a, b));
} break;
case dcmpg: {
@ -2125,7 +2151,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
frame->pushInt
(c->call
(c->constant(reinterpret_cast<intptr_t>(compareDoublesG)),
0, 0, 0, 4, 2, a, b));
context->indirection, 0, 0, 4, 2, a, b));
} break;
case dcmpl: {
@ -2135,7 +2161,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
frame->pushInt
(c->call
(c->constant(reinterpret_cast<intptr_t>(compareDoublesL)),
0, 0, 0, 4, 2, a, b));
context->indirection, 0, 0, 4, 2, a, b));
} break;
case dconst_0:
@ -2153,7 +2179,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
frame->pushLong
(c->call
(c->constant(reinterpret_cast<intptr_t>(divideDouble)),
0, 0, 0, 8, 2, a, b));
context->indirection, 0, 0, 8, 2, a, b));
} break;
case dmul: {
@ -2163,14 +2189,14 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
frame->pushLong
(c->call
(c->constant(reinterpret_cast<intptr_t>(multiplyDouble)),
0, 0, 0, 8, 2, a, b));
context->indirection, 0, 0, 8, 2, a, b));
} break;
case dneg: {
frame->pushLong
(c->call
(c->constant(reinterpret_cast<intptr_t>(negateDouble)),
0, 0, 0, 8, 1, frame->popLong()));
context->indirection, 0, 0, 8, 1, frame->popLong()));
} break;
case vm::drem: {
@ -2180,7 +2206,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
frame->pushLong
(c->call
(c->constant(reinterpret_cast<intptr_t>(moduloDouble)),
0, 0, 0, 8, 2, a, b));
context->indirection, 0, 0, 8, 2, a, b));
} break;
case dsub: {
@ -2190,7 +2216,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
frame->pushLong
(c->call
(c->constant(reinterpret_cast<intptr_t>(subtractDouble)),
0, 0, 0, 8, 2, a, b));
context->indirection, 0, 0, 8, 2, a, b));
} break;
case dup:
@ -2221,21 +2247,21 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
frame->pushLong
(c->call
(c->constant(reinterpret_cast<intptr_t>(floatToDouble)),
0, 0, 0, 8, 1, frame->popInt()));
context->indirection, 0, 0, 8, 1, frame->popInt()));
} break;
case f2i: {
frame->pushInt
(c->call
(c->constant(reinterpret_cast<intptr_t>(floatToInt)),
0, 0, 0, 4, 1, frame->popInt()));
context->indirection, 0, 0, 4, 1, frame->popInt()));
} break;
case f2l: {
frame->pushLong
(c->call
(c->constant(reinterpret_cast<intptr_t>(floatToLong)),
0, 0, 0, 8, 1, frame->popInt()));
context->indirection, 0, 0, 8, 1, frame->popInt()));
} break;
case fadd: {
@ -2245,7 +2271,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
frame->pushInt
(c->call
(c->constant(reinterpret_cast<intptr_t>(addFloat)),
0, 0, 0, 4, 2, a, b));
context->indirection, 0, 0, 4, 2, a, b));
} break;
case fcmpg: {
@ -2255,7 +2281,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
frame->pushInt
(c->call
(c->constant(reinterpret_cast<intptr_t>(compareFloatsG)),
0, 0, 0, 4, 2, a, b));
context->indirection, 0, 0, 4, 2, a, b));
} break;
case fcmpl: {
@ -2265,7 +2291,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
frame->pushInt
(c->call
(c->constant(reinterpret_cast<intptr_t>(compareFloatsL)),
0, 0, 0, 4, 2, a, b));
context->indirection, 0, 0, 4, 2, a, b));
} break;
case fconst_0:
@ -2287,7 +2313,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
frame->pushInt
(c->call
(c->constant(reinterpret_cast<intptr_t>(divideFloat)),
0, 0, 0, 4, 2, a, b));
context->indirection, 0, 0, 4, 2, a, b));
} break;
case fmul: {
@ -2297,14 +2323,14 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
frame->pushInt
(c->call
(c->constant(reinterpret_cast<intptr_t>(multiplyFloat)),
0, 0, 0, 4, 2, a, b));
context->indirection, 0, 0, 4, 2, a, b));
} break;
case fneg: {
frame->pushInt
(c->call
(c->constant(reinterpret_cast<intptr_t>(negateFloat)),
0, 0, 0, 4, 1, frame->popInt()));
context->indirection, 0, 0, 4, 1, frame->popInt()));
} break;
case vm::frem: {
@ -2314,7 +2340,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
frame->pushInt
(c->call
(c->constant(reinterpret_cast<intptr_t>(moduloFloat)),
0, 0, 0, 4, 2, a, b));
context->indirection, 0, 0, 4, 2, a, b));
} break;
case fsub: {
@ -2324,7 +2350,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
frame->pushInt
(c->call
(c->constant(reinterpret_cast<intptr_t>(subtractFloat)),
0, 0, 0, 4, 2, a, b));
context->indirection, 0, 0, 4, 2, a, b));
} break;
case getfield:
@ -2422,14 +2448,14 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
frame->pushLong
(c->call
(c->constant(reinterpret_cast<intptr_t>(intToDouble)),
0, 0, 0, 8, 1, frame->popInt()));
context->indirection, 0, 0, 8, 1, frame->popInt()));
} break;
case i2f: {
frame->pushInt
(c->call
(c->constant(reinterpret_cast<intptr_t>(intToFloat)),
0, 0, 0, 4, 1, frame->popInt()));
context->indirection, 0, 0, 4, 1, frame->popInt()));
} break;
case i2l:
@ -2656,7 +2682,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
frame->pushInt
(c->call
(c->constant(reinterpret_cast<intptr_t>(instanceOf)),
0, 0, 0, 4,
context->indirection, 0, 0, 4,
3, c->thread(), frame->append(class_), frame->popObject()));
} break;
@ -2852,14 +2878,14 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
frame->pushLong
(c->call
(c->constant(reinterpret_cast<intptr_t>(longToDouble)),
0, 0, 0, 8, 1, frame->popLong()));
context->indirection, 0, 0, 8, 1, frame->popLong()));
} break;
case l2f: {
frame->pushInt
(c->call
(c->constant(reinterpret_cast<intptr_t>(longToFloat)),
0, 0, 0, 4, 1, frame->popLong()));
context->indirection, 0, 0, 4, 1, frame->popLong()));
} break;
case l2i:
@ -3029,7 +3055,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
c->jmp
(c->call
(c->constant(reinterpret_cast<intptr_t>(lookUpAddress)),
0, 0, 0, BytesPerWord,
context->indirection, 0, 0, BytesPerWord,
4, key, start, c->constant(pairCount), default_));
for (int32_t i = 0; i < pairCount; ++i) {
@ -3327,7 +3353,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
} else {
c->call
(c->constant(reinterpret_cast<intptr_t>(set)),
0, 0, 0, 0,
context->indirection, 0, 0, 0,
4, c->thread(), table, c->constant(fieldOffset(t, field)), value);
}
break;
@ -3484,6 +3510,8 @@ translateExceptionHandlerTable(MyThread* t, Compiler* c, object code,
PROTECT(t, newIndex);
object newTable = makeArray(t, length + 1, false);
PROTECT(t, newTable);
set(t, newTable, ArrayBody, newIndex);
for (unsigned i = 0; i < length; ++i) {
@ -3499,10 +3527,14 @@ translateExceptionHandlerTable(MyThread* t, Compiler* c, object code,
intArrayBody(t, newIndex, (i * 3) + 2)
= c->machineIp(exceptionHandlerIp(oldHandler))->value() - start;
object type =
(exceptionHandlerCatchType(oldHandler) ?
singletonObject(t, codePool(t, code),
exceptionHandlerCatchType(oldHandler) - 1) : 0);
object type;
if (exceptionHandlerCatchType(oldHandler)) {
type = resolveClassInPool
(t, codePool(t, code), exceptionHandlerCatchType(oldHandler) - 1);
if (UNLIKELY(t->exception)) return;
} else {
type = 0;
}
set(t, newTable, ArrayBody + ((i + 1) * BytesPerWord), type);
}
@ -3739,6 +3771,7 @@ finish(MyThread* t, Context* context)
translateExceptionHandlerTable(t, c, methodCode(t, context->method),
reinterpret_cast<intptr_t>(start));
if (UNLIKELY(t->exception)) return 0;
translateLineNumberTable(t, c, methodCode(t, context->method),
reinterpret_cast<intptr_t>(start));
@ -3980,14 +4013,6 @@ compileMethod2(MyThread* t)
(t, resolveThisPointer(t, t->stack, target)), methodOffset(t, target))
= &singletonValue(t, methodCompiled(t, target), 0);
} else {
#ifndef VM_STRESS
// valgrind doesn't like this, since the effect of updateCall
// below does not propagate to valgrind's interpreter:
{ ACQUIRE(t, t->m->classLock);
removeCallNode(t, node);
}
#endif
Context context(t);
context.assembler->updateCall
(reinterpret_cast<void*>(callNodeAddress(t, node)),
@ -4196,11 +4221,11 @@ invokeNative(MyThread* t)
{
if (t->trace->nativeMethod == 0) {
object node = findCallNode(t, *static_cast<void**>(t->stack));
t->trace->nativeMethod = callNodeTarget(t, node);
object target = callNodeTarget(t, node);
if (callNodeVirtualCall(t, node)) {
t->trace->nativeMethod = resolveTarget
(t, t->stack, t->trace->nativeMethod);
target = resolveTarget(t, t->stack, target);
}
t->trace->nativeMethod = target;
}
uint64_t result = 0;
@ -4535,6 +4560,13 @@ invoke(Thread* thread, object method, ArgumentList* arguments)
arguments->position, returnType);
}
if (t->exception) {
if (t->backupHeap) {
collect(t, Heap::MinorCollection);
}
return 0;
}
object r;
switch (returnCode) {
case ByteField:
@ -4571,16 +4603,15 @@ traceSize(Thread* t)
{
class Counter: public Processor::StackVisitor {
public:
Counter(Thread* t): t(t), count(0) { }
Counter(): count(0) { }
virtual bool visit(Processor::StackWalker*) {
++ count;
return true;
}
Thread* t;
unsigned count;
} counter(t);
} counter;
t->m->processor->walkStack(t, &counter);
@ -4599,15 +4630,26 @@ class SegFaultHandler: public System::SignalHandler {
if (t->state == Thread::ActiveState) {
object node = methodForIp(t, *ip);
if (node) {
void* oldIp = t->ip;
void* oldBase = t->base;
void* oldStack = t->stack;
t->ip = *ip;
t->base = *base;
t->stack = *stack;
ensure(t, FixedSizeOfNullPointerException + traceSize(t));
t->tracing = true;
t->exception = makeNullPointerException(t);
t->tracing = false;
findUnwindTarget(t, ip, base, stack);
t->ip = oldIp;
t->base = oldBase;
t->stack = oldStack;
*thread = t;
return true;
}
@ -4618,6 +4660,11 @@ class SegFaultHandler: public System::SignalHandler {
Machine* m;
};
class MyProcessor;
MyProcessor*
processor(MyThread* t);
class MyProcessor: public Processor {
public:
class CodeAllocator: public Allocator {
@ -4929,22 +4976,54 @@ class MyProcessor: public Processor {
MyThread* t = static_cast<MyThread*>(vmt);
MyThread* target = static_cast<MyThread*>(vmTarget);
processor(t)->getDefaultCompiled(t);
processor(t)->getNativeCompiled(t);
class Visitor: public System::ThreadVisitor {
public:
Visitor(MyThread* t, MyThread* target): t(t), target(target) { }
virtual void visit(void* ip, void* base, void* stack) {
ensure(t, traceSize(t));
void* oldIp = target->ip;
void* oldBase = target->ip;
void* oldBase = target->base;
void* oldStack = target->stack;
if (methodForIp(t, ip)) {
target->ip = ip;
target->base = base;
target->stack = stack;
} else {
MyProcessor* p = processor(t);
uint8_t* default_ = reinterpret_cast<uint8_t*>
(&singletonValue(t, p->defaultCompiled, 0));
unsigned defaultSize = singletonLength(t, p->defaultCompiled);
uint8_t* native = reinterpret_cast<uint8_t*>
(&singletonValue(t, p->nativeCompiled, 0));
unsigned nativeSize = singletonLength(t, p->nativeCompiled);
if ((static_cast<uint8_t*>(ip) >= p->indirectCaller
and static_cast<uint8_t*>(ip)
< p->indirectCaller + p->indirectCallerSize)
or (static_cast<uint8_t*>(ip) >= default_
and static_cast<uint8_t*>(ip) < default_ + defaultSize)
or (static_cast<uint8_t*>(ip) >= native
and static_cast<uint8_t*>(ip) < native + nativeSize))
{
target->ip = *static_cast<void**>(stack);
target->base = base;
target->stack = stack;
}
}
ensure(t, traceSize(target));
t->tracing = true;
trace = makeTrace(t, target);
t->tracing = false;
target->ip = oldIp;
target->base = oldBase;
@ -4956,6 +5035,8 @@ class MyProcessor: public Processor {
object trace;
} visitor(t, target);
t->m->system->visit(t->systemThread, target->systemThread, &visitor);
if (t->backupHeap) {
PROTECT(t, visitor.trace);
@ -5082,7 +5163,7 @@ object
findCallNode(MyThread* t, void* address)
{
if (DebugCallTable) {
fprintf(stderr, "find trace node %p\n", address);
fprintf(stderr, "find call node %p\n", address);
}
MyProcessor* p = processor(t);
@ -5142,7 +5223,7 @@ void
insertCallNode(MyThread* t, object node)
{
if (DebugCallTable) {
fprintf(stderr, "insert trace node %p\n",
fprintf(stderr, "insert call node %p\n",
reinterpret_cast<void*>(callNodeAddress(t, node)));
}
@ -5164,50 +5245,6 @@ insertCallNode(MyThread* t, object node)
set(t, p->callTable, ArrayBody + (index * BytesPerWord), node);
}
void
removeCallNode(MyThread* t, object node)
{
if (DebugCallTable) {
fprintf(stderr, "remove call node %p\n",
reinterpret_cast<void*>(callNodeAddress(t, node)));
}
MyProcessor* p = processor(t);
PROTECT(t, node);
object oldNode = 0;
PROTECT(t, oldNode);
object newNode = 0;
PROTECT(t, newNode);
intptr_t key = callNodeAddress(t, node);
unsigned index = static_cast<uintptr_t>(key)
& (arrayLength(t, p->callTable) - 1);
for (oldNode = arrayBody(t, p->callTable, index);
oldNode;
oldNode = callNodeNext(t, oldNode))
{
if (oldNode != node) {
newNode = makeCallNode
(t, callNodeAddress(t, oldNode),
callNodeTarget(t, oldNode),
callNodeVirtualCall(t, oldNode),
newNode);
}
}
set(t, p->callTable, ArrayBody + (index * BytesPerWord), newNode);
-- p->callTableSize;
if (p->callTableSize <= arrayLength(t, p->callTable) / 3) {
p->callTable = resizeTable
(t, p->callTable, arrayLength(t, p->callTable) / 2);
}
}
object&
methodTree(MyThread* t)
{

View File

@ -1709,6 +1709,11 @@ class MyHeap: public Heap {
}
}
virtual bool needsMark(void* p, unsigned offset) {
return needsMark(p) and targetNeedsMark
(mask(*(static_cast<void**>(p) + offset)));
}
bool targetNeedsMark(void* target) {
return target
and not c.gen2.contains(target)

View File

@ -62,6 +62,7 @@ class Heap: public Allocator {
virtual void* allocateImmortal(Allocator* allocator, unsigned sizeInWords,
bool objectMask, unsigned* totalInBytes) = 0;
virtual bool needsMark(void* p) = 0;
virtual bool needsMark(void* p, unsigned offset) = 0;
virtual void mark(void* p, unsigned offset, unsigned count) = 0;
virtual void pad(void* p) = 0;
virtual void* follow(void* p) = 0;

View File

@ -490,6 +490,7 @@ postCollect(Thread* t)
if (t->backupHeap) {
t->m->heap->free
(t->backupHeap, t->backupHeapSizeInWords * BytesPerWord);
t->backupHeap = 0;
t->backupHeapIndex = 0;
t->backupHeapSizeInWords = 0;
}
@ -1735,7 +1736,8 @@ Thread::Thread(Machine* m, object javaThread, Thread* parent):
heap(defaultHeap),
backupHeap(0),
backupHeapIndex(0),
backupHeapSizeInWords(0)
backupHeapSizeInWords(0),
tracing(false)
#ifdef VM_STRESS
, stress(false)
#endif // VM_STRESS
@ -2046,6 +2048,10 @@ allocate3(Thread* t, Allocator* allocator, Machine::AllocationType type,
t->backupHeapIndex += ceiling(sizeInBytes, BytesPerWord);
cast<object>(o, 0) = 0;
return o;
} else if (t->tracing) {
expect(t, t->heapIndex + ceiling(sizeInBytes, BytesPerWord)
<= Thread::HeapSizeInWords);
return allocateSmall(t, sizeInBytes);
}
ACQUIRE_RAW(t, t->m->stateLock);
@ -2826,6 +2832,7 @@ makeTrace(Thread* t, Processor::StackWalker* walker)
}
object e = makeTraceElement(t, walker->method(), walker->ip());
assert(t, index < arrayLength(t, trace));
set(t, trace, ArrayBody + (index * BytesPerWord), e);
++ index;
return true;

View File

@ -1291,6 +1291,7 @@ class Thread {
uintptr_t* backupHeap;
unsigned backupHeapIndex;
unsigned backupHeapSizeInWords;
bool tracing;
#ifdef VM_STRESS
bool stress;
#endif // VM_STRESS
@ -1486,7 +1487,10 @@ mark(Thread* t, object o, unsigned offset, unsigned count)
inline void
mark(Thread* t, object o, unsigned offset)
{
mark(t, o, offset, 1);
if (t->m->heap->needsMark(o, offset / BytesPerWord)) {
ACQUIRE_RAW(t, t->m->heapLock);
t->m->heap->mark(o, offset / BytesPerWord, 1);
}
}
inline void

View File

@ -793,6 +793,8 @@ handleSignal(int signal, siginfo_t* info, void* context)
System::Thread* t = system->visitTarget;
system->visitTarget = 0;
ACQUIRE_MONITOR(t, system->visitLock);
system->visitLock->notifyAll(t);
} break;
@ -830,8 +832,14 @@ handleSignal(int signal, siginfo_t* info, void* context)
} else if (system->oldHandlers[index].sa_handler) {
system->oldHandlers[index].sa_handler(signal);
} else {
switch (signal) {
case VisitSignal:
break;
default:
abort();
}
}
}
} // namespace

View File

@ -223,7 +223,11 @@ findLineNumber(Thread* t, object method, unsigned ip)
}
}
abort(t);
if (top < lineNumberTableLength(t, lnt)) {
return lineNumberLine(lineNumberTableBody(t, lnt, top));
} else {
return UnknownLine;
}
} else {
return UnknownLine;
}

View File

@ -1793,11 +1793,14 @@ writeSizes(Output* out, Object* declarations)
out->write(typeFixedSize(o));
out->write(";\n\n");
int aes = typeArrayElementSize(o);
if (aes) {
out->write("const unsigned ArrayElementSizeOf");
out->write(capitalize(typeName(o)));
out->write(" = ");
out->write(typeArrayElementSize(o));
out->write(aes);
out->write(";\n\n");
}
} break;
default: break;

View File

@ -17,13 +17,14 @@
.globl vmNativeCall
vmNativeCall:
pushq %rbp
movq %rsp,%rbp
// %rdi aka 0(%rbp): function
// %rsi aka 8(%rbp): stack
// %rdx aka 16(%rbp): stackSize
// %rcx aka 24(%rbp): gprTable
// %r8 aka 32(%rbp): sseTable
// %r9 aka 40(%rbp): returnType
// %rdi aka -48(%rbp): function
// %rsi aka -40(%rbp): stack
// %rdx aka -32(%rbp): stackSize
// %rcx aka -24(%rbp): gprTable
// %r8 aka -16(%rbp): sseTable
// %r9 aka -8(%rbp): returnType
// save our argument registers so we can clobber them
pushq %r9
@ -33,8 +34,6 @@ vmNativeCall:
pushq %rsi
pushq %rdi
movq %rsp,%rbp
// reserve space for arguments passed via memory
subq %rdx,%rsp
@ -49,21 +48,21 @@ loop:
movq %rcx,%rax
movq %rcx,%rdx
addq %rsp,%rdx
addq 8(%rbp),%rax
addq -40(%rbp),%rax
movq (%rax),%rax
movq %rax,(%rdx)
addq $8,%rcx
test:
cmpq 16(%rbp),%rcx
cmpq -32(%rbp),%rcx
jb loop
// do we need to load the general-purpose registers?
cmpq $0,24(%rbp)
cmpq $0,-24(%rbp)
je sse
// yes, we do
movq 24(%rbp),%rax
movq -24(%rbp),%rax
movq 0(%rax),%rdi
movq 8(%rax),%rsi
movq 16(%rax),%rdx
@ -73,11 +72,11 @@ test:
sse:
// do we need to load the SSE registers?
cmpq $0,32(%rbp)
cmpq $0,-16(%rbp)
je call
// yes, we do
movq 32(%rbp),%rax
movq -16(%rbp),%rax
movq 0(%rax),%xmm0
movq 8(%rax),%xmm1
movq 16(%rax),%xmm2
@ -88,10 +87,10 @@ sse:
movq 64(%rax),%xmm7
call:
call *0(%rbp)
call *-48(%rbp)
// handle return value based on expected type
movq 40(%rbp),%rcx
movq -8(%rbp),%rcx
void:
cmpq $VOID_TYPE,%rcx
@ -109,10 +108,6 @@ copy:
exit:
movq %rbp,%rsp
// pop our argument registers
addq $48,%rsp
popq %rbp
ret