progress towards powerpc continuation and tail call support

This commit is contained in:
Joel Dice 2009-05-26 19:02:39 -06:00
parent 9ddbf14b6c
commit 2608a2ee43
9 changed files with 231 additions and 58 deletions

View File

@ -232,7 +232,7 @@ class Assembler {
class Memory: public Operand { class Memory: public Operand {
public: public:
Memory(int base, int offset, int index = NoRegister, unsigned scale = 0): Memory(int base, int offset, int index = NoRegister, unsigned scale = 1):
base(base), offset(offset), index(index), scale(scale) base(base), offset(offset), index(index), scale(scale)
{ } { }
@ -327,6 +327,7 @@ class Assembler {
virtual void saveFrame(unsigned stackOffset, unsigned baseOffset) = 0; virtual void saveFrame(unsigned stackOffset, unsigned baseOffset) = 0;
virtual void pushFrame(unsigned argumentCount, ...) = 0; virtual void pushFrame(unsigned argumentCount, ...) = 0;
virtual void allocateFrame(unsigned footprint) = 0; virtual void allocateFrame(unsigned footprint) = 0;
virtual void adjustFrame(unsigned footprint) = 0;
virtual void popFrame() = 0; virtual void popFrame() = 0;
virtual void popFrameForTailCall(unsigned footprint, int offset, virtual void popFrameForTailCall(unsigned footprint, int offset,
int returnAddressSurrogate, int returnAddressSurrogate,

View File

@ -30,7 +30,7 @@
#endif #endif
#if (defined __i386__) || (defined __POWERPC__) #if (defined __i386__) || (defined __POWERPC__)
# define LD "d" # define LD "ld"
# define LLD "lld" # define LLD "lld"
#ifdef __APPLE__ #ifdef __APPLE__
# define ULD "lu" # define ULD "lu"

View File

@ -17,14 +17,28 @@
#define ARGUMENT_BASE BYTES_PER_WORD * LINKAGE_AREA #define ARGUMENT_BASE BYTES_PER_WORD * LINKAGE_AREA
#define LOCAL(x) L##x #define LOCAL(x) L##x
#ifdef __APPLE__ #ifdef __APPLE__
.globl _vmInvoke # define GLOBAL(x) _##x
_vmInvoke:
#else #else
.globl vmInvoke # define GLOBAL(x) x
vmInvoke:
#endif #endif
#define THREAD_CONTINUATION 96
#define THREAD_EXCEPTION 36
#define THREAD_EXCEPTION_STACK_ADJUSTMENT 100
#define THREAD_EXCEPTION_OFFSET 104
#define THREAD_EXCEPTION_HANDLER 108
#define CONTINUATION_NEXT 4
#define CONTINUATION_ADDRESS 16
#define CONTINUATION_RETURN_ADDRESS_OFFSET 20
#define CONTINUATION_FRAME_POINTER_OFFSET 24
#define CONTINUATION_LENGTH 28
#define CONTINUATION_BODY 32
.globl GLOBAL(vmInvoke)
GLOBAL(vmInvoke):
// save return address // save return address
mflr r0 mflr r0
stw r0,8(r1) stw r0,8(r1)
@ -73,24 +87,95 @@ vmInvoke:
// copy arguments into place // copy arguments into place
li r16,0 li r16,0
b LOCAL(test) b LOCAL(vmInvoke_argumentTest)
LOCAL(loop): LOCAL(vmInvoke_argumentLoop):
lwzx r17,r16,r5 lwzx r17,r16,r5
addi r18,r16,ARGUMENT_BASE addi r18,r16,ARGUMENT_BASE
stwx r17,r18,r1 stwx r17,r18,r1
addi r16,r16,BYTES_PER_WORD addi r16,r16,BYTES_PER_WORD
LOCAL(test): LOCAL(vmInvoke_argumentTest):
cmplw r16,r6 cmplw r16,r6
blt LOCAL(loop) blt LOCAL(vmInvoke_argumentLoop)
// load and call function address // load and call function address
mtctr r4 mtctr r4
bctrl bctrl
.globl GLOBAL(vmInvoke_returnAddress)
GLOBAL(vmInvoke_returnAddress):
// restore stack pointer // restore stack pointer
lwz r1,0(r1) lwz r1,0(r1)
#ifdef AVIAN_CONTINUATIONS
// call the next continuation, if any
lwz r5,THREAD_CONTINUATION(r13)
cmplwi r5,0
beq LOCAL(vmInvoke_exit)
lwz r6,CONTINUATION_LENGTH(r5)
slwi r6,r6,2
subfic r6,r6,-80
stwux r1,r1,r6
addi r7,r5,CONTINUATION_BODY
li r8,0
b LOCAL(vmInvoke_continuationTest)
LOCAL(vmInvoke_continuationLoop):
lwzx r9,r7,r8
stwx r9,r1,r8
addi r8,r8,4
LOCAL(vmInvoke_continuationTest):
cmplw r8,r6
ble LOCAL(vmInvoke_continuationLoop)
lwz r7,CONTINUATION_RETURN_ADDRESS_OFFSET(r5)
bl LOCAL(vmInvoke_getPC)
LOCAL(vmInvoke_getPC):
mflr r10
addis r10,r10,ha16(GLOBAL(vmInvoke_returnAddress)-LOCAL(vmInvoke_getPC))
la r10,lo16(GLOBAL(vmInvoke_returnAddress)-LOCAL(vmInvoke_getPC))(r10)
stwx r10,r1,r7
lwz r7,CONTINUATION_FRAME_POINTER_OFFSET(r5)
lwz r8,0(r1)
add r7,r7,r1
stw r8,0(r7)
stw r7,0(r1)
lwz r7,CONTINUATION_NEXT(r5)
stw r7,THREAD_CONTINUATION(r13)
// call the continuation unless we're handling an exception
lwz r7,THREAD_EXCEPTION(r13)
cmpwi r7,0
bne LOCAL(vmInvoke_handleException)
addi r7,r5,CONTINUATION_ADDRESS
mtctr r7
bctr
LOCAL(vmInvoke_handleException):
// we're handling an exception - call the exception handler instead
li r8,0
stw r8,THREAD_EXCEPTION(r13)
lwz r8,THREAD_EXCEPTION_STACK_ADJUSTMENT(r13)
lwz r9,0(r1)
subfic r8,r8,0
stwux r9,r1,r8
lwz r8,THREAD_EXCEPTION_OFFSET(r13)
stwx r7,r1,r8
addi r7,r13,THREAD_EXCEPTION_HANDLER
mtctr r7
bctr
LOCAL(vmInvoke_exit):
#endif // AVIAN_CONTINUATIONS
// restore callee-saved registers // restore callee-saved registers
subi r9,r1,80 subi r9,r1,80
@ -118,23 +203,67 @@ LOCAL(test):
// handle return value based on expected type // handle return value based on expected type
lwz r8,44(r1) lwz r8,44(r1)
LOCAL(void): LOCAL(vmInvoke_void):
cmplwi r8,VOID_TYPE cmplwi r8,VOID_TYPE
bne LOCAL(int64) bne LOCAL(vmInvoke_int64)
b LOCAL(exit) b LOCAL(vmInvoke_return)
LOCAL(int64): LOCAL(vmInvoke_int64):
cmplwi r8,INT64_TYPE cmplwi r8,INT64_TYPE
bne LOCAL(int32) bne LOCAL(vmInvoke_int32)
b LOCAL(exit) b LOCAL(vmInvoke_return)
LOCAL(int32): LOCAL(vmInvoke_int32):
li r3,0 li r3,0
LOCAL(exit): LOCAL(vmInvoke_return):
// load return address // load return address
lwz r0,8(r1) lwz r0,8(r1)
mtlr r0 mtlr r0
// return // return
blr blr
.globl GLOBAL(vmJumpAndInvoke)
GLOBAL(vmJumpAndInvoke):
// r3: thread
// r4: address
// r5: (unused)
// r6: stack
// r7: argumentFootprint
// r8: arguments
// r9: frameSize
stw r6,0(r1)
mr r13,r3
// copy arguments into place
li r9,0
addi r10,r6,ARGUMENT_BASE
b LOCAL(vmJumpAndInvoke_argumentTest)
LOCAL(vmJumpAndInvoke_argumentLoop):
lwzx r11,r8,r9
stwx r11,r10,r9
addi r9,r9,4
LOCAL(vmJumpAndInvoke_argumentTest):
cmplw r9,r7
ble LOCAL(vmJumpAndInvoke_argumentLoop)
subf r7,r6,r9
stw r6,0(r7)
mr r1,r7
// set return address
bl LOCAL(vmJumpAndInvoke_getPC)
LOCAL(vmJumpAndInvoke_getPC):
mflr r10
addis r10,r10,ha16(GLOBAL(vmInvoke_returnAddress)-LOCAL(vmJumpAndInvoke_getPC))
la r10,lo16(GLOBAL(vmInvoke_returnAddress)-LOCAL(vmJumpAndInvoke_getPC))(r10)
mtlr r10
mtctr r4
bctr

View File

@ -148,15 +148,20 @@ LOCAL(vmInvoke_exit):
.globl vmJumpAndInvoke .globl vmJumpAndInvoke
vmJumpAndInvoke: vmJumpAndInvoke:
// %rdi: thread // %rdi: thread
// %rsi: address // %rsi: address
// %rdx: base // %rdx: base
// %rcx: stack // %rcx: stack
// %r8 : argumentFootprint // %r8 : argumentFootprint
// %r9 : arguments // %r9 : arguments
// 8(%rsp): frameSize
movq %rdx,%rbp
movq %rdi,%rbx movq %rdi,%rbx
subq 8(%rsp),%rcx
// set return address // set return address
movq vmInvoke_returnAddress@GOTPCREL(%rip),%r10 movq vmInvoke_returnAddress@GOTPCREL(%rip),%r10
movq %r10,(%rcx) movq %r10,(%rcx)
@ -174,7 +179,6 @@ LOCAL(vmJumpAndInvoke_argumentTest):
cmpq %r8,%r11 cmpq %r8,%r11
jb LOCAL(vmJumpAndInvoke_argumentLoop) jb LOCAL(vmJumpAndInvoke_argumentLoop)
movq %rdx,%rbp
movq %rcx,%rsp movq %rcx,%rsp
jmp *%rsi jmp *%rsi
@ -351,8 +355,12 @@ vmJumpAndInvoke:
// 16(%esp): stack // 16(%esp): stack
// 20(%esp): argumentFootprint // 20(%esp): argumentFootprint
// 24(%esp): arguments // 24(%esp): arguments
// 28(%esp): frameSize
movl 12(%esp),%ebp
movl 16(%esp),%ecx movl 16(%esp),%ecx
subl 28(%esp),%ecx
// set return address // set return address
call LOCAL(getPC) call LOCAL(getPC)
@ -377,7 +385,6 @@ LOCAL(vmJumpAndInvoke_argumentTest):
movl 4(%esp),%ebx movl 4(%esp),%ebx
movl 8(%esp),%esi movl 8(%esp),%esi
movl 12(%esp),%ebp
movl %ecx,%esp movl %ecx,%esp
jmp *%esi jmp *%esi

View File

@ -24,14 +24,15 @@ vmInvoke(void* thread, void* function, void* arguments,
extern "C" void extern "C" void
vmJumpAndInvoke(void* thread, void* function, void* base, void* stack, vmJumpAndInvoke(void* thread, void* function, void* base, void* stack,
unsigned argumentFootprint, uintptr_t* arguments); unsigned argumentFootprint, uintptr_t* arguments,
unsigned frameSize);
extern "C" void extern "C" void
vmCall(); vmCall();
namespace { namespace {
const bool DebugCompile = false; const bool DebugCompile = true;
const bool DebugNatives = false; const bool DebugNatives = false;
const bool DebugCallTable = false; const bool DebugCallTable = false;
const bool DebugMethodTree = false; const bool DebugMethodTree = false;
@ -5413,13 +5414,13 @@ jumpAndInvoke(MyThread* t, object method, void* base, void* stack,
vmJumpAndInvoke vmJumpAndInvoke
(t, reinterpret_cast<void*>(methodAddress(t, method)), (t, reinterpret_cast<void*>(methodAddress(t, method)),
base, base,
static_cast<void**>(stack) stack,
+ oldArgumentFootprint
- t->arch->argumentFootprint(argumentCount)
- t->arch->frameFooterSize()
- t->arch->frameReturnAddressSize(),
argumentCount * BytesPerWord, argumentCount * BytesPerWord,
arguments); arguments,
(oldArgumentFootprint
- t->arch->argumentFootprint(argumentCount)
- t->arch->frameFooterSize()
- t->arch->frameReturnAddressSize()) * BytesPerWord);
} }
void void

View File

@ -384,13 +384,13 @@ unsigned
RegisterResource::toString(Context* c, char* buffer, unsigned bufferSize) RegisterResource::toString(Context* c, char* buffer, unsigned bufferSize)
{ {
return snprintf return snprintf
(buffer, bufferSize, "register %"LD, this - c->registerResources); (buffer, bufferSize, "register %d", this - c->registerResources);
} }
unsigned unsigned
FrameResource::toString(Context* c, char* buffer, unsigned bufferSize) FrameResource::toString(Context* c, char* buffer, unsigned bufferSize)
{ {
return snprintf(buffer, bufferSize, "frame %"LD, this - c->frameResources); return snprintf(buffer, bufferSize, "frame %d", this - c->frameResources);
} }
class PoolPromise: public Promise { class PoolPromise: public Promise {
@ -679,8 +679,7 @@ offsetToFrameIndex(Context* c, unsigned offset)
{ {
assert(c, static_cast<int> assert(c, static_cast<int>
((offset / BytesPerWord) - c->arch->frameFooterSize()) >= 0); ((offset / BytesPerWord) - c->arch->frameFooterSize()) >= 0);
assert(c, ((offset / BytesPerWord) - c->arch->frameFooterSize())
assert(c, (offset / BytesPerWord) - c->arch->frameFooterSize()
< totalFrameSize(c)); < totalFrameSize(c));
return (offset / BytesPerWord) - c->arch->frameFooterSize(); return (offset / BytesPerWord) - c->arch->frameFooterSize();
@ -689,7 +688,8 @@ offsetToFrameIndex(Context* c, unsigned offset)
unsigned unsigned
frameBase(Context* c) frameBase(Context* c)
{ {
return c->alignedFrameSize - 1 return c->alignedFrameSize
- c->arch->frameReturnAddressSize()
- c->arch->frameFooterSize() - c->arch->frameFooterSize()
+ c->arch->frameHeaderSize(); + c->arch->frameHeaderSize();
} }
@ -1262,7 +1262,7 @@ pickTarget(Context* c, Read* read, bool intersectRead,
} }
} }
} }
best = pickTarget(c, read->value, mask, registerPenalty, best); best = pickTarget(c, read->value, mask, registerPenalty, best);
if (best.cost <= Target::MinimumFrameCost) { if (best.cost <= Target::MinimumFrameCost) {
return best; return best;
@ -2539,14 +2539,8 @@ class CallEvent: public Event {
(stackArgumentFootprint); (stackArgumentFootprint);
if (footprint > c->arch->stackAlignmentInWords()) { if (footprint > c->arch->stackAlignmentInWords()) {
Assembler::Register stack(c->arch->stack()); c->assembler->adjustFrame
ResolvedPromise adjustmentPromise (footprint - c->arch->stackAlignmentInWords());
((footprint - c->arch->stackAlignmentInWords()) * BytesPerWord);
Assembler::Constant adjustmentConstant(&adjustmentPromise);
c->assembler->apply
(Subtract, BytesPerWord, ConstantOperand, &adjustmentConstant,
BytesPerWord, RegisterOperand, &stack,
BytesPerWord, RegisterOperand, &stack);
} }
} }
} }

View File

@ -161,4 +161,6 @@ vmJump:
mtlr r3 mtlr r3
mr r1,r5 mr r1,r5
mr r13,r6 mr r13,r6
mr r4,r7
mr r3,r8
blr blr

View File

@ -65,7 +65,7 @@ inline int sth(int rs, int ra, int i) { return D(44, rs, ra, i); }
inline int sthx(int rs, int ra, int rb) { return X(31, rs, ra, rb, 407, 0); } inline int sthx(int rs, int ra, int rb) { return X(31, rs, ra, rb, 407, 0); }
inline int stw(int rs, int ra, int i) { return D(36, rs, ra, i); } inline int stw(int rs, int ra, int i) { return D(36, rs, ra, i); }
inline int stwu(int rs, int ra, int i) { return D(37, rs, ra, i); } inline int stwu(int rs, int ra, int i) { return D(37, rs, ra, i); }
inline int stwux(int rs, int ra, int i) { return X(31, rs, ra, rb, 183, 0); } inline int stwux(int rs, int ra, int rb) { return X(31, rs, ra, rb, 183, 0); }
inline int stwx(int rs, int ra, int rb) { return X(31, rs, ra, rb, 151, 0); } inline int stwx(int rs, int ra, int rb) { return X(31, rs, ra, rb, 151, 0); }
inline int add(int rt, int ra, int rb) { return XO(31, rt, ra, rb, 0, 266, 0); } inline int add(int rt, int ra, int rb) { return XO(31, rt, ra, rb, 0, 266, 0); }
inline int addc(int rt, int ra, int rb) { return XO(31, rt, ra, rb, 0, 10, 0); } inline int addc(int rt, int ra, int rb) { return XO(31, rt, ra, rb, 0, 10, 0); }
@ -137,7 +137,7 @@ inline int blt(int i) { return bc(12, 0, i, 0); }
inline int bgt(int i) { return bc(12, 1, i, 0); } inline int bgt(int i) { return bc(12, 1, i, 0); }
inline int bge(int i) { return bc(4, 0, i, 0); } inline int bge(int i) { return bc(4, 0, i, 0); }
inline int ble(int i) { return bc(4, 1, i, 0); } inline int ble(int i) { return bc(4, 1, i, 0); }
inline int be(int i) { return bc(12, 2, i, 0); } inline int beq(int i) { return bc(12, 2, i, 0); }
inline int bne(int i) { return bc(4, 2, i, 0); } inline int bne(int i) { return bc(4, 2, i, 0); }
inline int cmpw(int ra, int rb) { return cmp(0, ra, rb); } inline int cmpw(int ra, int rb) { return cmp(0, ra, rb); }
inline int cmplw(int ra, int rb) { return cmpl(0, ra, rb); } inline int cmplw(int ra, int rb) { return cmpl(0, ra, rb); }
@ -160,6 +160,7 @@ carry16(intptr_t v)
const unsigned FrameFooterSize = 6; const unsigned FrameFooterSize = 6;
const unsigned StackAlignmentInBytes = 16; const unsigned StackAlignmentInBytes = 16;
const unsigned StackAlignmentInWords = StackAlignmentInBytes / BytesPerWord;
const int StackRegister = 1; const int StackRegister = 1;
const int ThreadRegister = 13; const int ThreadRegister = 13;
@ -1507,7 +1508,7 @@ jumpIfEqualC(Context* c, unsigned size UNUSED, Assembler::Constant* target)
assert(c, size == BytesPerWord); assert(c, size == BytesPerWord);
appendOffsetTask(c, target->value, offset(c), true); appendOffsetTask(c, target->value, offset(c), true);
issue(c, be(0)); issue(c, beq(0));
} }
void void
@ -1688,6 +1689,14 @@ class MyArchitecture: public Assembler::Architecture {
return (BytesPerWord == 4 ? 3 : NoRegister); return (BytesPerWord == 4 ? 3 : NoRegister);
} }
virtual int virtualCallTarget() {
return 4;
}
virtual int virtualCallIndex() {
return 3;
}
virtual bool condensedAddressing() { virtual bool condensedAddressing() {
return false; return false;
} }
@ -1708,7 +1717,7 @@ class MyArchitecture: public Assembler::Architecture {
} }
} }
virtual unsigned stackPadding(unsigned footprint) { virtual unsigned frameFootprint(unsigned footprint) {
return max(footprint, StackAlignmentInWords); return max(footprint, StackAlignmentInWords);
} }
@ -1726,11 +1735,16 @@ class MyArchitecture: public Assembler::Architecture {
return index + 3; return index + 3;
} }
virtual unsigned stackAlignmentInWords() {
return StackAlignmentInWords;
}
virtual bool matchCall(void* returnAddress, void* target) { virtual bool matchCall(void* returnAddress, void* target) {
uint32_t* instruction = static_cast<uint32_t*>(returnAddress) - 1; uint32_t* instruction = static_cast<uint32_t*>(returnAddress) - 1;
return *instruction == bl(static_cast<uint8_t*>(target) return *instruction == static_cast<uint32_t>
- reinterpret_cast<uint8_t*>(instruction)); (bl(static_cast<uint8_t*>(target)
- reinterpret_cast<uint8_t*>(instruction)));
} }
virtual void updateCall(UnaryOperation op UNUSED, virtual void updateCall(UnaryOperation op UNUSED,
@ -1788,6 +1802,14 @@ class MyArchitecture: public Assembler::Architecture {
return FrameFooterSize; return FrameFooterSize;
} }
virtual int returnAddressOffset() {
return 8 / BytesPerWord;
}
virtual int framePointerOffset() {
return 0;
}
virtual void nextFrame(void** stack, void**) { virtual void nextFrame(void** stack, void**) {
assert(&c, *static_cast<void**>(*stack) != *stack); assert(&c, *static_cast<void**>(*stack) != *stack);
@ -1971,6 +1993,15 @@ class MyAssembler: public Assembler {
moveAndUpdateRM(&c, BytesPerWord, &stack, BytesPerWord, &stackDst); moveAndUpdateRM(&c, BytesPerWord, &stack, BytesPerWord, &stackDst);
} }
virtual void adjustFrame(unsigned footprint) {
Register nextStack(0);
Memory stackSrc(StackRegister, 0);
moveMR(&c, BytesPerWord, &stackSrc, BytesPerWord, &nextStack);
Memory stackDst(StackRegister, -footprint * BytesPerWord);
moveAndUpdateRM(&c, BytesPerWord, &nextStack, BytesPerWord, &stackDst);
}
virtual void popFrame() { virtual void popFrame() {
Register stack(StackRegister); Register stack(StackRegister);
Memory stackSrc(StackRegister, 0); Memory stackSrc(StackRegister, 0);
@ -2024,8 +2055,8 @@ class MyAssembler: public Assembler {
virtual void popFrameAndPopArgumentsAndReturn(unsigned argumentFootprint) { virtual void popFrameAndPopArgumentsAndReturn(unsigned argumentFootprint) {
popFrame(); popFrame();
assert(c, argumentFootprint >= StackAlignmentInWords); assert(&c, argumentFootprint >= StackAlignmentInWords);
assert(c, (argumentFootprint % StackAlignmentInWords) == 0); assert(&c, (argumentFootprint % StackAlignmentInWords) == 0);
if (argumentFootprint > StackAlignmentInWords) { if (argumentFootprint > StackAlignmentInWords) {
Register tmp(0); Register tmp(0);
@ -2049,7 +2080,7 @@ class MyAssembler: public Assembler {
moveMR(&c, BytesPerWord, &stackSrc, BytesPerWord, &tmp1); moveMR(&c, BytesPerWord, &stackSrc, BytesPerWord, &tmp1);
Register tmp2(arch_->returnLow()); Register tmp2(arch_->returnLow());
Memory newStackSrc(ThreadRegister, stackOffsetFromThread); Memory newStackSrc(ThreadRegister, stackOffset);
moveMR(&c, BytesPerWord, &stackSrc, BytesPerWord, &tmp2); moveMR(&c, BytesPerWord, &stackSrc, BytesPerWord, &tmp2);
Register stack(StackRegister); Register stack(StackRegister);

View File

@ -2398,6 +2398,14 @@ class MyAssembler: public Assembler {
BytesPerWord, RegisterOperand, &stack); BytesPerWord, RegisterOperand, &stack);
} }
virtual void adjustFrame(unsigned footprint) {
Register stack(rsp);
Constant footprintConstant(resolved(&c, footprint * BytesPerWord));
apply(Subtract, BytesPerWord, ConstantOperand, &footprintConstant,
BytesPerWord, RegisterOperand, &stack,
BytesPerWord, RegisterOperand, &stack);
}
virtual void popFrame() { virtual void popFrame() {
Register base(rbp); Register base(rbp);
Register stack(rsp); Register stack(rsp);