add SingleRead::successor; fix build errors

The SingleRead::successor field is used (when non-null) to further
constrain the SiteMask in SingleRead::intersect based on reads of
successor values (as in the cases of moves and condensed-addressing
combine and translate instructions).
This commit is contained in:
Joel Dice 2009-04-06 18:34:12 -06:00
parent fea92ed995
commit 35d1c6e068
7 changed files with 209 additions and 86 deletions

View File

@ -44,13 +44,19 @@ class BootImage {
uintptr_t codeBase; uintptr_t codeBase;
unsigned defaultThunk; unsigned defaultThunk;
unsigned defaultTailThunk;
unsigned defaultVirtualThunk;
unsigned tailHelperThunk;
unsigned nativeThunk; unsigned nativeThunk;
unsigned nativeTailThunk;
unsigned aioobThunk; unsigned aioobThunk;
unsigned thunkTable; unsigned thunkTable;
unsigned thunkSize; unsigned thunkSize;
unsigned compileMethodCall; unsigned compileMethodCall;
unsigned compileVirtualMethodCall;
unsigned tailCallCall;
unsigned invokeNativeCall; unsigned invokeNativeCall;
unsigned throwArrayIndexOutOfBoundsCall; unsigned throwArrayIndexOutOfBoundsCall;

View File

@ -76,6 +76,9 @@ class MyThread: public Thread {
ip(0), ip(0),
base(0), base(0),
stack(0), stack(0),
tailAddress(0),
virtualCallClass(0),
virtualCallIndex(0),
trace(0), trace(0),
reference(0), reference(0),
arch(parent ? parent->arch : makeArchitecture(m->system)) arch(parent ? parent->arch : makeArchitecture(m->system))
@ -86,6 +89,9 @@ class MyThread: public Thread {
void* ip; void* ip;
void* base; void* base;
void* stack; void* stack;
void* tailAddress;
void* virtualCallClass;
uintptr_t virtualCallIndex;
CallTrace* trace; CallTrace* trace;
Reference* reference; Reference* reference;
Assembler::Architecture* arch; Assembler::Architecture* arch;
@ -441,8 +447,8 @@ class Context;
class TraceElement: public TraceHandler { class TraceElement: public TraceHandler {
public: public:
const unsigned VirtualCall = 1 << 0; static const unsigned VirtualCall = 1 << 0;
const unsigned TailCall = 1 << 1; static const unsigned TailCall = 1 << 1;
TraceElement(Context* context, object target, TraceElement(Context* context, object target,
unsigned flags, TraceElement* next): unsigned flags, TraceElement* next):
@ -477,7 +483,7 @@ class TraceElement: public TraceHandler {
class TraceElementPromise: public Promise { class TraceElementPromise: public Promise {
public: public:
TraceElementPromise(TraceElement* trace): trace(trace) { } TraceElementPromise(System* s, TraceElement* trace): s(s), trace(trace) { }
virtual int64_t value() { virtual int64_t value() {
assert(s, resolved()); assert(s, resolved());
@ -1352,12 +1358,24 @@ objectPools(MyThread* t);
uintptr_t uintptr_t
defaultThunk(MyThread* t); defaultThunk(MyThread* t);
uintptr_t
defaultTailThunk(MyThread* t);
uintptr_t uintptr_t
nativeThunk(MyThread* t); nativeThunk(MyThread* t);
uintptr_t
nativeTailThunk(MyThread* t);
uintptr_t
tailHelperThunk(MyThread* t);
uintptr_t uintptr_t
aioobThunk(MyThread* t); aioobThunk(MyThread* t);
uintptr_t
virtualThunk(MyThread* t, unsigned index);
uintptr_t uintptr_t
methodAddress(Thread* t, object method) methodAddress(Thread* t, object method)
{ {
@ -1378,6 +1396,10 @@ tryInitClass(MyThread* t, object class_)
FixedAllocator* FixedAllocator*
codeAllocator(MyThread* t); codeAllocator(MyThread* t);
void
compile(MyThread* t, Allocator* allocator, BootContext* bootContext,
object method);
int64_t int64_t
findInterfaceMethodFromInstance(MyThread* t, object method, object instance) findInterfaceMethodFromInstance(MyThread* t, object method, object instance)
{ {
@ -1916,6 +1938,8 @@ Compiler::Operand*
compileDirectInvoke(MyThread* t, Frame* frame, object target, bool tailCall, compileDirectInvoke(MyThread* t, Frame* frame, object target, bool tailCall,
bool useThunk, unsigned rSize, Promise* addressPromise) bool useThunk, unsigned rSize, Promise* addressPromise)
{ {
Compiler* c = frame->c;
if (tailCall and methodParameterFootprint(t, target) if (tailCall and methodParameterFootprint(t, target)
> methodParameterFootprint(t, frame->context->method)) > methodParameterFootprint(t, frame->context->method))
{ {
@ -1930,7 +1954,6 @@ compileDirectInvoke(MyThread* t, Frame* frame, object target, bool tailCall,
(c->constant(nativeTailThunk(t)), (c->constant(nativeTailThunk(t)),
Compiler::TailCall, Compiler::TailCall,
frame->trace(target, TraceElement::TailCall), frame->trace(target, TraceElement::TailCall),
trace,
rSize, rSize,
methodParameterFootprint(t, target)); methodParameterFootprint(t, target));
} else { } else {
@ -1956,7 +1979,7 @@ compileDirectInvoke(MyThread* t, Frame* frame, object target, bool tailCall,
return result; return result;
} else { } else {
return c->stackCall return c->stackCall
(address, (c->constant(defaultThunk(t)),
flags | Compiler::Aligned, flags | Compiler::Aligned,
frame->trace(target, 0), frame->trace(target, 0),
rSize, rSize,
@ -1969,10 +1992,10 @@ compileDirectInvoke(MyThread* t, Frame* frame, object target, bool tailCall,
: c->constant(methodAddress(t, target))); : c->constant(methodAddress(t, target)));
return c->stackCall return c->stackCall
(c->promiseConstant(p), (address,
flags, flags,
tailCall ? 0 : frame->trace tailCall ? 0 : frame->trace
((methodFlags(t, target) & ACC_NATIVE) ? target, 0), ((methodFlags(t, target) & ACC_NATIVE) ? target : 0, 0),
rSize, rSize,
methodParameterFootprint(t, target)); methodParameterFootprint(t, target));
} }
@ -1982,8 +2005,6 @@ compileDirectInvoke(MyThread* t, Frame* frame, object target, bool tailCall,
void void
compileDirectInvoke(MyThread* t, Frame* frame, object target, bool tailCall) compileDirectInvoke(MyThread* t, Frame* frame, object target, bool tailCall)
{ {
Compiler* c = frame->c;
unsigned rSize = resultSize(t, methodReturnCode(t, target)); unsigned rSize = resultSize(t, methodReturnCode(t, target));
Compiler::Operand* result = 0; Compiler::Operand* result = 0;
@ -2043,7 +2064,7 @@ handleMonitorEvent(MyThread* t, Frame* frame, intptr_t function)
0, 0,
frame->trace(0, 0), frame->trace(0, 0),
0, 0,
2, c->thread(), lock); 2, c->register_(t->arch->thread()), lock);
} }
} }
@ -2229,7 +2250,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
0, 0,
frame->trace(0, 0), frame->trace(0, 0),
0, 0,
1, c->thread()); 1, c->register_(t->arch->thread()));
} }
// fprintf(stderr, "ip: %d map: %ld\n", ip, *(frame->map)); // fprintf(stderr, "ip: %d map: %ld\n", ip, *(frame->map));
@ -2329,7 +2350,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
0, 0,
frame->trace(0, 0), frame->trace(0, 0),
0, 0,
4, c->thread(), array, 4, c->register_(t->arch->thread()), array,
c->add(4, c->constant(ArrayBody), c->add(4, c->constant(ArrayBody),
c->shl(4, c->constant(log(BytesPerWord)), index)), c->shl(4, c->constant(log(BytesPerWord)), index)),
value); value);
@ -2397,7 +2418,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
0, 0,
frame->trace(0, 0), frame->trace(0, 0),
BytesPerWord, BytesPerWord,
3, c->thread(), frame->append(class_), length)); 3, c->register_(t->arch->thread()), frame->append(class_), length));
} break; } break;
case areturn: { case areturn: {
@ -2439,7 +2460,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
Compiler::NoReturn, Compiler::NoReturn,
frame->trace(0, 0), frame->trace(0, 0),
0, 0,
2, c->thread(), target); 2, c->register_(t->arch->thread()), target);
} return; } return;
case bipush: case bipush:
@ -2460,7 +2481,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
0, 0,
frame->trace(0, 0), frame->trace(0, 0),
0, 0,
3, c->thread(), frame->append(class_), instance); 3, c->register_(t->arch->thread()), frame->append(class_), instance);
} break; } break;
case d2f: { case d2f: {
@ -2741,7 +2762,8 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
0, 0,
frame->trace(0, 0), frame->trace(0, 0),
0, 0,
2, c->thread(), frame->append(fieldClass(t, field))); 2, c->register_(t->arch->thread()),
frame->append(fieldClass(t, field)));
} }
table = frame->append(classStaticTable(t, fieldClass(t, field))); table = frame->append(classStaticTable(t, fieldClass(t, field)));
@ -2766,7 +2788,8 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
c->call c->call
(c->constant(getThunk(t, acquireMonitorForObjectThunk)), (c->constant(getThunk(t, acquireMonitorForObjectThunk)),
0, frame->trace(0, 0), 0, 2, c->thread(), fieldOperand); 0, frame->trace(0, 0), 0, 2, c->register_(t->arch->thread()),
fieldOperand);
} }
switch (fieldCode(t, field)) { switch (fieldCode(t, field)) {
@ -2820,7 +2843,8 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
{ {
c->call c->call
(c->constant(getThunk(t, releaseMonitorForObjectThunk)), (c->constant(getThunk(t, releaseMonitorForObjectThunk)),
0, frame->trace(0, 0), 0, 2, c->thread(), fieldOperand); 0, frame->trace(0, 0), 0, 2, c->register_(t->arch->thread()),
fieldOperand);
} else { } else {
c->loadBarrier(); c->loadBarrier();
} }
@ -3094,7 +3118,8 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
(c->call (c->call
(c->constant(getThunk(t, instanceOf64Thunk)), (c->constant(getThunk(t, instanceOf64Thunk)),
0, 0, 4, 0, 0, 4,
3, c->thread(), frame->append(class_), frame->popObject())); 3, c->register_(t->arch->thread()), frame->append(class_),
frame->popObject()));
} break; } break;
case invokeinterface: { case invokeinterface: {
@ -3119,7 +3144,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
0, 0,
frame->trace(0, 0), frame->trace(0, 0),
BytesPerWord, BytesPerWord,
3, c->thread(), frame->append(target), 3, c->register_(t->arch->thread()), frame->append(target),
c->peek(1, instance)), c->peek(1, instance)),
0, 0,
frame->trace(0, 0), frame->trace(0, 0),
@ -3583,14 +3608,14 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
Compiler::Operand* target = frame->popObject(); Compiler::Operand* target = frame->popObject();
c->call c->call
(c->constant(getThunk(t, acquireMonitorForObjectThunk)), (c->constant(getThunk(t, acquireMonitorForObjectThunk)),
0, frame->trace(0, 0), 0, 2, c->thread(), target); 0, frame->trace(0, 0), 0, 2, c->register_(t->arch->thread()), target);
} break; } break;
case monitorexit: { case monitorexit: {
Compiler::Operand* target = frame->popObject(); Compiler::Operand* target = frame->popObject();
c->call c->call
(c->constant(getThunk(t, releaseMonitorForObjectThunk)), (c->constant(getThunk(t, releaseMonitorForObjectThunk)),
0, frame->trace(0, 0), 0, 2, c->thread(), target); 0, frame->trace(0, 0), 0, 2, c->register_(t->arch->thread()), target);
} break; } break;
case multianewarray: { case multianewarray: {
@ -3612,8 +3637,8 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
0, 0,
frame->trace(0, 0), frame->trace(0, 0),
BytesPerWord, BytesPerWord,
4, c->thread(), frame->append(class_), c->constant(dimensions), 4, c->register_(t->arch->thread()), frame->append(class_),
c->constant(offset)); c->constant(dimensions), c->constant(offset));
frame->pop(dimensions); frame->pop(dimensions);
frame->pushObject(result); frame->pushObject(result);
@ -3632,7 +3657,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
0, 0,
frame->trace(0, 0), frame->trace(0, 0),
BytesPerWord, BytesPerWord,
2, c->thread(), frame->append(class_))); 2, c->register_(t->arch->thread()), frame->append(class_)));
} else { } else {
frame->pushObject frame->pushObject
(c->call (c->call
@ -3640,7 +3665,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
0, 0,
frame->trace(0, 0), frame->trace(0, 0),
BytesPerWord, BytesPerWord,
2, c->thread(), frame->append(class_))); 2, c->register_(t->arch->thread()), frame->append(class_)));
} }
} break; } break;
@ -3655,7 +3680,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
0, 0,
frame->trace(0, 0), frame->trace(0, 0),
BytesPerWord, BytesPerWord,
3, c->thread(), c->constant(type), length)); 3, c->register_(t->arch->thread()), c->constant(type), length));
} break; } break;
case nop: break; case nop: break;
@ -3688,7 +3713,8 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
0, 0,
frame->trace(0, 0), frame->trace(0, 0),
0, 0,
2, c->thread(), frame->append(fieldClass(t, field))); 2, c->register_(t->arch->thread()),
frame->append(fieldClass(t, field)));
} }
staticTable = classStaticTable(t, fieldClass(t, field)); staticTable = classStaticTable(t, fieldClass(t, field));
@ -3742,7 +3768,8 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
c->call c->call
(c->constant(getThunk(t, acquireMonitorForObjectThunk)), (c->constant(getThunk(t, acquireMonitorForObjectThunk)),
0, frame->trace(0, 0), 0, 2, c->thread(), fieldOperand); 0, frame->trace(0, 0), 0, 2, c->register_(t->arch->thread()),
fieldOperand);
} else { } else {
c->storeStoreBarrier(); c->storeStoreBarrier();
} }
@ -3779,12 +3806,14 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
0, 0,
frame->trace(0, 0), frame->trace(0, 0),
0, 0,
4, c->thread(), table, c->constant(fieldOffset(t, field)), value); 4, c->register_(t->arch->thread()), table,
c->constant(fieldOffset(t, field)), value);
} else { } else {
c->call c->call
(c->constant(getThunk(t, setThunk)), (c->constant(getThunk(t, setThunk)),
0, 0, 0, 0, 0, 0,
4, c->thread(), table, c->constant(fieldOffset(t, field)), value); 4, c->register_(t->arch->thread()), table,
c->constant(fieldOffset(t, field)), value);
} }
break; break;
@ -3798,7 +3827,8 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
{ {
c->call c->call
(c->constant(getThunk(t, releaseMonitorForObjectThunk)), (c->constant(getThunk(t, releaseMonitorForObjectThunk)),
0, frame->trace(0, 0), 0, 2, c->thread(), fieldOperand); 0, frame->trace(0, 0), 0, 2, c->register_(t->arch->thread()),
fieldOperand);
} else { } else {
c->storeLoadBarrier(); c->storeLoadBarrier();
} }
@ -4562,12 +4592,8 @@ updateCall(MyThread* t, UnaryOperation op, bool assertAlignment,
t->arch->updateCall(op, assertAlignment, returnAddress, target); t->arch->updateCall(op, assertAlignment, returnAddress, target);
} }
void
compile(MyThread* t, Allocator* allocator, BootContext* bootContext,
object method);
void* void*
compileMethod2(MyThread* t, uintptr_t ip) compileMethod2(MyThread* t, void* ip)
{ {
object node = findCallNode(t, ip); object node = findCallNode(t, ip);
PROTECT(t, node); PROTECT(t, node);
@ -4583,12 +4609,12 @@ compileMethod2(MyThread* t, uintptr_t ip)
return 0; return 0;
} else { } else {
void* address = reinterpret_cast<void*>(methodAddress(t, target)); void* address = reinterpret_cast<void*>(methodAddress(t, target));
uintptr_t updateIp = ip; uint8_t* updateIp = static_cast<uint8_t*>(ip);
if (callNode(t, node) & TraceElement::TailCall) { if (callNodeFlags(t, node) & TraceElement::TailCall) {
updateIp -= t->arch->constantCallSize(); updateIp -= t->arch->constantCallSize();
} }
updateCall(t, Call, true, reinterpret_cast<void*>(updateIp), address); updateCall(t, Call, true, updateIp, address);
return address; return address;
} }
@ -4597,7 +4623,7 @@ compileMethod2(MyThread* t, uintptr_t ip)
uint64_t uint64_t
compileMethod(MyThread* t) compileMethod(MyThread* t)
{ {
uintptr_t ip; void* ip;
if (t->tailAddress) { if (t->tailAddress) {
ip = t->tailAddress; ip = t->tailAddress;
t->tailAddress = 0; t->tailAddress = 0;
@ -4629,7 +4655,7 @@ compileVirtualMethod2(MyThread* t, object class_, unsigned index)
return 0; return 0;
} else { } else {
void* address = reinterpret_cast<void*>(methodAddress(t, target)); void* address = reinterpret_cast<void*>(methodAddress(t, target));
if (address != nativeThunk(t, method)) { if (address != reinterpret_cast<void*>(nativeThunk(t))) {
classVtable(t, class_, methodOffset(t, target)) = address; classVtable(t, class_, methodOffset(t, target)) = address;
} }
return address; return address;
@ -4639,7 +4665,7 @@ compileVirtualMethod2(MyThread* t, object class_, unsigned index)
uint64_t uint64_t
compileVirtualMethod(MyThread* t) compileVirtualMethod(MyThread* t)
{ {
object class_ = t->virtualCallClass; object class_ = static_cast<object>(t->virtualCallClass);
t->virtualCallClass = 0; t->virtualCallClass = 0;
unsigned index = t->virtualCallIndex; unsigned index = t->virtualCallIndex;
@ -4655,7 +4681,7 @@ compileVirtualMethod(MyThread* t)
} }
void* void*
tailCall2(MyThread* t, uintptr_t ip) tailCall2(MyThread* t, void* ip)
{ {
object node = findCallNode(t, ip); object node = findCallNode(t, ip);
PROTECT(t, node); PROTECT(t, node);
@ -4678,7 +4704,9 @@ tailCall2(MyThread* t, uintptr_t ip)
void* stack = t->stack; void* stack = t->stack;
t->arch->nextFrame(&stack, &base); t->arch->nextFrame(&stack, &base);
if (t->arch->matchCall(t->arch->frameIp(stack), tailHelperThunk(t))) { if (t->arch->matchCall(t->arch->frameIp(stack),
reinterpret_cast<void*>(tailHelperThunk(t))))
{
void* nextBase = base; void* nextBase = base;
void* nextStack = stack; void* nextStack = stack;
t->arch->nextFrame(&nextStack, &nextBase); t->arch->nextFrame(&nextStack, &nextBase);
@ -4896,7 +4924,7 @@ uint64_t
invokeNative(MyThread* t) invokeNative(MyThread* t)
{ {
if (t->trace->nativeMethod == 0) { if (t->trace->nativeMethod == 0) {
uintptr_t ip; void* ip;
if (t->tailAddress) { if (t->tailAddress) {
ip = t->tailAddress; ip = t->tailAddress;
t->tailAddress = 0; t->tailAddress = 0;
@ -5276,7 +5304,11 @@ class MyProcessor: public Processor {
s(s), s(s),
allocator(allocator), allocator(allocator),
defaultThunk(0), defaultThunk(0),
defaultTailThunk(0),
defaultVirtualThunk(0),
tailHelperThunk(0),
nativeThunk(0), nativeThunk(0),
nativeTailThunk(0),
aioobThunk(0), aioobThunk(0),
callTable(0), callTable(0),
callTableSize(0), callTableSize(0),
@ -5346,7 +5378,8 @@ class MyProcessor: public Processor {
initVtable(Thread* t, object c) initVtable(Thread* t, object c)
{ {
for (unsigned i = 0; i < classLength(t, c); ++i) { for (unsigned i = 0; i < classLength(t, c); ++i) {
classVtable(t, c, i) = virtualThunk(static_cast<MyThread*>(t), i); classVtable(t, c, i) = reinterpret_cast<void*>
(virtualThunk(static_cast<MyThread*>(t), i));
} }
} }
@ -5650,7 +5683,7 @@ class MyProcessor: public Processor {
table[index++] = callNodeAddress(t, p) table[index++] = callNodeAddress(t, p)
- reinterpret_cast<uintptr_t>(code); - reinterpret_cast<uintptr_t>(code);
table[index++] = w->map()->find(callNodeTarget(t, p)) table[index++] = w->map()->find(callNodeTarget(t, p))
| (static_cast<unsigned>(callNodeFlags(t, node)) << BootShift); | (static_cast<unsigned>(callNodeFlags(t, p)) << BootShift);
} }
} }
@ -5658,7 +5691,8 @@ class MyProcessor: public Processor {
} }
virtual void boot(Thread* t, BootImage* image) { virtual void boot(Thread* t, BootImage* image) {
codeAllocator.base = s->tryAllocateExecutable(ExecutableAreaSizeInBytes); codeAllocator.base = static_cast<uint8_t*>
(s->tryAllocateExecutable(ExecutableAreaSizeInBytes));
codeAllocator.capacity = ExecutableAreaSizeInBytes; codeAllocator.capacity = ExecutableAreaSizeInBytes;
if (image) { if (image) {
@ -5681,7 +5715,11 @@ class MyProcessor: public Processor {
System* s; System* s;
Allocator* allocator; Allocator* allocator;
uint8_t* defaultThunk; uint8_t* defaultThunk;
uint8_t* defaultTailThunk;
uint8_t* defaultVirtualThunk;
uint8_t* tailHelperThunk;
uint8_t* nativeThunk; uint8_t* nativeThunk;
uint8_t* nativeTailThunk;
uint8_t* aioobThunk; uint8_t* aioobThunk;
uint8_t* thunkTable; uint8_t* thunkTable;
unsigned thunkSize; unsigned thunkSize;
@ -5965,11 +6003,23 @@ fixupThunks(MyThread* t, BootImage* image, uint8_t* code)
MyProcessor* p = processor(t); MyProcessor* p = processor(t);
p->defaultThunk = code + image->defaultThunk; p->defaultThunk = code + image->defaultThunk;
p->defaultTailThunk = code + image->defaultTailThunk;
updateCall(t, LongCall, false, code + image->compileMethodCall, updateCall(t, LongCall, false, code + image->compileMethodCall,
voidPointer(::compileMethod)); voidPointer(::compileMethod));
p->defaultVirtualThunk = code + image->defaultVirtualThunk;
updateCall(t, LongCall, false, code + image->compileVirtualMethodCall,
voidPointer(::compileVirtualMethod));
p->tailHelperThunk = code + image->tailHelperThunk;
updateCall(t, LongCall, false, code + image->tailCallCall,
voidPointer(::tailCall));
p->nativeThunk = code + image->nativeThunk; p->nativeThunk = code + image->nativeThunk;
p->nativeTailThunk = code + image->nativeTailThunk;
updateCall(t, LongCall, false, code + image->invokeNativeCall, updateCall(t, LongCall, false, code + image->invokeNativeCall,
voidPointer(invokeNative)); voidPointer(invokeNative));
@ -6263,7 +6313,7 @@ compileThunks(MyThread* t, Allocator* allocator, MyProcessor* p,
if (image) { if (image) {
image->tailHelperThunk = p->tailHelperThunk - imageBase; image->tailHelperThunk = p->tailHelperThunk - imageBase;
image->tailHelperCall = static_cast<uint8_t*>(call) - imageBase; image->tailCallCall = static_cast<uint8_t*>(call) - imageBase;
} }
} }
@ -6344,12 +6394,36 @@ defaultThunk(MyThread* t)
return reinterpret_cast<uintptr_t>(processor(t)->defaultThunk); return reinterpret_cast<uintptr_t>(processor(t)->defaultThunk);
} }
uintptr_t
defaultTailThunk(MyThread* t)
{
return reinterpret_cast<uintptr_t>(processor(t)->defaultTailThunk);
}
uintptr_t
defaultVirtualThunk(MyThread* t)
{
return reinterpret_cast<uintptr_t>(processor(t)->defaultVirtualThunk);
}
uintptr_t
tailHelperThunk(MyThread* t)
{
return reinterpret_cast<uintptr_t>(processor(t)->tailHelperThunk);
}
uintptr_t uintptr_t
nativeThunk(MyThread* t) nativeThunk(MyThread* t)
{ {
return reinterpret_cast<uintptr_t>(processor(t)->nativeThunk); return reinterpret_cast<uintptr_t>(processor(t)->nativeThunk);
} }
uintptr_t
nativeTailThunk(MyThread* t)
{
return reinterpret_cast<uintptr_t>(processor(t)->nativeTailThunk);
}
uintptr_t uintptr_t
aioobThunk(MyThread* t) aioobThunk(MyThread* t)
{ {
@ -6359,18 +6433,22 @@ aioobThunk(MyThread* t)
uintptr_t uintptr_t
compileVirtualThunk(MyThread* t, unsigned index) compileVirtualThunk(MyThread* t, unsigned index)
{ {
Context context; Context context(t);
Assembler* a = context.assembler; Assembler* a = context.assembler;
Assembler::Constant indexConstant(index); ResolvedPromise indexPromise(index);
Assembler::Constant indexConstant(&indexPromise);
Assembler::Register indexRegister(t->arch->returnHigh()); Assembler::Register indexRegister(t->arch->returnHigh());
a->apply(Move, BytesPerWord, ConstantOperand, &indexConstant, a->apply(Move, BytesPerWord, ConstantOperand, &indexConstant,
BytesPerWord, ConstantOperand, &indexRegister); BytesPerWord, ConstantOperand, &indexRegister);
Assembler::Constant thunk(defaultVirtualThunk(t)); ResolvedPromise defaultVirtualThunkPromise(defaultVirtualThunk(t));
Assembler::Constant thunk(&defaultVirtualThunkPromise);
a->apply(Jump, BytesPerWord, ConstantOperand, &thunk); a->apply(Jump, BytesPerWord, ConstantOperand, &thunk);
uint8_t* start = codeAllocator(t)->allocate(a->length()); uint8_t* start = static_cast<uint8_t*>
(codeAllocator(t)->allocate(a->length()));
a->writeTo(start); a->writeTo(start);
} }

View File

@ -286,6 +286,13 @@ intersect(const SiteMask& a, const SiteMask& b)
intersectFrameIndexes(a.frameIndex, b.frameIndex)); intersectFrameIndexes(a.frameIndex, b.frameIndex));
} }
bool
valid(const SiteMask& a)
{
return a.typeMask
and ((a.typeMask & ~(1 << RegisterOperand)) or a.registerMask);
}
class Value: public Compiler::Operand { class Value: public Compiler::Operand {
public: public:
Value(Site* site, Site* target): Value(Site* site, Site* target):
@ -1879,12 +1886,25 @@ release(Context* c, Resource* resource, Value* value UNUSED, Site* site UNUSED)
class SingleRead: public Read { class SingleRead: public Read {
public: public:
SingleRead(const SiteMask& mask): SingleRead(const SiteMask& mask, Value* successor):
next_(0), mask(mask) next_(0), mask(mask), successor(successor)
{ } { }
virtual bool intersect(SiteMask* mask) { virtual bool intersect(SiteMask* mask) {
*mask = ::intersect(*mask, this->mask); SiteMask result = ::intersect(*mask, this->mask);
if (successor) {
Read* r = live(successor);
if (r) {
SiteMask intersection = result;
bool valid = r->intersect(&intersection);
if (valid and ::valid(intersection)) {
result = intersection;
}
}
}
*mask = result;
return true; return true;
} }
@ -1904,15 +1924,16 @@ class SingleRead: public Read {
Read* next_; Read* next_;
SiteMask mask; SiteMask mask;
Value* successor;
}; };
Read* Read*
read(Context* c, const SiteMask& mask) read(Context* c, const SiteMask& mask, Value* successor = 0)
{ {
assert(c, (mask.typeMask != 1 << MemoryOperand) or mask.frameIndex >= 0); assert(c, (mask.typeMask != 1 << MemoryOperand) or mask.frameIndex >= 0);
return new (c->zone->allocate(sizeof(SingleRead))) return new (c->zone->allocate(sizeof(SingleRead)))
SingleRead(mask); SingleRead(mask, successor);
} }
Read* Read*
@ -2681,15 +2702,19 @@ class MoveEvent: public Event {
{ {
assert(c, srcSelectSize <= srcSize); assert(c, srcSelectSize <= srcSize);
addRead(c, this, src, read(c, srcLowMask)); bool noop = srcSelectSize >= dstSize;
if (srcSelectSize > BytesPerWord) {
maybeSplit(c, src);
addRead(c, this, src->high, read(c, srcHighMask));
}
if (dstSize > BytesPerWord) { if (dstSize > BytesPerWord) {
grow(c, dst); grow(c, dst);
} }
addRead(c, this, src, read(c, srcLowMask, noop ? dst : 0));
if (srcSelectSize > BytesPerWord) {
maybeSplit(c, src);
addRead(c, this, src->high, read
(c, srcHighMask,
noop and dstSize > BytesPerWord ? dst->high : 0));
}
} }
virtual const char* name() { virtual const char* name() {
@ -2966,14 +2991,17 @@ class CombineEvent: public Event {
addRead(c, this, first->high, read(c, firstHighMask)); addRead(c, this, first->high, read(c, firstHighMask));
} }
addRead(c, this, second, read(c, secondLowMask));
if (secondSize > BytesPerWord) {
addRead(c, this, second->high, read(c, secondHighMask));
}
if (resultSize > BytesPerWord) { if (resultSize > BytesPerWord) {
grow(c, result); grow(c, result);
} }
bool condensed = c->arch->condensedAddressing();
addRead(c, this, second, read(c, secondLowMask, condensed ? result : 0));
if (secondSize > BytesPerWord) {
addRead(c, this, second->high, read
(c, secondHighMask, condensed ? result->high : 0));
}
} }
virtual const char* name() { virtual const char* name() {
@ -3367,10 +3395,13 @@ class TranslateEvent: public Event {
Event(c), type(type), size(size), value(value), result(result), Event(c), type(type), size(size), value(value), result(result),
resultLowMask(resultLowMask), resultHighMask(resultHighMask) resultLowMask(resultLowMask), resultHighMask(resultHighMask)
{ {
addRead(c, this, value, read(c, valueLowMask)); bool condensed = c->arch->condensedAddressing();
addRead(c, this, value, read(c, valueLowMask, condensed ? result : 0));
if (size > BytesPerWord) { if (size > BytesPerWord) {
addRead(c, this, value->high, read(c, valueHighMask));
grow(c, result); grow(c, result);
addRead(c, this, value->high, read
(c, valueHighMask, condensed ? result->high : 0));
} }
} }

View File

@ -60,8 +60,7 @@ class Compiler {
Operand* index = 0, Operand* index = 0,
unsigned scale = 1) = 0; unsigned scale = 1) = 0;
virtual Operand* thread() = 0; virtual Operand* register_(int number) = 0;
virtual Operand* stack() = 0;
virtual void freezeRegister(int number, Operand* value) = 0; virtual void freezeRegister(int number, Operand* value) = 0;
virtual void thawRegister(int number) = 0; virtual void thawRegister(int number) = 0;

View File

@ -571,7 +571,7 @@ class MySystem: public System {
} }
virtual void freeExecutable(const void* p, unsigned sizeInBytes) { virtual void freeExecutable(const void* p, unsigned sizeInBytes) {
munmap(p, sizeInBytes); munmap(const_cast<void*>(p), sizeInBytes);
} }
virtual bool success(Status s) { virtual bool success(Status s) {
@ -766,11 +766,6 @@ class MySystem: public System {
registerHandler(0, VisitSignalIndex); registerHandler(0, VisitSignalIndex);
system = 0; system = 0;
if (executableArea) {
int r UNUSED = munmap(executableArea, ExecutableAreaSizeInBytes);
assert(this, r == 0);
}
::free(this); ::free(this);
} }

View File

@ -1713,6 +1713,13 @@ class MyArchitecture: public Assembler::Architecture {
return index + 3; return index + 3;
} }
virtual bool matchCall(void* returnAddress, void* target) {
uint32_t* instruction = static_cast<uint32_t*>(returnAddress) - 1;
return *instruction == bl(static_cast<uint8_t*>(target)
- reinterpret_cast<uint8_t*>(instruction));
}
virtual void updateCall(UnaryOperation op UNUSED, virtual void updateCall(UnaryOperation op UNUSED,
bool assertAlignment UNUSED, void* returnAddress, bool assertAlignment UNUSED, void* returnAddress,
void* newTarget) void* newTarget)

View File

@ -2057,9 +2057,16 @@ class MyArchitecture: public Assembler::Architecture {
} }
} }
virtual void updateCall(UnaryOperation op UNUSED, virtual bool matchCall(void* returnAddress, void* target) {
bool assertAlignment UNUSED, void* returnAddress, uint8_t* instruction = static_cast<uint8_t*>(returnAddress) - 5;
void* newTarget) int32_t actualOffset; memcpy(&actualOffset, instruction + 1, 4);
void* actualTarget = static_cast<uint8_t*>(returnAddress) + actualOffset;
return *instruction == 0xE8 and actualTarget == target;
}
virtual void updateCall(UnaryOperation op, bool assertAlignment UNUSED,
void* returnAddress, void* newTarget)
{ {
if (BytesPerWord == 4 or op == Call or op == Jump) { if (BytesPerWord == 4 or op == Call or op == Jump) {
uint8_t* instruction = static_cast<uint8_t*>(returnAddress) - 5; uint8_t* instruction = static_cast<uint8_t*>(returnAddress) - 5;