mirror of
https://github.com/corda/corda.git
synced 2025-01-22 04:18:31 +00:00
snapshot
This commit is contained in:
parent
4299063be9
commit
9efe6f1f05
@ -17,14 +17,14 @@
|
|||||||
namespace vm {
|
namespace vm {
|
||||||
|
|
||||||
enum Operation {
|
enum Operation {
|
||||||
|
PopFrame,
|
||||||
Return
|
Return
|
||||||
};
|
};
|
||||||
|
|
||||||
const unsigned OperationCount = Return + 1;
|
const unsigned OperationCount = Return + 1;
|
||||||
|
|
||||||
enum UnaryOperation {
|
enum UnaryOperation {
|
||||||
Push,
|
PushFrame,
|
||||||
Pop,
|
|
||||||
Call,
|
Call,
|
||||||
LongCall,
|
LongCall,
|
||||||
AlignedCall,
|
AlignedCall,
|
||||||
@ -42,13 +42,16 @@ enum UnaryOperation {
|
|||||||
const unsigned UnaryOperationCount = Negate + 1;
|
const unsigned UnaryOperationCount = Negate + 1;
|
||||||
|
|
||||||
enum BinaryOperation {
|
enum BinaryOperation {
|
||||||
LoadAddress,
|
|
||||||
Move,
|
Move,
|
||||||
MoveZ,
|
MoveZ,
|
||||||
Move4To8,
|
|
||||||
Swap,
|
Swap,
|
||||||
|
Compare
|
||||||
|
};
|
||||||
|
|
||||||
|
const unsigned BinaryOperationCount = Compare + 1;
|
||||||
|
|
||||||
|
enum TernaryOperation {
|
||||||
LongCompare,
|
LongCompare,
|
||||||
Compare,
|
|
||||||
Add,
|
Add,
|
||||||
Subtract,
|
Subtract,
|
||||||
Multiply,
|
Multiply,
|
||||||
@ -62,7 +65,7 @@ enum BinaryOperation {
|
|||||||
Xor
|
Xor
|
||||||
};
|
};
|
||||||
|
|
||||||
const unsigned BinaryOperationCount = Xor + 1;
|
const unsigned TernaryOperationCount = Xor + 1;
|
||||||
|
|
||||||
enum OperandType {
|
enum OperandType {
|
||||||
ConstantOperand,
|
ConstantOperand,
|
||||||
@ -156,14 +159,20 @@ class Assembler {
|
|||||||
virtual void restore(int r) = 0;
|
virtual void restore(int r) = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class Offset {
|
||||||
|
public:
|
||||||
|
virtual ~Offset() { }
|
||||||
|
|
||||||
|
virtual unsigned calculate(unsigned start) = 0;
|
||||||
|
};
|
||||||
|
|
||||||
virtual ~Assembler() { }
|
virtual ~Assembler() { }
|
||||||
|
|
||||||
virtual void setClient(Client* client) = 0;
|
virtual void setClient(Client* client) = 0;
|
||||||
|
|
||||||
virtual unsigned registerCount() = 0;
|
virtual unsigned registerCount() = 0;
|
||||||
|
|
||||||
virtual int base() = 0;
|
virtual int frame() = 0;
|
||||||
virtual int stack() = 0;
|
|
||||||
virtual int thread() = 0;
|
virtual int thread() = 0;
|
||||||
virtual int returnLow() = 0;
|
virtual int returnLow() = 0;
|
||||||
virtual int returnHigh() = 0;
|
virtual int returnHigh() = 0;
|
||||||
@ -171,22 +180,37 @@ class Assembler {
|
|||||||
virtual unsigned argumentRegisterCount() = 0;
|
virtual unsigned argumentRegisterCount() = 0;
|
||||||
virtual int argumentRegister(unsigned index) = 0;
|
virtual int argumentRegister(unsigned index) = 0;
|
||||||
|
|
||||||
virtual unsigned stackAlignment() = 0;
|
virtual void plan
|
||||||
|
(UnaryOperation op,
|
||||||
|
unsigned aSize, uint8_t* aTypeMask, uint64_t* aRegisterMask,
|
||||||
|
bool* thunk) = 0;
|
||||||
|
|
||||||
virtual void plan(UnaryOperation op, unsigned size, uint8_t* typeMask,
|
virtual void plan
|
||||||
uint64_t* registerMask, bool* thunk) = 0;
|
(BinaryOperation op,
|
||||||
|
unsigned aSize, uint8_t* aTypeMask, uint64_t* aRegisterMask,
|
||||||
|
unsigned bSize, uint8_t* bTypeMask, uint64_t* bRegisterMask,
|
||||||
|
bool* thunk) = 0;
|
||||||
|
|
||||||
virtual void plan(BinaryOperation op, unsigned size, uint8_t* aTypeMask,
|
virtual void plan
|
||||||
uint64_t* aRegisterMask, uint8_t* bTypeMask,
|
(TernaryOperation op,
|
||||||
uint64_t* bRegisterMask, bool* thunk) = 0;
|
unsigned aSize, uint8_t* aTypeMask, uint64_t* aRegisterMask,
|
||||||
|
unsigned bSize, uint8_t* bTypeMask, uint64_t* bRegisterMask,
|
||||||
|
unsigned cSize, uint8_t* cTypeMask, uint64_t* cRegisterMask,
|
||||||
|
bool* thunk) = 0;
|
||||||
|
|
||||||
virtual void apply(Operation op) = 0;
|
virtual void apply(Operation op) = 0;
|
||||||
|
|
||||||
virtual void apply(UnaryOperation op, unsigned size, OperandType type,
|
virtual void apply(UnaryOperation op,
|
||||||
Operand* operand) = 0;
|
unsigned aSize, OperandType aType, Operand* aOperand) = 0;
|
||||||
|
|
||||||
virtual void apply(BinaryOperation op, unsigned size, OperandType aType,
|
virtual void apply(BinaryOperation op,
|
||||||
Operand* a, OperandType bType, Operand* b) = 0;
|
unsigned aSize, OperandType aType, Operand* aOperand,
|
||||||
|
unsigned bSize, OperandType bType, Operand* bOperand) = 0;
|
||||||
|
|
||||||
|
virtual void apply(TernaryOperation op,
|
||||||
|
unsigned aSize, OperandType aType, Operand* aOperand,
|
||||||
|
unsigned bSize, OperandType bType, Operand* bOperand,
|
||||||
|
unsigned cSize, OperandType cType, Operand* cOperand) = 0;
|
||||||
|
|
||||||
virtual void writeTo(uint8_t* dst) = 0;
|
virtual void writeTo(uint8_t* dst) = 0;
|
||||||
|
|
||||||
@ -195,6 +219,8 @@ class Assembler {
|
|||||||
virtual void updateCall(void* returnAddress, void* newTarget) = 0;
|
virtual void updateCall(void* returnAddress, void* newTarget) = 0;
|
||||||
|
|
||||||
virtual void dispose() = 0;
|
virtual void dispose() = 0;
|
||||||
|
|
||||||
|
static unsigned alignFrameSize(unsigned sizeInWords);
|
||||||
};
|
};
|
||||||
|
|
||||||
Assembler*
|
Assembler*
|
||||||
|
@ -41,26 +41,22 @@ class MyThread: public Thread {
|
|||||||
public:
|
public:
|
||||||
CallTrace(MyThread* t):
|
CallTrace(MyThread* t):
|
||||||
t(t),
|
t(t),
|
||||||
base(t->base),
|
frame(t->frame),
|
||||||
stack(t->stack),
|
|
||||||
nativeMethod(0),
|
nativeMethod(0),
|
||||||
next(t->trace)
|
next(t->trace)
|
||||||
{
|
{
|
||||||
t->trace = this;
|
t->trace = this;
|
||||||
t->base = 0;
|
t->frame = 0;
|
||||||
t->stack = 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
~CallTrace() {
|
~CallTrace() {
|
||||||
t->stack = stack;
|
t->frame = frame;
|
||||||
t->base = base;
|
|
||||||
t->trace = next;
|
t->trace = next;
|
||||||
}
|
}
|
||||||
|
|
||||||
MyThread* t;
|
MyThread* t;
|
||||||
void* ip;
|
void* ip;
|
||||||
void* base;
|
void* frame;
|
||||||
void* stack;
|
|
||||||
object nativeMethod;
|
object nativeMethod;
|
||||||
CallTrace* next;
|
CallTrace* next;
|
||||||
};
|
};
|
||||||
@ -68,15 +64,13 @@ class MyThread: public Thread {
|
|||||||
MyThread(Machine* m, object javaThread, Thread* parent):
|
MyThread(Machine* m, object javaThread, Thread* parent):
|
||||||
Thread(m, javaThread, parent),
|
Thread(m, javaThread, parent),
|
||||||
ip(0),
|
ip(0),
|
||||||
base(0),
|
frame(0),
|
||||||
stack(0),
|
|
||||||
trace(0),
|
trace(0),
|
||||||
reference(0)
|
reference(0)
|
||||||
{ }
|
{ }
|
||||||
|
|
||||||
void* ip;
|
void* ip;
|
||||||
void* base;
|
void* frame;
|
||||||
void* stack;
|
|
||||||
CallTrace* trace;
|
CallTrace* trace;
|
||||||
Reference* reference;
|
Reference* reference;
|
||||||
};
|
};
|
||||||
@ -4311,42 +4305,34 @@ visitStack(MyThread* t, Heap::Visitor* v)
|
|||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
saveStackAndBase(MyThread* t, Assembler* a)
|
saveFrame(MyThread* t, Assembler* a)
|
||||||
{
|
{
|
||||||
Assembler::Register base(a->base());
|
Assembler::Register frame(a->frame());
|
||||||
Assembler::Memory baseDst(a->thread(), difference(&(t->base), t));
|
Assembler::Memory frameDst(a->thread(), difference(&(t->frame), t));
|
||||||
a->apply(Move, BytesPerWord, RegisterOperand, &base,
|
a->apply(Move, BytesPerWord, RegisterOperand, &frame,
|
||||||
MemoryOperand, &baseDst);
|
BytesPerWord, MemoryOperand, &frameDst);
|
||||||
|
|
||||||
Assembler::Register stack(a->stack());
|
|
||||||
Assembler::Memory stackDst(a->thread(), difference(&(t->stack), t));
|
|
||||||
a->apply(Move, BytesPerWord, RegisterOperand, &stack,
|
|
||||||
MemoryOperand, &stackDst);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
pushThread(MyThread*, Assembler* a)
|
pushFrame(MyThread* t, Assembler* a, unsigned size)
|
||||||
|
{
|
||||||
|
Assembler::Constant offset(resolved(c, Assembler::alignFrameSize(size)));
|
||||||
|
a->apply(PushFrame, BytesPerWord, ConstantOperand, &offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
setThreadArgument(MyThread*, Assembler* a)
|
||||||
{
|
{
|
||||||
Assembler::Register thread(a->thread());
|
Assembler::Register thread(a->thread());
|
||||||
|
|
||||||
if (a->argumentRegisterCount()) {
|
if (a->argumentRegisterCount()) {
|
||||||
Assembler::Register arg(a->argumentRegister(0));
|
Assembler::Register arg(a->argumentRegister(0));
|
||||||
a->apply(Move, BytesPerWord, RegisterOperand, &thread,
|
a->apply(Move, BytesPerWord, RegisterOperand, &thread,
|
||||||
RegisterOperand, &arg);
|
BytesPerWord, RegisterOperand, &arg);
|
||||||
} else {
|
} else {
|
||||||
a->apply(Push, BytesPerWord, RegisterOperand, &thread);
|
Assembler::Memory arg(a->frame(), a->argumentPosition(0));
|
||||||
}
|
a->apply(Move, BytesPerWord, RegisterOperand, &thread,
|
||||||
}
|
BytesPerWord, MemoryOperand, &arg);
|
||||||
|
|
||||||
void
|
|
||||||
popThread(MyThread*, Assembler* a)
|
|
||||||
{
|
|
||||||
if (a->argumentRegisterCount() == 0) {
|
|
||||||
ResolvedPromise bpwPromise(BytesPerWord);
|
|
||||||
Assembler::Constant bpw(&bpwPromise);
|
|
||||||
Assembler::Register stack(a->stack());
|
|
||||||
a->apply(Add, BytesPerWord, ConstantOperand, &bpw,
|
|
||||||
RegisterOperand, &stack);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -4984,8 +4970,9 @@ compileThunks(MyThread* t, MyProcessor* p)
|
|||||||
|
|
||||||
{ Assembler* a = defaultContext.context.assembler;
|
{ Assembler* a = defaultContext.context.assembler;
|
||||||
|
|
||||||
saveStackAndBase(t, a);
|
saveFrame(t, a);
|
||||||
pushThread(t, a);
|
pushFrame(t, a, 1);
|
||||||
|
setThreadArgument(t, a);
|
||||||
|
|
||||||
defaultContext.promise.resolved_ = true;
|
defaultContext.promise.resolved_ = true;
|
||||||
defaultContext.promise.value_ = reinterpret_cast<intptr_t>(compileMethod);
|
defaultContext.promise.value_ = reinterpret_cast<intptr_t>(compileMethod);
|
||||||
@ -4993,7 +4980,7 @@ compileThunks(MyThread* t, MyProcessor* p)
|
|||||||
Assembler::Constant proc(&(defaultContext.promise));
|
Assembler::Constant proc(&(defaultContext.promise));
|
||||||
a->apply(LongCall, BytesPerWord, ConstantOperand, &proc);
|
a->apply(LongCall, BytesPerWord, ConstantOperand, &proc);
|
||||||
|
|
||||||
popThread(t, a);
|
a->apply(PopFrame);
|
||||||
|
|
||||||
Assembler::Register result(a->returnLow());
|
Assembler::Register result(a->returnLow());
|
||||||
a->apply(Jump, BytesPerWord, RegisterOperand, &result);
|
a->apply(Jump, BytesPerWord, RegisterOperand, &result);
|
||||||
@ -5003,8 +4990,9 @@ compileThunks(MyThread* t, MyProcessor* p)
|
|||||||
|
|
||||||
{ Assembler* a = nativeContext.context.assembler;
|
{ Assembler* a = nativeContext.context.assembler;
|
||||||
|
|
||||||
saveStackAndBase(t, a);
|
saveFrame(t, a);
|
||||||
pushThread(t, a);
|
pushFrame(t, a, 1);
|
||||||
|
setThreadArgument(t, a);
|
||||||
|
|
||||||
nativeContext.promise.resolved_ = true;
|
nativeContext.promise.resolved_ = true;
|
||||||
nativeContext.promise.value_ = reinterpret_cast<intptr_t>(invokeNative);
|
nativeContext.promise.value_ = reinterpret_cast<intptr_t>(invokeNative);
|
||||||
@ -5012,7 +5000,7 @@ compileThunks(MyThread* t, MyProcessor* p)
|
|||||||
Assembler::Constant proc(&(nativeContext.promise));
|
Assembler::Constant proc(&(nativeContext.promise));
|
||||||
a->apply(LongCall, BytesPerWord, ConstantOperand, &proc);
|
a->apply(LongCall, BytesPerWord, ConstantOperand, &proc);
|
||||||
|
|
||||||
popThread(t, a);
|
a->apply(PopFrame);
|
||||||
|
|
||||||
a->apply(Return);
|
a->apply(Return);
|
||||||
}
|
}
|
||||||
@ -5021,8 +5009,9 @@ compileThunks(MyThread* t, MyProcessor* p)
|
|||||||
|
|
||||||
{ Assembler* a = aioobContext.context.assembler;
|
{ Assembler* a = aioobContext.context.assembler;
|
||||||
|
|
||||||
saveStackAndBase(t, a);
|
saveFrame(t, a);
|
||||||
pushThread(t, a);
|
pushFrame(t, a, 1);
|
||||||
|
setThreadArgument(t, a);
|
||||||
|
|
||||||
aioobContext.promise.resolved_ = true;
|
aioobContext.promise.resolved_ = true;
|
||||||
aioobContext.promise.value_ = reinterpret_cast<intptr_t>
|
aioobContext.promise.value_ = reinterpret_cast<intptr_t>
|
||||||
@ -5036,7 +5025,7 @@ compileThunks(MyThread* t, MyProcessor* p)
|
|||||||
|
|
||||||
{ Assembler* a = tableContext.context.assembler;
|
{ Assembler* a = tableContext.context.assembler;
|
||||||
|
|
||||||
saveStackAndBase(t, a);
|
saveFrame(t, a);
|
||||||
|
|
||||||
Assembler::Constant proc(&(tableContext.promise));
|
Assembler::Constant proc(&(tableContext.promise));
|
||||||
a->apply(LongJump, BytesPerWord, ConstantOperand, &proc);
|
a->apply(LongJump, BytesPerWord, ConstantOperand, &proc);
|
||||||
|
474
src/compiler.cpp
474
src/compiler.cpp
@ -32,10 +32,19 @@ class Read;
|
|||||||
void NO_RETURN abort(Context*);
|
void NO_RETURN abort(Context*);
|
||||||
|
|
||||||
void
|
void
|
||||||
apply(Context* c, UnaryOperation op, unsigned size, Site* a);
|
apply(Context* c, UnaryOperation op,
|
||||||
|
unsigned s1Size, Site* s1);
|
||||||
|
|
||||||
void
|
void
|
||||||
apply(Context* c, BinaryOperation op, unsigned size, Site* a, Site* b);
|
apply(Context* c, BinaryOperation op,
|
||||||
|
unsigned s1Size, Site* s1,
|
||||||
|
unsigned s2Size, Site* s2);
|
||||||
|
|
||||||
|
void
|
||||||
|
apply(Context* c, TernaryOperation op,
|
||||||
|
unsigned s1Size, Site* s1,
|
||||||
|
unsigned s2Size, Site* s2,
|
||||||
|
unsigned s3Size, Site* s3);
|
||||||
|
|
||||||
enum ConstantCompare {
|
enum ConstantCompare {
|
||||||
CompareNone,
|
CompareNone,
|
||||||
@ -221,7 +230,7 @@ class Context {
|
|||||||
Compiler::Client* client;
|
Compiler::Client* client;
|
||||||
int logicalIp;
|
int logicalIp;
|
||||||
State* state;
|
State* state;
|
||||||
LogicalInstruction* logicalCode;
|
LogicalInstruction** logicalCode;
|
||||||
unsigned logicalCodeLength;
|
unsigned logicalCodeLength;
|
||||||
unsigned parameterFootprint;
|
unsigned parameterFootprint;
|
||||||
unsigned localFootprint;
|
unsigned localFootprint;
|
||||||
@ -292,7 +301,7 @@ class IpPromise: public Promise {
|
|||||||
virtual int64_t value() {
|
virtual int64_t value() {
|
||||||
if (resolved()) {
|
if (resolved()) {
|
||||||
return reinterpret_cast<intptr_t>
|
return reinterpret_cast<intptr_t>
|
||||||
(c->machineCode + c->logicalCode[logicalIp].machineOffset);
|
(c->machineCode + c->logicalCode[logicalIp]->machineOffset->value());
|
||||||
}
|
}
|
||||||
|
|
||||||
abort(c);
|
abort(c);
|
||||||
@ -334,7 +343,7 @@ class Event {
|
|||||||
{
|
{
|
||||||
assert(c, c->logicalIp >= 0);
|
assert(c, c->logicalIp >= 0);
|
||||||
|
|
||||||
LogicalInstruction* i = c->logicalCode + c->logicalIp;
|
LogicalInstruction* i = c->logicalCode[c->logicalIp];
|
||||||
if (i->lastEvent) {
|
if (i->lastEvent) {
|
||||||
i->lastEvent->next = this;
|
i->lastEvent->next = this;
|
||||||
} else {
|
} else {
|
||||||
@ -1021,7 +1030,7 @@ trySteal(Context* c, Register* r, Stack* stack, Value** locals)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (saveSite) {
|
if (saveSite) {
|
||||||
apply(c, Move, r->size, r->site, saveSite);
|
apply(c, Move, r->size, r->site, r->size, saveSite);
|
||||||
addSite(c, 0, 0, r->size, v, saveSite);
|
addSite(c, 0, 0, r->size, v, saveSite);
|
||||||
} else {
|
} else {
|
||||||
if (DebugRegisters) {
|
if (DebugRegisters) {
|
||||||
@ -1106,7 +1115,8 @@ swap(Context* c, Register* a, Register* b)
|
|||||||
Assembler::Register ar(a->number);
|
Assembler::Register ar(a->number);
|
||||||
Assembler::Register br(b->number);
|
Assembler::Register br(b->number);
|
||||||
c->assembler->apply
|
c->assembler->apply
|
||||||
(Swap, BytesPerWord, RegisterOperand, &ar, RegisterOperand, &br);
|
(Swap, BytesPerWord, RegisterOperand, &ar,
|
||||||
|
BytesPerWord, RegisterOperand, &br);
|
||||||
|
|
||||||
c->registers[a->number] = b;
|
c->registers[a->number] = b;
|
||||||
c->registers[b->number] = a;
|
c->registers[b->number] = a;
|
||||||
@ -1209,31 +1219,56 @@ validate(Context* c, uint32_t mask, Stack* stack, Value** locals,
|
|||||||
Assembler::Register rr(r->number);
|
Assembler::Register rr(r->number);
|
||||||
Assembler::Register cr(current->number);
|
Assembler::Register cr(current->number);
|
||||||
c->assembler->apply
|
c->assembler->apply
|
||||||
(Move, BytesPerWord, RegisterOperand, &cr, RegisterOperand, &rr);
|
(Move, BytesPerWord, RegisterOperand, &cr,
|
||||||
|
BytesPerWord, RegisterOperand, &rr);
|
||||||
}
|
}
|
||||||
|
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
apply(Context* c, UnaryOperation op, unsigned size, Site* a)
|
apply(Context* c, UnaryOperation op,
|
||||||
|
unsigned s1Size, Site* s1)
|
||||||
{
|
{
|
||||||
OperandType type = a->type(c);
|
OperandType s1Type = s1->type(c);
|
||||||
Assembler::Operand* operand = a->asAssemblerOperand(c);
|
Assembler::Operand* s1Operand = s1->asAssemblerOperand(c);
|
||||||
|
|
||||||
c->assembler->apply(op, size, type, operand);
|
c->assembler->apply(op, s1Size, s1Type, s1Operand);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
apply(Context* c, BinaryOperation op, unsigned size, Site* a, Site* b)
|
apply(Context* c, BinaryOperation op,
|
||||||
|
unsigned s1Size, Site* s1,
|
||||||
|
unsigned s2Size, Site* s2)
|
||||||
{
|
{
|
||||||
OperandType aType = a->type(c);
|
OperandType s1Type = s1->type(c);
|
||||||
Assembler::Operand* aOperand = a->asAssemblerOperand(c);
|
Assembler::Operand* s1Operand = s1->asAssemblerOperand(c);
|
||||||
|
|
||||||
OperandType bType = b->type(c);
|
OperandType s2Type = s2->type(c);
|
||||||
Assembler::Operand* bOperand = b->asAssemblerOperand(c);
|
Assembler::Operand* s2Operand = s2->asAssemblerOperand(c);
|
||||||
|
|
||||||
c->assembler->apply(op, size, aType, aOperand, bType, bOperand);
|
c->assembler->apply(op, s1Size, s1Type, s1Operand,
|
||||||
|
s2Size, s2Type, s2Operand);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
apply(Context* c, TernaryOperation op,
|
||||||
|
unsigned s1Size, Site* s1,
|
||||||
|
unsigned s2Size, Site* s2,
|
||||||
|
unsigned s3Size, Site* s3)
|
||||||
|
{
|
||||||
|
OperandType s1Type = s1->type(c);
|
||||||
|
Assembler::Operand* s1Operand = s1->asAssemblerOperand(c);
|
||||||
|
|
||||||
|
OperandType s2Type = s2->type(c);
|
||||||
|
Assembler::Operand* s2Operand = s2->asAssemblerOperand(c);
|
||||||
|
|
||||||
|
OperandType s3Type = s3->type(c);
|
||||||
|
Assembler::Operand* s3Operand = s3->asAssemblerOperand(c);
|
||||||
|
|
||||||
|
c->assembler->apply(op, s1Size, s1Type, s1Operand,
|
||||||
|
s2Size, s2Type, s2Operand,
|
||||||
|
s3Size, s3Type, s3Operand);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -1269,7 +1304,7 @@ insertRead(Context* c, Event* event, int sequence, Value* v, Read* r)
|
|||||||
void
|
void
|
||||||
addRead(Context* c, Value* v, Read* r)
|
addRead(Context* c, Value* v, Read* r)
|
||||||
{
|
{
|
||||||
insertRead(c, c->logicalCode[c->logicalIp].lastEvent, -1, v, r);
|
insertRead(c, c->logicalCode[c->logicalIp]->lastEvent, -1, v, r);
|
||||||
}
|
}
|
||||||
|
|
||||||
Site*
|
Site*
|
||||||
@ -1355,17 +1390,6 @@ appendPushed(Context* c, Stack* s)
|
|||||||
void
|
void
|
||||||
push(Context* c, unsigned size, Value* v);
|
push(Context* c, unsigned size, Value* v);
|
||||||
|
|
||||||
void
|
|
||||||
ignore(Context* c, unsigned count)
|
|
||||||
{
|
|
||||||
if (count) {
|
|
||||||
Assembler::Register stack(c->assembler->stack());
|
|
||||||
Assembler::Constant offset(resolved(c, count * BytesPerWord));
|
|
||||||
c->assembler->apply
|
|
||||||
(Add, BytesPerWord, ConstantOperand, &offset, RegisterOperand, &stack);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
void
|
||||||
cleanStack(Context* c, Stack* stack, Local* locals, Read* reads)
|
cleanStack(Context* c, Stack* stack, Local* locals, Read* reads)
|
||||||
{
|
{
|
||||||
@ -1531,12 +1555,11 @@ class ReturnEvent: public Event {
|
|||||||
nextRead(c, value);
|
nextRead(c, value);
|
||||||
}
|
}
|
||||||
|
|
||||||
Assembler::Register base(c->assembler->base());
|
Assembler::Register frame(c->assembler->frame());
|
||||||
Assembler::Register stack(c->assembler->stack());
|
Assembler::Memory oldFrame(c->assembler->frame(), 0);
|
||||||
|
|
||||||
c->assembler->apply(Move, BytesPerWord, RegisterOperand, &base,
|
c->assembler->apply(Move, BytesPerWord, MemoryOperand, &oldFrame,
|
||||||
RegisterOperand, &stack);
|
BytesPerWord, RegisterOperand, &frame);
|
||||||
c->assembler->apply(Pop, BytesPerWord, RegisterOperand, &base);
|
|
||||||
c->assembler->apply(Return);
|
c->assembler->apply(Return);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1555,9 +1578,11 @@ appendReturn(Context* c, unsigned size, Value* value)
|
|||||||
|
|
||||||
class MoveEvent: public Event {
|
class MoveEvent: public Event {
|
||||||
public:
|
public:
|
||||||
MoveEvent(Context* c, BinaryOperation type, unsigned size, Value* src,
|
MoveEvent(Context* c, BinaryOperation type, unsigned srcSize, Value* src,
|
||||||
Value* dst, Site* srcTarget, VirtualSite* dstTarget):
|
unsigned dstSize, Value* dst, Site* srcTarget,
|
||||||
Event(c), type(type), size(size), src(src), dst(dst), dstTarget(dstTarget)
|
VirtualSite* dstTarget):
|
||||||
|
Event(c), type(type), srcSize(srcSize), src(src), dstSize(dstSize),
|
||||||
|
dst(dst), dstTarget(dstTarget)
|
||||||
{
|
{
|
||||||
addRead(c, src, size, srcTarget);
|
addRead(c, src, size, srcTarget);
|
||||||
}
|
}
|
||||||
@ -1583,25 +1608,25 @@ class MoveEvent: public Event {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (not isStore) {
|
if (not isStore) {
|
||||||
addSite(c, stack, locals, size, dst, target);
|
addSite(c, stack, locals, dstSize, dst, target);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (cost or type != Move) {
|
if (cost or type != Move) {
|
||||||
if (match(c, target, dstTarget->typeMask, dstTarget->registerMask)) {
|
if (match(c, target, dstTarget->typeMask, dstTarget->registerMask)) {
|
||||||
apply(c, type, size, src->source, target);
|
apply(c, type, srcSize, src->source, dstSize, target);
|
||||||
} else {
|
} else {
|
||||||
assert(c, dstTarget->typeMask & (1 << RegisterOperand));
|
assert(c, dstTarget->typeMask & (1 << RegisterOperand));
|
||||||
|
|
||||||
Site* tmpTarget = freeRegisterSite(c, dstTarget->registerMask);
|
Site* tmpTarget = freeRegisterSite(c, dstTarget->registerMask);
|
||||||
|
|
||||||
addSite(c, stack, locals, size, dst, tmpTarget);
|
addSite(c, stack, locals, dstSize, dst, tmpTarget);
|
||||||
|
|
||||||
apply(c, type, size, src->source, tmpTarget);
|
apply(c, type, srcSize, src->source, dstSize, tmpTarget);
|
||||||
|
|
||||||
if (isStore) {
|
if (isStore) {
|
||||||
removeSite(c, dst, tmpTarget);
|
removeSite(c, dst, tmpTarget);
|
||||||
|
|
||||||
apply(c, Move, size, tmpTarget, target);
|
apply(c, Move, dstSize, tmpTarget, dstSize, target);
|
||||||
} else {
|
} else {
|
||||||
removeSite(c, dst, target);
|
removeSite(c, dst, target);
|
||||||
}
|
}
|
||||||
@ -1616,8 +1641,9 @@ class MoveEvent: public Event {
|
|||||||
}
|
}
|
||||||
|
|
||||||
BinaryOperation type;
|
BinaryOperation type;
|
||||||
unsigned size;
|
unsigned srcSize;
|
||||||
Value* src;
|
Value* src;
|
||||||
|
unsigned dstSize;
|
||||||
Value* dst;
|
Value* dst;
|
||||||
VirtualSite* dstTarget;
|
VirtualSite* dstTarget;
|
||||||
};
|
};
|
||||||
@ -1688,7 +1714,7 @@ class CompareEvent: public Event {
|
|||||||
} else {
|
} else {
|
||||||
c->constantCompare = CompareNone;
|
c->constantCompare = CompareNone;
|
||||||
|
|
||||||
apply(c, Compare, size, first->source, second->source);
|
apply(c, Compare, size, first->source, size, second->source);
|
||||||
}
|
}
|
||||||
|
|
||||||
nextRead(c, first);
|
nextRead(c, first);
|
||||||
@ -1730,7 +1756,7 @@ preserve(Context* c, Stack* stack, unsigned size, Value* v, Site* s,
|
|||||||
Site* r = targetOrNull(c, read);
|
Site* r = targetOrNull(c, read);
|
||||||
if (r == 0 or r == s) r = freeRegisterSite(c);
|
if (r == 0 or r == s) r = freeRegisterSite(c);
|
||||||
addSite(c, stack, locals, size, v, r);
|
addSite(c, stack, locals, size, v, r);
|
||||||
apply(c, Move, size, s, r);
|
apply(c, Move, size, s, size, r);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -1743,30 +1769,17 @@ maybePreserve(Context* c, Stack* stack, unsigned size, Value* v, Site* s)
|
|||||||
|
|
||||||
class CombineEvent: public Event {
|
class CombineEvent: public Event {
|
||||||
public:
|
public:
|
||||||
CombineEvent(Context* c, BinaryOperation type, unsigned size, Value* first,
|
CombineEvent(Context* c, TernaryOperation type,
|
||||||
Value* second, Value* result, Site* firstTarget,
|
unsigned firstSize, Value* first,
|
||||||
Site* secondTarget):
|
unsigned secondSize, Value* second,
|
||||||
Event(c), type(type), size(size), first(first), second(second),
|
unsigned resultSize, Value* result,
|
||||||
|
Site* firstTarget, Site* secondTarget):
|
||||||
|
Event(c), type(type), firstSize(firstSize), first(first),
|
||||||
|
secondSize(secondSize), second(second), resultSize(resultSize),
|
||||||
result(result)
|
result(result)
|
||||||
{
|
{
|
||||||
// todo: we should really specify the sizes of each operand
|
|
||||||
// seperately for binary operations. The following is a hack
|
|
||||||
// until then.
|
|
||||||
unsigned firstSize;
|
|
||||||
switch (type) {
|
|
||||||
case ShiftLeft:
|
|
||||||
case ShiftRight:
|
|
||||||
case UnsignedShiftRight:
|
|
||||||
firstSize = 4;
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
firstSize = size;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
addRead(c, first, firstSize, firstTarget);
|
addRead(c, first, firstSize, firstTarget);
|
||||||
addRead(c, second, size, secondTarget);
|
addRead(c, second, secondSize, secondTarget);
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual void compile(Context* c) {
|
virtual void compile(Context* c) {
|
||||||
@ -1774,23 +1787,26 @@ class CombineEvent: public Event {
|
|||||||
fprintf(stderr, "CombineEvent.compile\n");
|
fprintf(stderr, "CombineEvent.compile\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
maybePreserve(c, stack, size, second, second->source);
|
maybePreserve(c, stack, secondSize, second, second->source);
|
||||||
|
|
||||||
apply(c, type, size, first->source, second->source);
|
Site* target = targetOrRegister(c, result);
|
||||||
|
apply(c, type, firstSize, first->source, secondSize, second->source,
|
||||||
|
resultSize, target);
|
||||||
|
|
||||||
nextRead(c, first);
|
nextRead(c, first);
|
||||||
nextRead(c, second);
|
nextRead(c, second);
|
||||||
|
|
||||||
removeSite(c, second, second->source);
|
if (live(target)) {
|
||||||
if (live(result)) {
|
addSite(c, 0, 0, resultSize, result, target);
|
||||||
addSite(c, 0, 0, size, result, second->source);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
BinaryOperation type;
|
BinaryOperation type;
|
||||||
unsigned size;
|
unsigned firstSize;
|
||||||
Value* first;
|
Value* first;
|
||||||
|
unsigned secondSize;
|
||||||
Value* second;
|
Value* second;
|
||||||
|
unsigned resultSize;
|
||||||
Value* result;
|
Value* result;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -2340,122 +2356,210 @@ propagateJunctionSites(Context* c, Event* e, Site** sites)
|
|||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
frameCount(Context* c, Stack* s)
|
||||||
|
{
|
||||||
|
return c->localCount + s->index + footprint(s);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
populateSiteTables(Context* c, Event* e)
|
||||||
|
{
|
||||||
|
Event* successor = static_cast<Event*>(e->successors->value);
|
||||||
|
|
||||||
|
unsigned frameCount = ::frameCount(c, successor->stack);
|
||||||
|
|
||||||
|
{ Site* frozenSites[frameCount];
|
||||||
|
unsigned frozenSiteIndex = 0;
|
||||||
|
|
||||||
|
if (e->junctionSites) {
|
||||||
|
for (unsigned i = 0; i < frameCount; ++i) {
|
||||||
|
Site* site = e->junctionSites[i];
|
||||||
|
if (site) {
|
||||||
|
frozenSites[frozenSiteIndex++] = site;
|
||||||
|
site->freeze(c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
for (Cell* sc = e->successors; sc; sc = sc->next) {
|
||||||
|
Event* s = static_cast<Event*>(sc->value);
|
||||||
|
if (s->predecessors->next) {
|
||||||
|
unsigned size = sizeof(Site*) * frameCount;
|
||||||
|
Site** junctionSites = static_cast<Site**>
|
||||||
|
(c->zone->allocate(size));
|
||||||
|
memset(junctionSites, 0, size);
|
||||||
|
|
||||||
|
propagateJunctionSites(c, s, junctionSites);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (e->junctionSites) {
|
||||||
|
Event* s = e->next;
|
||||||
|
for (unsigned i = 0; i < c->localCount; ++i) {
|
||||||
|
frozenSiteIndex = resolveJunctionSite
|
||||||
|
(c, e, s, s->locals[i], i, frozenSites, frozenSiteIndex);
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned i = s->stack->index + c->localCount;
|
||||||
|
for (Stack* stack = s->stack; stack; stack = stack->next) {
|
||||||
|
frozenSiteIndex = resolveJunctionSite
|
||||||
|
(c, e, s, stack->value, i, frozenSites, frozenSiteIndex);
|
||||||
|
|
||||||
|
i -= footprint(stack);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
while (frozenSiteIndex) {
|
||||||
|
frozenSites[--frozenSiteIndex]->thaw(c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (e->successors->next) {
|
||||||
|
unsigned size = sizeof(Site*) * frameCount;
|
||||||
|
Site** savedSites = static_cast<Site**>(c->zone->allocate(size));
|
||||||
|
|
||||||
|
for (unsigned i = 0; i < c->localCount; ++i) {
|
||||||
|
savedSites = successor->locals[i]->sites;
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned i = successor->stack->index + c->localCount;
|
||||||
|
for (Stack* stack = successor->stack; stack; stack = stack->next) {
|
||||||
|
savedSites = stack->value->sites;
|
||||||
|
|
||||||
|
i -= footprint(stack);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
setSites(Context* c, Event* e, Site** sites)
|
||||||
|
{
|
||||||
|
for (unsigned i = 0; i < c->localCount; ++i) {
|
||||||
|
Value* v = e->locals[i];
|
||||||
|
clearSites(c, v);
|
||||||
|
addSites(c, v, sites);
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned i = e->stack->index + c->localCount;
|
||||||
|
for (Stack* stack = e->stack; stack; stack = stack->next) {
|
||||||
|
Value* v = stack->value;
|
||||||
|
clearSites(c, v);
|
||||||
|
addSites(c, v, sites);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
populateSources(Context* c, Event* e)
|
||||||
|
{
|
||||||
|
Site* frozenSites[e->readCount];
|
||||||
|
unsigned frozenSiteIndex = 0;
|
||||||
|
for (Read* r = e->reads; r; r = r->eventNext) {
|
||||||
|
r->value->source = readSource(c, e->stack, e->locals, r);
|
||||||
|
|
||||||
|
if (r->value->source) {
|
||||||
|
assert(c, frozenSiteIndex < e->readCount);
|
||||||
|
frozenSites[frozenSiteIndex++] = r->value->source;
|
||||||
|
r->value->source->freeze(c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
while (frozenSiteIndex) {
|
||||||
|
frozenSites[--frozenSiteIndex]->thaw(c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
LogicalInstruction*
|
||||||
|
next(Context* c, LogicalInstruction* i)
|
||||||
|
{
|
||||||
|
for (unsigned n = i->index + 1; n < c->logicalCodeLength; ++n) {
|
||||||
|
i = c->logicalCode[n];
|
||||||
|
if (i) return i;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
class Block {
|
||||||
|
public:
|
||||||
|
Block(Event* head):
|
||||||
|
head(head), nextInstruction(0), offset(0), start(0)
|
||||||
|
{ }
|
||||||
|
|
||||||
|
Event* head;
|
||||||
|
LogicalInstruction* nextInstruction;
|
||||||
|
Assembler::Offset* offset;
|
||||||
|
unsigned start;
|
||||||
|
};
|
||||||
|
|
||||||
|
Block*
|
||||||
|
block(Context* c, Event* head)
|
||||||
|
{
|
||||||
|
return new (c->zone->allocate(sizeof(Block))) Block(head);
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned
|
||||||
compile(Context* c)
|
compile(Context* c)
|
||||||
{
|
{
|
||||||
Assembler* a = c->assembler;
|
Assembler* a = c->assembler;
|
||||||
|
|
||||||
c->pass = CompilePass;
|
c->pass = CompilePass;
|
||||||
|
|
||||||
Assembler::Register base(a->base());
|
Block* firstBlock = block(c, c->firstEvent);
|
||||||
Assembler::Register stack(a->stack());
|
Block* block = firstBlock;
|
||||||
a->apply(Push, BytesPerWord, RegisterOperand, &base);
|
|
||||||
a->apply(Move, BytesPerWord, RegisterOperand, &stack,
|
|
||||||
RegisterOperand, &base);
|
|
||||||
|
|
||||||
if (stackOffset(c)) {
|
Assembler::Constant offset(resolved(c, alignedFrameSize(c)));
|
||||||
Assembler::Constant offset(resolved(c, stackOffset(c) * BytesPerWord));
|
a->apply(PushFrame, BytesPerWord, ConstantOperand, &offset);
|
||||||
a->apply(Subtract, BytesPerWord, ConstantOperand, &offset,
|
|
||||||
RegisterOperand, &stack);
|
|
||||||
}
|
|
||||||
|
|
||||||
for (Event* e = c->firstEvent; e; e = e->next) {
|
for (Event* e = c->firstEvent; e; e = e->next) {
|
||||||
|
e->block = block;
|
||||||
|
|
||||||
if (e->predecessors->next) {
|
if (e->predecessors->next) {
|
||||||
setSites(e, static_cast<Event*>(e->predecessors->value)->junctionSites);
|
setSites
|
||||||
|
(c, e, static_cast<Event*>(e->predecessors->value)->junctionSites);
|
||||||
} else if (e->predecessors->successors->next) {
|
} else if (e->predecessors->successors->next) {
|
||||||
setSites(e, static_cast<Event*>(e->predecessors->value)->savedSites);
|
setSites
|
||||||
|
(c, e, static_cast<Event*>(e->predecessors->value)->savedSites);
|
||||||
}
|
}
|
||||||
|
|
||||||
{ Site* frozenSites[e->readCount];
|
populateSources(c, e);
|
||||||
unsigned frozenSiteIndex = 0;
|
|
||||||
for (Read* r = e->reads; r; r = r->eventNext) {
|
|
||||||
r->value->source = readSource(c, e->stack, e->locals, r);
|
|
||||||
|
|
||||||
if (r->value->source) {
|
e->compile(c);
|
||||||
assert(c, frozenSiteIndex < e->readCount);
|
|
||||||
frozenSites[frozenSiteIndex++] = r->value->source;
|
|
||||||
r->value->source->freeze(c);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
while (frozenSiteIndex) {
|
if (e->successors) {
|
||||||
frozenSites[--frozenSiteIndex]->thaw(c);
|
populateSiteTables(c, e);
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
e->compilePresync(c);
|
|
||||||
|
|
||||||
unsigned frameCount
|
|
||||||
= c->localCount + s->stack->index + footprint(s->stack);
|
|
||||||
|
|
||||||
{ Site* frozenSites[frameCount];
|
|
||||||
unsigned frozenSiteIndex = 0;
|
|
||||||
|
|
||||||
if (e->junctionSites) {
|
|
||||||
for (unsigned i = 0; i < frameCount; ++i) {
|
|
||||||
Site* site = e->junctionSites[i];
|
|
||||||
if (site) {
|
|
||||||
frozenSites[frozenSiteIndex++] = site;
|
|
||||||
site->freeze(c);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
for (Cell* sc = e->successors; sc; sc = sc->next) {
|
|
||||||
Event* s = static_cast<Event*>(sc->value);
|
|
||||||
if (s->predecessors->next) {
|
|
||||||
unsigned size = sizeof(Site*) * frameCount;
|
|
||||||
Site** junctionSites = static_cast<Site**>
|
|
||||||
(c->zone->allocate(size));
|
|
||||||
memset(junctionSites, 0, size);
|
|
||||||
|
|
||||||
propagateJunctionSites(c, s, junctionSites);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (e->junctionSites) {
|
|
||||||
Event* s = e->next;
|
|
||||||
for (unsigned i = 0; i < c->localCount; ++i) {
|
|
||||||
frozenSiteIndex = resolveJunctionSite
|
|
||||||
(c, e, s, s->locals[i], i, frozenSites, frozenSiteIndex);
|
|
||||||
}
|
|
||||||
|
|
||||||
unsigned i = s->stack->index + c->localCount;
|
|
||||||
for (Stack* stack = s->stack; stack; stack = stack->next) {
|
|
||||||
frozenSiteIndex = resolveJunctionSite
|
|
||||||
(c, e, s, stack->value, i, frozenSites, frozenSiteIndex);
|
|
||||||
|
|
||||||
i -= footprint(stack);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
while (frozenSiteIndex) {
|
|
||||||
frozenSites[--frozenSiteIndex]->thaw(c);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (e->successors->next) {
|
|
||||||
unsigned size = sizeof(Site*) * frameCount;
|
|
||||||
Site** savedSites = static_cast<Site**>(c->zone->allocate(size));
|
|
||||||
|
|
||||||
for (unsigned i = 0; i < c->localCount; ++i) {
|
|
||||||
savedSites = s->locals[i]->sites;
|
|
||||||
}
|
|
||||||
|
|
||||||
unsigned i = s->stack->index + c->localCount;
|
|
||||||
for (Stack* stack = s->stack; stack; stack = stack->next) {
|
|
||||||
savedSites = stack->value->sites;
|
|
||||||
|
|
||||||
i -= footprint(stack);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
e->compilePostsync(c);
|
e->compilePostsync(c);
|
||||||
|
|
||||||
for (CodePromise* p = e->promises; p; p = p->next) {
|
for (CodePromise* p = e->promises; p; p = p->next) {
|
||||||
p->offset = a->offset();
|
p->offset = a->offset();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (e->next and e->logicalInstruction->lastEvent == e) {
|
||||||
|
LogicalInstruction* nextInstruction = next(c, e->logicalInstruction);
|
||||||
|
if (nextInstruction != e->next->logicalInstruction) {
|
||||||
|
a->endBlock();
|
||||||
|
|
||||||
|
block->nextInstruction = nextInstruction;
|
||||||
|
block->offset = a->offset();
|
||||||
|
Block* block = block(c, e->next);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
a->endBlock();
|
||||||
|
|
||||||
|
block->nextInstruction = 0;
|
||||||
|
block->offset = a->offset();
|
||||||
|
|
||||||
|
block = firstBlock;
|
||||||
|
while (block->nextInstruction) {
|
||||||
|
Block* next = block->nextInstruction->firstEvent->block;
|
||||||
|
next->start = block->offset->calculate(block->start);
|
||||||
|
block = next;
|
||||||
|
}
|
||||||
|
|
||||||
|
return block->offset->calculate(block->start);
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned
|
unsigned
|
||||||
@ -2486,10 +2590,11 @@ pushState(Context* c)
|
|||||||
void
|
void
|
||||||
saveStack(Context* c)
|
saveStack(Context* c)
|
||||||
{
|
{
|
||||||
if (c->logicalIp >= 0 and not c->logicalCode[c->logicalIp].stackSaved) {
|
if (c->logicalIp >= 0 and not c->logicalCode[c->logicalIp]->stackSaved) {
|
||||||
c->logicalCode[c->logicalIp].stackSaved = true;
|
LogicalInstruction* i = c->logicalCode[c->logicalIp];
|
||||||
c->logicalCode[c->logicalIp].stack = c->state->stack;
|
i->stackSaved = true;
|
||||||
c->logicalCode[c->logicalIp].locals = c->locals;
|
i->stack = c->state->stack;
|
||||||
|
i->locals = c->locals;
|
||||||
|
|
||||||
if (DebugAppend) {
|
if (DebugAppend) {
|
||||||
unsigned count = 0;
|
unsigned count = 0;
|
||||||
@ -2547,7 +2652,7 @@ void
|
|||||||
updateJunctions(Context* c)
|
updateJunctions(Context* c)
|
||||||
{
|
{
|
||||||
for (Junction* j = c->junctions; j; j = j->next) {
|
for (Junction* j = c->junctions; j; j = j->next) {
|
||||||
LogicalInstruction* i = c->logicalCode + j->logicalIp;
|
LogicalInstruction* i = c->logicalCode[j->logicalIp];
|
||||||
LogicalInstruction* p = i->immediatePredecessor;
|
LogicalInstruction* p = i->immediatePredecessor;
|
||||||
|
|
||||||
p->lastEvent = p->lastEvent->next
|
p->lastEvent = p->lastEvent->next
|
||||||
@ -2561,10 +2666,14 @@ visit(Context* c, unsigned logicalIp)
|
|||||||
{
|
{
|
||||||
assert(c, logicalIp < c->logicalCodeLength);
|
assert(c, logicalIp < c->logicalCodeLength);
|
||||||
|
|
||||||
|
LogicalInstruction* i = new (c->zone->allocate(sizeof(LogicalInstruction)))
|
||||||
|
LogicalInstruction;
|
||||||
|
|
||||||
|
c->logicalCode[logicalIp] = i;
|
||||||
|
|
||||||
if (c->logicalIp >= 0 and (not c->stackReset)) {
|
if (c->logicalIp >= 0 and (not c->stackReset)) {
|
||||||
assert(c, c->logicalCode[logicalIp].immediatePredecessor == 0);
|
assert(c, i->immediatePredecessor == 0);
|
||||||
c->logicalCode[logicalIp].immediatePredecessor
|
i->immediatePredecessor = c->logicalCode[c->logicalIp];
|
||||||
= c->logicalCode + c->logicalIp;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2635,9 +2744,9 @@ class MyCompiler: public Compiler {
|
|||||||
c.parameterFootprint = parameterFootprint;
|
c.parameterFootprint = parameterFootprint;
|
||||||
c.localFootprint = localFootprint;
|
c.localFootprint = localFootprint;
|
||||||
|
|
||||||
c.logicalCode = static_cast<LogicalInstruction*>
|
c.logicalCode = static_cast<LogicalInstruction**>
|
||||||
(c.zone->allocate(sizeof(LogicalInstruction) * logicalCodeLength));
|
(c.zone->allocate(sizeof(LogicalInstruction*) * logicalCodeLength));
|
||||||
memset(c.logicalCode, 0, sizeof(LogicalInstruction) * logicalCodeLength);
|
memset(c.logicalCode, 0, sizeof(LogicalInstruction*) * logicalCodeLength);
|
||||||
|
|
||||||
c.localTable = static_cast<Local**>
|
c.localTable = static_cast<Local**>
|
||||||
(c.zone->allocate(sizeof(Local*) * localFootprint));
|
(c.zone->allocate(sizeof(Local*) * localFootprint));
|
||||||
@ -2649,7 +2758,7 @@ class MyCompiler: public Compiler {
|
|||||||
|
|
||||||
c.stackReset = false;
|
c.stackReset = false;
|
||||||
|
|
||||||
if (c.logicalCode[logicalIp].immediatePredecessor) {
|
if (c.logicalCode[logicalIp]->immediatePredecessor) {
|
||||||
c.junctions = new (c.zone->allocate(sizeof(Junction)))
|
c.junctions = new (c.zone->allocate(sizeof(Junction)))
|
||||||
Junction(logicalIp, c.junctions);
|
Junction(logicalIp, c.junctions);
|
||||||
}
|
}
|
||||||
@ -2739,7 +2848,7 @@ class MyCompiler: public Compiler {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Promise* machineIp() {
|
Promise* machineIp() {
|
||||||
return codePromise(&c, c.logicalCode[c.logicalIp].lastEvent);
|
return codePromise(&c, c.logicalCode[c.logicalIp]->lastEvent);
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual void mark(Operand* label) {
|
virtual void mark(Operand* label) {
|
||||||
@ -3075,8 +3184,7 @@ class MyCompiler: public Compiler {
|
|||||||
|
|
||||||
virtual unsigned compile() {
|
virtual unsigned compile() {
|
||||||
updateJunctions(&c);
|
updateJunctions(&c);
|
||||||
::compile(&c);
|
return ::compile(&c);
|
||||||
return c.assembler->length();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual unsigned poolSize() {
|
virtual unsigned poolSize() {
|
||||||
@ -3085,7 +3193,7 @@ class MyCompiler: public Compiler {
|
|||||||
|
|
||||||
virtual void writeTo(uint8_t* dst) {
|
virtual void writeTo(uint8_t* dst) {
|
||||||
c.machineCode = dst;
|
c.machineCode = dst;
|
||||||
c.assembler->writeTo(dst);
|
c.assembler->writeTo(&c, dst);
|
||||||
|
|
||||||
int i = 0;
|
int i = 0;
|
||||||
for (ConstantPoolNode* n = c.firstConstant; n; n = n->next) {
|
for (ConstantPoolNode* n = c.firstConstant; n; n = n->next) {
|
||||||
|
Loading…
Reference in New Issue
Block a user