mirror of
https://github.com/corda/corda.git
synced 2025-01-07 13:38:47 +00:00
Merge branch 'master' into compiler
Conflicts: src/compile.cpp src/compiler.cpp src/compiler.h
This commit is contained in:
commit
9f8d77a2a9
@ -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_;
|
||||
}
|
||||
|
@ -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) {
|
||||
|
@ -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);
|
||||
|
||||
|
@ -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__
|
||||
|
383
src/compile.cpp
383
src/compile.cpp
@ -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)
|
||||
{
|
||||
|
@ -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)
|
||||
|
@ -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;
|
||||
|
@ -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;
|
||||
|
@ -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
|
||||
|
@ -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
|
||||
|
@ -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;
|
||||
}
|
||||
|
@ -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;
|
||||
|
35
src/x86.S
35
src/x86.S
@ -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
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user