Index: test/TableGen/GlobalISelEmitter.td =================================================================== --- test/TableGen/GlobalISelEmitter.td +++ test/TableGen/GlobalISelEmitter.td @@ -18,12 +18,55 @@ let Pattern = Pat; } +def complex : Operand, ComplexPattern { + let MIOperandInfo = (ops i32imm, i32imm); +} +def gi_complex : + GIComplexOperandMatcher, + GIComplexPatternEquiv; + //===- Test the function definition boilerplate. --------------------------===// // CHECK: bool MyTargetInstructionSelector::selectImpl(MachineInstr &I) const { // CHECK: MachineFunction &MF = *I.getParent()->getParent(); // CHECK: const MachineRegisterInfo &MRI = MF.getRegInfo(); +//===- Test a pattern with multiple ComplexPattern operands. --------------===// +// + +// CHECK-LABEL: if ([&]() { +// CHECK-NEXT: MachineInstr &MI0 = I; +// CHECK-NEXT: if (MI0.getNumOperands() < 4) +// CHECK-NEXT: return false; +// CHECK-NEXT: if ((MI0.getOpcode() == TargetOpcode::G_SELECT) && +// CHECK-NEXT: ((/* dst */ (MRI.getType(MI0.getOperand(0).getReg()) == (LLT::scalar(32))) && +// CHECK-NEXT: ((&RBI.getRegBankFromRegClass(MyTarget::GPR32RegClass) == RBI.getRegBank(MI0.getOperand(0).getReg(), MRI, TRI))))) && +// CHECK-NEXT: ((/* src1 */ (MRI.getType(MI0.getOperand(1).getReg()) == (LLT::scalar(32))) && +// CHECK-NEXT: ((&RBI.getRegBankFromRegClass(MyTarget::GPR32RegClass) == RBI.getRegBank(MI0.getOperand(1).getReg(), MRI, TRI))))) && +// CHECK-NEXT: ((/* src2 */ (MRI.getType(MI0.getOperand(2).getReg()) == (LLT::scalar(32))) && +// CHECK-NEXT: (selectComplexPattern(MI0.getOperand(2), TempOp0, TempOp1)))) && +// CHECK-NEXT: ((/* src3 */ (MRI.getType(MI0.getOperand(3).getReg()) == (LLT::scalar(32))) && +// CHECK-NEXT: (selectComplexPattern(MI0.getOperand(3), TempOp2, TempOp3))))) { +// CHECK-NEXT: // (select:i32 GPR32:i32:$src1, complex:i32:$src2, complex:i32:$src3) => (INSN2:i32 GPR32:i32:$src1, complex:i32:$src3, complex:i32:$src2) +// CHECK-NEXT: MachineInstrBuilder MIB = BuildMI(*I.getParent(), I, I.getDebugLoc(), TII.get(MyTarget::INSN2)); +// CHECK-NEXT: MIB.add(MI0.getOperand(0)/*dst*/); +// CHECK-NEXT: MIB.add(MI0.getOperand(1)/*src1*/); +// CHECK-NEXT: MIB.add(TempOp2); +// CHECK-NEXT: MIB.add(TempOp3); +// CHECK-NEXT: MIB.add(TempOp0); +// CHECK-NEXT: MIB.add(TempOp1); +// CHECK-NEXT: MIB.setMemRefs(I.memoperands_begin(), I.memoperands_end()); +// CHECK-NEXT: I.eraseFromParent(); +// CHECK-NEXT: MachineInstr &NewI = *MIB; +// CHECK-NEXT: constrainSelectedInstRegOperands(NewI, TII, TRI, RBI); +// CHECK-NEXT: return true; +// CHECK-NEXT: } + +def : GINodeEquiv; +def INSN2 : I<(outs GPR32:$dst), (ins GPR32:$src1, complex:$src2, complex:$src3), []>; +def : Pat<(select GPR32:$src1, complex:$src2, complex:$src3), + (INSN2 GPR32:$src1, complex:$src3, complex:$src2)>; + //===- Test a simple pattern with regclass operands. ----------------------===// // CHECK-LABEL: if ([&]() { @@ -160,6 +203,36 @@ def MUL : I<(outs GPR32:$dst), (ins GPR32:$src2, GPR32:$src1), [(set GPR32:$dst, (mul GPR32:$src1, GPR32:$src2))]>; +//===- Test a pattern with ComplexPattern operands. -----------------------===// +// + +// CHECK-LABEL: if ([&]() { +// CHECK-NEXT: MachineInstr &MI0 = I; +// CHECK-NEXT: if (MI0.getNumOperands() < 3) +// CHECK-NEXT: return false; +// CHECK-NEXT: if ((MI0.getOpcode() == TargetOpcode::G_SUB) && +// CHECK-NEXT: ((/* dst */ (MRI.getType(MI0.getOperand(0).getReg()) == (LLT::scalar(32))) && +// CHECK-NEXT: ((&RBI.getRegBankFromRegClass(MyTarget::GPR32RegClass) == RBI.getRegBank(MI0.getOperand(0).getReg(), MRI, TRI))))) && +// CHECK-NEXT: ((/* src1 */ (MRI.getType(MI0.getOperand(1).getReg()) == (LLT::scalar(32))) && +// CHECK-NEXT: ((&RBI.getRegBankFromRegClass(MyTarget::GPR32RegClass) == RBI.getRegBank(MI0.getOperand(1).getReg(), MRI, TRI))))) && +// CHECK-NEXT: ((/* src2 */ (MRI.getType(MI0.getOperand(2).getReg()) == (LLT::scalar(32))) && +// CHECK-NEXT: (selectComplexPattern(MI0.getOperand(2), TempOp0, TempOp1))))) { +// CHECK-NEXT: // (sub:i32 GPR32:i32:$src1, complex:i32:$src2) => (INSN1:i32 GPR32:i32:$src1, complex:i32:$src2) +// CHECK-NEXT: MachineInstrBuilder MIB = BuildMI(*I.getParent(), I, I.getDebugLoc(), TII.get(MyTarget::INSN1)); +// CHECK-NEXT: MIB.add(MI0.getOperand(0)/*dst*/); +// CHECK-NEXT: MIB.add(MI0.getOperand(1)/*src1*/); +// CHECK-NEXT: MIB.add(TempOp0); +// CHECK-NEXT: MIB.add(TempOp1); +// CHECK-NEXT: MIB.setMemRefs(I.memoperands_begin(), I.memoperands_end()); +// CHECK-NEXT: I.eraseFromParent(); +// CHECK-NEXT: MachineInstr &NewI = *MIB; +// CHECK-NEXT: constrainSelectedInstRegOperands(NewI, TII, TRI, RBI); +// CHECK-NEXT: return true; +// CHECK-NEXT: } + +def INSN1 : I<(outs GPR32:$dst), (ins GPR32:$src1, complex:$src2), []>; +def : Pat<(sub GPR32:$src1, complex:$src2), (INSN1 GPR32:$src1, complex:$src2)>; + //===- Test a simple pattern with constant immediate operands. ------------===// // // This must precede the 3-register variants because constant immediates have Index: utils/TableGen/GlobalISelEmitter.cpp =================================================================== --- utils/TableGen/GlobalISelEmitter.cpp +++ utils/TableGen/GlobalISelEmitter.cpp @@ -327,26 +327,31 @@ /// Generates code to check that an operand is a particular target constant. class ComplexPatternOperandMatcher : public OperandPredicateMatcher { protected: + const OperandMatcher &Operand; const Record &TheDef; - /// The index of the first temporary operand to allocate to this - /// ComplexPattern. - unsigned BaseTemporaryID; unsigned getNumOperands() const { return TheDef.getValueAsDag("Operands")->getNumArgs(); } + unsigned getAllocatedTemporariesBaseID() const; + public: - ComplexPatternOperandMatcher(const Record &TheDef, unsigned BaseTemporaryID) - : OperandPredicateMatcher(OPM_ComplexPattern), TheDef(TheDef), - BaseTemporaryID(BaseTemporaryID) {} + ComplexPatternOperandMatcher(const OperandMatcher &Operand, + const Record &TheDef) + : OperandPredicateMatcher(OPM_ComplexPattern), Operand(Operand), + TheDef(TheDef) {} + + static bool classof(const OperandPredicateMatcher *P) { + return P->getKind() == OPM_ComplexPattern; + } void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule, StringRef OperandExpr) const override { OS << TheDef.getValueAsString("MatcherFn") << "(" << OperandExpr; for (unsigned I = 0; I < getNumOperands(); ++I) { OS << ", "; - OperandPlaceholder::CreateTemporary(BaseTemporaryID + I) + OperandPlaceholder::CreateTemporary(getAllocatedTemporariesBaseID() + I) .emitCxxValueExpr(OS); } OS << ")"; @@ -420,10 +425,17 @@ unsigned OpIdx; std::string SymbolicName; + /// The index of the first temporary variable allocated to this operand. The + /// number of allocated temporaries can be found with + /// countTemporaryOperands(). + unsigned AllocatedTemporariesBaseID; + public: OperandMatcher(InstructionMatcher &Insn, unsigned OpIdx, - const std::string &SymbolicName) - : Insn(Insn), OpIdx(OpIdx), SymbolicName(SymbolicName) {} + const std::string &SymbolicName, + unsigned AllocatedTemporariesBaseID) + : Insn(Insn), OpIdx(OpIdx), SymbolicName(SymbolicName), + AllocatedTemporariesBaseID(AllocatedTemporariesBaseID) {} bool hasSymbolicName() const { return !SymbolicName.empty(); } const StringRef getSymbolicName() const { return SymbolicName; } @@ -504,8 +516,16 @@ return A + Predicate->countTemporaryOperands(); }); } + + unsigned getAllocatedTemporariesBaseID() const { + return AllocatedTemporariesBaseID; + } }; +unsigned ComplexPatternOperandMatcher::getAllocatedTemporariesBaseID() const { + return Operand.getAllocatedTemporariesBaseID(); +} + /// Generates code to check a predicate on an instruction. /// /// Typical predicates include: @@ -593,7 +613,7 @@ class InstructionMatcher : public PredicateListMatcher { protected: - typedef std::vector OperandVec; + typedef std::vector> OperandVec; /// The operands to match. All rendered operands must be present even if the /// condition is always true. @@ -601,18 +621,20 @@ public: /// Add an operand to the matcher. - OperandMatcher &addOperand(unsigned OpIdx, const std::string &SymbolicName) { - Operands.emplace_back(*this, OpIdx, SymbolicName); - return Operands.back(); + OperandMatcher &addOperand(unsigned OpIdx, const std::string &SymbolicName, + unsigned AllocatedTemporariesBaseID) { + Operands.emplace_back(new OperandMatcher(*this, OpIdx, SymbolicName, + AllocatedTemporariesBaseID)); + return *Operands.back(); } OperandMatcher &getOperand(unsigned OpIdx) { auto I = std::find_if(Operands.begin(), Operands.end(), - [&OpIdx](const OperandMatcher &X) { - return X.getOperandIndex() == OpIdx; + [&OpIdx](const std::unique_ptr &X) { + return X->getOperandIndex() == OpIdx; }); if (I != Operands.end()) - return *I; + return **I; llvm_unreachable("Failed to lookup operand"); } @@ -620,7 +642,7 @@ getOptionalOperand(StringRef SymbolicName) const { assert(!SymbolicName.empty() && "Cannot lookup unnamed operand"); for (const auto &Operand : Operands) { - const auto &OM = Operand.getOptionalOperand(SymbolicName); + const auto &OM = Operand->getOptionalOperand(SymbolicName); if (OM.hasValue()) return OM.getValue(); } @@ -655,8 +677,8 @@ OS << "if (" << Expr << ".getNumOperands() < " << getNumOperands() << ")\n" << " return false;\n"; for (const auto &Operand : Operands) { - Operand.emitCxxCaptureStmtsForOperand(OS, Rule, - Operand.getOperandExpr(Expr)); + Operand->emitCxxCaptureStmtsForOperand(OS, Rule, + Operand->getOperandExpr(Expr)); } } @@ -667,7 +689,7 @@ emitCxxPredicateListExpr(OS, Rule, InsnVarName); for (const auto &Operand : Operands) { OS << " &&\n("; - Operand.emitCxxPredicateExpr(OS, Rule, InsnVarName); + Operand->emitCxxPredicateExpr(OS, Rule, InsnVarName); OS << ")"; } } @@ -690,9 +712,9 @@ } for (const auto &Operand : zip(Operands, B.Operands)) { - if (std::get<0>(Operand).isHigherPriorityThan(std::get<1>(Operand))) + if (std::get<0>(Operand)->isHigherPriorityThan(*std::get<1>(Operand))) return true; - if (std::get<1>(Operand).isHigherPriorityThan(std::get<0>(Operand))) + if (std::get<1>(Operand)->isHigherPriorityThan(*std::get<0>(Operand))) return false; } @@ -708,10 +730,11 @@ &Predicate) { return A + Predicate->countTemporaryOperands(); }) + - std::accumulate(Operands.begin(), Operands.end(), 0, - [](unsigned A, const OperandMatcher &Operand) { - return A + Operand.countTemporaryOperands(); - }); + std::accumulate( + Operands.begin(), Operands.end(), 0, + [](unsigned A, const std::unique_ptr &Operand) { + return A + Operand->countTemporaryOperands(); + }); } }; @@ -1138,9 +1161,10 @@ Expected importInstructionRenderer(RuleMatcher &M, const TreePatternNode *Dst, const InstructionMatcher &InsnMatcher) const; - Expected importExplicitUseRenderer( - BuildMIAction &DstMIBuilder, TreePatternNode *DstChild, - const InstructionMatcher &InsnMatcher, unsigned &TempOpIdx) const; + Expected + importExplicitUseRenderer(BuildMIAction &DstMIBuilder, + TreePatternNode *DstChild, + const InstructionMatcher &InsnMatcher) const; Expected importImplicitDefRenderers(BuildMIAction &DstMIBuilder, const std::vector &ImplicitDefs) const; @@ -1203,6 +1227,7 @@ InsnMatcher.addPredicate(&SrcGI); unsigned OpIdx = 0; + unsigned TempOpIdx = 0; for (const EEVT::TypeSet &Ty : Src->getExtTypes()) { auto OpTyOrNone = MVTToLLT(Ty.getConcrete()); @@ -1212,11 +1237,10 @@ // Results don't have a name unless they are the root node. The caller will // set the name if appropriate. - OperandMatcher &OM = InsnMatcher.addOperand(OpIdx++, ""); + OperandMatcher &OM = InsnMatcher.addOperand(OpIdx++, "", TempOpIdx); OM.addPredicate(*OpTyOrNone); } - unsigned TempOpIdx = 0; // Match the used operands (i.e. the children of the operator). for (unsigned i = 0, e = Src->getNumChildren(); i != e; ++i) { auto TrueOrError = @@ -1232,7 +1256,7 @@ GlobalISelEmitter::importChildMatcher(InstructionMatcher &InsnMatcher, TreePatternNode *SrcChild, unsigned OpIdx, unsigned &TempOpIdx) const { - OperandMatcher &OM = InsnMatcher.addOperand(OpIdx, SrcChild->getName()); + OperandMatcher &OM = InsnMatcher.addOperand(OpIdx, SrcChild->getName(), TempOpIdx); if (SrcChild->hasAnyPredicate()) return failedImport("Src pattern child has predicate"); @@ -1295,7 +1319,7 @@ "SelectionDAG ComplexPattern not mapped to GlobalISel"); const auto &Predicate = OM.addPredicate( - *ComplexPattern->second, TempOpIdx); + OM, *ComplexPattern->second); TempOpIdx += Predicate.countTemporaryOperands(); return true; } @@ -1309,7 +1333,7 @@ Expected GlobalISelEmitter::importExplicitUseRenderer( BuildMIAction &DstMIBuilder, TreePatternNode *DstChild, - const InstructionMatcher &InsnMatcher, unsigned &TempOpIdx) const { + const InstructionMatcher &InsnMatcher) const { // The only non-leaf child we accept is 'bb': it's an operator because // BasicBlockSDNode isn't inline, but in MI it's just another operand. if (!DstChild->isLeaf()) { @@ -1356,13 +1380,10 @@ "SelectionDAG ComplexPattern not mapped to GlobalISel"); SmallVector RenderedOperands; - for (unsigned I = 0; - I < - InsnMatcher.getOperand(DstChild->getName()).countTemporaryOperands(); - ++I) { - RenderedOperands.push_back(OperandPlaceholder::CreateTemporary(I)); - TempOpIdx++; - } + const OperandMatcher &OM = InsnMatcher.getOperand(DstChild->getName()); + for (unsigned I = 0; I < OM.countTemporaryOperands(); ++I) + RenderedOperands.push_back(OperandPlaceholder::CreateTemporary( + OM.getAllocatedTemporariesBaseID() + I)); DstMIBuilder.addRenderer( *ComplexPattern->second, RenderedOperands); return true; @@ -1392,10 +1413,9 @@ } // Render the explicit uses. - unsigned TempOpIdx = 0; for (unsigned i = 0, e = Dst->getNumChildren(); i != e; ++i) { - auto TrueOrError = importExplicitUseRenderer(DstMIBuilder, Dst->getChild(i), - InsnMatcher, TempOpIdx); + auto TrueOrError = + importExplicitUseRenderer(DstMIBuilder, Dst->getChild(i), InsnMatcher); if (auto Error = TrueOrError.takeError()) return std::move(Error); }