2007-09-24 01:39:03 +00:00
|
|
|
#include "common.h"
|
|
|
|
#include "system.h"
|
|
|
|
#include "constants.h"
|
|
|
|
#include "machine.h"
|
2007-09-26 23:23:03 +00:00
|
|
|
#include "processor.h"
|
|
|
|
#include "process.h"
|
2007-09-24 01:39:03 +00:00
|
|
|
|
|
|
|
using namespace vm;
|
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
extern "C" uint64_t
|
|
|
|
vmInvoke(void* function, void* stack, unsigned stackSize,
|
|
|
|
unsigned returnType);
|
|
|
|
|
2007-09-24 01:39:03 +00:00
|
|
|
namespace {
|
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
const unsigned FrameThread = BytesPerWord * 2;
|
|
|
|
const unsigned FrameMethod = FrameThread + BytesPerWord;
|
|
|
|
const unsigned FrameNext = FrameNext + BytesPerWord;
|
|
|
|
const unsigned FrameFootprint = BytesPerWord * 3;
|
2007-09-25 23:53:11 +00:00
|
|
|
|
2007-09-28 23:41:03 +00:00
|
|
|
class Buffer {
|
2007-09-24 01:39:03 +00:00
|
|
|
public:
|
2007-09-28 23:41:03 +00:00
|
|
|
Buffer(System* s, unsigned minimumCapacity):
|
2007-09-24 01:39:03 +00:00
|
|
|
s(s),
|
2007-09-28 23:41:03 +00:00
|
|
|
data(0),
|
|
|
|
position(0),
|
|
|
|
capacity(0),
|
|
|
|
minimumCapacity(minimumCapacity)
|
2007-09-24 01:39:03 +00:00
|
|
|
{ }
|
|
|
|
|
2007-09-28 23:41:03 +00:00
|
|
|
~Buffer() {
|
|
|
|
if (data) {
|
|
|
|
s->free(data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ensure(unsigned space) {
|
|
|
|
if (position + space > capacity) {
|
|
|
|
unsigned newCapacity = max
|
|
|
|
(position + space, max(minimumCapacity, capacity * 2));
|
|
|
|
uint8_t* newData = static_cast<uint8_t*>(s->allocate(newCapacity));
|
|
|
|
if (data) {
|
|
|
|
memcpy(newData, data, position);
|
|
|
|
s->free(data);
|
2007-09-24 01:39:03 +00:00
|
|
|
}
|
2007-09-28 23:41:03 +00:00
|
|
|
data = newData;
|
2007-09-24 01:39:03 +00:00
|
|
|
}
|
2007-09-28 23:41:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void append(uint8_t v) {
|
|
|
|
ensure(1);
|
|
|
|
data[position++] = v;
|
|
|
|
}
|
2007-09-24 01:39:03 +00:00
|
|
|
|
2007-09-29 20:24:14 +00:00
|
|
|
void append2(uint16_t v) {
|
2007-09-28 23:41:03 +00:00
|
|
|
ensure(2);
|
2007-09-29 20:24:14 +00:00
|
|
|
memcpy(data + position, &v, 2);
|
|
|
|
position += 2;
|
2007-09-24 01:39:03 +00:00
|
|
|
}
|
|
|
|
|
2007-09-27 00:01:38 +00:00
|
|
|
void append4(uint32_t v) {
|
2007-09-28 23:41:03 +00:00
|
|
|
ensure(4);
|
2007-09-29 20:24:14 +00:00
|
|
|
memcpy(data + position, &v, 4);
|
|
|
|
position += 4;
|
2007-09-28 23:41:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void set2(unsigned offset, uint32_t v) {
|
2007-09-29 20:24:14 +00:00
|
|
|
assert(s, offset + 2 <= position);
|
|
|
|
memcpy(data + offset, &v, 2);
|
2007-09-28 23:41:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void set4(unsigned offset, uint32_t v) {
|
2007-09-29 20:24:14 +00:00
|
|
|
assert(s, offset + 4 <= position);
|
|
|
|
memcpy(data + offset, &v, 4);
|
2007-09-28 23:41:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t get2(unsigned offset) {
|
2007-09-29 20:24:14 +00:00
|
|
|
assert(s, offset + 2 <= position);
|
|
|
|
uint16_t v; memcpy(&v, data + offset, 2);
|
|
|
|
return v;
|
2007-09-28 23:41:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t get4(unsigned offset) {
|
2007-09-29 20:24:14 +00:00
|
|
|
assert(s, offset + 4 <= position);
|
|
|
|
uint32_t v; memcpy(&v, data + offset, 4);
|
|
|
|
return v;
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
2007-09-27 00:01:38 +00:00
|
|
|
void appendAddress(uintptr_t v) {
|
|
|
|
append4(v);
|
|
|
|
if (BytesPerWord == 8) {
|
2007-09-29 18:34:56 +00:00
|
|
|
// we have to use the preprocessor here to avoid a warning on
|
|
|
|
// 32-bit systems
|
|
|
|
#ifdef __x86_64__
|
2007-09-27 00:01:38 +00:00
|
|
|
append4(v >> 32);
|
2007-09-29 18:34:56 +00:00
|
|
|
#endif
|
2007-09-27 00:01:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-09-24 01:39:03 +00:00
|
|
|
unsigned length() {
|
2007-09-28 23:41:03 +00:00
|
|
|
return position;
|
2007-09-24 01:39:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void copyTo(uint8_t* b) {
|
2007-09-28 23:41:03 +00:00
|
|
|
if (data) {
|
|
|
|
memcpy(b, data, position);
|
2007-09-24 01:39:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
System* s;
|
2007-09-28 23:41:03 +00:00
|
|
|
uint8_t* data;
|
2007-09-24 01:39:03 +00:00
|
|
|
unsigned position;
|
2007-09-28 23:41:03 +00:00
|
|
|
unsigned capacity;
|
|
|
|
unsigned minimumCapacity;
|
2007-09-24 01:39:03 +00:00
|
|
|
};
|
|
|
|
|
2007-09-25 23:53:11 +00:00
|
|
|
class ArgumentList;
|
|
|
|
|
|
|
|
class MyThread: public Thread {
|
|
|
|
public:
|
|
|
|
MyThread(Machine* m, object javaThread, vm::Thread* parent):
|
|
|
|
vm::Thread(m, javaThread, parent),
|
|
|
|
argumentList(0),
|
|
|
|
frame(0)
|
|
|
|
{ }
|
|
|
|
|
|
|
|
ArgumentList* argumentList;
|
|
|
|
void* frame;
|
|
|
|
};
|
|
|
|
|
2007-09-27 00:01:38 +00:00
|
|
|
inline bool
|
|
|
|
isByte(int32_t v)
|
|
|
|
{
|
|
|
|
return v == static_cast<int8_t>(v);
|
|
|
|
}
|
|
|
|
|
2007-09-28 23:41:03 +00:00
|
|
|
enum Register {
|
|
|
|
rax = 0,
|
|
|
|
rcx = 1,
|
|
|
|
rdx = 2,
|
|
|
|
rbx = 3,
|
|
|
|
rsp = 4,
|
|
|
|
rbp = 5,
|
|
|
|
rsi = 6,
|
|
|
|
rdi = 7,
|
|
|
|
r8 = 8,
|
|
|
|
r9 = 9,
|
|
|
|
r10 = 10,
|
|
|
|
r11 = 11,
|
|
|
|
r12 = 12,
|
|
|
|
r13 = 13,
|
|
|
|
r14 = 14,
|
|
|
|
r15 = 15,
|
|
|
|
};
|
|
|
|
|
|
|
|
enum SSERegister {
|
|
|
|
xmm0 = 0,
|
|
|
|
xmm1 = 1,
|
|
|
|
xmm2 = 2,
|
|
|
|
xmm3 = 3,
|
|
|
|
xmm4 = 4,
|
|
|
|
xmm5 = 5,
|
|
|
|
xmm6 = 6,
|
|
|
|
xmm7 = 7
|
|
|
|
};
|
|
|
|
|
2007-09-24 01:39:03 +00:00
|
|
|
class Assembler {
|
|
|
|
public:
|
2007-09-25 23:53:11 +00:00
|
|
|
class Label {
|
|
|
|
public:
|
|
|
|
static const unsigned Capacity = 8;
|
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
Label(Assembler* a):
|
|
|
|
code(&(a->code)),
|
2007-09-25 23:53:11 +00:00
|
|
|
unresolvedCount(0),
|
|
|
|
mark_(-1)
|
|
|
|
{ }
|
|
|
|
|
2007-09-28 23:41:03 +00:00
|
|
|
void reference() {
|
2007-09-25 23:53:11 +00:00
|
|
|
if (mark_ == -1) {
|
2007-09-26 23:23:03 +00:00
|
|
|
expect(code->s, unresolvedCount < Capacity);
|
2007-09-28 23:41:03 +00:00
|
|
|
unresolved[unresolvedCount] = code->length();
|
2007-09-25 23:53:11 +00:00
|
|
|
++ unresolvedCount;
|
|
|
|
|
2007-09-28 23:41:03 +00:00
|
|
|
code->append4(0);
|
2007-09-25 23:53:11 +00:00
|
|
|
} else {
|
2007-09-28 23:41:03 +00:00
|
|
|
code->append4(mark_ - (code->length() + 4));
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
void mark() {
|
|
|
|
mark_ = code->length();
|
2007-09-25 23:53:11 +00:00
|
|
|
for (unsigned i = 0; i < unresolvedCount; ++i) {
|
2007-09-28 23:41:03 +00:00
|
|
|
code->set4(unresolved[i], mark_ - (unresolved[i] + 4));
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-09-28 23:41:03 +00:00
|
|
|
Buffer* code;
|
|
|
|
unsigned unresolved[Capacity];
|
2007-09-25 23:53:11 +00:00
|
|
|
unsigned unresolvedCount;
|
|
|
|
int mark_;
|
|
|
|
};
|
|
|
|
|
2007-09-24 01:39:03 +00:00
|
|
|
Assembler(System* s):
|
2007-09-28 23:41:03 +00:00
|
|
|
code(s, 1024),
|
|
|
|
jumps(s, 32)
|
2007-09-24 01:39:03 +00:00
|
|
|
{ }
|
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
void rex() {
|
|
|
|
if (BytesPerWord == 8) {
|
|
|
|
code.append(0x48);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-09-25 23:53:11 +00:00
|
|
|
void mov(Register src, Register dst) {
|
2007-09-26 23:23:03 +00:00
|
|
|
rex();
|
|
|
|
code.append(0x89);
|
|
|
|
code.append(0xc0 | (src << 3) | dst);
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
2007-09-28 14:45:26 +00:00
|
|
|
void offsetInstruction(uint8_t instruction, uint8_t zeroPrefix,
|
|
|
|
uint8_t bytePrefix, uint8_t wordPrefix,
|
2007-09-28 23:41:03 +00:00
|
|
|
unsigned a, unsigned b, int32_t offset)
|
2007-09-28 14:45:26 +00:00
|
|
|
{
|
|
|
|
code.append(instruction);
|
|
|
|
|
|
|
|
uint8_t prefix;
|
|
|
|
if (offset == 0 and b != rbp) {
|
|
|
|
prefix = zeroPrefix;
|
|
|
|
} else if (isByte(offset)) {
|
|
|
|
prefix = bytePrefix;
|
2007-09-25 23:53:11 +00:00
|
|
|
} else {
|
2007-09-28 14:45:26 +00:00
|
|
|
prefix = wordPrefix;
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
2007-09-28 14:45:26 +00:00
|
|
|
|
|
|
|
code.append(prefix | (a << 3) | b);
|
|
|
|
|
|
|
|
if (b == rsp) {
|
|
|
|
code.append(0x24);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (offset == 0 and b != rbp) {
|
|
|
|
// do nothing
|
|
|
|
} else if (isByte(offset)) {
|
|
|
|
code.append(offset);
|
|
|
|
} else {
|
|
|
|
code.append4(offset);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-09-29 20:24:14 +00:00
|
|
|
void movz1(Register src, int32_t srcOffset, Register dst) {
|
|
|
|
code.append(0x0f);
|
|
|
|
offsetInstruction(0x86, 0, 0x40, 0x80, dst, src, srcOffset);
|
|
|
|
}
|
|
|
|
|
|
|
|
void movs1(Register src, int32_t srcOffset, Register dst) {
|
|
|
|
code.append(0x0f);
|
|
|
|
offsetInstruction(0x8e, 0, 0x40, 0x80, dst, src, srcOffset);
|
|
|
|
}
|
|
|
|
|
|
|
|
void movz2(Register src, int32_t srcOffset, Register dst) {
|
|
|
|
code.append(0x0f);
|
|
|
|
offsetInstruction(0x87, 0, 0x40, 0x80, dst, src, srcOffset);
|
|
|
|
}
|
|
|
|
|
|
|
|
void movs2(Register src, int32_t srcOffset, Register dst) {
|
|
|
|
code.append(0x0f);
|
|
|
|
offsetInstruction(0x8f, 0, 0x40, 0x80, dst, src, srcOffset);
|
|
|
|
}
|
|
|
|
|
2007-09-28 14:45:26 +00:00
|
|
|
void mov4(Register src, int32_t srcOffset, Register dst) {
|
|
|
|
offsetInstruction(0x8b, 0, 0x40, 0x80, dst, src, srcOffset);
|
|
|
|
}
|
|
|
|
|
2007-09-29 20:24:14 +00:00
|
|
|
void mov1(Register src, Register dst, int32_t dstOffset) {
|
|
|
|
offsetInstruction(0x88, 0, 0x40, 0x80, src, dst, dstOffset);
|
|
|
|
}
|
|
|
|
|
|
|
|
void mov2(Register src, Register dst, int32_t dstOffset) {
|
|
|
|
code.append(0x66);
|
|
|
|
offsetInstruction(0x89, 0, 0x40, 0x80, src, dst, dstOffset);
|
|
|
|
}
|
|
|
|
|
2007-09-28 14:45:26 +00:00
|
|
|
void mov4(Register src, Register dst, int32_t dstOffset) {
|
|
|
|
offsetInstruction(0x89, 0, 0x40, 0x80, src, dst, dstOffset);
|
|
|
|
}
|
|
|
|
|
2007-09-28 23:41:03 +00:00
|
|
|
void mov(Register src, int32_t srcOffset, SSERegister dst) {
|
|
|
|
code.append(0xf3);
|
|
|
|
code.append(0x0f);
|
|
|
|
offsetInstruction(0x7e, 0, 0x40, 0x80, dst, src, srcOffset);
|
|
|
|
}
|
|
|
|
|
2007-09-28 14:45:26 +00:00
|
|
|
void mov(Register src, int32_t srcOffset, Register dst) {
|
|
|
|
rex();
|
|
|
|
mov4(src, srcOffset, dst);
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
2007-09-27 00:01:38 +00:00
|
|
|
void mov(Register src, Register dst, int32_t dstOffset) {
|
2007-09-26 23:23:03 +00:00
|
|
|
rex();
|
2007-09-28 14:45:26 +00:00
|
|
|
mov4(src, dst, dstOffset);
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
void mov(uintptr_t v, Register dst) {
|
|
|
|
rex();
|
|
|
|
code.append(0xb8 | dst);
|
|
|
|
code.appendAddress(v);
|
|
|
|
}
|
|
|
|
|
|
|
|
void alignedMov(uintptr_t v, Register dst) {
|
|
|
|
while ((code.length() + (BytesPerWord == 8 ? 2 : 1)) % BytesPerWord) {
|
|
|
|
nop();
|
|
|
|
}
|
|
|
|
rex();
|
|
|
|
code.append(0xb8 | dst);
|
|
|
|
code.appendAddress(v);
|
|
|
|
}
|
|
|
|
|
|
|
|
void nop() {
|
|
|
|
code.append(0x90);
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void push(Register reg) {
|
2007-09-26 23:23:03 +00:00
|
|
|
code.append(0x50 | reg);
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
2007-09-27 00:01:38 +00:00
|
|
|
void push(Register reg, int32_t offset) {
|
2007-09-28 14:45:26 +00:00
|
|
|
offsetInstruction(0xff, 0x30, 0x70, 0xb0, rax, reg, offset);
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
2007-09-27 00:01:38 +00:00
|
|
|
void push(int32_t v) {
|
2007-09-29 18:34:56 +00:00
|
|
|
if (isByte(v)) {
|
|
|
|
code.append(0x6a);
|
|
|
|
code.append(v);
|
|
|
|
} else {
|
|
|
|
code.append(0x68);
|
|
|
|
code.append4(v);
|
|
|
|
}
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
2007-09-27 22:20:54 +00:00
|
|
|
void pushAddress(uintptr_t v) {
|
|
|
|
if (BytesPerWord == 8) {
|
|
|
|
mov(v, rsi);
|
|
|
|
push(rsi);
|
|
|
|
} else {
|
|
|
|
push(v);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-09-28 14:45:26 +00:00
|
|
|
void push4(Register reg, int32_t offset) {
|
|
|
|
if (BytesPerWord == 8) {
|
|
|
|
mov4(reg, offset, rsi);
|
|
|
|
push(rsi);
|
|
|
|
} else {
|
|
|
|
push(reg, offset);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-09-25 23:53:11 +00:00
|
|
|
void pop(Register dst) {
|
2007-09-26 23:23:03 +00:00
|
|
|
code.append(0x58 | dst);
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
2007-09-27 00:01:38 +00:00
|
|
|
void pop(Register dst, int32_t offset) {
|
2007-09-28 14:45:26 +00:00
|
|
|
offsetInstruction(0x8f, 0, 0x40, 0x80, rax, dst, offset);
|
|
|
|
}
|
2007-09-26 23:23:03 +00:00
|
|
|
|
2007-09-28 14:45:26 +00:00
|
|
|
void pop4(Register reg, int32_t offset) {
|
|
|
|
if (BytesPerWord == 8) {
|
|
|
|
pop(rsi);
|
|
|
|
mov4(rsi, reg, offset);
|
|
|
|
} else {
|
|
|
|
pop(reg, offset);
|
|
|
|
}
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void add(Register src, Register dst) {
|
2007-09-26 23:23:03 +00:00
|
|
|
rex();
|
|
|
|
code.append(0x01);
|
|
|
|
code.append(0xc0 | (src << 3) | dst);
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
2007-09-27 00:01:38 +00:00
|
|
|
void add(int32_t v, Register dst) {
|
|
|
|
assert(code.s, isByte(v)); // todo
|
2007-09-26 23:23:03 +00:00
|
|
|
|
|
|
|
rex();
|
|
|
|
code.append(0x83);
|
|
|
|
code.append(0xc0 | dst);
|
|
|
|
code.append(v);
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void sub(Register src, Register dst) {
|
2007-09-26 23:23:03 +00:00
|
|
|
rex();
|
|
|
|
code.append(0x29);
|
|
|
|
code.append(0xc0 | (src << 3) | dst);
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
2007-09-27 00:01:38 +00:00
|
|
|
void sub(int32_t v, Register dst) {
|
|
|
|
assert(code.s, isByte(v)); // todo
|
2007-09-26 23:23:03 +00:00
|
|
|
|
|
|
|
rex();
|
|
|
|
code.append(0x83);
|
|
|
|
code.append(0xe8 | dst);
|
|
|
|
code.append(v);
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void or_(Register src, Register dst) {
|
2007-09-26 23:23:03 +00:00
|
|
|
rex();
|
|
|
|
code.append(0x09);
|
|
|
|
code.append(0xc0 | (src << 3) | dst);
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
2007-09-27 00:01:38 +00:00
|
|
|
void or_(int32_t v, Register dst) {
|
|
|
|
assert(code.s, isByte(v)); // todo
|
2007-09-26 23:23:03 +00:00
|
|
|
|
|
|
|
rex();
|
|
|
|
code.append(0x83);
|
|
|
|
code.append(0xc8 | dst);
|
|
|
|
code.append(v);
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void and_(Register src, Register dst) {
|
2007-09-26 23:23:03 +00:00
|
|
|
rex();
|
|
|
|
code.append(0x21);
|
|
|
|
code.append(0xc0 | (src << 3) | dst);
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
2007-09-27 00:01:38 +00:00
|
|
|
void and_(int32_t v, Register dst) {
|
|
|
|
assert(code.s, isByte(v)); // todo
|
2007-09-26 23:23:03 +00:00
|
|
|
|
|
|
|
rex();
|
|
|
|
code.append(0x83);
|
|
|
|
code.append(0xe0 | dst);
|
|
|
|
code.append(v);
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ret() {
|
2007-09-26 23:23:03 +00:00
|
|
|
code.append(0xc3);
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void jmp(Label& label) {
|
2007-09-26 23:23:03 +00:00
|
|
|
code.append(0xE9);
|
2007-09-28 23:41:03 +00:00
|
|
|
label.reference();
|
|
|
|
}
|
|
|
|
|
|
|
|
void jmp(unsigned javaIP) {
|
|
|
|
code.append(0xE9);
|
|
|
|
|
|
|
|
jumps.append4(javaIP);
|
|
|
|
jumps.append4(code.length());
|
|
|
|
|
|
|
|
code.append4(0);
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void jmp(Register reg) {
|
2007-09-26 23:23:03 +00:00
|
|
|
code.append(0xff);
|
|
|
|
code.append(0xe0 | reg);
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
void jz(Label& label) {
|
2007-09-29 20:24:14 +00:00
|
|
|
code.append(0x0f);
|
2007-09-26 23:23:03 +00:00
|
|
|
code.append(0x84);
|
2007-09-28 23:41:03 +00:00
|
|
|
label.reference();
|
2007-09-26 23:23:03 +00:00
|
|
|
}
|
|
|
|
|
2007-09-29 20:24:14 +00:00
|
|
|
void conditional(unsigned javaIP, unsigned condition) {
|
|
|
|
code.append(0x0f);
|
|
|
|
code.append(condition);
|
2007-09-28 23:41:03 +00:00
|
|
|
|
|
|
|
jumps.append4(javaIP);
|
|
|
|
jumps.append4(code.length());
|
|
|
|
|
|
|
|
code.append4(0);
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
2007-09-29 20:24:14 +00:00
|
|
|
void jz(unsigned javaIP) {
|
|
|
|
conditional(javaIP, 0x84);
|
|
|
|
}
|
|
|
|
|
2007-09-25 23:53:11 +00:00
|
|
|
void jnz(Label& label) {
|
2007-09-29 20:24:14 +00:00
|
|
|
code.append(0x0f);
|
2007-09-26 23:23:03 +00:00
|
|
|
code.append(0x85);
|
2007-09-28 23:41:03 +00:00
|
|
|
label.reference();
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
2007-09-28 23:41:03 +00:00
|
|
|
void jnz(unsigned javaIP) {
|
2007-09-29 20:24:14 +00:00
|
|
|
conditional(javaIP, 0x85);
|
|
|
|
}
|
2007-09-28 23:41:03 +00:00
|
|
|
|
2007-09-29 20:24:14 +00:00
|
|
|
void jg(unsigned javaIP) {
|
|
|
|
conditional(javaIP, 0x8f);
|
|
|
|
}
|
2007-09-28 23:41:03 +00:00
|
|
|
|
2007-09-29 20:24:14 +00:00
|
|
|
void jge(unsigned javaIP) {
|
|
|
|
conditional(javaIP, 0x8d);
|
|
|
|
}
|
|
|
|
|
|
|
|
void jl(unsigned javaIP) {
|
|
|
|
conditional(javaIP, 0x8c);
|
|
|
|
}
|
|
|
|
|
|
|
|
void jle(unsigned javaIP) {
|
|
|
|
conditional(javaIP, 0x8e);
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void cmp(int v, Register reg) {
|
2007-09-26 23:23:03 +00:00
|
|
|
code.append(0x83);
|
|
|
|
code.append(0xf8 | reg);
|
|
|
|
code.append(v);
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void call(Register reg) {
|
2007-09-26 23:23:03 +00:00
|
|
|
code.append(0xff);
|
|
|
|
code.append(0xd0 | reg);
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
|
2007-09-28 23:41:03 +00:00
|
|
|
Buffer code;
|
|
|
|
Buffer jumps;
|
2007-09-24 01:39:03 +00:00
|
|
|
};
|
|
|
|
|
2007-09-25 23:53:11 +00:00
|
|
|
void
|
2007-09-26 23:23:03 +00:00
|
|
|
compileMethod(MyThread* t, object method);
|
2007-09-25 23:53:11 +00:00
|
|
|
|
|
|
|
int
|
|
|
|
localOffset(int v, int parameterFootprint)
|
|
|
|
{
|
2007-09-26 23:23:03 +00:00
|
|
|
v *= BytesPerWord;
|
2007-09-25 23:53:11 +00:00
|
|
|
if (v < parameterFootprint) {
|
2007-09-26 23:23:03 +00:00
|
|
|
return v + (BytesPerWord * 2) + FrameFootprint;
|
2007-09-25 23:53:11 +00:00
|
|
|
} else {
|
2007-09-26 23:23:03 +00:00
|
|
|
return -(v + BytesPerWord - parameterFootprint);
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-09-28 23:41:03 +00:00
|
|
|
Register
|
|
|
|
gpRegister(Thread* t, unsigned index)
|
|
|
|
{
|
|
|
|
switch (index) {
|
|
|
|
case 0:
|
|
|
|
return rdi;
|
|
|
|
case 1:
|
|
|
|
return rsi;
|
|
|
|
case 2:
|
|
|
|
return rdx;
|
|
|
|
case 3:
|
|
|
|
return rcx;
|
|
|
|
case 4:
|
|
|
|
return r8;
|
|
|
|
case 5:
|
|
|
|
return r9;
|
|
|
|
default:
|
|
|
|
abort(t);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SSERegister
|
|
|
|
sseRegister(Thread* t, unsigned index)
|
|
|
|
{
|
|
|
|
assert(t, index < 8);
|
|
|
|
return static_cast<SSERegister>(index);
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned
|
|
|
|
parameterOffset(Thread* t, object method, unsigned index)
|
|
|
|
{
|
|
|
|
return FrameFootprint
|
|
|
|
+ (((methodParameterFootprint(t, method) - index - 1) + 2)
|
|
|
|
* BytesPerWord);
|
|
|
|
}
|
|
|
|
|
2007-09-25 23:53:11 +00:00
|
|
|
class Compiler: public Assembler {
|
2007-09-24 01:39:03 +00:00
|
|
|
public:
|
|
|
|
Compiler(System* s):
|
2007-09-28 23:41:03 +00:00
|
|
|
Assembler(s),
|
|
|
|
javaIPs(s, 1024),
|
|
|
|
machineIPs(s, 1024)
|
2007-09-24 01:39:03 +00:00
|
|
|
{ }
|
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
void pushReturnValue(Thread* t, unsigned code) {
|
|
|
|
switch (code) {
|
|
|
|
case ByteField:
|
|
|
|
case BooleanField:
|
|
|
|
case CharField:
|
|
|
|
case ShortField:
|
|
|
|
case FloatField:
|
|
|
|
case IntField:
|
|
|
|
case ObjectField:
|
|
|
|
push(rax);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LongField:
|
|
|
|
case DoubleField:
|
|
|
|
push(rax);
|
|
|
|
push(rdx);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VoidField:
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
abort(t);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-09-28 14:45:26 +00:00
|
|
|
void compileDirectInvoke(Thread* t, object target) {
|
|
|
|
unsigned footprint = FrameFootprint
|
|
|
|
+ (methodParameterFootprint(t, target) * BytesPerWord);
|
|
|
|
|
|
|
|
uint8_t* code = &compiledBody(t, methodCompiled(t, target), 0);
|
|
|
|
|
|
|
|
push(rbp);
|
|
|
|
pushAddress(reinterpret_cast<uintptr_t>(target));
|
|
|
|
push(rbp, FrameThread);
|
|
|
|
|
|
|
|
alignedMov(reinterpret_cast<uintptr_t>(code), rax);
|
|
|
|
call(rax);
|
|
|
|
|
|
|
|
add(footprint, rsp); // pop arguments
|
|
|
|
|
|
|
|
pushReturnValue(t, methodReturnCode(t, target));
|
|
|
|
}
|
|
|
|
|
|
|
|
void compileCall1(uintptr_t function, uintptr_t arg1) {
|
|
|
|
if (BytesPerWord == 4) {
|
|
|
|
pushAddress(arg1);
|
|
|
|
push(rbp, FrameThread);
|
|
|
|
} else {
|
|
|
|
mov(arg1, rsi);
|
|
|
|
mov(rbp, FrameThread, rdi);
|
|
|
|
}
|
|
|
|
|
|
|
|
mov(function, rax);
|
|
|
|
call(rax);
|
|
|
|
|
|
|
|
if (BytesPerWord == 4) {
|
|
|
|
add(BytesPerWord * 2, rsp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-09-28 23:41:03 +00:00
|
|
|
void compile(MyThread* t, object method) {
|
|
|
|
if (methodFlags(t, method) & ACC_NATIVE) {
|
|
|
|
compileNative(t, method);
|
|
|
|
} else {
|
|
|
|
compileJava(t, method);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void compileNative(MyThread* t, object method) {
|
|
|
|
unsigned frameOffset = reinterpret_cast<uintptr_t>(&(t->frame))
|
|
|
|
- reinterpret_cast<uintptr_t>(t);
|
|
|
|
|
|
|
|
void* function = resolveNativeMethod(t, method);
|
2007-09-26 23:23:03 +00:00
|
|
|
|
|
|
|
push(rbp);
|
|
|
|
mov(rsp, rbp);
|
2007-09-28 23:41:03 +00:00
|
|
|
|
|
|
|
mov(rbp, FrameThread, rax);
|
|
|
|
mov(rbp, rax, frameOffset); // set thread frame to current
|
|
|
|
|
|
|
|
unsigned index;
|
|
|
|
if (methodFlags(t, method) & ACC_STATIC) {
|
|
|
|
pushAddress(reinterpret_cast<uintptr_t>(methodClass(t, method)));
|
|
|
|
index = 0;
|
|
|
|
} else {
|
|
|
|
index = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
MethodSpecIterator it(t, reinterpret_cast<const char*>
|
|
|
|
(&byteArrayBody(t, methodSpec(t, method), 0)));
|
|
|
|
|
|
|
|
unsigned stackFootprint;
|
|
|
|
|
|
|
|
if (BytesPerWord == 4) {
|
|
|
|
while (it.hasNext()) {
|
|
|
|
unsigned offset = parameterOffset(t, method, index);
|
|
|
|
|
|
|
|
switch (fieldCode(t, *it.next())) {
|
|
|
|
case BooleanField:
|
|
|
|
case ByteField:
|
|
|
|
case ShortField:
|
|
|
|
case CharField:
|
|
|
|
case IntField:
|
|
|
|
case FloatField: {
|
|
|
|
push(rbp, offset);
|
|
|
|
++ index;
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case LongField:
|
|
|
|
case DoubleField: {
|
|
|
|
push(rbp, offset);
|
|
|
|
push(rbp, offset - BytesPerWord);
|
|
|
|
index += 2;
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case ObjectField: {
|
|
|
|
mov(rbp, rax);
|
|
|
|
add(offset, rax);
|
|
|
|
push(rax);
|
|
|
|
++ index;
|
|
|
|
} break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
abort(t);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (methodFlags(t, method) & ACC_STATIC) {
|
|
|
|
mov(rbp, rax);
|
|
|
|
sub(BytesPerWord, rax);
|
|
|
|
push(rax); // push pointer to class pointer
|
|
|
|
} else {
|
|
|
|
unsigned offset = parameterOffset(t, method, 0);
|
|
|
|
mov(rbp, rax);
|
|
|
|
add(offset, rax);
|
|
|
|
push(rax); // push pointer to this pointer
|
|
|
|
}
|
|
|
|
|
|
|
|
push(rbp, FrameThread); // push thread pointer
|
|
|
|
|
|
|
|
stackFootprint = FrameFootprint
|
|
|
|
+ (methodParameterFootprint(t, method) * BytesPerWord);
|
|
|
|
} else {
|
|
|
|
const unsigned GprCount = 6;
|
|
|
|
unsigned gprIndex = 0;
|
|
|
|
|
|
|
|
const unsigned SseCount = 8;
|
|
|
|
unsigned sseIndex = 0;
|
|
|
|
|
|
|
|
stackFootprint = 0;
|
|
|
|
|
|
|
|
while (it.hasNext()) {
|
|
|
|
unsigned offset = parameterOffset(t, method, index);
|
|
|
|
|
|
|
|
switch (fieldCode(t, *it.next())) {
|
|
|
|
case BooleanField:
|
|
|
|
case ByteField:
|
|
|
|
case ShortField:
|
|
|
|
case CharField:
|
|
|
|
case IntField:
|
|
|
|
case LongField: {
|
|
|
|
if (gprIndex < GprCount - 2) {
|
|
|
|
Register reg = gpRegister(t, gprIndex + 2);
|
|
|
|
mov(rbp, offset, reg);
|
|
|
|
++ gprIndex;
|
|
|
|
} else {
|
|
|
|
push(rbp, offset);
|
|
|
|
stackFootprint += BytesPerWord;
|
|
|
|
}
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case ObjectField: {
|
|
|
|
if (gprIndex < GprCount - 2) {
|
|
|
|
Register reg = gpRegister(t, gprIndex + 2);
|
|
|
|
mov(rbp, reg);
|
|
|
|
add(offset, reg);
|
|
|
|
++ gprIndex;
|
|
|
|
} else {
|
|
|
|
mov(rbp, rax);
|
|
|
|
add(offset, rax);
|
|
|
|
push(rax);
|
|
|
|
stackFootprint += BytesPerWord;
|
|
|
|
}
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case FloatField:
|
|
|
|
case DoubleField: {
|
|
|
|
if (sseIndex < SseCount) {
|
|
|
|
SSERegister reg = sseRegister(t, sseIndex);
|
|
|
|
mov(rbp, offset, reg);
|
|
|
|
++ sseIndex;
|
|
|
|
} else {
|
|
|
|
push(rbp, offset);
|
|
|
|
stackFootprint += BytesPerWord;
|
|
|
|
}
|
|
|
|
} break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
abort(t);
|
|
|
|
}
|
|
|
|
|
|
|
|
++ index;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (methodFlags(t, method) & ACC_STATIC) {
|
|
|
|
mov(rbp, rsi);
|
|
|
|
sub(BytesPerWord, rsi); // push pointer to class pointer
|
|
|
|
} else {
|
|
|
|
unsigned offset = parameterOffset(t, method, 0);
|
|
|
|
mov(rbp, rsi);
|
|
|
|
add(offset, rsi); // push pointer to this pointer
|
|
|
|
}
|
|
|
|
|
|
|
|
mov(rbp, FrameThread, rdi); // push thread pointer
|
|
|
|
}
|
|
|
|
|
|
|
|
mov(reinterpret_cast<uintptr_t>(function), rax);
|
|
|
|
call(rax);
|
|
|
|
|
|
|
|
if (stackFootprint) {
|
|
|
|
add(stackFootprint, rsp);
|
|
|
|
}
|
|
|
|
|
|
|
|
mov(rbp, rsp);
|
|
|
|
pop(rbp);
|
|
|
|
ret();
|
|
|
|
}
|
|
|
|
|
|
|
|
void compileJava(MyThread* t, object method) {
|
|
|
|
PROTECT(t, method);
|
2007-09-24 01:39:03 +00:00
|
|
|
|
|
|
|
object code = methodCode(t, method);
|
2007-09-26 23:23:03 +00:00
|
|
|
PROTECT(t, code);
|
|
|
|
|
|
|
|
unsigned parameterFootprint
|
|
|
|
= methodParameterFootprint(t, method) * BytesPerWord;
|
|
|
|
|
|
|
|
unsigned localFootprint = codeMaxLocals(t, code) * BytesPerWord;
|
2007-09-24 01:39:03 +00:00
|
|
|
|
2007-09-28 23:41:03 +00:00
|
|
|
push(rbp);
|
|
|
|
mov(rsp, rbp);
|
|
|
|
|
2007-09-27 22:20:54 +00:00
|
|
|
if (localFootprint > parameterFootprint) {
|
|
|
|
// reserve space for local variables
|
|
|
|
sub(localFootprint - parameterFootprint, rsp);
|
|
|
|
}
|
2007-09-24 01:39:03 +00:00
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
for (unsigned ip = 0; ip < codeLength(t, code);) {
|
2007-09-28 23:41:03 +00:00
|
|
|
javaIPs.append2(ip);
|
|
|
|
machineIPs.append4(this->code.length());
|
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
unsigned instruction = codeBody(t, code, ip++);
|
|
|
|
|
|
|
|
switch (instruction) {
|
2007-09-29 20:24:14 +00:00
|
|
|
case aconst_null:
|
|
|
|
push(0);
|
|
|
|
break;
|
|
|
|
|
2007-09-28 23:41:03 +00:00
|
|
|
case aload:
|
|
|
|
case iload:
|
|
|
|
case fload:
|
|
|
|
push(rbp, localOffset(codeBody(t, code, ip++), parameterFootprint));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case aload_0:
|
|
|
|
case iload_0:
|
|
|
|
case fload_0:
|
|
|
|
push(rbp, localOffset(0, parameterFootprint));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case aload_1:
|
|
|
|
case iload_1:
|
|
|
|
case fload_1:
|
|
|
|
push(rbp, localOffset(1, parameterFootprint));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case aload_2:
|
|
|
|
case iload_2:
|
|
|
|
case fload_2:
|
|
|
|
push(rbp, localOffset(2, parameterFootprint));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case aload_3:
|
|
|
|
case iload_3:
|
|
|
|
case fload_3:
|
|
|
|
push(rbp, localOffset(3, parameterFootprint));
|
|
|
|
break;
|
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
case areturn:
|
|
|
|
pop(rax);
|
|
|
|
mov(rbp, rsp);
|
|
|
|
pop(rbp);
|
|
|
|
ret();
|
|
|
|
break;
|
|
|
|
|
2007-09-28 23:41:03 +00:00
|
|
|
case astore:
|
|
|
|
case istore:
|
|
|
|
case fstore:
|
|
|
|
pop(rbp, localOffset(codeBody(t, code, ip++), parameterFootprint));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case astore_0:
|
|
|
|
case istore_0:
|
|
|
|
case fstore_0:
|
|
|
|
pop(rbp, localOffset(0, parameterFootprint));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case astore_1:
|
|
|
|
case istore_1:
|
|
|
|
case fstore_1:
|
|
|
|
pop(rbp, localOffset(1, parameterFootprint));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case astore_2:
|
|
|
|
case istore_2:
|
|
|
|
case fstore_2:
|
|
|
|
pop(rbp, localOffset(2, parameterFootprint));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case astore_3:
|
|
|
|
case istore_3:
|
|
|
|
case fstore_3:
|
|
|
|
pop(rbp, localOffset(3, parameterFootprint));
|
|
|
|
break;
|
|
|
|
|
2007-09-28 14:45:26 +00:00
|
|
|
case bipush: {
|
|
|
|
push(static_cast<int8_t>(codeBody(t, code, ip++)));
|
|
|
|
} break;
|
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
case dup:
|
2007-09-27 22:20:54 +00:00
|
|
|
push(rsp, BytesPerWord);
|
2007-09-26 23:23:03 +00:00
|
|
|
break;
|
|
|
|
|
2007-09-29 20:24:14 +00:00
|
|
|
case getfield: {
|
|
|
|
uint16_t index = codeReadInt16(t, code, ip);
|
|
|
|
|
|
|
|
object field = resolveField(t, codePool(t, code), index - 1);
|
|
|
|
if (UNLIKELY(t->exception)) return;
|
|
|
|
|
|
|
|
pop(rax);
|
|
|
|
|
|
|
|
switch (fieldCode(t, field)) {
|
|
|
|
case ByteField:
|
|
|
|
case BooleanField:
|
|
|
|
movs1(rax, fieldOffset(t, field), rax);
|
|
|
|
push(rax);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CharField:
|
|
|
|
movz2(rax, fieldOffset(t, field), rax);
|
|
|
|
push(rax);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ShortField:
|
|
|
|
movs2(rax, fieldOffset(t, field), rax);
|
|
|
|
push(rax);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case FloatField:
|
|
|
|
case IntField:
|
|
|
|
push4(rax, fieldOffset(t, field));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DoubleField:
|
|
|
|
case LongField:
|
|
|
|
push4(rax, fieldOffset(t, field));
|
|
|
|
push4(rax, fieldOffset(t, field) + 4);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ObjectField:
|
|
|
|
push(rax, fieldOffset(t, field));
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
abort(t);
|
|
|
|
}
|
|
|
|
} break;
|
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
case getstatic: {
|
|
|
|
uint16_t index = codeReadInt16(t, code, ip);
|
|
|
|
|
|
|
|
object field = resolveField(t, codePool(t, code), index - 1);
|
|
|
|
if (UNLIKELY(t->exception)) return;
|
|
|
|
PROTECT(t, field);
|
|
|
|
|
|
|
|
initClass(t, fieldClass(t, field));
|
|
|
|
if (UNLIKELY(t->exception)) return;
|
|
|
|
|
|
|
|
object table = classStaticTable(t, fieldClass(t, field));
|
|
|
|
|
|
|
|
mov(reinterpret_cast<uintptr_t>(table), rax);
|
|
|
|
add(fieldOffset(t, field), rax);
|
|
|
|
|
|
|
|
switch (fieldCode(t, field)) {
|
|
|
|
case ByteField:
|
|
|
|
case BooleanField:
|
|
|
|
case CharField:
|
|
|
|
case ShortField:
|
|
|
|
case FloatField:
|
|
|
|
case IntField: {
|
|
|
|
Label zero(this);
|
|
|
|
Label next(this);
|
|
|
|
|
|
|
|
cmp(0, rax);
|
2007-09-28 23:41:03 +00:00
|
|
|
jz(zero);
|
2007-09-26 23:23:03 +00:00
|
|
|
|
2007-09-28 14:45:26 +00:00
|
|
|
push4(rax, IntValue);
|
2007-09-26 23:23:03 +00:00
|
|
|
jmp(next);
|
|
|
|
|
|
|
|
zero.mark();
|
|
|
|
push(0);
|
|
|
|
|
|
|
|
next.mark();
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case DoubleField:
|
|
|
|
case LongField: {
|
|
|
|
Label zero(this);
|
|
|
|
Label next(this);
|
|
|
|
|
|
|
|
cmp(0, rax);
|
2007-09-28 23:41:03 +00:00
|
|
|
jz(zero);
|
2007-09-26 23:23:03 +00:00
|
|
|
|
2007-09-28 14:45:26 +00:00
|
|
|
push4(rax, LongValue);
|
|
|
|
push4(rax, LongValue + 4);
|
2007-09-26 23:23:03 +00:00
|
|
|
jmp(next);
|
|
|
|
|
|
|
|
zero.mark();
|
|
|
|
push(0);
|
|
|
|
push(0);
|
|
|
|
|
|
|
|
next.mark();
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case ObjectField: {
|
2007-09-28 14:45:26 +00:00
|
|
|
push(rax, 0);
|
2007-09-26 23:23:03 +00:00
|
|
|
} break;
|
|
|
|
|
|
|
|
default: abort(t);
|
|
|
|
}
|
|
|
|
} break;
|
|
|
|
|
2007-09-24 01:39:03 +00:00
|
|
|
case iadd:
|
2007-09-26 23:23:03 +00:00
|
|
|
pop(rax);
|
2007-09-27 22:20:54 +00:00
|
|
|
pop(rcx);
|
|
|
|
add(rax, rcx);
|
|
|
|
push(rcx);
|
2007-09-24 01:39:03 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case iconst_m1:
|
|
|
|
push(-1);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case iconst_0:
|
|
|
|
push(0);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case iconst_1:
|
|
|
|
push(1);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case iconst_2:
|
|
|
|
push(2);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case iconst_3:
|
|
|
|
push(3);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case iconst_4:
|
|
|
|
push(4);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case iconst_5:
|
|
|
|
push(5);
|
|
|
|
break;
|
|
|
|
|
2007-09-29 20:24:14 +00:00
|
|
|
case if_acmpeq:
|
|
|
|
case if_icmpeq: {
|
|
|
|
int16_t offset = codeReadInt16(t, code, ip);
|
|
|
|
|
|
|
|
pop(rax);
|
|
|
|
pop(rcx);
|
|
|
|
cmp(rax, rcx);
|
|
|
|
jz((ip - 3) + offset);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case if_acmpne:
|
|
|
|
case if_icmpne: {
|
|
|
|
int16_t offset = codeReadInt16(t, code, ip);
|
|
|
|
|
|
|
|
pop(rax);
|
|
|
|
pop(rcx);
|
|
|
|
cmp(rax, rcx);
|
|
|
|
jnz((ip - 3) + offset);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case if_icmpgt: {
|
|
|
|
int16_t offset = codeReadInt16(t, code, ip);
|
|
|
|
|
|
|
|
pop(rax);
|
|
|
|
pop(rcx);
|
|
|
|
cmp(rax, rcx);
|
|
|
|
jg((ip - 3) + offset);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case if_icmpge: {
|
|
|
|
int16_t offset = codeReadInt16(t, code, ip);
|
|
|
|
|
|
|
|
pop(rax);
|
|
|
|
pop(rcx);
|
|
|
|
cmp(rax, rcx);
|
|
|
|
jge((ip - 3) + offset);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case if_icmplt: {
|
|
|
|
int16_t offset = codeReadInt16(t, code, ip);
|
|
|
|
|
|
|
|
pop(rax);
|
|
|
|
pop(rcx);
|
|
|
|
cmp(rax, rcx);
|
|
|
|
jl((ip - 3) + offset);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case if_icmple: {
|
|
|
|
int16_t offset = codeReadInt16(t, code, ip);
|
|
|
|
|
|
|
|
pop(rax);
|
|
|
|
pop(rcx);
|
|
|
|
cmp(rax, rcx);
|
|
|
|
jle((ip - 3) + offset);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case ifeq:
|
2007-09-28 23:41:03 +00:00
|
|
|
case ifnull: {
|
|
|
|
int16_t offset = codeReadInt16(t, code, ip);
|
|
|
|
|
|
|
|
pop(rax);
|
|
|
|
cmp(0, rax);
|
|
|
|
jz((ip - 3) + offset);
|
|
|
|
} break;
|
2007-09-24 01:39:03 +00:00
|
|
|
|
2007-09-29 20:24:14 +00:00
|
|
|
case ifne:
|
|
|
|
case ifnonnull: {
|
|
|
|
int16_t offset = codeReadInt16(t, code, ip);
|
|
|
|
|
|
|
|
pop(rax);
|
|
|
|
cmp(0, rax);
|
|
|
|
jnz((ip - 3) + offset);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case ifgt: {
|
|
|
|
int16_t offset = codeReadInt16(t, code, ip);
|
|
|
|
|
|
|
|
pop(rax);
|
|
|
|
cmp(0, rax);
|
|
|
|
jg((ip - 3) + offset);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case ifge: {
|
|
|
|
int16_t offset = codeReadInt16(t, code, ip);
|
|
|
|
|
|
|
|
pop(rax);
|
|
|
|
cmp(0, rax);
|
|
|
|
jge((ip - 3) + offset);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case iflt: {
|
|
|
|
int16_t offset = codeReadInt16(t, code, ip);
|
|
|
|
|
|
|
|
pop(rax);
|
|
|
|
cmp(0, rax);
|
|
|
|
jl((ip - 3) + offset);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case ifle: {
|
|
|
|
int16_t offset = codeReadInt16(t, code, ip);
|
|
|
|
|
|
|
|
pop(rax);
|
|
|
|
cmp(0, rax);
|
|
|
|
jle((ip - 3) + offset);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case goto_: {
|
|
|
|
int16_t offset = codeReadInt16(t, code, ip);
|
|
|
|
jmp((ip - 3) + offset);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case goto_w: {
|
|
|
|
int32_t offset = codeReadInt32(t, code, ip);
|
|
|
|
jmp((ip - 5) + offset);
|
|
|
|
} break;
|
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
case invokespecial: {
|
|
|
|
uint16_t index = codeReadInt16(t, code, ip);
|
|
|
|
|
|
|
|
object target = resolveMethod(t, codePool(t, code), index - 1);
|
|
|
|
if (UNLIKELY(t->exception)) return;
|
|
|
|
|
2007-09-29 20:24:14 +00:00
|
|
|
object class_ = methodClass(t, target);
|
2007-09-26 23:23:03 +00:00
|
|
|
if (isSpecialMethod(t, target, class_)) {
|
2007-09-28 14:45:26 +00:00
|
|
|
target = findMethod(t, target, classSuper(t, class_));
|
2007-09-26 23:23:03 +00:00
|
|
|
}
|
|
|
|
|
2007-09-28 14:45:26 +00:00
|
|
|
compileDirectInvoke(t, target);
|
|
|
|
} break;
|
2007-09-26 23:23:03 +00:00
|
|
|
|
2007-09-28 14:45:26 +00:00
|
|
|
case invokestatic: {
|
|
|
|
uint16_t index = codeReadInt16(t, code, ip);
|
2007-09-26 23:23:03 +00:00
|
|
|
|
2007-09-28 14:45:26 +00:00
|
|
|
object target = resolveMethod(t, codePool(t, code), index - 1);
|
|
|
|
if (UNLIKELY(t->exception)) return;
|
|
|
|
PROTECT(t, target);
|
2007-09-26 23:23:03 +00:00
|
|
|
|
2007-09-29 20:24:14 +00:00
|
|
|
initClass(t, methodClass(t, target));
|
2007-09-28 14:45:26 +00:00
|
|
|
if (UNLIKELY(t->exception)) return;
|
2007-09-26 23:23:03 +00:00
|
|
|
|
2007-09-28 14:45:26 +00:00
|
|
|
compileDirectInvoke(t, target);
|
2007-09-26 23:23:03 +00:00
|
|
|
} break;
|
|
|
|
|
|
|
|
case invokevirtual: {
|
|
|
|
uint16_t index = codeReadInt16(t, code, ip);
|
|
|
|
|
|
|
|
object target = resolveMethod(t, codePool(t, code), index - 1);
|
|
|
|
if (UNLIKELY(t->exception)) return;
|
|
|
|
|
2007-09-27 22:20:54 +00:00
|
|
|
unsigned parameterFootprint
|
|
|
|
= methodParameterFootprint(t, target) * BytesPerWord;
|
|
|
|
|
|
|
|
unsigned instance = parameterFootprint - BytesPerWord;
|
|
|
|
|
|
|
|
unsigned footprint = FrameFootprint + parameterFootprint;
|
2007-09-26 23:23:03 +00:00
|
|
|
|
|
|
|
unsigned offset = ArrayBody + (methodOffset(t, target) * BytesPerWord);
|
2007-09-27 22:20:54 +00:00
|
|
|
|
|
|
|
mov(rsp, instance, rax); // load instance
|
|
|
|
mov(rax, 0, rax); // load class
|
2007-09-26 23:23:03 +00:00
|
|
|
mov(rax, ClassVirtualTable, rax); // load vtable
|
|
|
|
mov(rax, offset, rax); // load method
|
2007-09-27 22:20:54 +00:00
|
|
|
|
2007-09-28 14:45:26 +00:00
|
|
|
push(rbp);
|
2007-09-27 22:20:54 +00:00
|
|
|
push(rax);
|
|
|
|
push(rbp, FrameThread);
|
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
mov(rax, MethodCompiled, rax); // load compiled code
|
|
|
|
add(CompiledBody, rax);
|
|
|
|
call(rax); // call compiled code
|
|
|
|
|
|
|
|
add(footprint, rsp); // pop arguments
|
|
|
|
|
2007-09-29 20:24:14 +00:00
|
|
|
pushReturnValue(t, methodReturnCode(t, target));
|
2007-09-26 23:23:03 +00:00
|
|
|
} break;
|
|
|
|
|
|
|
|
case ldc:
|
|
|
|
case ldc_w: {
|
|
|
|
uint16_t index;
|
|
|
|
|
|
|
|
if (instruction == ldc) {
|
|
|
|
index = codeBody(t, code, ip++);
|
|
|
|
} else {
|
|
|
|
uint8_t index1 = codeBody(t, code, ip++);
|
|
|
|
uint8_t index2 = codeBody(t, code, ip++);
|
|
|
|
index = (index1 << 8) | index2;
|
|
|
|
}
|
|
|
|
|
|
|
|
object v = arrayBody(t, codePool(t, code), index - 1);
|
|
|
|
|
|
|
|
if (objectClass(t, v) == arrayBody(t, t->m->types, Machine::IntType)) {
|
|
|
|
push(intValue(t, v));
|
|
|
|
} else if (objectClass(t, v)
|
|
|
|
== arrayBody(t, t->m->types, Machine::FloatType))
|
|
|
|
{
|
|
|
|
push(floatValue(t, v));
|
|
|
|
} else if (objectClass(t, v)
|
|
|
|
== arrayBody(t, t->m->types, Machine::StringType))
|
|
|
|
{
|
2007-09-27 22:20:54 +00:00
|
|
|
pushAddress(reinterpret_cast<uintptr_t>(v));
|
2007-09-26 23:23:03 +00:00
|
|
|
} else {
|
|
|
|
object class_ = resolveClass(t, codePool(t, code), index - 1);
|
|
|
|
|
2007-09-27 22:20:54 +00:00
|
|
|
pushAddress(reinterpret_cast<uintptr_t>(class_));
|
2007-09-26 23:23:03 +00:00
|
|
|
}
|
|
|
|
} break;
|
|
|
|
|
2007-09-27 22:20:54 +00:00
|
|
|
case new_: {
|
|
|
|
uint16_t index = codeReadInt16(t, code, ip);
|
|
|
|
|
|
|
|
object class_ = resolveClass(t, codePool(t, code), index - 1);
|
|
|
|
if (UNLIKELY(t->exception)) return;
|
|
|
|
PROTECT(t, class_);
|
|
|
|
|
|
|
|
initClass(t, class_);
|
|
|
|
if (UNLIKELY(t->exception)) return;
|
2007-09-28 14:45:26 +00:00
|
|
|
|
2007-09-27 22:20:54 +00:00
|
|
|
if (classVmFlags(t, class_) & WeakReferenceFlag) {
|
2007-09-28 14:45:26 +00:00
|
|
|
compileCall1(reinterpret_cast<uintptr_t>(makeNewWeakReference),
|
|
|
|
reinterpret_cast<uintptr_t>(class_));
|
2007-09-27 22:20:54 +00:00
|
|
|
} else {
|
2007-09-28 14:45:26 +00:00
|
|
|
compileCall1(reinterpret_cast<uintptr_t>(makeNew),
|
|
|
|
reinterpret_cast<uintptr_t>(class_));
|
2007-09-27 22:20:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
push(rax);
|
|
|
|
} break;
|
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
case pop_: {
|
|
|
|
add(BytesPerWord, rsp);
|
|
|
|
} break;
|
|
|
|
|
2007-09-29 20:24:14 +00:00
|
|
|
case putfield: {
|
|
|
|
uint16_t index = codeReadInt16(t, code, ip);
|
|
|
|
|
|
|
|
object field = resolveField(t, codePool(t, code), index - 1);
|
|
|
|
if (UNLIKELY(t->exception)) return;
|
|
|
|
|
|
|
|
switch (fieldCode(t, field)) {
|
|
|
|
case ByteField:
|
|
|
|
case BooleanField:
|
|
|
|
case CharField:
|
|
|
|
case ShortField:
|
|
|
|
case FloatField:
|
|
|
|
case IntField: {
|
|
|
|
pop(rcx);
|
|
|
|
pop(rax);
|
|
|
|
switch (fieldCode(t, field)) {
|
|
|
|
case ByteField:
|
|
|
|
case BooleanField:
|
|
|
|
mov1(rcx, rax, fieldOffset(t, field));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CharField:
|
|
|
|
case ShortField:
|
|
|
|
mov2(rcx, rax, fieldOffset(t, field));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case FloatField:
|
|
|
|
case IntField:
|
|
|
|
mov4(rcx, rax, fieldOffset(t, field));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case DoubleField:
|
|
|
|
case LongField: {
|
|
|
|
pop(rcx);
|
|
|
|
pop(rdx);
|
|
|
|
pop(rax);
|
|
|
|
mov4(rcx, rax, fieldOffset(t, field));
|
|
|
|
mov4(rdx, rax, fieldOffset(t, field) + 4);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case ObjectField: {
|
|
|
|
pop(rcx);
|
|
|
|
pop(rax);
|
|
|
|
mov(rcx, rax, fieldOffset(t, field));
|
|
|
|
} break;
|
|
|
|
|
|
|
|
default: abort(t);
|
|
|
|
}
|
|
|
|
} break;
|
|
|
|
|
2007-09-28 14:45:26 +00:00
|
|
|
case putstatic: {
|
|
|
|
uint16_t index = codeReadInt16(t, code, ip);
|
|
|
|
|
|
|
|
object field = resolveField(t, codePool(t, code), index - 1);
|
|
|
|
if (UNLIKELY(t->exception)) return;
|
|
|
|
|
|
|
|
initClass(t, fieldClass(t, field));
|
|
|
|
if (UNLIKELY(t->exception)) return;
|
|
|
|
|
|
|
|
object table = classStaticTable(t, fieldClass(t, field));
|
|
|
|
|
|
|
|
mov(reinterpret_cast<uintptr_t>(table), rax);
|
|
|
|
add(fieldOffset(t, field), rax);
|
|
|
|
|
|
|
|
switch (fieldCode(t, field)) {
|
|
|
|
case ByteField:
|
|
|
|
case BooleanField:
|
|
|
|
case CharField:
|
|
|
|
case ShortField:
|
|
|
|
case FloatField:
|
|
|
|
case IntField: {
|
|
|
|
compileCall1(reinterpret_cast<uintptr_t>(makeNew),
|
|
|
|
reinterpret_cast<uintptr_t>
|
|
|
|
(arrayBody(t, t->m->types, Machine::IntType)));
|
|
|
|
|
|
|
|
pop4(rax, IntValue);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case DoubleField:
|
|
|
|
case LongField: {
|
|
|
|
compileCall1(reinterpret_cast<uintptr_t>(makeNew),
|
|
|
|
reinterpret_cast<uintptr_t>
|
|
|
|
(arrayBody(t, t->m->types, Machine::LongType)));
|
|
|
|
|
|
|
|
pop4(rax, LongValue);
|
|
|
|
pop4(rax, LongValue + 4);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case ObjectField:
|
|
|
|
pop(rax, 0);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default: abort(t);
|
|
|
|
}
|
|
|
|
} break;
|
|
|
|
|
2007-09-24 01:39:03 +00:00
|
|
|
case return_:
|
2007-09-26 23:23:03 +00:00
|
|
|
mov(rbp, rsp);
|
|
|
|
pop(rbp);
|
2007-09-24 01:39:03 +00:00
|
|
|
ret();
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
abort(t);
|
|
|
|
}
|
|
|
|
}
|
2007-09-28 23:41:03 +00:00
|
|
|
|
|
|
|
resolveJumps();
|
|
|
|
}
|
|
|
|
|
|
|
|
void resolveJumps() {
|
|
|
|
for (unsigned i = 0; i < jumps.length(); i += 8) {
|
|
|
|
uint32_t ip = jumps.get4(i);
|
|
|
|
uint32_t offset = jumps.get4(i + 4);
|
|
|
|
|
|
|
|
unsigned bottom = 0;
|
|
|
|
unsigned top = javaIPs.length() / 2;
|
2007-09-29 20:24:14 +00:00
|
|
|
bool success = false;
|
2007-09-28 23:41:03 +00:00
|
|
|
for (unsigned span = top - bottom; span; span = top - bottom) {
|
|
|
|
unsigned middle = bottom + (span / 2);
|
|
|
|
uint32_t k = javaIPs.get2(middle * 2);
|
|
|
|
|
|
|
|
if (ip < k) {
|
|
|
|
top = middle;
|
|
|
|
} else if (ip > k) {
|
|
|
|
bottom = middle + 1;
|
|
|
|
} else {
|
|
|
|
code.set4(offset, machineIPs.get4(middle * 4) - (offset + 4));
|
2007-09-29 20:24:14 +00:00
|
|
|
success = true;
|
2007-09-28 23:41:03 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2007-09-29 20:24:14 +00:00
|
|
|
assert(code.s, success);
|
2007-09-28 23:41:03 +00:00
|
|
|
}
|
2007-09-24 01:39:03 +00:00
|
|
|
}
|
2007-09-25 23:53:11 +00:00
|
|
|
|
|
|
|
void compileStub(MyThread* t) {
|
|
|
|
unsigned frameOffset = reinterpret_cast<uintptr_t>(&(t->frame))
|
|
|
|
- reinterpret_cast<uintptr_t>(t);
|
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
push(rbp);
|
|
|
|
mov(rsp, rbp);
|
|
|
|
|
|
|
|
mov(rbp, FrameThread, rax);
|
|
|
|
mov(rbp, rax, frameOffset); // set thread frame to current
|
2007-09-25 23:53:11 +00:00
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
if (BytesPerWord == 4) {
|
|
|
|
push(rbp, FrameMethod);
|
|
|
|
push(rbp, FrameThread);
|
|
|
|
} else {
|
|
|
|
mov(rbp, FrameMethod, rsi);
|
|
|
|
mov(rbp, FrameThread, rdi);
|
|
|
|
}
|
2007-09-25 23:53:11 +00:00
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
mov(reinterpret_cast<uintptr_t>(compileMethod), rax);
|
|
|
|
call(rax);
|
|
|
|
|
|
|
|
if (BytesPerWord == 4) {
|
|
|
|
add(BytesPerWord * 2, rsp);
|
|
|
|
}
|
2007-09-25 23:53:11 +00:00
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
mov(rbp, FrameMethod, rax);
|
|
|
|
mov(rax, MethodCompiled, rax); // load compiled code
|
2007-09-25 23:53:11 +00:00
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
mov(rbp, rsp);
|
|
|
|
pop(rbp);
|
2007-09-25 23:53:11 +00:00
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
add(CompiledBody, rax);
|
|
|
|
jmp(rax); // call compiled code
|
2007-09-25 23:53:11 +00:00
|
|
|
}
|
2007-09-28 23:41:03 +00:00
|
|
|
|
|
|
|
Buffer javaIPs;
|
|
|
|
Buffer machineIPs;
|
2007-09-24 01:39:03 +00:00
|
|
|
};
|
|
|
|
|
2007-09-25 23:53:11 +00:00
|
|
|
void
|
2007-09-26 23:23:03 +00:00
|
|
|
compileMethod2(MyThread* t, object method)
|
2007-09-25 23:53:11 +00:00
|
|
|
{
|
|
|
|
if (methodCompiled(t, method) == t->m->processor->methodStub(t)) {
|
|
|
|
PROTECT(t, method);
|
|
|
|
|
|
|
|
ACQUIRE(t, t->m->classLock);
|
|
|
|
|
|
|
|
if (methodCompiled(t, method) == t->m->processor->methodStub(t)) {
|
2007-09-29 20:24:14 +00:00
|
|
|
fprintf(stderr, "compiling %s.%s\n",
|
|
|
|
&byteArrayBody(t, className(t, methodClass(t, method)), 0),
|
|
|
|
&byteArrayBody(t, methodName(t, method), 0));
|
|
|
|
|
2007-09-25 23:53:11 +00:00
|
|
|
Compiler c(t->m->system);
|
|
|
|
c.compile(t, method);
|
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
object compiled = makeCompiled(t, 0, c.code.length(), false);
|
|
|
|
if (UNLIKELY(t->exception)) return;
|
|
|
|
|
|
|
|
c.code.copyTo(&compiledBody(t, compiled, 0));
|
2007-09-25 23:53:11 +00:00
|
|
|
|
2007-09-29 20:24:14 +00:00
|
|
|
fprintf(stderr, "compiled %s.%s from %p to %p\n",
|
|
|
|
&byteArrayBody(t, className(t, methodClass(t, method)), 0),
|
|
|
|
&byteArrayBody(t, methodName(t, method), 0),
|
|
|
|
&compiledBody(t, compiled, 0),
|
|
|
|
&compiledBody(t, compiled, 0) + compiledLength(t, compiled));
|
|
|
|
|
2007-09-25 23:53:11 +00:00
|
|
|
set(t, methodCompiled(t, method), compiled);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
void
|
|
|
|
updateCaller(MyThread* t, object method)
|
|
|
|
{
|
|
|
|
uintptr_t stub = reinterpret_cast<uintptr_t>
|
|
|
|
(&compiledBody(t, t->m->processor->methodStub(t), 0));
|
|
|
|
|
|
|
|
Assembler a(t->m->system);
|
2007-09-28 23:41:03 +00:00
|
|
|
a.mov(stub, rax);
|
2007-09-26 23:23:03 +00:00
|
|
|
unsigned offset = a.code.length() - BytesPerWord;
|
|
|
|
|
2007-09-28 23:41:03 +00:00
|
|
|
a.call(rax);
|
2007-09-26 23:23:03 +00:00
|
|
|
|
|
|
|
uint8_t* caller = static_cast<uint8_t**>(t->frame)[1] - a.code.length();
|
2007-09-28 23:41:03 +00:00
|
|
|
if (memcmp(a.code.data, caller, a.code.length()) == 0) {
|
2007-09-26 23:23:03 +00:00
|
|
|
// it's a direct call - update caller to point to new code
|
|
|
|
|
|
|
|
// address must be aligned on a word boundary for this write to
|
|
|
|
// be atomic
|
|
|
|
assert(t, reinterpret_cast<uintptr_t>(caller + offset)
|
|
|
|
% BytesPerWord == 0);
|
|
|
|
|
|
|
|
*reinterpret_cast<void**>(caller + offset)
|
|
|
|
= &compiledBody(t, methodCompiled(t, method), 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
unwind(Thread* t)
|
|
|
|
{
|
|
|
|
// todo
|
|
|
|
abort(t);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
compileMethod(MyThread* t, object method)
|
|
|
|
{
|
|
|
|
compileMethod2(t, method);
|
2007-09-27 22:20:54 +00:00
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
if (UNLIKELY(t->exception)) {
|
|
|
|
unwind(t);
|
2007-09-27 22:20:54 +00:00
|
|
|
} else if (not methodVirtual(t, method)) {
|
2007-09-26 23:23:03 +00:00
|
|
|
updateCaller(t, method);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-09-24 01:39:03 +00:00
|
|
|
object
|
2007-09-25 23:53:11 +00:00
|
|
|
compileStub(Thread* t)
|
2007-09-24 01:39:03 +00:00
|
|
|
{
|
2007-09-25 23:53:11 +00:00
|
|
|
Compiler c(t->m->system);
|
|
|
|
c.compileStub(static_cast<MyThread*>(t));
|
2007-09-24 01:39:03 +00:00
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
object stub = makeCompiled(t, 0, c.code.length(), false);
|
|
|
|
c.code.copyTo(&compiledBody(t, stub, 0));
|
2007-09-25 23:53:11 +00:00
|
|
|
|
|
|
|
return stub;
|
|
|
|
}
|
|
|
|
|
|
|
|
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)),
|
|
|
|
next(this->t->argumentList),
|
|
|
|
array(array),
|
|
|
|
objectMask(objectMask),
|
|
|
|
position(0)
|
|
|
|
{
|
|
|
|
this->t->argumentList = this;
|
|
|
|
|
|
|
|
addInt(reinterpret_cast<uintptr_t>(t));
|
|
|
|
addObject(0); // reserve space for method
|
|
|
|
addInt(reinterpret_cast<uintptr_t>(this->t->frame));
|
|
|
|
|
|
|
|
if (this_) {
|
|
|
|
addObject(this_);
|
|
|
|
}
|
|
|
|
|
|
|
|
const char* s = spec;
|
|
|
|
++ s; // skip '('
|
|
|
|
while (*s and *s != ')') {
|
|
|
|
switch (*s) {
|
|
|
|
case 'L':
|
|
|
|
while (*s and *s != ';') ++ s;
|
|
|
|
++ s;
|
|
|
|
|
|
|
|
if (indirectObjects) {
|
|
|
|
object* v = va_arg(arguments, object*);
|
|
|
|
addObject(v ? *v : 0);
|
|
|
|
} else {
|
|
|
|
addObject(va_arg(arguments, object));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case '[':
|
|
|
|
while (*s == '[') ++ s;
|
|
|
|
switch (*s) {
|
|
|
|
case 'L':
|
|
|
|
while (*s and *s != ';') ++ s;
|
|
|
|
++ s;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
++ s;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (indirectObjects) {
|
|
|
|
object* v = va_arg(arguments, object*);
|
|
|
|
addObject(v ? *v : 0);
|
|
|
|
} else {
|
|
|
|
addObject(va_arg(arguments, object));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'J':
|
|
|
|
case 'D':
|
|
|
|
++ s;
|
|
|
|
addLong(va_arg(arguments, uint64_t));
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
++ s;
|
|
|
|
addInt(va_arg(arguments, uint32_t));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ArgumentList(Thread* t, uintptr_t* array, bool* objectMask, object this_,
|
|
|
|
const char* spec, object arguments):
|
|
|
|
t(static_cast<MyThread*>(t)),
|
|
|
|
next(this->t->argumentList),
|
|
|
|
array(array),
|
|
|
|
objectMask(objectMask),
|
|
|
|
position(0)
|
|
|
|
{
|
|
|
|
this->t->argumentList = this;
|
|
|
|
|
|
|
|
addInt(0); // reserve space for trace pointer
|
|
|
|
addObject(0); // reserve space for method pointer
|
|
|
|
|
|
|
|
if (this_) {
|
|
|
|
addObject(this_);
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned index = 0;
|
|
|
|
const char* s = spec;
|
|
|
|
++ s; // skip '('
|
|
|
|
while (*s and *s != ')') {
|
|
|
|
switch (*s) {
|
|
|
|
case 'L':
|
|
|
|
while (*s and *s != ';') ++ s;
|
|
|
|
++ s;
|
|
|
|
addObject(objectArrayBody(t, arguments, index++));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case '[':
|
|
|
|
while (*s == '[') ++ s;
|
|
|
|
switch (*s) {
|
|
|
|
case 'L':
|
|
|
|
while (*s and *s != ';') ++ s;
|
|
|
|
++ s;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
++ s;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
addObject(objectArrayBody(t, arguments, index++));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'J':
|
|
|
|
case 'D':
|
|
|
|
++ s;
|
|
|
|
addLong(cast<int64_t>(objectArrayBody(t, arguments, index++),
|
|
|
|
BytesPerWord));
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
++ s;
|
|
|
|
addInt(cast<int32_t>(objectArrayBody(t, arguments, index++),
|
|
|
|
BytesPerWord));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
~ArgumentList() {
|
|
|
|
t->argumentList = next;
|
|
|
|
}
|
|
|
|
|
|
|
|
void addObject(object v) {
|
|
|
|
array[position] = reinterpret_cast<uintptr_t>(v);
|
|
|
|
objectMask[position] = true;
|
|
|
|
++ position;
|
|
|
|
}
|
|
|
|
|
|
|
|
void addInt(uint32_t v) {
|
|
|
|
array[position] = v;
|
|
|
|
objectMask[position] = false;
|
|
|
|
++ position;
|
|
|
|
}
|
|
|
|
|
|
|
|
void addLong(uint64_t v) {
|
|
|
|
memcpy(array + position, &v, 8);
|
|
|
|
objectMask[position] = false;
|
|
|
|
objectMask[position] = false;
|
|
|
|
position += 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
MyThread* t;
|
|
|
|
ArgumentList* next;
|
|
|
|
uintptr_t* array;
|
|
|
|
bool* objectMask;
|
|
|
|
unsigned position;
|
|
|
|
};
|
|
|
|
|
|
|
|
object
|
|
|
|
invoke(Thread* thread, object method, ArgumentList* arguments)
|
|
|
|
{
|
|
|
|
MyThread* t = static_cast<MyThread*>(thread);
|
|
|
|
|
|
|
|
arguments->array[1] = reinterpret_cast<uintptr_t>(method);
|
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
unsigned returnCode = methodReturnCode(t, method);
|
2007-09-25 23:53:11 +00:00
|
|
|
unsigned returnType = fieldType(t, returnCode);
|
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
uint64_t result = vmInvoke
|
2007-09-25 23:53:11 +00:00
|
|
|
(&compiledBody(t, methodCompiled(t, method), 0), arguments->array,
|
2007-09-26 23:23:03 +00:00
|
|
|
arguments->position * BytesPerWord, 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:
|
|
|
|
r = (result == 0 ? 0 :
|
|
|
|
*reinterpret_cast<object*>(static_cast<uintptr_t>(result)));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VoidField:
|
|
|
|
r = 0;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
abort(t);
|
|
|
|
};
|
2007-09-24 01:39:03 +00:00
|
|
|
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2007-09-25 23:53:11 +00:00
|
|
|
class MyProcessor: public Processor {
|
2007-09-24 01:39:03 +00:00
|
|
|
public:
|
2007-09-25 23:53:11 +00:00
|
|
|
MyProcessor(System* s):
|
|
|
|
s(s),
|
|
|
|
stub(0)
|
2007-09-24 01:39:03 +00:00
|
|
|
{ }
|
|
|
|
|
2007-09-25 23:53:11 +00:00
|
|
|
virtual Thread*
|
|
|
|
makeThread(Machine* m, object javaThread, Thread* parent)
|
|
|
|
{
|
|
|
|
return new (s->allocate(sizeof(MyThread))) MyThread(m, javaThread, parent);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual object
|
|
|
|
methodStub(Thread* t)
|
|
|
|
{
|
|
|
|
if (stub == 0) {
|
|
|
|
stub = compileStub(t);
|
|
|
|
}
|
|
|
|
return stub;
|
|
|
|
}
|
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
virtual unsigned
|
|
|
|
parameterFootprint(vm::Thread*, const char* s, bool static_)
|
|
|
|
{
|
|
|
|
unsigned footprint = 0;
|
|
|
|
++ s; // skip '('
|
|
|
|
while (*s and *s != ')') {
|
|
|
|
switch (*s) {
|
|
|
|
case 'L':
|
|
|
|
while (*s and *s != ';') ++ s;
|
|
|
|
++ s;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case '[':
|
|
|
|
while (*s == '[') ++ s;
|
|
|
|
switch (*s) {
|
|
|
|
case 'L':
|
|
|
|
while (*s and *s != ';') ++ s;
|
|
|
|
++ s;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
++ s;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'J':
|
|
|
|
case 'D':
|
|
|
|
++ s;
|
|
|
|
if (BytesPerWord == 4) {
|
|
|
|
++ footprint;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
++ s;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
++ footprint;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (not static_) {
|
|
|
|
++ footprint;
|
|
|
|
}
|
|
|
|
return footprint;
|
|
|
|
}
|
|
|
|
|
|
|
|
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)
|
|
|
|
{
|
2007-09-28 23:41:03 +00:00
|
|
|
classVmFlags(t, c) |= InitFlag;
|
2007-09-26 23:23:03 +00:00
|
|
|
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* t, Heap::Visitor*)
|
|
|
|
{
|
|
|
|
abort(t);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual uintptr_t
|
|
|
|
frameStart(Thread* t)
|
|
|
|
{
|
|
|
|
abort(t);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual uintptr_t
|
|
|
|
frameNext(Thread* t, uintptr_t)
|
|
|
|
{
|
|
|
|
abort(t);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual bool
|
|
|
|
frameValid(Thread* t, uintptr_t)
|
|
|
|
{
|
|
|
|
abort(t);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual object
|
|
|
|
frameMethod(Thread* t, uintptr_t)
|
|
|
|
{
|
|
|
|
abort(t);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual unsigned
|
|
|
|
frameIp(Thread* t, uintptr_t)
|
|
|
|
{
|
|
|
|
abort(t);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual object*
|
|
|
|
makeLocalReference(Thread* t, object)
|
|
|
|
{
|
|
|
|
abort(t);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void
|
|
|
|
disposeLocalReference(Thread* t, object*)
|
|
|
|
{
|
|
|
|
abort(t);
|
|
|
|
}
|
|
|
|
|
2007-09-24 01:39:03 +00:00
|
|
|
virtual object
|
|
|
|
invokeArray(Thread* t, object method, object this_, object arguments)
|
|
|
|
{
|
|
|
|
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-09-26 23:23:03 +00:00
|
|
|
unsigned size = methodParameterFootprint(t, method) + FrameFootprint;
|
2007-09-25 23:53:11 +00:00
|
|
|
uintptr_t array[size];
|
|
|
|
bool objectMask[size];
|
|
|
|
ArgumentList list(t, array, objectMask, this_, spec, arguments);
|
2007-09-24 01:39:03 +00:00
|
|
|
|
2007-09-25 23:53:11 +00:00
|
|
|
return ::invoke(t, method, &list);
|
2007-09-24 01:39:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual object
|
|
|
|
invokeList(Thread* t, object method, object this_, bool indirectObjects,
|
|
|
|
va_list arguments)
|
|
|
|
{
|
|
|
|
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-09-26 23:23:03 +00:00
|
|
|
unsigned size = methodParameterFootprint(t, method) + FrameFootprint;
|
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-09-24 01:39:03 +00:00
|
|
|
|
2007-09-25 23:53:11 +00:00
|
|
|
return ::invoke(t, method, &list);
|
2007-09-24 01:39:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual object
|
|
|
|
invokeList(Thread* t, const char* className, const char* methodName,
|
|
|
|
const char* methodSpec, object this_, va_list arguments)
|
|
|
|
{
|
|
|
|
assert(t, t->state == Thread::ActiveState
|
|
|
|
or t->state == Thread::ExclusiveState);
|
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
unsigned size = parameterFootprint(t, methodSpec, false) + FrameFootprint;
|
2007-09-25 23:53:11 +00:00
|
|
|
uintptr_t array[size];
|
|
|
|
bool objectMask[size];
|
|
|
|
ArgumentList list
|
|
|
|
(t, array, objectMask, this_, methodSpec, false, arguments);
|
2007-09-24 01:39:03 +00:00
|
|
|
|
|
|
|
object method = resolveMethod(t, className, methodName, methodSpec);
|
|
|
|
if (LIKELY(t->exception == 0)) {
|
|
|
|
assert(t, ((methodFlags(t, method) & ACC_STATIC) == 0) xor (this_ == 0));
|
|
|
|
|
2007-09-25 23:53:11 +00:00
|
|
|
return ::invoke(t, method, &list);
|
2007-09-24 01:39:03 +00:00
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void dispose() {
|
|
|
|
s->free(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
System* s;
|
2007-09-25 23:53:11 +00:00
|
|
|
object stub;
|
2007-09-24 01:39:03 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
namespace vm {
|
|
|
|
|
2007-09-25 23:53:11 +00:00
|
|
|
Processor*
|
|
|
|
makeProcessor(System* system)
|
2007-09-24 01:39:03 +00:00
|
|
|
{
|
2007-09-25 23:53:11 +00:00
|
|
|
return new (system->allocate(sizeof(MyProcessor))) MyProcessor(system);
|
2007-09-24 01:39:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace vm
|
2007-09-25 23:53:11 +00:00
|
|
|
|