Skip to content

Commit 13bdae8

Browse files
committed
Revert r372285 "GlobalISel: Don't materialize immarg arguments to intrinsics"
This broke the Chromium build, causing it to fail with e.g. fatal error: error in backend: Cannot select: t362: v4i32 = X86ISD::VSHLI t392, Constant:i8<15> See llvm-commits thread of r372285 for details. This also reverts r372286, r372287, r372288, r372289, r372290, r372291, r372292, r372293, r372296, and r372297, which seemed to depend on the main commit. > Encode them directly as an imm argument to G_INTRINSIC*. > > Since now intrinsics can now define what parameters are required to be > immediates, avoid using registers for them. Intrinsics could > potentially want a constant that isn't a legal register type. Also, > since G_CONSTANT is subject to CSE and legalization, transforms could > potentially obscure the value (and create extra work for the > selector). The register bank of a G_CONSTANT is also meaningful, so > this could throw off future folding and legalization logic for AMDGPU. > > This will be much more convenient to work with than needing to call > getConstantVRegVal and checking if it may have failed for every > constant intrinsic parameter. AMDGPU has quite a lot of intrinsics wth > immarg operands, many of which need inspection during lowering. Having > to find the value in a register is going to add a lot of boilerplate > and waste compile time. > > SelectionDAG has always provided TargetConstant for constants which > should not be legalized or materialized in a register. The distinction > between Constant and TargetConstant was somewhat fuzzy, and there was > no automatic way to force usage of TargetConstant for certain > intrinsic parameters. They were both ultimately ConstantSDNode, and it > was inconsistently used. It was quite easy to mis-select an > instruction requiring an immediate. For SelectionDAG, start emitting > TargetConstant for these arguments, and using timm to match them. > > Most of the work here is to cleanup target handling of constants. Some > targets process intrinsics through intermediate custom nodes, which > need to preserve TargetConstant usage to match the intrinsic > expectation. Pattern inputs now need to distinguish whether a constant > is merely compatible with an operand or whether it is mandatory. > > The GlobalISelEmitter needs to treat timm as a special case of a leaf > node, simlar to MachineBasicBlock operands. This should also enable > handling of patterns for some G_* instructions with immediates, like > G_FENCE or G_EXTRACT. > > This does include a workaround for a crash in GlobalISelEmitter when > ARM tries to uses "imm" in an output with a "timm" pattern source. llvm-svn: 372314
1 parent 0cfb78e commit 13bdae8

File tree

79 files changed

+1364
-5009
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

79 files changed

+1364
-5009
lines changed

llvm/include/llvm/CodeGen/GlobalISel/InstructionSelector.h

-5
Original file line numberDiff line numberDiff line change
@@ -220,11 +220,6 @@ enum {
220220
/// - OpIdx - Operand index
221221
GIM_CheckIsMBB,
222222

223-
/// Check the specified operand is an Imm
224-
/// - InsnID - Instruction ID
225-
/// - OpIdx - Operand index
226-
GIM_CheckIsImm,
227-
228223
/// Check if the specified operand is safe to fold into the current
229224
/// instruction.
230225
/// - InsnID - Instruction ID

llvm/include/llvm/CodeGen/GlobalISel/InstructionSelectorImpl.h

+1-13
Original file line numberDiff line numberDiff line change
@@ -690,19 +690,7 @@ bool InstructionSelector::executeMatchTable(
690690
}
691691
break;
692692
}
693-
case GIM_CheckIsImm: {
694-
int64_t InsnID = MatchTable[CurrentIdx++];
695-
int64_t OpIdx = MatchTable[CurrentIdx++];
696-
DEBUG_WITH_TYPE(TgtInstructionSelector::getName(),
697-
dbgs() << CurrentIdx << ": GIM_CheckIsImm(MIs[" << InsnID
698-
<< "]->getOperand(" << OpIdx << "))\n");
699-
assert(State.MIs[InsnID] != nullptr && "Used insn before defined");
700-
if (!State.MIs[InsnID]->getOperand(OpIdx).isImm()) {
701-
if (handleReject() == RejectAndGiveUp)
702-
return false;
703-
}
704-
break;
705-
}
693+
706694
case GIM_CheckIsSafeToFold: {
707695
int64_t InsnID = MatchTable[CurrentIdx++];
708696
DEBUG_WITH_TYPE(TgtInstructionSelector::getName(),

llvm/include/llvm/CodeGen/ScheduleDAGInstrs.h

-3
Original file line numberDiff line numberDiff line change
@@ -374,9 +374,6 @@ namespace llvm {
374374
/// Returns a mask for which lanes get read/written by the given (register)
375375
/// machine operand.
376376
LaneBitmask getLaneMaskForMO(const MachineOperand &MO) const;
377-
378-
/// Returns true if the def register in \p MO has no uses.
379-
bool deadDefHasNoUse(const MachineOperand &MO);
380377
};
381378

382379
/// Creates a new SUnit and return a ptr to it.

llvm/include/llvm/Target/TargetSelectionDAG.td

-5
Original file line numberDiff line numberDiff line change
@@ -848,11 +848,6 @@ class ImmLeaf<ValueType vt, code pred, SDNodeXForm xform = NOOP_SDNodeXForm,
848848
bit IsAPFloat = 0;
849849
}
850850

851-
// Convenience wrapper for ImmLeaf to use timm/TargetConstant instead
852-
// of imm/Constant.
853-
class TImmLeaf<ValueType vt, code pred, SDNodeXForm xform = NOOP_SDNodeXForm,
854-
SDNode ImmNode = timm> : ImmLeaf<vt, pred, xform, ImmNode>;
855-
856851
// An ImmLeaf except that Imm is an APInt. This is useful when you need to
857852
// zero-extend the immediate instead of sign-extend it.
858853
//

llvm/lib/CodeGen/GlobalISel/IRTranslator.cpp

+6-21
Original file line numberDiff line numberDiff line change
@@ -1617,29 +1617,14 @@ bool IRTranslator::translateCall(const User &U, MachineIRBuilder &MIRBuilder) {
16171617
if (isa<FPMathOperator>(CI))
16181618
MIB->copyIRFlags(CI);
16191619

1620-
for (auto &Arg : enumerate(CI.arg_operands())) {
1620+
for (auto &Arg : CI.arg_operands()) {
16211621
// Some intrinsics take metadata parameters. Reject them.
1622-
if (isa<MetadataAsValue>(Arg.value()))
1622+
if (isa<MetadataAsValue>(Arg))
16231623
return false;
1624-
1625-
// If this is required to be an immediate, don't materialize it in a
1626-
// register.
1627-
if (CI.paramHasAttr(Arg.index(), Attribute::ImmArg)) {
1628-
if (ConstantInt *CI = dyn_cast<ConstantInt>(Arg.value())) {
1629-
// imm arguments are more convenient than cimm (and realistically
1630-
// probably sufficient), so use them.
1631-
assert(CI->getBitWidth() <= 64 &&
1632-
"large intrinsic immediates not handled");
1633-
MIB.addImm(CI->getSExtValue());
1634-
} else {
1635-
MIB.addFPImm(cast<ConstantFP>(Arg.value()));
1636-
}
1637-
} else {
1638-
ArrayRef<Register> VRegs = getOrCreateVRegs(*Arg.value());
1639-
if (VRegs.size() > 1)
1640-
return false;
1641-
MIB.addUse(VRegs[0]);
1642-
}
1624+
ArrayRef<Register> VRegs = getOrCreateVRegs(*Arg);
1625+
if (VRegs.size() > 1)
1626+
return false;
1627+
MIB.addUse(VRegs[0]);
16431628
}
16441629

16451630
// Add a MachineMemOperand if it is a target mem intrinsic.

llvm/lib/CodeGen/ScheduleDAGInstrs.cpp

+2-8
Original file line numberDiff line numberDiff line change
@@ -373,13 +373,6 @@ LaneBitmask ScheduleDAGInstrs::getLaneMaskForMO(const MachineOperand &MO) const
373373
return TRI->getSubRegIndexLaneMask(SubReg);
374374
}
375375

376-
bool ScheduleDAGInstrs::deadDefHasNoUse(const MachineOperand &MO) {
377-
auto RegUse = CurrentVRegUses.find(MO.getReg());
378-
if (RegUse == CurrentVRegUses.end())
379-
return true;
380-
return (RegUse->LaneMask & getLaneMaskForMO(MO)).none();
381-
}
382-
383376
/// Adds register output and data dependencies from this SUnit to instructions
384377
/// that occur later in the same scheduling region if they read from or write to
385378
/// the virtual register defined at OperIdx.
@@ -409,7 +402,8 @@ void ScheduleDAGInstrs::addVRegDefDeps(SUnit *SU, unsigned OperIdx) {
409402
}
410403

411404
if (MO.isDead()) {
412-
assert(deadDefHasNoUse(MO) && "Dead defs should have no uses");
405+
assert(CurrentVRegUses.find(Reg) == CurrentVRegUses.end() &&
406+
"Dead defs should have no uses");
413407
} else {
414408
// Add data dependence to all uses we found so far.
415409
const TargetSubtargetInfo &ST = MF.getSubtarget();

llvm/lib/CodeGen/SelectionDAG/SelectionDAGBuilder.cpp

+2-16
Original file line numberDiff line numberDiff line change
@@ -4768,22 +4768,8 @@ void SelectionDAGBuilder::visitTargetIntrinsic(const CallInst &I,
47684768

47694769
// Add all operands of the call to the operand list.
47704770
for (unsigned i = 0, e = I.getNumArgOperands(); i != e; ++i) {
4771-
const Value *Arg = I.getArgOperand(i);
4772-
if (!I.paramHasAttr(i, Attribute::ImmArg)) {
4773-
Ops.push_back(getValue(Arg));
4774-
continue;
4775-
}
4776-
4777-
// Use TargetConstant instead of a regular constant for immarg.
4778-
EVT VT = TLI.getValueType(*DL, Arg->getType(), true);
4779-
if (const ConstantInt *CI = dyn_cast<ConstantInt>(Arg)) {
4780-
assert(CI->getBitWidth() <= 64 &&
4781-
"large intrinsic immediates not handled");
4782-
Ops.push_back(DAG.getTargetConstant(*CI, SDLoc(), VT));
4783-
} else {
4784-
Ops.push_back(
4785-
DAG.getTargetConstantFP(*cast<ConstantFP>(Arg), SDLoc(), VT));
4786-
}
4771+
SDValue Op = getValue(I.getArgOperand(i));
4772+
Ops.push_back(Op);
47874773
}
47884774

47894775
SmallVector<EVT, 4> ValueVTs;

llvm/lib/Target/AArch64/AArch64InstrFormats.td

+2-2
Original file line numberDiff line numberDiff line change
@@ -685,11 +685,11 @@ def logical_imm64_not : Operand<i64> {
685685

686686
// iXX_imm0_65535 predicates - True if the immediate is in the range [0,65535].
687687
let ParserMatchClass = AsmImmRange<0, 65535>, PrintMethod = "printImmHex" in {
688-
def i32_imm0_65535 : Operand<i32>, TImmLeaf<i32, [{
688+
def i32_imm0_65535 : Operand<i32>, ImmLeaf<i32, [{
689689
return ((uint32_t)Imm) < 65536;
690690
}]>;
691691

692-
def i64_imm0_65535 : Operand<i64>, TImmLeaf<i64, [{
692+
def i64_imm0_65535 : Operand<i64>, ImmLeaf<i64, [{
693693
return ((uint64_t)Imm) < 65536;
694694
}]>;
695695
}

llvm/lib/Target/AArch64/AArch64InstrInfo.td

+1-1
Original file line numberDiff line numberDiff line change
@@ -798,7 +798,7 @@ def MOVbaseTLS : Pseudo<(outs GPR64:$dst), (ins),
798798
let Uses = [ X9 ], Defs = [ X16, X17, LR, NZCV ] in {
799799
def HWASAN_CHECK_MEMACCESS : Pseudo<
800800
(outs), (ins GPR64noip:$ptr, i32imm:$accessinfo),
801-
[(int_hwasan_check_memaccess X9, GPR64noip:$ptr, (i32 timm:$accessinfo))]>,
801+
[(int_hwasan_check_memaccess X9, GPR64noip:$ptr, (i32 imm:$accessinfo))]>,
802802
Sched<[]>;
803803
}
804804

0 commit comments

Comments
 (0)