2009-02-16 08:21:12 -07:00
|
|
|
/* Copyright (c) 2009, 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 "types.h"
|
|
|
|
|
|
|
|
.text
|
|
|
|
|
|
|
|
#define BYTES_PER_WORD 4
|
|
|
|
#define LINKAGE_AREA 6
|
2009-02-16 19:49:28 -07:00
|
|
|
#define ARGUMENT_BASE BYTES_PER_WORD * LINKAGE_AREA
|
2009-02-16 08:21:12 -07:00
|
|
|
|
|
|
|
#define LOCAL(x) L##x
|
2009-05-26 19:02:39 -06:00
|
|
|
|
2009-02-16 08:21:12 -07:00
|
|
|
#ifdef __APPLE__
|
2009-05-26 19:02:39 -06:00
|
|
|
# define GLOBAL(x) _##x
|
2009-02-16 08:21:12 -07:00
|
|
|
#else
|
2009-05-26 19:02:39 -06:00
|
|
|
# define GLOBAL(x) x
|
2009-02-16 08:21:12 -07:00
|
|
|
#endif
|
2009-05-26 19:02:39 -06:00
|
|
|
|
|
|
|
#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):
|
2009-02-16 08:21:12 -07:00
|
|
|
// save return address
|
|
|
|
mflr r0
|
|
|
|
stw r0,8(r1)
|
|
|
|
|
|
|
|
// r3: thread
|
|
|
|
// r4: function
|
2009-02-16 19:49:28 -07:00
|
|
|
// r5: arguments
|
|
|
|
// r6: argumentFootprint
|
2009-02-16 08:21:12 -07:00
|
|
|
// r7: frameSize
|
2009-02-16 19:49:28 -07:00
|
|
|
// r8: returnType
|
2009-02-16 08:21:12 -07:00
|
|
|
|
|
|
|
// r9: temporary
|
|
|
|
|
2009-02-16 19:49:28 -07:00
|
|
|
// save return type
|
|
|
|
stw r8,44(r1)
|
|
|
|
|
2009-02-16 08:21:12 -07:00
|
|
|
// allocate stack space, adding room for callee-saved registers
|
2009-02-24 18:28:05 -07:00
|
|
|
subfic r9,r7,-80
|
2009-02-16 08:21:12 -07:00
|
|
|
stwux r1,r1,r9
|
|
|
|
|
|
|
|
// save callee-saved registers
|
|
|
|
add r9,r7,r1
|
|
|
|
|
|
|
|
stw r13,0(r9)
|
|
|
|
stw r14,4(r9)
|
|
|
|
stw r15,8(r9)
|
|
|
|
stw r16,12(r9)
|
|
|
|
stw r17,16(r9)
|
|
|
|
stw r18,20(r9)
|
|
|
|
stw r19,24(r9)
|
|
|
|
stw r20,28(r9)
|
|
|
|
stw r21,32(r9)
|
|
|
|
stw r22,36(r9)
|
|
|
|
stw r23,40(r9)
|
|
|
|
stw r24,44(r9)
|
|
|
|
stw r25,48(r9)
|
|
|
|
stw r26,52(r9)
|
|
|
|
stw r27,56(r9)
|
|
|
|
stw r28,60(r9)
|
|
|
|
stw r29,64(r9)
|
|
|
|
stw r30,68(r9)
|
|
|
|
stw r31,72(r9)
|
|
|
|
|
|
|
|
// we use r13 to hold the thread pointer, by convention
|
|
|
|
mr r13,r3
|
|
|
|
|
|
|
|
// copy arguments into place
|
|
|
|
li r16,0
|
2009-05-28 18:56:05 -06:00
|
|
|
addi r18,r1,ARGUMENT_BASE
|
2009-05-26 19:02:39 -06:00
|
|
|
b LOCAL(vmInvoke_argumentTest)
|
2009-02-16 08:21:12 -07:00
|
|
|
|
2009-05-26 19:02:39 -06:00
|
|
|
LOCAL(vmInvoke_argumentLoop):
|
2009-02-16 08:21:12 -07:00
|
|
|
lwzx r17,r16,r5
|
2009-05-28 18:56:05 -06:00
|
|
|
stwx r17,r16,r18
|
2009-02-16 08:21:12 -07:00
|
|
|
addi r16,r16,BYTES_PER_WORD
|
|
|
|
|
2009-05-26 19:02:39 -06:00
|
|
|
LOCAL(vmInvoke_argumentTest):
|
2009-02-16 08:21:12 -07:00
|
|
|
cmplw r16,r6
|
2009-05-26 19:02:39 -06:00
|
|
|
blt LOCAL(vmInvoke_argumentLoop)
|
2009-02-16 08:21:12 -07:00
|
|
|
|
|
|
|
// load and call function address
|
|
|
|
mtctr r4
|
|
|
|
bctrl
|
2009-05-28 18:56:05 -06:00
|
|
|
|
|
|
|
LOCAL(vmInvoke_returnAddress):
|
2009-02-16 08:21:12 -07:00
|
|
|
// restore stack pointer
|
|
|
|
lwz r1,0(r1)
|
2009-05-26 19:02:39 -06:00
|
|
|
|
|
|
|
#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
|
2009-05-28 18:56:05 -06:00
|
|
|
subfic r7,r6,-80
|
|
|
|
stwux r1,r1,r7
|
2009-05-26 19:02:39 -06:00
|
|
|
|
|
|
|
addi r7,r5,CONTINUATION_BODY
|
|
|
|
|
|
|
|
li r8,0
|
2009-05-28 18:56:05 -06:00
|
|
|
addi r10,r1,ARGUMENT_BASE
|
2009-05-26 19:02:39 -06:00
|
|
|
b LOCAL(vmInvoke_continuationTest)
|
|
|
|
|
|
|
|
LOCAL(vmInvoke_continuationLoop):
|
|
|
|
lwzx r9,r7,r8
|
2009-05-28 18:56:05 -06:00
|
|
|
stwx r9,r10,r8
|
2009-05-26 19:02:39 -06:00
|
|
|
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
|
2009-05-28 18:56:05 -06:00
|
|
|
la r10,lo16(LOCAL(vmInvoke_returnAddress)-LOCAL(vmInvoke_getPC))(r10)
|
2009-05-26 19:02:39 -06:00
|
|
|
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)
|
2009-05-28 18:56:05 -06:00
|
|
|
lwz r7,CONTINUATION_ADDRESS(r5)
|
2009-05-26 19:02:39 -06:00
|
|
|
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
|
|
|
|
|
2009-05-28 18:56:05 -06:00
|
|
|
lwz r7,THREAD_EXCEPTION_HANDLER(r13)
|
2009-05-26 19:02:39 -06:00
|
|
|
mtctr r7
|
|
|
|
bctr
|
|
|
|
|
|
|
|
LOCAL(vmInvoke_exit):
|
|
|
|
#endif // AVIAN_CONTINUATIONS
|
2009-02-16 08:21:12 -07:00
|
|
|
|
|
|
|
// restore callee-saved registers
|
|
|
|
subi r9,r1,80
|
|
|
|
|
|
|
|
lwz r13,0(r9)
|
|
|
|
lwz r14,4(r9)
|
|
|
|
lwz r15,8(r9)
|
|
|
|
lwz r16,12(r9)
|
|
|
|
lwz r17,16(r9)
|
|
|
|
lwz r18,20(r9)
|
|
|
|
lwz r19,24(r9)
|
|
|
|
lwz r20,28(r9)
|
|
|
|
lwz r21,32(r9)
|
|
|
|
lwz r22,36(r9)
|
|
|
|
lwz r23,40(r9)
|
|
|
|
lwz r24,44(r9)
|
|
|
|
lwz r25,48(r9)
|
|
|
|
lwz r26,52(r9)
|
|
|
|
lwz r27,56(r9)
|
|
|
|
lwz r28,60(r9)
|
|
|
|
lwz r29,64(r9)
|
|
|
|
lwz r30,68(r9)
|
|
|
|
lwz r31,72(r9)
|
|
|
|
|
2009-02-16 19:49:28 -07:00
|
|
|
// handle return value based on expected type
|
|
|
|
lwz r8,44(r1)
|
|
|
|
|
2009-05-26 19:02:39 -06:00
|
|
|
LOCAL(vmInvoke_void):
|
2009-02-18 17:46:07 -07:00
|
|
|
cmplwi r8,VOID_TYPE
|
2009-05-26 19:02:39 -06:00
|
|
|
bne LOCAL(vmInvoke_int64)
|
|
|
|
b LOCAL(vmInvoke_return)
|
2009-02-16 19:49:28 -07:00
|
|
|
|
2009-05-26 19:02:39 -06:00
|
|
|
LOCAL(vmInvoke_int64):
|
2009-02-18 17:46:07 -07:00
|
|
|
cmplwi r8,INT64_TYPE
|
2009-05-26 19:02:39 -06:00
|
|
|
bne LOCAL(vmInvoke_int32)
|
|
|
|
b LOCAL(vmInvoke_return)
|
2009-02-16 19:49:28 -07:00
|
|
|
|
2009-05-26 19:02:39 -06:00
|
|
|
LOCAL(vmInvoke_int32):
|
2009-02-16 19:49:28 -07:00
|
|
|
li r3,0
|
|
|
|
|
2009-05-26 19:02:39 -06:00
|
|
|
LOCAL(vmInvoke_return):
|
2009-02-16 08:21:12 -07:00
|
|
|
// load return address
|
|
|
|
lwz r0,8(r1)
|
|
|
|
mtlr r0
|
|
|
|
|
|
|
|
// return
|
|
|
|
blr
|
2009-05-26 19:02:39 -06:00
|
|
|
|
|
|
|
.globl GLOBAL(vmJumpAndInvoke)
|
|
|
|
GLOBAL(vmJumpAndInvoke):
|
2009-05-28 18:56:05 -06:00
|
|
|
#ifdef AVIAN_CONTINUATIONS
|
2009-05-26 19:02:39 -06:00
|
|
|
// r3: thread
|
|
|
|
// r4: address
|
|
|
|
// r5: (unused)
|
|
|
|
// r6: stack
|
|
|
|
// r7: argumentFootprint
|
|
|
|
// r8: arguments
|
|
|
|
// r9: frameSize
|
|
|
|
|
2009-05-28 18:56:05 -06:00
|
|
|
// restore (pseudo)-stack pointer (we don't want to touch the real
|
|
|
|
// stack pointer, since we haven't copied the arguments yet)
|
|
|
|
lwz r6,0(r6)
|
|
|
|
|
|
|
|
// make everything between r1 and r6 one big stack frame while we
|
|
|
|
// shuffle things around
|
2009-05-26 19:02:39 -06:00
|
|
|
stw r6,0(r1)
|
2009-05-28 18:56:05 -06:00
|
|
|
|
|
|
|
// allocate new frame, adding room for callee-saved registers
|
|
|
|
subfic r10,r9,-80
|
|
|
|
stwux r6,r6,r10
|
2009-05-26 19:02:39 -06:00
|
|
|
|
|
|
|
mr r13,r3
|
|
|
|
|
|
|
|
// copy arguments into place
|
|
|
|
li r9,0
|
2009-05-28 18:56:05 -06:00
|
|
|
addi r11,r6,ARGUMENT_BASE
|
2009-05-26 19:02:39 -06:00
|
|
|
b LOCAL(vmJumpAndInvoke_argumentTest)
|
|
|
|
|
|
|
|
LOCAL(vmJumpAndInvoke_argumentLoop):
|
2009-05-28 18:56:05 -06:00
|
|
|
lwzx r12,r8,r9
|
|
|
|
stwx r12,r11,r9
|
2009-05-26 19:02:39 -06:00
|
|
|
addi r9,r9,4
|
|
|
|
|
|
|
|
LOCAL(vmJumpAndInvoke_argumentTest):
|
|
|
|
cmplw r9,r7
|
|
|
|
ble LOCAL(vmJumpAndInvoke_argumentLoop)
|
|
|
|
|
2009-05-28 18:56:05 -06:00
|
|
|
// the arguments have been copied, so we can set the real stack
|
|
|
|
// pointer now
|
|
|
|
mr r1,r6
|
2009-05-26 19:02:39 -06:00
|
|
|
|
2009-05-28 18:56:05 -06:00
|
|
|
// set return address to vmInvoke_returnAddress
|
2009-05-26 19:02:39 -06:00
|
|
|
bl LOCAL(vmJumpAndInvoke_getPC)
|
|
|
|
|
|
|
|
LOCAL(vmJumpAndInvoke_getPC):
|
|
|
|
mflr r10
|
2009-05-28 18:56:05 -06:00
|
|
|
la r10,lo16(LOCAL(vmInvoke_returnAddress)-LOCAL(vmJumpAndInvoke_getPC))(r10)
|
2009-05-26 19:02:39 -06:00
|
|
|
mtlr r10
|
|
|
|
|
|
|
|
mtctr r4
|
|
|
|
bctr
|
2009-05-28 18:56:05 -06:00
|
|
|
#else // not AVIAN_CONTINUATIONS
|
|
|
|
// vmJumpAndInvoke should only be called when continuations are
|
|
|
|
// enabled
|
|
|
|
trap
|
|
|
|
#endif // not AVIAN_CONTINUATIONS
|