corda/src/powerpc.cpp

1036 lines
28 KiB
C++
Raw Normal View History

2009-02-24 02:16:42 +00:00
/* Copyright (c) 2009, Avian Contributors
2008-06-04 22:21:27 +00:00
2009-02-24 02:16:42 +00:00
Permission to use, copy, modify, and/or distribute this software
for any purpose with or without fee is hereby granted, provided
that the above copyright notice and this permission notice appear
in all copies.
2009-02-24 02:16:42 +00:00
There is NO WARRANTY for this software. See license.txt for
details. */
2009-02-24 02:16:42 +00:00
#include "assembler.h"
#include "vector.h"
2009-02-24 02:16:42 +00:00
#define CAST1(x) reinterpret_cast<UnaryOperationType>(x)
#define CAST2(x) reinterpret_cast<BinaryOperationType>(x)
2009-02-24 02:16:42 +00:00
#define CAST3(x) reinterpret_cast<TernaryOperationType>(x)
2009-02-24 02:16:42 +00:00
using namespace vm;
2009-02-24 02:16:42 +00:00
namespace field {
// BITFIELD MASKS
const int64_t MASK_LO32 = 0x0ffffffff;
const int MASK_LO16 = 0x0ffff;
const int MASK_LO8 = 0x0ff;
// BITFIELD EXTRACTORS
inline int lo32(int64_t i) { return (int)(i & MASK_LO32); }
inline int hi32(int64_t i) { return lo32(i >> 32); }
inline int lo16(int64_t i) { return (int)(i & MASK_LO16); }
inline int hi16(int64_t i) { return lo16(i >> 16); }
inline int lo8(int64_t i) { return (int)(i & MASK_LO8); }
inline int hi8(int64_t i) { return lo8(i >> 8); }
}
namespace isa {
// INSTRUCTION FORMATS
inline int D(int op, int rt, int ra, int d) { return op<<26|rt<<21|ra<<16|d; }
inline int DS(int op, int rt, int ra, int ds, int xo) { return op<<26|rt<<21|ra<<16|ds<<2|xo; }
inline int I(int op, int li, int aa, int lk) { return op<<26|li<<2|aa<<1|lk; }
inline int B(int op, int bo, int bi, int bd, int aa, int lk) { return op<<26|bo<<21|bi<<16|bd<<2|aa<<1|lk; }
inline int SC(int op, int lev) { return op<<26|lev<<5|2; }
inline int X(int op, int rt, int ra, int rb, int xo, int rc) { return op<<26|rt<<21|ra<<16|rb<<11|xo<<1|rc; }
inline int XL(int op, int bt, int ba, int bb, int xo, int lk) { return op<<26|bt<<21|ba<<16|bb<<11|xo<<1|lk; }
inline int XFX(int op, int rt, int spr, int xo) { return op<<26|rt<<21|spr<<11|xo<<1; }
inline int XFL(int op, int flm, int frb, int xo, int rc) { return op<<26|flm<<17|frb<<11|xo<<1|rc; }
inline int XS(int op, int rs, int ra, int sh, int xo, int sh2, int rc) { return op<<26|rs<<21|ra<<16|sh<<11|xo<<2|sh2<<1|rc; }
inline int XO(int op, int rt, int ra, int rb, int oe, int xo, int rc) { return op<<26|rt<<21|ra<<16|rb<<11|oe<<10|xo<<1|rc; }
inline int A(int op, int frt, int fra, int frb, int frc, int xo, int rc) { return op<<26|frt<<21|fra<<16|frb<<11|frc<<6|xo<<1|rc; }
inline int M(int op, int rs, int ra, int rb, int mb, int me, int rc) { return op<<26|rs<<21|ra<<16|rb<<11|mb<<6|me<<1|rc; }
inline int MD(int op, int rs, int ra, int sh, int mb, int xo, int sh2, int rc) { return op<<26|rs<<21|ra<<16|sh<<11|mb<<5|xo<<2|sh2<<1|rc; }
inline int MDS(int op, int rs, int ra, int rb, int mb, int xo, int rc) { return op<<26|rs<<21|ra<<16|rb<<11|mb<<5|xo<<1|rc; }
// INSTRUCTIONS
inline int lbz(int rt, int ra, int i) { return D(34, rt, ra, i); }
inline int lbzx(int rt, int ra, int rb) { return X(34, rt, ra, rb, 87, 0); }
inline int lhz(int rt, int ra, int i) { return D(40, rt, ra, i); }
inline int lha(int rt, int ra, int i) { return D(42, rt, ra, i); }
inline int lhax(int rt, int ra, int rb) { return X(31, rt, ra, rb, 343, 0); }
inline int lwz(int rt, int ra, int i) { return D(32, rt, ra, i); }
inline int lwzx(int rt, int ra, int rb) { return X(31, rt, ra, rb, 23, 0); }
inline int stb(int rs, int ra, int i) { return D(38, rs, ra, i); }
inline int stbx(int rs, int ra, int rb) { return X(31, rs, ra, rb, 215, 0); }
inline int sth(int rs, int ra, int i) { return D(44, rs, ra, i); }
inline int sthx(int rs, int ra, int rb) { return X(31, rs, ra, rb, 407, 0); }
inline int stw(int rs, int ra, int i) { return D(36, rs, ra, i); }
inline int stwu(int rs, int ra, int i) { return D(37, rs, ra, i); }
inline int stwx(int rs, int ra, int rb) { return X(31, rs, ra, rb, 151, 0); }
inline int add(int rt, int ra, int rb) { return XO(31, rt, ra, rb, 0, 266, 0); }
inline int addc(int rt, int ra, int rb) { return XO(31, rt, ra, rb, 0, 10, 0); }
inline int adde(int rt, int ra, int rb) { return XO(31, rt, ra, rb, 0, 138, 0); }
inline int addi(int rt, int ra, int i) { return D(14, rt, ra, i); }
inline int addis(int rt, int ra, int i) { return D(15, rt, ra, i); }
inline int subf(int rt, int ra, int rb) { return XO(31, rt, ra, rb, 0, 40, 0); }
inline int subfc(int rt, int ra, int rb) { return XO(31, rt, ra, rb, 0, 8, 0); }
inline int subfe(int rt, int ra, int rb) { return XO(31, rt, ra, rb, 0, 136, 0); }
inline int _and(int rt, int ra, int rb) { return X(31, ra, rt, rb, 28, 0); }
inline int andi(int rt, int ra, int i) { return D(28, ra, rt, i); }
inline int andis(int rt, int ra, int i) { return D(29, ra, rt, i); }
inline int _or(int rt, int ra, int rb) { return X(31, ra, rt, rb, 444, 0); }
inline int ori(int rt, int ra, int i) { return D(24, rt, ra, i); }
inline int oris(int rt, int ra, int i) { return D(25, rt, ra, i); }
inline int rlwinm(int rt, int ra, int i, int mb, int me) { return M(21, ra, rt, i, mb, me, 0); }
inline int rlwimi(int rt, int ra, int i, int mb, int me) { return M(20, ra, rt, i, mb, me, 0); }
inline int slw(int rt, int ra, int sh) { return X(31, ra, rt, sh, 21, 0); }
inline int sld(int rt, int ra, int rb) { return X(31, ra, rt, rb, 27, 0); }
inline int srw(int rt, int ra, int sh) { return X(31, ra, rt, sh, 536, 0); }
inline int sraw(int rt, int ra, int sh) { return X(31, ra, rt, sh, 792, 0); }
inline int srawi(int rt, int ra, int sh) { return X(31, ra, rt, sh, 824, 0); }
inline int extsb(int rt, int rs) { return X(31, rs, rt, 0, 954, 0); }
inline int extsh(int rt, int rs) { return X(31, rs, rt, 0, 922, 0); }
inline int mfspr(int rt, int spr) { return XFX(31, rt, spr, 339); }
inline int mtspr(int spr, int rs) { return XFX(31, rs, spr, 467); }
// PSEUDO-INSTRUCTIONS
inline int li(int rt, int i) { return ori(rt, 0, i); }
inline int lis(int rt, int i) { return oris(rt, 0, i); }
inline int slwi(int rt, int ra, int i) { return rlwinm(rt, ra, i, 0, 31-i); }
inline int srwi(int rt, int ra, int i) { return rlwinm(rt, ra, 32-i, i, 31); }
inline int sub(int rt, int ra, int rb) { return subf(rt, rb, ra); }
inline int subc(int rt, int ra, int rb) { return subfc(rt, rb, ra); }
inline int subi(int rt, int ra, int i) { return addi(rt, ra, -i); }
inline int mr(int rt, int ra) { return _or(rt, ra, ra); }
inline int mflr(int rx) { return mfspr(rx, 8); }
inline int mtlr(int rx) { return mtspr(8, rx); }
}
2009-02-24 02:16:42 +00:00
namespace {
2009-02-24 02:16:42 +00:00
const unsigned FrameFooterSize = 6;
const int StackRegister = 1;
const int ThreadRegister = 20;
2009-02-24 02:16:42 +00:00
class MyBlock: public Assembler::Block {
public:
2009-02-24 02:16:42 +00:00
MyBlock(unsigned offset):
next(0), offset(offset), start(~0), size(0)
{ }
2009-02-24 02:16:42 +00:00
virtual unsigned resolve(unsigned start, Assembler::Block* next) {
this->start = start;
this->next = static_cast<MyBlock*>(next);
2009-02-24 02:16:42 +00:00
return start + size;
}
2009-02-24 02:16:42 +00:00
MyBlock* next;
unsigned offset;
2009-02-24 02:16:42 +00:00
unsigned start;
unsigned size;
};
2009-02-24 02:16:42 +00:00
class Task;
2009-02-24 02:16:42 +00:00
class Context {
public:
2009-02-24 02:16:42 +00:00
Context(System* s, Allocator* a, Zone* zone):
s(s), zone(zone), client(0), code(s, a, 1024), tasks(0), result(0),
firstBlock(new (zone->allocate(sizeof(MyBlock))) MyBlock(0)),
lastBlock(firstBlock)
{ }
2009-02-24 02:16:42 +00:00
System* s;
Zone* zone;
Assembler::Client* client;
Vector code;
Task* tasks;
uint8_t* result;
MyBlock* firstBlock;
MyBlock* lastBlock;
};
2009-02-24 02:16:42 +00:00
class Task {
public:
2009-02-24 02:16:42 +00:00
Task(Task* next): next(next) { }
2009-02-24 02:16:42 +00:00
virtual void run(Context* c) = 0;
2009-02-24 02:16:42 +00:00
Task* next;
};
typedef void (*OperationType)(Context*);
typedef void (*UnaryOperationType)(Context*, unsigned, Assembler::Operand*);
typedef void (*BinaryOperationType)
2009-02-24 02:16:42 +00:00
(Context*, unsigned, Assembler::Operand*, unsigned, Assembler::Operand*);
2009-02-24 02:16:42 +00:00
typedef void (*TernaryOperationType)
(Context*, unsigned, Assembler::Operand*, unsigned, Assembler::Operand*,
unsigned, Assembler::Operand*);
2009-02-24 02:16:42 +00:00
class ArchitectureContext {
public:
ArchitectureContext(System* s): s(s) { }
System* s;
OperationType operations[OperationCount];
UnaryOperationType unaryOperations[UnaryOperationCount
* OperandTypeCount];
BinaryOperationType binaryOperations
[BinaryOperationCount * OperandTypeCount * OperandTypeCount];
TernaryOperationType ternaryOperations
[TernaryOperationCount * OperandTypeCount];
};
2009-02-24 02:16:42 +00:00
inline void NO_RETURN
abort(Context* c)
{
2009-02-24 02:16:42 +00:00
abort(c->s);
}
2009-02-24 02:16:42 +00:00
inline void NO_RETURN
abort(ArchitectureContext* c)
{
2009-02-24 02:16:42 +00:00
abort(c->s);
}
2009-02-24 02:16:42 +00:00
#ifndef NDEBUG
inline void
assert(Context* c, bool v)
{
2009-02-24 02:16:42 +00:00
assert(c->s, v);
}
2009-02-24 02:16:42 +00:00
inline void
assert(ArchitectureContext* c, bool v)
{
2009-02-24 02:16:42 +00:00
assert(c->s, v);
}
2009-02-24 02:16:42 +00:00
#endif // not NDEBUG
2009-02-24 02:16:42 +00:00
inline void
expect(Context* c, bool v)
{
2009-02-24 02:16:42 +00:00
expect(c->s, v);
}
2009-02-24 02:16:42 +00:00
class Offset: public Promise {
public:
Offset(Context* c, MyBlock* block, unsigned offset):
c(c), block(block), offset(offset)
{ }
2009-02-24 02:16:42 +00:00
virtual bool resolved() {
return block->start != static_cast<unsigned>(~0);
}
2009-02-24 02:16:42 +00:00
virtual int64_t value() {
assert(c, resolved());
2009-02-24 02:16:42 +00:00
return block->start + (offset - block->offset);
}
2009-02-24 02:16:42 +00:00
Context* c;
MyBlock* block;
unsigned offset;
};
2009-02-24 02:16:42 +00:00
Promise*
offset(Context* c)
{
2009-02-24 02:16:42 +00:00
return new (c->zone->allocate(sizeof(Offset)))
Offset(c, c->lastBlock, c->code.length());
}
2009-02-24 02:16:42 +00:00
inline unsigned
index(UnaryOperation operation, OperandType operand)
{
2009-02-24 02:16:42 +00:00
return operation + (UnaryOperationCount * operand);
}
2009-02-24 02:16:42 +00:00
inline unsigned
index(BinaryOperation operation,
OperandType operand1,
OperandType operand2)
{
2009-02-24 02:16:42 +00:00
return operation
+ (BinaryOperationCount * operand1)
+ (BinaryOperationCount * OperandTypeCount * operand2);
}
2009-02-24 02:16:42 +00:00
inline unsigned
index(TernaryOperation operation,
OperandType operand1)
{
return operation + (TernaryOperationCount * operand1);
}
2009-02-24 02:16:42 +00:00
// BEGIN OPERATION COMPILERS
using namespace field;
using namespace isa;
typedef Assembler::Register Reg;
typedef Assembler::Constant Const;
inline void issue(Context* con, int code) { con->code.append4(code); }
inline int getTemp(Context* con) { return con->client->acquireTemporary(); }
inline void freeTemp(Context* con, int r) { con->client->releaseTemporary(r); }
inline int64_t getVal(Const* c) { return c->value->value(); }
inline int R(Reg* r) { return r->low; }
inline int H(Reg* r) { return r->high; }
void shiftLeftRRR(Context* con, unsigned size, Reg* a, Reg* b, Reg* t)
{
if(size == 8) {
2009-02-24 02:16:42 +00:00
abort(con); // todo
} else
2009-02-24 02:16:42 +00:00
issue(con, slw(R(t), R(b), R(a)));
}
2009-02-24 02:16:42 +00:00
void shiftLeftCRR(Context* con, unsigned size, Const* a, Reg* b, Reg* t)
{
2009-02-24 02:16:42 +00:00
int sh = getVal(a);
if (size == 8) {
abort(con); // todo
} else
2009-02-24 02:16:42 +00:00
issue(con, slwi(R(t), R(b), sh));
}
2009-02-24 02:16:42 +00:00
void shiftRightRRR(Context* con, unsigned size, Reg* a, Reg* b, Reg* t)
{
if(size == 8) {
2009-02-24 02:16:42 +00:00
abort(con); // todo
} else
2009-02-24 02:16:42 +00:00
issue(con, sraw(R(t), R(b), R(a)));
}
2009-02-24 02:16:42 +00:00
void shiftRightCRR(Context* con, unsigned size, Const* a, Reg* b, Reg* t)
{
2009-02-24 02:16:42 +00:00
int sh = getVal(a);
if(size == 8) {
2009-02-24 02:16:42 +00:00
abort(con); // todo
} else
2009-02-24 02:16:42 +00:00
issue(con, srawi(R(t), R(b), sh));
}
2009-02-24 02:16:42 +00:00
void unsignedShiftRightRRR(Context* con, unsigned size, Reg* a, Reg* b, Reg* t)
{
if(size == 8) {
2009-02-24 02:16:42 +00:00
abort(con); // todo
} else
2009-02-24 02:16:42 +00:00
issue(con, srw(R(t), R(b), R(a)));
}
2009-02-24 02:16:42 +00:00
void unsignedShiftRightCRR(Context* con, unsigned size, Const* a, Reg* b, Reg* t)
{
2009-02-24 02:16:42 +00:00
int sh = getVal(a);
if (size == 8) {
abort(con); // todo
} else
2009-02-24 02:16:42 +00:00
issue(con, srwi(R(t), R(b), sh));
}
void
2009-02-24 02:16:42 +00:00
moveRR(Context* c, unsigned srcSize, Assembler::Register* src,
unsigned dstSize, Assembler::Register* dst);
2009-02-24 02:16:42 +00:00
void
swapRR(Context* c, unsigned aSize, Assembler::Register* a,
unsigned bSize, Assembler::Register* b)
{
assert(c, aSize == BytesPerWord);
assert(c, bSize == BytesPerWord);
2009-02-24 02:16:42 +00:00
Assembler::Register tmp(c->client->acquireTemporary());
moveRR(c, aSize, a, bSize, &tmp);
moveRR(c, bSize, b, aSize, a);
moveRR(c, bSize, &tmp, bSize, b);
c->client->releaseTemporary(tmp.low);
}
void
2009-02-24 02:16:42 +00:00
moveRR(Context* c, unsigned srcSize, Assembler::Register* src,
unsigned dstSize, Assembler::Register* dst)
{
2009-02-24 02:16:42 +00:00
switch (srcSize) {
case 1:
issue(c, extsb(src->low, dst->low));
break;
case 2:
issue(c, extsh(src->low, dst->low));
break;
case 4:
case 8:
if (srcSize == 4 and dstSize == 8) {
Assembler::Register dstHigh(dst->high);
moveRR(c, 4, src, 4, dst);
moveRR(c, 4, src, 4, &dstHigh);
issue(c, srawi(dst->high, dst->high, 31));
} else if (srcSize == 8 and dstSize == 8) {
Assembler::Register srcHigh(src->high);
Assembler::Register dstHigh(dst->high);
if (src->high == dst->low) {
if (src->low == dst->high) {
swapRR(c, 4, src, 4, dst);
} else {
moveRR(c, 4, &srcHigh, 4, &dstHigh);
moveRR(c, 4, src, 4, dst);
}
} else {
moveRR(c, 4, src, 4, dst);
moveRR(c, 4, &srcHigh, 4, &dstHigh);
}
} else if (src->low != dst->low) {
issue(c, mr(dst->low, src->low));
}
break;
}
}
2009-02-24 02:16:42 +00:00
void moveCR(Context* con, unsigned aSize, Const* a, unsigned tSize, Reg* t) {
int64_t i = getVal(a);
if(tSize == 8) {
int64_t j;
if(aSize == 8) j = i; // 64-bit const -> load high bits into high register
else j = 0; // 32-bit const -> clear high register
issue(con, lis(H(t), hi16(hi32(j))));
issue(con, ori(H(t), H(t), lo16(hi32(j))));
}
issue(con, lis(R(t), hi16(i)));
issue(con, ori(R(t), R(t), lo16(i)));
}
2009-02-24 02:16:42 +00:00
int
normalize(Context* c, int offset, int index, unsigned scale,
bool* preserveIndex)
{
2009-02-24 02:16:42 +00:00
if (offset != 0 or scale != 1) {
Assembler::Register normalizedIndex
(*preserveIndex ? c->client->acquireTemporary() : index);
*preserveIndex = false;
2009-02-24 02:16:42 +00:00
int scaled;
2009-02-24 02:16:42 +00:00
if (scale != 1) {
Assembler::Register unscaledIndex(index);
2009-02-24 02:16:42 +00:00
ResolvedPromise scalePromise(log(scale));
Assembler::Constant scaleConstant(&scalePromise);
shiftLeftCRR(c, BytesPerWord, &scaleConstant,
&unscaledIndex, &normalizedIndex);
2009-02-24 02:16:42 +00:00
scaled = normalizedIndex.low;
} else {
2009-02-24 02:16:42 +00:00
scaled = index;
}
if (offset != 0) {
Assembler::Register untranslatedIndex(scaled);
ResolvedPromise offsetPromise(offset);
Assembler::Constant offsetConstant(&offsetPromise);
addCRR(c, BytesPerWord, &offsetConstant,
&untranslatedIndex, &normalizedIndex);
}
2009-02-24 02:16:42 +00:00
return normalizedIndex.low;
} else {
return index;
}
}
void
2009-02-24 02:16:42 +00:00
store(Context* c, unsigned size, Assembler::Register* src,
int base, int offset, int index, unsigned scale, bool preserveIndex)
{
2009-02-24 02:16:42 +00:00
if (index != NoRegister) {
int normalized = normalize(c, offset, index, scale, &preserveIndex);
2009-02-24 02:16:42 +00:00
switch (size) {
case 1:
issue(c, stbx(src->low, base, normalized));
break;
2009-02-24 02:16:42 +00:00
case 2:
issue(c, sthx(src->low, base, normalized));
break;
2009-02-24 02:16:42 +00:00
case 4:
issue(c, stwx(src->low, base, normalized));
break;
2009-02-24 02:16:42 +00:00
case 8: {
Assembler::Register srcHigh(src->high);
store(c, 4, &srcHigh, base, 0, normalized, 1, preserveIndex);
store(c, 4, src, base, 4, normalized, 1, preserveIndex);
} break;
2009-02-24 02:16:42 +00:00
default: abort(c);
}
2009-02-24 02:16:42 +00:00
} else {
switch (size) {
case 1:
issue(c, stb(src->low, base, offset));
break;
2009-02-24 02:16:42 +00:00
case 2:
issue(c, sth(src->low, base, offset));
break;
2009-02-24 02:16:42 +00:00
case 4:
issue(c, stw(src->low, base, offset));
break;
case 8: {
Assembler::Register srcHigh(src->high);
store(c, 4, &srcHigh, base, offset, NoRegister, 1, false);
store(c, 4, src, base, offset + 4, NoRegister, 1, false);
} break;
default: abort(c);
}
}
}
void
2009-02-24 02:16:42 +00:00
moveRM(Context* c, unsigned srcSize, Assembler::Register* src,
unsigned dstSize UNUSED, Assembler::Memory* dst)
{
2009-02-24 02:16:42 +00:00
assert(c, srcSize == dstSize);
2009-02-24 02:16:42 +00:00
store(c, srcSize, src, dst->base, dst->offset, dst->index, dst->scale, true);
}
void
2009-02-24 02:16:42 +00:00
moveAndUpdateRM(Context* c, unsigned srcSize, Assembler::Register* src,
unsigned dstSize UNUSED, Assembler::Memory* dst)
{
2009-02-24 02:16:42 +00:00
assert(c, srcSize == BytesPerWord);
assert(c, dstSize == BytesPerWord);
assert(c, dst->index == NoRegister);
2009-02-24 02:16:42 +00:00
issue(c, stwu(src->low, dst->base, dst->offset));
}
void
2009-02-24 02:16:42 +00:00
load(Context* c, unsigned srcSize, int base, int offset, int index,
unsigned scale, unsigned dstSize, Assembler::Register* dst,
bool preserveIndex)
{
2009-02-24 02:16:42 +00:00
if (index != NoRegister) {
int normalized = normalize(c, offset, index, scale, &preserveIndex);
2009-02-24 02:16:42 +00:00
switch (srcSize) {
case 1:
issue(c, lbzx(dst->low, base, normalized));
moveRR(c, 1, dst, BytesPerWord, dst);
break;
2009-02-24 02:16:42 +00:00
case 2:
issue(c, lhax(dst->low, base, normalized));
break;
2009-02-24 02:16:42 +00:00
case 4:
case 8: {
if (srcSize == 4 and dstSize == 8) {
load(c, 4, base, 0, normalized, 1, 4, dst, preserveIndex);
moveRR(c, 4, dst, 8, dst);
} else if (srcSize == 8 and dstSize == 8) {
Assembler::Register dstHigh(dst->high);
load(c, 4, base, 0, normalized, 1, 4, &dstHigh, preserveIndex);
load(c, 4, base, 4, normalized, 1, 4, dst, preserveIndex);
} else {
issue(c, lwzx(dst->low, base, offset));
}
} break;
2009-02-24 02:16:42 +00:00
default: abort(c);
}
} else {
2009-02-24 02:16:42 +00:00
switch (srcSize) {
case 1:
issue(c, lbz(dst->low, base, offset));
issue(c, extsb(dst->low, dst->low));
break;
2009-02-24 02:16:42 +00:00
case 2:
issue(c, lha(dst->low, base, offset));
break;
2009-02-24 02:16:42 +00:00
case 4:
issue(c, lwz(dst->low, base, offset));
break;
2009-02-24 02:16:42 +00:00
case 8: {
if (srcSize == 4 and dstSize == 8) {
load(c, 4, base, offset, NoRegister, 1, 4, dst, false);
moveRR(c, 4, dst, 8, dst);
} else if (srcSize == 8 and dstSize == 8) {
Assembler::Register dstHigh(dst->high);
load(c, 4, base, offset, NoRegister, 1, 4, &dstHigh, false);
load(c, 4, base, offset + 4, NoRegister, 1, 4, dst, false);
} else {
issue(c, lwzx(dst->low, base, offset));
}
} break;
2009-02-24 02:16:42 +00:00
default: abort(c);
}
}
}
2009-02-24 02:16:42 +00:00
void
moveMR(Context* c, unsigned srcSize, Assembler::Memory* src,
unsigned dstSize, Assembler::Register* dst)
{
load(c, srcSize, src->base, src->offset, src->index, src->scale,
dstSize, dst, true);
}
2009-02-24 02:16:42 +00:00
void
loadLinkRegisterR(Context* c, unsigned dstSize, Assembler::Register* dst)
{
assert(c, dstSize == BytesPerWord);
2009-02-24 02:16:42 +00:00
issue(c, mflr(dst->low));
}
2009-02-24 02:16:42 +00:00
void
storeLinkRegisterR(Context* c, unsigned srcSize, Assembler::Register* src)
{
assert(c, srcSize == BytesPerWord);
2009-02-24 02:16:42 +00:00
issue(c, mtlr(src->low));
}
2009-02-24 02:16:42 +00:00
void addRRR(Context* con, unsigned size, Reg* a, Reg* b, Reg* t) {
if(size == 8) {
issue(con, addc(R(t), R(a), R(b)));
issue(con, adde(H(t), H(a), H(b)));
} else {
issue(con, add(R(t), R(a), R(b)));
}
}
2009-02-24 02:16:42 +00:00
void addCRR(Context* con, unsigned size, Const* a, Reg* b, Reg* t) {
assert(con, size == BytesPerWord);
2009-02-24 02:16:42 +00:00
int64_t i = getVal(a);
if(i) {
issue(con, addi(R(t), R(b), lo16(i)));
issue(con, addi(R(t), R(t), hi16(i)));
}
}
2009-02-24 02:16:42 +00:00
void subRRR(Context* con, unsigned size, Reg* a, Reg* b, Reg* t) {
if(size == 8) {
issue(con, subfc(R(t), R(a), R(b)));
issue(con, subfe(H(t), H(a), H(b)));
} else {
issue(con, subf(R(t), R(a), R(b)));
}
}
2009-02-24 02:16:42 +00:00
void subCRR(Context* con, unsigned size, Const* a, Reg* b, Reg* t) {
assert(con, size == BytesPerWord);
2009-02-24 02:16:42 +00:00
int64_t i = getVal(a);
if(i) {
issue(con, subi(R(t), R(b), lo16(i)));
issue(con, subi(R(t), R(t), hi16(i)));
}
}
2009-02-24 02:16:42 +00:00
// END OPERATION COMPILERS
2009-02-24 02:16:42 +00:00
void
populateTables(ArchitectureContext* /*c*/)
{
// const OperandType C = ConstantOperand;
// const OperandType A = AddressOperand;
// const OperandType R = RegisterOperand;
// const OperandType M = MemoryOperand;
2009-02-24 02:16:42 +00:00
// OperationType* zo = c->operations;
// UnaryOperationType* uo = c->unaryOperations;
// BinaryOperationType* bo = c->binaryOperations;
// TernaryOperationType* to = c->ternaryOperations;
2009-02-24 02:16:42 +00:00
}
2009-02-24 02:16:42 +00:00
class MyArchitecture: public Assembler::Architecture {
public:
MyArchitecture(System* system): c(system), referenceCount(0) {
populateTables(&c);
}
2009-02-24 02:16:42 +00:00
virtual unsigned registerCount() {
return 32;
}
2009-02-24 02:16:42 +00:00
virtual int stack() {
return StackRegister;
}
2009-02-24 02:16:42 +00:00
virtual int thread() {
return ThreadRegister;
}
2009-02-24 02:16:42 +00:00
virtual int returnLow(unsigned size) {
return (size > BytesPerWord ? 4 : 3);
}
2009-02-24 02:16:42 +00:00
virtual int returnHigh() {
return (BytesPerWord == 4 ? 3 : NoRegister);
}
2009-02-24 02:16:42 +00:00
virtual bool condensedAddressing() {
return false;
}
2009-02-24 02:16:42 +00:00
virtual bool reserved(int register_) {
switch (register_) {
case StackRegister:
case ThreadRegister:
return true;
2009-02-24 02:16:42 +00:00
default:
return false;
}
}
2009-02-24 02:16:42 +00:00
virtual unsigned argumentRegisterCount() {
return 8;
}
2009-02-24 02:16:42 +00:00
virtual int argumentRegister(unsigned index) {
assert(&c, index < argumentRegisterCount());
2009-02-24 02:16:42 +00:00
return index + 3;
}
2009-02-24 02:16:42 +00:00
virtual void updateCall(UnaryOperation op UNUSED,
bool assertAlignment UNUSED, void* /*returnAddress*/,
void* /*newTarget*/)
{
// todo
abort(&c);
}
2009-02-24 02:16:42 +00:00
virtual unsigned alignFrameSize(unsigned sizeInWords) {
const unsigned alignment = 16 / BytesPerWord;
return (ceiling(sizeInWords + FrameFooterSize, alignment) * alignment)
- FrameFooterSize;
}
2009-02-24 02:16:42 +00:00
virtual void* frameIp(void* stack) {
return stack ? *static_cast<void**>(stack) : 0;
}
2009-02-24 02:16:42 +00:00
virtual unsigned frameHeaderSize() {
return 0;
}
2009-02-24 02:16:42 +00:00
virtual unsigned frameReturnAddressSize() {
return 1;
}
2009-02-24 02:16:42 +00:00
virtual unsigned frameFooterSize() {
return FrameFooterSize;
}
2009-02-24 02:16:42 +00:00
virtual void nextFrame(void** stack, void**) {
*stack = static_cast<void**>(*stack);
}
2009-02-24 02:16:42 +00:00
virtual void plan
(UnaryOperation,
unsigned, uint8_t* aTypeMask, uint64_t* aRegisterMask,
bool* thunk)
{
2009-02-24 02:16:42 +00:00
*aTypeMask = (1 << RegisterOperand);
*aRegisterMask = ~static_cast<uint64_t>(0);
*thunk = false;
}
2009-02-24 02:16:42 +00:00
virtual void plan
(BinaryOperation op,
unsigned, uint8_t* aTypeMask, uint64_t* aRegisterMask,
unsigned, uint8_t* bTypeMask, uint64_t* bRegisterMask,
bool* thunk)
{
*aTypeMask = ~0;
*aRegisterMask = ~static_cast<uint64_t>(0);
*bTypeMask = (1 << RegisterOperand) | (1 << MemoryOperand);
*bRegisterMask = ~static_cast<uint64_t>(0);
*thunk = false;
switch (op) {
case Compare:
2009-02-24 02:16:42 +00:00
*aTypeMask = (1 << RegisterOperand) | (1 << ConstantOperand);
*bTypeMask = (1 << RegisterOperand);
break;
2009-02-24 02:16:42 +00:00
case Negate:
*aTypeMask = (1 << RegisterOperand);
*bTypeMask = (1 << RegisterOperand);
break;
2009-02-24 02:16:42 +00:00
default:
break;
2009-02-24 02:16:42 +00:00
}
}
virtual void plan
(TernaryOperation op,
unsigned aSize, uint8_t* aTypeMask, uint64_t* aRegisterMask,
unsigned, uint8_t* bTypeMask, uint64_t* bRegisterMask,
unsigned, uint8_t* cTypeMask, uint64_t* cRegisterMask,
bool* thunk)
{
*aTypeMask = (1 << RegisterOperand) | (1 << ConstantOperand);
*aRegisterMask = ~static_cast<uint64_t>(0);
*bTypeMask = (1 << RegisterOperand);
*bRegisterMask = ~static_cast<uint64_t>(0);
*thunk = false;
2009-02-24 02:16:42 +00:00
switch (op) {
case Add:
case Subtract:
if (BytesPerWord == 4 and aSize == 8) {
aTypeMask = bTypeMask = (1 << RegisterOperand);
}
break;
case Divide:
2009-02-24 02:16:42 +00:00
if (BytesPerWord == 4 and aSize == 8) {
*bTypeMask = ~0;
*thunk = true;
}
break;
case Remainder:
2009-02-24 02:16:42 +00:00
if (BytesPerWord == 4 and aSize == 8) {
*bTypeMask = ~0;
*thunk = true;
}
break;
default:
break;
}
2009-02-24 02:16:42 +00:00
*cTypeMask = *bTypeMask;
*cRegisterMask = *bRegisterMask;
}
virtual void acquire() {
++ referenceCount;
}
virtual void release() {
if (-- referenceCount == 0) {
c.s->free(this);
}
}
ArchitectureContext c;
unsigned referenceCount;
};
class MyAssembler: public Assembler {
public:
MyAssembler(System* s, Allocator* a, Zone* zone, MyArchitecture* arch):
c(s, a, zone), arch_(arch)
{ }
virtual void setClient(Client* client) {
assert(&c, c.client == 0);
c.client = client;
}
virtual Architecture* arch() {
return arch_;
}
virtual void saveFrame(unsigned stackOffset, unsigned) {
Register stack(StackRegister);
Memory stackDst(ThreadRegister, stackOffset);
moveRM(&c, BytesPerWord, &stack, BytesPerWord, &stackDst);
}
virtual void pushFrame(unsigned argumentCount, ...) {
struct {
unsigned size;
OperandType type;
Operand* operand;
} arguments[argumentCount];
va_list a; va_start(a, argumentCount);
unsigned footprint = 0;
for (unsigned i = 0; i < argumentCount; ++i) {
arguments[i].size = va_arg(a, unsigned);
arguments[i].type = static_cast<OperandType>(va_arg(a, int));
arguments[i].operand = va_arg(a, Operand*);
footprint += ceiling(arguments[i].size, BytesPerWord);
}
va_end(a);
allocateFrame(arch_->alignFrameSize(footprint));
unsigned offset = 0;
for (unsigned i = 0; i < argumentCount; ++i) {
if (i < arch_->argumentRegisterCount()) {
Register dst(arch_->argumentRegister(i));
apply(Move,
arguments[i].size, arguments[i].type, arguments[i].operand,
pad(arguments[i].size), RegisterOperand, &dst);
offset += ceiling(arguments[i].size, BytesPerWord);
} else {
Memory dst(ThreadRegister, (offset + FrameFooterSize) * BytesPerWord);
apply(Move,
arguments[i].size, arguments[i].type, arguments[i].operand,
pad(arguments[i].size), MemoryOperand, &dst);
offset += ceiling(arguments[i].size, BytesPerWord);
}
}
}
virtual void allocateFrame(unsigned footprint) {
Register returnAddress(0);
loadLinkRegisterR(&c, BytesPerWord, &returnAddress);
Memory returnAddressDst(StackRegister, 8);
moveRM(&c, BytesPerWord, &returnAddress, BytesPerWord, &returnAddressDst);
Register stack(StackRegister);
Memory stackDst(StackRegister, -footprint * BytesPerWord);
moveAndUpdateRM(&c, BytesPerWord, &stack, BytesPerWord, &stackDst);
}
virtual void popFrame() {
Register stack(StackRegister);
Memory stackSrc(StackRegister, 0);
moveMR(&c, BytesPerWord, &stackSrc, BytesPerWord, &stack);
Register returnAddress(0);
Memory returnAddressSrc(StackRegister, 8);
moveMR(&c, BytesPerWord, &returnAddressSrc, BytesPerWord, &returnAddress);
storeLinkRegisterR(&c, BytesPerWord, &returnAddress);
}
virtual void apply(Operation op) {
2009-02-24 02:16:42 +00:00
arch_->c.operations[op](&c);
}
2009-02-24 02:16:42 +00:00
virtual void apply(UnaryOperation op,
unsigned aSize, OperandType aType, Operand* aOperand)
{
2009-02-24 02:16:42 +00:00
arch_->c.unaryOperations[index(op, aType)](&c, aSize, aOperand);
}
2009-02-24 02:16:42 +00:00
virtual void apply(BinaryOperation op,
unsigned aSize, OperandType aType, Operand* aOperand,
unsigned bSize, OperandType bType, Operand* bOperand)
{
2009-02-24 02:16:42 +00:00
arch_->c.binaryOperations[index(op, aType, bType)]
(&c, aSize, aOperand, bSize, bOperand);
}
virtual void apply(TernaryOperation op,
unsigned aSize, OperandType aType, Operand* aOperand,
unsigned bSize, OperandType bType UNUSED,
Operand* bOperand,
unsigned cSize, OperandType cType UNUSED,
Operand* cOperand)
{
assert(&c, bSize == cSize);
assert(&c, bType == RegisterOperand);
assert(&c, cType == RegisterOperand);
arch_->c.ternaryOperations[index(op, aType)]
(&c, aSize, aOperand, bSize, bOperand, cSize, cOperand);
}
virtual void writeTo(uint8_t* dst) {
c.result = dst;
2009-02-24 02:16:42 +00:00
for (MyBlock* b = c.firstBlock; b; b = b->next) {
memcpy(dst + b->start, c.code.data + b->offset, b->size);
}
for (Task* t = c.tasks; t; t = t->next) {
t->run(&c);
}
}
2009-02-24 02:16:42 +00:00
virtual Promise* offset() {
return ::offset(&c);
}
2009-02-24 02:16:42 +00:00
virtual Block* endBlock(bool startNew) {
MyBlock* b = c.lastBlock;
b->size = c.code.length() - b->offset;
if (startNew) {
c.lastBlock = new (c.zone->allocate(sizeof(MyBlock)))
MyBlock(c.code.length());
} else {
c.lastBlock = 0;
}
return b;
}
2009-02-24 02:16:42 +00:00
virtual unsigned length() {
return c.code.length();
}
virtual void dispose() {
c.code.dispose();
}
Context c;
2009-02-24 02:16:42 +00:00
MyArchitecture* arch_;
};
} // namespace
namespace vm {
2009-02-24 02:16:42 +00:00
Assembler::Architecture*
makeArchitecture(System* system)
{
return new (allocate(system, sizeof(MyArchitecture))) MyArchitecture(system);
}
Assembler*
2009-02-24 02:16:42 +00:00
makeAssembler(System* system, Allocator* allocator, Zone* zone,
Assembler::Architecture* architecture)
{
return new (zone->allocate(sizeof(MyAssembler)))
2009-02-24 02:16:42 +00:00
MyAssembler(system, allocator, zone,
static_cast<MyArchitecture*>(architecture));
}
} // namespace vm