mirror of
https://github.com/corda/corda.git
synced 2025-01-22 12:28:11 +00:00
enable Linux PowerPC build
This commit is contained in:
parent
944e223335
commit
9e1ee7e974
6
makefile
6
makefile
@ -3,7 +3,11 @@ MAKEFLAGS = -s
|
|||||||
name = avian
|
name = avian
|
||||||
version = 0.4
|
version = 0.4
|
||||||
|
|
||||||
build-arch := $(shell uname -m | sed 's/^i.86$$/i386/' | sed 's/^arm.*$$/arm/')
|
build-arch := $(shell uname -m \
|
||||||
|
| sed 's/^i.86$$/i386/' \
|
||||||
|
| sed 's/^arm.*$$/arm/' \
|
||||||
|
| sed 's/ppc/powerpc/')
|
||||||
|
|
||||||
ifeq (Power,$(filter Power,$(build-arch)))
|
ifeq (Power,$(filter Power,$(build-arch)))
|
||||||
build-arch = powerpc
|
build-arch = powerpc
|
||||||
endif
|
endif
|
||||||
|
@ -35,6 +35,7 @@
|
|||||||
#define EV_CURRENT 1
|
#define EV_CURRENT 1
|
||||||
|
|
||||||
#define ELFDATA2LSB 1
|
#define ELFDATA2LSB 1
|
||||||
|
#define ELFDATA2MSB 2
|
||||||
|
|
||||||
#define ELFOSABI_SYSV 0
|
#define ELFOSABI_SYSV 0
|
||||||
|
|
||||||
@ -43,6 +44,7 @@
|
|||||||
#define EM_386 3
|
#define EM_386 3
|
||||||
#define EM_X86_64 62
|
#define EM_X86_64 62
|
||||||
#define EM_ARM 40
|
#define EM_ARM 40
|
||||||
|
#define EM_PPC 20
|
||||||
|
|
||||||
#define SHT_PROGBITS 1
|
#define SHT_PROGBITS 1
|
||||||
#define SHT_SYMTAB 2
|
#define SHT_SYMTAB 2
|
||||||
@ -77,7 +79,6 @@
|
|||||||
# error
|
# error
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#define Data ELFDATA2LSB
|
|
||||||
#define OSABI ELFOSABI_SYSV
|
#define OSABI ELFOSABI_SYSV
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
@ -178,7 +179,7 @@ void
|
|||||||
writeObject(const uint8_t* data, unsigned size, FILE* out,
|
writeObject(const uint8_t* data, unsigned size, FILE* out,
|
||||||
const char* startName, const char* endName,
|
const char* startName, const char* endName,
|
||||||
const char* sectionName, unsigned sectionFlags,
|
const char* sectionName, unsigned sectionFlags,
|
||||||
unsigned alignment, int machine)
|
unsigned alignment, int machine, int encoding)
|
||||||
{
|
{
|
||||||
const unsigned sectionCount = 5;
|
const unsigned sectionCount = 5;
|
||||||
const unsigned symbolCount = 2;
|
const unsigned symbolCount = 2;
|
||||||
@ -222,7 +223,7 @@ writeObject(const uint8_t* data, unsigned size, FILE* out,
|
|||||||
fileHeader.e_ident[EI_MAG2] = ELFMAG2;
|
fileHeader.e_ident[EI_MAG2] = ELFMAG2;
|
||||||
fileHeader.e_ident[EI_MAG3] = ELFMAG3;
|
fileHeader.e_ident[EI_MAG3] = ELFMAG3;
|
||||||
fileHeader.e_ident[EI_CLASS] = Class;
|
fileHeader.e_ident[EI_CLASS] = Class;
|
||||||
fileHeader.e_ident[EI_DATA] = Data;
|
fileHeader.e_ident[EI_DATA] = encoding;
|
||||||
fileHeader.e_ident[EI_VERSION] = EV_CURRENT;
|
fileHeader.e_ident[EI_VERSION] = EV_CURRENT;
|
||||||
fileHeader.e_ident[EI_OSABI] = OSABI;
|
fileHeader.e_ident[EI_OSABI] = OSABI;
|
||||||
fileHeader.e_ident[EI_ABIVERSION] = 0;
|
fileHeader.e_ident[EI_ABIVERSION] = 0;
|
||||||
@ -349,12 +350,19 @@ MAKE_NAME(writeElf, BITS_PER_WORD, Object)
|
|||||||
bool writable, bool executable)
|
bool writable, bool executable)
|
||||||
{
|
{
|
||||||
int machine;
|
int machine;
|
||||||
|
int encoding;
|
||||||
if (strcmp(architecture, "x86_64") == 0) {
|
if (strcmp(architecture, "x86_64") == 0) {
|
||||||
machine = EM_X86_64;
|
machine = EM_X86_64;
|
||||||
|
encoding = ELFDATA2LSB;
|
||||||
} else if (strcmp(architecture, "i386") == 0) {
|
} else if (strcmp(architecture, "i386") == 0) {
|
||||||
machine = EM_386;
|
machine = EM_386;
|
||||||
|
encoding = ELFDATA2LSB;
|
||||||
} else if (strcmp(architecture, "arm") == 0) {
|
} else if (strcmp(architecture, "arm") == 0) {
|
||||||
machine = EM_ARM;
|
machine = EM_ARM;
|
||||||
|
encoding = ELFDATA2LSB;
|
||||||
|
} else if (strcmp(architecture, "powerpc") == 0) {
|
||||||
|
machine = EM_PPC;
|
||||||
|
encoding = ELFDATA2MSB;
|
||||||
} else {
|
} else {
|
||||||
fprintf(stderr, "unsupported architecture: %s\n", architecture);
|
fprintf(stderr, "unsupported architecture: %s\n", architecture);
|
||||||
return false;
|
return false;
|
||||||
@ -376,7 +384,7 @@ MAKE_NAME(writeElf, BITS_PER_WORD, Object)
|
|||||||
}
|
}
|
||||||
|
|
||||||
writeObject(data, size, out, startName, endName, sectionName, sectionFlags,
|
writeObject(data, size, out, startName, endName, sectionName, sectionFlags,
|
||||||
alignment, machine);
|
alignment, machine, encoding);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -73,8 +73,10 @@ writeObject(uint8_t* data, unsigned size, FILE* out, const char* startName,
|
|||||||
success = writeElf64Object
|
success = writeElf64Object
|
||||||
(data, size, out, startName, endName, architecture, alignment,
|
(data, size, out, startName, endName, architecture, alignment,
|
||||||
writable, executable);
|
writable, executable);
|
||||||
} else if (strcmp("i386", architecture) == 0 ||
|
} else if (strcmp("i386", architecture) == 0
|
||||||
strcmp("arm", architecture) == 0) {
|
or strcmp("arm", architecture) == 0
|
||||||
|
or strcmp("powerpc", architecture) == 0)
|
||||||
|
{
|
||||||
found = true;
|
found = true;
|
||||||
success = writeElf32Object
|
success = writeElf32Object
|
||||||
(data, size, out, startName, endName, architecture, alignment,
|
(data, size, out, startName, endName, architecture, alignment,
|
||||||
|
@ -88,7 +88,7 @@ alias(void* p, unsigned offset)
|
|||||||
# define ARCH_x86_32
|
# define ARCH_x86_32
|
||||||
# elif defined __x86_64__
|
# elif defined __x86_64__
|
||||||
# define ARCH_x86_64
|
# define ARCH_x86_64
|
||||||
# elif defined __POWERPC__
|
# elif (defined __POWERPC__) || (defined __powerpc__)
|
||||||
# define ARCH_powerpc
|
# define ARCH_powerpc
|
||||||
# elif defined __arm__
|
# elif defined __arm__
|
||||||
# define ARCH_arm
|
# define ARCH_arm
|
||||||
|
@ -13,17 +13,22 @@
|
|||||||
.text
|
.text
|
||||||
|
|
||||||
#define BYTES_PER_WORD 4
|
#define BYTES_PER_WORD 4
|
||||||
#define LINKAGE_AREA 6
|
|
||||||
#define ARGUMENT_BASE BYTES_PER_WORD * LINKAGE_AREA
|
|
||||||
|
|
||||||
#define LOCAL(x) L##x
|
|
||||||
|
|
||||||
#ifdef __APPLE__
|
#ifdef __APPLE__
|
||||||
# define GLOBAL(x) _##x
|
# define GLOBAL(x) _##x
|
||||||
|
# define LOCAL(x) L##x
|
||||||
|
# define LINKAGE_AREA 6
|
||||||
|
# define RETURN_ADDRESS_OFFSET 8
|
||||||
#else
|
#else
|
||||||
# define GLOBAL(x) x
|
# define GLOBAL(x) x
|
||||||
|
# define LOCAL(x) .L##x
|
||||||
|
# define LINKAGE_AREA 2
|
||||||
|
# define RETURN_ADDRESS_OFFSET 4
|
||||||
|
# include "powerpc-regs.S"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#define ARGUMENT_BASE BYTES_PER_WORD * LINKAGE_AREA
|
||||||
|
|
||||||
#define THREAD_STACK 2148
|
#define THREAD_STACK 2148
|
||||||
#define THREAD_CONTINUATION 2156
|
#define THREAD_CONTINUATION 2156
|
||||||
#define THREAD_EXCEPTION 44
|
#define THREAD_EXCEPTION 44
|
||||||
@ -42,7 +47,7 @@
|
|||||||
GLOBAL(vmInvoke):
|
GLOBAL(vmInvoke):
|
||||||
// save return address
|
// save return address
|
||||||
mflr r0
|
mflr r0
|
||||||
stw r0,8(r1)
|
stw r0,RETURN_ADDRESS_OFFSET(r1)
|
||||||
|
|
||||||
// r3: thread
|
// r3: thread
|
||||||
// r4: function
|
// r4: function
|
||||||
@ -53,10 +58,8 @@ GLOBAL(vmInvoke):
|
|||||||
|
|
||||||
// r9: temporary
|
// r9: temporary
|
||||||
|
|
||||||
// save return type
|
// allocate stack space, adding room for callee-saved registers and
|
||||||
stw r8,44(r1)
|
// return type
|
||||||
|
|
||||||
// allocate stack space, adding room for callee-saved registers
|
|
||||||
subfic r9,r7,-80
|
subfic r9,r7,-80
|
||||||
stwux r1,r1,r9
|
stwux r1,r1,r9
|
||||||
|
|
||||||
@ -83,6 +86,9 @@ GLOBAL(vmInvoke):
|
|||||||
stw r30,68(r9)
|
stw r30,68(r9)
|
||||||
stw r31,72(r9)
|
stw r31,72(r9)
|
||||||
|
|
||||||
|
// save return type
|
||||||
|
stw r8,74(r9)
|
||||||
|
|
||||||
// we use r13 to hold the thread pointer, by convention
|
// we use r13 to hold the thread pointer, by convention
|
||||||
mr r13,r3
|
mr r13,r3
|
||||||
|
|
||||||
@ -213,11 +219,11 @@ LOCAL(vmInvoke_exit):
|
|||||||
lwz r31,72(r9)
|
lwz r31,72(r9)
|
||||||
|
|
||||||
// handle return value based on expected type
|
// handle return value based on expected type
|
||||||
lwz r8,44(r1)
|
lwz r8,74(r9)
|
||||||
|
|
||||||
LOCAL(vmInvoke_return):
|
LOCAL(vmInvoke_return):
|
||||||
// load return address
|
// load return address
|
||||||
lwz r0,8(r1)
|
lwz r0,RETURN_ADDRESS_OFFSET(r1)
|
||||||
mtlr r0
|
mtlr r0
|
||||||
|
|
||||||
// return
|
// return
|
||||||
|
@ -2171,9 +2171,9 @@ class MemorySite: public Site {
|
|||||||
}
|
}
|
||||||
|
|
||||||
virtual Site* makeNextWord(Context* c, unsigned index) {
|
virtual Site* makeNextWord(Context* c, unsigned index) {
|
||||||
// todo: endianness?
|
|
||||||
return memorySite
|
return memorySite
|
||||||
(c, base, offset + (index == 1 ? BytesPerWord : -BytesPerWord),
|
(c, base, offset + ((index == 1) xor c->arch->bigEndian()
|
||||||
|
? BytesPerWord : -BytesPerWord),
|
||||||
this->index, scale);
|
this->index, scale);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2184,12 +2184,11 @@ class MemorySite: public Site {
|
|||||||
}
|
}
|
||||||
|
|
||||||
virtual SiteMask nextWordMask(Context* c, unsigned index) {
|
virtual SiteMask nextWordMask(Context* c, unsigned index) {
|
||||||
// todo: endianness?
|
|
||||||
int frameIndex;
|
int frameIndex;
|
||||||
if (base == c->arch->stack()) {
|
if (base == c->arch->stack()) {
|
||||||
assert(c, this->index == NoRegister);
|
assert(c, this->index == NoRegister);
|
||||||
frameIndex = static_cast<int>(offsetToFrameIndex(c, offset))
|
frameIndex = static_cast<int>(offsetToFrameIndex(c, offset))
|
||||||
+ (index == 1 ? 1 : -1);
|
+ ((index == 1) xor c->arch->bigEndian() ? 1 : -1);
|
||||||
} else {
|
} else {
|
||||||
frameIndex = NoFrameIndex;
|
frameIndex = NoFrameIndex;
|
||||||
}
|
}
|
||||||
|
64
src/powerpc-regs.S
Normal file
64
src/powerpc-regs.S
Normal file
@ -0,0 +1,64 @@
|
|||||||
|
#define r0 0
|
||||||
|
#define r1 1
|
||||||
|
#define r2 2
|
||||||
|
#define r3 3
|
||||||
|
#define r4 4
|
||||||
|
#define r5 5
|
||||||
|
#define r6 6
|
||||||
|
#define r7 7
|
||||||
|
#define r8 8
|
||||||
|
#define r9 9
|
||||||
|
#define r10 10
|
||||||
|
#define r11 11
|
||||||
|
#define r12 12
|
||||||
|
#define r13 13
|
||||||
|
#define r14 14
|
||||||
|
#define r15 15
|
||||||
|
#define r16 16
|
||||||
|
#define r17 17
|
||||||
|
#define r18 18
|
||||||
|
#define r19 19
|
||||||
|
#define r20 20
|
||||||
|
#define r21 21
|
||||||
|
#define r22 22
|
||||||
|
#define r23 23
|
||||||
|
#define r24 24
|
||||||
|
#define r25 25
|
||||||
|
#define r26 26
|
||||||
|
#define r27 27
|
||||||
|
#define r28 28
|
||||||
|
#define r29 29
|
||||||
|
#define r30 30
|
||||||
|
#define r31 31
|
||||||
|
#define f0 0
|
||||||
|
#define f1 1
|
||||||
|
#define f2 2
|
||||||
|
#define f3 3
|
||||||
|
#define f4 4
|
||||||
|
#define f5 5
|
||||||
|
#define f6 6
|
||||||
|
#define f7 7
|
||||||
|
#define f8 8
|
||||||
|
#define f9 9
|
||||||
|
#define f10 10
|
||||||
|
#define f11 11
|
||||||
|
#define f12 12
|
||||||
|
#define f13 13
|
||||||
|
#define f14 14
|
||||||
|
#define f15 15
|
||||||
|
#define f16 16
|
||||||
|
#define f17 17
|
||||||
|
#define f18 18
|
||||||
|
#define f19 19
|
||||||
|
#define f20 20
|
||||||
|
#define f21 21
|
||||||
|
#define f22 22
|
||||||
|
#define f23 23
|
||||||
|
#define f24 24
|
||||||
|
#define f25 25
|
||||||
|
#define f26 26
|
||||||
|
#define f27 27
|
||||||
|
#define f28 28
|
||||||
|
#define f29 29
|
||||||
|
#define f30 30
|
||||||
|
#define f31 31
|
@ -13,22 +13,28 @@
|
|||||||
.text
|
.text
|
||||||
|
|
||||||
#define BYTES_PER_WORD 4
|
#define BYTES_PER_WORD 4
|
||||||
#define LINKAGE_AREA 6
|
|
||||||
#define GPR_COUNT 8
|
#define GPR_COUNT 8
|
||||||
#define MEMORY_BASE BYTES_PER_WORD * (LINKAGE_AREA + GPR_COUNT)
|
|
||||||
#define LOCAL(x) L##x
|
|
||||||
|
|
||||||
#ifdef __APPLE__
|
#ifdef __APPLE__
|
||||||
# define GLOBAL(x) _##x
|
# define GLOBAL(x) _##x
|
||||||
|
# define LOCAL(x) L##x
|
||||||
|
# define LINKAGE_AREA 6
|
||||||
|
# define MEMORY_BASE BYTES_PER_WORD * (LINKAGE_AREA + GPR_COUNT)
|
||||||
|
# define RETURN_ADDRESS_OFFSET 8
|
||||||
#else
|
#else
|
||||||
# define GLOBAL(x) x
|
# define GLOBAL(x) x
|
||||||
|
# define LOCAL(x) .L##x
|
||||||
|
# define LINKAGE_AREA 2
|
||||||
|
# define MEMORY_BASE BYTES_PER_WORD * LINKAGE_AREA
|
||||||
|
# define RETURN_ADDRESS_OFFSET 4
|
||||||
|
# include "powerpc-regs.S"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
.globl GLOBAL(vmNativeCall)
|
.globl GLOBAL(vmNativeCall)
|
||||||
GLOBAL(vmNativeCall):
|
GLOBAL(vmNativeCall):
|
||||||
// save return address
|
// save return address
|
||||||
mflr r0
|
mflr r0
|
||||||
stw r0,8(r1)
|
stw r0,RETURN_ADDRESS_OFFSET(r1)
|
||||||
|
|
||||||
// r3 aka r13: function
|
// r3 aka r13: function
|
||||||
// r4 : stackTotal
|
// r4 : stackTotal
|
||||||
@ -43,16 +49,25 @@ GLOBAL(vmNativeCall):
|
|||||||
// r17 : temporary
|
// r17 : temporary
|
||||||
// r18 : temporary
|
// r18 : temporary
|
||||||
|
|
||||||
// save registers used for local variables
|
// allocate stack space, adding room for callee-saved registers and
|
||||||
stw r13,24(r1)
|
// scratch space for copying a FP return value into GPRs
|
||||||
stw r14,28(r1)
|
subfic r10,r4,-48
|
||||||
stw r15,32(r1)
|
stwux r1,r1,r10
|
||||||
stw r16,36(r1)
|
|
||||||
stw r17,40(r1)
|
|
||||||
stw r18,44(r1)
|
|
||||||
|
|
||||||
// allocate stack space
|
// save callee-saved registers used for local variables
|
||||||
stwux r1,r1,r4
|
add r10,r4,r1
|
||||||
|
|
||||||
|
// save registers used for local variables
|
||||||
|
stw r13,0(r10)
|
||||||
|
stw r14,4(r10)
|
||||||
|
stw r15,8(r10)
|
||||||
|
stw r16,12(r10)
|
||||||
|
stw r17,16(r10)
|
||||||
|
stw r18,20(r10)
|
||||||
|
stw r19,24(r10)
|
||||||
|
|
||||||
|
// remember where we saved the local variables
|
||||||
|
mr r19,r10
|
||||||
|
|
||||||
// save our argument registers so we can clobber them
|
// save our argument registers so we can clobber them
|
||||||
mr r13,r3
|
mr r13,r3
|
||||||
@ -84,11 +99,13 @@ LOCAL(test):
|
|||||||
lfd f6,40(r8)
|
lfd f6,40(r8)
|
||||||
lfd f7,48(r8)
|
lfd f7,48(r8)
|
||||||
lfd f8,56(r8)
|
lfd f8,56(r8)
|
||||||
|
#ifdef __APPLE__
|
||||||
lfd f9,64(r8)
|
lfd f9,64(r8)
|
||||||
lfd f10,72(r8)
|
lfd f10,72(r8)
|
||||||
lfd f11,80(r8)
|
lfd f11,80(r8)
|
||||||
lfd f12,88(r8)
|
lfd f12,88(r8)
|
||||||
lfd f13,96(r8)
|
lfd f13,96(r8)
|
||||||
|
#endif
|
||||||
|
|
||||||
LOCAL(gpr):
|
LOCAL(gpr):
|
||||||
// do we need to load the general-purpose registers?
|
// do we need to load the general-purpose registers?
|
||||||
@ -128,25 +145,26 @@ LOCAL(float):
|
|||||||
|
|
||||||
LOCAL(copy):
|
LOCAL(copy):
|
||||||
// move floating point return value to GPRs via memory
|
// move floating point return value to GPRs via memory
|
||||||
stfd f1,8(r1)
|
stfd f1,32(r19)
|
||||||
lwz r3,8(r1)
|
lwz r3,32(r19)
|
||||||
lwz r4,12(r1)
|
lwz r4,36(r19)
|
||||||
b LOCAL(exit)
|
b LOCAL(exit)
|
||||||
|
|
||||||
LOCAL(exit):
|
LOCAL(exit):
|
||||||
|
// restore callee-saved registers used for local variables
|
||||||
|
lwz r13,0(r19)
|
||||||
|
lwz r14,4(r19)
|
||||||
|
lwz r15,8(r19)
|
||||||
|
lwz r16,12(r19)
|
||||||
|
lwz r17,16(r19)
|
||||||
|
lwz r18,20(r19)
|
||||||
|
lwz r19,24(r19)
|
||||||
|
|
||||||
// restore stack pointer
|
// restore stack pointer
|
||||||
lwz r1,0(r1)
|
lwz r1,0(r1)
|
||||||
|
|
||||||
// restore registers used for local variables
|
|
||||||
lwz r13,24(r1)
|
|
||||||
lwz r14,28(r1)
|
|
||||||
lwz r15,32(r1)
|
|
||||||
lwz r16,36(r1)
|
|
||||||
lwz r17,40(r1)
|
|
||||||
lwz r18,44(r1)
|
|
||||||
|
|
||||||
// load return address
|
// load return address
|
||||||
lwz r0,8(r1)
|
lwz r0,RETURN_ADDRESS_OFFSET(r1)
|
||||||
mtlr r0
|
mtlr r0
|
||||||
|
|
||||||
// return
|
// return
|
||||||
@ -171,7 +189,7 @@ GLOBAL(vmRun):
|
|||||||
// r5: checkpoint
|
// r5: checkpoint
|
||||||
|
|
||||||
mflr r0
|
mflr r0
|
||||||
stw r0,8(r1)
|
stw r0,RETURN_ADDRESS_OFFSET(r1)
|
||||||
|
|
||||||
stwu r1,-(MEMORY_BASE+88)(r1)
|
stwu r1,-(MEMORY_BASE+88)(r1)
|
||||||
|
|
||||||
@ -226,6 +244,6 @@ GLOBAL(vmRun_returnAddress):
|
|||||||
lwz r31,MEMORY_BASE+72(r1)
|
lwz r31,MEMORY_BASE+72(r1)
|
||||||
|
|
||||||
lwz r1,0(r1)
|
lwz r1,0(r1)
|
||||||
lwz r0,8(r1)
|
lwz r0,RETURN_ADDRESS_OFFSET(r1)
|
||||||
mtlr r0
|
mtlr r0
|
||||||
blr
|
blr
|
||||||
|
@ -162,7 +162,15 @@ carry16(intptr_t v)
|
|||||||
return static_cast<int16_t>(v) < 0 ? 1 : 0;
|
return static_cast<int16_t>(v) < 0 ? 1 : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef __APPLE__
|
||||||
const unsigned FrameFooterSize = 6;
|
const unsigned FrameFooterSize = 6;
|
||||||
|
const unsigned ReturnAddressOffset = 2;
|
||||||
|
const unsigned AlignArguments = false;
|
||||||
|
#else
|
||||||
|
const unsigned FrameFooterSize = 2;
|
||||||
|
const unsigned ReturnAddressOffset = 1;
|
||||||
|
const unsigned AlignArguments = true;
|
||||||
|
#endif
|
||||||
|
|
||||||
const unsigned StackAlignmentInBytes = 16;
|
const unsigned StackAlignmentInBytes = 16;
|
||||||
const unsigned StackAlignmentInWords = StackAlignmentInBytes / BytesPerWord;
|
const unsigned StackAlignmentInWords = StackAlignmentInBytes / BytesPerWord;
|
||||||
@ -1777,7 +1785,7 @@ nextFrame(ArchitectureContext* c UNUSED, int32_t* start, unsigned size,
|
|||||||
// todo: check for and handle tail calls
|
// todo: check for and handle tail calls
|
||||||
}
|
}
|
||||||
|
|
||||||
*ip = static_cast<void**>(*stack)[offset + 2];
|
*ip = static_cast<void**>(*stack)[offset + ReturnAddressOffset];
|
||||||
*stack = static_cast<void**>(*stack) + offset;
|
*stack = static_cast<void**>(*stack) + offset;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1940,7 +1948,7 @@ class MyArchitecture: public Assembler::Architecture {
|
|||||||
}
|
}
|
||||||
|
|
||||||
virtual bool argumentAlignment() {
|
virtual bool argumentAlignment() {
|
||||||
return false;
|
return AlignArguments;
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual unsigned argumentRegisterCount() {
|
virtual unsigned argumentRegisterCount() {
|
||||||
@ -2023,7 +2031,7 @@ class MyArchitecture: public Assembler::Architecture {
|
|||||||
}
|
}
|
||||||
|
|
||||||
virtual void* frameIp(void* stack) {
|
virtual void* frameIp(void* stack) {
|
||||||
return stack ? static_cast<void**>(stack)[2] : 0;
|
return stack ? static_cast<void**>(stack)[ReturnAddressOffset] : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual unsigned frameHeaderSize() {
|
virtual unsigned frameHeaderSize() {
|
||||||
@ -2039,7 +2047,7 @@ class MyArchitecture: public Assembler::Architecture {
|
|||||||
}
|
}
|
||||||
|
|
||||||
virtual int returnAddressOffset() {
|
virtual int returnAddressOffset() {
|
||||||
return 8 / BytesPerWord;
|
return ReturnAddressOffset;
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual int framePointerOffset() {
|
virtual int framePointerOffset() {
|
||||||
@ -2261,7 +2269,7 @@ class MyAssembler: public Assembler {
|
|||||||
Register returnAddress(0);
|
Register returnAddress(0);
|
||||||
emit(&c, mflr(returnAddress.low));
|
emit(&c, mflr(returnAddress.low));
|
||||||
|
|
||||||
Memory returnAddressDst(StackRegister, 8);
|
Memory returnAddressDst(StackRegister, ReturnAddressOffset * BytesPerWord);
|
||||||
moveRM(&c, BytesPerWord, &returnAddress, BytesPerWord, &returnAddressDst);
|
moveRM(&c, BytesPerWord, &returnAddress, BytesPerWord, &returnAddressDst);
|
||||||
|
|
||||||
Register stack(StackRegister);
|
Register stack(StackRegister);
|
||||||
@ -2314,7 +2322,7 @@ class MyAssembler: public Assembler {
|
|||||||
Register returnAddress(0);
|
Register returnAddress(0);
|
||||||
emit(&c, mflr(returnAddress.low));
|
emit(&c, mflr(returnAddress.low));
|
||||||
|
|
||||||
Memory returnAddressDst(StackRegister, 8);
|
Memory returnAddressDst(StackRegister, ReturnAddressOffset * BytesPerWord);
|
||||||
moveRM(&c, BytesPerWord, &returnAddress, BytesPerWord, &returnAddressDst);
|
moveRM(&c, BytesPerWord, &returnAddress, BytesPerWord, &returnAddressDst);
|
||||||
|
|
||||||
Register stack(StackRegister);
|
Register stack(StackRegister);
|
||||||
@ -2337,7 +2345,7 @@ class MyAssembler: public Assembler {
|
|||||||
moveMR(&c, BytesPerWord, &stackSrc, BytesPerWord, &stack);
|
moveMR(&c, BytesPerWord, &stackSrc, BytesPerWord, &stack);
|
||||||
|
|
||||||
Register returnAddress(0);
|
Register returnAddress(0);
|
||||||
Memory returnAddressSrc(StackRegister, 8);
|
Memory returnAddressSrc(StackRegister, ReturnAddressOffset * BytesPerWord);
|
||||||
moveMR(&c, BytesPerWord, &returnAddressSrc, BytesPerWord, &returnAddress);
|
moveMR(&c, BytesPerWord, &returnAddressSrc, BytesPerWord, &returnAddress);
|
||||||
|
|
||||||
emit(&c, mtlr(returnAddress.low));
|
emit(&c, mtlr(returnAddress.low));
|
||||||
@ -2351,7 +2359,8 @@ class MyAssembler: public Assembler {
|
|||||||
if (TailCalls) {
|
if (TailCalls) {
|
||||||
if (offset) {
|
if (offset) {
|
||||||
Register tmp(0);
|
Register tmp(0);
|
||||||
Memory returnAddressSrc(StackRegister, 8 + (footprint * BytesPerWord));
|
Memory returnAddressSrc
|
||||||
|
(StackRegister, (ReturnAddressOffset + footprint) * BytesPerWord);
|
||||||
moveMR(&c, BytesPerWord, &returnAddressSrc, BytesPerWord, &tmp);
|
moveMR(&c, BytesPerWord, &returnAddressSrc, BytesPerWord, &tmp);
|
||||||
|
|
||||||
emit(&c, mtlr(tmp.low));
|
emit(&c, mtlr(tmp.low));
|
||||||
@ -2366,7 +2375,8 @@ class MyAssembler: public Assembler {
|
|||||||
assert(&c, offset > 0);
|
assert(&c, offset > 0);
|
||||||
|
|
||||||
Register ras(returnAddressSurrogate);
|
Register ras(returnAddressSurrogate);
|
||||||
Memory dst(StackRegister, 8 + (offset * BytesPerWord));
|
Memory dst
|
||||||
|
(StackRegister, (ReturnAddressOffset + offset) * BytesPerWord);
|
||||||
moveRM(&c, BytesPerWord, &ras, BytesPerWord, &dst);
|
moveRM(&c, BytesPerWord, &ras, BytesPerWord, &dst);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -14,8 +14,6 @@
|
|||||||
#include "types.h"
|
#include "types.h"
|
||||||
#include "common.h"
|
#include "common.h"
|
||||||
|
|
||||||
#define VA_LIST(x) (&(x))
|
|
||||||
|
|
||||||
#ifdef __APPLE__
|
#ifdef __APPLE__
|
||||||
# include "mach/mach_types.h"
|
# include "mach/mach_types.h"
|
||||||
# include "mach/ppc/thread_act.h"
|
# include "mach/ppc/thread_act.h"
|
||||||
@ -45,9 +43,17 @@
|
|||||||
# define LINK_REGISTER(context) \
|
# define LINK_REGISTER(context) \
|
||||||
THREAD_STATE_LINK(context->uc_mcontext->FIELD(ss))
|
THREAD_STATE_LINK(context->uc_mcontext->FIELD(ss))
|
||||||
|
|
||||||
#else
|
#define VA_LIST(x) (&(x))
|
||||||
# error "non-Apple PowerPC-based platforms not yet supported"
|
|
||||||
#endif
|
#else // not __APPLE__
|
||||||
|
# define IP_REGISTER(context) (context->uc_mcontext.regs->gpr[32])
|
||||||
|
# define STACK_REGISTER(context) (context->uc_mcontext.regs->gpr[1])
|
||||||
|
# define THREAD_REGISTER(context) (context->uc_mcontext.regs->gpr[13])
|
||||||
|
# define LINK_REGISTER(context) (context->uc_mcontext.regs->gpr[36])
|
||||||
|
|
||||||
|
#define VA_LIST(x) (x)
|
||||||
|
|
||||||
|
#endif // not __APPLE__
|
||||||
|
|
||||||
extern "C" uint64_t
|
extern "C" uint64_t
|
||||||
vmNativeCall(void* function, unsigned stackTotal, void* memoryTable,
|
vmNativeCall(void* function, unsigned stackTotal, void* memoryTable,
|
||||||
@ -150,13 +156,22 @@ dynamicCall(void* function, uintptr_t* arguments, uint8_t* argumentTypes,
|
|||||||
unsigned argumentCount, unsigned argumentsSize,
|
unsigned argumentCount, unsigned argumentsSize,
|
||||||
unsigned returnType)
|
unsigned returnType)
|
||||||
{
|
{
|
||||||
|
#ifdef __APPLE__
|
||||||
|
# define SKIP(var, count) var += count;
|
||||||
|
# define ALIGN(var)
|
||||||
const unsigned LinkageArea = 24;
|
const unsigned LinkageArea = 24;
|
||||||
|
const unsigned FprCount = 13;
|
||||||
|
#else
|
||||||
|
# define SKIP(var, count)
|
||||||
|
# define ALIGN(var) if (var & 1) ++var;
|
||||||
|
const unsigned LinkageArea = 8;
|
||||||
|
const unsigned FprCount = 8;
|
||||||
|
#endif
|
||||||
|
|
||||||
const unsigned GprCount = 8;
|
const unsigned GprCount = 8;
|
||||||
uintptr_t gprTable[GprCount];
|
uintptr_t gprTable[GprCount];
|
||||||
unsigned gprIndex = 0;
|
unsigned gprIndex = 0;
|
||||||
|
|
||||||
const unsigned FprCount = 13;
|
|
||||||
uint64_t fprTable[FprCount];
|
uint64_t fprTable[FprCount];
|
||||||
unsigned fprIndex = 0;
|
unsigned fprIndex = 0;
|
||||||
|
|
||||||
@ -172,8 +187,8 @@ dynamicCall(void* function, uintptr_t* arguments, uint8_t* argumentTypes,
|
|||||||
double d = bitsToFloat(arguments[ai]);
|
double d = bitsToFloat(arguments[ai]);
|
||||||
memcpy(fprTable + fprIndex, &d, 8);
|
memcpy(fprTable + fprIndex, &d, 8);
|
||||||
++ fprIndex;
|
++ fprIndex;
|
||||||
++ gprIndex;
|
SKIP(gprIndex, 1);
|
||||||
++ stackSkip;
|
SKIP(stackSkip, 1);
|
||||||
} else {
|
} else {
|
||||||
stack[stackIndex++] = arguments[ai];
|
stack[stackIndex++] = arguments[ai];
|
||||||
}
|
}
|
||||||
@ -184,9 +199,10 @@ dynamicCall(void* function, uintptr_t* arguments, uint8_t* argumentTypes,
|
|||||||
if (fprIndex + (8 / BytesPerWord) <= FprCount) {
|
if (fprIndex + (8 / BytesPerWord) <= FprCount) {
|
||||||
memcpy(fprTable + fprIndex, arguments + ai, 8);
|
memcpy(fprTable + fprIndex, arguments + ai, 8);
|
||||||
++ fprIndex;
|
++ fprIndex;
|
||||||
gprIndex += 8 / BytesPerWord;
|
SKIP(gprIndex, 8 / BytesPerWord);
|
||||||
stackSkip += 8 / BytesPerWord;
|
SKIP(stackSkip, 8 / BytesPerWord);
|
||||||
} else {
|
} else {
|
||||||
|
ALIGN(stackIndex);
|
||||||
memcpy(stack + stackIndex, arguments + ai, 8);
|
memcpy(stack + stackIndex, arguments + ai, 8);
|
||||||
stackIndex += 8 / BytesPerWord;
|
stackIndex += 8 / BytesPerWord;
|
||||||
}
|
}
|
||||||
@ -195,10 +211,12 @@ dynamicCall(void* function, uintptr_t* arguments, uint8_t* argumentTypes,
|
|||||||
|
|
||||||
case INT64_TYPE: {
|
case INT64_TYPE: {
|
||||||
if (gprIndex + (8 / BytesPerWord) <= GprCount) {
|
if (gprIndex + (8 / BytesPerWord) <= GprCount) {
|
||||||
|
ALIGN(gprIndex);
|
||||||
memcpy(gprTable + gprIndex, arguments + ai, 8);
|
memcpy(gprTable + gprIndex, arguments + ai, 8);
|
||||||
gprIndex += 8 / BytesPerWord;
|
gprIndex += 8 / BytesPerWord;
|
||||||
stackSkip += 8 / BytesPerWord;
|
SKIP(stackSkip, 8 / BytesPerWord);
|
||||||
} else {
|
} else {
|
||||||
|
ALIGN(stackIndex);
|
||||||
memcpy(stack + stackIndex, arguments + ai, 8);
|
memcpy(stack + stackIndex, arguments + ai, 8);
|
||||||
stackIndex += 8 / BytesPerWord;
|
stackIndex += 8 / BytesPerWord;
|
||||||
}
|
}
|
||||||
@ -208,7 +226,7 @@ dynamicCall(void* function, uintptr_t* arguments, uint8_t* argumentTypes,
|
|||||||
default: {
|
default: {
|
||||||
if (gprIndex < GprCount) {
|
if (gprIndex < GprCount) {
|
||||||
gprTable[gprIndex++] = arguments[ai];
|
gprTable[gprIndex++] = arguments[ai];
|
||||||
++ stackSkip;
|
SKIP(stackSkip, 1);
|
||||||
} else {
|
} else {
|
||||||
stack[stackIndex++] = arguments[ai];
|
stack[stackIndex++] = arguments[ai];
|
||||||
}
|
}
|
||||||
@ -219,8 +237,7 @@ dynamicCall(void* function, uintptr_t* arguments, uint8_t* argumentTypes,
|
|||||||
|
|
||||||
return vmNativeCall
|
return vmNativeCall
|
||||||
(function,
|
(function,
|
||||||
- ((((1 + stackSkip + stackIndex) * BytesPerWord) + LinkageArea + 15)
|
(((1 + stackSkip + stackIndex) * BytesPerWord) + LinkageArea + 15) & -16,
|
||||||
& -16),
|
|
||||||
stack, stackIndex * BytesPerWord,
|
stack, stackIndex * BytesPerWord,
|
||||||
(gprIndex ? gprTable : 0),
|
(gprIndex ? gprTable : 0),
|
||||||
(fprIndex ? fprTable : 0), returnType);
|
(fprIndex ? fprTable : 0), returnType);
|
||||||
|
Loading…
Reference in New Issue
Block a user