2014-04-20 20:14:48 -06:00
|
|
|
/* Copyright (c) 2008-2014, Avian Contributors
|
2013-02-11 08:07:46 -07: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.
|
|
|
|
|
|
|
|
There is NO WARRANTY for this software. See license.txt for
|
|
|
|
details. */
|
|
|
|
|
|
|
|
#ifndef AVIAN_CODEGEN_LIR_H
|
|
|
|
#define AVIAN_CODEGEN_LIR_H
|
|
|
|
|
|
|
|
namespace avian {
|
|
|
|
namespace codegen {
|
|
|
|
class Promise;
|
|
|
|
|
|
|
|
namespace lir {
|
|
|
|
enum Operation {
|
2014-07-11 09:50:18 -06:00
|
|
|
#define LIR_OP_0(x) x,
|
|
|
|
#define LIR_OP_1(x)
|
|
|
|
#define LIR_OP_2(x)
|
|
|
|
#define LIR_OP_3(x)
|
|
|
|
#include "lir-ops.inc.cpp"
|
|
|
|
#undef LIR_OP_0
|
|
|
|
#undef LIR_OP_1
|
|
|
|
#undef LIR_OP_2
|
|
|
|
#undef LIR_OP_3
|
2013-02-11 08:07:46 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
const unsigned OperationCount = Trap + 1;
|
|
|
|
|
|
|
|
enum UnaryOperation {
|
2014-07-11 09:50:18 -06:00
|
|
|
#define LIR_OP_0(x)
|
|
|
|
#define LIR_OP_1(x) x,
|
|
|
|
#define LIR_OP_2(x)
|
|
|
|
#define LIR_OP_3(x)
|
|
|
|
#include "lir-ops.inc.cpp"
|
|
|
|
#undef LIR_OP_0
|
|
|
|
#undef LIR_OP_1
|
|
|
|
#undef LIR_OP_2
|
|
|
|
#undef LIR_OP_3
|
2013-02-11 08:07:46 -07:00
|
|
|
NoUnaryOperation = -1
|
|
|
|
};
|
|
|
|
|
|
|
|
const unsigned UnaryOperationCount = AlignedJump + 1;
|
|
|
|
|
|
|
|
enum BinaryOperation {
|
2014-07-11 09:50:18 -06:00
|
|
|
#define LIR_OP_0(x)
|
|
|
|
#define LIR_OP_1(x)
|
|
|
|
#define LIR_OP_2(x) x,
|
|
|
|
#define LIR_OP_3(x)
|
|
|
|
#include "lir-ops.inc.cpp"
|
|
|
|
#undef LIR_OP_0
|
|
|
|
#undef LIR_OP_1
|
|
|
|
#undef LIR_OP_2
|
|
|
|
#undef LIR_OP_3
|
2013-02-11 08:07:46 -07:00
|
|
|
NoBinaryOperation = -1
|
|
|
|
};
|
|
|
|
|
|
|
|
const unsigned BinaryOperationCount = Absolute + 1;
|
|
|
|
|
|
|
|
enum TernaryOperation {
|
2014-07-11 09:50:18 -06:00
|
|
|
#define LIR_OP_0(x)
|
|
|
|
#define LIR_OP_1(x)
|
|
|
|
#define LIR_OP_2(x)
|
|
|
|
#define LIR_OP_3(x) x,
|
|
|
|
#include "lir-ops.inc.cpp"
|
|
|
|
#undef LIR_OP_0
|
|
|
|
#undef LIR_OP_1
|
|
|
|
#undef LIR_OP_2
|
|
|
|
#undef LIR_OP_3
|
2013-02-11 08:07:46 -07:00
|
|
|
NoTernaryOperation = -1
|
|
|
|
};
|
|
|
|
|
2014-07-11 09:50:18 -06:00
|
|
|
const unsigned TernaryOperationCount = JumpIfFloatGreaterOrEqualOrUnordered + 1;
|
2013-02-11 08:07:46 -07:00
|
|
|
|
|
|
|
const unsigned NonBranchTernaryOperationCount = FloatMin + 1;
|
2014-07-11 09:50:18 -06:00
|
|
|
const unsigned BranchOperationCount = JumpIfFloatGreaterOrEqualOrUnordered
|
|
|
|
- FloatMin;
|
2013-02-11 08:07:46 -07:00
|
|
|
|
|
|
|
enum OperandType {
|
|
|
|
ConstantOperand,
|
|
|
|
AddressOperand,
|
|
|
|
RegisterOperand,
|
|
|
|
MemoryOperand
|
|
|
|
};
|
|
|
|
|
2014-07-11 09:50:18 -06:00
|
|
|
enum ValueType { ValueGeneral, ValueFloat };
|
2013-02-11 08:07:46 -07:00
|
|
|
|
|
|
|
const unsigned OperandTypeCount = MemoryOperand + 1;
|
|
|
|
|
|
|
|
const int NoRegister = -1;
|
|
|
|
|
2014-07-11 09:50:18 -06:00
|
|
|
inline bool isBranch(lir::TernaryOperation op)
|
|
|
|
{
|
2013-02-11 08:07:46 -07:00
|
|
|
return op > FloatMin;
|
|
|
|
}
|
|
|
|
|
2014-07-11 09:50:18 -06:00
|
|
|
inline bool isFloatBranch(lir::TernaryOperation op)
|
|
|
|
{
|
2013-02-11 08:07:46 -07:00
|
|
|
return op > JumpIfNotEqual;
|
|
|
|
}
|
|
|
|
|
2014-07-11 09:50:18 -06:00
|
|
|
inline bool isGeneralBranch(lir::TernaryOperation op)
|
|
|
|
{
|
2013-12-18 14:38:05 -07:00
|
|
|
return isBranch(op) && !isFloatBranch(op);
|
|
|
|
}
|
|
|
|
|
2014-07-11 09:50:18 -06:00
|
|
|
inline bool isGeneralBinaryOp(lir::TernaryOperation op)
|
|
|
|
{
|
2013-12-18 15:11:30 -07:00
|
|
|
return op < FloatAdd;
|
|
|
|
}
|
|
|
|
|
2014-07-11 09:50:18 -06:00
|
|
|
inline bool isFloatBinaryOp(lir::TernaryOperation op)
|
|
|
|
{
|
2013-12-18 15:22:38 -07:00
|
|
|
return op >= FloatAdd && op <= FloatMin;
|
2013-12-18 15:11:30 -07:00
|
|
|
}
|
|
|
|
|
2014-07-11 09:50:18 -06:00
|
|
|
inline bool isGeneralUnaryOp(lir::BinaryOperation op)
|
|
|
|
{
|
2013-12-18 16:43:15 -07:00
|
|
|
return op == Negate || op == Absolute;
|
|
|
|
}
|
|
|
|
|
2014-07-11 09:50:18 -06:00
|
|
|
inline bool isFloatUnaryOp(lir::BinaryOperation op)
|
|
|
|
{
|
2013-12-18 16:43:15 -07:00
|
|
|
return op == FloatNegate || op == FloatSquareRoot || op == FloatAbsolute;
|
|
|
|
}
|
|
|
|
|
2014-07-11 09:50:18 -06:00
|
|
|
class Operand {
|
|
|
|
};
|
2013-02-11 08:07:46 -07:00
|
|
|
|
2014-07-11 09:50:18 -06:00
|
|
|
class Constant : public Operand {
|
2013-02-11 08:07:46 -07:00
|
|
|
public:
|
2014-07-11 09:50:18 -06:00
|
|
|
Constant(Promise* value) : value(value)
|
|
|
|
{
|
|
|
|
}
|
2013-02-11 08:07:46 -07:00
|
|
|
|
|
|
|
Promise* value;
|
|
|
|
};
|
|
|
|
|
2014-07-11 09:50:18 -06:00
|
|
|
class Address : public Operand {
|
2013-02-11 08:07:46 -07:00
|
|
|
public:
|
2014-07-11 09:50:18 -06:00
|
|
|
Address(Promise* address) : address(address)
|
|
|
|
{
|
|
|
|
}
|
2013-02-11 08:07:46 -07:00
|
|
|
|
|
|
|
Promise* address;
|
|
|
|
};
|
|
|
|
|
2014-07-11 09:50:18 -06:00
|
|
|
class Register : public Operand {
|
2013-02-11 08:07:46 -07:00
|
|
|
public:
|
2014-07-11 09:50:18 -06:00
|
|
|
Register(int low, int high = NoRegister) : low(low), high(high)
|
|
|
|
{
|
|
|
|
}
|
2013-02-11 08:07:46 -07:00
|
|
|
|
|
|
|
int low;
|
|
|
|
int high;
|
|
|
|
};
|
|
|
|
|
2014-07-11 09:50:18 -06:00
|
|
|
class Memory : public Operand {
|
2013-02-11 08:07:46 -07:00
|
|
|
public:
|
2014-07-11 09:50:18 -06:00
|
|
|
Memory(int base, int offset, int index = NoRegister, unsigned scale = 1)
|
|
|
|
: base(base), offset(offset), index(index), scale(scale)
|
|
|
|
{
|
|
|
|
}
|
2013-02-11 08:07:46 -07:00
|
|
|
|
|
|
|
int base;
|
|
|
|
int offset;
|
|
|
|
int index;
|
|
|
|
unsigned scale;
|
|
|
|
};
|
|
|
|
|
2014-07-11 09:50:18 -06:00
|
|
|
} // namespace lir
|
|
|
|
} // namespace codegen
|
|
|
|
} // namespace avian
|
2013-02-11 08:07:46 -07:00
|
|
|
|
2014-07-11 09:50:18 -06:00
|
|
|
#endif // AVIAN_CODEGEN_LIR_H
|