Skip to content

Commit b32f365

Browse files
committed
[SystemZ] Do not use glue to represent condition code dependencies
Currently, an instruction setting the condition code is linked to the instruction using the condition code via a "glue" link in the SelectionDAG. This has a number of drawbacks; in particular, it means the same CC cannot be used by multiple users. It also makes it more difficult to efficiently implement SADDO et. al. This patch changes the back-end to represent CC dependencies as normal values during SelectionDAG matching, along the lines of how this is handled in the X86 back-end already. In addition to the core mechanics of updating all relevant patterns, this requires a number of additional changes: - We now need to be able to spill/restore a CC value into a GPR if necessary. This means providing a copyPhysReg implementation for moves involving CC, and defining getCrossCopyRegClass. - Since we still prefer to avoid such spills, we provide an override for IsProfitableToFold to avoid creating a merged LOAD / ICMP if this would result in multiple users of the CC. - combineCCMask no longer requires a single CC user, and no longer need to be careful about preventing invalid glue/chain cycles. - emitSelect needs to be more careful in marking CC live-in to the basic block it generates. Also, we can now optimize the case of multiple subsequent selects with the same condition just like X86 does. llvm-svn: 331202
1 parent 2de9d4a commit b32f365

15 files changed

+540
-249
lines changed

llvm/lib/Target/SystemZ/SystemZISelDAGToDAG.cpp

+49-2
Original file line numberDiff line numberDiff line change
@@ -351,6 +351,7 @@ class SystemZDAGToDAGISel : public SelectionDAGISel {
351351
void Select(SDNode *Node) override;
352352
bool SelectInlineAsmMemoryOperand(const SDValue &Op, unsigned ConstraintID,
353353
std::vector<SDValue> &OutOps) override;
354+
bool IsProfitableToFold(SDValue N, SDNode *U, SDNode *Root) const override;
354355
void PreprocessISelDAG() override;
355356

356357
// Include the pieces autogenerated from the target description.
@@ -1445,6 +1446,52 @@ SelectInlineAsmMemoryOperand(const SDValue &Op,
14451446
return true;
14461447
}
14471448

1449+
// IsProfitableToFold - Returns true if is profitable to fold the specific
1450+
// operand node N of U during instruction selection that starts at Root.
1451+
bool
1452+
SystemZDAGToDAGISel::IsProfitableToFold(SDValue N, SDNode *U,
1453+
SDNode *Root) const {
1454+
// We want to avoid folding a LOAD into an ICMP node if as a result
1455+
// we would be forced to spill the condition code into a GPR.
1456+
if (N.getOpcode() == ISD::LOAD && U->getOpcode() == SystemZISD::ICMP) {
1457+
if (!N.hasOneUse() || !U->hasOneUse())
1458+
return false;
1459+
1460+
// The user of the CC value will usually be a CopyToReg into the
1461+
// physical CC register, which in turn is glued and chained to the
1462+
// actual instruction that uses the CC value. Bail out if we have
1463+
// anything else than that.
1464+
SDNode *CCUser = *U->use_begin();
1465+
SDNode *CCRegUser = nullptr;
1466+
if (CCUser->getOpcode() == ISD::CopyToReg ||
1467+
cast<RegisterSDNode>(CCUser->getOperand(1))->getReg() == SystemZ::CC) {
1468+
for (auto *U : CCUser->uses()) {
1469+
if (CCRegUser == nullptr)
1470+
CCRegUser = U;
1471+
else if (CCRegUser != U)
1472+
return false;
1473+
}
1474+
}
1475+
if (CCRegUser == nullptr)
1476+
return false;
1477+
1478+
// If the actual instruction is a branch, the only thing that remains to be
1479+
// checked is whether the CCUser chain is a predecessor of the load.
1480+
if (CCRegUser->isMachineOpcode() &&
1481+
CCRegUser->getMachineOpcode() == SystemZ::BRC)
1482+
return !N->isPredecessorOf(CCUser->getOperand(0).getNode());
1483+
1484+
// Otherwise, the instruction may have multiple operands, and we need to
1485+
// verify that none of them are a predecessor of the load. This is exactly
1486+
// the same check that would be done by common code if the CC setter were
1487+
// glued to the CC user, so simply invoke that check here.
1488+
if (!IsLegalToFold(N, U, CCRegUser, OptLevel, false))
1489+
return false;
1490+
}
1491+
1492+
return true;
1493+
}
1494+
14481495
namespace {
14491496
// Represents a sequence for extracting a 0/1 value from an IPM result:
14501497
// (((X ^ XORValue) + AddValue) >> Bit)
@@ -1543,9 +1590,9 @@ SDValue SystemZDAGToDAGISel::expandSelectBoolean(SDNode *Node) {
15431590
int CCMask = CCMaskOp->getZExtValue();
15441591

15451592
SDLoc DL(Node);
1546-
SDValue Glue = Node->getOperand(4);
1593+
SDValue CCReg = Node->getOperand(4);
15471594
IPMConversion IPM = getIPMConversion(CCValid, CCMask);
1548-
SDValue Result = CurDAG->getNode(SystemZISD::IPM, DL, MVT::i32, Glue);
1595+
SDValue Result = CurDAG->getNode(SystemZISD::IPM, DL, MVT::i32, CCReg);
15491596

15501597
if (IPM.XORValue)
15511598
Result = CurDAG->getNode(ISD::XOR, DL, MVT::i32, Result,

0 commit comments

Comments
 (0)