mirror of
https://github.com/corda/corda.git
synced 2025-01-23 21:08:48 +00:00
refactor logical stack mechanism in JIT code
This commit is contained in:
parent
f8cda0cd85
commit
6b1f01511b
104
src/compiler.cpp
104
src/compiler.cpp
@ -37,7 +37,7 @@ class Context {
|
|||||||
Context(System* s, void* indirectCaller):
|
Context(System* s, void* indirectCaller):
|
||||||
s(s),
|
s(s),
|
||||||
code(s, 1024),
|
code(s, 1024),
|
||||||
virtualStack(s, BytesPerWord * 32),
|
logicalStack(s, BytesPerWord * 32),
|
||||||
operands(s, 8 * 1024),
|
operands(s, 8 * 1024),
|
||||||
ipTable(s, sizeof(IpMapping) * 512),
|
ipTable(s, sizeof(IpMapping) * 512),
|
||||||
constantPool(s, BytesPerWord * 32),
|
constantPool(s, BytesPerWord * 32),
|
||||||
@ -53,13 +53,13 @@ class Context {
|
|||||||
registerPool.dispose();
|
registerPool.dispose();
|
||||||
ipTable.dispose();
|
ipTable.dispose();
|
||||||
operands.dispose();
|
operands.dispose();
|
||||||
virtualStack.dispose();
|
logicalStack.dispose();
|
||||||
code.dispose();
|
code.dispose();
|
||||||
}
|
}
|
||||||
|
|
||||||
System* s;
|
System* s;
|
||||||
Vector code;
|
Vector code;
|
||||||
Vector virtualStack;
|
Vector logicalStack;
|
||||||
Vector operands;
|
Vector operands;
|
||||||
Vector ipTable;
|
Vector ipTable;
|
||||||
Vector constantPool;
|
Vector constantPool;
|
||||||
@ -157,6 +157,10 @@ class MyOperand: public Operand {
|
|||||||
|
|
||||||
virtual unsigned size() = 0;
|
virtual unsigned size() = 0;
|
||||||
|
|
||||||
|
virtual void logicalPush(Context* c) { abort(c); }
|
||||||
|
|
||||||
|
virtual void logicalFlush(Context* c) { abort(c); }
|
||||||
|
|
||||||
virtual void push(Context* c) { abort(c); }
|
virtual void push(Context* c) { abort(c); }
|
||||||
|
|
||||||
virtual void pop(Context* c) { abort(c); }
|
virtual void pop(Context* c) { abort(c); }
|
||||||
@ -446,10 +450,6 @@ class MemoryOperand: public MyOperand {
|
|||||||
return sizeof(MemoryOperand);
|
return sizeof(MemoryOperand);
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual bool isStackReference() {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual void push(Context* c) {
|
virtual void push(Context* c) {
|
||||||
assert(c, index == 0);
|
assert(c, index == 0);
|
||||||
assert(c, scale == 0);
|
assert(c, scale == 0);
|
||||||
@ -482,21 +482,6 @@ class MemoryOperand: public MyOperand {
|
|||||||
unsigned scale;
|
unsigned scale;
|
||||||
};
|
};
|
||||||
|
|
||||||
class StackOperand: public MemoryOperand {
|
|
||||||
public:
|
|
||||||
StackOperand(MyOperand* base, int displacement):
|
|
||||||
MemoryOperand(base, displacement, 0, 1)
|
|
||||||
{ }
|
|
||||||
|
|
||||||
virtual unsigned size() {
|
|
||||||
return sizeof(StackOperand);
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual bool isStackReference() {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
class SelectionOperand: public MyOperand {
|
class SelectionOperand: public MyOperand {
|
||||||
public:
|
public:
|
||||||
enum SelectionType {
|
enum SelectionType {
|
||||||
@ -556,10 +541,11 @@ memory(Context* c, MyOperand* base, int displacement,
|
|||||||
return c->operands.push(MemoryOperand(base, displacement, index, scale));
|
return c->operands.push(MemoryOperand(base, displacement, index, scale));
|
||||||
}
|
}
|
||||||
|
|
||||||
StackOperand*
|
MemoryOperand*
|
||||||
stack(Context* c, int displacement)
|
stack(Context* c, int displacement)
|
||||||
{
|
{
|
||||||
return c->operands.push(StackOperand(register_(c, rbp), displacement));
|
return c->operands.push
|
||||||
|
(MemoryOperand(register_(c, rbp), displacement, 0, 1));
|
||||||
}
|
}
|
||||||
|
|
||||||
MyOperand*
|
MyOperand*
|
||||||
@ -574,41 +560,40 @@ selection(Context* c, SelectionOperand::SelectionType type, MyOperand* base)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool
|
|
||||||
isStackReference(Context*, MyOperand* v)
|
|
||||||
{
|
|
||||||
return v->type() == MyOperand::MemoryOperandType
|
|
||||||
and static_cast<MemoryOperand*>(v)->isStackReference();
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
void
|
||||||
flushStack(Context* c)
|
flushStack(Context* c)
|
||||||
{
|
{
|
||||||
Vector newVirtualStack(c->s, c->virtualStack.length() * 2);
|
for (unsigned i = 0; i < c->logicalStack.length(); i += BytesPerWord) {
|
||||||
for (unsigned i = 0; i < c->virtualStack.length();) {
|
(*c->logicalStack.peek<MyOperand*>(i))->logicalFlush(c);
|
||||||
MyOperand* v = c->virtualStack.peek<MyOperand>(i);
|
|
||||||
|
|
||||||
if (not isStackReference(c, v)) {
|
|
||||||
v->push(c);
|
|
||||||
|
|
||||||
if (v->footprint() / BytesPerWord == 2) {
|
|
||||||
newVirtualStack.push(stack(c, c->stackIndex + 4));
|
|
||||||
} else {
|
|
||||||
newVirtualStack.push(stack(c, c->stackIndex));
|
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
newVirtualStack.append(v, v->size());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
i += v->size();
|
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);
|
||||||
}
|
}
|
||||||
|
|
||||||
c->virtualStack.update(&newVirtualStack);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned
|
unsigned
|
||||||
pushArguments(Context* c, unsigned count, va_list list)
|
pushArguments(Context* c, unsigned count, va_list list)
|
||||||
{
|
{
|
||||||
|
flushStack(c);
|
||||||
|
|
||||||
MyOperand* arguments[count];
|
MyOperand* arguments[count];
|
||||||
unsigned footprint = 0;
|
unsigned footprint = 0;
|
||||||
for (unsigned i = 0; i < count; ++i) {
|
for (unsigned i = 0; i < count; ++i) {
|
||||||
@ -616,12 +601,25 @@ pushArguments(Context* c, unsigned count, va_list list)
|
|||||||
footprint += pad(arguments[i]->footprint());
|
footprint += pad(arguments[i]->footprint());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const int GprCount = 6;
|
||||||
for (int i = count - 1; i >= 0; --i) {
|
for (int i = count - 1; i >= 0; --i) {
|
||||||
|
if (BytesPerWord == 8 and i < GprCount) {
|
||||||
|
arguments[i]->mov(c, register_(c, gpRegister(c, i)));
|
||||||
|
} else {
|
||||||
arguments[i]->push(c);
|
arguments[i]->push(c);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (BytesPerWord == 8) {
|
||||||
|
if (footprint > GprCount * BytesPerWord) {
|
||||||
|
return footprint - GprCount * BytesPerWord;
|
||||||
|
} else {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
return footprint;
|
return footprint;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
class MyCompiler: public Compiler {
|
class MyCompiler: public Compiler {
|
||||||
public:
|
public:
|
||||||
@ -649,17 +647,17 @@ class MyCompiler: public Compiler {
|
|||||||
}
|
}
|
||||||
|
|
||||||
virtual void push(Operand* v) {
|
virtual void push(Operand* v) {
|
||||||
c.virtualStack.push(static_cast<MyOperand*>(v));
|
static_cast<MyOperand*>(v)->logicalPush(&c);
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual void push2(Operand* v) {
|
virtual void push2(Operand* v) {
|
||||||
push(v);
|
push(v);
|
||||||
if (BytesPerWord == 8) push(0);
|
if (BytesPerWord == 8) push(immediate(&c, 0));
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual Operand* stack(unsigned index) {
|
virtual Operand* stack(unsigned index) {
|
||||||
return c.virtualStack.peek<MyOperand>
|
return c.logicalStack.peek<MyOperand>
|
||||||
(c.virtualStack.length() - ((index + 1) * BytesPerWord));
|
(c.logicalStack.length() - ((index + 1) * BytesPerWord));
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual Operand* stack2(unsigned index) {
|
virtual Operand* stack2(unsigned index) {
|
||||||
@ -667,7 +665,7 @@ class MyCompiler: public Compiler {
|
|||||||
}
|
}
|
||||||
|
|
||||||
virtual Operand* pop() {
|
virtual Operand* pop() {
|
||||||
return c.virtualStack.pop<MyOperand*>();
|
return c.logicalStack.pop<MyOperand*>();
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual Operand* pop2() {
|
virtual Operand* pop2() {
|
||||||
@ -676,7 +674,7 @@ class MyCompiler: public Compiler {
|
|||||||
}
|
}
|
||||||
|
|
||||||
virtual void pop(Operand* dst) {
|
virtual void pop(Operand* dst) {
|
||||||
c.virtualStack.pop<MyOperand*>()->mov(&c, static_cast<MyOperand*>(dst));
|
c.logicalStack.pop<MyOperand*>()->mov(&c, static_cast<MyOperand*>(dst));
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual void pop2(Operand* dst) {
|
virtual void pop2(Operand* dst) {
|
||||||
|
@ -103,15 +103,6 @@ class Vector {
|
|||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
void update(Vector* v) {
|
|
||||||
dispose();
|
|
||||||
|
|
||||||
data = v->data;
|
|
||||||
position = v->position;
|
|
||||||
capacity = v->capacity;
|
|
||||||
minimumCapacity = v->minimumCapacity;
|
|
||||||
}
|
|
||||||
|
|
||||||
System* s;
|
System* s;
|
||||||
uint8_t* data;
|
uint8_t* data;
|
||||||
unsigned position;
|
unsigned position;
|
||||||
|
Loading…
Reference in New Issue
Block a user