corda/src/compiler.cpp

2705 lines
59 KiB
C++
Raw Normal View History

2007-12-08 23:22:13 +00:00
#include "compiler.h"
2007-12-09 20:03:21 +00:00
#include "vector.h"
2007-12-11 00:48:09 +00:00
#include "zone.h"
2007-12-08 23:22:13 +00:00
using namespace vm;
namespace {
2007-12-09 22:45:43 +00:00
enum Register {
NoRegister = -1,
2007-12-09 22:45:43 +00:00
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-15 01:11:01 +00:00
const bool Verbose = false;
2007-12-11 21:26:59 +00:00
const unsigned RegisterCount = BytesPerWord * 2;
2007-12-16 00:24:15 +00:00
const unsigned GprParameterCount = 6;
2007-12-11 21:26:59 +00:00
class Context;
2007-12-16 00:24:15 +00:00
class MyOperand;
2007-12-16 21:30:19 +00:00
class AddressOperand;
2007-12-11 21:26:59 +00:00
class ImmediateOperand;
2007-12-11 23:52:28 +00:00
class AbsoluteOperand;
2007-12-11 21:26:59 +00:00
class RegisterOperand;
class MemoryOperand;
class CodePromise;
class MyPromise;
class RegisterReference;
2007-12-11 21:26:59 +00:00
int64_t
divideLong(int64_t a, int64_t b)
{
return a / b;
}
int64_t
moduloLong(int64_t a, int64_t b)
{
return a % b;
}
2007-12-11 00:48:09 +00:00
inline bool
isInt8(intptr_t v)
2007-12-09 20:03:21 +00:00
{
2007-12-11 00:48:09 +00:00
return v == static_cast<int8_t>(v);
2007-12-09 20:03:21 +00:00
}
2007-12-11 00:48:09 +00:00
inline bool
isInt32(intptr_t v)
2007-12-09 20:03:21 +00:00
{
2007-12-11 00:48:09 +00:00
return v == static_cast<int32_t>(v);
2007-12-09 20:03:21 +00:00
}
2007-12-16 00:24:15 +00:00
class RegisterNode {
public:
RegisterNode(Register value, RegisterNode* next):
value(value), next(next)
{ }
Register value;
RegisterNode* next;
};
class Task {
2007-12-08 23:22:13 +00:00
public:
Task(Task* next): next(next) { }
2007-12-08 23:22:13 +00:00
virtual ~Task() { }
2007-12-11 23:52:28 +00:00
virtual void run(Context*, unsigned) = 0;
Task* next;
2007-12-09 20:03:21 +00:00
};
class Event {
public:
2007-12-15 01:11:01 +00:00
Event(Event* next): next(next), task(0) {
if (next) {
count = next->count + 1;
} else {
count = 1;
}
}
virtual ~Event() { }
2007-12-09 20:03:21 +00:00
virtual void run(Context*) { }
2007-12-11 23:52:28 +00:00
Event* next;
Task* task;
unsigned count;
2007-12-11 21:26:59 +00:00
};
2007-12-11 00:48:09 +00:00
class Segment {
2007-12-11 21:26:59 +00:00
public:
Segment(int logicalIp, Event* event):
2007-12-15 01:11:01 +00:00
logicalIp(logicalIp), offset(-1), event(event)
{ }
int logicalIp;
2007-12-15 01:11:01 +00:00
int offset;
Event* event;
2007-12-11 21:26:59 +00:00
};
2007-12-11 00:48:09 +00:00
2007-12-16 00:24:15 +00:00
class MyStack: public Stack {
public:
MyStack(MyOperand* value, int index, MyStack* next):
value(value), index(index), next(next)
{ }
MyOperand* value;
int index;
MyStack* next;
};
2007-12-16 21:30:19 +00:00
class RegisterData {
public:
RegisterData(): reserved(false) { }
2007-12-16 21:30:19 +00:00
bool reserved;
};
class Context {
public:
Context(System* s, void* indirectCaller):
s(s),
constantPool(s, BytesPerWord * 32),
plan(s, 1024),
code(s, 1024),
zone(s, 8 * 1024),
indirectCaller(reinterpret_cast<intptr_t>(indirectCaller)),
segmentTable(0),
reserved(0),
codeLength(-1)
{
plan.appendAddress(new (zone.allocate(sizeof(Segment))) Segment
(-1, new (zone.allocate(sizeof(Event))) Event(0)));
registers[rsp].reserved = true;
registers[rbp].reserved = true;
registers[rbx].reserved = true;
}
void dispose() {
zone.dispose();
plan.dispose();
code.dispose();
constantPool.dispose();
if (segmentTable) s->free(segmentTable);
}
System* s;
Vector constantPool;
Vector plan;
Vector code;
Zone zone;
intptr_t indirectCaller;
Segment** segmentTable;
unsigned reserved;
int codeLength;
RegisterData registers[RegisterCount];
};
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)
{
expect(c->s, v);
}
class MyPromise: public Promise {
public:
virtual intptr_t value(Compiler*);
virtual intptr_t value(Context*) = 0;
virtual bool resolved(Context*) = 0;
};
class ResolvedPromise: public MyPromise {
public:
ResolvedPromise(intptr_t value): value_(value) { }
virtual intptr_t value(Context*) {
return value_;
}
virtual bool resolved(Context*) {
return true;
}
intptr_t value_;
};
class PoolPromise: public MyPromise {
public:
PoolPromise(intptr_t key): key(key) { }
virtual intptr_t value(Context* c) {
if (resolved(c)) {
return reinterpret_cast<intptr_t>(c->code.data + c->codeLength + key);
}
abort(c);
}
virtual bool resolved(Context* c) {
return c->codeLength >= 0;
}
intptr_t key;
};
class CodePromise: public MyPromise {
public:
CodePromise():
offset(-1)
{ }
virtual intptr_t value(Context* c) {
if (resolved(c)) {
return reinterpret_cast<intptr_t>(c->code.data + offset);
}
abort(c);
}
virtual bool resolved(Context*) {
return offset >= 0;
}
intptr_t offset;
};
class IpPromise: public MyPromise {
public:
IpPromise(intptr_t logicalIp):
logicalIp(logicalIp)
{ }
virtual intptr_t value(Context* c) {
if (resolved(c)) {
unsigned bottom = 0;
unsigned top = c->plan.length() / BytesPerWord;
for (unsigned span = top - bottom; span; span = top - bottom) {
unsigned middle = bottom + (span / 2);
Segment* s = c->segmentTable[middle];
if (logicalIp == s->logicalIp) {
return reinterpret_cast<intptr_t>(c->code.data + s->offset);
} else if (logicalIp < s->logicalIp) {
top = middle;
} else if (logicalIp > s->logicalIp) {
bottom = middle + 1;
}
}
}
abort(c);
}
virtual bool resolved(Context* c) {
return c->codeLength >= 0;
}
intptr_t logicalIp;
};
AddressOperand*
address(Context* c, MyPromise* p);
ImmediateOperand*
immediate(Context* c, int64_t v);
2007-12-16 21:30:19 +00:00
AbsoluteOperand*
absolute(Context* c, MyPromise* v);
RegisterOperand*
register_(Context* c, RegisterReference*);
2007-12-17 01:46:46 +00:00
RegisterOperand*
register_(Context* c, Register = NoRegister, Register = NoRegister);
2007-12-16 21:30:19 +00:00
MemoryOperand*
memory(Context* c, MyOperand* base, int displacement,
MyOperand* index, unsigned scale);
2007-12-16 21:30:19 +00:00
2007-12-09 20:03:21 +00:00
class MyOperand: public Operand {
public:
2007-12-11 00:48:09 +00:00
enum Operation {
push1,
push2,
push2z,
push4,
push8,
pop4,
pop8,
2007-12-11 00:48:09 +00:00
call,
alignedCall,
ret,
mov1,
mov2,
mov4,
mov8,
mov1ToW,
mov2ToW,
mov2zToW,
mov4To8,
cmp4,
cmp8,
2007-12-11 00:48:09 +00:00
jl,
jg,
jle,
jge,
je,
jne,
jmp,
add4,
add8,
sub4,
sub8,
mul4,
mul8,
div4,
div8,
rem4,
rem8,
shl4,
shl8,
shr4,
shr8,
ushr4,
ushr8,
and4,
and8,
or4,
or8,
xor4,
xor8,
neg4,
neg8,
addc,
subb
2007-12-09 20:03:21 +00:00
};
static const Operation push = (BytesPerWord == 8 ? push8 : push4);
static const Operation pop = (BytesPerWord == 8 ? pop8 : pop4);
static const Operation mov = (BytesPerWord == 8 ? mov8 : mov4);
static const Operation cmp = (BytesPerWord == 8 ? cmp8 : cmp4);
static const Operation add = (BytesPerWord == 8 ? add8 : add4);
static const Operation sub = (BytesPerWord == 8 ? sub8 : sub4);
static const Operation mul = (BytesPerWord == 8 ? mul8 : mul4);
static const Operation neg = (BytesPerWord == 8 ? neg8 : neg4);
2007-12-09 20:03:21 +00:00
virtual ~MyOperand() { }
2007-12-09 20:03:21 +00:00
2007-12-11 00:48:09 +00:00
virtual Register asRegister(Context* c) { abort(c); }
2007-12-09 22:45:43 +00:00
2007-12-16 00:24:15 +00:00
virtual RegisterNode* dependencies(Context*, RegisterNode* next)
{ return next; }
2007-12-11 00:48:09 +00:00
virtual void release(Context*) { /* ignore */ }
2007-12-09 20:03:21 +00:00
2007-12-16 00:24:15 +00:00
virtual void setLabelValue(Context* c, MyPromise*) { abort(c); }
2007-12-09 20:03:21 +00:00
virtual void apply(Context*, Operation) = 0;
2007-12-09 20:03:21 +00:00
virtual void apply(Context*, Operation, MyOperand*) = 0;
virtual void accept(Context* c, Operation, RegisterOperand*) = 0;
2007-12-09 20:03:21 +00:00
virtual void accept(Context* c, Operation, ImmediateOperand*) = 0;
2007-12-11 23:52:28 +00:00
virtual void accept(Context* c, Operation, AddressOperand*) = 0;
virtual void accept(Context* c, Operation, AbsoluteOperand*) = 0;
virtual void accept(Context* c, Operation, MemoryOperand*) = 0;
2007-12-16 21:30:19 +00:00
};
void
acquire(Context* c, Register v)
{
assert(c, not c->registers[v].reserved);
if (Verbose) {
fprintf(stderr, "acquire %d\n", v);
}
c->registers[v].reserved = true;
}
2007-12-16 21:30:19 +00:00
Register
acquire(Context* c)
{
// we don't yet support using r9-r15
for (int i = 8/*RegisterCount*/ - 1; i >= 0; --i) {
if (not c->registers[i].reserved) {
acquire(c, static_cast<Register>(i));
return static_cast<Register>(i);
}
}
2007-12-15 01:11:01 +00:00
2007-12-16 21:30:19 +00:00
abort(c);
}
void
release(Context* c, Register v)
{
assert(c, c->registers[v].reserved);
if (Verbose) {
fprintf(stderr, "release %d\n", v);
}
c->registers[v].reserved = false;
}
2007-12-09 20:03:21 +00:00
2007-12-19 01:28:55 +00:00
class RegisterReference {
public:
2007-12-20 01:42:12 +00:00
RegisterReference(Register value = NoRegister, Register high = NoRegister):
value_(value), defaultValue(value), high_(high), defaultHigh(high),
acquired(true)
2007-12-20 01:42:12 +00:00
{ }
2007-12-19 01:28:55 +00:00
2007-12-20 01:42:12 +00:00
void acquire(Context* c) {
if (defaultValue != NoRegister) {
::acquire(c, defaultValue);
}
if (defaultHigh != NoRegister) {
::acquire(c, defaultHigh);
2007-12-20 01:42:12 +00:00
}
value_ = defaultValue;
high_ = defaultHigh;
acquired = true;
}
2007-12-19 01:28:55 +00:00
void release(Context* c) {
assert(c, acquired);
if (value_ != NoRegister) {
::release(c, value_);
}
if (high_ != NoRegister) {
::release(c, high_);
}
value_ = NoRegister;
high_ = NoRegister;
acquired = false;
2007-12-19 01:28:55 +00:00
}
Register value(Context* c) {
assert(c, acquired);
if (value_ == NoRegister) {
value_ = ::acquire(c);
}
2007-12-19 01:28:55 +00:00
return value_;
}
Register high(Context* c) {
assert(c, acquired);
if (high_ == NoRegister) {
high_ = ::acquire(c);
}
return high_;
}
2007-12-19 01:28:55 +00:00
Register value_;
Register defaultValue;
Register high_;
Register defaultHigh;
bool acquired;
2007-12-19 01:28:55 +00:00
};
2007-12-09 20:03:21 +00:00
class RegisterOperand: public MyOperand {
public:
RegisterOperand(RegisterReference* reference):
reference(reference)
2007-12-09 20:03:21 +00:00
{ }
2007-12-19 01:28:55 +00:00
Register value(Context* c) {
return reference->value(c);
}
Register high(Context* c) {
return reference->high(c);
}
2007-12-19 01:28:55 +00:00
virtual Register asRegister(Context* c) {
return value(c);
2007-12-11 21:26:59 +00:00
}
2007-12-16 21:30:19 +00:00
virtual RegisterNode* dependencies(Context* c, RegisterNode* next) {
return new (c->zone.allocate(sizeof(RegisterNode)))
2007-12-19 01:28:55 +00:00
RegisterNode(value(c), next);
}
2007-12-16 21:30:19 +00:00
virtual void release(Context* c) {
2007-12-19 01:28:55 +00:00
reference->release(c);
2007-12-09 20:03:21 +00:00
}
2007-12-11 21:26:59 +00:00
virtual void apply(Context*, Operation);
2007-12-09 20:03:21 +00:00
2007-12-11 00:48:09 +00:00
virtual void apply(Context* c, Operation operation, MyOperand* operand) {
operand->accept(c, operation, this);
2007-12-09 20:03:21 +00:00
}
2007-12-11 21:26:59 +00:00
virtual void accept(Context*, Operation, RegisterOperand*);
virtual void accept(Context*, Operation, ImmediateOperand*);
virtual void accept(Context*, Operation, AddressOperand*);
2007-12-11 23:52:28 +00:00
virtual void accept(Context*, Operation, AbsoluteOperand*);
2007-12-11 21:26:59 +00:00
virtual void accept(Context*, Operation, MemoryOperand*);
2007-12-19 01:28:55 +00:00
RegisterReference* reference;
2007-12-08 23:22:13 +00:00
};
2007-12-09 22:45:43 +00:00
class ImmediateOperand: public MyOperand {
public:
ImmediateOperand(int64_t value):
value(value)
2007-12-09 22:45:43 +00:00
{ }
2007-12-09 20:03:21 +00:00
2007-12-16 21:30:19 +00:00
virtual void apply(Context* c, Operation operation);
2007-12-11 00:48:09 +00:00
virtual void apply(Context* c, Operation operation, MyOperand* operand) {
operand->accept(c, operation, this);
2007-12-09 22:45:43 +00:00
}
virtual void accept(Context* c, Operation, RegisterOperand*) { abort(c); }
virtual void accept(Context* c, Operation, ImmediateOperand*) { abort(c); }
virtual void accept(Context* c, Operation, AddressOperand*) { abort(c); }
virtual void accept(Context* c, Operation, AbsoluteOperand*) { abort(c); }
virtual void accept(Context* c, Operation, MemoryOperand*) { abort(c); }
2007-12-16 21:30:19 +00:00
int64_t value;
};
class AddressOperand: public MyOperand {
public:
AddressOperand(MyPromise* promise):
promise(promise)
{ }
2007-12-16 00:24:15 +00:00
virtual Register asRegister(Context* c);
virtual void setLabelValue(Context*, MyPromise*);
virtual void apply(Context*, Operation);
virtual void apply(Context* c, Operation operation, MyOperand* operand) {
operand->accept(c, operation, this);
}
virtual void accept(Context* c, Operation, RegisterOperand*) { abort(c); }
virtual void accept(Context* c, Operation, ImmediateOperand*) { abort(c); }
virtual void accept(Context* c, Operation, AddressOperand*) { abort(c); }
virtual void accept(Context* c, Operation, AbsoluteOperand*) { abort(c); }
virtual void accept(Context* c, Operation, MemoryOperand*) { abort(c); }
MyPromise* promise;
2007-12-09 22:45:43 +00:00
};
class AbsoluteOperand: public MyOperand {
public:
AbsoluteOperand(MyPromise* promise):
promise(promise)
2007-12-09 22:45:43 +00:00
{ }
2007-12-15 01:11:01 +00:00
virtual Register asRegister(Context* c);
virtual void apply(Context*, Operation);
2007-12-11 21:26:59 +00:00
2007-12-11 23:52:28 +00:00
virtual void apply(Context* c, Operation operation, MyOperand* operand) {
operand->accept(c, operation, this);
}
virtual void accept(Context* c, Operation, RegisterOperand*) { abort(c); }
virtual void accept(Context* c, Operation, ImmediateOperand*) { abort(c); }
virtual void accept(Context* c, Operation, AddressOperand*) { abort(c); }
virtual void accept(Context* c, Operation, AbsoluteOperand*) { abort(c); }
virtual void accept(Context* c, Operation, MemoryOperand*) { abort(c); }
MyPromise* promise;
2007-12-09 22:45:43 +00:00
};
2007-12-09 20:03:21 +00:00
class MemoryOperand: public MyOperand {
public:
MemoryOperand(MyOperand* base, int displacement, MyOperand* index,
unsigned scale):
2007-12-09 20:03:21 +00:00
base(base),
displacement(displacement),
index(index),
scale(scale)
{ }
2007-12-16 21:30:19 +00:00
virtual Register asRegister(Context*);
2007-12-16 21:30:19 +00:00
virtual RegisterNode* dependencies(Context* c, RegisterNode* next) {
next = base->dependencies(c, next);
if (index) {
return index->dependencies(c, next);
} else {
return next;
}
}
2007-12-16 21:30:19 +00:00
virtual void apply(Context*, Operation);
virtual void apply(Context* c, Operation operation, MyOperand* operand) {
operand->accept(c, operation, this);
}
2007-12-16 21:30:19 +00:00
virtual void accept(Context*, Operation, RegisterOperand*);
virtual void accept(Context*, Operation, ImmediateOperand*);
virtual void accept(Context* c, Operation, AddressOperand*) { abort(c); }
2007-12-16 21:30:19 +00:00
virtual void accept(Context*, Operation, AbsoluteOperand*);
virtual void accept(Context*, Operation, MemoryOperand*);
MyOperand* base;
int displacement;
MyOperand* index;
unsigned scale;
};
AddressOperand*
address(Context* c, MyPromise* p)
{
return new (c->zone.allocate(sizeof(AddressOperand))) AddressOperand(p);
}
2007-12-09 20:03:21 +00:00
ImmediateOperand*
immediate(Context* c, int64_t v)
2007-12-09 20:03:21 +00:00
{
2007-12-16 21:30:19 +00:00
return new (c->zone.allocate(sizeof(ImmediateOperand)))
ImmediateOperand(v);
2007-12-09 20:03:21 +00:00
}
AbsoluteOperand*
2007-12-11 00:48:09 +00:00
absolute(Context* c, MyPromise* v)
2007-12-09 20:03:21 +00:00
{
2007-12-11 00:48:09 +00:00
return new (c->zone.allocate(sizeof(AbsoluteOperand))) AbsoluteOperand(v);
2007-12-09 20:03:21 +00:00
}
RegisterOperand*
register_(Context* c, RegisterReference* r)
{
return new (c->zone.allocate(sizeof(RegisterOperand)))
RegisterOperand(r);
}
2007-12-09 20:03:21 +00:00
RegisterOperand*
register_(Context* c, Register v, Register h)
2007-12-09 20:03:21 +00:00
{
2007-12-19 01:28:55 +00:00
RegisterReference* r = new (c->zone.allocate(sizeof(RegisterReference)))
2007-12-20 01:42:12 +00:00
RegisterReference(v, h);
return register_(c, r);
2007-12-09 20:03:21 +00:00
}
MemoryOperand*
memory(Context* c, MyOperand* base, int displacement,
MyOperand* index, unsigned scale)
2007-12-09 20:03:21 +00:00
{
2007-12-11 00:48:09 +00:00
return new (c->zone.allocate(sizeof(MemoryOperand)))
MemoryOperand(base, displacement, index, scale);
2007-12-09 20:03:21 +00:00
}
2007-12-16 21:30:19 +00:00
RegisterOperand*
temporary(Context* c)
2007-12-11 21:26:59 +00:00
{
2007-12-16 21:30:19 +00:00
return register_(c, acquire(c));
2007-12-11 21:26:59 +00:00
}
2007-12-11 00:48:09 +00:00
RegisterOperand*
2007-12-16 21:30:19 +00:00
temporary(Context* c, Register v)
2007-12-11 00:48:09 +00:00
{
2007-12-16 21:30:19 +00:00
acquire(c, v);
return register_(c, v);
2007-12-11 00:48:09 +00:00
}
RegisterOperand*
temporary(Context* c, Register v, Register h)
{
acquire(c, v);
acquire(c, h);
return register_(c, v, h);
}
2007-12-16 21:30:19 +00:00
Segment*
currentSegment(Context* c)
{
2007-12-16 21:30:19 +00:00
Segment* s; c->plan.get(c->plan.length() - BytesPerWord, &s, BytesPerWord);
return s;
}
void
apply(Context* c, MyOperand::Operation op)
{
switch (op) {
case MyOperand::ret:
c->code.append(0xc3);
break;
default: abort(c);
}
}
class OpEvent: public Event {
public:
OpEvent(MyOperand::Operation operation, Event* next):
Event(next), operation(operation)
{ }
virtual void run(Context* c) {
apply(c, operation);
}
MyOperand::Operation operation;
};
class UnaryOpEvent: public Event {
public:
UnaryOpEvent(MyOperand::Operation operation, Operand* operand, Event* next):
Event(next),
operation(operation),
operand(static_cast<MyOperand*>(operand))
{ }
virtual void run(Context* c) {
2007-12-16 21:30:19 +00:00
if (Verbose) {
fprintf(stderr, "unary %d\n", operation);
}
operand->apply(c, operation);
}
MyOperand::Operation operation;
MyOperand* operand;
};
class BinaryOpEvent: public Event {
public:
BinaryOpEvent(MyOperand::Operation operation, Operand* a, Operand* b,
Event* next):
Event(next),
operation(operation),
a(static_cast<MyOperand*>(a)),
b(static_cast<MyOperand*>(b))
{ }
virtual void run(Context* c) {
2007-12-16 21:30:19 +00:00
if (Verbose) {
fprintf(stderr, "binary %d\n", operation);
}
a->apply(c, operation, b);
}
MyOperand::Operation operation;
MyOperand* a;
MyOperand* b;
};
class AcquireEvent: public Event {
public:
AcquireEvent(RegisterOperand* operand, Event* next):
Event(next),
operand(operand)
{ }
virtual void run(Context* c) {
if (Verbose) {
fprintf(stderr, "acquire register\n");
}
2007-12-20 01:42:12 +00:00
operand->reference->acquire(c);
}
RegisterOperand* operand;
};
class ReleaseEvent: public Event {
public:
ReleaseEvent(Operand* operand, Event* next):
Event(next),
operand(static_cast<MyOperand*>(operand))
{ }
virtual void run(Context* c) {
if (Verbose) {
fprintf(stderr, "release register\n");
}
operand->release(c);
}
MyOperand* operand;
};
2007-12-16 00:24:15 +00:00
class Movement {
public:
MyOperand* source;
Register destination;
RegisterNode* dependencies;
};
void
push(Context* c, Movement* table, unsigned size)
{
int pushed[size];
unsigned pushIndex = 0;
for (unsigned i = 0; i < size; ++i) {
Movement* mi = table + i;
for (unsigned j = i + 1; j < size; ++j) {
Movement* mj = table + j;
for (RegisterNode* d = mj->dependencies; d; d = d->next) {
if (mi->destination == d->value) {
mi->source->apply(c, MyOperand::push);
pushed[pushIndex++] = i;
goto loop;
}
}
}
mi->source->apply(c, MyOperand::mov, register_(c, mi->destination));
loop:;
}
for (int i = pushIndex - 1; i >= 0; --i) {
register_(c, table[pushed[i]].destination)->apply
(c, MyOperand::pop);
}
}
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);
}
}
class ArgumentEvent: public Event {
public:
ArgumentEvent(MyOperand** arguments, unsigned count, Event* next):
Event(next),
arguments(arguments),
count(count)
{ }
virtual void run(Context* c) {
if (BytesPerWord == 8) {
const unsigned size = min(count, GprParameterCount);
Movement moveTable[size];
for (int i = count - 1; i >= 0; --i) {
if (static_cast<unsigned>(i) < GprParameterCount) {
Movement* m = moveTable + (size - i - 1);
m->source = arguments[i];
m->destination = gpRegister(c, i);
m->dependencies = arguments[i]->dependencies(c, 0);
} else {
arguments[i]->apply(c, MyOperand::push8);
2007-12-16 00:24:15 +00:00
}
}
push(c, moveTable, size);
} else {
for (int i = count - 1; i >= 0; --i) {
if (i > 0 and arguments[i - 1] == 0) {
arguments[i]->apply(c, MyOperand::push8);
-- i;
} else {
arguments[i]->apply(c, MyOperand::push4);
}
2007-12-16 00:24:15 +00:00
}
}
}
MyOperand** arguments;
unsigned count;
};
void
appendOperation(Context* c, MyOperand::Operation operation)
{
Segment* s = currentSegment(c);
s->event = new (c->zone.allocate(sizeof(OpEvent)))
OpEvent(operation, s->event);
}
void
appendOperation(Context* c, MyOperand::Operation operation, Operand* operand)
{
Segment* s = currentSegment(c);
s->event = new (c->zone.allocate(sizeof(UnaryOpEvent)))
UnaryOpEvent(operation, operand, s->event);
}
void
appendOperation(Context* c, MyOperand::Operation operation, Operand* a, Operand* b)
{
Segment* s = currentSegment(c);
s->event = new (c->zone.allocate(sizeof(BinaryOpEvent)))
BinaryOpEvent(operation, a, b, s->event);
}
void
appendAcquire(Context* c, RegisterOperand* operand)
{
Segment* s = currentSegment(c);
s->event = new (c->zone.allocate(sizeof(AcquireEvent)))
AcquireEvent(operand, s->event);
}
void
appendRelease(Context* c, Operand* operand)
{
Segment* s = currentSegment(c);
s->event = new (c->zone.allocate(sizeof(ReleaseEvent)))
ReleaseEvent(operand, s->event);
}
2007-12-16 00:24:15 +00:00
void
appendArgumentEvent(Context* c, MyOperand** arguments, unsigned count)
{
Segment* s = currentSegment(c);
s->event = new (c->zone.allocate(sizeof(ArgumentEvent)))
ArgumentEvent(arguments, count, s->event);
}
void
logStack(Context* c, MyStack* stack)
{
fprintf(stderr, "ip %3d: ", currentSegment(c)->logicalIp);
if (stack) {
fprintf(stderr, " %d",
static_cast<MemoryOperand*>(stack->value)->displacement);
}
for (MyStack* s = stack; s; s = s->next) {
fprintf(stderr, "*");
}
fprintf(stderr, "\n");
}
2007-12-16 00:24:15 +00:00
MyStack*
pushed(Context* c, MyStack* stack)
2007-12-11 00:48:09 +00:00
{
int index = (stack ? stack->index + 1 : 0);
2007-12-16 00:24:15 +00:00
MyOperand* value = memory
(c, register_(c, rbp), - (c->reserved + index + 1) * BytesPerWord, 0, 1);
stack = new (c->zone.allocate(sizeof(MyStack))) MyStack(value, index, stack);
if (Verbose) {
logStack(c, stack);
}
return stack;
2007-12-11 00:48:09 +00:00
}
2007-12-16 00:24:15 +00:00
MyStack*
push(Context* c, MyStack* stack, MyOperand::Operation operation, MyOperand* v)
{
appendOperation(c, operation, v);
if (BytesPerWord == 4 and operation == MyOperand::push8) {
stack = pushed(c, stack);
}
return pushed(c, stack);
}
2007-12-16 00:24:15 +00:00
MyStack*
pop(Context* c, MyStack* stack, int count)
{
appendOperation
(c, MyOperand::add, immediate(c, count * BytesPerWord), register_(c, rsp));
2007-12-13 00:18:31 +00:00
while (count) {
-- count;
assert(c, count >= 0);
2007-12-16 00:24:15 +00:00
stack = stack->next;
}
2007-12-16 00:24:15 +00:00
if (Verbose) {
logStack(c, stack);
}
2007-12-16 00:24:15 +00:00
return stack;
}
2007-12-16 00:24:15 +00:00
MyStack*
pop(Context* c, MyStack* stack, MyOperand::Operation operation, MyOperand* dst)
2007-12-11 00:48:09 +00:00
{
appendOperation(c, operation, dst);
2007-12-11 00:48:09 +00:00
if (BytesPerWord == 4 and operation == MyOperand::pop8) {
stack = stack->next;
}
if (Verbose) {
logStack(c, stack->next);
}
2007-12-16 00:24:15 +00:00
return stack->next;
2007-12-09 22:45:43 +00:00
}
void
2007-12-09 22:45:43 +00:00
pushArguments(Context* c, unsigned count, va_list list)
{
2007-12-16 00:24:15 +00:00
MyOperand** arguments = static_cast<MyOperand**>
(c->zone.allocate(count * BytesPerWord));
unsigned index = 0;
2007-12-09 22:45:43 +00:00
for (unsigned i = 0; i < count; ++i) {
if (BytesPerWord == 8) {
arguments[index] = va_arg(list, MyOperand*);
if (arguments[index]) {
++ index;
}
} else {
arguments[index++] = va_arg(list, MyOperand*);
}
2007-12-09 22:45:43 +00:00
}
appendArgumentEvent(c, arguments, index);
}
2007-12-09 22:45:43 +00:00
unsigned
argumentFootprint(unsigned count)
{
if (BytesPerWord == 8) {
if (count > GprParameterCount) {
return (count - GprParameterCount) * BytesPerWord;
} else {
return 0;
}
} else {
return count * BytesPerWord;
}
2007-12-09 20:03:21 +00:00
}
2007-12-11 00:48:09 +00:00
void
rex(Context* c)
{
if (BytesPerWord == 8) {
c->code.append(0x48);
}
}
void
2007-12-18 00:22:37 +00:00
encode(Context* c, uint8_t* instruction, unsigned length, int a, Register b,
int32_t displacement, int index, unsigned scale)
2007-12-11 00:48:09 +00:00
{
2007-12-18 00:22:37 +00:00
c->code.append(instruction, length);
2007-12-11 00:48:09 +00:00
uint8_t width;
if (displacement == 0 and b != rbp) {
width = 0;
} else if (isInt8(displacement)) {
width = 0x40;
2007-12-11 00:48:09 +00:00
} else {
width = 0x80;
2007-12-11 00:48:09 +00:00
}
if (index == -1) {
c->code.append(width | (a << 3) | b);
if (b == rsp) {
c->code.append(0x24);
}
} else {
assert(c, b != rsp);
c->code.append(width | (a << 3) | 4);
c->code.append((log(scale) << 6) | (index << 3) | b);
2007-12-11 00:48:09 +00:00
}
if (displacement == 0 and b != rbp) {
2007-12-11 00:48:09 +00:00
// do nothing
} else if (isInt8(displacement)) {
c->code.append(displacement);
2007-12-11 00:48:09 +00:00
} else {
c->code.append4(displacement);
2007-12-11 00:48:09 +00:00
}
}
void
encode(Context* c, uint8_t instruction, int a, MemoryOperand* b, bool rex)
{
Register r = b->base->asRegister(c);
int index = b->index ? b->index->asRegister(c) : -1;
if (rex) {
::rex(c);
}
2007-12-18 00:22:37 +00:00
encode(c, &instruction, 1, a, r, b->displacement, index, b->scale);
}
void
encode2(Context* c, uint16_t instruction, int a, MemoryOperand* b, bool rex)
{
Register r = b->base->asRegister(c);
int index = b->index ? b->index->asRegister(c) : -1;
if (rex) {
::rex(c);
}
uint8_t i[2] = { instruction >> 8, instruction & 0xff };
encode(c, i, 2, a, r, b->displacement, index, b->scale);
}
2007-12-11 21:26:59 +00:00
void
RegisterOperand::apply(Context* c, Operation operation)
{
switch (operation) {
case call:
c->code.append(0xff);
c->code.append(0xd0 | value(c));
2007-12-11 21:26:59 +00:00
break;
case jmp:
c->code.append(0xff);
c->code.append(0xe0 | value(c));
2007-12-11 21:26:59 +00:00
break;
case pop4:
case pop8:
if (BytesPerWord == 4 and operation == pop8) {
register_(c, value(c))->apply(c, pop);
register_(c, high(c))->apply(c, pop);
} else {
c->code.append(0x58 | value(c));
2007-12-20 01:42:12 +00:00
}
break;
case push4:
case push8:
if (BytesPerWord == 4 and operation == push8) {
register_(c, high(c))->apply(c, push);
register_(c, value(c))->apply(c, push);
2007-12-20 01:42:12 +00:00
} else {
c->code.append(0x50 | value(c));
2007-12-20 01:42:12 +00:00
}
break;
case neg4:
case neg8:
assert(c, BytesPerWord == 8 or operation == neg4); // todo
rex(c);
c->code.append(0xf7);
c->code.append(0xd8 | value(c));
break;
2007-12-11 21:26:59 +00:00
default: abort(c);
}
}
void
RegisterOperand::accept(Context* c, Operation operation,
RegisterOperand* operand)
{
switch (operation) {
case add:
rex(c);
c->code.append(0x01);
c->code.append(0xc0 | (operand->value(c) << 3) | value(c));
break;
case cmp4:
case cmp8:
if (BytesPerWord == 4 and operation == cmp8) {
2007-12-20 01:42:12 +00:00
register_(c, high(c))->accept
(c, cmp, register_(c, operand->high(c)));
2007-12-20 01:42:12 +00:00
// if the high order bits are equal, we compare the low order
// bits; otherwise, we jump past that comparison
c->code.append(0x0f);
c->code.append(0x85); // jne
c->code.append4(2);
register_(c, value(c))->accept
(c, cmp, register_(c, operand->value(c)));
} else {
if (operation == cmp8) rex(c);
c->code.append(0x39);
c->code.append(0xc0 | (operand->value(c) << 3) | value(c));
2007-12-20 01:42:12 +00:00
}
break;
case mov4:
case mov8:
if (BytesPerWord == 4 and operation == mov8) {
register_(c, value(c))->accept
(c, mov, register_(c, operand->value(c)));
register_(c, high(c))->accept
(c, mov, register_(c, operand->high(c)));
} else if (value(c) != operand->value(c)) {
rex(c);
c->code.append(0x89);
c->code.append(0xc0 | (operand->value(c) << 3) | value(c));
2007-12-15 01:11:01 +00:00
}
2007-12-16 21:30:19 +00:00
break;
2007-12-15 01:11:01 +00:00
case mov1ToW:
c->code.append(0xbe);
c->code.append(0xc0 | (operand->value(c) << 3) | value(c));
break;
case mov2ToW:
c->code.append(0xbf);
c->code.append(0xc0 | (operand->value(c) << 3) | value(c));
break;
case mov2zToW:
c->code.append(0xb7);
c->code.append(0xc0 | (operand->value(c) << 3) | value(c));
break;
case mov4To8:
assert(c, BytesPerWord == 8);
rex(c);
c->code.append(0x63);
c->code.append(0xc0 | (operand->value(c) << 3) | value(c));
break;
case mul4:
case mul8:
assert(c, BytesPerWord == 8 or operation == mul4); // todo
rex(c);
c->code.append(0x0f);
c->code.append(0xaf);
c->code.append(0xc0 | (value(c) << 3) | operand->value(c));
break;
2007-12-15 01:11:01 +00:00
default: abort(c);
}
}
2007-12-11 21:26:59 +00:00
void
RegisterOperand::accept(Context* c, Operation operation,
ImmediateOperand* operand)
{
switch (operation) {
case add4:
case add8:
assert(c, BytesPerWord == 8 or operation == add4); // todo
if (operand->value) {
rex(c);
if (isInt8(operand->value)) {
c->code.append(0x83);
c->code.append(0xc0 | value(c));
c->code.append(operand->value);
} else if (isInt32(operand->value)) {
c->code.append(0x81);
c->code.append(0xc0 | value(c));
c->code.append4(operand->value);
} else {
abort(c);
}
}
break;
case addc:
if (isInt8(operand->value)) {
c->code.append(0x83);
c->code.append(0xd0 | value(c));
c->code.append(operand->value);
} else {
abort(c);
}
break;
case and4:
case and8:
assert(c, BytesPerWord == 8 or operation == and4); // todo
rex(c);
2007-12-16 00:24:15 +00:00
if (isInt8(operand->value)) {
c->code.append(0x83);
c->code.append(0xe0 | value(c));
2007-12-16 00:24:15 +00:00
c->code.append(operand->value);
} else {
assert(c, isInt32(operand->value));
2007-12-16 00:24:15 +00:00
c->code.append(0x81);
c->code.append(0xe0 | value(c));
2007-12-16 00:24:15 +00:00
c->code.append(operand->value);
}
break;
case cmp4:
case cmp8: {
assert(c, isInt8(operand->value)); // todo
assert(c, BytesPerWord == 8 or operation == cmp4); // todo
rex(c);
2007-12-13 00:18:31 +00:00
c->code.append(0x83);
c->code.append(0xf8 | value(c));
c->code.append(operand->value);
} break;
case mov4:
case mov8: {
assert(c, BytesPerWord == 8 or operation == mov4); // todo
2007-12-11 23:52:28 +00:00
rex(c);
c->code.append(0xb8 | value(c));
c->code.appendAddress(operand->value);
} break;
2007-12-11 23:52:28 +00:00
case shl4:
case shl8: {
assert(c, BytesPerWord == 8 or operation == shl4); // todo
2007-12-16 00:24:15 +00:00
if (operand->value) {
rex(c);
2007-12-16 00:24:15 +00:00
if (operand->value == 1) {
c->code.append(0xd1);
c->code.append(0xe0 | value(c));
2007-12-16 00:24:15 +00:00
} else {
assert(c, isInt8(operand->value));
c->code.append(0xc1);
c->code.append(0xe0 | value(c));
2007-12-16 00:24:15 +00:00
c->code.append(operand->value);
}
}
} break;
case sub4:
case sub8: {
assert(c, BytesPerWord == 8 or operation == sub4); // todo
if (operand->value) {
rex(c);
if (isInt8(operand->value)) {
c->code.append(0x83);
c->code.append(0xe8 | value(c));
c->code.append(operand->value);
} else if (isInt32(operand->value)) {
c->code.append(0x81);
c->code.append(0xe8 | value(c));
c->code.append4(operand->value);
} else {
abort(c);
}
2007-12-11 23:52:28 +00:00
}
} break;
2007-12-11 21:26:59 +00:00
default: abort(c);
}
}
ImmediateOperand*
value(Context* c, AddressOperand* operand)
{
if (operand->promise->resolved(c)) {
return immediate(c, operand->promise->value(c));
} else {
return immediate(c, 0);
}
}
void
RegisterOperand::accept(Context* c, Operation operation,
AddressOperand* operand)
{
switch (operation) {
case mov: {
accept(c, operation, ::value(c, operand));
} break;
default: abort(c);
}
}
2007-12-11 21:26:59 +00:00
void
RegisterOperand::accept(Context* c, Operation operation,
MemoryOperand* operand)
{
switch (operation) {
case cmp4:
case cmp8:
assert(c, BytesPerWord == 8 or operation == cmp4); // todo
2007-12-13 00:18:31 +00:00
encode(c, 0x3b, value(c), operand, true);
break;
case mov4:
case mov8:
if (BytesPerWord == 4 and operation == mov8) {
register_(c, value(c))->accept
(c, mov, memory
(c, operand->base, operand->displacement,
operand->index, operand->scale));
register_(c, high(c))->accept
(c, mov, memory
(c, operand->base, operand->displacement + BytesPerWord,
operand->index, operand->scale));
} else if (BytesPerWord == 8 and operation == mov4) {
encode(c, 0x63, value(c), operand, true);
} else {
encode(c, 0x8b, value(c), operand, true);
}
break;
case mov1ToW:
encode2(c, 0x0fbe, value(c), operand, true);
break;
case mov2ToW:
encode2(c, 0x0fbf, value(c), operand, true);
break;
case mov2zToW:
encode2(c, 0x0fb7, value(c), operand, true);
break;
case mov4To8:
assert(c, BytesPerWord == 8); // todo
encode(c, 0x63, value(c), operand, true);
break;
case mul4:
case mul8:
assert(c, BytesPerWord == 8 or operation == mul4); // todo
encode2(c, 0x0faf, value(c), operand, true);
break;
default: abort(c);
}
}
ImmediateOperand*
value(Context* c, AbsoluteOperand* operand)
{
2007-12-16 21:30:19 +00:00
if (operand->promise->resolved(c)) {
2007-12-16 00:24:15 +00:00
return immediate(c, operand->promise->value(c));
} else {
return immediate(c, 0);
}
}
2007-12-11 23:52:28 +00:00
void
RegisterOperand::accept(Context* c, Operation operation,
AbsoluteOperand* operand)
{
switch (operation) {
case cmp4:
case cmp8: {
assert(c, BytesPerWord == 8 or operation == cmp4); // todo
2007-12-13 00:18:31 +00:00
RegisterOperand* tmp = temporary(c);
tmp->accept(c, mov, ::value(c, operand));
2007-12-13 00:18:31 +00:00
accept(c, cmp, memory(c, tmp, 0, 0, 1));
tmp->release(c);
} break;
case mov4:
case mov8: {
assert(c, BytesPerWord == 8 or operation == mov4); // todo
accept(c, mov, ::value(c, operand));
2007-12-11 23:52:28 +00:00
accept(c, mov, memory(c, this, 0, 0, 1));
} break;
default: abort(c);
}
}
void
unconditional(Context* c, unsigned jump, AddressOperand* operand)
2007-12-11 21:26:59 +00:00
{
intptr_t v;
2007-12-16 21:30:19 +00:00
if (operand->promise->resolved(c)) {
uint8_t* instruction = c->code.data + c->code.length();
v = reinterpret_cast<uint8_t*>(operand->promise->value(c))
- instruction - 5;
} else {
v = 0;
}
c->code.append(jump);
c->code.append4(v);
}
void
conditional(Context* c, unsigned condition, AddressOperand* operand)
{
intptr_t v;
2007-12-16 21:30:19 +00:00
if (operand->promise->resolved(c)) {
uint8_t* instruction = c->code.data + c->code.length();
v = reinterpret_cast<uint8_t*>(operand->promise->value(c))
- instruction - 6;
} else {
v = 0;
}
c->code.append(0x0f);
c->code.append(condition);
c->code.append4(v);
2007-12-11 21:26:59 +00:00
}
void
2007-12-16 00:24:15 +00:00
AddressOperand::setLabelValue(Context*, MyPromise* p)
{
promise = p;
}
void
AddressOperand::apply(Context* c, Operation operation)
2007-12-11 21:26:59 +00:00
{
switch (operation) {
case alignedCall: {
while ((c->code.length() + 1) % 4) {
c->code.append(0x90);
}
apply(c, call);
} break;
case call:
unconditional(c, 0xe8, this);
break;
case jmp:
unconditional(c, 0xe9, this);
break;
case je:
conditional(c, 0x84, this);
break;
case jne:
conditional(c, 0x85, this);
break;
case jg:
conditional(c, 0x8f, this);
break;
case jge:
conditional(c, 0x8d, this);
break;
case jl:
conditional(c, 0x8c, this);
break;
case jle:
conditional(c, 0x8e, this);
break;
case push4:
case push8: {
assert(c, BytesPerWord == 8 or operation == push4); // todo
RegisterOperand* tmp = temporary(c);
tmp->accept(c, mov, this);
tmp->apply(c, push);
tmp->release(c);
} break;
default: abort(c);
}
}
2007-12-16 00:24:15 +00:00
Register
AddressOperand::asRegister(Context* c)
{
intptr_t v;
if (c->codeLength >= 0) {
v = promise->value(c);
} else {
v = 0;
}
RegisterOperand* tmp = temporary(c);
tmp->accept(c, mov, immediate(c, v));
Register r = tmp->value(c);
2007-12-16 00:24:15 +00:00
tmp->release(c);
return r;
2007-12-16 00:24:15 +00:00
}
void
ImmediateOperand::apply(Context* c, Operation operation)
{
switch (operation) {
case alignedCall:
case call:
case jmp:
address(c, new (c->zone.allocate(sizeof(ResolvedPromise)))
ResolvedPromise(value))->apply(c, operation);
break;
case push4:
case push8:
if (BytesPerWord == 4 and operation == push8) {
immediate(c, (value >> 32) & 0xFFFFFFFF)->apply(c, push);
immediate(c, (value ) & 0xFFFFFFFF)->apply(c, push);
} else {
2007-12-17 01:46:46 +00:00
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);
tmp->accept(c, mov, this);
tmp->apply(c, push);
tmp->release(c);
}
}
break;
2007-12-11 21:26:59 +00:00
default: abort(c);
}
}
2007-12-15 01:11:01 +00:00
Register
AbsoluteOperand::asRegister(Context* c)
{
RegisterOperand* tmp = temporary(c);
tmp->accept(c, mov, this);
Register v = tmp->value(c);
2007-12-15 01:11:01 +00:00
tmp->release(c);
return v;
2007-12-15 01:11:01 +00:00
}
void
absoluteApply(Context* c, MyOperand::Operation operation,
AbsoluteOperand* operand)
{
RegisterOperand* tmp = temporary(c);
tmp->accept(c, MyOperand::mov, value(c, operand));
memory(c, tmp, 0, 0, 1)->apply(c, operation);
tmp->release(c);
}
2007-12-11 21:26:59 +00:00
void
AbsoluteOperand::apply(Context* c, Operation operation)
{
switch (operation) {
case push:
absoluteApply(c, operation, this);
break;
2007-12-11 21:26:59 +00:00
default: abort(c);
}
}
Register
MemoryOperand::asRegister(Context* c)
2007-12-11 21:26:59 +00:00
{
RegisterOperand* tmp = temporary(c);
2007-12-11 21:26:59 +00:00
tmp->accept(c, mov, this);
Register v = tmp->value(c);
tmp->release(c);
return v;
2007-12-11 21:26:59 +00:00
}
void
MemoryOperand::apply(Context* c, Operation operation)
{
switch (operation) {
case call:
encode(c, 0xff, 2, this, false);
break;
2007-12-16 00:24:15 +00:00
case jmp:
encode(c, 0xff, 4, this, false);
break;
case neg4:
case neg8:
if (BytesPerWord == 4 and operation == neg8) {
2007-12-20 01:42:12 +00:00
RegisterOperand* ax = temporary(c, rax);
RegisterOperand* dx = temporary(c, rdx);
MemoryOperand* low = memory(c, base, displacement, index, scale);
MemoryOperand* high = memory
(c, base, displacement + BytesPerWord, index, scale);
ax->accept(c, mov, low);
dx->accept(c, mov, high);
ax->apply(c, neg);
2007-12-20 23:19:48 +00:00
dx->accept(c, addc, immediate(c, 0));
2007-12-20 01:42:12 +00:00
dx->apply(c, neg);
low->accept(c, mov, ax);
high->accept(c, mov, dx);
ax->release(c);
dx->release(c);
} else {
encode(c, 0xf7, 2, this, true);
2007-12-20 01:42:12 +00:00
}
break;
case pop4:
case pop8:
if (BytesPerWord == 4 and operation == pop8) {
MemoryOperand* low = memory(c, base, displacement, index, scale);
MemoryOperand* high = memory
(c, base, displacement + BytesPerWord, index, scale);
low->apply(c, pop);
high->apply(c, pop);
} else if (BytesPerWord == 8 and operation == pop4) {
abort(c);
} else {
encode(c, 0x8f, 0, this, false);
}
2007-12-11 23:52:28 +00:00
break;
case push4:
case push8:
if (BytesPerWord == 4 and operation == push8) {
MemoryOperand* low = memory(c, base, displacement, index, scale);
MemoryOperand* high = memory
(c, base, displacement + BytesPerWord, index, scale);
2007-12-22 00:26:55 +00:00
high->apply(c, push);
low->apply(c, push);
} else if (BytesPerWord == 8 and operation == push4) {
RegisterOperand* tmp = temporary(c);
tmp->accept(c, mov4, this);
tmp->apply(c, operation);
tmp->release(c);
} else {
encode(c, 0xff, 6, this, false);
2007-12-16 21:30:19 +00:00
}
break;
case push1:
case push2:
case push2z: {
RegisterOperand* tmp = temporary(c);
switch (operation) {
case push1:
tmp->accept(c, mov1ToW, this);
break;
case push2:
tmp->accept(c, mov2ToW, this);
break;
case push2z:
tmp->accept(c, mov2zToW, this);
break;
default: abort(c);
}
tmp->apply(c, push);
tmp->release(c);
} break;
2007-12-11 21:26:59 +00:00
default: abort(c);
}
}
void
MemoryOperand::accept(Context* c, Operation operation,
RegisterOperand* operand)
{
switch (operation) {
case and4:
case and8:
assert(c, BytesPerWord == 8 or operation == and4); // todo
2007-12-16 00:24:15 +00:00
encode(c, 0x21, operand->value(c), this, true);
break;
case add4:
case add8:
if (BytesPerWord == 4 and operation == add8) {
RegisterOperand* ax = temporary(c, rax);
RegisterOperand* dx = temporary(c, rdx);
ax->accept(c, mov, register_(c, operand->value(c)));
dx->accept(c, mov, register_(c, operand->high(c)));
memory(c, base, displacement, index, scale)->accept(c, add, ax);
memory(c, base, displacement + BytesPerWord, index, scale)->accept
(c, addc, dx);
ax->release(c);
dx->release(c);
} else {
encode(c, 0x01, operand->value(c), this, true);
}
break;
case addc:
encode(c, 0x11, operand->value(c), this, true);
break;
case div4:
case div8:
if (BytesPerWord == 4 and operation == div8) {
operand->apply(c, push8);
memory(c, base, displacement, index, scale)->apply(c, push8);
immediate(c, reinterpret_cast<intptr_t>(divideLong))->apply(c, call);
register_(c, rsp)->accept(c, add, immediate(c, 16));
accept(c, mov8, register_(c, rax, rdx));
} else {
2007-12-20 01:42:12 +00:00
RegisterOperand* ax = temporary(c, rax);
RegisterOperand* dx = temporary(c, rdx);
ax->accept(c, mov, this);
rex(c);
2007-12-20 01:42:12 +00:00
c->code.append(0x99);
rex(c);
2007-12-20 01:42:12 +00:00
c->code.append(0xf7);
c->code.append(0xf8 | operand->value(c));
2007-12-20 01:42:12 +00:00
accept(c, mov, ax);
2007-12-16 21:30:19 +00:00
2007-12-20 01:42:12 +00:00
ax->release(c);
dx->release(c);
}
break;
2007-12-16 21:30:19 +00:00
case mov4:
case mov8:
if (BytesPerWord == 4 and operation == mov8) {
memory(c, base, displacement, index, scale)->accept
(c, mov, register_(c, operand->value(c)));
2007-12-20 01:42:12 +00:00
memory(c, base, displacement + BytesPerWord, index, scale)->accept
(c, mov, register_(c, operand->high(c)));
} else if (BytesPerWord == 8 and operation == mov4) {
encode(c, 0x89, operand->value(c), this, false);
} else {
encode(c, 0x89, operand->value(c), this, true);
}
break;
2007-12-20 01:42:12 +00:00
case mov1:
if (BytesPerWord == 8) {
if (operand->value(c) > rbx) {
c->code.append(0x40);
}
encode(c, 0x88, operand->value(c), this, false);
} else {
if (operand->value(c) > rbx) {
RegisterOperand* ax = temporary(c, rax);
ax->accept(c, mov, register_(c, operand->value(c)));
accept(c, mov1, register_(c, rax));
ax->release(c);
} else {
encode(c, 0x88, operand->value(c), this, false);
2007-12-16 21:30:19 +00:00
}
}
break;
2007-12-16 21:30:19 +00:00
case mov2:
c->code.append(0x66);
encode(c, 0x89, operand->value(c), this, false);
break;
case mov4To8:
assert(c, BytesPerWord == 8);
encode(c, 0x89, operand->value(c), this, false);
break;
case mul4:
case mul8:
if (BytesPerWord == 4 and operation == mul8) {
RegisterOperand* tmp = temporary(c, rcx);
RegisterOperand* ax = temporary(c, rax);
RegisterOperand* dx = temporary(c, rdx);
RegisterOperand* lowSrc = register_(c, operand->value(c));
RegisterOperand* highSrc = register_(c, operand->high(c));
MemoryOperand* lowDst = memory(c, base, displacement, index, scale);
MemoryOperand* highDst = memory
(c, base, displacement + BytesPerWord, index, scale);
tmp->accept(c, mov, highSrc);
tmp->accept(c, mul, lowDst);
ax->accept(c, mov, highDst);
ax->accept(c, mul, lowSrc);
tmp->accept(c, add, ax);
ax->accept(c, mov, lowDst);
// mul lowSrc,%eax
c->code.append(0xf7);
c->code.append(0xe8 | lowSrc->value(c));
dx->accept(c, add, tmp);
2007-12-20 23:19:48 +00:00
lowDst->accept(c, mov, ax);
highDst->accept(c, mov, dx);
tmp->release(c);
ax->release(c);
dx->release(c);
} else {
RegisterOperand* tmp = temporary(c);
tmp->accept(c, mov, this);
tmp->accept(c, mul, operand);
accept(c, mov, tmp);
tmp->release(c);
}
break;
case or4:
case or8:
assert(c, BytesPerWord == 8 or operation == or4); // todo
encode(c, 0x09, operand->value(c), this, true);
break;
2007-12-18 00:22:37 +00:00
case rem4:
case rem8:
if (BytesPerWord == 4 and operation == rem8) {
operand->apply(c, push8);
memory(c, base, displacement, index, scale)->apply(c, push8);
immediate(c, reinterpret_cast<intptr_t>(moduloLong))->apply(c, call);
register_(c, rsp)->accept(c, add, immediate(c, 16));
accept(c, mov8, register_(c, rax, rdx));
} else {
RegisterOperand* ax = temporary(c, rax);
RegisterOperand* dx = temporary(c, rdx);
ax->accept(c, mov, this);
2007-12-16 21:30:19 +00:00
rex(c);
c->code.append(0x99);
rex(c);
c->code.append(0xf7);
c->code.append(0xf8 | operand->value(c));
2007-12-16 21:30:19 +00:00
accept(c, mov, dx);
2007-12-16 21:30:19 +00:00
ax->release(c);
dx->release(c);
}
break;
2007-12-11 21:26:59 +00:00
case shl4:
case shl8: {
assert(c, BytesPerWord == 8 or operation == shl4); // todo
RegisterOperand* cx = temporary(c, rcx);
cx->accept(c, mov, operand);
encode(c, 0xd3, 4, this, true);
cx->release(c);
} break;
case shr4:
case shr8: {
assert(c, BytesPerWord == 8 or operation == shr4); // todo
RegisterOperand* cx = temporary(c, rcx);
cx->accept(c, mov, operand);
encode(c, 0xd3, 5, this, true);
cx->release(c);
} break;
case ushr4:
case ushr8: {
assert(c, BytesPerWord == 8 or operation == ushr4); // todo
RegisterOperand* cx = temporary(c, rcx);
cx->accept(c, mov, operand);
encode(c, 0xd3, 7, this, true);
cx->release(c);
} break;
case sub4:
case sub8:
if (BytesPerWord == 4 and operation == sub8) {
RegisterOperand* ax = temporary(c, rax);
RegisterOperand* dx = temporary(c, rdx);
ax->accept(c, mov, register_(c, operand->value(c)));
dx->accept(c, mov, register_(c, operand->high(c)));
memory(c, base, displacement, index, scale)->accept(c, sub, ax);
memory(c, base, displacement + BytesPerWord, index, scale)->accept
(c, subb, dx);
ax->release(c);
dx->release(c);
} else {
encode(c, 0x29, operand->value(c), this, true);
}
break;
case subb:
encode(c, 0x19, operand->value(c), this, true);
break;
case xor4:
case xor8: {
assert(c, BytesPerWord == 8 or operation == xor4); // todo
encode(c, 0x31, operand->value(c), this, true);
2007-12-18 00:22:37 +00:00
} break;
2007-12-11 21:26:59 +00:00
default: abort(c);
}
}
void
MemoryOperand::accept(Context* c, Operation operation,
ImmediateOperand* operand)
{
switch (operation) {
case add4:
case add8: {
assert(c, BytesPerWord == 8 or operation == add4); // todo
unsigned i = (isInt8(operand->value) ? 0x83 : 0x81);
encode(c, i, 0, this, true);
if (isInt8(operand->value)) {
c->code.append(operand->value);
} else if (isInt32(operand->value)) {
c->code.append4(operand->value);
} else {
abort(c);
}
} break;
case mov4:
case mov8: {
assert(c, isInt32(operand->value)); // todo
assert(c, BytesPerWord == 8 or operation == mov4); // todo
2007-12-11 21:26:59 +00:00
encode(c, 0xc7, 0, this, true);
c->code.append4(operand->value);
} break;
2007-12-11 21:26:59 +00:00
default: abort(c);
}
}
void
MemoryOperand::accept(Context* c, Operation operation,
AbsoluteOperand* operand)
{
2007-12-16 21:30:19 +00:00
RegisterOperand* tmp = temporary(c);
2007-12-16 21:30:19 +00:00
tmp->accept(c, mov, operand);
accept(c, operation, tmp);
2007-12-16 21:30:19 +00:00
tmp->release(c);
}
void
MemoryOperand::accept(Context* c, Operation operation,
2007-12-16 21:30:19 +00:00
MemoryOperand* operand)
{
switch (operation) {
case mov1ToW:
case mov2ToW:
case mov2zToW:
case mov4To8: {
if (BytesPerWord == 4 and operation == mov4To8) {
RegisterOperand* ax = temporary(c, rax);
RegisterOperand* dx = temporary(c, rdx);
ax->accept(c, mov4, operand);
c->code.append(0x99); // cdq
accept(c, mov8, register_(c, rax, rdx));
ax->release(c);
dx->release(c);
} else {
RegisterOperand* tmp = temporary(c);
tmp->accept(c, operation, operand);
accept(c, mov, tmp);
tmp->release(c);
}
} break;
case mov4:
case mov8:
case and4: {
RegisterOperand* tmp = temporary(c);
tmp->accept(c, mov, operand);
accept(c, operation, tmp);
tmp->release(c);
} break;
default: abort(c);
}
}
int
compareSegmentPointers(const void* a, const void* b)
2007-12-11 23:52:28 +00:00
{
return (*static_cast<Segment* const*>(a))->logicalIp
- (*static_cast<Segment* const*>(b))->logicalIp;
2007-12-11 23:52:28 +00:00
}
void
writeCode(Context* c)
{
unsigned tableSize = (c->plan.length() / BytesPerWord);
if (c->codeLength < 0) {
c->segmentTable = static_cast<Segment**>(c->s->allocate(c->plan.length()));
for (unsigned i = 0; i < tableSize; ++i) {
c->plan.get(i * BytesPerWord, c->segmentTable + i, BytesPerWord);
}
qsort(c->segmentTable, tableSize, BytesPerWord, compareSegmentPointers);
2007-12-11 23:52:28 +00:00
}
for (unsigned i = 0; i < tableSize; ++i) {
Segment* s = c->segmentTable[i];
2007-12-16 21:30:19 +00:00
if (Verbose) {
fprintf(stderr, "\nip %d\n", s->logicalIp);
}
2007-12-15 01:11:01 +00:00
if (c->codeLength >= 0) {
assert(c, s->offset == static_cast<int>(c->code.length()));
} else {
s->offset = c->code.length();
}
Event* events[s->event->count];
unsigned ei = s->event->count;
for (Event* e = s->event; e; e = e->next) {
events[--ei] = e;
}
2007-12-11 23:52:28 +00:00
for (unsigned ei = 0; ei < s->event->count; ++ei) {
2007-12-16 21:30:19 +00:00
if (Verbose and ei) {
fprintf(stderr, "address %p\n", c->code.data + c->code.length());
}
events[ei]->run(c);
if (c->codeLength < 0) {
for (Task* t = events[ei]->task; t; t = t->next) {
t->run(c, c->code.length());
}
2007-12-11 23:52:28 +00:00
}
}
}
c->codeLength = pad(c->code.length());
2007-12-11 23:52:28 +00:00
}
class CodePromiseTask: public Task {
public:
CodePromiseTask(CodePromise* promise, Task* next):
Task(next), promise(promise)
{ }
2007-12-11 23:52:28 +00:00
virtual void run(Context* c UNUSED, unsigned offset) {
promise->offset = offset;
2007-12-11 23:52:28 +00:00
}
CodePromise* promise;
};
2007-12-11 23:52:28 +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
{ }
2007-12-16 00:24:15 +00:00
virtual Promise* machineIp() {
CodePromise* p = new (c.zone.allocate(sizeof(CodePromise))) CodePromise();
2007-12-11 21:26:59 +00:00
Segment* s = currentSegment(&c);
s->event->task = new (c.zone.allocate(sizeof(CodePromiseTask)))
CodePromiseTask(p, s->event->task);
2007-12-11 21:26:59 +00:00
return p;
2007-12-08 23:22:13 +00:00
}
2007-12-16 00:24:15 +00:00
virtual Promise* machineIp(unsigned logicalIp) {
return new (c.zone.allocate(sizeof(IpPromise))) IpPromise(logicalIp);
}
virtual Promise* poolAppend(intptr_t v) {
return poolAppendPromise
(new (c.zone.allocate(sizeof(ResolvedPromise))) ResolvedPromise(v));
}
virtual Promise* poolAppendPromise(Promise* v) {
Promise* p = new (c.zone.allocate(sizeof(PoolPromise)))
PoolPromise(c.constantPool.length());
2007-12-11 00:48:09 +00:00
c.constantPool.appendAddress(v);
2007-12-16 00:24:15 +00:00
return p;
2007-12-08 23:22:13 +00:00
}
2007-12-16 21:30:19 +00:00
virtual Operand* constant(int64_t v) {
2007-12-09 20:03:21 +00:00
return immediate(&c, v);
2007-12-08 23:22:13 +00:00
}
2007-12-16 00:24:15 +00:00
virtual Operand* promiseConstant(Promise* p) {
return address(&c, static_cast<MyPromise*>(p));
}
2007-12-16 00:24:15 +00:00
virtual Operand* absolute(Promise* p) {
return ::absolute(&c, static_cast<MyPromise*>(p));
2007-12-08 23:22:13 +00:00
}
2007-12-16 00:24:15 +00:00
virtual Stack* push(Stack* s, unsigned count) {
appendOperation
(&c, MyOperand::sub, immediate(&c, count * BytesPerWord),
register_(&c, rsp));
return pushed(s, count);
}
virtual Stack* pushed(Stack* s, unsigned count) {
MyStack* stack = static_cast<MyStack*>(s);
while (count) {
-- count;
stack = ::pushed(&c, stack);
}
return stack;
2007-12-08 23:22:13 +00:00
}
virtual Stack* push1(Stack* s, Operand* v) {
return ::push(&c, static_cast<MyStack*>(s), MyOperand::push1,
static_cast<MyOperand*>(v));
}
virtual Stack* push2(Stack* s, Operand* v) {
return ::push(&c, static_cast<MyStack*>(s), MyOperand::push2,
static_cast<MyOperand*>(v));
}
virtual Stack* push2z(Stack* s, Operand* v) {
return ::push(&c, static_cast<MyStack*>(s), MyOperand::push2z,
static_cast<MyOperand*>(v));
}
virtual Stack* push4(Stack* s, Operand* v) {
return ::push(&c, static_cast<MyStack*>(s), MyOperand::push4,
static_cast<MyOperand*>(v));
}
virtual Stack* push8(Stack* s, Operand* v) {
return ::push(&c, static_cast<MyStack*>(s), MyOperand::push8,
static_cast<MyOperand*>(v));
2007-12-16 00:24:15 +00:00
}
2007-12-08 23:22:13 +00:00
2007-12-16 00:24:15 +00:00
virtual Operand* stack(Stack* s, unsigned index) {
MyStack* stack = static_cast<MyStack*>(s);
while (index) {
-- index;
2007-12-16 00:24:15 +00:00
stack = stack->next;
}
2007-12-16 00:24:15 +00:00
return stack->value;
2007-12-08 23:22:13 +00:00
}
2007-12-16 00:24:15 +00:00
virtual Stack* pop(Stack* s, unsigned count) {
return ::pop(&c, static_cast<MyStack*>(s), count);
2007-12-08 23:22:13 +00:00
}
virtual Stack* pop4(Stack* s, Operand* dst) {
return ::pop(&c, static_cast<MyStack*>(s), MyOperand::pop4,
static_cast<MyOperand*>(dst));
}
virtual Stack* pop8(Stack* s, Operand* dst) {
return ::pop(&c, static_cast<MyStack*>(s), MyOperand::pop8,
static_cast<MyOperand*>(dst));
2007-12-08 23:22:13 +00:00
}
virtual Operand* stack() {
2007-12-09 20:03:21 +00:00
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() {
RegisterOperand* r = register_(&c);
appendAcquire(&c, r);
return r;
2007-12-08 23:22:13 +00:00
}
virtual void release(Operand* v) {
appendRelease(&c, v);
2007-12-08 23:22:13 +00:00
}
virtual Operand* label() {
return address(&c, 0);
2007-12-08 23:22:13 +00:00
}
virtual void mark(Operand* label) {
2007-12-16 00:24:15 +00:00
static_cast<MyOperand*>(label)->setLabelValue
(&c, static_cast<MyPromise*>(machineIp()));
2007-12-08 23:22:13 +00:00
}
virtual Promise* indirectCall
2007-12-08 23:22:13 +00:00
(Operand* address, unsigned argumentCount, ...)
{
va_list a; va_start(a, argumentCount);
pushArguments(&c, argumentCount, a);
2007-12-08 23:22:13 +00:00
va_end(a);
appendOperation(&c, MyOperand::mov, address, register_(&c, rax));
call(immediate(&c, c.indirectCaller));
Promise* p = machineIp();
2007-12-11 00:48:09 +00:00
appendOperation
(&c, MyOperand::add, immediate(&c, argumentFootprint(argumentCount)),
register_(&c, rsp));
return p;
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);
appendOperation(&c, MyOperand::mov, address, register_(&c, rax));
2007-12-11 00:48:09 +00:00
call(immediate(&c, c.indirectCaller));
2007-12-08 23:22:13 +00:00
}
virtual Promise* directCall
2007-12-08 23:22:13 +00:00
(Operand* address, unsigned argumentCount, ...)
{
va_list a; va_start(a, argumentCount);
pushArguments(&c, argumentCount, a);
2007-12-08 23:22:13 +00:00
va_end(a);
call(address);
Promise* p = machineIp();
2007-12-08 23:22:13 +00:00
appendOperation
(&c, MyOperand::add, immediate(&c, argumentFootprint(argumentCount)),
register_(&c, rsp));
return p;
2007-12-22 00:26:55 +00:00
}
2007-12-09 22:45:43 +00:00
virtual Operand* result4() {
RegisterOperand* r = register_(&c, rax);
appendAcquire(&c, r);
return r;
2007-12-08 23:22:13 +00:00
}
virtual Operand* result8() {
if (BytesPerWord == 8) {
return result4();
} else {
RegisterOperand* r = register_(&c, rax, rdx);
appendAcquire(&c, r);
return r;
}
}
virtual void return4(Operand* v) {
appendOperation(&c, MyOperand::mov, v, register_(&c, rax));
epilogue();
2007-12-09 22:45:43 +00:00
ret();
2007-12-08 23:22:13 +00:00
}
virtual void return8(Operand* v) {
if (BytesPerWord == 8) {
return4(v);
} else {
appendOperation(&c, MyOperand::mov, v, register_(&c, rax, rdx));
epilogue();
ret();
}
}
2007-12-22 00:26:55 +00:00
virtual void call(Operand* v) {
appendOperation(&c, MyOperand::call, v);
2007-12-11 00:48:09 +00:00
}
2007-12-22 00:26:55 +00:00
virtual void alignedCall(Operand* v) {
appendOperation(&c, MyOperand::alignedCall, v);
2007-12-11 00:48:09 +00:00
}
2007-12-08 23:22:13 +00:00
virtual void ret() {
appendOperation(&c, MyOperand::ret);
2007-12-08 23:22:13 +00:00
}
virtual void mov1(Operand* src, Operand* dst) {
appendOperation(&c, MyOperand::mov1, src, dst);
}
virtual void mov2(Operand* src, Operand* dst) {
appendOperation(&c, MyOperand::mov2, src, dst);
}
virtual void mov4(Operand* src, Operand* dst) {
appendOperation(&c, MyOperand::mov4, src, dst);
}
virtual void mov8(Operand* src, Operand* dst) {
appendOperation(&c, MyOperand::mov8, src, dst);
}
virtual void mov1ToW(Operand* src, Operand* dst) {
appendOperation(&c, MyOperand::mov1ToW, src, dst);
}
virtual void mov2ToW(Operand* src, Operand* dst) {
appendOperation(&c, MyOperand::mov2ToW, src, dst);
}
virtual void mov2zToW(Operand* src, Operand* dst) {
appendOperation(&c, MyOperand::mov2zToW, src, dst);
2007-12-08 23:22:13 +00:00
}
virtual void mov4To8(Operand* src, Operand* dst) {
appendOperation(&c, MyOperand::mov4To8, src, dst);
}
virtual void cmp4(Operand* subtrahend, Operand* minuend) {
appendOperation(&c, MyOperand::cmp4, subtrahend, minuend);
}
virtual void cmp8(Operand* subtrahend, Operand* minuend) {
appendOperation(&c, MyOperand::cmp8, subtrahend, minuend);
2007-12-08 23:22:13 +00:00
}
virtual void jl(Operand* v) {
appendOperation(&c, MyOperand::jl, v);
2007-12-08 23:22:13 +00:00
}
virtual void jg(Operand* v) {
appendOperation(&c, MyOperand::jg, v);
2007-12-08 23:22:13 +00:00
}
virtual void jle(Operand* v) {
appendOperation(&c, MyOperand::jle, v);
2007-12-08 23:22:13 +00:00
}
virtual void jge(Operand* v) {
appendOperation(&c, MyOperand::jge, v);
2007-12-08 23:22:13 +00:00
}
virtual void je(Operand* v) {
appendOperation(&c, MyOperand::je, v);
2007-12-08 23:22:13 +00:00
}
virtual void jne(Operand* v) {
appendOperation(&c, MyOperand::jne, v);
2007-12-08 23:22:13 +00:00
}
virtual void jmp(Operand* v) {
appendOperation(&c, MyOperand::jmp, v);
2007-12-08 23:22:13 +00:00
}
virtual void add4(Operand* v, Operand* dst) {
appendOperation(&c, MyOperand::add4, v, dst);
2007-12-08 23:22:13 +00:00
}
virtual void add8(Operand* v, Operand* dst) {
appendOperation(&c, MyOperand::add8, v, dst);
2007-12-08 23:22:13 +00:00
}
virtual void sub4(Operand* v, Operand* dst) {
appendOperation(&c, MyOperand::sub4, v, dst);
2007-12-08 23:22:13 +00:00
}
virtual void sub8(Operand* v, Operand* dst) {
appendOperation(&c, MyOperand::sub8, v, dst);
2007-12-08 23:22:13 +00:00
}
virtual void mul4(Operand* v, Operand* dst) {
appendOperation(&c, MyOperand::mul4, v, dst);
2007-12-08 23:22:13 +00:00
}
virtual void mul8(Operand* v, Operand* dst) {
appendOperation(&c, MyOperand::mul8, v, dst);
2007-12-08 23:22:13 +00:00
}
virtual void div4(Operand* v, Operand* dst) {
appendOperation(&c, MyOperand::div4, v, dst);
2007-12-08 23:22:13 +00:00
}
virtual void div8(Operand* v, Operand* dst) {
appendOperation(&c, MyOperand::div8, v, dst);
2007-12-08 23:22:13 +00:00
}
virtual void rem4(Operand* v, Operand* dst) {
appendOperation(&c, MyOperand::rem4, v, dst);
2007-12-08 23:22:13 +00:00
}
virtual void rem8(Operand* v, Operand* dst) {
appendOperation(&c, MyOperand::rem8, v, dst);
2007-12-08 23:22:13 +00:00
}
virtual void shl4(Operand* v, Operand* dst) {
appendOperation(&c, MyOperand::shl4, v, dst);
2007-12-08 23:22:13 +00:00
}
virtual void shl8(Operand* v, Operand* dst) {
appendOperation(&c, MyOperand::shl8, v, dst);
2007-12-08 23:22:13 +00:00
}
virtual void shr4(Operand* v, Operand* dst) {
appendOperation(&c, MyOperand::shr4, v, dst);
}
virtual void shr8(Operand* v, Operand* dst) {
appendOperation(&c, MyOperand::shr8, v, dst);
}
virtual void ushr4(Operand* v, Operand* dst) {
appendOperation(&c, MyOperand::ushr4, v, dst);
}
virtual void ushr8(Operand* v, Operand* dst) {
appendOperation(&c, MyOperand::ushr8, v, dst);
2007-12-09 22:45:43 +00:00
}
virtual void and4(Operand* v, Operand* dst) {
appendOperation(&c, MyOperand::and4, v, dst);
2007-12-09 22:45:43 +00:00
}
virtual void and8(Operand* v, Operand* dst) {
appendOperation(&c, MyOperand::and8, v, dst);
2007-12-09 22:45:43 +00:00
}
virtual void or4(Operand* v, Operand* dst) {
appendOperation(&c, MyOperand::or4, v, dst);
}
virtual void or8(Operand* v, Operand* dst) {
appendOperation(&c, MyOperand::or8, v, dst);
2007-12-08 23:22:13 +00:00
}
virtual void xor4(Operand* v, Operand* dst) {
appendOperation(&c, MyOperand::xor4, v, dst);
2007-12-22 00:26:55 +00:00
}
virtual void xor8(Operand* v, Operand* dst) {
appendOperation(&c, MyOperand::xor8, v, dst);
}
virtual void neg4(Operand* v) {
appendOperation(&c, MyOperand::neg4, v);
}
virtual void neg8(Operand* v) {
appendOperation(&c, MyOperand::neg8, v);
}
virtual Operand* memory(Operand* base, int displacement,
Operand* index, unsigned scale)
{
return ::memory(&c, static_cast<MyOperand*>(base), displacement,
static_cast<MyOperand*>(index), scale);
2007-12-08 23:22:13 +00:00
}
virtual void prologue() {
appendOperation(&c, MyOperand::push, register_(&c, rbp));
appendOperation
(&c, MyOperand::mov, register_(&c, rsp), register_(&c, rbp));
2007-12-08 23:22:13 +00:00
}
virtual void reserve(unsigned size) {
appendOperation
(&c, MyOperand::sub, immediate(&c, size * BytesPerWord),
register_(&c, rsp));
c.reserved = size;
}
2007-12-08 23:22:13 +00:00
virtual void epilogue() {
appendOperation
(&c, MyOperand::mov, register_(&c, rbp), register_(&c, rsp));
appendOperation(&c, MyOperand::pop, register_(&c, rbp));
2007-12-08 23:22:13 +00:00
}
2007-12-09 20:03:21 +00:00
virtual void startLogicalIp(unsigned ip) {
c.plan.appendAddress
(new (c.zone.allocate(sizeof(Segment)))
Segment(ip, new (c.zone.allocate(sizeof(Event))) Event(0)));
2007-12-08 23:22:13 +00:00
}
2007-12-11 23:52:28 +00:00
virtual unsigned codeSize() {
if (c.codeLength < 0) {
assert(&c, c.code.length() == 0);
writeCode(&c);
}
return c.codeLength;
2007-12-11 23:52:28 +00:00
}
virtual unsigned poolSize() {
return c.constantPool.length();
2007-12-11 00:48:09 +00:00
}
2007-12-11 21:26:59 +00:00
virtual void writeTo(uint8_t* out) {
c.code.wrap(out, codeSize());
writeCode(&c);
2007-12-11 23:52:28 +00:00
2007-12-16 00:24:15 +00:00
for (unsigned i = 0; i < c.constantPool.length(); i += BytesPerWord) {
Promise* p; c.constantPool.get(i, &p, BytesPerWord);
*reinterpret_cast<intptr_t*>(out + codeSize() + i) = p->value(this);
}
2007-12-08 23:22:13 +00:00
}
2007-12-11 00:48:09 +00:00
virtual void updateCall(void* returnAddress, void* newTarget) {
uint8_t* instruction = static_cast<uint8_t*>(returnAddress) - 5;
assert(&c, *instruction == 0xE8);
2007-12-11 23:52:28 +00:00
assert(&c, reinterpret_cast<uintptr_t>(instruction + 1) % 4 == 0);
2007-12-11 00:48:09 +00:00
int32_t v = static_cast<uint8_t*>(newTarget)
- static_cast<uint8_t*>(returnAddress);
memcpy(instruction + 1, &v, 4);
2007-12-09 20:03:21 +00:00
}
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
};
intptr_t
2007-12-11 23:52:28 +00:00
MyPromise::value(Compiler* compiler)
2007-12-11 00:48:09 +00:00
{
2007-12-11 23:52:28 +00:00
return value(&(static_cast<MyCompiler*>(compiler)->c));
2007-12-11 00:48:09 +00:00
}
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 v