2007-12-08 23:22:13 +00:00
|
|
|
#include "compiler.h"
|
2007-12-09 20:03:21 +00:00
|
|
|
#include "vector.h"
|
2007-12-08 23:22:13 +00:00
|
|
|
|
|
|
|
using namespace vm;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2007-12-09 22:45:43 +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,
|
|
|
|
};
|
|
|
|
|
2007-12-09 20:03:21 +00:00
|
|
|
class IpMapping {
|
|
|
|
public:
|
|
|
|
IpMapping(unsigned ip, unsigned offset): ip(ip), offset(offset) { }
|
|
|
|
|
|
|
|
const unsigned ip;
|
|
|
|
const unsigned offset;
|
|
|
|
};
|
|
|
|
|
|
|
|
class Context {
|
|
|
|
public:
|
|
|
|
Context(System* s, void* indirectCaller):
|
|
|
|
s(s),
|
|
|
|
code(s, 1024),
|
2007-12-09 23:06:47 +00:00
|
|
|
logicalStack(s, BytesPerWord * 32),
|
2007-12-09 20:03:21 +00:00
|
|
|
operands(s, 8 * 1024),
|
|
|
|
ipTable(s, sizeof(IpMapping) * 512),
|
|
|
|
constantPool(s, BytesPerWord * 32),
|
|
|
|
registerPool(s, BytesPerWord * 8),
|
|
|
|
promises(s, 1024),
|
2007-12-09 22:45:43 +00:00
|
|
|
indirectCaller(reinterpret_cast<intptr_t>(indirectCaller)),
|
|
|
|
stackIndex(- BytesPerWord)
|
2007-12-09 20:03:21 +00:00
|
|
|
{ }
|
|
|
|
|
|
|
|
void dispose() {
|
|
|
|
promises.dispose();
|
|
|
|
constantPool.dispose();
|
|
|
|
registerPool.dispose();
|
|
|
|
ipTable.dispose();
|
|
|
|
operands.dispose();
|
2007-12-09 23:06:47 +00:00
|
|
|
logicalStack.dispose();
|
2007-12-09 20:03:21 +00:00
|
|
|
code.dispose();
|
|
|
|
}
|
|
|
|
|
|
|
|
System* s;
|
|
|
|
Vector code;
|
2007-12-09 23:06:47 +00:00
|
|
|
Vector logicalStack;
|
2007-12-09 20:03:21 +00:00
|
|
|
Vector operands;
|
2007-12-09 22:45:43 +00:00
|
|
|
Vector ipTable;
|
|
|
|
Vector constantPool;
|
|
|
|
Vector registerPool;
|
2007-12-09 20:03:21 +00:00
|
|
|
Vector promises;
|
2007-12-09 22:45:43 +00:00
|
|
|
intptr_t indirectCaller;
|
|
|
|
int stackIndex;
|
2007-12-09 20:03:21 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
inline void NO_RETURN
|
|
|
|
abort(Context* c)
|
|
|
|
{
|
|
|
|
abort(c->s);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifndef NDEBUG
|
|
|
|
inline void
|
|
|
|
assert(Context* c, bool v)
|
|
|
|
{
|
|
|
|
assert(c->s, v);
|
|
|
|
}
|
|
|
|
#endif // not NDEBUG
|
|
|
|
|
|
|
|
inline void
|
|
|
|
expect(Context* c, bool v)
|
|
|
|
{
|
2007-12-09 22:45:43 +00:00
|
|
|
expect(c->s, v);
|
2007-12-09 20:03:21 +00:00
|
|
|
}
|
|
|
|
|
2007-12-08 23:22:13 +00:00
|
|
|
class MyPromise: public Promise {
|
|
|
|
public:
|
2007-12-09 20:03:21 +00:00
|
|
|
enum PromiseType {
|
|
|
|
PoolPromiseType,
|
|
|
|
CodePromiseType,
|
|
|
|
IpPromiseType
|
|
|
|
};
|
|
|
|
|
|
|
|
MyPromise(intptr_t value): resolved(false), value_(value) { }
|
2007-12-08 23:22:13 +00:00
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
virtual unsigned value(System* s UNUSED) {
|
2007-12-08 23:22:13 +00:00
|
|
|
assert(s, resolved);
|
|
|
|
return value_;
|
|
|
|
}
|
|
|
|
|
2007-12-09 20:03:21 +00:00
|
|
|
virtual PromiseType type() = 0;
|
|
|
|
|
2007-12-08 23:22:13 +00:00
|
|
|
bool resolved;
|
2007-12-09 20:03:21 +00:00
|
|
|
intptr_t value_;
|
|
|
|
};
|
|
|
|
|
|
|
|
class PoolPromise: public MyPromise {
|
|
|
|
public:
|
|
|
|
PoolPromise(intptr_t value): MyPromise(value) { }
|
|
|
|
|
|
|
|
virtual PromiseType type() {
|
|
|
|
return PoolPromiseType;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class CodePromise: public MyPromise {
|
|
|
|
public:
|
2007-12-09 22:45:43 +00:00
|
|
|
CodePromise(intptr_t value): MyPromise(value) { }
|
2007-12-09 20:03:21 +00:00
|
|
|
|
|
|
|
virtual PromiseType type() {
|
|
|
|
return CodePromiseType;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class IpPromise: public MyPromise {
|
|
|
|
public:
|
2007-12-09 22:45:43 +00:00
|
|
|
IpPromise(intptr_t value): MyPromise(value) { }
|
2007-12-09 20:03:21 +00:00
|
|
|
|
|
|
|
virtual PromiseType type() {
|
|
|
|
return IpPromiseType;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class MyOperand: public Operand {
|
|
|
|
public:
|
|
|
|
enum OperandType {
|
|
|
|
ImmediateOperandType,
|
2007-12-09 22:45:43 +00:00
|
|
|
AbsoluteOperandType,
|
2007-12-09 20:03:21 +00:00
|
|
|
RegisterOperandType,
|
2007-12-09 22:45:43 +00:00
|
|
|
MemoryOperandType,
|
|
|
|
SelectionOperandType
|
2007-12-09 20:03:21 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
virtual ~MyOperand() { }
|
|
|
|
|
|
|
|
virtual OperandType type() = 0;
|
|
|
|
|
|
|
|
virtual unsigned footprint() {
|
|
|
|
return BytesPerWord;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual unsigned size() = 0;
|
|
|
|
|
2007-12-09 23:06:47 +00:00
|
|
|
virtual void logicalPush(Context* c) { abort(c); }
|
|
|
|
|
|
|
|
virtual void logicalFlush(Context* c) { abort(c); }
|
|
|
|
|
2007-12-09 20:03:21 +00:00
|
|
|
virtual void push(Context* c) { abort(c); }
|
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
virtual void pop(Context* c) { abort(c); }
|
|
|
|
|
2007-12-09 20:03:21 +00:00
|
|
|
virtual void mov(Context* c, MyOperand*) { abort(c); }
|
|
|
|
|
|
|
|
virtual void cmp(Context* c, MyOperand*) { abort(c); }
|
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
virtual void call(Context* c) { abort(c); }
|
|
|
|
|
|
|
|
virtual void alignedCall(Context* c) { abort(c); }
|
|
|
|
|
2007-12-09 20:03:21 +00:00
|
|
|
virtual void jl(Context* c) { abort(c); }
|
|
|
|
|
|
|
|
virtual void jg(Context* c) { abort(c); }
|
|
|
|
|
|
|
|
virtual void jle(Context* c) { abort(c); }
|
|
|
|
|
|
|
|
virtual void jge(Context* c) { abort(c); }
|
|
|
|
|
|
|
|
virtual void je(Context* c) { abort(c); }
|
|
|
|
|
|
|
|
virtual void jne(Context* c) { abort(c); }
|
|
|
|
|
|
|
|
virtual void jmp(Context* c) { abort(c); }
|
|
|
|
|
|
|
|
virtual void add(Context* c, MyOperand*) { abort(c); }
|
|
|
|
|
|
|
|
virtual void sub(Context* c, MyOperand*) { abort(c); }
|
|
|
|
|
|
|
|
virtual void mul(Context* c, MyOperand*) { abort(c); }
|
|
|
|
|
|
|
|
virtual void div(Context* c, MyOperand*) { abort(c); }
|
|
|
|
|
|
|
|
virtual void rem(Context* c, MyOperand*) { abort(c); }
|
|
|
|
|
|
|
|
virtual void shl(Context* c, MyOperand*) { abort(c); }
|
|
|
|
|
|
|
|
virtual void shr(Context* c, MyOperand*) { abort(c); }
|
|
|
|
|
|
|
|
virtual void ushr(Context* c, MyOperand*) { abort(c); }
|
|
|
|
|
|
|
|
virtual void and_(Context* c, MyOperand*) { abort(c); }
|
|
|
|
|
|
|
|
virtual void or_(Context* c, MyOperand*) { abort(c); }
|
|
|
|
|
|
|
|
virtual void xor_(Context* c, MyOperand*) { abort(c); }
|
|
|
|
|
|
|
|
virtual void neg(Context* c) { abort(c); }
|
|
|
|
};
|
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
inline bool
|
|
|
|
isInt8(intptr_t v)
|
|
|
|
{
|
|
|
|
return v == static_cast<int8_t>(v);
|
|
|
|
}
|
2007-12-09 20:03:21 +00:00
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
inline bool
|
|
|
|
isInt32(intptr_t v)
|
|
|
|
{
|
|
|
|
return v == static_cast<int32_t>(v);
|
|
|
|
}
|
2007-12-09 20:03:21 +00:00
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
Register
|
|
|
|
registerValue(Context* c, MyOperand* v);
|
2007-12-09 20:03:21 +00:00
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
Promise*
|
|
|
|
absoluteValue(Context* c, MyOperand* v);
|
2007-12-09 20:03:21 +00:00
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
void
|
|
|
|
setAbsoluteValue(Context* c, MyOperand* v, Promise* value);
|
2007-12-09 20:03:21 +00:00
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
MyOperand*
|
|
|
|
memoryBase(Context* c, MyOperand* v);
|
2007-12-09 20:03:21 +00:00
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
int
|
|
|
|
memoryDisplacement(Context* c, MyOperand* v);
|
|
|
|
|
|
|
|
MyOperand*
|
|
|
|
memoryIndex(Context* c, MyOperand* v);
|
|
|
|
|
|
|
|
unsigned
|
|
|
|
memoryScale(Context* c, MyOperand* v);
|
|
|
|
|
|
|
|
Register
|
|
|
|
asRegister(Context* c, MyOperand* v);
|
|
|
|
|
|
|
|
void
|
|
|
|
rex(Context* c)
|
|
|
|
{
|
|
|
|
if (BytesPerWord == 8) {
|
|
|
|
c->code.append(0x48);
|
2007-12-09 20:03:21 +00:00
|
|
|
}
|
2007-12-09 22:45:43 +00:00
|
|
|
}
|
2007-12-09 20:03:21 +00:00
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
void
|
|
|
|
ret(Context* c)
|
|
|
|
{
|
|
|
|
c->code.append(0xc3);
|
|
|
|
}
|
2007-12-09 20:03:21 +00:00
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
void
|
|
|
|
encode(Context* c, uint8_t instruction, uint8_t zeroPrefix,
|
|
|
|
uint8_t bytePrefix, uint8_t wordPrefix,
|
|
|
|
Register a, Register b, int32_t offset)
|
|
|
|
{
|
|
|
|
c->code.append(instruction);
|
2007-12-09 20:03:21 +00:00
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
uint8_t prefix;
|
|
|
|
if (offset == 0 and b != rbp) {
|
|
|
|
prefix = zeroPrefix;
|
|
|
|
} else if (isInt8(offset)) {
|
|
|
|
prefix = bytePrefix;
|
|
|
|
} else {
|
|
|
|
prefix = wordPrefix;
|
2007-12-09 20:03:21 +00:00
|
|
|
}
|
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
c->code.append(prefix | (a << 3) | b);
|
|
|
|
|
|
|
|
if (b == rsp) {
|
|
|
|
c->code.append(0x24);
|
2007-12-09 20:03:21 +00:00
|
|
|
}
|
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
if (offset == 0 and b != rbp) {
|
|
|
|
// do nothing
|
|
|
|
} else if (isInt8(offset)) {
|
|
|
|
c->code.append(offset);
|
|
|
|
} else {
|
|
|
|
c->code.append4(offset);
|
|
|
|
}
|
|
|
|
}
|
2007-12-09 20:03:21 +00:00
|
|
|
|
|
|
|
class RegisterOperand: public MyOperand {
|
|
|
|
public:
|
|
|
|
RegisterOperand(Register value):
|
|
|
|
value(value)
|
|
|
|
{ }
|
|
|
|
|
|
|
|
virtual OperandType type() {
|
|
|
|
return RegisterOperandType;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual unsigned size() {
|
|
|
|
return sizeof(RegisterOperand);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void push(Context* c) {
|
|
|
|
c->code.append(0x50 | value);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void mov(Context* c, MyOperand* dst) {
|
|
|
|
switch (dst->type()) {
|
|
|
|
case RegisterOperandType:
|
|
|
|
if (value != registerValue(c, dst)) {
|
|
|
|
rex(c);
|
|
|
|
c->code.append(0x89);
|
|
|
|
c->code.append(0xc0 | (value << 3) | registerValue(c, dst));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MemoryOperandType:
|
|
|
|
rex(c);
|
2007-12-09 22:45:43 +00:00
|
|
|
encode(c, 0x89, 0, 0x40, 0x80, value,
|
|
|
|
asRegister(c, memoryBase(c, dst)),
|
2007-12-09 20:03:21 +00:00
|
|
|
memoryDisplacement(c, dst));
|
|
|
|
break;
|
|
|
|
|
|
|
|
default: abort(c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Register value;
|
2007-12-08 23:22:13 +00:00
|
|
|
};
|
|
|
|
|
2007-12-09 20:03:21 +00:00
|
|
|
RegisterOperand*
|
|
|
|
temporary(Context* c)
|
|
|
|
{
|
|
|
|
return c->registerPool.pop<RegisterOperand*>();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
release(Context* c, RegisterOperand* v)
|
|
|
|
{
|
2007-12-09 22:45:43 +00:00
|
|
|
c->registerPool.push(v);
|
2007-12-09 20:03:21 +00:00
|
|
|
}
|
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
class ImmediateOperand: public MyOperand {
|
|
|
|
public:
|
|
|
|
ImmediateOperand(intptr_t value):
|
|
|
|
value(value)
|
|
|
|
{ }
|
2007-12-09 20:03:21 +00:00
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
virtual OperandType type() {
|
|
|
|
return ImmediateOperandType;
|
2007-12-09 20:03:21 +00:00
|
|
|
}
|
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
virtual unsigned size() {
|
|
|
|
return sizeof(ImmediateOperand);
|
|
|
|
}
|
2007-12-09 20:03:21 +00:00
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
virtual void push(Context* c) {
|
|
|
|
if (isInt8(value)) {
|
|
|
|
c->code.append(0x6a);
|
|
|
|
c->code.append(value);
|
|
|
|
} else if (isInt32(value)) {
|
|
|
|
c->code.append(0x68);
|
|
|
|
c->code.append4(value);
|
|
|
|
} else {
|
|
|
|
RegisterOperand* tmp = temporary(c);
|
|
|
|
mov(c, tmp);
|
|
|
|
tmp->push(c);
|
|
|
|
release(c, tmp);
|
|
|
|
}
|
2007-12-09 20:03:21 +00:00
|
|
|
}
|
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
virtual void mov(Context* c, MyOperand* dst) {
|
|
|
|
switch (dst->type()) {
|
|
|
|
case RegisterOperandType:
|
|
|
|
rex(c);
|
|
|
|
c->code.append(0xb8 | registerValue(c, dst));
|
|
|
|
c->code.appendAddress(value);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MemoryOperandType:
|
|
|
|
rex(c);
|
|
|
|
encode(c, 0xc7, 0, 0x40, 0x80, rax,
|
|
|
|
asRegister(c, memoryBase(c, dst)),
|
|
|
|
memoryDisplacement(c, dst));
|
|
|
|
c->code.append4(value);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default: abort(c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
intptr_t value;
|
|
|
|
};
|
|
|
|
|
|
|
|
class AbsoluteOperand: public MyOperand {
|
|
|
|
public:
|
|
|
|
AbsoluteOperand(Promise* value):
|
|
|
|
value(value)
|
|
|
|
{ }
|
|
|
|
|
|
|
|
virtual OperandType type() {
|
|
|
|
return AbsoluteOperandType;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual unsigned size() {
|
|
|
|
return sizeof(AbsoluteOperand);
|
2007-12-09 20:03:21 +00:00
|
|
|
}
|
2007-12-09 22:45:43 +00:00
|
|
|
|
|
|
|
Promise* value;
|
|
|
|
};
|
2007-12-09 20:03:21 +00:00
|
|
|
|
|
|
|
Register
|
|
|
|
asRegister(Context* c, MyOperand* v)
|
|
|
|
{
|
2007-12-09 22:45:43 +00:00
|
|
|
if (v->type() == MyOperand::RegisterOperandType) {
|
|
|
|
return registerValue(c, v);
|
2007-12-09 20:03:21 +00:00
|
|
|
} else {
|
2007-12-09 22:45:43 +00:00
|
|
|
assert(c, v->type() == MyOperand::MemoryOperandType);
|
2007-12-09 20:03:21 +00:00
|
|
|
|
|
|
|
RegisterOperand* tmp = temporary(c);
|
|
|
|
v->mov(c, tmp);
|
|
|
|
Register r = tmp->value;
|
|
|
|
release(c, tmp);
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class MemoryOperand: public MyOperand {
|
|
|
|
public:
|
|
|
|
MemoryOperand(MyOperand* base, int displacement, MyOperand* index,
|
|
|
|
unsigned scale):
|
|
|
|
base(base),
|
|
|
|
displacement(displacement),
|
|
|
|
index(index),
|
|
|
|
scale(scale)
|
|
|
|
{ }
|
|
|
|
|
|
|
|
virtual OperandType type() {
|
|
|
|
return MemoryOperandType;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual unsigned size() {
|
|
|
|
return sizeof(MemoryOperand);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void push(Context* c) {
|
|
|
|
assert(c, index == 0);
|
|
|
|
assert(c, scale == 0);
|
|
|
|
|
|
|
|
encode(c, 0xff, 0x30, 0x70, 0xb0, rax, asRegister(c, base), displacement);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void mov(Context* c, MyOperand* dst) {
|
|
|
|
switch (dst->type()) {
|
|
|
|
case RegisterOperandType:
|
|
|
|
rex(c);
|
|
|
|
encode(c, 0x8b, 0, 0x40, 0x80, registerValue(c, dst),
|
2007-12-09 22:45:43 +00:00
|
|
|
asRegister(c, base), displacement);
|
2007-12-09 20:03:21 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case MemoryOperandType: {
|
|
|
|
RegisterOperand* tmp = temporary(c);
|
|
|
|
mov(c, tmp);
|
|
|
|
tmp->mov(c, dst);
|
|
|
|
release(c, tmp);
|
|
|
|
} break;
|
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
default: abort(c);
|
2007-12-09 20:03:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
MyOperand* base;
|
|
|
|
int displacement;
|
|
|
|
MyOperand* index;
|
|
|
|
unsigned scale;
|
|
|
|
};
|
|
|
|
|
|
|
|
class SelectionOperand: public MyOperand {
|
|
|
|
public:
|
2007-12-09 22:45:43 +00:00
|
|
|
enum SelectionType {
|
|
|
|
S1Selection,
|
|
|
|
S2Selection,
|
|
|
|
Z2Selection,
|
|
|
|
S4Selection,
|
|
|
|
S8Selection
|
|
|
|
};
|
|
|
|
|
|
|
|
SelectionOperand(SelectionType type, MyOperand* base):
|
|
|
|
selectionType(type), base(base)
|
2007-12-09 20:03:21 +00:00
|
|
|
{ }
|
|
|
|
|
|
|
|
virtual OperandType type() {
|
|
|
|
return SelectionOperandType;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual unsigned footprint() {
|
2007-12-09 22:45:43 +00:00
|
|
|
if (selectionType == S8Selection) {
|
2007-12-09 20:03:21 +00:00
|
|
|
return 8;
|
|
|
|
} else {
|
|
|
|
return 4;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual unsigned size() {
|
|
|
|
return sizeof(SelectionOperand);
|
|
|
|
}
|
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
SelectionType selectionType;
|
2007-12-09 20:03:21 +00:00
|
|
|
MyOperand* base;
|
|
|
|
};
|
|
|
|
|
|
|
|
ImmediateOperand*
|
|
|
|
immediate(Context* c, intptr_t v)
|
|
|
|
{
|
|
|
|
return c->operands.push(ImmediateOperand(v));
|
|
|
|
}
|
|
|
|
|
|
|
|
AbsoluteOperand*
|
2007-12-09 22:45:43 +00:00
|
|
|
absolute(Context* c, Promise* v)
|
2007-12-09 20:03:21 +00:00
|
|
|
{
|
|
|
|
return c->operands.push(AbsoluteOperand(v));
|
|
|
|
}
|
|
|
|
|
|
|
|
RegisterOperand*
|
|
|
|
register_(Context* c, Register v)
|
|
|
|
{
|
|
|
|
return c->operands.push(RegisterOperand(v));
|
|
|
|
}
|
|
|
|
|
|
|
|
MemoryOperand*
|
|
|
|
memory(Context* c, MyOperand* base, int displacement,
|
|
|
|
MyOperand* index, unsigned scale)
|
|
|
|
{
|
|
|
|
return c->operands.push(MemoryOperand(base, displacement, index, scale));
|
|
|
|
}
|
|
|
|
|
2007-12-09 23:06:47 +00:00
|
|
|
MemoryOperand*
|
2007-12-09 20:03:21 +00:00
|
|
|
stack(Context* c, int displacement)
|
|
|
|
{
|
2007-12-09 23:06:47 +00:00
|
|
|
return c->operands.push
|
|
|
|
(MemoryOperand(register_(c, rbp), displacement, 0, 1));
|
2007-12-09 20:03:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
MyOperand*
|
2007-12-09 22:45:43 +00:00
|
|
|
selection(Context* c, SelectionOperand::SelectionType type, MyOperand* base)
|
2007-12-09 20:03:21 +00:00
|
|
|
{
|
2007-12-09 22:45:43 +00:00
|
|
|
if ((type == SelectionOperand::S4Selection and BytesPerWord == 4)
|
|
|
|
or (type == SelectionOperand::S8Selection and BytesPerWord == 8))
|
2007-12-09 20:03:21 +00:00
|
|
|
{
|
|
|
|
return base;
|
|
|
|
} else {
|
|
|
|
return c->operands.push(SelectionOperand(type, base));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
flushStack(Context* c)
|
|
|
|
{
|
2007-12-09 23:06:47 +00:00
|
|
|
for (unsigned i = 0; i < c->logicalStack.length(); i += BytesPerWord) {
|
|
|
|
(*c->logicalStack.peek<MyOperand*>(i))->logicalFlush(c);
|
2007-12-09 20:03:21 +00:00
|
|
|
}
|
2007-12-09 23:06:47 +00:00
|
|
|
}
|
2007-12-09 20:03:21 +00:00
|
|
|
|
2007-12-09 23:06:47 +00:00
|
|
|
Register
|
|
|
|
gpRegister(Context* c, 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(c);
|
|
|
|
}
|
2007-12-09 22:45:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
unsigned
|
|
|
|
pushArguments(Context* c, unsigned count, va_list list)
|
|
|
|
{
|
2007-12-09 23:06:47 +00:00
|
|
|
flushStack(c);
|
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
MyOperand* arguments[count];
|
|
|
|
unsigned footprint = 0;
|
|
|
|
for (unsigned i = 0; i < count; ++i) {
|
|
|
|
arguments[i] = va_arg(list, MyOperand*);
|
|
|
|
footprint += pad(arguments[i]->footprint());
|
|
|
|
}
|
|
|
|
|
2007-12-09 23:06:47 +00:00
|
|
|
const int GprCount = 6;
|
2007-12-09 22:45:43 +00:00
|
|
|
for (int i = count - 1; i >= 0; --i) {
|
2007-12-09 23:06:47 +00:00
|
|
|
if (BytesPerWord == 8 and i < GprCount) {
|
|
|
|
arguments[i]->mov(c, register_(c, gpRegister(c, i)));
|
|
|
|
} else {
|
|
|
|
arguments[i]->push(c);
|
|
|
|
}
|
2007-12-09 22:45:43 +00:00
|
|
|
}
|
|
|
|
|
2007-12-09 23:06:47 +00:00
|
|
|
if (BytesPerWord == 8) {
|
|
|
|
if (footprint > GprCount * BytesPerWord) {
|
|
|
|
return footprint - GprCount * BytesPerWord;
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return footprint;
|
|
|
|
}
|
2007-12-09 20:03:21 +00:00
|
|
|
}
|
|
|
|
|
2007-12-08 23:22:13 +00:00
|
|
|
class MyCompiler: public Compiler {
|
|
|
|
public:
|
|
|
|
MyCompiler(System* s, void* indirectCaller):
|
2007-12-09 20:03:21 +00:00
|
|
|
c(s, indirectCaller)
|
2007-12-08 23:22:13 +00:00
|
|
|
{ }
|
|
|
|
|
|
|
|
virtual Promise* poolOffset() {
|
2007-12-09 22:45:43 +00:00
|
|
|
return c.promises.push
|
|
|
|
(PoolPromise(c.constantPool.length() / BytesPerWord));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual Promise* codeOffset() {
|
2007-12-09 22:45:43 +00:00
|
|
|
return c.promises.push(CodePromise(c.code.length()));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual Operand* poolAppend(Operand* v) {
|
2007-12-09 20:03:21 +00:00
|
|
|
Operand* r = absolute(&c, poolOffset());
|
2007-12-09 22:45:43 +00:00
|
|
|
c.constantPool.push(static_cast<MyOperand*>(v));
|
2007-12-08 23:22:13 +00:00
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual Operand* constant(intptr_t v) {
|
2007-12-09 20:03:21 +00:00
|
|
|
return immediate(&c, v);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void push(Operand* v) {
|
2007-12-09 23:06:47 +00:00
|
|
|
static_cast<MyOperand*>(v)->logicalPush(&c);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void push2(Operand* v) {
|
2007-12-09 20:03:21 +00:00
|
|
|
push(v);
|
2007-12-09 23:06:47 +00:00
|
|
|
if (BytesPerWord == 8) push(immediate(&c, 0));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual Operand* stack(unsigned index) {
|
2007-12-09 23:06:47 +00:00
|
|
|
return c.logicalStack.peek<MyOperand>
|
|
|
|
(c.logicalStack.length() - ((index + 1) * BytesPerWord));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual Operand* stack2(unsigned index) {
|
2007-12-09 22:45:43 +00:00
|
|
|
return stack(index);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual Operand* pop() {
|
2007-12-09 23:06:47 +00:00
|
|
|
return c.logicalStack.pop<MyOperand*>();
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual Operand* pop2() {
|
2007-12-09 20:03:21 +00:00
|
|
|
if (BytesPerWord == 8) pop();
|
|
|
|
return pop();
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void pop(Operand* dst) {
|
2007-12-09 23:06:47 +00:00
|
|
|
c.logicalStack.pop<MyOperand*>()->mov(&c, static_cast<MyOperand*>(dst));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void pop2(Operand* dst) {
|
2007-12-09 20:03:21 +00:00
|
|
|
if (BytesPerWord == 8) pop();
|
|
|
|
pop(dst);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual Operand* stack() {
|
2007-12-09 20:03:21 +00:00
|
|
|
flushStack(&c);
|
|
|
|
return register_(&c, rsp);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual Operand* base() {
|
2007-12-09 20:03:21 +00:00
|
|
|
return register_(&c, rbp);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual Operand* thread() {
|
2007-12-09 20:03:21 +00:00
|
|
|
return register_(&c, rbx);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual Operand* indirectTarget() {
|
2007-12-09 20:03:21 +00:00
|
|
|
return register_(&c, rax);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual Operand* temporary() {
|
2007-12-09 20:03:21 +00:00
|
|
|
return ::temporary(&c);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void release(Operand* v) {
|
2007-12-09 20:03:21 +00:00
|
|
|
assert(&c, static_cast<MyOperand>(v)->type() == RegisterOperandType);
|
|
|
|
return ::release(&c, static_cast<RegisterOperand*>(v));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual Operand* label() {
|
2007-12-09 20:03:21 +00:00
|
|
|
return absolute(&c, 0);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void mark(Operand* label) {
|
2007-12-09 20:03:21 +00:00
|
|
|
setAbsoluteValue(&c, static_cast<MyOperand*>(label), codeOffset());
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual Operand* call(Operand* v) {
|
2007-12-09 20:03:21 +00:00
|
|
|
flushStack(&c);
|
|
|
|
static_cast<MyOperand*>(v)->call(&c);
|
|
|
|
return register_(&c, rax);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual Operand* alignedCall(Operand* v) {
|
2007-12-09 20:03:21 +00:00
|
|
|
flushStack(&c);
|
|
|
|
static_cast<MyOperand*>(v)->alignedCall(&c);
|
|
|
|
return register_(&c, rax);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual Operand* indirectCall
|
|
|
|
(Operand* address, unsigned argumentCount, ...)
|
|
|
|
{
|
|
|
|
va_list a; va_start(a, argumentCount);
|
2007-12-09 22:45:43 +00:00
|
|
|
unsigned footprint = pushArguments(&c, argumentCount, a);
|
2007-12-08 23:22:13 +00:00
|
|
|
va_end(a);
|
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
static_cast<MyOperand*>(address)->mov(&c, register_(&c, rax));
|
|
|
|
immediate(&c, c.indirectCaller)->call(&c);
|
2007-12-08 23:22:13 +00:00
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
immediate(&c, footprint)->sub(&c, register_(&c, rsp));
|
|
|
|
|
|
|
|
return register_(&c, rax);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
virtual void indirectCallNoReturn
|
2007-12-08 23:22:13 +00:00
|
|
|
(Operand* address, unsigned argumentCount, ...)
|
|
|
|
{
|
|
|
|
va_list a; va_start(a, argumentCount);
|
2007-12-09 20:03:21 +00:00
|
|
|
pushArguments(&c, argumentCount, a);
|
2007-12-08 23:22:13 +00:00
|
|
|
va_end(a);
|
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
static_cast<MyOperand*>(address)->mov(&c, register_(&c, rax));
|
|
|
|
immediate(&c, c.indirectCaller)->call(&c);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual Operand* directCall
|
|
|
|
(Operand* address, unsigned argumentCount, ...)
|
|
|
|
{
|
|
|
|
va_list a; va_start(a, argumentCount);
|
2007-12-09 22:45:43 +00:00
|
|
|
unsigned footprint = pushArguments(&c, argumentCount, a);
|
2007-12-08 23:22:13 +00:00
|
|
|
va_end(a);
|
|
|
|
|
2007-12-09 20:03:21 +00:00
|
|
|
static_cast<MyOperand*>(address)->call(&c);
|
2007-12-08 23:22:13 +00:00
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
immediate(&c, footprint)->sub(&c, register_(&c, rsp));
|
|
|
|
|
|
|
|
return register_(&c, rax);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void return_(Operand* v) {
|
2007-12-09 22:45:43 +00:00
|
|
|
static_cast<MyOperand*>(v)->mov(&c, register_(&c, rax));
|
|
|
|
ret();
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void ret() {
|
2007-12-09 22:45:43 +00:00
|
|
|
::ret(&c);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void mov(Operand* src, Operand* dst) {
|
2007-12-09 20:03:21 +00:00
|
|
|
static_cast<MyOperand*>(src)->mov(&c, static_cast<MyOperand*>(dst));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void cmp(Operand* subtrahend, Operand* minuend) {
|
2007-12-09 20:03:21 +00:00
|
|
|
static_cast<MyOperand*>(subtrahend)->mov
|
|
|
|
(&c, static_cast<MyOperand*>(minuend));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void jl(Operand* v) {
|
2007-12-09 20:03:21 +00:00
|
|
|
static_cast<MyOperand*>(v)->jl(&c);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void jg(Operand* v) {
|
2007-12-09 20:03:21 +00:00
|
|
|
static_cast<MyOperand*>(v)->jg(&c);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void jle(Operand* v) {
|
2007-12-09 20:03:21 +00:00
|
|
|
static_cast<MyOperand*>(v)->jle(&c);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void jge(Operand* v) {
|
2007-12-09 20:03:21 +00:00
|
|
|
static_cast<MyOperand*>(v)->jge(&c);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void je(Operand* v) {
|
2007-12-09 20:03:21 +00:00
|
|
|
static_cast<MyOperand*>(v)->je(&c);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void jne(Operand* v) {
|
2007-12-09 20:03:21 +00:00
|
|
|
static_cast<MyOperand*>(v)->jne(&c);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void jmp(Operand* v) {
|
2007-12-09 20:03:21 +00:00
|
|
|
static_cast<MyOperand*>(v)->jmp(&c);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void add(Operand* v, Operand* dst) {
|
2007-12-09 20:03:21 +00:00
|
|
|
static_cast<MyOperand*>(v)->add(&c, static_cast<MyOperand*>(dst));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void sub(Operand* v, Operand* dst) {
|
2007-12-09 20:03:21 +00:00
|
|
|
static_cast<MyOperand*>(v)->sub(&c, static_cast<MyOperand*>(dst));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void mul(Operand* v, Operand* dst) {
|
2007-12-09 20:03:21 +00:00
|
|
|
static_cast<MyOperand*>(v)->mul(&c, static_cast<MyOperand*>(dst));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void div(Operand* v, Operand* dst) {
|
2007-12-09 20:03:21 +00:00
|
|
|
static_cast<MyOperand*>(v)->div(&c, static_cast<MyOperand*>(dst));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void rem(Operand* v, Operand* dst) {
|
2007-12-09 20:03:21 +00:00
|
|
|
static_cast<MyOperand*>(v)->rem(&c, static_cast<MyOperand*>(dst));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void shl(Operand* v, Operand* dst) {
|
2007-12-09 20:03:21 +00:00
|
|
|
static_cast<MyOperand*>(v)->shl(&c, static_cast<MyOperand*>(dst));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void shr(Operand* v, Operand* dst) {
|
2007-12-09 20:03:21 +00:00
|
|
|
static_cast<MyOperand*>(v)->shr(&c, static_cast<MyOperand*>(dst));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void ushr(Operand* v, Operand* dst) {
|
2007-12-09 20:03:21 +00:00
|
|
|
static_cast<MyOperand*>(v)->ushr(&c, static_cast<MyOperand*>(dst));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void and_(Operand* v, Operand* dst) {
|
2007-12-09 20:03:21 +00:00
|
|
|
static_cast<MyOperand*>(v)->and_(&c, static_cast<MyOperand*>(dst));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void or_(Operand* v, Operand* dst) {
|
2007-12-09 20:03:21 +00:00
|
|
|
static_cast<MyOperand*>(v)->or_(&c, static_cast<MyOperand*>(dst));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void xor_(Operand* v, Operand* dst) {
|
2007-12-09 20:03:21 +00:00
|
|
|
static_cast<MyOperand*>(v)->xor_(&c, static_cast<MyOperand*>(dst));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void neg(Operand* v) {
|
2007-12-09 20:03:21 +00:00
|
|
|
static_cast<MyOperand*>(v)->neg(&c);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
2007-12-09 20:03:21 +00:00
|
|
|
virtual Operand* memory(Operand* base, int displacement,
|
2007-12-08 23:22:13 +00:00
|
|
|
Operand* index, unsigned scale)
|
|
|
|
{
|
2007-12-09 22:45:43 +00:00
|
|
|
return ::memory(&c, static_cast<MyOperand*>(base), displacement,
|
|
|
|
static_cast<MyOperand*>(index), scale);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual Operand* select1(Operand* v) {
|
|
|
|
return selection(&c, SelectionOperand::S1Selection,
|
|
|
|
static_cast<MyOperand*>(v));
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual Operand* select2(Operand* v) {
|
|
|
|
return selection(&c, SelectionOperand::S2Selection,
|
|
|
|
static_cast<MyOperand*>(v));
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual Operand* select2z(Operand* v) {
|
|
|
|
return selection(&c, SelectionOperand::Z2Selection,
|
|
|
|
static_cast<MyOperand*>(v));
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual Operand* select4(Operand* v) {
|
|
|
|
return selection(&c, SelectionOperand::S4Selection,
|
|
|
|
static_cast<MyOperand*>(v));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
virtual Operand* select8(Operand* v) {
|
|
|
|
return selection(&c, SelectionOperand::S8Selection,
|
|
|
|
static_cast<MyOperand*>(v));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void prologue() {
|
2007-12-09 20:03:21 +00:00
|
|
|
register_(&c, rbp)->push(&c);
|
|
|
|
register_(&c, rsp)->mov(&c, register_(&c, rbp));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void epilogue() {
|
2007-12-09 20:03:21 +00:00
|
|
|
register_(&c, rbp)->mov(&c, register_(&c, rsp));
|
|
|
|
register_(&c, rbp)->pop(&c);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
2007-12-09 20:03:21 +00:00
|
|
|
virtual void startLogicalIp(unsigned ip) {
|
2007-12-09 22:45:43 +00:00
|
|
|
c.ipTable.push(IpMapping(ip, c.code.length()));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
2007-12-09 20:03:21 +00:00
|
|
|
virtual Operand* logicalIp(unsigned ip) {
|
2007-12-09 22:45:43 +00:00
|
|
|
return absolute(&c, c.promises.push(IpPromise(ip)));
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual unsigned logicalIpToOffset(unsigned ip) {
|
|
|
|
unsigned bottom = 0;
|
2007-12-09 22:45:43 +00:00
|
|
|
unsigned top = c.ipTable.length() / sizeof(IpMapping);
|
2007-12-08 23:22:13 +00:00
|
|
|
for (unsigned span = top - bottom; span; span = top - bottom) {
|
|
|
|
unsigned middle = bottom + (span / 2);
|
2007-12-09 22:45:43 +00:00
|
|
|
IpMapping* mapping = c.ipTable.peek<IpMapping>
|
|
|
|
(middle * sizeof(IpMapping));
|
2007-12-08 23:22:13 +00:00
|
|
|
|
|
|
|
if (ip == mapping->ip) {
|
|
|
|
return mapping->offset;
|
|
|
|
} else if (ip < mapping->ip) {
|
|
|
|
top = middle;
|
|
|
|
} else if (ip > mapping->ip) {
|
|
|
|
bottom = middle + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
abort(&c);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual unsigned size() {
|
2007-12-09 20:03:21 +00:00
|
|
|
return c.code.length();
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
virtual void writeTo(void*) {
|
2007-12-08 23:22:13 +00:00
|
|
|
// todo
|
|
|
|
}
|
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
virtual void updateCall(void*, void*) {
|
2007-12-09 20:03:21 +00:00
|
|
|
// todo
|
|
|
|
}
|
2007-12-08 23:22:13 +00:00
|
|
|
|
|
|
|
virtual void dispose() {
|
2007-12-09 20:03:21 +00:00
|
|
|
c.dispose();
|
2007-12-08 23:22:13 +00:00
|
|
|
|
2007-12-09 22:45:43 +00:00
|
|
|
c.s->free(this);
|
2007-12-08 23:22:13 +00:00
|
|
|
}
|
|
|
|
|
2007-12-09 20:03:21 +00:00
|
|
|
Context c;
|
2007-12-08 23:22:13 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
namespace vm {
|
|
|
|
|
|
|
|
Compiler*
|
|
|
|
makeCompiler(System* system, void* indirectCaller)
|
|
|
|
{
|
|
|
|
return new (system->allocate(sizeof(MyCompiler)))
|
|
|
|
MyCompiler(system, indirectCaller);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace vm
|