corda/src/compile.cpp

5207 lines
116 KiB
C++
Raw Normal View History

/* Copyright (c) 2008, Avian Contributors
Permission to use, copy, modify, and/or distribute this software
for any purpose with or without fee is hereby granted, provided
that the above copyright notice and this permission notice appear
in all copies.
There is NO WARRANTY for this software. See license.txt for
details. */
#include "machine.h"
2007-12-09 22:45:43 +00:00
#include "util.h"
#include "vector.h"
#include "process.h"
2007-12-09 22:45:43 +00:00
#include "compiler.h"
#include "x86.h"
using namespace vm;
extern "C" uint64_t
2007-12-09 22:45:43 +00:00
vmInvoke(void* thread, void* function, void* stack, unsigned stackSize,
unsigned returnType);
2007-10-04 03:19:39 +00:00
extern "C" void
vmCall();
namespace {
const bool Verbose = true;
const bool DebugNatives = false;
const bool DebugCallTable = false;
const bool DebugMethodTree = true;
const bool DebugFrameMaps = false;
2007-12-11 23:52:28 +00:00
const bool CheckArrayBounds = true;
2007-12-09 22:45:43 +00:00
class MyThread: public Thread {
public:
class CallTrace {
public:
CallTrace(MyThread* t):
t(t),
2007-12-30 22:24:48 +00:00
ip(t->ip),
2007-12-09 22:45:43 +00:00
base(t->base),
stack(t->stack),
nativeMethod(0),
2007-12-09 22:45:43 +00:00
next(t->trace)
{
t->trace = this;
2007-12-30 22:24:48 +00:00
t->ip = 0;
2007-12-09 22:45:43 +00:00
t->base = 0;
t->stack = 0;
}
2007-09-30 15:52:21 +00:00
2007-12-09 22:45:43 +00:00
~CallTrace() {
t->stack = stack;
t->base = base;
2007-12-30 22:24:48 +00:00
t->ip = ip;
2007-12-09 22:45:43 +00:00
t->trace = next;
}
2007-09-25 23:53:11 +00:00
2007-12-09 22:45:43 +00:00
MyThread* t;
2007-12-30 22:24:48 +00:00
void* ip;
2007-12-09 22:45:43 +00:00
void* base;
void* stack;
object nativeMethod;
2007-12-09 22:45:43 +00:00
CallTrace* next;
};
2007-12-09 22:45:43 +00:00
MyThread(Machine* m, object javaThread, Thread* parent):
Thread(m, javaThread, parent),
2007-12-30 22:24:48 +00:00
ip(0),
2007-12-09 22:45:43 +00:00
base(0),
stack(0),
trace(0),
reference(0)
2007-10-03 00:22:48 +00:00
{ }
2007-12-30 22:24:48 +00:00
void* ip;
2007-12-09 22:45:43 +00:00
void* base;
void* stack;
CallTrace* trace;
Reference* reference;
};
object
resolveThisPointer(MyThread* t, void* stack, object method)
2007-12-09 22:45:43 +00:00
{
return reinterpret_cast<object*>(stack)[methodParameterFootprint(t, method)];
}
2007-12-23 19:26:35 +00:00
object
resolveTarget(MyThread* t, void* stack, object method)
{
object class_ = objectClass(t, resolveThisPointer(t, stack, method));
2007-10-03 00:22:48 +00:00
if (classVmFlags(t, class_) & BootstrapFlag) {
PROTECT(t, method);
PROTECT(t, class_);
resolveClass(t, className(t, class_));
if (UNLIKELY(t->exception)) return 0;
2007-10-03 00:22:48 +00:00
}
if (classFlags(t, methodClass(t, method)) & ACC_INTERFACE) {
return findInterfaceMethod(t, method, class_);
} else {
return findMethod(t, method, class_);
}
2007-12-09 22:45:43 +00:00
}
2007-10-03 00:22:48 +00:00
object&
methodTree(MyThread* t);
2007-12-09 22:45:43 +00:00
object
methodTreeSentinal(MyThread* t);
2007-10-03 00:22:48 +00:00
intptr_t
compareIpToMethodBounds(Thread* t, intptr_t ip, object method)
{
intptr_t start = reinterpret_cast<intptr_t>
(&singletonValue(t, methodCompiled(t, method), 0));
if (DebugMethodTree) {
fprintf(stderr, "find 0x%lx in (0x%lx,0x%lx)\n", ip, start,
start + (singletonCount(t, methodCompiled(t, method))
* BytesPerWord));
}
if (ip < start) {
return -1;
} else if (ip < start + (singletonCount(t, methodCompiled(t, method))
* BytesPerWord))
{
return 0;
} else {
return 1;
}
}
object
methodForIp(MyThread* t, void* ip)
{
if (DebugMethodTree) {
fprintf(stderr, "query for method containing %p\n", ip);
}
return treeQuery(t, methodTree(t), reinterpret_cast<intptr_t>(ip),
methodTreeSentinal(t), compareIpToMethodBounds);
}
2007-10-03 00:22:48 +00:00
2007-12-09 22:45:43 +00:00
class MyStackWalker: public Processor::StackWalker {
public:
class MyProtector: public Thread::Protector {
public:
MyProtector(MyStackWalker* walker):
Protector(walker->t), walker(walker)
{ }
2007-10-03 00:22:48 +00:00
2007-12-09 22:45:43 +00:00
virtual void visit(Heap::Visitor* v) {
v->visit(&(walker->method_));
2007-12-09 22:45:43 +00:00
v->visit(&(walker->nativeMethod));
}
2007-10-03 00:22:48 +00:00
2007-12-09 22:45:43 +00:00
MyStackWalker* walker;
};
MyStackWalker(MyThread* t):
t(t),
ip_(t->ip ? t->ip : (t->stack ? *static_cast<void**>(t->stack) : 0)),
2007-12-09 22:45:43 +00:00
base(t->base),
stack(t->stack),
trace(t->trace),
nativeMethod(trace->nativeMethod),
method_(ip_ ? methodForIp(t, ip_) : 0),
2007-12-09 22:45:43 +00:00
protector(this)
{ }
MyStackWalker(MyStackWalker* w):
t(w->t),
ip_(w->ip_),
2007-12-09 22:45:43 +00:00
base(w->base),
stack(w->stack),
trace(w->trace),
nativeMethod(w->nativeMethod),
method_(w->method_),
2007-12-09 22:45:43 +00:00
protector(this)
{ }
2007-12-09 22:45:43 +00:00
virtual void walk(Processor::StackVisitor* v) {
fprintf(stderr, "ip: %p stack: %p method: %p\n", ip_, stack, method_);
if (stack == 0) {
return;
}
2007-12-09 22:45:43 +00:00
if (not v->visit(this)) {
return;
}
for (MyStackWalker it(this); it.next();) {
MyStackWalker walker(it);
if (not v->visit(&walker)) {
break;
}
}
}
bool next() {
if (nativeMethod) {
nativeMethod = 0;
} else {
stack = static_cast<void**>(base) + 1;
base = *static_cast<void**>(base);
method_ = methodForIp(t, *static_cast<void**>(stack));
if (method_ == 0) {
if (trace and trace->stack) {
2007-12-09 22:45:43 +00:00
base = trace->base;
stack = static_cast<void**>(trace->stack);
nativeMethod = trace->nativeMethod;
2007-12-09 22:45:43 +00:00
trace = trace->next;
} else {
return false;
}
}
}
return true;
}
2007-12-09 22:45:43 +00:00
virtual object method() {
if (nativeMethod) {
return nativeMethod;
} else {
return method_;
2007-12-09 22:45:43 +00:00
}
2007-10-03 00:22:48 +00:00
}
2007-12-09 22:45:43 +00:00
virtual int ip() {
if (nativeMethod) {
return 0;
} else {
return reinterpret_cast<intptr_t>(ip_) - reinterpret_cast<intptr_t>
(&singletonValue(t, methodCompiled(t, method_), 0));
2007-10-03 00:22:48 +00:00
}
}
2007-12-09 22:45:43 +00:00
virtual unsigned count() {
class Visitor: public Processor::StackVisitor {
public:
Visitor(): count(0) { }
virtual bool visit(Processor::StackWalker*) {
++ count;
return true;
}
unsigned count;
} v;
MyStackWalker walker(this);
walker.walk(&v);
return v.count;
2007-10-03 00:22:48 +00:00
}
2007-12-09 22:45:43 +00:00
MyThread* t;
void* ip_;
2007-12-09 22:45:43 +00:00
void* base;
void* stack;
MyThread::CallTrace* trace;
object nativeMethod;
object method_;
2007-12-09 22:45:43 +00:00
MyProtector protector;
};
int
localOffset(MyThread* t, int v, object method)
{
int parameterFootprint = methodParameterFootprint(t, method) * BytesPerWord;
v *= BytesPerWord;
if (v < parameterFootprint) {
return (parameterFootprint - v - BytesPerWord) + (BytesPerWord * 2);
} else {
return -(v + BytesPerWord - parameterFootprint);
2007-10-03 00:22:48 +00:00
}
2007-12-09 22:45:43 +00:00
}
2007-10-03 00:22:48 +00:00
inline object*
localObject(MyThread* t, void* base, object method, unsigned index)
{
return reinterpret_cast<object*>
(static_cast<uint8_t*>(base) + localOffset(t, index, method));
}
2007-12-09 22:45:43 +00:00
class PoolElement {
public:
PoolElement(object value, Promise* address, PoolElement* next):
value(value), address(address), next(next)
2007-12-16 00:24:15 +00:00
{ }
2007-12-09 22:45:43 +00:00
object value;
2007-12-16 00:24:15 +00:00
Promise* address;
PoolElement* next;
2007-12-09 22:45:43 +00:00
};
class Context;
class TraceElement: public Compiler::TraceHandler {
2007-12-09 22:45:43 +00:00
public:
TraceElement(Context* context, object target,
bool virtualCall, TraceElement* next):
context(context),
address(0),
target(target),
virtualCall(virtualCall),
next(next)
{ }
2007-12-09 22:45:43 +00:00
virtual void handleTrace(Promise* address) {
if (this->address == 0) {
this->address = address;
}
}
Context* context;
Promise* address;
intptr_t addressValue;
2007-12-09 22:45:43 +00:00
object target;
bool virtualCall;
TraceElement* next;
uintptr_t map[0];
2007-10-03 00:22:48 +00:00
};
inline Stack*
push(Compiler* c, Stack* s, Operand* v)
{
if (BytesPerWord == 8) {
return c->push8(s, v);
} else {
return c->push4(s, v);
}
}
inline Stack*
pop(Compiler* c, Stack* s, Operand* v)
{
if (BytesPerWord == 8) {
return c->pop8(s, v);
} else {
return c->pop4(s, v);
}
}
inline void
mov(Compiler* c, Operand* src, Operand* dst)
{
if (BytesPerWord == 8) {
c->mov8(src, dst);
} else {
c->mov4(src, dst);
}
}
inline Operand*
result(Compiler* c)
{
if (BytesPerWord == 8) {
return c->result8();
} else {
return c->result4();
}
}
inline void
returnW(Compiler* c, Operand* v)
{
if (BytesPerWord == 8) {
c->return8(v);
} else {
c->return4(v);
}
}
inline void
cmp(Compiler* c, Operand* src, Operand* dst)
{
if (BytesPerWord == 8) {
c->cmp8(src, dst);
} else {
c->cmp4(src, dst);
}
}
inline void
and_(Compiler* c, Operand* src, Operand* dst)
{
if (BytesPerWord == 8) {
c->and8(src, dst);
} else {
c->and4(src, dst);
}
}
2008-01-07 14:51:07 +00:00
enum Event {
PushEvent,
PopEvent,
IpEvent,
MarkEvent,
ClearEvent,
TraceEvent
};
unsigned
localSize(MyThread* t, object method)
{
unsigned size = codeMaxLocals(t, methodCode(t, method));
if ((methodFlags(t, method) & (ACC_SYNCHRONIZED | ACC_STATIC))
== ACC_SYNCHRONIZED)
{
++ size;
}
return size;
}
2008-01-07 14:51:07 +00:00
unsigned
frameSize(MyThread* t, object method)
{
return localSize(t, method) + codeMaxStack(t, methodCode(t, method));
2008-01-07 14:51:07 +00:00
}
unsigned
stackMapSizeInWords(MyThread* t, object method)
{
return ceiling(codeMaxStack(t, methodCode(t, method)), BitsPerWord)
* BytesPerWord;
}
unsigned
frameMapSizeInWords(MyThread* t, object method)
{
return ceiling(frameSize(t, method), BitsPerWord) * BytesPerWord;
}
uint16_t*
makeVisitTable(MyThread* t, Zone* zone, object method)
{
unsigned size = codeLength(t, methodCode(t, method)) * 2;
uint16_t* table = static_cast<uint16_t*>(zone->allocate(size));
memset(table, 0, size);
return table;
}
uintptr_t*
makeRootTable(MyThread* t, Zone* zone, object method)
2008-01-07 14:51:07 +00:00
{
unsigned size = frameMapSizeInWords(t, method)
* codeLength(t, methodCode(t, method))
* BytesPerWord;
uintptr_t* table = static_cast<uintptr_t*>(zone->allocate(size));
memset(table, 0xFF, size);
2008-01-07 14:51:07 +00:00
return table;
}
class Context {
2007-10-10 22:39:40 +00:00
public:
2007-12-09 22:45:43 +00:00
class MyProtector: public Thread::Protector {
public:
MyProtector(Context* c): Protector(c->t), c(c) { }
2007-12-09 22:45:43 +00:00
virtual void visit(Heap::Visitor* v) {
v->visit(&(c->method));
2007-12-09 22:45:43 +00:00
for (PoolElement* p = c->objectPool; p; p = p->next) {
v->visit(&(p->value));
}
2007-12-09 22:45:43 +00:00
for (TraceElement* p = c->traceLog; p; p = p->next) {
v->visit(&(p->target));
2007-12-09 22:45:43 +00:00
}
}
Context* c;
2007-10-10 22:39:40 +00:00
};
Context(MyThread* t, object method, uint8_t* indirectCaller):
2007-10-10 22:39:40 +00:00
t(t),
zone(t->m->system, t->m->heap, false, 16 * 1024),
c(makeCompiler(t->m->system, t->m->heap, &zone, indirectCaller)),
2007-10-10 22:39:40 +00:00
method(method),
objectPool(0),
traceLog(0),
traceLogCount(0),
2008-01-07 14:51:07 +00:00
visitTable(makeVisitTable(t, &zone, method)),
rootTable(makeRootTable(t, &zone, method)),
eventLog(t->m->system, t->m->heap, 1024),
protector(this)
{ }
Context(MyThread* t):
t(t),
zone(t->m->system, t->m->heap, false, LikelyPageSizeInBytes),
c(makeCompiler(t->m->system, t->m->heap, &zone, 0)),
method(0),
objectPool(0),
traceLog(0),
traceLogCount(0),
2008-01-07 14:51:07 +00:00
visitTable(0),
rootTable(0),
eventLog(t->m->system, t->m->heap, 0),
protector(this)
{ }
~Context() {
c->dispose();
}
MyThread* t;
Zone zone;
Compiler* c;
object method;
PoolElement* objectPool;
TraceElement* traceLog;
unsigned traceLogCount;
2008-01-07 14:51:07 +00:00
uint16_t* visitTable;
uintptr_t* rootTable;
bool dirtyRoots;
2008-01-07 14:51:07 +00:00
Vector eventLog;
MyProtector protector;
};
class Frame {
public:
2008-01-07 14:51:07 +00:00
Frame(Context* context, uintptr_t* stackMap):
context(context),
t(context->t),
c(context->c),
stack(0),
2008-01-07 14:51:07 +00:00
stackMap(stackMap),
2007-12-09 22:45:43 +00:00
ip(0),
sp(localSize()),
2008-01-07 14:51:07 +00:00
level(0)
2007-12-09 22:45:43 +00:00
{
2008-01-07 14:51:07 +00:00
memset(stackMap, 0,
stackMapSizeInWords(t, context->method) * BytesPerWord);
2007-12-09 22:45:43 +00:00
}
Frame(Frame* f, uintptr_t* stackMap):
context(f->context),
t(context->t),
c(context->c),
2007-12-16 00:24:15 +00:00
stack(f->stack),
2008-01-07 14:51:07 +00:00
stackMap(stackMap),
2007-12-09 22:45:43 +00:00
ip(f->ip),
2008-01-07 14:51:07 +00:00
sp(f->sp),
level(f->level + 1)
2007-12-09 22:45:43 +00:00
{
2008-01-07 14:51:07 +00:00
memcpy(stackMap, f->stackMap,
stackMapSizeInWords(t, context->method) * BytesPerWord);
if (level > 1) {
context->eventLog.append(PushEvent);
}
}
~Frame() {
if (level > 1 and t->exception == 0) {
context->eventLog.append(PopEvent);
}
2007-10-10 22:39:40 +00:00
}
2007-12-09 22:45:43 +00:00
Operand* append(object o) {
2007-12-16 00:24:15 +00:00
Promise* p = c->poolAppend(0);
context->objectPool = new
(context->zone.allocate(sizeof(PoolElement)))
PoolElement(o, p, context->objectPool);
2007-12-16 00:24:15 +00:00
return c->absolute(p);
2007-10-10 22:39:40 +00:00
}
2008-01-07 14:51:07 +00:00
unsigned localSize() {
return ::localSize(t, context->method);
2007-12-09 22:45:43 +00:00
}
2008-01-07 14:51:07 +00:00
unsigned stackSize() {
return codeMaxStack(t, methodCode(t, context->method));
}
2008-01-07 14:51:07 +00:00
unsigned frameSize() {
return localSize() + stackSize();
2007-12-09 22:45:43 +00:00
}
void mark(unsigned index) {
2008-01-07 14:51:07 +00:00
assert(t, index < frameSize());
2007-10-10 22:39:40 +00:00
2008-01-07 14:51:07 +00:00
context->eventLog.append(MarkEvent);
context->eventLog.append2(index);
int si = index - localSize();
if (si >= 0) {
markBit(stackMap, si);
}
}
void clear(unsigned index) {
2008-01-07 14:51:07 +00:00
assert(t, index < frameSize());
context->eventLog.append(ClearEvent);
context->eventLog.append2(index);
int si = index - localSize();
if (si >= 0) {
clearBit(stackMap, si);
}
2007-10-10 22:39:40 +00:00
}
unsigned get(unsigned index) {
2008-01-07 14:51:07 +00:00
assert(t, index < frameSize());
int si = index - localSize();
assert(t, si >= 0);
return getBit(stackMap, si);
2007-10-10 22:39:40 +00:00
}
void pushedInt() {
2008-01-07 14:51:07 +00:00
assert(t, sp + 1 <= frameSize());
assert(t, get(sp) == 0);
2007-12-09 22:45:43 +00:00
++ sp;
2007-10-10 22:39:40 +00:00
}
void pushedObject() {
2008-01-07 14:51:07 +00:00
assert(t, sp + 1 <= frameSize());
mark(sp++);
2007-12-09 22:45:43 +00:00
}
void popped(unsigned count) {
assert(t, sp >= count);
2008-01-07 14:51:07 +00:00
assert(t, sp - count >= localSize());
while (count) {
2008-01-07 14:51:07 +00:00
clear(-- sp);
-- count;
}
}
2007-12-09 22:45:43 +00:00
void poppedInt() {
assert(t, sp >= 1);
2008-01-07 14:51:07 +00:00
assert(t, sp - 1 >= localSize());
assert(t, get(sp - 1) == 0);
2007-12-09 22:45:43 +00:00
-- sp;
}
void poppedObject() {
assert(t, sp >= 1);
2008-01-07 14:51:07 +00:00
assert(t, sp - 1 >= localSize());
assert(t, get(sp - 1) != 0);
clear(-- sp);
2007-12-09 22:45:43 +00:00
}
void storedInt(unsigned index) {
2008-01-07 14:51:07 +00:00
assert(t, index < localSize());
clear(index);
2007-12-09 22:45:43 +00:00
}
void storedObject(unsigned index) {
2008-01-07 14:51:07 +00:00
assert(t, index < localSize());
mark(index);
2007-10-10 22:39:40 +00:00
}
void dupped() {
2008-01-07 14:51:07 +00:00
assert(t, sp + 1 <= frameSize());
assert(t, sp - 1 >= localSize());
if (get(sp - 1)) {
mark(sp);
2007-12-09 22:45:43 +00:00
}
++ sp;
2007-10-10 22:39:40 +00:00
}
void duppedX1() {
2008-01-07 14:51:07 +00:00
assert(t, sp + 1 <= frameSize());
assert(t, sp - 2 >= localSize());
2007-12-09 22:45:43 +00:00
2008-01-07 14:51:07 +00:00
unsigned b2 = get(sp - 2);
unsigned b1 = get(sp - 1);
2007-12-09 22:45:43 +00:00
if (b2) {
2008-01-07 14:51:07 +00:00
mark(sp - 1);
2007-12-09 22:45:43 +00:00
} else {
2008-01-07 14:51:07 +00:00
clear(sp - 1);
2007-12-09 22:45:43 +00:00
}
if (b1) {
2008-01-07 14:51:07 +00:00
mark(sp - 2);
mark(sp);
2007-12-09 22:45:43 +00:00
} else {
2008-01-07 14:51:07 +00:00
clear(sp - 2);
2007-12-09 22:45:43 +00:00
}
++ sp;
}
void duppedX2() {
2008-01-07 14:51:07 +00:00
assert(t, sp + 1 <= frameSize());
assert(t, sp - 3 >= localSize());
2007-12-09 22:45:43 +00:00
2008-01-07 14:51:07 +00:00
unsigned b3 = get(sp - 3);
unsigned b2 = get(sp - 2);
unsigned b1 = get(sp - 1);
2007-12-09 22:45:43 +00:00
if (b3) {
2008-01-07 14:51:07 +00:00
mark(sp - 2);
2007-12-09 22:45:43 +00:00
} else {
2008-01-07 14:51:07 +00:00
clear(sp - 2);
2007-12-09 22:45:43 +00:00
}
if (b2) {
2008-01-07 14:51:07 +00:00
mark(sp - 1);
2007-12-09 22:45:43 +00:00
} else {
2008-01-07 14:51:07 +00:00
clear(sp - 1);
2007-12-09 22:45:43 +00:00
}
if (b1) {
2008-01-07 14:51:07 +00:00
mark(sp - 3);
mark(sp);
2007-12-09 22:45:43 +00:00
} else {
2008-01-07 14:51:07 +00:00
clear(sp - 3);
2007-12-09 22:45:43 +00:00
}
++ sp;
}
void dupped2() {
2008-01-07 14:51:07 +00:00
assert(t, sp + 2 <= frameSize());
assert(t, sp - 2 >= localSize());
2007-12-09 22:45:43 +00:00
2008-01-07 14:51:07 +00:00
unsigned b2 = get(sp - 2);
unsigned b1 = get(sp - 1);
2007-12-09 22:45:43 +00:00
if (b2) {
2008-01-07 14:51:07 +00:00
mark(sp);
2007-12-09 22:45:43 +00:00
}
if (b1) {
2008-01-07 14:51:07 +00:00
mark(sp + 1);
2007-12-09 22:45:43 +00:00
}
sp += 2;
}
void dupped2X1() {
2008-01-07 14:51:07 +00:00
assert(t, sp + 2 <= frameSize());
assert(t, sp - 3 >= localSize());
2007-12-09 22:45:43 +00:00
2008-01-07 14:51:07 +00:00
unsigned b3 = get(sp - 3);
unsigned b2 = get(sp - 2);
unsigned b1 = get(sp - 1);
2007-12-09 22:45:43 +00:00
if (b3) {
2008-01-07 14:51:07 +00:00
mark(sp - 1);
2007-12-09 22:45:43 +00:00
} else {
2008-01-07 14:51:07 +00:00
clear(sp - 1);
2007-12-09 22:45:43 +00:00
}
if (b2) {
2008-01-07 14:51:07 +00:00
mark(sp - 3);
mark(sp);
2007-12-09 22:45:43 +00:00
} else {
2008-01-07 14:51:07 +00:00
clear(sp - 3);
2007-12-09 22:45:43 +00:00
}
if (b1) {
2008-01-07 14:51:07 +00:00
mark(sp - 2);
mark(sp + 1);
2007-12-09 22:45:43 +00:00
} else {
2008-01-07 14:51:07 +00:00
clear(sp - 2);
2007-12-09 22:45:43 +00:00
}
sp += 2;
}
void dupped2X2() {
2008-01-07 14:51:07 +00:00
assert(t, sp + 2 <= frameSize());
assert(t, sp - 4 >= localSize());
2008-01-07 14:51:07 +00:00
unsigned b4 = get(sp - 4);
unsigned b3 = get(sp - 3);
unsigned b2 = get(sp - 2);
unsigned b1 = get(sp - 1);
2007-10-12 00:30:46 +00:00
2007-12-09 22:45:43 +00:00
if (b4) {
2008-01-07 14:51:07 +00:00
mark(sp - 2);
2007-12-09 22:45:43 +00:00
} else {
2008-01-07 14:51:07 +00:00
clear(sp - 2);
2007-12-09 22:45:43 +00:00
}
2007-10-10 22:39:40 +00:00
2007-12-09 22:45:43 +00:00
if (b3) {
2008-01-07 14:51:07 +00:00
mark(sp - 1);
2007-12-09 22:45:43 +00:00
} else {
2008-01-07 14:51:07 +00:00
clear(sp - 1);
2007-12-09 22:45:43 +00:00
}
2007-10-10 22:39:40 +00:00
2007-12-09 22:45:43 +00:00
if (b2) {
2008-01-07 14:51:07 +00:00
mark(sp - 4);
mark(sp);
2007-12-09 22:45:43 +00:00
} else {
2008-01-07 14:51:07 +00:00
clear(sp - 4);
2007-12-09 22:45:43 +00:00
}
2007-10-10 22:39:40 +00:00
2007-12-09 22:45:43 +00:00
if (b1) {
2008-01-07 14:51:07 +00:00
mark(sp - 3);
mark(sp + 1);
2007-12-09 22:45:43 +00:00
} else {
2008-01-07 14:51:07 +00:00
clear(sp - 3);
}
2007-12-09 22:45:43 +00:00
sp += 2;
}
2007-12-09 22:45:43 +00:00
void swapped() {
assert(t, sp - 2 >= localSize());
2007-12-09 22:45:43 +00:00
2008-01-07 14:51:07 +00:00
bool savedBit = get(sp - 1);
if (get(sp - 2)) {
mark(sp - 1);
2007-12-09 22:45:43 +00:00
} else {
2008-01-07 14:51:07 +00:00
clear(sp - 1);
}
2007-10-10 22:39:40 +00:00
2007-12-09 22:45:43 +00:00
if (savedBit) {
2008-01-07 14:51:07 +00:00
mark(sp - 2);
2007-12-09 22:45:43 +00:00
} else {
2008-01-07 14:51:07 +00:00
clear(sp - 2);
}
2007-10-10 22:39:40 +00:00
}
2007-12-16 00:24:15 +00:00
Operand* machineIp(unsigned logicalIp) {
return c->promiseConstant(c->machineIp(logicalIp));
}
2008-01-07 14:51:07 +00:00
void visitLogicalIp(unsigned ip) {
context->eventLog.append(IpEvent);
context->eventLog.append2(ip);
2008-01-07 14:51:07 +00:00
}
2007-12-09 22:45:43 +00:00
void startLogicalIp(unsigned ip) {
c->startLogicalIp(ip);
this->ip = ip;
}
void topIntToLong() {
dup();
if (BytesPerWord == 4) {
c->mov4To8(c->stack(stack, 0), c->stack(stack, 0));
}
}
void topLongToInt() {
mov(c, c->stack(stack, 0), c->stack(stack, 1));
stack = c->pop(stack, 1);
poppedInt();
}
2007-12-09 22:45:43 +00:00
void pushInt(Operand* o) {
stack = push(c, stack, o);
pushedInt();
}
void pushInt1(Operand* o) {
stack = c->push1(stack, o);
pushedInt();
}
void pushInt2(Operand* o) {
stack = c->push2(stack, o);
pushedInt();
}
void pushInt2z(Operand* o) {
stack = c->push2z(stack, o);
pushedInt();
}
void pushInt4(Operand* o) {
stack = c->push4(stack, o);
2007-12-09 22:45:43 +00:00
pushedInt();
}
void pushAddress(Operand* o) {
stack = push(c, stack, o);
pushedInt();
}
2007-12-09 22:45:43 +00:00
void pushObject(Operand* o) {
stack = push(c, stack, o);
2007-12-09 22:45:43 +00:00
pushedObject();
}
void pushObject() {
stack = c->pushed(stack, 1);
pushedObject();
}
2007-12-09 22:45:43 +00:00
void pushLong(Operand* o) {
2007-12-16 21:30:19 +00:00
if (BytesPerWord == 8) {
stack = c->push(stack, 1);
}
stack = c->push8(stack, o);
2007-12-16 21:30:19 +00:00
2007-12-09 22:45:43 +00:00
pushedInt();
pushedInt();
}
2007-12-09 22:45:43 +00:00
void pop(unsigned count) {
popped(count);
2007-12-16 00:24:15 +00:00
stack = c->pop(stack, count);
}
2007-12-09 22:45:43 +00:00
Operand* topInt() {
assert(t, sp >= 1);
assert(t, sp - 1 >= localSize());
2008-01-07 14:51:07 +00:00
assert(t, get(sp - 1) == 0);
2007-12-16 00:24:15 +00:00
return c->stack(stack, 0);
}
2007-12-09 22:45:43 +00:00
Operand* topLong() {
assert(t, sp >= 2);
assert(t, sp - 2 >= localSize());
2008-01-07 14:51:07 +00:00
assert(t, get(sp - 1) == 0);
assert(t, get(sp - 2) == 0);
return c->stack(stack, 0);
}
2007-12-09 22:45:43 +00:00
Operand* topObject() {
assert(t, sp >= 1);
assert(t, sp - 1 >= localSize());
2008-01-07 14:51:07 +00:00
assert(t, get(sp - 1) != 0);
2007-12-16 00:24:15 +00:00
return c->stack(stack, 0);
}
2007-12-09 22:45:43 +00:00
Operand* popInt() {
Operand* tmp = c->temporary();
2007-12-16 00:24:15 +00:00
popInt(tmp);
return tmp;
2007-12-09 22:45:43 +00:00
}
Operand* popInt4() {
Operand* tmp = c->temporary();
popInt4(tmp);
return tmp;
}
2007-12-09 22:45:43 +00:00
Operand* popLong() {
Operand* tmp = c->temporary();
2007-12-16 00:24:15 +00:00
popLong(tmp);
return tmp;
2007-12-09 22:45:43 +00:00
}
2007-12-09 22:45:43 +00:00
Operand* popObject() {
2007-12-16 00:24:15 +00:00
Operand* tmp = c->temporary();
popObject(tmp);
return tmp;
2007-12-09 22:45:43 +00:00
}
2007-12-09 22:45:43 +00:00
void popInt(Operand* o) {
stack = ::pop(c, stack, o);
2007-12-09 22:45:43 +00:00
poppedInt();
}
void popInt4(Operand* o) {
stack = c->pop4(stack, o);
poppedInt();
}
2007-12-09 22:45:43 +00:00
void popLong(Operand* o) {
stack = c->pop8(stack, o);
2007-12-16 21:30:19 +00:00
if (BytesPerWord == 8) {
stack = c->pop(stack, 1);
}
2007-12-09 22:45:43 +00:00
poppedInt();
poppedInt();
}
2007-12-09 22:45:43 +00:00
void popObject(Operand* o) {
stack = ::pop(c, stack, o);
2007-12-09 22:45:43 +00:00
poppedObject();
}
2007-12-09 22:45:43 +00:00
void loadInt(unsigned index) {
2008-01-07 14:51:07 +00:00
assert(t, index < localSize());
pushInt(c->memory(c->base(), localOffset(t, index, context->method)));
2007-10-10 22:39:40 +00:00
}
2007-12-09 22:45:43 +00:00
void loadLong(unsigned index) {
2007-12-11 23:52:28 +00:00
assert(t, index < static_cast<unsigned>
2008-01-07 14:51:07 +00:00
(localSize() - 1));
pushLong(c->memory(c->base(), localOffset(t, index + 1, context->method)));
2007-12-09 22:45:43 +00:00
}
2007-12-09 22:45:43 +00:00
void loadObject(unsigned index) {
2008-01-07 14:51:07 +00:00
assert(t, index < localSize());
pushObject(c->memory(c->base(), localOffset(t, index, context->method)));
2007-12-09 22:45:43 +00:00
}
2007-12-09 22:45:43 +00:00
void storeInt(unsigned index) {
popInt(c->memory(c->base(), localOffset(t, index, context->method)));
2007-12-09 22:45:43 +00:00
storedInt(index);
}
2007-12-09 22:45:43 +00:00
void storeLong(unsigned index) {
popLong(c->memory(c->base(), localOffset(t, index + 1, context->method)));
2007-12-09 22:45:43 +00:00
storedInt(index);
storedInt(index + 1);
}
2007-12-09 22:45:43 +00:00
void storeObject(unsigned index) {
popObject(c->memory(c->base(), localOffset(t, index, context->method)));
2007-12-09 22:45:43 +00:00
storedObject(index);
}
void storeObjectOrAddress(unsigned index) {
stack = ::pop
(c, stack, c->memory(c->base(), localOffset(t, index, context->method)));
assert(t, sp >= 1);
2008-01-07 14:51:07 +00:00
assert(t, sp - 1 >= localSize());
if (get(sp - 1)) {
storedObject(index);
} else {
storedInt(index);
}
2008-01-07 14:51:07 +00:00
popped(1);
2007-12-09 22:45:43 +00:00
}
2007-12-09 22:45:43 +00:00
void dup() {
stack = push(c, stack, c->stack(stack, 0));
2007-12-09 22:45:43 +00:00
dupped();
}
2007-12-09 22:45:43 +00:00
void dupX1() {
stack = push(c, stack, c->stack(stack, 0));
mov(c, c->stack(stack, 2), c->stack(stack, 1));
mov(c, c->stack(stack, 0), c->stack(stack, 2));
2007-12-09 22:45:43 +00:00
duppedX1();
}
2007-12-09 22:45:43 +00:00
void dupX2() {
stack = push(c, stack, c->stack(stack, 0));
mov(c, c->stack(stack, 2), c->stack(stack, 1));
mov(c, c->stack(stack, 3), c->stack(stack, 2));
mov(c, c->stack(stack, 0), c->stack(stack, 3));
2007-12-09 22:45:43 +00:00
duppedX2();
2007-10-10 22:39:40 +00:00
}
2007-12-09 22:45:43 +00:00
void dup2() {
stack = push(c, stack, c->stack(stack, 1));
stack = push(c, stack, c->stack(stack, 1));
2007-12-09 22:45:43 +00:00
dupped2();
}
2007-12-09 22:45:43 +00:00
void dup2X1() {
stack = push(c, stack, c->stack(stack, 1));
stack = push(c, stack, c->stack(stack, 1));
mov(c, c->stack(stack, 4), c->stack(stack, 2));
mov(c, c->stack(stack, 1), c->stack(stack, 4));
mov(c, c->stack(stack, 0), c->stack(stack, 3));
2007-12-09 22:45:43 +00:00
dupped2X1();
}
2007-12-09 22:45:43 +00:00
void dup2X2() {
stack = push(c, stack, c->stack(stack, 1));
stack = push(c, stack, c->stack(stack, 1));
mov(c, c->stack(stack, 5), c->stack(stack, 3));
mov(c, c->stack(stack, 4), c->stack(stack, 2));
mov(c, c->stack(stack, 1), c->stack(stack, 5));
mov(c, c->stack(stack, 0), c->stack(stack, 4));
2007-10-02 00:08:17 +00:00
2007-12-09 22:45:43 +00:00
dupped2X2();
}
2007-10-02 00:08:17 +00:00
2007-12-09 22:45:43 +00:00
void swap() {
2007-12-16 00:24:15 +00:00
Operand* s0 = c->stack(stack, 0);
Operand* s1 = c->stack(stack, 1);
2007-12-09 22:45:43 +00:00
Operand* tmp = c->temporary();
2007-10-02 00:08:17 +00:00
mov(c, s0, tmp);
mov(c, s1, s0);
mov(c, tmp, s1);
2007-10-02 00:08:17 +00:00
2007-12-09 22:45:43 +00:00
c->release(tmp);
2007-10-02 00:08:17 +00:00
2007-12-09 22:45:43 +00:00
swapped();
2007-10-02 00:08:17 +00:00
}
TraceElement* trace(object target, bool virtualCall) {
2008-01-07 14:51:07 +00:00
unsigned mapSize = frameMapSizeInWords(t, context->method);
TraceElement* e = context->traceLog = new
(context->zone.allocate(sizeof(TraceElement) + (mapSize * BytesPerWord)))
TraceElement(context, target, virtualCall, context->traceLog);
++ context->traceLogCount;
2008-01-07 14:51:07 +00:00
context->eventLog.append(TraceEvent);
context->eventLog.appendAddress(e);
return e;
}
2007-12-09 22:45:43 +00:00
Context* context;
2007-12-09 22:45:43 +00:00
MyThread* t;
Compiler* c;
2007-12-16 00:24:15 +00:00
Stack* stack;
2008-01-07 14:51:07 +00:00
uintptr_t* stackMap;
2007-12-09 22:45:43 +00:00
unsigned ip;
unsigned sp;
2008-01-07 14:51:07 +00:00
unsigned level;
2007-12-30 22:24:48 +00:00
};
unsigned
savedTargetIndex(MyThread* t, object method)
{
return codeMaxLocals(t, methodCode(t, method));
}
object
findCallNode(MyThread* t, void* address);
void
insertCallNode(MyThread* t, object node);
void
removeCallNode(MyThread* t, object node);
void
findUnwindTarget(MyThread* t, void** targetIp, void** targetBase,
void** targetStack)
2007-09-29 21:08:29 +00:00
{
2007-12-30 22:24:48 +00:00
void* ip = t->ip;
2007-12-09 22:45:43 +00:00
void* base = t->base;
void** stack = static_cast<void**>(t->stack);
if (ip) {
t->ip = 0;
} else {
2007-12-30 22:24:48 +00:00
ip = *stack;
}
*targetIp = 0;
while (*targetIp == 0) {
object method = methodForIp(t, ip);
if (method) {
PROTECT(t, method);
2007-12-09 22:45:43 +00:00
uint8_t* compiled = reinterpret_cast<uint8_t*>
(&singletonValue(t, methodCompiled(t, method), 0));
ExceptionHandler* handler = findExceptionHandler
2007-12-30 22:24:48 +00:00
(t, method, difference(ip, compiled));
2007-12-09 22:45:43 +00:00
if (handler) {
2007-10-04 22:41:19 +00:00
unsigned parameterFootprint = methodParameterFootprint(t, method);
unsigned localFootprint = localSize(t, method);
2007-10-04 22:41:19 +00:00
stack = static_cast<void**>(base)
- (localFootprint - parameterFootprint);
2007-10-04 22:41:19 +00:00
*(--stack) = t->exception;
t->exception = 0;
*targetIp = compiled + exceptionHandlerIp(handler);
*targetBase = base;
*targetStack = stack;
2007-12-09 22:45:43 +00:00
} else {
if (methodFlags(t, method) & ACC_SYNCHRONIZED) {
object lock;
if (methodFlags(t, method) & ACC_STATIC) {
lock = methodClass(t, method);
} else {
lock = *localObject(t, base, method, savedTargetIndex(t, method));
}
release(t, lock);
}
2007-12-09 22:45:43 +00:00
stack = static_cast<void**>(base) + 1;
2007-12-30 22:24:48 +00:00
ip = *stack;
2007-12-09 22:45:43 +00:00
base = *static_cast<void**>(base);
}
2007-10-12 22:06:33 +00:00
} else {
*targetIp = ip;
*targetBase = base;
*targetStack = stack + 1;
}
}
}
void NO_RETURN
unwind(MyThread* t)
{
void* ip;
void* base;
void* stack;
findUnwindTarget(t, &ip, &base, &stack);
vmJump(ip, base, stack, t);
}
2008-01-08 19:36:34 +00:00
void FORCE_ALIGN
tryInitClass(MyThread* t, object class_)
{
initClass(t, class_);
if (UNLIKELY(t->exception)) unwind(t);
}
void* FORCE_ALIGN
2007-12-30 22:24:48 +00:00
findInterfaceMethodFromInstance(MyThread* t, object method, object instance)
{
2007-12-30 22:24:48 +00:00
if (instance) {
return &singletonValue
(t, methodCompiled
(t, findInterfaceMethod(t, method, objectClass(t, instance))), 0);
} else {
t->exception = makeNullPointerException(t);
unwind(t);
}
}
intptr_t
compareDoublesG(uint64_t bi, uint64_t ai)
{
double a = bitsToDouble(ai);
double b = bitsToDouble(bi);
if (a < b) {
return -1;
} else if (a > b) {
return 1;
} else if (a == b) {
return 0;
} else {
return 1;
}
}
intptr_t
compareDoublesL(uint64_t bi, uint64_t ai)
{
double a = bitsToDouble(ai);
double b = bitsToDouble(bi);
if (a < b) {
return -1;
} else if (a > b) {
return 1;
} else if (a == b) {
return 0;
} else {
return -1;
}
}
intptr_t
compareFloatsG(uint32_t bi, uint32_t ai)
{
float a = bitsToFloat(ai);
float b = bitsToFloat(bi);
if (a < b) {
return -1;
} else if (a > b) {
return 1;
} else if (a == b) {
return 0;
} else {
return 1;
}
}
intptr_t
compareFloatsL(uint32_t bi, uint32_t ai)
{
float a = bitsToFloat(ai);
float b = bitsToFloat(bi);
if (a < b) {
return -1;
} else if (a > b) {
return 1;
} else if (a == b) {
return 0;
} else {
return -1;
}
}
uint64_t
addDouble(uint64_t b, uint64_t a)
{
return doubleToBits(bitsToDouble(a) + bitsToDouble(b));
}
uint64_t
subtractDouble(uint64_t b, uint64_t a)
{
return doubleToBits(bitsToDouble(a) - bitsToDouble(b));
}
uint64_t
multiplyDouble(uint64_t b, uint64_t a)
{
return doubleToBits(bitsToDouble(a) * bitsToDouble(b));
}
uint64_t
divideDouble(uint64_t b, uint64_t a)
{
return doubleToBits(bitsToDouble(a) / bitsToDouble(b));
}
uint64_t
moduloDouble(uint64_t b, uint64_t a)
{
return doubleToBits(fmod(bitsToDouble(a), bitsToDouble(b)));
}
2007-12-09 22:45:43 +00:00
uint64_t
negateDouble(uint64_t a)
{
return doubleToBits(- bitsToDouble(a));
}
uint32_t
doubleToFloat(int64_t a)
{
return floatToBits(static_cast<float>(bitsToDouble(a)));
}
int32_t
doubleToInt(int64_t a)
{
return static_cast<int32_t>(bitsToDouble(a));
}
int64_t
doubleToLong(int64_t a)
{
return static_cast<int64_t>(bitsToDouble(a));
}
uint32_t
addFloat(uint32_t b, uint32_t a)
{
return floatToBits(bitsToFloat(a) + bitsToFloat(b));
}
uint32_t
subtractFloat(uint32_t b, uint32_t a)
{
return floatToBits(bitsToFloat(a) - bitsToFloat(b));
}
uint32_t
multiplyFloat(uint32_t b, uint32_t a)
{
return floatToBits(bitsToFloat(a) * bitsToFloat(b));
}
uint32_t
divideFloat(uint32_t b, uint32_t a)
{
return floatToBits(bitsToFloat(a) / bitsToFloat(b));
}
uint32_t
moduloFloat(uint32_t b, uint32_t a)
{
return floatToBits(fmod(bitsToFloat(a), bitsToFloat(b)));
}
2007-12-09 22:45:43 +00:00
uint32_t
negateFloat(uint32_t a)
{
return floatToBits(- bitsToFloat(a));
}
int64_t
divideLong(int64_t b, int64_t a)
{
return a / b;
}
int64_t
moduloLong(int64_t b, int64_t a)
{
return a % b;
}
2007-12-09 22:45:43 +00:00
uint64_t
floatToDouble(int32_t a)
{
return doubleToBits(static_cast<double>(bitsToFloat(a)));
}
int32_t
floatToInt(int32_t a)
{
return static_cast<int32_t>(bitsToFloat(a));
}
int64_t
floatToLong(int32_t a)
{
return static_cast<int64_t>(bitsToFloat(a));
}
uint64_t
intToDouble(int32_t a)
{
return doubleToBits(static_cast<double>(a));
}
uint32_t
intToFloat(int32_t a)
{
return floatToBits(static_cast<float>(a));
}
2008-03-21 00:37:58 +00:00
uint64_t
longToDouble(int64_t a)
{
return doubleToBits(static_cast<double>(a));
}
uint32_t
longToFloat(int64_t a)
{
return floatToBits(static_cast<float>(a));
}
object FORCE_ALIGN
2007-09-30 02:48:27 +00:00
makeBlankObjectArray(Thread* t, object class_, int32_t length)
{
return makeObjectArray(t, class_, length, true);
}
object FORCE_ALIGN
2007-09-30 02:48:27 +00:00
makeBlankArray(Thread* t, object (*constructor)(Thread*, uintptr_t, bool),
int32_t length)
{
return constructor(t, length, true);
}
2007-12-09 22:45:43 +00:00
uintptr_t
lookUpAddress(int32_t key, uintptr_t* start, int32_t count,
uintptr_t default_)
{
2007-12-09 22:45:43 +00:00
int32_t bottom = 0;
int32_t top = count;
for (int32_t span = top - bottom; span; span = top - bottom) {
int32_t middle = bottom + (span / 2);
uintptr_t* p = start + (middle * 2);
int32_t k = *p;
if (key < k) {
top = middle;
} else if (key > k) {
bottom = middle + 1;
} else {
return p[1];
}
}
return default_;
2007-12-09 22:45:43 +00:00
}
void FORCE_ALIGN
2007-12-30 22:24:48 +00:00
setMaybeNull(MyThread* t, object o, unsigned offset, object value)
2007-12-09 22:45:43 +00:00
{
if (LIKELY(o)) {
2007-12-30 22:24:48 +00:00
set(t, o, offset, value);
} else {
t->exception = makeNullPointerException(t);
unwind(t);
}
}
void FORCE_ALIGN
2007-12-30 22:24:48 +00:00
acquireMonitorForObject(MyThread* t, object o)
{
if (LIKELY(o)) {
2007-12-30 22:24:48 +00:00
acquire(t, o);
} else {
t->exception = makeNullPointerException(t);
unwind(t);
}
2007-12-09 22:45:43 +00:00
}
void FORCE_ALIGN
2007-12-30 22:24:48 +00:00
releaseMonitorForObject(MyThread* t, object o)
2007-12-09 22:45:43 +00:00
{
if (LIKELY(o)) {
2007-12-30 22:24:48 +00:00
release(t, o);
} else {
t->exception = makeNullPointerException(t);
unwind(t);
}
2007-12-09 22:45:43 +00:00
}
2007-12-09 22:45:43 +00:00
object
makeMultidimensionalArray2(MyThread* t, object class_, uintptr_t* stack,
int32_t dimensions)
{
PROTECT(t, class_);
2007-12-09 22:45:43 +00:00
int32_t counts[dimensions];
for (int i = dimensions - 1; i >= 0; --i) {
counts[i] = stack[dimensions - i - 1];
if (UNLIKELY(counts[i] < 0)) {
object message = makeString(t, "%d", counts[i]);
t->exception = makeNegativeArraySizeException(t, message);
return 0;
}
}
2007-12-09 22:45:43 +00:00
object array = makeArray(t, counts[0], true);
setObjectClass(t, array, class_);
PROTECT(t, array);
2007-12-09 22:45:43 +00:00
populateMultiArray(t, array, counts, 0, dimensions);
2007-12-09 22:45:43 +00:00
return array;
}
object FORCE_ALIGN
2007-12-09 22:45:43 +00:00
makeMultidimensionalArray(MyThread* t, object class_, uintptr_t* stack,
int32_t dimensions)
{
2007-12-09 22:45:43 +00:00
object r = makeMultidimensionalArray2(t, class_, stack, dimensions);
if (UNLIKELY(t->exception)) {
unwind(t);
} else {
2007-12-09 22:45:43 +00:00
return r;
}
}
void NO_RETURN FORCE_ALIGN
throwArrayIndexOutOfBounds(MyThread* t, object array, int32_t index)
{
object message = makeString
(t, "array of length %d indexed at %d", arrayLength(t, array), index);
t->exception = makeArrayIndexOutOfBoundsException(t, message);
unwind(t);
}
void NO_RETURN FORCE_ALIGN
throwNegativeArraySize(MyThread* t, int32_t length)
{
object message = makeString(t, "%d", length);
t->exception = makeArrayIndexOutOfBoundsException(t, message);
2007-12-09 22:45:43 +00:00
unwind(t);
}
void NO_RETURN FORCE_ALIGN
2007-12-09 22:45:43 +00:00
throw_(MyThread* t, object o)
2007-10-09 17:15:40 +00:00
{
if (LIKELY(o)) {
2007-12-09 22:45:43 +00:00
t->exception = o;
} else {
t->exception = makeNullPointerException(t);
}
2007-12-09 22:45:43 +00:00
unwind(t);
}
void FORCE_ALIGN
checkCast(MyThread* t, object class_, object o)
{
if (UNLIKELY(o and not isAssignableFrom(t, class_, objectClass(t, o)))) {
object message = makeString
(t, "%s as %s",
&byteArrayBody(t, className(t, objectClass(t, o)), 0),
&byteArrayBody(t, className(t, class_), 0));
t->exception = makeClassCastException(t, message);
unwind(t);
}
}
2008-04-09 19:08:13 +00:00
void FORCE_ALIGN
gcIfNecessary(MyThread* t)
{
if (UNLIKELY(t->backupHeap)) {
collect(t, Heap::MinorCollection);
}
}
2007-12-09 22:45:43 +00:00
void
2007-12-22 00:26:55 +00:00
pushReturnValue(MyThread* t, Frame* frame, unsigned code)
2007-12-09 22:45:43 +00:00
{
Compiler* c = frame->c;
2007-12-22 00:26:55 +00:00
2007-12-09 22:45:43 +00:00
switch (code) {
case ByteField:
case BooleanField:
case CharField:
case ShortField:
case FloatField:
case IntField: {
Operand* result = c->result4();
frame->pushInt(result);
c->release(result);
} break;
case ObjectField: {
Operand* result = ::result(c);
2007-12-09 22:45:43 +00:00
frame->pushObject(result);
c->release(result);
} break;
2007-09-25 23:53:11 +00:00
2007-12-09 22:45:43 +00:00
case LongField:
case DoubleField: {
Operand* result = c->result8();
frame->pushLong(result);
c->release(result);
} break;
2007-12-09 22:45:43 +00:00
case VoidField:
break;
2007-09-25 23:53:11 +00:00
2007-12-09 22:45:43 +00:00
default:
abort(t);
}
2007-12-09 22:45:43 +00:00
}
2008-04-09 19:08:13 +00:00
object
defaultCompiled(MyThread* t);
object
nativeCompiled(MyThread* t);
2007-12-09 22:45:43 +00:00
void
compileDirectInvoke(MyThread* t, Frame* frame, object target)
{
Compiler* c = frame->c;
2008-04-09 19:08:13 +00:00
if (methodFlags(t, target) & ACC_NATIVE) {
c->call
(c->constant
(reinterpret_cast<intptr_t>
(&singletonBody(t, nativeCompiled(t), 0))),
frame->trace(target, false));
} else if (methodCompiled(t, target) == defaultCompiled(t)) {
c->alignedCall
(c->constant
(reinterpret_cast<intptr_t>
(&singletonBody(t, defaultCompiled(t), 0))),
frame->trace(target, false));
} else {
c->call
(c->constant
(reinterpret_cast<intptr_t>
(&singletonBody(t, methodCompiled(t, target), 0))),
frame->trace(0, false));
}
2007-09-25 23:53:11 +00:00
2007-12-09 22:45:43 +00:00
frame->pop(methodParameterFootprint(t, target));
2007-09-25 23:53:11 +00:00
pushReturnValue(t, frame, methodReturnCode(t, target));
2007-12-09 22:45:43 +00:00
}
void
handleMonitorEvent(MyThread* t, Frame* frame, intptr_t function)
{
Compiler* c = frame->c;
object method = frame->context->method;
if (methodFlags(t, method) & ACC_SYNCHRONIZED) {
Operand* lock;
if (methodFlags(t, method) & ACC_STATIC) {
lock = frame->append(methodClass(t, method));
} else {
lock = c->memory
(c->base(), localOffset(t, savedTargetIndex(t, method), method));
}
c->indirectCall
(c->constant(function),
frame->trace(0, false),
2, c->thread(), lock);
}
}
void
handleEntrance(MyThread* t, Frame* frame)
{
object method = frame->context->method;
if ((methodFlags(t, method) & (ACC_SYNCHRONIZED | ACC_STATIC))
== ACC_SYNCHRONIZED)
{
Compiler* c = frame->c;
// save 'this' pointer in case it is overwritten.
unsigned index = savedTargetIndex(t, method);
mov(c, c->memory(c->base(), localOffset(t, 0, method)),
c->memory(c->base(), localOffset(t, index, method)));
frame->mark(index);
}
handleMonitorEvent
(t, frame, reinterpret_cast<intptr_t>(acquireMonitorForObject));
}
void
handleExit(MyThread* t, Frame* frame)
{
handleMonitorEvent
(t, frame, reinterpret_cast<intptr_t>(releaseMonitorForObject));
}
2007-12-09 22:45:43 +00:00
void
2008-04-09 19:08:13 +00:00
compile(MyThread* t, Frame* initialFrame, unsigned ip,
bool exceptionHandler = false)
2007-12-09 22:45:43 +00:00
{
2008-01-07 14:51:07 +00:00
uintptr_t stackMap[stackMapSizeInWords(t, initialFrame->context->method)];
Frame myFrame(initialFrame, stackMap);
2007-12-09 22:45:43 +00:00
Frame* frame = &myFrame;
Compiler* c = frame->c;
Context* context = frame->context;
2007-09-25 23:53:11 +00:00
object code = methodCode(t, context->method);
2007-12-09 22:45:43 +00:00
PROTECT(t, code);
while (ip < codeLength(t, code)) {
2008-01-07 14:51:07 +00:00
frame->visitLogicalIp(ip);
if (context->visitTable[ip] ++) {
2007-12-09 22:45:43 +00:00
// we've already visited this part of the code
return;
2007-09-30 04:07:22 +00:00
}
2007-12-09 22:45:43 +00:00
frame->startLogicalIp(ip);
2008-04-09 19:08:13 +00:00
if (exceptionHandler) {
exceptionHandler = false;
c->indirectCall
(c->constant(reinterpret_cast<intptr_t>(gcIfNecessary)),
frame->trace(0, false),
1, c->thread());
}
2008-01-07 14:51:07 +00:00
// fprintf(stderr, "ip: %d map: %ld\n", ip, *(frame->map));
2007-12-09 22:45:43 +00:00
unsigned instruction = codeBody(t, code, ip++);
2007-09-25 23:53:11 +00:00
2007-12-09 22:45:43 +00:00
switch (instruction) {
case aaload:
case baload:
case caload:
case daload:
case faload:
case iaload:
case laload:
case saload: {
Operand* index = frame->popInt4();
2007-12-09 22:45:43 +00:00
Operand* array = frame->popObject();
if (CheckArrayBounds) {
Operand* load = c->label();
Operand* throw_ = c->label();
c->cmp4(c->constant(0), index);
c->jl(throw_);
2008-04-09 19:08:13 +00:00
c->cmp4(c->memory(array, ArrayLength, 0, 1), index);
c->jl(load);
c->mark(throw_);
c->indirectCallNoReturn
(c->constant(reinterpret_cast<intptr_t>(throwArrayIndexOutOfBounds)),
frame->trace(0, false),
3, c->thread(), array, index);
c->mark(load);
}
2007-09-25 23:53:11 +00:00
2007-12-09 22:45:43 +00:00
switch (instruction) {
case aaload:
2007-12-30 22:24:48 +00:00
frame->pushObject
(c->memory(array, ArrayBody, index, BytesPerWord));
2007-12-09 22:45:43 +00:00
break;
2007-09-25 23:53:11 +00:00
2007-12-09 22:45:43 +00:00
case faload:
case iaload:
frame->pushInt4(c->memory(array, ArrayBody, index, 4));
2007-12-09 22:45:43 +00:00
break;
2007-12-09 22:45:43 +00:00
case baload:
frame->pushInt1(c->memory(array, ArrayBody, index, 1));
2007-12-09 22:45:43 +00:00
break;
2007-12-09 22:45:43 +00:00
case caload:
frame->pushInt2z(c->memory(array, ArrayBody, index, 2));
2007-12-09 22:45:43 +00:00
break;
2007-12-09 22:45:43 +00:00
case daload:
case laload:
frame->pushLong(c->memory(array, ArrayBody, index, 8));
2007-12-09 22:45:43 +00:00
break;
2007-09-25 23:53:11 +00:00
2007-12-09 22:45:43 +00:00
case saload:
frame->pushInt2(c->memory(array, ArrayBody, index, 2));
2007-12-09 22:45:43 +00:00
break;
}
2007-09-30 02:48:27 +00:00
c->release(index);
c->release(array);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case aastore:
case bastore:
case castore:
case dastore:
case fastore:
case iastore:
case lastore:
case sastore: {
Operand* value;
if (instruction == dastore or instruction == lastore) {
value = frame->popLong();
} else if (instruction == aastore) {
value = frame->popObject();
} else {
value = frame->popInt();
}
2007-09-30 02:48:27 +00:00
Operand* index = frame->popInt4();
2007-12-09 22:45:43 +00:00
Operand* array = frame->popObject();
if (CheckArrayBounds) {
Operand* store = c->label();
Operand* throw_ = c->label();
2007-09-30 02:48:27 +00:00
c->cmp4(c->constant(0), index);
c->jl(throw_);
2008-04-09 19:08:13 +00:00
c->cmp4(c->memory(array, ArrayLength, 0, 1), index);
c->jl(store);
c->mark(throw_);
c->indirectCallNoReturn
(c->constant(reinterpret_cast<intptr_t>(throwArrayIndexOutOfBounds)),
frame->trace(0, false),
3, c->thread(), array, index);
c->mark(store);
}
2007-12-09 22:45:43 +00:00
switch (instruction) {
case aastore: {
c->shl4(c->constant(log(BytesPerWord)), index);
c->add4(c->constant(ArrayBody), index);
2007-12-09 22:45:43 +00:00
c->indirectCall
2007-12-30 22:24:48 +00:00
(c->constant(reinterpret_cast<intptr_t>(setMaybeNull)),
frame->trace(0, false),
2007-12-09 22:45:43 +00:00
4, c->thread(), array, index, value);
} break;
2007-12-09 22:45:43 +00:00
case fastore:
case iastore:
c->mov4(value, c->memory(array, ArrayBody, index, 4));
2007-12-09 22:45:43 +00:00
break;
2007-09-30 02:48:27 +00:00
2007-12-09 22:45:43 +00:00
case bastore:
c->mov1(value, c->memory(array, ArrayBody, index, 1));
2007-12-09 22:45:43 +00:00
break;
2007-09-25 23:53:11 +00:00
2007-12-09 22:45:43 +00:00
case castore:
case sastore:
c->mov2(value, c->memory(array, ArrayBody, index, 2));
2007-12-09 22:45:43 +00:00
break;
2007-12-09 22:45:43 +00:00
case dastore:
case lastore:
c->mov8(value, c->memory(array, ArrayBody, index, 8));
2007-12-09 22:45:43 +00:00
break;
}
c->release(value);
c->release(index);
c->release(array);
2007-12-09 22:45:43 +00:00
} break;
2007-09-30 15:52:21 +00:00
2007-12-09 22:45:43 +00:00
case aconst_null:
frame->pushObject(c->constant(0));
break;
2007-09-25 23:53:11 +00:00
2007-12-09 22:45:43 +00:00
case aload:
frame->loadObject(codeBody(t, code, ip++));
break;
2007-12-09 22:45:43 +00:00
case aload_0:
frame->loadObject(0);
break;
2007-10-08 23:13:55 +00:00
2007-12-09 22:45:43 +00:00
case aload_1:
frame->loadObject(1);
break;
2007-12-09 22:45:43 +00:00
case aload_2:
frame->loadObject(2);
break;
2007-12-09 22:45:43 +00:00
case aload_3:
frame->loadObject(3);
break;
2007-12-09 22:45:43 +00:00
case anewarray: {
uint16_t index = codeReadInt16(t, code, ip);
object class_ = resolveClassInPool(t, codePool(t, code), index - 1);
if (UNLIKELY(t->exception)) return;
2007-12-09 22:45:43 +00:00
Operand* nonnegative = c->label();
Operand* length = frame->popInt4();
c->cmp4(c->constant(0), length);
2007-12-09 22:45:43 +00:00
c->jge(nonnegative);
c->indirectCallNoReturn
(c->constant(reinterpret_cast<intptr_t>(throwNegativeArraySize)),
frame->trace(0, false),
2, c->thread(), length);
2007-12-09 22:45:43 +00:00
c->mark(nonnegative);
2007-10-08 23:13:55 +00:00
c->indirectCall
(c->constant(reinterpret_cast<intptr_t>(makeBlankObjectArray)),
frame->trace(0, false),
3, c->thread(), frame->append(class_), length);
Operand* result = ::result(c);
c->release(length);
frame->pushObject(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
case areturn: {
handleExit(t, frame);
Operand* result = frame->popObject();
returnW(c, result);
c->release(result);
} return;
case arraylength: {
Operand* array = frame->popObject();
2008-04-09 19:08:13 +00:00
frame->pushInt4(c->memory(array, ArrayLength, 0, 1));
c->release(array);
} break;
2007-10-04 03:19:39 +00:00
2007-12-09 22:45:43 +00:00
case astore:
frame->storeObjectOrAddress(codeBody(t, code, ip++));
2007-12-09 22:45:43 +00:00
break;
2007-10-10 21:34:04 +00:00
2007-12-09 22:45:43 +00:00
case astore_0:
frame->storeObjectOrAddress(0);
2007-12-09 22:45:43 +00:00
break;
2007-12-09 22:45:43 +00:00
case astore_1:
frame->storeObjectOrAddress(1);
2007-12-09 22:45:43 +00:00
break;
2007-10-10 21:34:04 +00:00
2007-12-09 22:45:43 +00:00
case astore_2:
frame->storeObjectOrAddress(2);
2007-12-09 22:45:43 +00:00
break;
2007-10-10 21:34:04 +00:00
2007-12-09 22:45:43 +00:00
case astore_3:
frame->storeObjectOrAddress(3);
2007-12-09 22:45:43 +00:00
break;
2007-10-10 21:34:04 +00:00
case athrow: {
Operand* e = frame->popObject();
2007-12-09 22:45:43 +00:00
c->indirectCallNoReturn
(c->constant(reinterpret_cast<intptr_t>(throw_)),
frame->trace(0, false),
2, c->thread(), e);
c->release(e);
} return;
2007-12-09 22:45:43 +00:00
case bipush:
frame->pushInt
(c->constant(static_cast<int8_t>(codeBody(t, code, ip++))));
break;
2007-12-09 22:45:43 +00:00
case checkcast: {
uint16_t index = codeReadInt16(t, code, ip);
2007-12-09 22:45:43 +00:00
object class_ = resolveClassInPool(t, codePool(t, code), index - 1);
if (UNLIKELY(t->exception)) return;
2007-12-09 22:45:43 +00:00
Operand* instance = frame->topObject();
2007-12-09 22:45:43 +00:00
Operand* classOperand = frame->append(class_);
2007-10-10 21:34:04 +00:00
c->indirectCall
(c->constant(reinterpret_cast<intptr_t>(checkCast)),
frame->trace(0, false),
3, c->thread(), classOperand, instance);
2007-12-09 22:45:43 +00:00
} break;
2007-10-10 21:34:04 +00:00
2007-12-09 22:45:43 +00:00
case d2f: {
Operand* a = frame->popLong();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(doubleToFloat)), 2, 0, a);
c->release(a);
Operand* result = c->result4();
frame->pushInt(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case d2i: {
Operand* a = frame->popLong();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(doubleToInt)), 2, 0, a);
c->release(a);
Operand* result = c->result4();
frame->pushInt(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case d2l: {
Operand* a = frame->popLong();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(doubleToLong)), 2, 0, a);
c->release(a);
Operand* result = c->result8();
frame->pushLong(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-10-10 21:34:04 +00:00
2007-12-09 22:45:43 +00:00
case dadd: {
Operand* a = frame->popLong();
Operand* b = frame->popLong();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(addDouble)), 4, 0, a, 0, b);
c->release(a);
c->release(b);
Operand* result = c->result8();
frame->pushLong(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-10-10 21:34:04 +00:00
2007-12-09 22:45:43 +00:00
case dcmpg: {
Operand* a = frame->popLong();
Operand* b = frame->popLong();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(compareDoublesG)),
4, 0, a, 0, b);
c->release(a);
c->release(b);
Operand* result = c->result4();
frame->pushInt(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-10-10 21:34:04 +00:00
2007-12-09 22:45:43 +00:00
case dcmpl: {
Operand* a = frame->popLong();
Operand* b = frame->popLong();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(compareDoublesL)),
4, 0, a, 0, b);
c->release(a);
c->release(b);
Operand* result = c->result4();
frame->pushInt(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case dconst_0:
frame->pushLong(c->constant(doubleToBits(0.0)));
2007-12-09 22:45:43 +00:00
break;
case dconst_1:
frame->pushLong(c->constant(doubleToBits(1.0)));
2007-12-09 22:45:43 +00:00
break;
2007-12-09 22:45:43 +00:00
case ddiv: {
Operand* a = frame->popLong();
Operand* b = frame->popLong();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(divideDouble)),
4, 0, a, 0, b);
c->release(a);
c->release(b);
Operand* result = c->result8();
frame->pushLong(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case dmul: {
Operand* a = frame->popLong();
Operand* b = frame->popLong();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(multiplyDouble)),
4, 0, a, 0, b);
c->release(a);
c->release(b);
Operand* result = c->result8();
frame->pushLong(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case dneg: {
Operand* a = frame->popLong();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(negateDouble)), 2, 0, a);
c->release(a);
Operand* result = c->result8();
frame->pushLong(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case vm::drem: {
Operand* a = frame->popLong();
Operand* b = frame->popLong();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(moduloDouble)), 4, 0, a, 0, b);
c->release(a);
c->release(b);
Operand* result = c->result8();
frame->pushLong(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case dsub: {
Operand* a = frame->popLong();
Operand* b = frame->popLong();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(subtractDouble)),
4, 0, a, 0, b);
c->release(a);
c->release(b);
Operand* result = c->result8();
frame->pushLong(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case dup:
frame->dup();
break;
2007-12-09 22:45:43 +00:00
case dup_x1:
frame->dupX1();
break;
2007-12-09 22:45:43 +00:00
case dup_x2:
frame->dupX2();
break;
2007-12-09 22:45:43 +00:00
case dup2:
frame->dup2();
break;
2007-12-09 22:45:43 +00:00
case dup2_x1:
frame->dup2X1();
break;
2007-12-09 22:45:43 +00:00
case dup2_x2:
frame->dup2X2();
break;
2007-10-09 17:15:40 +00:00
2007-12-09 22:45:43 +00:00
case f2d: {
Operand* a = frame->popInt();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(floatToDouble)), 1, a);
c->release(a);
Operand* result = c->result8();
frame->pushLong(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case f2i: {
Operand* a = frame->popInt();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(floatToInt)), 1, a);
c->release(a);
Operand* result = c->result4();
frame->pushInt(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case f2l: {
Operand* a = frame->popInt();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(floatToLong)), 1, a);
c->release(a);
Operand* result = c->result8();
frame->pushLong(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case fadd: {
Operand* a = frame->popInt();
Operand* b = frame->popInt();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(addFloat)), 2, a, b);
c->release(a);
c->release(b);
Operand* result = c->result4();
frame->pushInt(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case fcmpg: {
Operand* a = frame->popInt();
Operand* b = frame->popInt();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(compareFloatsG)), 2, a, b);
c->release(a);
c->release(b);
Operand* result = c->result4();
frame->pushInt(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case fcmpl: {
Operand* a = frame->popInt();
Operand* b = frame->popInt();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(compareFloatsL)), 2, a, b);
c->release(a);
c->release(b);
Operand* result = c->result4();
frame->pushInt(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case fconst_0:
frame->pushInt(c->constant(floatToBits(0.0)));
break;
case fconst_1:
frame->pushInt(c->constant(floatToBits(1.0)));
break;
case fconst_2:
frame->pushInt(c->constant(floatToBits(2.0)));
break;
2007-12-09 22:45:43 +00:00
case fdiv: {
Operand* a = frame->popInt();
Operand* b = frame->popInt();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(divideFloat)), 2, a, b);
c->release(a);
c->release(b);
Operand* result = c->result4();
frame->pushInt(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case fmul: {
Operand* a = frame->popInt();
Operand* b = frame->popInt();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(multiplyFloat)), 2, a, b);
c->release(a);
c->release(b);
Operand* result = c->result4();
frame->pushInt(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case fneg: {
Operand* a = frame->popInt();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(negateFloat)), 1, a);
c->release(a);
Operand* result = c->result4();
frame->pushInt(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case vm::frem: {
Operand* a = frame->popInt();
Operand* b = frame->popInt();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(moduloFloat)), 2, a, b);
c->release(a);
c->release(b);
Operand* result = c->result4();
frame->pushInt(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case fsub: {
Operand* a = frame->popInt();
Operand* b = frame->popInt();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(subtractFloat)), 2, a, b);
c->release(a);
c->release(b);
Operand* result = c->result4();
frame->pushInt(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case getfield:
case getstatic: {
uint16_t index = codeReadInt16(t, code, ip);
object field = resolveField(t, codePool(t, code), index - 1);
if (UNLIKELY(t->exception)) return;
2007-12-09 22:45:43 +00:00
Operand* table;
2007-12-09 22:45:43 +00:00
if (instruction == getstatic) {
2008-01-08 19:36:34 +00:00
c->indirectCall
(c->constant(reinterpret_cast<intptr_t>(tryInitClass)),
frame->trace(0, false),
2, c->thread(), frame->append(fieldClass(t, field)));
2007-12-09 22:45:43 +00:00
table = frame->append(classStaticTable(t, fieldClass(t, field)));
} else {
table = frame->popObject();
}
2007-12-09 22:45:43 +00:00
switch (fieldCode(t, field)) {
case ByteField:
case BooleanField:
2008-04-09 19:08:13 +00:00
frame->pushInt1(c->memory(table, fieldOffset(t, field), 0, 1));
2007-12-09 22:45:43 +00:00
break;
2007-12-09 22:45:43 +00:00
case CharField:
2008-04-09 19:08:13 +00:00
frame->pushInt2z(c->memory(table, fieldOffset(t, field), 0, 1));
2007-12-09 22:45:43 +00:00
break;
2007-12-09 22:45:43 +00:00
case ShortField:
2008-04-09 19:08:13 +00:00
frame->pushInt2(c->memory(table, fieldOffset(t, field), 0, 1));
2007-12-09 22:45:43 +00:00
break;
2007-12-09 22:45:43 +00:00
case FloatField:
case IntField:
2008-04-09 19:08:13 +00:00
frame->pushInt4(c->memory(table, fieldOffset(t, field), 0, 1));
2007-12-09 22:45:43 +00:00
break;
2007-12-09 22:45:43 +00:00
case DoubleField:
case LongField:
2008-04-09 19:08:13 +00:00
frame->pushLong(c->memory(table, fieldOffset(t, field), 0, 1));
2007-12-09 22:45:43 +00:00
break;
2007-12-09 22:45:43 +00:00
case ObjectField:
2008-04-09 19:08:13 +00:00
frame->pushObject(c->memory(table, fieldOffset(t, field), 0, 1));
2007-12-09 22:45:43 +00:00
break;
2007-12-09 22:45:43 +00:00
default:
abort(t);
}
if (instruction == getfield) {
c->release(table);
}
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case goto_: {
uint32_t newIp = (ip - 3) + codeReadInt16(t, code, ip);
assert(t, newIp < codeLength(t, code));
2007-12-16 00:24:15 +00:00
c->jmp(frame->machineIp(newIp));
2007-12-09 22:45:43 +00:00
ip = newIp;
} break;
2007-12-09 22:45:43 +00:00
case goto_w: {
uint32_t newIp = (ip - 5) + codeReadInt32(t, code, ip);
assert(t, newIp < codeLength(t, code));
2007-12-16 00:24:15 +00:00
c->jmp(frame->machineIp(newIp));
2007-12-09 22:45:43 +00:00
ip = newIp;
} break;
2007-12-09 22:45:43 +00:00
case i2b: {
2007-12-16 21:30:19 +00:00
Operand* top = frame->topInt();
c->mov1ToW(top, top);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case i2c: {
2007-12-16 21:30:19 +00:00
Operand* top = frame->topInt();
c->mov2zToW(top, top);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case i2d: {
Operand* a = frame->popInt();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(intToDouble)), 1, a);
Operand* result = c->result8();
frame->pushLong(result);
c->release(result);
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case i2f: {
Operand* a = frame->popInt();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(intToFloat)), 1, a);
Operand* result = c->result4();
frame->pushInt(result);
c->release(result);
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
case i2l:
frame->topIntToLong();
break;
2007-10-04 03:19:39 +00:00
2007-12-09 22:45:43 +00:00
case i2s: {
2007-12-16 21:30:19 +00:00
Operand* top = frame->topInt();
c->mov2ToW(top, top);
2007-12-09 22:45:43 +00:00
} break;
case iadd: {
Operand* a = frame->popInt();
c->add4(a, frame->topInt());
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
case iand: {
Operand* a = frame->popInt();
c->and4(a, frame->topInt());
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
2007-10-04 00:41:54 +00:00
2007-12-09 22:45:43 +00:00
case iconst_m1:
frame->pushInt(c->constant(-1));
break;
2007-10-04 00:41:54 +00:00
2007-12-09 22:45:43 +00:00
case iconst_0:
frame->pushInt(c->constant(0));
break;
2007-10-04 00:41:54 +00:00
2007-12-09 22:45:43 +00:00
case iconst_1:
frame->pushInt(c->constant(1));
break;
2007-12-09 22:45:43 +00:00
case iconst_2:
frame->pushInt(c->constant(2));
break;
2007-12-09 22:45:43 +00:00
case iconst_3:
frame->pushInt(c->constant(3));
break;
2007-09-30 02:48:27 +00:00
2007-12-09 22:45:43 +00:00
case iconst_4:
frame->pushInt(c->constant(4));
break;
2007-09-30 02:48:27 +00:00
2007-12-09 22:45:43 +00:00
case iconst_5:
frame->pushInt(c->constant(5));
break;
2007-12-09 22:45:43 +00:00
case idiv: {
Operand* a = frame->popInt();
c->div4(a, frame->topInt());
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case if_acmpeq:
case if_acmpne: {
uint32_t newIp = (ip - 3) + codeReadInt16(t, code, ip);
assert(t, newIp < codeLength(t, code));
Operand* a = frame->popObject();
Operand* b = frame->popObject();
cmp(c, a, b);
c->release(a);
c->release(b);
2007-12-16 00:24:15 +00:00
Operand* target = frame->machineIp(newIp);
2007-12-09 22:45:43 +00:00
if (instruction == if_acmpeq) {
c->je(target);
} else {
c->jne(target);
}
compile(t, frame, newIp);
if (UNLIKELY(t->exception)) return;
} break;
2007-12-09 22:45:43 +00:00
case if_icmpeq:
case if_icmpne:
case if_icmpgt:
case if_icmpge:
case if_icmplt:
case if_icmple: {
uint32_t newIp = (ip - 3) + codeReadInt16(t, code, ip);
assert(t, newIp < codeLength(t, code));
Operand* a = frame->popInt();
Operand* b = frame->popInt();
c->cmp4(a, b);
c->release(a);
c->release(b);
2007-12-16 00:24:15 +00:00
Operand* target = frame->machineIp(newIp);
2007-12-09 22:45:43 +00:00
switch (instruction) {
case if_icmpeq:
c->je(target);
break;
case if_icmpne:
c->jne(target);
break;
case if_icmpgt:
c->jg(target);
break;
case if_icmpge:
c->jge(target);
break;
case if_icmplt:
c->jl(target);
break;
case if_icmple:
c->jle(target);
break;
}
compile(t, frame, newIp);
if (UNLIKELY(t->exception)) return;
} break;
2007-10-03 00:22:48 +00:00
2007-12-09 22:45:43 +00:00
case ifeq:
case ifne:
case ifgt:
case ifge:
case iflt:
case ifle: {
uint32_t newIp = (ip - 3) + codeReadInt16(t, code, ip);
assert(t, newIp < codeLength(t, code));
2007-10-03 00:22:48 +00:00
Operand* a = frame->popInt();
c->cmp4(c->constant(0), a);
c->release(a);
2007-12-16 00:24:15 +00:00
Operand* target = frame->machineIp(newIp);
2007-12-09 22:45:43 +00:00
switch (instruction) {
case ifeq:
c->je(target);
break;
case ifne:
c->jne(target);
break;
case ifgt:
c->jg(target);
break;
case ifge:
c->jge(target);
break;
case iflt:
c->jl(target);
break;
case ifle:
c->jle(target);
break;
}
compile(t, frame, newIp);
if (UNLIKELY(t->exception)) return;
} break;
2007-12-09 22:45:43 +00:00
case ifnull:
case ifnonnull: {
uint32_t newIp = (ip - 3) + codeReadInt16(t, code, ip);
assert(t, newIp < codeLength(t, code));
2007-12-16 00:24:15 +00:00
Operand* a = frame->popObject();
cmp(c, c->constant(0), a);
c->release(a);
2007-09-30 02:48:27 +00:00
2007-12-16 00:24:15 +00:00
Operand* target = frame->machineIp(newIp);
2007-12-09 22:45:43 +00:00
if (instruction == ifnull) {
c->je(target);
} else {
c->jne(target);
}
compile(t, frame, newIp);
if (UNLIKELY(t->exception)) return;
} break;
2007-09-29 21:08:29 +00:00
2007-12-09 22:45:43 +00:00
case iinc: {
uint8_t index = codeBody(t, code, ip++);
int8_t count = codeBody(t, code, ip++);
2007-09-29 21:08:29 +00:00
2008-01-07 14:51:07 +00:00
c->add4(c->constant(count),
c->memory(c->base(), localOffset(t, index, context->method)));
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case iload:
case fload:
frame->loadInt(codeBody(t, code, ip++));
break;
2007-12-09 22:45:43 +00:00
case iload_0:
case fload_0:
frame->loadInt(0);
break;
2007-12-09 22:45:43 +00:00
case iload_1:
case fload_1:
frame->loadInt(1);
break;
2007-12-09 22:45:43 +00:00
case iload_2:
case fload_2:
frame->loadInt(2);
break;
2007-10-03 00:22:48 +00:00
2007-12-09 22:45:43 +00:00
case iload_3:
case fload_3:
frame->loadInt(3);
break;
2007-10-03 00:22:48 +00:00
2007-12-09 22:45:43 +00:00
case imul: {
Operand* a = frame->popInt();
c->mul4(a, frame->topInt());
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
case ineg: {
c->neg4(frame->topInt());
} break;
2007-12-09 22:45:43 +00:00
case instanceof: {
uint16_t index = codeReadInt16(t, code, ip);
2007-09-30 04:07:22 +00:00
2007-12-09 22:45:43 +00:00
object class_ = resolveClassInPool(t, codePool(t, code), index - 1);
if (UNLIKELY(t->exception)) return;
2007-09-30 04:07:22 +00:00
Operand* instance = frame->popObject();
2007-09-30 15:52:21 +00:00
2007-12-09 22:45:43 +00:00
Operand* classOperand = frame->append(class_);
2007-09-30 15:52:21 +00:00
c->directCall
(c->constant(reinterpret_cast<intptr_t>(instanceOf)),
3, c->thread(), classOperand, instance);
Operand* result = c->result4();
2007-12-09 22:45:43 +00:00
frame->pushInt(result);
c->release(result);
c->release(instance);
2007-12-09 22:45:43 +00:00
} break;
2007-09-30 02:48:27 +00:00
2007-12-09 22:45:43 +00:00
case invokeinterface: {
uint16_t index = codeReadInt16(t, code, ip);
ip += 2;
2007-09-30 02:48:27 +00:00
2007-12-09 22:45:43 +00:00
object target = resolveMethod(t, codePool(t, code), index - 1);
if (UNLIKELY(t->exception)) return;
2007-09-30 02:48:27 +00:00
2007-12-09 22:45:43 +00:00
unsigned parameterFootprint = methodParameterFootprint(t, target);
2007-09-30 02:48:27 +00:00
2007-12-09 22:45:43 +00:00
unsigned instance = parameterFootprint - 1;
2007-12-30 22:24:48 +00:00
c->indirectCall
(c->constant
(reinterpret_cast<intptr_t>(findInterfaceMethodFromInstance)),
frame->trace(0, false),
3, c->thread(), frame->append(target),
c->stack(frame->stack, instance));
Operand* result = ::result(c);
c->call(result, frame->trace(target, true));
c->release(result);
2007-09-30 15:52:21 +00:00
2007-12-09 22:45:43 +00:00
frame->pop(parameterFootprint);
pushReturnValue(t, frame, methodReturnCode(t, target));
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case invokespecial: {
uint16_t index = codeReadInt16(t, code, ip);
2007-12-09 22:45:43 +00:00
object target = resolveMethod(t, codePool(t, code), index - 1);
if (UNLIKELY(t->exception)) return;
object class_ = methodClass(t, context->method);
2007-12-09 22:45:43 +00:00
if (isSpecialMethod(t, target, class_)) {
target = findMethod(t, target, classSuper(t, class_));
}
2007-12-09 22:45:43 +00:00
compileDirectInvoke(t, frame, target);
} break;
2007-12-09 22:45:43 +00:00
case invokestatic: {
uint16_t index = codeReadInt16(t, code, ip);
2007-09-30 02:48:27 +00:00
2007-12-09 22:45:43 +00:00
object target = resolveMethod(t, codePool(t, code), index - 1);
if (UNLIKELY(t->exception)) return;
2007-09-29 21:08:29 +00:00
2007-12-09 22:45:43 +00:00
compileDirectInvoke(t, frame, target);
} break;
2007-09-29 21:08:29 +00:00
2007-12-09 22:45:43 +00:00
case invokevirtual: {
uint16_t index = codeReadInt16(t, code, ip);
2007-09-29 21:08:29 +00:00
2007-12-09 22:45:43 +00:00
object target = resolveMethod(t, codePool(t, code), index - 1);
if (UNLIKELY(t->exception)) return;
2007-09-29 21:08:29 +00:00
2007-12-09 22:45:43 +00:00
unsigned parameterFootprint = methodParameterFootprint(t, target);
2007-12-09 22:45:43 +00:00
unsigned offset = ClassVtable + (methodOffset(t, target) * BytesPerWord);
2007-09-29 21:08:29 +00:00
2007-12-16 00:24:15 +00:00
Operand* instance = c->stack(frame->stack, parameterFootprint - 1);
2007-12-09 22:45:43 +00:00
Operand* class_ = c->temporary();
2008-04-09 19:08:13 +00:00
mov(c, c->memory(instance, 0, 0, 1), class_);
and_(c, c->constant(PointerMask), class_);
2007-09-29 21:08:29 +00:00
c->call(c->memory(class_, offset, 0, 1), frame->trace(target, true));
2007-12-09 22:45:43 +00:00
c->release(class_);
2007-12-09 22:45:43 +00:00
frame->pop(parameterFootprint);
pushReturnValue(t, frame, methodReturnCode(t, target));
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case ior: {
Operand* a = frame->popInt();
c->or4(a, frame->topInt());
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case irem: {
Operand* a = frame->popInt();
c->rem4(a, frame->topInt());
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case ireturn:
case freturn: {
handleExit(t, frame);
Operand* a = frame->popInt();
c->return4(a);
c->release(a);
} return;
2007-12-09 22:45:43 +00:00
case ishl: {
Operand* a = frame->popInt();
c->shl4(a, frame->topInt());
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case ishr: {
Operand* a = frame->popInt();
c->shr4(a, frame->topInt());
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case istore:
case fstore:
frame->storeInt(codeBody(t, code, ip++));
break;
2007-12-09 22:45:43 +00:00
case istore_0:
case fstore_0:
frame->storeInt(0);
break;
2007-12-09 22:45:43 +00:00
case istore_1:
case fstore_1:
frame->storeInt(1);
break;
2007-12-09 22:45:43 +00:00
case istore_2:
case fstore_2:
frame->storeInt(2);
break;
2007-12-09 22:45:43 +00:00
case istore_3:
case fstore_3:
frame->storeInt(3);
break;
2007-12-09 22:45:43 +00:00
case isub: {
Operand* a = frame->popInt();
c->sub4(a, frame->topInt());
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case iushr: {
Operand* a = frame->popInt();
c->ushr4(a, frame->topInt());
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case ixor: {
Operand* a = frame->popInt();
c->xor4(a, frame->topInt());
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case jsr:
case jsr_w: {
uint32_t newIp;
if (instruction == jsr) {
newIp = (ip - 3) + codeReadInt16(t, code, ip);
} else {
newIp = (ip - 5) + codeReadInt32(t, code, ip);
}
assert(t, newIp < codeLength(t, code));
frame->pushAddress(frame->machineIp(ip));
c->jmp(frame->machineIp(newIp));
// NB: we assume that the stack will look the same on return
// from the subroutine as at call time.
compile(t, frame, newIp);
if (UNLIKELY(t->exception)) return;
frame->pop(1);
2007-12-16 21:30:19 +00:00
} break;
2008-03-21 00:37:58 +00:00
case l2d: {
Operand* a = frame->popLong();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(longToDouble)), 2, 0, a);
Operand* result = c->result8();
frame->pushLong(result);
c->release(result);
c->release(a);
} break;
case l2f: {
Operand* a = frame->popLong();
c->directCall
(c->constant(reinterpret_cast<intptr_t>(longToDouble)), 2, 0, a);
Operand* result = c->result4();
frame->pushInt(result);
c->release(result);
c->release(a);
} break;
case l2i:
frame->topLongToInt();
break;
2007-12-09 22:45:43 +00:00
case ladd: {
Operand* a = frame->popLong();
c->add8(a, frame->topLong());
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
case land: {
Operand* a = frame->popLong();
c->and8(a, frame->topLong());
c->release(a);
} break;
2007-12-09 22:45:43 +00:00
case lcmp: {
Operand* next = c->label();
Operand* less = c->label();
Operand* greater = c->label();
2007-12-09 22:45:43 +00:00
Operand* a = frame->popLong();
Operand* b = frame->popLong();
Operand* result = c->temporary();
c->cmp8(a, b);
c->release(a);
c->release(b);
2007-12-09 22:45:43 +00:00
c->jl(less);
c->jg(greater);
c->mov4(c->constant(0), result);
2007-12-09 22:45:43 +00:00
c->jmp(next);
c->mark(less);
c->mov4(c->constant(-1), result);
2007-12-09 22:45:43 +00:00
c->jmp(next);
2007-12-09 22:45:43 +00:00
c->mark(greater);
c->mov4(c->constant(1), result);
2007-12-09 22:45:43 +00:00
c->mark(next);
frame->pushInt(result);
2007-12-09 22:45:43 +00:00
c->release(result);
} break;
2007-12-09 22:45:43 +00:00
case lconst_0:
frame->pushLong(c->constant(0));
2007-12-09 22:45:43 +00:00
break;
2007-12-09 22:45:43 +00:00
case lconst_1:
frame->pushLong(c->constant(1));
2007-12-09 22:45:43 +00:00
break;
2007-09-30 02:48:27 +00:00
2007-12-09 22:45:43 +00:00
case ldc:
case ldc_w: {
uint16_t index;
2007-09-30 02:48:27 +00:00
2007-12-09 22:45:43 +00:00
if (instruction == ldc) {
index = codeBody(t, code, ip++);
} else {
index = codeReadInt16(t, code, ip);
}
2007-09-30 15:52:21 +00:00
2007-12-09 22:45:43 +00:00
object pool = codePool(t, code);
2007-09-30 15:52:21 +00:00
2007-12-09 22:45:43 +00:00
if (singletonIsObject(t, pool, index - 1)) {
object v = singletonObject(t, pool, index - 1);
if (objectClass(t, v)
== arrayBody(t, t->m->types, Machine::ByteArrayType))
{
object class_ = resolveClassInPool(t, pool, index - 1);
if (UNLIKELY(t->exception)) return;
2007-09-30 15:52:21 +00:00
2007-12-09 22:45:43 +00:00
frame->pushObject(frame->append(class_));
} else {
2007-12-09 22:45:43 +00:00
frame->pushObject(frame->append(v));
}
2007-12-09 22:45:43 +00:00
} else {
frame->pushInt(c->constant(singletonValue(t, pool, index - 1)));
}
} break;
2007-10-04 22:41:19 +00:00
2007-12-09 22:45:43 +00:00
case ldc2_w: {
uint16_t index = codeReadInt16(t, code, ip);
2007-12-09 22:45:43 +00:00
object pool = codePool(t, code);
2007-12-09 22:45:43 +00:00
uint64_t v;
memcpy(&v, &singletonValue(t, pool, index - 1), 8);
frame->pushLong(c->constant(v));
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case ldiv_: {
Operand* a = frame->popLong();
c->div8(a, frame->topLong());
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case lload:
case dload:
frame->loadLong(codeBody(t, code, ip++));
break;
2007-12-09 22:45:43 +00:00
case lload_0:
case dload_0:
frame->loadLong(0);
break;
2007-12-09 22:45:43 +00:00
case lload_1:
case dload_1:
frame->loadLong(1);
break;
2007-12-09 22:45:43 +00:00
case lload_2:
case dload_2:
frame->loadLong(2);
break;
2007-12-09 22:45:43 +00:00
case lload_3:
case dload_3:
frame->loadLong(3);
break;
2007-12-09 22:45:43 +00:00
case lmul: {
Operand* a = frame->popLong();
c->mul8(a, frame->topLong());
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case lneg:
c->neg8(frame->topLong());
2007-12-09 22:45:43 +00:00
break;
2007-12-09 22:45:43 +00:00
case lookupswitch: {
int32_t base = ip - 1;
2007-12-09 22:45:43 +00:00
ip = (ip + 3) & ~3; // pad to four byte boundary
Operand* key = frame->popInt4();
2007-12-09 22:45:43 +00:00
uint32_t defaultIp = base + codeReadInt32(t, code, ip);
assert(t, defaultIp < codeLength(t, code));
2007-12-16 00:24:15 +00:00
Operand* default_ = c->absolute
(c->poolAppendPromise(c->machineIp(defaultIp)));
2007-12-09 22:45:43 +00:00
int32_t pairCount = codeReadInt32(t, code, ip);
2007-12-16 00:24:15 +00:00
Operand* start = 0;
uint32_t ipTable[pairCount];
2007-12-09 22:45:43 +00:00
for (int32_t i = 0; i < pairCount; ++i) {
unsigned index = ip + (i * 8);
int32_t key = codeReadInt32(t, code, index);
uint32_t newIp = base + codeReadInt32(t, code, index);
assert(t, newIp < codeLength(t, code));
2007-12-16 00:24:15 +00:00
ipTable[i] = newIp;
2007-12-16 00:24:15 +00:00
Promise* p = c->poolAppend(key);
2007-12-09 22:45:43 +00:00
if (i == 0) {
2007-12-16 00:24:15 +00:00
start = c->promiseConstant(p);
2007-12-09 22:45:43 +00:00
}
2007-12-16 00:24:15 +00:00
c->poolAppendPromise(c->machineIp(newIp));
2007-12-09 22:45:43 +00:00
}
2007-12-16 00:24:15 +00:00
assert(t, start);
c->directCall
(c->constant(reinterpret_cast<intptr_t>(lookUpAddress)),
4, key, start, c->constant(pairCount), default_);
Operand* result = ::result(c);
c->jmp(result);
c->release(result);
c->release(key);
2007-12-16 00:24:15 +00:00
for (int32_t i = 0; i < pairCount; ++i) {
compile(t, frame, ipTable[i]);
if (UNLIKELY(t->exception)) return;
}
2008-01-07 14:51:07 +00:00
ip = defaultIp;
} break;
2007-12-09 22:45:43 +00:00
case lor: {
Operand* a = frame->popLong();
c->or8(a, frame->topLong());
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case lrem: {
Operand* a = frame->popLong();
c->rem8(a, frame->topLong());
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case lreturn:
case dreturn: {
handleExit(t, frame);
Operand* a = frame->popLong();
c->return8(a);
c->release(a);
} return;
2007-12-09 22:45:43 +00:00
case lshl: {
Operand* a = frame->popInt();
c->shl8(a, frame->topLong());
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case lshr: {
Operand* a = frame->popInt();
c->shr8(a, frame->topLong());
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case lstore:
case dstore:
frame->storeLong(codeBody(t, code, ip++));
break;
2007-12-09 22:45:43 +00:00
case lstore_0:
case dstore_0:
frame->storeLong(0);
break;
2007-12-09 22:45:43 +00:00
case lstore_1:
case dstore_1:
frame->storeLong(1);
break;
case lstore_2:
case dstore_2:
frame->storeLong(2);
break;
case lstore_3:
case dstore_3:
frame->storeLong(3);
break;
2007-12-09 22:45:43 +00:00
case lsub: {
Operand* a = frame->popLong();
c->sub8(a, frame->topLong());
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case lushr: {
Operand* a = frame->popInt();
c->ushr8(a, frame->topLong());
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case lxor: {
Operand* a = frame->popLong();
c->xor8(a, frame->topLong());
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case monitorenter: {
Operand* a = frame->popObject();
c->indirectCall
2007-12-09 22:45:43 +00:00
(c->constant(reinterpret_cast<intptr_t>(acquireMonitorForObject)),
frame->trace(0, false),
2, c->thread(), a);
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
2007-09-29 21:08:29 +00:00
2007-12-09 22:45:43 +00:00
case monitorexit: {
Operand* a = frame->popObject();
c->indirectCall
2007-12-09 22:45:43 +00:00
(c->constant(reinterpret_cast<intptr_t>(releaseMonitorForObject)),
frame->trace(0, false),
2, c->thread(), a);
c->release(a);
2007-12-09 22:45:43 +00:00
} break;
2007-09-29 21:08:29 +00:00
2007-12-09 22:45:43 +00:00
case multianewarray: {
uint16_t index = codeReadInt16(t, code, ip);
uint8_t dimensions = codeBody(t, code, ip++);
2007-12-09 22:45:43 +00:00
object class_ = resolveClassInPool(t, codePool(t, code), index - 1);
if (UNLIKELY(t->exception)) return;
PROTECT(t, class_);
Operand* stack = c->temporary();
mov(c, c->stack(), stack);
c->indirectCall
2007-12-09 22:45:43 +00:00
(c->constant(reinterpret_cast<intptr_t>(makeMultidimensionalArray)),
frame->trace(0, false),
4, c->thread(), frame->append(class_), stack,
2007-12-09 22:45:43 +00:00
c->constant(dimensions));
c->release(stack);
2007-09-29 21:08:29 +00:00
Operand* result = ::result(c);
2007-12-09 22:45:43 +00:00
frame->pop(dimensions);
frame->pushObject(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-10-12 00:30:46 +00:00
2007-12-09 22:45:43 +00:00
case new_: {
uint16_t index = codeReadInt16(t, code, ip);
object class_ = resolveClassInPool(t, codePool(t, code), index - 1);
if (UNLIKELY(t->exception)) return;
if (classVmFlags(t, class_) & WeakReferenceFlag) {
c->indirectCall
2007-12-09 22:45:43 +00:00
(c->constant(reinterpret_cast<intptr_t>(makeNewWeakReference)),
frame->trace(0, false),
2007-12-09 22:45:43 +00:00
2, c->thread(), frame->append(class_));
} else {
c->indirectCall
2007-12-09 22:45:43 +00:00
(c->constant(reinterpret_cast<intptr_t>(makeNew)),
frame->trace(0, false),
2007-12-09 22:45:43 +00:00
2, c->thread(), frame->append(class_));
}
2007-09-29 21:08:29 +00:00
Operand* result = ::result(c);
2007-12-09 22:45:43 +00:00
frame->pushObject(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case newarray: {
uint8_t type = codeBody(t, code, ip++);
2007-12-09 22:45:43 +00:00
Operand* nonnegative = c->label();
Operand* length = frame->popInt4();
c->cmp4(c->constant(0), length);
2007-12-09 22:45:43 +00:00
c->jge(nonnegative);
c->indirectCallNoReturn
(c->constant(reinterpret_cast<intptr_t>(throwNegativeArraySize)),
frame->trace(0, false),
2, c->thread(), length);
2007-12-09 22:45:43 +00:00
c->mark(nonnegative);
2007-12-09 22:45:43 +00:00
object (*constructor)(Thread*, uintptr_t, bool);
switch (type) {
case T_BOOLEAN:
constructor = makeBooleanArray;
break;
2007-12-09 22:45:43 +00:00
case T_CHAR:
constructor = makeCharArray;
break;
2007-12-09 22:45:43 +00:00
case T_FLOAT:
constructor = makeFloatArray;
break;
2007-12-09 22:45:43 +00:00
case T_DOUBLE:
constructor = makeDoubleArray;
break;
2007-12-09 22:45:43 +00:00
case T_BYTE:
constructor = makeByteArray;
break;
2007-12-09 22:45:43 +00:00
case T_SHORT:
constructor = makeShortArray;
break;
2007-12-09 22:45:43 +00:00
case T_INT:
constructor = makeIntArray;
break;
2007-12-09 22:45:43 +00:00
case T_LONG:
constructor = makeLongArray;
break;
2007-12-09 22:45:43 +00:00
default: abort(t);
}
c->indirectCall
2007-12-09 22:45:43 +00:00
(c->constant(reinterpret_cast<intptr_t>(makeBlankArray)),
frame->trace(0, false),
2007-12-16 00:24:15 +00:00
3, c->thread(), c->constant(reinterpret_cast<intptr_t>(constructor)),
length);
Operand* result = ::result(c);
2007-12-16 00:24:15 +00:00
c->release(length);
2007-12-09 22:45:43 +00:00
frame->pushObject(result);
c->release(result);
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case nop: break;
2007-12-09 22:45:43 +00:00
case pop_:
frame->pop(1);
break;
2007-12-09 22:45:43 +00:00
case pop2:
frame->pop(2);
break;
2007-12-09 22:45:43 +00:00
case putfield:
case putstatic: {
uint16_t index = codeReadInt16(t, code, ip);
object field = resolveField(t, codePool(t, code), index - 1);
if (UNLIKELY(t->exception)) return;
object staticTable = 0;
2007-12-09 22:45:43 +00:00
if (instruction == putstatic) {
2008-01-08 19:36:34 +00:00
c->indirectCall
(c->constant(reinterpret_cast<intptr_t>(tryInitClass)),
frame->trace(0, false),
2, c->thread(), frame->append(fieldClass(t, field)));
2007-12-09 22:45:43 +00:00
staticTable = classStaticTable(t, fieldClass(t, field));
}
2007-10-12 22:06:33 +00:00
2007-12-09 22:45:43 +00:00
Operand* value;
switch (fieldCode(t, field)) {
case ByteField:
case BooleanField:
case CharField:
case ShortField:
case FloatField:
case IntField: {
value = frame->popInt();
} break;
2007-12-09 22:45:43 +00:00
case DoubleField:
case LongField: {
value = frame->popLong();
} break;
2007-12-09 22:45:43 +00:00
case ObjectField: {
value = frame->popObject();
} break;
2007-12-09 22:45:43 +00:00
default: abort(t);
}
2007-12-09 22:45:43 +00:00
Operand* table;
2007-12-09 22:45:43 +00:00
if (instruction == putstatic) {
table = frame->append(staticTable);
} else {
table = frame->popObject();
}
2007-12-09 22:45:43 +00:00
switch (fieldCode(t, field)) {
case ByteField:
case BooleanField:
2008-04-09 19:08:13 +00:00
c->mov1(value, c->memory(table, fieldOffset(t, field), 0, 1));
break;
2007-12-09 22:45:43 +00:00
case CharField:
case ShortField:
2008-04-09 19:08:13 +00:00
c->mov2(value, c->memory(table, fieldOffset(t, field), 0, 1));
break;
2007-12-09 22:45:43 +00:00
case FloatField:
case IntField:
2008-04-09 19:08:13 +00:00
c->mov4(value, c->memory(table, fieldOffset(t, field), 0, 1));
break;
2007-12-09 22:45:43 +00:00
case DoubleField:
case LongField:
2008-04-09 19:08:13 +00:00
c->mov8(value, c->memory(table, fieldOffset(t, field), 0, 1));
break;
2007-12-09 22:45:43 +00:00
case ObjectField:
if (instruction == putfield) {
2007-12-30 22:24:48 +00:00
c->indirectCall
(c->constant(reinterpret_cast<intptr_t>(setMaybeNull)),
frame->trace(0, false),
2007-12-30 22:24:48 +00:00
4, c->thread(), table, c->constant(fieldOffset(t, field)), value);
} else {
c->directCall
(c->constant(reinterpret_cast<intptr_t>(set)),
4, c->thread(), table, c->constant(fieldOffset(t, field)), value);
}
break;
2007-12-09 22:45:43 +00:00
default: abort(t);
}
if (instruction == putfield) {
c->release(table);
}
c->release(value);
2007-12-09 22:45:43 +00:00
} break;
case ret:
c->jmp
(c->memory
(c->base(), localOffset
(t, codeBody(t, code, ip), context->method)));
return;
2007-12-09 22:45:43 +00:00
case return_:
handleExit(t, frame);
2007-12-09 22:45:43 +00:00
c->epilogue();
c->ret();
return;
2007-12-09 22:45:43 +00:00
case sipush:
frame->pushInt
(c->constant(static_cast<int16_t>(codeReadInt16(t, code, ip))));
break;
2007-12-09 22:45:43 +00:00
case swap:
frame->swap();
break;
2007-12-09 22:45:43 +00:00
case tableswitch: {
int32_t base = ip - 1;
2007-12-09 22:45:43 +00:00
ip = (ip + 3) & ~3; // pad to four byte boundary
2007-09-30 02:48:27 +00:00
Operand* key = frame->popInt4();
2007-12-09 22:45:43 +00:00
uint32_t defaultIp = base + codeReadInt32(t, code, ip);
assert(t, defaultIp < codeLength(t, code));
int32_t bottom = codeReadInt32(t, code, ip);
int32_t top = codeReadInt32(t, code, ip);
2007-12-16 00:24:15 +00:00
Operand* start = 0;
uint32_t ipTable[top - bottom + 1];
for (int32_t i = 0; i < top - bottom + 1; ++i) {
2007-12-09 22:45:43 +00:00
unsigned index = ip + (i * 4);
uint32_t newIp = base + codeReadInt32(t, code, index);
assert(t, newIp < codeLength(t, code));
2007-12-16 00:24:15 +00:00
ipTable[i] = newIp;
Promise* p = c->poolAppendPromise(c->machineIp(newIp));
2007-12-09 22:45:43 +00:00
if (i == 0) {
2007-12-16 00:24:15 +00:00
start = c->promiseConstant(p);
}
2007-12-09 22:45:43 +00:00
}
2007-12-16 00:24:15 +00:00
assert(t, start);
2007-12-09 22:45:43 +00:00
Operand* defaultCase = c->label();
c->cmp4(c->constant(bottom), key);
2007-12-09 22:45:43 +00:00
c->jl(defaultCase);
2007-10-04 22:41:19 +00:00
c->cmp4(c->constant(top), key);
2007-12-09 22:45:43 +00:00
c->jg(defaultCase);
2007-10-04 22:41:19 +00:00
c->sub4(c->constant(bottom), key);
2007-12-09 22:45:43 +00:00
c->jmp(c->memory(start, 0, key, BytesPerWord));
c->mark(defaultCase);
2007-12-16 00:24:15 +00:00
c->jmp(frame->machineIp(defaultIp));
c->release(key);
2007-12-16 00:24:15 +00:00
for (int32_t i = 0; i < top - bottom + 1; ++i) {
compile(t, frame, ipTable[i]);
if (UNLIKELY(t->exception)) return;
}
2008-01-07 14:51:07 +00:00
ip = defaultIp;
} break;
2007-12-09 22:45:43 +00:00
case wide: {
switch (codeBody(t, code, ip++)) {
case aload: {
frame->loadObject(codeReadInt16(t, code, ip));
2007-10-04 22:41:19 +00:00
} break;
2007-12-09 22:45:43 +00:00
case astore: {
frame->storeObject(codeReadInt16(t, code, ip));
} break;
2007-10-04 22:41:19 +00:00
2007-12-09 22:45:43 +00:00
case iinc: {
uint16_t index = codeReadInt16(t, code, ip);
uint16_t count = codeReadInt16(t, code, ip);
2007-10-04 22:41:19 +00:00
2008-01-07 14:51:07 +00:00
c->add4(c->constant(count),
c->memory(c->base(), localOffset(t, index, context->method)));
2007-12-09 22:45:43 +00:00
} break;
2007-10-04 22:41:19 +00:00
2007-12-09 22:45:43 +00:00
case iload: {
frame->loadInt(codeReadInt16(t, code, ip));
} break;
2007-10-04 22:41:19 +00:00
2007-12-09 22:45:43 +00:00
case istore: {
frame->storeInt(codeReadInt16(t, code, ip));
} break;
2007-10-04 22:41:19 +00:00
2007-12-09 22:45:43 +00:00
case lload: {
frame->loadLong(codeReadInt16(t, code, ip));
} break;
2007-10-04 22:41:19 +00:00
2007-12-09 22:45:43 +00:00
case lstore: {
frame->storeLong(codeReadInt16(t, code, ip));
} break;
2007-10-12 14:26:36 +00:00
2007-12-09 22:45:43 +00:00
case ret:
c->jmp
(c->memory
(c->base(), localOffset
(t, codeReadInt16(t, code, ip), context->method)));
return;
2007-10-04 22:41:19 +00:00
2007-12-09 22:45:43 +00:00
default: abort(t);
}
} break;
default: abort(t);
2007-12-09 22:45:43 +00:00
}
}
}
2007-10-04 22:41:19 +00:00
void
logCompile(const void* code, unsigned size, const char* class_,
const char* name, const char* spec)
{
fprintf(stderr, "%s.%s%s from %p to %p\n",
class_, name, spec, code,
static_cast<const uint8_t*>(code) + size);
}
2008-01-07 14:51:07 +00:00
void
updateExceptionHandlerTable(MyThread* t, Compiler* c, object code,
intptr_t start)
2008-01-07 14:51:07 +00:00
{
object oldTable = codeExceptionHandlerTable(t, code);
if (oldTable) {
PROTECT(t, code);
PROTECT(t, oldTable);
unsigned length = exceptionHandlerTableLength(t, oldTable);
object newTable = makeExceptionHandlerTable(t, length, false);
for (unsigned i = 0; i < length; ++i) {
ExceptionHandler* oldHandler = exceptionHandlerTableBody
(t, oldTable, i);
ExceptionHandler* newHandler = exceptionHandlerTableBody
(t, newTable, i);
exceptionHandlerStart(newHandler)
= c->machineIp(exceptionHandlerStart(oldHandler))->value(c) - start;
2008-01-07 14:51:07 +00:00
exceptionHandlerEnd(newHandler)
= c->machineIp(exceptionHandlerEnd(oldHandler))->value(c) - start;
2008-01-07 14:51:07 +00:00
exceptionHandlerIp(newHandler)
= c->machineIp(exceptionHandlerIp(oldHandler))->value(c) - start;
2008-01-07 14:51:07 +00:00
exceptionHandlerCatchType(newHandler)
= exceptionHandlerCatchType(oldHandler);
}
set(t, code, CodeExceptionHandlerTable, newTable);
}
}
void
updateLineNumberTable(MyThread* t, Compiler* c, object code,
intptr_t start)
2008-01-07 14:51:07 +00:00
{
object oldTable = codeLineNumberTable(t, code);
if (oldTable) {
PROTECT(t, code);
PROTECT(t, oldTable);
unsigned length = lineNumberTableLength(t, oldTable);
object newTable = makeLineNumberTable(t, length, false);
for (unsigned i = 0; i < length; ++i) {
LineNumber* oldLine = lineNumberTableBody(t, oldTable, i);
LineNumber* newLine = lineNumberTableBody(t, newTable, i);
lineNumberIp(newLine)
= c->machineIp(lineNumberIp(oldLine))->value(c) - start;
2008-01-07 14:51:07 +00:00
lineNumberLine(newLine) = lineNumberLine(oldLine);
}
set(t, code, CodeLineNumberTable, newTable);
}
}
void
printSet(uintptr_t m)
{
for (unsigned i = 0; i < 16; ++i) {
if ((m >> i) & 1) {
fprintf(stderr, "1");
} else {
fprintf(stderr, "_");
}
}
}
unsigned
calculateFrameMaps(MyThread* t, Context* context, uintptr_t* originalRoots,
unsigned eventIndex)
2008-01-07 14:51:07 +00:00
{
// for each instruction with more than one predecessor, and for each
// stack position, determine if there exists a path to that
// instruction such that there is not an object pointer left at that
// stack position (i.e. it is uninitialized or contains primitive
// data).
2008-01-07 14:51:07 +00:00
unsigned mapSize = frameMapSizeInWords(t, context->method);
uintptr_t roots[mapSize];
if (originalRoots) {
memcpy(roots, originalRoots, mapSize * BytesPerWord);
} else {
memset(roots, 0, mapSize * BytesPerWord);
}
2008-01-07 14:51:07 +00:00
int32_t ip = -1;
// invariant: for each stack position, roots contains a zero at that
// position if there exists some path to the current instruction
// such that there is definitely not an object pointer at that
// position. Otherwise, roots contains a one at that position,
// meaning either all known paths result in an object pointer at
// that position, or the contents of that position are as yet
// unknown.
unsigned length = context->eventLog.length();
while (eventIndex < length) {
Event e = static_cast<Event>(context->eventLog.get(eventIndex++));
2008-01-07 14:51:07 +00:00
switch (e) {
case PushEvent: {
eventIndex = calculateFrameMaps(t, context, roots, eventIndex);
2008-01-07 14:51:07 +00:00
} break;
case PopEvent:
return eventIndex;
2008-01-07 14:51:07 +00:00
case IpEvent: {
ip = context->eventLog.get2(eventIndex);
eventIndex += 2;
2008-01-07 14:51:07 +00:00
if (DebugFrameMaps) {
fprintf(stderr, " roots at ip %3d: ", ip);
printSet(*roots);
fprintf(stderr, "\n");
}
uintptr_t* tableRoots = context->rootTable + (ip * mapSize);
2008-01-07 14:51:07 +00:00
if (context->visitTable[ip] > 1) {
for (unsigned wi = 0; wi < mapSize; ++wi) {
uintptr_t newRoots = tableRoots[wi] & roots[wi];
if ((eventIndex == length
or context->eventLog.get(eventIndex) == PopEvent)
and newRoots != tableRoots[wi])
{
if (DebugFrameMaps) {
fprintf(stderr, "dirty roots!\n");
}
context->dirtyRoots = true;
}
tableRoots[wi] = newRoots;
roots[wi] &= tableRoots[wi];
2008-01-07 14:51:07 +00:00
}
if (DebugFrameMaps) {
fprintf(stderr, "table roots at ip %3d: ", ip);
printSet(*tableRoots);
fprintf(stderr, "\n");
}
} else {
memcpy(tableRoots, roots, mapSize * BytesPerWord);
2008-01-07 14:51:07 +00:00
}
} break;
case MarkEvent: {
unsigned i = context->eventLog.get2(eventIndex);
eventIndex += 2;
2008-01-07 14:51:07 +00:00
markBit(roots, i);
} break;
case ClearEvent: {
unsigned i = context->eventLog.get2(eventIndex);
eventIndex += 2;
clearBit(roots, i);
2008-01-07 14:51:07 +00:00
} break;
case TraceEvent: {
TraceElement* te; context->eventLog.get(eventIndex, &te, BytesPerWord);
memcpy(te->map, roots, mapSize * BytesPerWord);
eventIndex += BytesPerWord;
2008-01-07 14:51:07 +00:00
} break;
default: abort(t);
}
}
return eventIndex;
2008-01-07 14:51:07 +00:00
}
Allocator*
codeAllocator(MyThread* t);
int
compareTraceElementPointers(const void* va, const void* vb)
{
TraceElement* a = *static_cast<TraceElement* const*>(va);
TraceElement* b = *static_cast<TraceElement* const*>(vb);
if (a->addressValue > b->addressValue) {
return 1;
} else if (a->addressValue < b->addressValue) {
return -1;
} else {
return 0;
}
}
intptr_t
compareMethodBounds(Thread* t, object a, object b)
{
if (DebugMethodTree) {
fprintf(stderr, "compare %p to %p\n",
&singletonValue(t, methodCompiled(t, a), 0),
&singletonValue(t, methodCompiled(t, b), 0));
}
return reinterpret_cast<intptr_t>
(&singletonValue(t, methodCompiled(t, a), 0))
- reinterpret_cast<intptr_t>
(&singletonValue(t, methodCompiled(t, b), 0));
}
2007-12-09 22:45:43 +00:00
object
finish(MyThread* t, Context* context, const char* name)
2007-12-09 22:45:43 +00:00
{
Compiler* c = context->c;
2007-12-18 02:09:32 +00:00
2007-12-11 23:52:28 +00:00
unsigned count = ceiling(c->codeSize() + c->poolSize(), BytesPerWord);
2007-12-09 22:45:43 +00:00
unsigned size = count + singletonMaskSize(count);
object result = allocate3
(t, codeAllocator(t), Machine::ImmortalAllocation,
SingletonBody + (size * BytesPerWord), true, true);
initSingleton(t, result, size, true);
mark(t, result, 0);
2007-12-09 22:45:43 +00:00
singletonMask(t, result)[0] = 1;
2007-10-04 22:41:19 +00:00
2007-12-09 22:45:43 +00:00
uint8_t* start = reinterpret_cast<uint8_t*>(&singletonValue(t, result, 0));
2007-10-04 22:41:19 +00:00
2007-12-09 22:45:43 +00:00
c->writeTo(start);
2007-10-04 22:41:19 +00:00
if (context->method) {
2007-12-16 22:41:07 +00:00
PROTECT(t, result);
2007-10-12 14:26:36 +00:00
{ object code = methodCode(t, context->method);
code = makeCode(t, 0,
codeExceptionHandlerTable(t, code),
codeLineNumberTable(t, code),
codeMaxStack(t, code),
codeMaxLocals(t, code),
0, false);
set(t, context->method, MethodCode, code);
}
if (context->traceLogCount) {
TraceElement* elements[context->traceLogCount];
unsigned index = 0;
for (TraceElement* p = context->traceLog; p; p = p->next) {
elements[index++] = p;
p->addressValue = p->address->value(c);
if (p->target) {
insertCallNode
(t, makeCallNode
(t, p->address->value(c), p->target, p->virtualCall, 0));
}
2007-12-09 22:45:43 +00:00
}
qsort(elements, context->traceLogCount, sizeof(TraceElement*),
compareTraceElementPointers);
unsigned size = frameSize(t, context->method);
object map = makeIntArray
(t, context->traceLogCount
+ ceiling(context->traceLogCount * size, 32),
false);
for (unsigned i = 0; i < context->traceLogCount; ++i) {
TraceElement* p = elements[i];
intArrayBody(t, map, i) = static_cast<intptr_t>(p->addressValue)
- reinterpret_cast<intptr_t>(start);
for (unsigned j = 0; j < size; ++j) {
unsigned index = ((i * size) + j);
int32_t* v = &intArrayBody
(t, map, context->traceLogCount + (index / 32));
if (getBit(p->map, j)) {
*v |= static_cast<int32_t>(1) << (index % 32);
} else {
*v &= ~(static_cast<int32_t>(1) << (index % 32));
}
}
}
set(t, methodCode(t, context->method), CodePool, map);
2007-12-09 22:45:43 +00:00
}
for (PoolElement* p = context->objectPool; p; p = p->next) {
intptr_t offset = p->address->value(c)
- reinterpret_cast<intptr_t>(start);
singletonMarkObject(t, result, offset / BytesPerWord);
set(t, result, SingletonBody + offset, p->value);
}
updateExceptionHandlerTable(t, c, methodCode(t, context->method),
reinterpret_cast<intptr_t>(start));
updateLineNumberTable(t, c, methodCode(t, context->method),
reinterpret_cast<intptr_t>(start));
2007-12-11 23:52:28 +00:00
if (Verbose) {
logCompile
(start, c->codeSize(),
reinterpret_cast<const char*>
(&byteArrayBody(t, className(t, methodClass(t, context->method)), 0)),
reinterpret_cast<const char*>
(&byteArrayBody(t, methodName(t, context->method), 0)),
reinterpret_cast<const char*>
(&byteArrayBody(t, methodSpec(t, context->method), 0)));
2007-12-11 23:52:28 +00:00
}
2007-12-13 00:18:31 +00:00
// for debugging:
if (false and
strcmp
(reinterpret_cast<const char*>
(&byteArrayBody(t, className(t, methodClass(t, context->method)), 0)),
"java/lang/System") == 0 and
strcmp
(reinterpret_cast<const char*>
(&byteArrayBody(t, methodName(t, context->method), 0)),
"getProperty") == 0)
2007-12-13 00:18:31 +00:00
{
asm("int3");
}
} else {
if (Verbose) {
logCompile(start, c->codeSize(), 0, name, 0);
}
2007-12-09 22:45:43 +00:00
}
2007-12-09 22:45:43 +00:00
return result;
}
2007-09-30 02:48:27 +00:00
2007-12-09 22:45:43 +00:00
object
compile(MyThread* t, Context* context)
2007-12-09 22:45:43 +00:00
{
Compiler* c = context->c;
2007-09-30 02:48:27 +00:00
// fprintf(stderr, "compiling %s.%s%s\n",
// &byteArrayBody(t, className(t, methodClass(t, context->method)), 0),
// &byteArrayBody(t, methodName(t, context->method), 0),
// &byteArrayBody(t, methodSpec(t, context->method), 0));
2007-12-09 22:45:43 +00:00
c->prologue();
2007-09-30 02:48:27 +00:00
unsigned footprint = methodParameterFootprint(t, context->method);
unsigned locals = localSize(t, context->method);
c->reserve(locals - footprint);
2007-09-30 02:48:27 +00:00
2008-01-07 14:51:07 +00:00
uintptr_t stackMap[stackMapSizeInWords(t, context->method)];
Frame frame(context, stackMap);
2007-09-30 02:48:27 +00:00
2008-01-07 14:51:07 +00:00
unsigned index = 0;
if ((methodFlags(t, context->method) & ACC_STATIC) == 0) {
2008-01-07 14:51:07 +00:00
frame.mark(index++);
}
for (MethodSpecIterator it
(t, reinterpret_cast<const char*>
(&byteArrayBody(t, methodSpec(t, context->method), 0)));
it.hasNext();)
{
2008-01-07 14:51:07 +00:00
switch (*it.next()) {
case 'L':
case '[':
frame.mark(index++);
break;
case 'J':
case 'D':
index += 2;
break;
default:
++ index;
break;
}
}
handleEntrance(t, &frame);
2007-12-09 22:45:43 +00:00
compile(t, &frame, 0);
if (UNLIKELY(t->exception)) return 0;
2007-09-30 02:48:27 +00:00
context->dirtyRoots = false;
unsigned eventIndex = calculateFrameMaps(t, context, 0, 0);
2008-01-07 14:51:07 +00:00
object eht = codeExceptionHandlerTable(t, methodCode(t, context->method));
2007-12-09 22:45:43 +00:00
if (eht) {
PROTECT(t, eht);
2007-09-30 02:48:27 +00:00
unsigned visitCount = exceptionHandlerTableLength(t, eht);
bool visited[visitCount];
memset(visited, 0, visitCount);
while (visitCount) {
bool progress = false;
for (unsigned i = 0; i < exceptionHandlerTableLength(t, eht); ++i) {
ExceptionHandler* eh = exceptionHandlerTableBody(t, eht, i);
unsigned start = exceptionHandlerStart(eh);
if (not visited[i] and context->visitTable[start]) {
-- visitCount;
visited[i] = true;
progress = true;
uintptr_t stackMap[stackMapSizeInWords(t, context->method)];
Frame frame2(&frame, stackMap);
uintptr_t* roots = context->rootTable
+ (start * frameMapSizeInWords(t, context->method));
for (unsigned i = 0; i < localSize(t, context->method); ++ i) {
if (getBit(roots, i)) {
frame2.mark(i);
} else {
frame2.clear(i);
}
}
2007-09-30 02:48:27 +00:00
frame2.pushObject();
2008-01-07 14:51:07 +00:00
for (unsigned i = 1;
i < codeMaxStack(t, methodCode(t, context->method));
++i)
{
frame2.clear(localSize(t, context->method) + i);
}
2008-01-07 14:51:07 +00:00
2008-04-09 19:08:13 +00:00
compile(t, &frame2, exceptionHandlerIp(eh), true);
if (UNLIKELY(t->exception)) return 0;
eventIndex = calculateFrameMaps(t, context, 0, eventIndex);
}
}
assert(t, progress);
2007-12-09 22:45:43 +00:00
}
}
2007-09-30 02:48:27 +00:00
while (context->dirtyRoots) {
context->dirtyRoots = false;
calculateFrameMaps(t, context, 0, 0);
}
return finish(t, context, 0);
2007-12-09 22:45:43 +00:00
}
2007-09-30 02:48:27 +00:00
2007-12-09 22:45:43 +00:00
void
compile(MyThread* t, object method);
2007-09-30 02:48:27 +00:00
2007-12-09 22:45:43 +00:00
void*
2007-12-16 22:41:07 +00:00
compileMethod2(MyThread* t)
2007-12-09 22:45:43 +00:00
{
object node = findCallNode(t, *static_cast<void**>(t->stack));
2007-12-16 22:41:07 +00:00
PROTECT(t, node);
object target = callNodeTarget(t, node);
2007-12-16 22:41:07 +00:00
PROTECT(t, target);
2007-09-30 02:48:27 +00:00
if (callNodeVirtualCall(t, node)) {
target = resolveTarget(t, t->stack, target);
2007-12-18 00:22:37 +00:00
}
2007-12-09 22:45:43 +00:00
if (LIKELY(t->exception == 0)) {
compile(t, target);
}
2007-12-09 22:45:43 +00:00
if (UNLIKELY(t->exception)) {
2007-12-16 22:41:07 +00:00
return 0;
2007-12-09 22:45:43 +00:00
} else {
if (callNodeVirtualCall(t, node)) {
classVtable
(t, objectClass
(t, resolveThisPointer(t, t->stack, target)), methodOffset(t, target))
= &singletonValue(t, methodCompiled(t, target), 0);
} else {
{ ACQUIRE(t, t->m->classLock);
removeCallNode(t, node);
}
Context context(t);
context.c->updateCall
(reinterpret_cast<void*>(callNodeAddress(t, node)),
&singletonValue(t, methodCompiled(t, target), 0));
2007-12-09 22:45:43 +00:00
}
return &singletonValue(t, methodCompiled(t, target), 0);
}
}
2007-09-30 02:48:27 +00:00
void* FORCE_ALIGN
2007-12-16 22:41:07 +00:00
compileMethod(MyThread* t)
{
void* r = compileMethod2(t);
if (UNLIKELY(t->exception)) {
unwind(t);
} else {
return r;
}
}
2007-12-09 22:45:43 +00:00
uint64_t
invokeNative2(MyThread* t, object method)
{
PROTECT(t, method);
assert(t, methodFlags(t, method) & ACC_NATIVE);
2008-01-08 19:36:34 +00:00
initClass(t, methodClass(t, method));
if (UNLIKELY(t->exception)) return 0;
if (methodCode(t, method) == 0) {
2007-12-09 22:45:43 +00:00
void* function = resolveNativeMethod(t, method);
if (UNLIKELY(function == 0)) {
object message = makeString
2008-01-31 00:08:47 +00:00
(t, "%s.%s%s",
&byteArrayBody(t, className(t, methodClass(t, method)), 0),
&byteArrayBody(t, methodName(t, method), 0),
&byteArrayBody(t, methodSpec(t, method), 0));
2007-12-09 22:45:43 +00:00
t->exception = makeUnsatisfiedLinkError(t, message);
return 0;
}
2007-12-09 22:45:43 +00:00
object p = makePointer(t, function);
set(t, method, MethodCode, p);
}
2007-12-09 22:45:43 +00:00
object class_ = methodClass(t, method);
PROTECT(t, class_);
2007-12-09 22:45:43 +00:00
unsigned footprint = methodParameterFootprint(t, method) + 1;
if (methodFlags(t, method) & ACC_STATIC) {
++ footprint;
}
2008-01-03 19:49:07 +00:00
unsigned count = methodParameterCount(t, method) + 2;
2007-12-09 22:45:43 +00:00
uintptr_t args[footprint];
unsigned argOffset = 0;
uint8_t types[count];
unsigned typeOffset = 0;
2007-09-30 02:48:27 +00:00
2007-12-09 22:45:43 +00:00
args[argOffset++] = reinterpret_cast<uintptr_t>(t);
types[typeOffset++] = POINTER_TYPE;
2007-12-09 22:45:43 +00:00
uintptr_t* sp = static_cast<uintptr_t*>(t->stack)
2007-12-15 01:11:01 +00:00
+ methodParameterFootprint(t, method);
2007-12-09 22:45:43 +00:00
if (methodFlags(t, method) & ACC_STATIC) {
args[argOffset++] = reinterpret_cast<uintptr_t>(&class_);
} else {
args[argOffset++] = reinterpret_cast<uintptr_t>(sp--);
}
types[typeOffset++] = POINTER_TYPE;
2007-12-09 22:45:43 +00:00
MethodSpecIterator it
(t, reinterpret_cast<const char*>
(&byteArrayBody(t, methodSpec(t, method), 0)));
while (it.hasNext()) {
unsigned type = types[typeOffset++]
= fieldType(t, fieldCode(t, *it.next()));
2007-12-09 22:45:43 +00:00
switch (type) {
case INT8_TYPE:
case INT16_TYPE:
case INT32_TYPE:
case FLOAT_TYPE:
args[argOffset++] = *(sp--);
break;
2007-12-09 22:45:43 +00:00
case INT64_TYPE:
case DOUBLE_TYPE: {
2007-12-23 19:18:34 +00:00
memcpy(args + argOffset, sp - 1, 8);
2007-12-18 02:09:32 +00:00
argOffset += (8 / BytesPerWord);
2007-12-18 00:22:37 +00:00
sp -= 2;
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
case POINTER_TYPE: {
if (*sp) {
args[argOffset++] = reinterpret_cast<uintptr_t>(sp);
} else {
args[argOffset++] = 0;
}
-- sp;
2007-12-09 22:45:43 +00:00
} break;
2007-12-09 22:45:43 +00:00
default: abort(t);
2007-10-03 00:22:48 +00:00
}
}
2007-12-09 22:45:43 +00:00
void* function = pointerValue(t, methodCode(t, method));
unsigned returnCode = methodReturnCode(t, method);
unsigned returnType = fieldType(t, returnCode);
2007-12-09 22:45:43 +00:00
uint64_t result;
if (DebugNatives) {
2007-12-09 22:45:43 +00:00
fprintf(stderr, "invoke native method %s.%s\n",
&byteArrayBody(t, className(t, methodClass(t, method)), 0),
&byteArrayBody(t, methodName(t, method), 0));
}
2007-10-04 00:41:54 +00:00
if (methodFlags(t, method) & ACC_SYNCHRONIZED) {
if (methodFlags(t, method) & ACC_STATIC) {
acquire(t, methodClass(t, method));
} else {
acquire(t, *reinterpret_cast<object*>(args[0]));
}
}
Reference* reference = t->reference;
2007-12-09 22:45:43 +00:00
{ ENTER(t, Thread::IdleState);
2007-10-03 00:22:48 +00:00
2007-12-09 22:45:43 +00:00
result = t->m->system->call
(function,
args,
types,
2008-01-03 19:49:07 +00:00
count,
2007-12-09 22:45:43 +00:00
footprint * BytesPerWord,
returnType);
}
2007-09-25 23:53:11 +00:00
if (methodFlags(t, method) & ACC_SYNCHRONIZED) {
if (methodFlags(t, method) & ACC_STATIC) {
release(t, methodClass(t, method));
} else {
release(t, *reinterpret_cast<object*>(args[0]));
}
}
if (DebugNatives) {
2007-12-09 22:45:43 +00:00
fprintf(stderr, "return from native method %s.%s\n",
&byteArrayBody(t, className(t, methodClass(t, method)), 0),
&byteArrayBody(t, methodName(t, method), 0));
2007-10-03 00:22:48 +00:00
}
if (LIKELY(t->exception == 0)) {
switch (returnCode) {
case ByteField:
case BooleanField:
result = static_cast<int8_t>(result);
break;
case CharField:
result = static_cast<uint16_t>(result);
break;
case ShortField:
result = static_cast<int16_t>(result);
break;
case FloatField:
case IntField:
result = static_cast<int32_t>(result);
break;
2007-12-18 02:09:32 +00:00
case LongField:
case DoubleField:
result = result;
break;
case ObjectField:
result = static_cast<uintptr_t>(result) ? *reinterpret_cast<uintptr_t*>
(static_cast<uintptr_t>(result)) : 0;
break;
case VoidField:
result = 0;
break;
default: abort(t);
2007-12-18 02:09:32 +00:00
}
2007-12-09 22:45:43 +00:00
} else {
result = 0;
2007-10-03 00:22:48 +00:00
}
while (t->reference != reference) {
dispose(t, t->reference);
}
return result;
2007-12-09 22:45:43 +00:00
}
uint64_t FORCE_ALIGN
2007-12-09 22:45:43 +00:00
invokeNative(MyThread* t)
{
if (t->trace->nativeMethod == 0) {
object node = findCallNode(t, *static_cast<void**>(t->stack));
t->trace->nativeMethod = callNodeTarget(t, node);
if (callNodeVirtualCall(t, node)) {
t->trace->nativeMethod = resolveTarget
(t, t->stack, t->trace->nativeMethod);
2008-04-01 17:37:59 +00:00
}
}
2008-04-01 17:37:59 +00:00
uint64_t result = 0;
2007-10-03 00:22:48 +00:00
2007-12-09 22:45:43 +00:00
if (LIKELY(t->exception == 0)) {
result = invokeNative2(t, t->trace->nativeMethod);
2007-10-03 00:22:48 +00:00
}
t->trace->nativeMethod = 0;
2007-12-09 22:45:43 +00:00
if (UNLIKELY(t->exception)) {
unwind(t);
} else {
return result;
2007-10-03 00:22:48 +00:00
}
2007-12-09 22:45:43 +00:00
}
unsigned
frameMapIndex(MyThread* t, object method, int32_t offset)
{
object map = codePool(t, methodCode(t, method));
unsigned mapSize = ceiling
(intArrayLength(t, map), (32 / frameSize(t, method)) + 1);
unsigned indexSize = intArrayLength(t, map) - mapSize;
unsigned bottom = 0;
unsigned top = indexSize;
for (unsigned span = top - bottom; span; span = top - bottom) {
unsigned middle = bottom + (span / 2);
int32_t v = intArrayBody(t, map, middle);
if (offset == v) {
return (indexSize * 32) + (frameSize(t, method) * middle);
} else if (offset < v) {
top = middle;
} else {
bottom = middle + 1;
}
}
abort(t);
}
2007-10-03 00:22:48 +00:00
2007-12-09 22:45:43 +00:00
void
visitStackAndLocals(MyThread* t, Heap::Visitor* v, void* base, object method,
void* ip, void* calleeBase, unsigned argumentFootprint)
2007-12-09 22:45:43 +00:00
{
2008-01-07 14:51:07 +00:00
unsigned count;
if (calleeBase) {
unsigned parameterFootprint = methodParameterFootprint(t, method);
unsigned height = static_cast<uintptr_t*>(base)
- static_cast<uintptr_t*>(calleeBase) - 2;
2007-12-09 22:45:43 +00:00
2008-01-07 14:51:07 +00:00
count = parameterFootprint + height - argumentFootprint;
} else {
count = frameSize(t, method);
2007-12-09 22:45:43 +00:00
}
if (count) {
object map = codePool(t, methodCode(t, method));
int index = frameMapIndex
(t, method, difference
(ip, &singletonValue(t, methodCompiled(t, method), 0)));
2007-12-09 22:45:43 +00:00
for (unsigned i = 0; i < count; ++i) {
int j = index + i;
if ((intArrayBody(t, map, j / 32)
& (static_cast<int32_t>(1) << (j % 32))))
{
v->visit(localObject(t, base, method, i));
}
2007-09-25 23:53:11 +00:00
}
}
}
void
2007-12-09 22:45:43 +00:00
visitStack(MyThread* t, Heap::Visitor* v)
{
2007-12-30 22:24:48 +00:00
void* ip = t->ip;
2007-12-09 22:45:43 +00:00
void* base = t->base;
void** stack = static_cast<void**>(t->stack);
2008-01-02 01:07:12 +00:00
if (ip == 0 and stack) {
2007-12-30 22:24:48 +00:00
ip = *stack;
}
2007-12-09 22:45:43 +00:00
MyThread::CallTrace* trace = t->trace;
2008-01-07 14:51:07 +00:00
void* calleeBase = 0;
unsigned argumentFootprint = 0;
2007-12-09 22:45:43 +00:00
2007-12-16 22:41:07 +00:00
while (stack) {
object method = methodForIp(t, ip);
if (method) {
PROTECT(t, method);
visitStackAndLocals
(t, v, base, method, ip, calleeBase, argumentFootprint);
2008-01-07 14:51:07 +00:00
calleeBase = base;
argumentFootprint = methodParameterFootprint(t, method);
2007-12-09 22:45:43 +00:00
stack = static_cast<void**>(base) + 1;
2008-01-02 01:07:12 +00:00
if (stack) {
ip = *stack;
}
2007-12-09 22:45:43 +00:00
base = *static_cast<void**>(base);
} else if (trace) {
2008-01-07 14:51:07 +00:00
calleeBase = 0;
argumentFootprint = 0;
2007-12-09 22:45:43 +00:00
base = trace->base;
stack = static_cast<void**>(trace->stack);
2008-01-02 01:07:12 +00:00
if (stack) {
ip = *stack;
}
2007-12-09 22:45:43 +00:00
trace = trace->next;
} else {
break;
}
}
}
2007-12-09 22:45:43 +00:00
object
compileDefault(MyThread* t, Context* context)
2007-12-09 22:45:43 +00:00
{
Compiler* c = context->c;
mov(c, c->base(), c->memory(c->thread(), difference(&(t->base), t)));
mov(c, c->stack(), c->memory(c->thread(), difference(&(t->stack), t)));
c->directCall
2007-12-09 22:45:43 +00:00
(c->constant(reinterpret_cast<intptr_t>(compileMethod)),
1, c->thread());
Operand* result = ::result(c);
c->jmp(result);
c->release(result);
return finish(t, context, "default");
}
2007-12-09 22:45:43 +00:00
object
compileNative(MyThread* t, Context* context)
{
Compiler* c = context->c;
mov(c, c->base(), c->memory(c->thread(), difference(&(t->base), t)));
mov(c, c->stack(), c->memory(c->thread(), difference(&(t->stack), t)));
2007-12-09 22:45:43 +00:00
c->directCall
(c->constant(reinterpret_cast<intptr_t>(invokeNative)), 1, c->thread());
2007-12-09 22:45:43 +00:00
c->ret();
return finish(t, context, "native");
}
2007-09-25 23:53:11 +00:00
class ArgumentList {
public:
ArgumentList(Thread* t, uintptr_t* array, bool* objectMask, object this_,
const char* spec, bool indirectObjects, va_list arguments):
t(static_cast<MyThread*>(t)),
array(array),
objectMask(objectMask),
position(0),
protector(this)
2007-09-25 23:53:11 +00:00
{
if (this_) {
addObject(this_);
}
for (MethodSpecIterator it(t, spec); it.hasNext();) {
switch (*it.next()) {
2007-09-25 23:53:11 +00:00
case 'L':
case '[':
if (indirectObjects) {
object* v = va_arg(arguments, object*);
addObject(v ? *v : 0);
} else {
addObject(va_arg(arguments, object));
}
break;
case 'J':
case 'D':
addLong(va_arg(arguments, uint64_t));
break;
2007-09-25 23:53:11 +00:00
default:
addInt(va_arg(arguments, uint32_t));
break;
2007-09-25 23:53:11 +00:00
}
}
2007-09-25 23:53:11 +00:00
}
ArgumentList(Thread* t, uintptr_t* array, bool* objectMask, object this_,
const char* spec, object arguments):
t(static_cast<MyThread*>(t)),
array(array),
objectMask(objectMask),
position(0),
protector(this)
2007-09-25 23:53:11 +00:00
{
if (this_) {
addObject(this_);
}
unsigned index = 0;
for (MethodSpecIterator it(t, spec); it.hasNext();) {
switch (*it.next()) {
2007-09-25 23:53:11 +00:00
case 'L':
case '[':
addObject(objectArrayBody(t, arguments, index++));
break;
case 'J':
case 'D':
addLong(cast<int64_t>(objectArrayBody(t, arguments, index++),
BytesPerWord));
break;
default:
addInt(cast<int32_t>(objectArrayBody(t, arguments, index++),
BytesPerWord));
break;
2007-09-25 23:53:11 +00:00
}
}
}
void addObject(object v) {
array[position] = reinterpret_cast<uintptr_t>(v);
objectMask[position] = true;
++ position;
}
2007-10-12 22:06:33 +00:00
void addInt(uintptr_t v) {
2007-09-25 23:53:11 +00:00
array[position] = v;
objectMask[position] = false;
++ position;
}
void addLong(uint64_t v) {
2007-12-23 20:06:24 +00:00
if (BytesPerWord == 8) {
memcpy(array + position + 1, &v, 8);
} else {
// push words in reverse order, since they will be switched back
// when pushed on the stack:
array[position] = v >> 32;
array[position + 1] = v;
2007-12-23 20:06:24 +00:00
}
2007-09-25 23:53:11 +00:00
objectMask[position] = false;
2007-12-23 20:06:24 +00:00
objectMask[position + 1] = false;
2007-09-25 23:53:11 +00:00
position += 2;
}
MyThread* t;
uintptr_t* array;
bool* objectMask;
unsigned position;
class MyProtector: public Thread::Protector {
public:
MyProtector(ArgumentList* list): Protector(list->t), list(list) { }
virtual void visit(Heap::Visitor* v) {
for (unsigned i = 0; i < list->position; ++i) {
if (list->objectMask[i]) {
v->visit(reinterpret_cast<object*>(list->array + i));
}
}
}
ArgumentList* list;
} protector;
2007-09-25 23:53:11 +00:00
};
object
invoke(Thread* thread, object method, ArgumentList* arguments)
{
MyThread* t = static_cast<MyThread*>(thread);
unsigned returnCode = methodReturnCode(t, method);
2007-09-25 23:53:11 +00:00
unsigned returnType = fieldType(t, returnCode);
2007-12-30 22:24:48 +00:00
uint64_t result;
2007-09-30 02:48:27 +00:00
2007-12-30 22:24:48 +00:00
{ MyThread::CallTrace trace(t);
if (methodFlags(t, method) & ACC_NATIVE) {
trace.nativeMethod = method;
}
2007-12-30 22:24:48 +00:00
result = vmInvoke
(t, &singletonValue(t, methodCompiled(t, method), 0), arguments->array,
arguments->position, returnType);
}
2007-09-25 23:53:11 +00:00
object r;
switch (returnCode) {
case ByteField:
case BooleanField:
case CharField:
case ShortField:
case FloatField:
case IntField:
r = makeInt(t, result);
break;
case LongField:
case DoubleField:
r = makeLong(t, result);
break;
case ObjectField:
2007-12-16 00:24:15 +00:00
r = reinterpret_cast<object>(result);
2007-09-25 23:53:11 +00:00
break;
case VoidField:
r = 0;
break;
default:
abort(t);
};
return r;
}
2008-04-09 19:08:13 +00:00
unsigned
traceSize(Thread* t)
{
class Counter: public Processor::StackVisitor {
public:
Counter(Thread* t): t(t), count(0) { }
virtual bool visit(Processor::StackWalker*) {
++ count;
return true;
}
Thread* t;
unsigned count;
} counter(t);
t->m->processor->walkStack(t, &counter);
return FixedSizeOfArray + (counter.count * ArrayElementSizeOfArray)
+ (counter.count * FixedSizeOfTraceElement);
}
2007-12-30 22:24:48 +00:00
class SegFaultHandler: public System::SignalHandler {
public:
SegFaultHandler(): m(0) { }
virtual bool handleSignal(void** ip, void** base, void** stack,
void** thread)
{
2007-12-30 22:24:48 +00:00
MyThread* t = static_cast<MyThread*>(m->localThread->get());
2008-01-02 01:07:12 +00:00
if (t->state == Thread::ActiveState) {
object node = methodForIp(t, *ip);
2008-01-02 01:07:12 +00:00
if (node) {
t->ip = *ip;
t->base = *base;
t->stack = *stack;
2008-04-09 19:08:13 +00:00
ensure(t, FixedSizeOfNullPointerException + traceSize(t));
2008-01-02 01:07:12 +00:00
t->exception = makeNullPointerException(t);
findUnwindTarget(t, ip, base, stack);
*thread = t;
return true;
}
2007-12-30 22:24:48 +00:00
}
2008-01-02 01:07:12 +00:00
return false;
2007-12-30 22:24:48 +00:00
}
Machine* m;
};
2007-09-25 23:53:11 +00:00
class MyProcessor: public Processor {
public:
MyProcessor(System* s, Allocator* allocator):
2007-09-25 23:53:11 +00:00
s(s),
allocator(allocator),
2007-12-09 22:45:43 +00:00
defaultCompiled(0),
nativeCompiled(0),
callTable(0),
callTableSize(0),
methodTree(0),
methodTreeSentinal(0),
indirectCaller(0),
indirectCallerSize(0),
codeAllocator(s, allocator, true, 64 * 1024)
{ }
2007-09-25 23:53:11 +00:00
virtual Thread*
makeThread(Machine* m, object javaThread, Thread* parent)
{
MyThread* t = new (m->heap->allocate(sizeof(MyThread), false))
MyThread(m, javaThread, parent);
t->init();
return t;
2007-09-25 23:53:11 +00:00
}
2007-12-09 22:45:43 +00:00
object getDefaultCompiled(MyThread* t) {
if (defaultCompiled == 0) {
Context context(t);
defaultCompiled = compileDefault(t, &context);
}
2007-12-09 22:45:43 +00:00
return defaultCompiled;
}
2007-12-09 22:45:43 +00:00
object getNativeCompiled(MyThread* t) {
if (nativeCompiled == 0) {
Context context(t);
nativeCompiled = compileNative(t, &context);
2007-09-25 23:53:11 +00:00
}
2007-12-09 22:45:43 +00:00
return nativeCompiled;
2007-09-25 23:53:11 +00:00
}
2007-12-09 22:45:43 +00:00
virtual object
makeMethod(vm::Thread* t,
uint8_t vmFlags,
uint8_t returnCode,
uint8_t parameterCount,
uint8_t parameterFootprint,
uint16_t flags,
uint16_t offset,
object name,
object spec,
object class_,
object code)
2007-10-04 03:19:39 +00:00
{
2007-12-09 22:45:43 +00:00
return vm::makeMethod
(t, vmFlags, returnCode, parameterCount, parameterFootprint, flags,
offset, name, spec, class_, code,
getDefaultCompiled(static_cast<MyThread*>(t)));
2007-10-04 03:19:39 +00:00
}
2007-12-09 22:45:43 +00:00
virtual object
makeClass(vm::Thread* t,
uint16_t flags,
uint8_t vmFlags,
uint8_t arrayDimensions,
uint16_t fixedSize,
uint16_t arrayElementSize,
object objectMask,
object name,
object super,
object interfaceTable,
object virtualTable,
object fieldTable,
object methodTable,
object staticTable,
object loader,
unsigned vtableLength)
{
2007-12-11 21:26:59 +00:00
return vm::makeClass
2007-12-09 22:45:43 +00:00
(t, flags, vmFlags, arrayDimensions, fixedSize, arrayElementSize,
objectMask, name, super, interfaceTable, virtualTable, fieldTable,
methodTable, staticTable, loader, vtableLength, false);
2007-12-11 21:26:59 +00:00
}
2007-12-11 21:26:59 +00:00
virtual void
initVtable(Thread* t, object c)
{
void* compiled = &singletonBody
(t, getDefaultCompiled(static_cast<MyThread*>(t)), 0);
for (unsigned i = 0; i < classLength(t, c); ++i) {
classVtable(t, c, i) = compiled;
}
}
virtual void
initClass(Thread* t, object c)
{
PROTECT(t, c);
ACQUIRE(t, t->m->classLock);
if (classVmFlags(t, c) & NeedInitFlag
and (classVmFlags(t, c) & InitFlag) == 0)
{
classVmFlags(t, c) |= InitFlag;
invoke(t, classInitializer(t, c), 0);
if (t->exception) {
t->exception = makeExceptionInInitializerError(t, t->exception);
}
classVmFlags(t, c) &= ~(NeedInitFlag | InitFlag);
}
}
2007-09-25 23:53:11 +00:00
virtual void
visitObjects(Thread* vmt, Heap::Visitor* v)
2007-09-25 23:53:11 +00:00
{
MyThread* t = static_cast<MyThread*>(vmt);
2007-12-09 22:45:43 +00:00
if (t == t->m->rootThread) {
v->visit(&defaultCompiled);
v->visit(&nativeCompiled);
v->visit(&callTable);
v->visit(&methodTree);
v->visit(&methodTreeSentinal);
}
2007-09-25 23:53:11 +00:00
for (MyThread::CallTrace* trace = t->trace; trace; trace = trace->next) {
v->visit(&(trace->nativeMethod));
}
2008-04-01 17:37:59 +00:00
2007-12-09 22:45:43 +00:00
for (Reference* r = t->reference; r; r = r->next) {
v->visit(&(r->target));
}
2007-09-25 23:53:11 +00:00
2007-12-09 22:45:43 +00:00
visitStack(t, v);
2007-09-25 23:53:11 +00:00
}
2007-12-09 22:45:43 +00:00
virtual void
walkStack(Thread* vmt, StackVisitor* v)
2007-09-25 23:53:11 +00:00
{
2007-12-09 22:45:43 +00:00
MyThread* t = static_cast<MyThread*>(vmt);
2007-09-25 23:53:11 +00:00
2007-12-09 22:45:43 +00:00
MyStackWalker walker(t);
walker.walk(v);
2007-09-25 23:53:11 +00:00
}
2007-10-04 22:41:19 +00:00
virtual int
2007-12-09 22:45:43 +00:00
lineNumber(Thread* vmt, object method, int ip)
2007-10-04 22:41:19 +00:00
{
2007-12-09 22:45:43 +00:00
return findLineNumber(static_cast<MyThread*>(vmt), method, ip);
2007-10-04 22:41:19 +00:00
}
2007-09-25 23:53:11 +00:00
virtual object*
makeLocalReference(Thread* vmt, object o)
2007-09-25 23:53:11 +00:00
{
if (o) {
MyThread* t = static_cast<MyThread*>(vmt);
PROTECT(t, o);
Reference* r = new (t->m->heap->allocate(sizeof(Reference), false))
Reference(o, &(t->reference));
return &(r->target);
} else {
return 0;
}
2007-09-25 23:53:11 +00:00
}
virtual void
disposeLocalReference(Thread* t, object* r)
2007-09-25 23:53:11 +00:00
{
if (r) {
vm::dispose(t, reinterpret_cast<Reference*>(r));
}
2007-09-25 23:53:11 +00:00
}
virtual object
invokeArray(Thread* t, object method, object this_, object arguments)
{
2008-04-01 17:37:59 +00:00
if (UNLIKELY(t->exception)) return 0;
assert(t, t->state == Thread::ActiveState
or t->state == Thread::ExclusiveState);
assert(t, ((methodFlags(t, method) & ACC_STATIC) == 0) xor (this_ == 0));
2007-09-25 23:53:11 +00:00
const char* spec = reinterpret_cast<char*>
(&byteArrayBody(t, methodSpec(t, method), 0));
2007-12-09 22:45:43 +00:00
unsigned size = methodParameterFootprint(t, method);
2007-09-25 23:53:11 +00:00
uintptr_t array[size];
bool objectMask[size];
ArgumentList list(t, array, objectMask, this_, spec, arguments);
2007-12-09 22:45:43 +00:00
PROTECT(t, method);
compile(static_cast<MyThread*>(t), method);
if (LIKELY(t->exception == 0)) {
return ::invoke(t, method, &list);
}
return 0;
}
virtual object
invokeList(Thread* t, object method, object this_, bool indirectObjects,
va_list arguments)
{
2008-04-01 17:37:59 +00:00
if (UNLIKELY(t->exception)) return 0;
assert(t, t->state == Thread::ActiveState
or t->state == Thread::ExclusiveState);
assert(t, ((methodFlags(t, method) & ACC_STATIC) == 0) xor (this_ == 0));
2007-09-25 23:53:11 +00:00
const char* spec = reinterpret_cast<char*>
(&byteArrayBody(t, methodSpec(t, method), 0));
2007-12-09 22:45:43 +00:00
unsigned size = methodParameterFootprint(t, method);
2007-09-25 23:53:11 +00:00
uintptr_t array[size];
bool objectMask[size];
ArgumentList list
(t, array, objectMask, this_, spec, indirectObjects, arguments);
2007-12-09 22:45:43 +00:00
PROTECT(t, method);
compile(static_cast<MyThread*>(t), method);
if (LIKELY(t->exception == 0)) {
return ::invoke(t, method, &list);
}
return 0;
}
virtual object
invokeList(Thread* t, const char* className, const char* methodName,
const char* methodSpec, object this_, va_list arguments)
{
2008-04-01 17:37:59 +00:00
if (UNLIKELY(t->exception)) return 0;
assert(t, t->state == Thread::ActiveState
or t->state == Thread::ExclusiveState);
2007-12-09 22:45:43 +00:00
unsigned size = parameterFootprint(t, methodSpec, false);
2007-09-25 23:53:11 +00:00
uintptr_t array[size];
bool objectMask[size];
ArgumentList list
(t, array, objectMask, this_, methodSpec, false, arguments);
object method = resolveMethod(t, className, methodName, methodSpec);
if (LIKELY(t->exception == 0)) {
assert(t, ((methodFlags(t, method) & ACC_STATIC) == 0) xor (this_ == 0));
2007-12-09 22:45:43 +00:00
PROTECT(t, method);
compile(static_cast<MyThread*>(t), method);
2007-12-09 22:45:43 +00:00
if (LIKELY(t->exception == 0)) {
return ::invoke(t, method, &list);
}
2007-10-03 00:22:48 +00:00
}
2007-12-09 22:45:43 +00:00
return 0;
}
2007-10-03 00:22:48 +00:00
2007-12-30 22:24:48 +00:00
virtual void dispose(Thread* vmt) {
MyThread* t = static_cast<MyThread*>(vmt);
2007-12-11 21:26:59 +00:00
while (t->reference) {
vm::dispose(t, t->reference);
}
t->m->heap->free(t, sizeof(*t), false);
2007-12-11 21:26:59 +00:00
}
2007-12-09 22:45:43 +00:00
virtual void dispose() {
codeAllocator.dispose();
s->handleSegFault(0);
allocator->free(this, sizeof(*this), false);
}
2008-04-09 19:08:13 +00:00
virtual object getStackTrace(Thread* vmt, Thread* vmTarget) {
MyThread* t = static_cast<MyThread*>(vmt);
MyThread* target = static_cast<MyThread*>(vmTarget);
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* oldStack = target->stack;
target->ip = ip;
target->base = base;
target->stack = stack;
trace = makeTrace(t, target);
target->ip = oldIp;
target->base = oldBase;
target->stack = oldStack;
}
MyThread* t;
MyThread* target;
object trace;
} visitor(t, target);
if (t->backupHeap) {
PROTECT(t, visitor.trace);
collect(t, Heap::MinorCollection);
}
return visitor.trace;
}
System* s;
Allocator* allocator;
2007-12-09 22:45:43 +00:00
object defaultCompiled;
object nativeCompiled;
object callTable;
unsigned callTableSize;
object methodTree;
object methodTreeSentinal;
2007-12-11 21:26:59 +00:00
uint8_t* indirectCaller;
unsigned indirectCallerSize;
2007-12-30 22:24:48 +00:00
SegFaultHandler segFaultHandler;
Zone codeAllocator;
};
2007-12-09 22:45:43 +00:00
MyProcessor*
processor(MyThread* t)
2007-10-04 03:19:39 +00:00
{
2007-12-09 22:45:43 +00:00
MyProcessor* p = static_cast<MyProcessor*>(t->m->processor);
if (p->callTable == 0) {
2007-12-09 22:45:43 +00:00
ACQUIRE(t, t->m->classLock);
2007-10-04 03:19:39 +00:00
if (p->callTable == 0) {
p->callTable = makeArray(t, 128, true);
p->methodTree = p->methodTreeSentinal = makeTreeNode(t, 0, 0, 0);
set(t, p->methodTree, TreeNodeLeft, p->methodTreeSentinal);
set(t, p->methodTree, TreeNodeRight, p->methodTreeSentinal);
Context context(t);
Compiler* c = context.c;
2007-12-09 22:45:43 +00:00
mov(c, c->base(), c->memory(c->thread(), difference(&(t->base), t)));
mov(c, c->stack(), c->memory(c->thread(), difference(&(t->stack), t)));
2007-12-09 22:45:43 +00:00
c->jmp(c->indirectTarget());
p->indirectCallerSize = c->codeSize();
2007-12-11 21:26:59 +00:00
p->indirectCaller = static_cast<uint8_t*>
(p->codeAllocator.allocate(p->indirectCallerSize));
2007-12-09 22:45:43 +00:00
c->writeTo(p->indirectCaller);
2007-12-11 21:26:59 +00:00
if (Verbose) {
logCompile(p->indirectCaller, c->codeSize(), 0, "indirect caller", 0);
}
2007-12-30 22:24:48 +00:00
p->segFaultHandler.m = t->m;
expect(t, t->m->system->success
(t->m->system->handleSegFault(&(p->segFaultHandler))));
}
2007-12-09 22:45:43 +00:00
}
return p;
}
2008-04-09 19:08:13 +00:00
object
defaultCompiled(MyThread* t)
{
return processor(t)->getDefaultCompiled(t);
}
object
nativeCompiled(MyThread* t)
{
return processor(t)->getNativeCompiled(t);
}
void
2007-12-09 22:45:43 +00:00
compile(MyThread* t, object method)
{
MyProcessor* p = processor(t);
2007-12-11 21:26:59 +00:00
if (methodCompiled(t, method) == p->getDefaultCompiled(t)) {
2007-12-09 22:45:43 +00:00
PROTECT(t, method);
ACQUIRE(t, t->m->classLock);
2007-12-11 21:26:59 +00:00
if (methodCompiled(t, method) == p->getDefaultCompiled(t)) {
2008-01-08 19:36:34 +00:00
initClass(t, methodClass(t, method));
if (UNLIKELY(t->exception)) return;
if (methodCompiled(t, method) == p->getDefaultCompiled(t)) {
object compiled;
if (methodFlags(t, method) & ACC_NATIVE) {
compiled = p->getNativeCompiled(t);
} else {
Context context(t, method, p->indirectCaller);
compiled = compile(t, &context);
if (UNLIKELY(t->exception)) return;
}
set(t, method, MethodCompiled, compiled);
2007-12-09 22:45:43 +00:00
if (methodVirtual(t, method)) {
classVtable(t, methodClass(t, method), methodOffset(t, method))
= &singletonValue(t, compiled, 0);
}
if ((methodFlags(t, method) & ACC_NATIVE) == 0) {
if (DebugMethodTree) {
fprintf(stderr, "insert method at %p\n",
&singletonValue(t, methodCompiled(t, method), 0));
}
methodTree(t) = treeInsert
(t, methodTree(t), method, methodTreeSentinal(t),
compareMethodBounds);
}
2007-12-11 23:52:28 +00:00
}
2007-12-09 22:45:43 +00:00
}
}
}
object
findCallNode(MyThread* t, void* address)
{
if (DebugCallTable) {
fprintf(stderr, "find trace node %p\n", address);
}
2007-12-09 22:45:43 +00:00
MyProcessor* p = processor(t);
object table = p->callTable;
2007-12-11 00:48:09 +00:00
intptr_t key = reinterpret_cast<intptr_t>(address);
unsigned index = static_cast<uintptr_t>(key)
& (arrayLength(t, table) - 1);
2007-12-11 00:48:09 +00:00
for (object n = arrayBody(t, table, index);
n; n = callNodeNext(t, n))
2007-12-11 00:48:09 +00:00
{
intptr_t k = callNodeAddress(t, n);
2007-12-11 00:48:09 +00:00
if (k == key) {
return n;
}
}
return 0;
2007-12-11 00:48:09 +00:00
}
object
resizeTable(MyThread* t, object oldTable, unsigned newLength)
{
PROTECT(t, oldTable);
object oldNode = 0;
PROTECT(t, oldNode);
2007-12-11 00:48:09 +00:00
object newTable = makeArray(t, newLength, true);
2008-01-02 01:07:12 +00:00
PROTECT(t, newTable);
2007-12-11 00:48:09 +00:00
for (unsigned i = 0; i < arrayLength(t, oldTable); ++i) {
for (oldNode = arrayBody(t, oldTable, i);
oldNode;
oldNode = callNodeNext(t, oldNode))
{
intptr_t k = callNodeAddress(t, oldNode);
2007-12-11 00:48:09 +00:00
unsigned index = k & (newLength - 1);
object newNode = makeCallNode
(t, callNodeAddress(t, oldNode),
callNodeTarget(t, oldNode),
callNodeVirtualCall(t, oldNode),
arrayBody(t, newTable, index));
set(t, newTable, ArrayBody + (index * BytesPerWord), newNode);
2007-12-11 00:48:09 +00:00
}
}
return newTable;
2007-12-09 22:45:43 +00:00
}
2007-12-09 22:45:43 +00:00
void
insertCallNode(MyThread* t, object node)
2007-12-09 22:45:43 +00:00
{
if (DebugCallTable) {
fprintf(stderr, "insert trace node %p\n",
reinterpret_cast<void*>(callNodeAddress(t, node)));
}
2007-12-09 22:45:43 +00:00
MyProcessor* p = processor(t);
2007-12-11 21:26:59 +00:00
PROTECT(t, node);
2007-12-11 00:48:09 +00:00
++ p->callTableSize;
if (p->callTableSize >= arrayLength(t, p->callTable) * 2) {
p->callTable = resizeTable
(t, p->callTable, arrayLength(t, p->callTable) * 2);
}
intptr_t key = callNodeAddress(t, node);
unsigned index = static_cast<uintptr_t>(key)
& (arrayLength(t, p->callTable) - 1);
set(t, node, CallNodeNext, arrayBody(t, p->callTable, index));
set(t, p->callTable, ArrayBody + (index * BytesPerWord), node);
}
2007-12-11 00:48:09 +00:00
void
removeCallNode(MyThread* t, object node)
{
if (DebugCallTable) {
fprintf(stderr, "remove call node %p\n",
reinterpret_cast<void*>(callNodeAddress(t, node)));
2007-12-11 00:48:09 +00:00
}
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);
2007-12-11 00:48:09 +00:00
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;
2007-12-11 00:48:09 +00:00
if (p->callTableSize <= arrayLength(t, p->callTable) / 3) {
p->callTable = resizeTable
(t, p->callTable, arrayLength(t, p->callTable) / 2);
}
}
object&
methodTree(MyThread* t)
{
return processor(t)->methodTree;
}
object
methodTreeSentinal(MyThread* t)
{
return processor(t)->methodTreeSentinal;
}
Allocator*
codeAllocator(MyThread* t) {
return &(processor(t)->codeAllocator);
}
2007-12-09 22:45:43 +00:00
} // namespace
namespace vm {
2007-09-25 23:53:11 +00:00
Processor*
makeProcessor(System* system, Allocator* allocator)
{
return new (allocator->allocate(sizeof(MyProcessor), false))
MyProcessor(system, allocator);
}
} // namespace vm