mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2024-12-14 11:32:34 +00:00
move IR-level optimization flags into their own struct
This is a preliminary step to using the IR-level floating-point fast-math-flags in the SDAG (D8900). In this patch, we introduce the optimization flags as their own struct. As noted in the TODO comment, we should eventually share this data between the IR passes and the backend. We also switch the existing nsw / nuw / exact bit functionality of the BinaryWithFlagsSDNode class to use the new struct. The tradeoff is that instead of using the free but limited space of SDNode's SubclassData, we add a data member to the subclass. This means we don't have to repeat all of the get/set methods per flag, but we're potentially adding size to all nodes of this subclassi type. In practice on 64-bit systems (measured on Linux and MacOS X), there is no size difference between an SDNode and BinaryWithFlagsSDNode after this change: they're both 80 bytes. This means that we had at least one free byte to play with due to struct alignment. Differential Revision: http://reviews.llvm.org/D9325 git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@235997 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
a34b8b2e55
commit
2a7841dd4d
@ -926,6 +926,62 @@ inline void SDUse::setNode(SDNode *N) {
|
||||
if (N) N->addUse(*this);
|
||||
}
|
||||
|
||||
/// These are IR-level optimization flags that may be propagated to SDNodes.
|
||||
/// TODO: This data structure should be shared by the IR optimizer and the
|
||||
/// the backend.
|
||||
struct SDNodeFlags {
|
||||
private:
|
||||
bool NoUnsignedWrap : 1;
|
||||
bool NoSignedWrap : 1;
|
||||
bool Exact : 1;
|
||||
bool UnsafeAlgebra : 1;
|
||||
bool NoNaNs : 1;
|
||||
bool NoInfs : 1;
|
||||
bool NoSignedZeros : 1;
|
||||
bool AllowReciprocal : 1;
|
||||
|
||||
public:
|
||||
/// Default constructor turns off all optimization flags.
|
||||
SDNodeFlags() {
|
||||
NoUnsignedWrap = false;
|
||||
NoSignedWrap = false;
|
||||
Exact = false;
|
||||
UnsafeAlgebra = false;
|
||||
NoNaNs = false;
|
||||
NoInfs = false;
|
||||
NoSignedZeros = false;
|
||||
AllowReciprocal = false;
|
||||
}
|
||||
|
||||
// These are mutators for each flag.
|
||||
void setNoUnsignedWrap(bool b) { NoUnsignedWrap = b; }
|
||||
void setNoSignedWrap(bool b) { NoSignedWrap = b; }
|
||||
void setExact(bool b) { Exact = b; }
|
||||
void setUnsafeAlgebra(bool b) { UnsafeAlgebra = b; }
|
||||
void setNoNaNs(bool b) { NoNaNs = b; }
|
||||
void setNoInfs(bool b) { NoInfs = b; }
|
||||
void setNoSignedZeros(bool b) { NoSignedZeros = b; }
|
||||
void setAllowReciprocal(bool b) { AllowReciprocal = b; }
|
||||
|
||||
// These are accessors for each flag.
|
||||
bool hasNoUnsignedWrap() const { return NoUnsignedWrap; }
|
||||
bool hasNoSignedWrap() const { return NoSignedWrap; }
|
||||
bool hasExact() const { return Exact; }
|
||||
bool hasUnsafeAlgebra() const { return UnsafeAlgebra; }
|
||||
bool hasNoNaNs() const { return NoNaNs; }
|
||||
bool hasNoInfs() const { return NoInfs; }
|
||||
bool hasNoSignedZeros() const { return NoSignedZeros; }
|
||||
bool hasAllowReciprocal() const { return AllowReciprocal; }
|
||||
|
||||
/// Return a raw encoding of the flags.
|
||||
/// This function should only be used to add data to the NodeID value.
|
||||
unsigned getRawFlags() const {
|
||||
return (NoUnsignedWrap << 0) | (NoSignedWrap << 1) | (Exact << 2) |
|
||||
(UnsafeAlgebra << 3) | (NoNaNs << 4) | (NoInfs << 5) |
|
||||
(NoSignedZeros << 6) | (AllowReciprocal << 7);
|
||||
}
|
||||
};
|
||||
|
||||
/// This class is used for single-operand SDNodes. This is solely
|
||||
/// to allow co-allocation of node operands with the node itself.
|
||||
class UnarySDNode : public SDNode {
|
||||
@ -970,27 +1026,11 @@ static bool isBinOpWithFlags(unsigned Opcode) {
|
||||
/// This class is an extension of BinarySDNode
|
||||
/// used from those opcodes that have associated extra flags.
|
||||
class BinaryWithFlagsSDNode : public BinarySDNode {
|
||||
enum { NUW = (1 << 0), NSW = (1 << 1), EXACT = (1 << 2) };
|
||||
|
||||
public:
|
||||
SDNodeFlags Flags;
|
||||
BinaryWithFlagsSDNode(unsigned Opc, unsigned Order, DebugLoc dl, SDVTList VTs,
|
||||
SDValue X, SDValue Y)
|
||||
: BinarySDNode(Opc, Order, dl, VTs, X, Y) {}
|
||||
/// Return the SubclassData value, which contains an encoding of the flags.
|
||||
/// This function should be used to add subclass data to the NodeID value.
|
||||
unsigned getRawSubclassData() const { return SubclassData; }
|
||||
void setHasNoUnsignedWrap(bool b) {
|
||||
SubclassData = (SubclassData & ~NUW) | (b ? NUW : 0);
|
||||
}
|
||||
void setHasNoSignedWrap(bool b) {
|
||||
SubclassData = (SubclassData & ~NSW) | (b ? NSW : 0);
|
||||
}
|
||||
void setIsExact(bool b) {
|
||||
SubclassData = (SubclassData & ~EXACT) | (b ? EXACT : 0);
|
||||
}
|
||||
bool hasNoUnsignedWrap() const { return SubclassData & NUW; }
|
||||
bool hasNoSignedWrap() const { return SubclassData & NSW; }
|
||||
bool isExact() const { return SubclassData & EXACT; }
|
||||
: BinarySDNode(Opc, Order, dl, VTs, X, Y), Flags() { }
|
||||
static bool classof(const SDNode *N) {
|
||||
return isBinOpWithFlags(N->getOpcode());
|
||||
}
|
||||
|
@ -1450,9 +1450,10 @@ SDValue DAGCombiner::combine(SDNode *N) {
|
||||
SDNode *CSENode;
|
||||
if (const BinaryWithFlagsSDNode *BinNode =
|
||||
dyn_cast<BinaryWithFlagsSDNode>(N)) {
|
||||
CSENode = DAG.getNodeIfExists(
|
||||
N->getOpcode(), N->getVTList(), Ops, BinNode->hasNoUnsignedWrap(),
|
||||
BinNode->hasNoSignedWrap(), BinNode->isExact());
|
||||
CSENode = DAG.getNodeIfExists(N->getOpcode(), N->getVTList(), Ops,
|
||||
BinNode->Flags.hasNoUnsignedWrap(),
|
||||
BinNode->Flags.hasNoSignedWrap(),
|
||||
BinNode->Flags.hasExact());
|
||||
} else {
|
||||
CSENode = DAG.getNodeIfExists(N->getOpcode(), N->getVTList(), Ops);
|
||||
}
|
||||
|
@ -515,8 +515,10 @@ static void AddNodeIDCustom(FoldingSetNodeID &ID, const SDNode *N) {
|
||||
case ISD::SUB:
|
||||
case ISD::SHL: {
|
||||
const BinaryWithFlagsSDNode *BinNode = cast<BinaryWithFlagsSDNode>(N);
|
||||
AddBinaryNodeIDCustom(ID, N->getOpcode(), BinNode->hasNoUnsignedWrap(),
|
||||
BinNode->hasNoSignedWrap(), BinNode->isExact());
|
||||
AddBinaryNodeIDCustom(ID, N->getOpcode(),
|
||||
BinNode->Flags.hasNoUnsignedWrap(),
|
||||
BinNode->Flags.hasNoSignedWrap(),
|
||||
BinNode->Flags.hasExact());
|
||||
break;
|
||||
}
|
||||
case ISD::ATOMIC_CMP_SWAP:
|
||||
@ -963,9 +965,9 @@ BinarySDNode *SelectionDAG::GetBinarySDNode(unsigned Opcode, SDLoc DL,
|
||||
if (isBinOpWithFlags(Opcode)) {
|
||||
BinaryWithFlagsSDNode *FN = new (NodeAllocator) BinaryWithFlagsSDNode(
|
||||
Opcode, DL.getIROrder(), DL.getDebugLoc(), VTs, N1, N2);
|
||||
FN->setHasNoUnsignedWrap(nuw);
|
||||
FN->setHasNoSignedWrap(nsw);
|
||||
FN->setIsExact(exact);
|
||||
FN->Flags.setNoUnsignedWrap(nuw);
|
||||
FN->Flags.setNoSignedWrap(nsw);
|
||||
FN->Flags.setExact(exact);
|
||||
|
||||
return FN;
|
||||
}
|
||||
|
@ -12560,7 +12560,7 @@ SDValue X86TargetLowering::EmitTest(SDValue Op, unsigned X86CC, SDLoc dl,
|
||||
case ISD::SHL: {
|
||||
const BinaryWithFlagsSDNode *BinNode =
|
||||
cast<BinaryWithFlagsSDNode>(Op.getNode());
|
||||
if (BinNode->hasNoSignedWrap())
|
||||
if (BinNode->Flags.hasNoSignedWrap())
|
||||
break;
|
||||
}
|
||||
default:
|
||||
|
Loading…
Reference in New Issue
Block a user