blob: 7acc65e349ea60ef7b0000aef335a29060ff8fef [file] [log] [blame]
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001//===- GlobalISelEmitter.cpp - Generate an instruction selector -----------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10/// \file
11/// This tablegen backend emits code for use by the GlobalISel instruction
12/// selector. See include/llvm/CodeGen/TargetGlobalISel.td.
13///
14/// This file analyzes the patterns recognized by the SelectionDAGISel tablegen
15/// backend, filters out the ones that are unsupported, maps
16/// SelectionDAG-specific constructs to their GlobalISel counterpart
17/// (when applicable: MVT to LLT; SDNode to generic Instruction).
18///
19/// Not all patterns are supported: pass the tablegen invocation
20/// "-warn-on-skipped-patterns" to emit a warning when a pattern is skipped,
21/// as well as why.
22///
23/// The generated file defines a single method:
24/// bool <Target>InstructionSelector::selectImpl(MachineInstr &I) const;
25/// intended to be used in InstructionSelector::select as the first-step
26/// selector for the patterns that don't require complex C++.
27///
28/// FIXME: We'll probably want to eventually define a base
29/// "TargetGenInstructionSelector" class.
30///
31//===----------------------------------------------------------------------===//
32
33#include "CodeGenDAGPatterns.h"
34#include "llvm/ADT/Optional.h"
Daniel Sanders0ed28822017-04-12 08:23:08 +000035#include "llvm/ADT/SmallSet.h"
Ahmed Bougacha36f70352016-12-21 23:26:20 +000036#include "llvm/ADT/Statistic.h"
37#include "llvm/CodeGen/MachineValueType.h"
38#include "llvm/Support/CommandLine.h"
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +000039#include "llvm/Support/Error.h"
Daniel Sanders52b4ce72017-03-07 23:20:35 +000040#include "llvm/Support/LowLevelTypeImpl.h"
Pavel Labath52a82e22017-02-21 09:19:41 +000041#include "llvm/Support/ScopedPrinter.h"
Ahmed Bougacha36f70352016-12-21 23:26:20 +000042#include "llvm/TableGen/Error.h"
43#include "llvm/TableGen/Record.h"
44#include "llvm/TableGen/TableGenBackend.h"
45#include <string>
Daniel Sanders8a4bae92017-03-14 21:32:08 +000046#include <numeric>
Ahmed Bougacha36f70352016-12-21 23:26:20 +000047using namespace llvm;
48
49#define DEBUG_TYPE "gisel-emitter"
50
51STATISTIC(NumPatternTotal, "Total number of patterns");
Daniel Sandersb41ce2b2017-02-20 14:31:27 +000052STATISTIC(NumPatternImported, "Number of patterns imported from SelectionDAG");
53STATISTIC(NumPatternImportsSkipped, "Number of SelectionDAG imports skipped");
Ahmed Bougacha36f70352016-12-21 23:26:20 +000054STATISTIC(NumPatternEmitted, "Number of patterns emitted");
55
Daniel Sanders0848b232017-03-27 13:15:13 +000056cl::OptionCategory GlobalISelEmitterCat("Options for -gen-global-isel");
57
Ahmed Bougacha36f70352016-12-21 23:26:20 +000058static cl::opt<bool> WarnOnSkippedPatterns(
59 "warn-on-skipped-patterns",
60 cl::desc("Explain why a pattern was skipped for inclusion "
61 "in the GlobalISel selector"),
Daniel Sanders0848b232017-03-27 13:15:13 +000062 cl::init(false), cl::cat(GlobalISelEmitterCat));
Ahmed Bougacha36f70352016-12-21 23:26:20 +000063
Daniel Sandersbdfebb82017-03-15 20:18:38 +000064namespace {
Ahmed Bougacha36f70352016-12-21 23:26:20 +000065//===- Helper functions ---------------------------------------------------===//
66
Daniel Sanders52b4ce72017-03-07 23:20:35 +000067/// This class stands in for LLT wherever we want to tablegen-erate an
68/// equivalent at compiler run-time.
69class LLTCodeGen {
70private:
71 LLT Ty;
72
73public:
74 LLTCodeGen(const LLT &Ty) : Ty(Ty) {}
75
76 void emitCxxConstructorCall(raw_ostream &OS) const {
77 if (Ty.isScalar()) {
78 OS << "LLT::scalar(" << Ty.getSizeInBits() << ")";
79 return;
80 }
81 if (Ty.isVector()) {
82 OS << "LLT::vector(" << Ty.getNumElements() << ", " << Ty.getSizeInBits()
83 << ")";
84 return;
85 }
86 llvm_unreachable("Unhandled LLT");
87 }
Daniel Sanders8a4bae92017-03-14 21:32:08 +000088
89 const LLT &get() const { return Ty; }
90};
91
92class InstructionMatcher;
93class OperandPlaceholder {
94private:
95 enum PlaceholderKind {
96 OP_MatchReference,
97 OP_Temporary,
98 } Kind;
99
100 struct MatchReferenceData {
101 InstructionMatcher *InsnMatcher;
102 StringRef InsnVarName;
103 StringRef SymbolicName;
104 };
105
106 struct TemporaryData {
107 unsigned OpIdx;
108 };
109
110 union {
111 struct MatchReferenceData MatchReference;
112 struct TemporaryData Temporary;
113 };
114
115 OperandPlaceholder(PlaceholderKind Kind) : Kind(Kind) {}
116
117public:
118 ~OperandPlaceholder() {}
119
120 static OperandPlaceholder
121 CreateMatchReference(InstructionMatcher *InsnMatcher,
Daniel Sandersdb7ed372017-04-04 13:52:00 +0000122 StringRef InsnVarName, StringRef SymbolicName) {
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000123 OperandPlaceholder Result(OP_MatchReference);
124 Result.MatchReference.InsnMatcher = InsnMatcher;
125 Result.MatchReference.InsnVarName = InsnVarName;
126 Result.MatchReference.SymbolicName = SymbolicName;
127 return Result;
128 }
129
130 static OperandPlaceholder CreateTemporary(unsigned OpIdx) {
131 OperandPlaceholder Result(OP_Temporary);
132 Result.Temporary.OpIdx = OpIdx;
133 return Result;
134 }
135
136 void emitCxxValueExpr(raw_ostream &OS) const;
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000137};
138
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000139/// Convert an MVT to an equivalent LLT if possible, or the invalid LLT() for
140/// MVTs that don't map cleanly to an LLT (e.g., iPTR, *any, ...).
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000141static Optional<LLTCodeGen> MVTToLLT(MVT::SimpleValueType SVT) {
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000142 MVT VT(SVT);
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000143 if (VT.isVector() && VT.getVectorNumElements() != 1)
144 return LLTCodeGen(LLT::vector(VT.getVectorNumElements(), VT.getScalarSizeInBits()));
145 if (VT.isInteger() || VT.isFloatingPoint())
146 return LLTCodeGen(LLT::scalar(VT.getSizeInBits()));
147 return None;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000148}
149
Daniel Sandersd0656a32017-04-13 09:45:37 +0000150static std::string explainPredicates(const TreePatternNode *N) {
151 std::string Explanation = "";
152 StringRef Separator = "";
153 for (const auto &P : N->getPredicateFns()) {
154 Explanation +=
155 (Separator + P.getOrigPatFragRecord()->getRecord()->getName()).str();
156 if (P.isAlwaysTrue())
157 Explanation += " always-true";
158 if (P.isImmediatePattern())
159 Explanation += " immediate";
160 }
161 return Explanation;
162}
163
164static std::string explainRulePredicates(const ArrayRef<Init *> Predicates) {
165 std::string Explanation = "";
166 StringRef Separator = "";
167 for (const auto *P : Predicates) {
168 Explanation += Separator;
169
170 if (const DefInit *PDef = dyn_cast<DefInit>(P)) {
171 Explanation += PDef->getDef()->getName();
172 } else
173 Explanation += "<unknown>";
174 }
175 return Explanation;
176}
177
178std::string explainOperator(Record *Operator) {
179 if (Operator->isSubClassOf("SDNode"))
180 return " (" + Operator->getValueAsString("Opcode") + ")";
181
182 if (Operator->isSubClassOf("Intrinsic"))
183 return (" (Operator is an Intrinsic, " + Operator->getName() + ")").str();
184
185 return " (Operator not understood)";
186}
187
188/// Helper function to let the emitter report skip reason error messages.
189static Error failedImport(const Twine &Reason) {
190 return make_error<StringError>(Reason, inconvertibleErrorCode());
191}
192
193static Error isTrivialOperatorNode(const TreePatternNode *N) {
194 std::string Explanation = "";
195 std::string Separator = "";
196 if (N->isLeaf()) {
197 Explanation = "Is a leaf";
198 Separator = ", ";
199 }
200
201 if (N->hasAnyPredicate()) {
202 Explanation = Separator + "Has a predicate (" + explainPredicates(N) + ")";
203 Separator = ", ";
204 }
205
206 if (N->getTransformFn()) {
207 Explanation += Separator + "Has a transform function";
208 Separator = ", ";
209 }
210
211 if (!N->isLeaf() && !N->hasAnyPredicate() && !N->getTransformFn())
212 return Error::success();
213
214 return failedImport(Explanation);
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000215}
216
217//===- Matchers -----------------------------------------------------------===//
218
Daniel Sandersbee57392017-04-04 13:25:23 +0000219class OperandMatcher;
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000220class MatchAction;
221
222/// Generates code to check that a match rule matches.
223class RuleMatcher {
224 /// A list of matchers that all need to succeed for the current rule to match.
225 /// FIXME: This currently supports a single match position but could be
226 /// extended to support multiple positions to support div/rem fusion or
227 /// load-multiple instructions.
228 std::vector<std::unique_ptr<InstructionMatcher>> Matchers;
229
230 /// A list of actions that need to be taken when all predicates in this rule
231 /// have succeeded.
232 std::vector<std::unique_ptr<MatchAction>> Actions;
233
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000234 /// A map of instruction matchers to the local variables created by
235 /// emitCxxCaptureStmts().
236 std::map<const InstructionMatcher *, std::string> InsnVariableNames;
237
238 /// ID for the next instruction variable defined with defineInsnVar()
239 unsigned NextInsnVarID;
240
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000241public:
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000242 RuleMatcher()
243 : Matchers(), Actions(), InsnVariableNames(), NextInsnVarID(0) {}
Zachary Turnerb7dbd872017-03-20 19:56:52 +0000244 RuleMatcher(RuleMatcher &&Other) = default;
245 RuleMatcher &operator=(RuleMatcher &&Other) = default;
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000246
247 InstructionMatcher &addInstructionMatcher();
248
249 template <class Kind, class... Args> Kind &addAction(Args &&... args);
250
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000251 std::string defineInsnVar(raw_ostream &OS, const InstructionMatcher &Matcher,
252 StringRef Value);
253 StringRef getInsnVarName(const InstructionMatcher &InsnMatcher) const;
254
Daniel Sandersbee57392017-04-04 13:25:23 +0000255 void emitCxxCapturedInsnList(raw_ostream &OS);
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000256 void emitCxxCaptureStmts(raw_ostream &OS, StringRef Expr);
257
258 void emit(raw_ostream &OS);
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000259
260 /// Compare the priority of this object and B.
261 ///
262 /// Returns true if this object is more important than B.
263 bool isHigherPriorityThan(const RuleMatcher &B) const;
264
265 /// Report the maximum number of temporary operands needed by the rule
266 /// matcher.
267 unsigned countTemporaryOperands() const;
268};
269
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000270template <class PredicateTy> class PredicateListMatcher {
271private:
272 typedef std::vector<std::unique_ptr<PredicateTy>> PredicateVec;
273 PredicateVec Predicates;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000274
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000275public:
276 /// Construct a new operand predicate and add it to the matcher.
277 template <class Kind, class... Args>
278 Kind &addPredicate(Args&&... args) {
Ahmed Bougachab67a3ce2017-01-26 22:07:37 +0000279 Predicates.emplace_back(
280 llvm::make_unique<Kind>(std::forward<Args>(args)...));
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000281 return *static_cast<Kind *>(Predicates.back().get());
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000282 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000283
284 typename PredicateVec::const_iterator predicates_begin() const { return Predicates.begin(); }
285 typename PredicateVec::const_iterator predicates_end() const { return Predicates.end(); }
286 iterator_range<typename PredicateVec::const_iterator> predicates() const {
287 return make_range(predicates_begin(), predicates_end());
288 }
Daniel Sanders759ff412017-02-24 13:58:11 +0000289 typename PredicateVec::size_type predicates_size() const { return Predicates.size(); }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000290
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000291 /// Emit a C++ expression that tests whether all the predicates are met.
Ahmed Bougachab67a3ce2017-01-26 22:07:37 +0000292 template <class... Args>
Daniel Sandersf8c804f2017-01-28 11:10:42 +0000293 void emitCxxPredicateListExpr(raw_ostream &OS, Args &&... args) const {
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000294 if (Predicates.empty()) {
295 OS << "true";
296 return;
297 }
298
299 StringRef Separator = "";
300 for (const auto &Predicate : predicates()) {
301 OS << Separator << "(";
Ahmed Bougachab67a3ce2017-01-26 22:07:37 +0000302 Predicate->emitCxxPredicateExpr(OS, std::forward<Args>(args)...);
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000303 OS << ")";
Ahmed Bougacha905af9f2017-02-04 00:47:02 +0000304 Separator = " &&\n";
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000305 }
306 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000307};
308
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000309/// Generates code to check a predicate of an operand.
310///
311/// Typical predicates include:
312/// * Operand is a particular register.
313/// * Operand is assigned a particular register bank.
314/// * Operand is an MBB.
315class OperandPredicateMatcher {
316public:
Daniel Sanders759ff412017-02-24 13:58:11 +0000317 /// This enum is used for RTTI and also defines the priority that is given to
318 /// the predicate when generating the matcher code. Kinds with higher priority
319 /// must be tested first.
320 ///
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000321 /// The relative priority of OPM_LLT, OPM_RegBank, and OPM_MBB do not matter
322 /// but OPM_Int must have priority over OPM_RegBank since constant integers
323 /// are represented by a virtual register defined by a G_CONSTANT instruction.
Daniel Sanders759ff412017-02-24 13:58:11 +0000324 enum PredicateKind {
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000325 OPM_ComplexPattern,
Daniel Sandersbee57392017-04-04 13:25:23 +0000326 OPM_Instruction,
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000327 OPM_Int,
Daniel Sanders759ff412017-02-24 13:58:11 +0000328 OPM_LLT,
329 OPM_RegBank,
330 OPM_MBB,
331 };
332
333protected:
334 PredicateKind Kind;
335
336public:
337 OperandPredicateMatcher(PredicateKind Kind) : Kind(Kind) {}
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000338 virtual ~OperandPredicateMatcher() {}
339
Daniel Sanders759ff412017-02-24 13:58:11 +0000340 PredicateKind getKind() const { return Kind; }
341
Daniel Sandersbee57392017-04-04 13:25:23 +0000342 /// Return the OperandMatcher for the specified operand or nullptr if there
343 /// isn't one by that name in this operand predicate matcher.
344 ///
345 /// InstructionOperandMatcher is the only subclass that can return non-null
346 /// for this.
347 virtual Optional<const OperandMatcher *>
Daniel Sandersdb7ed372017-04-04 13:52:00 +0000348 getOptionalOperand(StringRef SymbolicName) const {
Daniel Sandersbee57392017-04-04 13:25:23 +0000349 assert(!SymbolicName.empty() && "Cannot lookup unnamed operand");
350 return None;
351 }
352
353 /// Emit C++ statements to capture instructions into local variables.
354 ///
355 /// Only InstructionOperandMatcher needs to do anything for this method.
356 virtual void emitCxxCaptureStmts(raw_ostream &OS, RuleMatcher &Rule,
357 StringRef Expr) const {}
358
Daniel Sanderse604ef52017-02-20 15:30:43 +0000359 /// Emit a C++ expression that checks the predicate for the given operand.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000360 virtual void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanderse604ef52017-02-20 15:30:43 +0000361 StringRef OperandExpr) const = 0;
Daniel Sanders759ff412017-02-24 13:58:11 +0000362
363 /// Compare the priority of this object and B.
364 ///
365 /// Returns true if this object is more important than B.
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000366 virtual bool isHigherPriorityThan(const OperandPredicateMatcher &B) const {
367 return Kind < B.Kind;
Daniel Sanders759ff412017-02-24 13:58:11 +0000368 };
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000369
370 /// Report the maximum number of temporary operands needed by the predicate
371 /// matcher.
372 virtual unsigned countTemporaryOperands() const { return 0; }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000373};
374
375/// Generates code to check that an operand is a particular LLT.
376class LLTOperandMatcher : public OperandPredicateMatcher {
377protected:
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000378 LLTCodeGen Ty;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000379
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000380public:
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000381 LLTOperandMatcher(const LLTCodeGen &Ty)
Daniel Sanders759ff412017-02-24 13:58:11 +0000382 : OperandPredicateMatcher(OPM_LLT), Ty(Ty) {}
383
384 static bool classof(const OperandPredicateMatcher *P) {
385 return P->getKind() == OPM_LLT;
386 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000387
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000388 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanderse604ef52017-02-20 15:30:43 +0000389 StringRef OperandExpr) const override {
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000390 OS << "MRI.getType(" << OperandExpr << ".getReg()) == (";
391 Ty.emitCxxConstructorCall(OS);
392 OS << ")";
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000393 }
394};
395
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000396/// Generates code to check that an operand is a particular target constant.
397class ComplexPatternOperandMatcher : public OperandPredicateMatcher {
398protected:
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000399 const OperandMatcher &Operand;
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000400 const Record &TheDef;
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000401
402 unsigned getNumOperands() const {
403 return TheDef.getValueAsDag("Operands")->getNumArgs();
404 }
405
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000406 unsigned getAllocatedTemporariesBaseID() const;
407
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000408public:
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000409 ComplexPatternOperandMatcher(const OperandMatcher &Operand,
410 const Record &TheDef)
411 : OperandPredicateMatcher(OPM_ComplexPattern), Operand(Operand),
412 TheDef(TheDef) {}
413
414 static bool classof(const OperandPredicateMatcher *P) {
415 return P->getKind() == OPM_ComplexPattern;
416 }
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000417
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000418 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000419 StringRef OperandExpr) const override {
420 OS << TheDef.getValueAsString("MatcherFn") << "(" << OperandExpr;
421 for (unsigned I = 0; I < getNumOperands(); ++I) {
422 OS << ", ";
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000423 OperandPlaceholder::CreateTemporary(getAllocatedTemporariesBaseID() + I)
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000424 .emitCxxValueExpr(OS);
425 }
426 OS << ")";
427 }
428
429 unsigned countTemporaryOperands() const override {
430 return getNumOperands();
431 }
432};
433
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000434/// Generates code to check that an operand is in a particular register bank.
435class RegisterBankOperandMatcher : public OperandPredicateMatcher {
436protected:
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000437 const CodeGenRegisterClass &RC;
438
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000439public:
Daniel Sanders759ff412017-02-24 13:58:11 +0000440 RegisterBankOperandMatcher(const CodeGenRegisterClass &RC)
441 : OperandPredicateMatcher(OPM_RegBank), RC(RC) {}
442
443 static bool classof(const OperandPredicateMatcher *P) {
444 return P->getKind() == OPM_RegBank;
445 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000446
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000447 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanderse604ef52017-02-20 15:30:43 +0000448 StringRef OperandExpr) const override {
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000449 OS << "(&RBI.getRegBankFromRegClass(" << RC.getQualifiedName()
Daniel Sanderse604ef52017-02-20 15:30:43 +0000450 << "RegClass) == RBI.getRegBank(" << OperandExpr
451 << ".getReg(), MRI, TRI))";
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000452 }
453};
454
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000455/// Generates code to check that an operand is a basic block.
456class MBBOperandMatcher : public OperandPredicateMatcher {
457public:
Daniel Sanders759ff412017-02-24 13:58:11 +0000458 MBBOperandMatcher() : OperandPredicateMatcher(OPM_MBB) {}
459
460 static bool classof(const OperandPredicateMatcher *P) {
461 return P->getKind() == OPM_MBB;
462 }
463
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000464 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanderse604ef52017-02-20 15:30:43 +0000465 StringRef OperandExpr) const override {
466 OS << OperandExpr << ".isMBB()";
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000467 }
468};
469
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000470/// Generates code to check that an operand is a particular int.
471class IntOperandMatcher : public OperandPredicateMatcher {
472protected:
473 int64_t Value;
474
475public:
476 IntOperandMatcher(int64_t Value)
477 : OperandPredicateMatcher(OPM_Int), Value(Value) {}
478
479 static bool classof(const OperandPredicateMatcher *P) {
480 return P->getKind() == OPM_Int;
481 }
482
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000483 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Simon Pilgrimd0302912017-02-24 17:20:27 +0000484 StringRef OperandExpr) const override {
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000485 OS << "isOperandImmEqual(" << OperandExpr << ", " << Value << ", MRI)";
486 }
487};
488
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000489/// Generates code to check that a set of predicates match for a particular
490/// operand.
491class OperandMatcher : public PredicateListMatcher<OperandPredicateMatcher> {
492protected:
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000493 InstructionMatcher &Insn;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000494 unsigned OpIdx;
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000495 std::string SymbolicName;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000496
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000497 /// The index of the first temporary variable allocated to this operand. The
498 /// number of allocated temporaries can be found with
499 /// countTemporaryOperands().
500 unsigned AllocatedTemporariesBaseID;
501
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000502public:
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000503 OperandMatcher(InstructionMatcher &Insn, unsigned OpIdx,
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000504 const std::string &SymbolicName,
505 unsigned AllocatedTemporariesBaseID)
506 : Insn(Insn), OpIdx(OpIdx), SymbolicName(SymbolicName),
507 AllocatedTemporariesBaseID(AllocatedTemporariesBaseID) {}
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000508
509 bool hasSymbolicName() const { return !SymbolicName.empty(); }
510 const StringRef getSymbolicName() const { return SymbolicName; }
Daniel Sandersffc7d582017-03-29 15:37:18 +0000511 void setSymbolicName(StringRef Name) {
512 assert(SymbolicName.empty() && "Operand already has a symbolic name");
513 SymbolicName = Name;
514 }
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000515 unsigned getOperandIndex() const { return OpIdx; }
516
Daniel Sandersdb7ed372017-04-04 13:52:00 +0000517 std::string getOperandExpr(StringRef InsnVarName) const {
Pavel Labath52a82e22017-02-21 09:19:41 +0000518 return (InsnVarName + ".getOperand(" + llvm::to_string(OpIdx) + ")").str();
Daniel Sanderse604ef52017-02-20 15:30:43 +0000519 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000520
Daniel Sandersbee57392017-04-04 13:25:23 +0000521 Optional<const OperandMatcher *>
522 getOptionalOperand(StringRef DesiredSymbolicName) const {
523 assert(!DesiredSymbolicName.empty() && "Cannot lookup unnamed operand");
524 if (DesiredSymbolicName == SymbolicName)
525 return this;
526 for (const auto &OP : predicates()) {
527 const auto &MaybeOperand = OP->getOptionalOperand(DesiredSymbolicName);
528 if (MaybeOperand.hasValue())
529 return MaybeOperand.getValue();
530 }
531 return None;
532 }
533
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000534 InstructionMatcher &getInstructionMatcher() const { return Insn; }
535
Daniel Sandersbee57392017-04-04 13:25:23 +0000536 /// Emit C++ statements to capture instructions into local variables.
537 void emitCxxCaptureStmts(raw_ostream &OS, RuleMatcher &Rule,
538 StringRef OperandExpr) const {
539 for (const auto &Predicate : predicates())
540 Predicate->emitCxxCaptureStmts(OS, Rule, OperandExpr);
541 }
542
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000543 /// Emit a C++ expression that tests whether the instruction named in
544 /// InsnVarName matches all the predicate and all the operands.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000545 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sandersdb7ed372017-04-04 13:52:00 +0000546 StringRef InsnVarName) const {
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000547 OS << "(/* ";
548 if (SymbolicName.empty())
549 OS << "Operand " << OpIdx;
550 else
551 OS << SymbolicName;
552 OS << " */ ";
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000553 emitCxxPredicateListExpr(OS, Rule, getOperandExpr(InsnVarName));
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000554 OS << ")";
555 }
Daniel Sanders759ff412017-02-24 13:58:11 +0000556
557 /// Compare the priority of this object and B.
558 ///
559 /// Returns true if this object is more important than B.
560 bool isHigherPriorityThan(const OperandMatcher &B) const {
561 // Operand matchers involving more predicates have higher priority.
562 if (predicates_size() > B.predicates_size())
563 return true;
564 if (predicates_size() < B.predicates_size())
565 return false;
566
567 // This assumes that predicates are added in a consistent order.
568 for (const auto &Predicate : zip(predicates(), B.predicates())) {
569 if (std::get<0>(Predicate)->isHigherPriorityThan(*std::get<1>(Predicate)))
570 return true;
571 if (std::get<1>(Predicate)->isHigherPriorityThan(*std::get<0>(Predicate)))
572 return false;
573 }
574
575 return false;
576 };
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000577
578 /// Report the maximum number of temporary operands needed by the operand
579 /// matcher.
580 unsigned countTemporaryOperands() const {
581 return std::accumulate(
582 predicates().begin(), predicates().end(), 0,
583 [](unsigned A,
584 const std::unique_ptr<OperandPredicateMatcher> &Predicate) {
585 return A + Predicate->countTemporaryOperands();
586 });
587 }
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000588
589 unsigned getAllocatedTemporariesBaseID() const {
590 return AllocatedTemporariesBaseID;
591 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000592};
593
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000594unsigned ComplexPatternOperandMatcher::getAllocatedTemporariesBaseID() const {
595 return Operand.getAllocatedTemporariesBaseID();
596}
597
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000598/// Generates code to check a predicate on an instruction.
599///
600/// Typical predicates include:
601/// * The opcode of the instruction is a particular value.
602/// * The nsw/nuw flag is/isn't set.
603class InstructionPredicateMatcher {
Daniel Sanders759ff412017-02-24 13:58:11 +0000604protected:
605 /// This enum is used for RTTI and also defines the priority that is given to
606 /// the predicate when generating the matcher code. Kinds with higher priority
607 /// must be tested first.
608 enum PredicateKind {
609 IPM_Opcode,
610 };
611
612 PredicateKind Kind;
613
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000614public:
Daniel Sanders8d4d72f2017-02-24 14:53:35 +0000615 InstructionPredicateMatcher(PredicateKind Kind) : Kind(Kind) {}
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000616 virtual ~InstructionPredicateMatcher() {}
617
Daniel Sanders759ff412017-02-24 13:58:11 +0000618 PredicateKind getKind() const { return Kind; }
619
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000620 /// Emit a C++ expression that tests whether the instruction named in
621 /// InsnVarName matches the predicate.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000622 virtual void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Ahmed Bougacha6a1ac5a2017-02-09 02:50:01 +0000623 StringRef InsnVarName) const = 0;
Daniel Sanders759ff412017-02-24 13:58:11 +0000624
625 /// Compare the priority of this object and B.
626 ///
627 /// Returns true if this object is more important than B.
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000628 virtual bool isHigherPriorityThan(const InstructionPredicateMatcher &B) const {
Daniel Sanders759ff412017-02-24 13:58:11 +0000629 return Kind < B.Kind;
630 };
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000631
632 /// Report the maximum number of temporary operands needed by the predicate
633 /// matcher.
634 virtual unsigned countTemporaryOperands() const { return 0; }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000635};
636
637/// Generates code to check the opcode of an instruction.
638class InstructionOpcodeMatcher : public InstructionPredicateMatcher {
639protected:
640 const CodeGenInstruction *I;
641
642public:
Daniel Sanders8d4d72f2017-02-24 14:53:35 +0000643 InstructionOpcodeMatcher(const CodeGenInstruction *I)
644 : InstructionPredicateMatcher(IPM_Opcode), I(I) {}
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000645
Daniel Sanders759ff412017-02-24 13:58:11 +0000646 static bool classof(const InstructionPredicateMatcher *P) {
647 return P->getKind() == IPM_Opcode;
648 }
649
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000650 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Ahmed Bougacha6a1ac5a2017-02-09 02:50:01 +0000651 StringRef InsnVarName) const override {
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000652 OS << InsnVarName << ".getOpcode() == " << I->Namespace
653 << "::" << I->TheDef->getName();
654 }
Daniel Sanders759ff412017-02-24 13:58:11 +0000655
656 /// Compare the priority of this object and B.
657 ///
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000658 /// Returns true if this object is more important than B.
659 bool isHigherPriorityThan(const InstructionPredicateMatcher &B) const override {
Daniel Sanders759ff412017-02-24 13:58:11 +0000660 if (InstructionPredicateMatcher::isHigherPriorityThan(B))
661 return true;
662 if (B.InstructionPredicateMatcher::isHigherPriorityThan(*this))
663 return false;
664
665 // Prioritize opcodes for cosmetic reasons in the generated source. Although
666 // this is cosmetic at the moment, we may want to drive a similar ordering
667 // using instruction frequency information to improve compile time.
668 if (const InstructionOpcodeMatcher *BO =
669 dyn_cast<InstructionOpcodeMatcher>(&B))
670 return I->TheDef->getName() < BO->I->TheDef->getName();
671
672 return false;
673 };
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000674};
675
676/// Generates code to check that a set of predicates and operands match for a
677/// particular instruction.
678///
679/// Typical predicates include:
680/// * Has a specific opcode.
681/// * Has an nsw/nuw flag or doesn't.
682class InstructionMatcher
683 : public PredicateListMatcher<InstructionPredicateMatcher> {
684protected:
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000685 typedef std::vector<std::unique_ptr<OperandMatcher>> OperandVec;
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000686
687 /// The operands to match. All rendered operands must be present even if the
688 /// condition is always true.
689 OperandVec Operands;
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000690
691public:
692 /// Add an operand to the matcher.
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000693 OperandMatcher &addOperand(unsigned OpIdx, const std::string &SymbolicName,
694 unsigned AllocatedTemporariesBaseID) {
695 Operands.emplace_back(new OperandMatcher(*this, OpIdx, SymbolicName,
696 AllocatedTemporariesBaseID));
697 return *Operands.back();
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000698 }
699
Daniel Sandersffc7d582017-03-29 15:37:18 +0000700 OperandMatcher &getOperand(unsigned OpIdx) {
701 auto I = std::find_if(Operands.begin(), Operands.end(),
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000702 [&OpIdx](const std::unique_ptr<OperandMatcher> &X) {
703 return X->getOperandIndex() == OpIdx;
Daniel Sandersffc7d582017-03-29 15:37:18 +0000704 });
705 if (I != Operands.end())
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000706 return **I;
Daniel Sandersffc7d582017-03-29 15:37:18 +0000707 llvm_unreachable("Failed to lookup operand");
708 }
709
Daniel Sandersbee57392017-04-04 13:25:23 +0000710 Optional<const OperandMatcher *>
711 getOptionalOperand(StringRef SymbolicName) const {
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000712 assert(!SymbolicName.empty() && "Cannot lookup unnamed operand");
Daniel Sandersbee57392017-04-04 13:25:23 +0000713 for (const auto &Operand : Operands) {
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000714 const auto &OM = Operand->getOptionalOperand(SymbolicName);
Daniel Sandersbee57392017-04-04 13:25:23 +0000715 if (OM.hasValue())
716 return OM.getValue();
717 }
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000718 return None;
719 }
720
Daniel Sandersdb7ed372017-04-04 13:52:00 +0000721 const OperandMatcher &getOperand(StringRef SymbolicName) const {
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000722 Optional<const OperandMatcher *>OM = getOptionalOperand(SymbolicName);
723 if (OM.hasValue())
724 return *OM.getValue();
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000725 llvm_unreachable("Failed to lookup operand");
726 }
727
728 unsigned getNumOperands() const { return Operands.size(); }
Daniel Sandersbee57392017-04-04 13:25:23 +0000729 OperandVec::iterator operands_begin() { return Operands.begin(); }
730 OperandVec::iterator operands_end() { return Operands.end(); }
731 iterator_range<OperandVec::iterator> operands() {
732 return make_range(operands_begin(), operands_end());
733 }
Daniel Sandersffc7d582017-03-29 15:37:18 +0000734 OperandVec::const_iterator operands_begin() const { return Operands.begin(); }
735 OperandVec::const_iterator operands_end() const { return Operands.end(); }
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000736 iterator_range<OperandVec::const_iterator> operands() const {
737 return make_range(operands_begin(), operands_end());
738 }
739
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000740 /// Emit C++ statements to check the shape of the match and capture
741 /// instructions into local variables.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000742 void emitCxxCaptureStmts(raw_ostream &OS, RuleMatcher &Rule, StringRef Expr) {
743 OS << "if (" << Expr << ".getNumOperands() < " << getNumOperands() << ")\n"
744 << " return false;\n";
Daniel Sandersbee57392017-04-04 13:25:23 +0000745 for (const auto &Operand : Operands) {
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000746 Operand->emitCxxCaptureStmts(OS, Rule, Operand->getOperandExpr(Expr));
Daniel Sandersbee57392017-04-04 13:25:23 +0000747 }
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000748 }
749
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000750 /// Emit a C++ expression that tests whether the instruction named in
751 /// InsnVarName matches all the predicates and all the operands.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000752 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
753 StringRef InsnVarName) const {
754 emitCxxPredicateListExpr(OS, Rule, InsnVarName);
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000755 for (const auto &Operand : Operands) {
Ahmed Bougacha905af9f2017-02-04 00:47:02 +0000756 OS << " &&\n(";
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000757 Operand->emitCxxPredicateExpr(OS, Rule, InsnVarName);
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000758 OS << ")";
759 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000760 }
Daniel Sanders759ff412017-02-24 13:58:11 +0000761
762 /// Compare the priority of this object and B.
763 ///
764 /// Returns true if this object is more important than B.
765 bool isHigherPriorityThan(const InstructionMatcher &B) const {
766 // Instruction matchers involving more operands have higher priority.
767 if (Operands.size() > B.Operands.size())
768 return true;
769 if (Operands.size() < B.Operands.size())
770 return false;
771
772 for (const auto &Predicate : zip(predicates(), B.predicates())) {
773 if (std::get<0>(Predicate)->isHigherPriorityThan(*std::get<1>(Predicate)))
774 return true;
775 if (std::get<1>(Predicate)->isHigherPriorityThan(*std::get<0>(Predicate)))
776 return false;
777 }
778
779 for (const auto &Operand : zip(Operands, B.Operands)) {
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000780 if (std::get<0>(Operand)->isHigherPriorityThan(*std::get<1>(Operand)))
Daniel Sanders759ff412017-02-24 13:58:11 +0000781 return true;
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000782 if (std::get<1>(Operand)->isHigherPriorityThan(*std::get<0>(Operand)))
Daniel Sanders759ff412017-02-24 13:58:11 +0000783 return false;
784 }
785
786 return false;
787 };
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000788
789 /// Report the maximum number of temporary operands needed by the instruction
790 /// matcher.
791 unsigned countTemporaryOperands() const {
792 return std::accumulate(predicates().begin(), predicates().end(), 0,
793 [](unsigned A,
794 const std::unique_ptr<InstructionPredicateMatcher>
795 &Predicate) {
796 return A + Predicate->countTemporaryOperands();
797 }) +
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000798 std::accumulate(
799 Operands.begin(), Operands.end(), 0,
800 [](unsigned A, const std::unique_ptr<OperandMatcher> &Operand) {
801 return A + Operand->countTemporaryOperands();
802 });
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000803 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000804};
805
Daniel Sandersbee57392017-04-04 13:25:23 +0000806/// Generates code to check that the operand is a register defined by an
807/// instruction that matches the given instruction matcher.
808///
809/// For example, the pattern:
810/// (set $dst, (G_MUL (G_ADD $src1, $src2), $src3))
811/// would use an InstructionOperandMatcher for operand 1 of the G_MUL to match
812/// the:
813/// (G_ADD $src1, $src2)
814/// subpattern.
815class InstructionOperandMatcher : public OperandPredicateMatcher {
816protected:
817 std::unique_ptr<InstructionMatcher> InsnMatcher;
818
819public:
820 InstructionOperandMatcher()
821 : OperandPredicateMatcher(OPM_Instruction),
822 InsnMatcher(new InstructionMatcher()) {}
823
824 static bool classof(const OperandPredicateMatcher *P) {
825 return P->getKind() == OPM_Instruction;
826 }
827
828 InstructionMatcher &getInsnMatcher() const { return *InsnMatcher; }
829
830 Optional<const OperandMatcher *>
831 getOptionalOperand(StringRef SymbolicName) const override {
832 assert(!SymbolicName.empty() && "Cannot lookup unnamed operand");
833 return InsnMatcher->getOptionalOperand(SymbolicName);
834 }
835
836 void emitCxxCaptureStmts(raw_ostream &OS, RuleMatcher &Rule,
837 StringRef OperandExpr) const override {
838 OS << "if (!" << OperandExpr + ".isReg())\n"
839 << " return false;\n";
840 std::string InsnVarName = Rule.defineInsnVar(
841 OS, *InsnMatcher,
842 ("*MRI.getVRegDef(" + OperandExpr + ".getReg())").str());
843 InsnMatcher->emitCxxCaptureStmts(OS, Rule, InsnVarName);
844 }
845
846 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
847 StringRef OperandExpr) const override {
848 OperandExpr = Rule.getInsnVarName(*InsnMatcher);
849 OS << "(";
850 InsnMatcher->emitCxxPredicateExpr(OS, Rule, OperandExpr);
851 OS << ")\n";
852 }
853};
854
Daniel Sanders43c882c2017-02-01 10:53:10 +0000855//===- Actions ------------------------------------------------------------===//
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000856void OperandPlaceholder::emitCxxValueExpr(raw_ostream &OS) const {
857 switch (Kind) {
858 case OP_MatchReference:
859 OS << MatchReference.InsnMatcher->getOperand(MatchReference.SymbolicName)
860 .getOperandExpr(MatchReference.InsnVarName);
861 break;
862 case OP_Temporary:
863 OS << "TempOp" << Temporary.OpIdx;
864 break;
865 }
866}
Daniel Sanders43c882c2017-02-01 10:53:10 +0000867
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000868class OperandRenderer {
869public:
Daniel Sanders0ed28822017-04-12 08:23:08 +0000870 enum RendererKind { OR_Copy, OR_Imm, OR_Register, OR_ComplexPattern };
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000871
872protected:
873 RendererKind Kind;
874
875public:
876 OperandRenderer(RendererKind Kind) : Kind(Kind) {}
877 virtual ~OperandRenderer() {}
878
879 RendererKind getKind() const { return Kind; }
880
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000881 virtual void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const = 0;
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000882};
883
884/// A CopyRenderer emits code to copy a single operand from an existing
885/// instruction to the one being built.
886class CopyRenderer : public OperandRenderer {
887protected:
888 /// The matcher for the instruction that this operand is copied from.
889 /// This provides the facility for looking up an a operand by it's name so
890 /// that it can be used as a source for the instruction being built.
891 const InstructionMatcher &Matched;
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000892 /// The name of the operand.
893 const StringRef SymbolicName;
894
895public:
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000896 CopyRenderer(const InstructionMatcher &Matched, StringRef SymbolicName)
897 : OperandRenderer(OR_Copy), Matched(Matched), SymbolicName(SymbolicName) {
898 }
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000899
900 static bool classof(const OperandRenderer *R) {
901 return R->getKind() == OR_Copy;
902 }
903
904 const StringRef getSymbolicName() const { return SymbolicName; }
905
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000906 void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const override {
907 const OperandMatcher &Operand = Matched.getOperand(SymbolicName);
908 StringRef InsnVarName =
909 Rule.getInsnVarName(Operand.getInstructionMatcher());
910 std::string OperandExpr = Operand.getOperandExpr(InsnVarName);
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000911 OS << " MIB.add(" << OperandExpr << "/*" << SymbolicName << "*/);\n";
912 }
913};
914
915/// Adds a specific physical register to the instruction being built.
916/// This is typically useful for WZR/XZR on AArch64.
917class AddRegisterRenderer : public OperandRenderer {
918protected:
919 const Record *RegisterDef;
920
921public:
922 AddRegisterRenderer(const Record *RegisterDef)
923 : OperandRenderer(OR_Register), RegisterDef(RegisterDef) {}
924
925 static bool classof(const OperandRenderer *R) {
926 return R->getKind() == OR_Register;
927 }
928
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000929 void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const override {
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000930 OS << " MIB.addReg(" << RegisterDef->getValueAsString("Namespace")
931 << "::" << RegisterDef->getName() << ");\n";
932 }
933};
934
Daniel Sanders0ed28822017-04-12 08:23:08 +0000935/// Adds a specific immediate to the instruction being built.
936class ImmRenderer : public OperandRenderer {
937protected:
938 int64_t Imm;
939
940public:
941 ImmRenderer(int64_t Imm)
942 : OperandRenderer(OR_Imm), Imm(Imm) {}
943
944 static bool classof(const OperandRenderer *R) {
945 return R->getKind() == OR_Imm;
946 }
947
948 void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const override {
949 OS << " MIB.addImm(" << Imm << ");\n";
950 }
951};
952
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000953class RenderComplexPatternOperand : public OperandRenderer {
954private:
955 const Record &TheDef;
956 std::vector<OperandPlaceholder> Sources;
957
958 unsigned getNumOperands() const {
959 return TheDef.getValueAsDag("Operands")->getNumArgs();
960 }
961
962public:
963 RenderComplexPatternOperand(const Record &TheDef,
964 const ArrayRef<OperandPlaceholder> Sources)
965 : OperandRenderer(OR_ComplexPattern), TheDef(TheDef), Sources(Sources) {}
966
967 static bool classof(const OperandRenderer *R) {
968 return R->getKind() == OR_ComplexPattern;
969 }
970
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000971 void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const override {
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000972 assert(Sources.size() == getNumOperands() && "Inconsistent number of operands");
973 for (const auto &Source : Sources) {
974 OS << "MIB.add(";
975 Source.emitCxxValueExpr(OS);
976 OS << ");\n";
977 }
978 }
979};
980
Ahmed Bougacha56ca3a92017-02-04 00:47:10 +0000981/// An action taken when all Matcher predicates succeeded for a parent rule.
982///
983/// Typical actions include:
984/// * Changing the opcode of an instruction.
985/// * Adding an operand to an instruction.
Daniel Sanders43c882c2017-02-01 10:53:10 +0000986class MatchAction {
987public:
988 virtual ~MatchAction() {}
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000989
990 /// Emit the C++ statements to implement the action.
991 ///
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000992 /// \param RecycleVarName If given, it's an instruction to recycle. The
993 /// requirements on the instruction vary from action to
994 /// action.
995 virtual void emitCxxActionStmts(raw_ostream &OS, RuleMatcher &Rule,
996 StringRef RecycleVarName) const = 0;
Daniel Sanders43c882c2017-02-01 10:53:10 +0000997};
998
Ahmed Bougacha9aa4c102017-02-04 00:47:08 +0000999/// Generates a comment describing the matched rule being acted upon.
1000class DebugCommentAction : public MatchAction {
1001private:
1002 const PatternToMatch &P;
1003
1004public:
1005 DebugCommentAction(const PatternToMatch &P) : P(P) {}
1006
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001007 void emitCxxActionStmts(raw_ostream &OS, RuleMatcher &Rule,
1008 StringRef RecycleVarName) const override {
1009 OS << "// " << *P.getSrcPattern() << " => " << *P.getDstPattern() << "\n";
Ahmed Bougacha9aa4c102017-02-04 00:47:08 +00001010 }
1011};
1012
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001013/// Generates code to build an instruction or mutate an existing instruction
1014/// into the desired instruction when this is possible.
1015class BuildMIAction : public MatchAction {
Daniel Sanders43c882c2017-02-01 10:53:10 +00001016private:
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001017 const CodeGenInstruction *I;
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001018 const InstructionMatcher &Matched;
1019 std::vector<std::unique_ptr<OperandRenderer>> OperandRenderers;
1020
1021 /// True if the instruction can be built solely by mutating the opcode.
1022 bool canMutate() const {
1023 for (const auto &Renderer : enumerate(OperandRenderers)) {
Zachary Turner309a0882017-03-13 16:24:10 +00001024 if (const auto *Copy = dyn_cast<CopyRenderer>(&*Renderer.value())) {
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001025 if (Matched.getOperand(Copy->getSymbolicName()).getOperandIndex() !=
Zachary Turner309a0882017-03-13 16:24:10 +00001026 Renderer.index())
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001027 return false;
1028 } else
1029 return false;
1030 }
1031
1032 return true;
1033 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001034
Daniel Sanders43c882c2017-02-01 10:53:10 +00001035public:
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001036 BuildMIAction(const CodeGenInstruction *I, const InstructionMatcher &Matched)
1037 : I(I), Matched(Matched) {}
Daniel Sanders43c882c2017-02-01 10:53:10 +00001038
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001039 template <class Kind, class... Args>
1040 Kind &addRenderer(Args&&... args) {
1041 OperandRenderers.emplace_back(
1042 llvm::make_unique<Kind>(std::forward<Args>(args)...));
1043 return *static_cast<Kind *>(OperandRenderers.back().get());
1044 }
1045
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001046 void emitCxxActionStmts(raw_ostream &OS, RuleMatcher &Rule,
1047 StringRef RecycleVarName) const override {
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001048 if (canMutate()) {
Tim Northover4340d642017-03-20 21:58:23 +00001049 OS << " " << RecycleVarName << ".setDesc(TII.get(" << I->Namespace
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001050 << "::" << I->TheDef->getName() << "));\n";
Tim Northover4340d642017-03-20 21:58:23 +00001051
1052 if (!I->ImplicitDefs.empty() || !I->ImplicitUses.empty()) {
1053 OS << " auto MIB = MachineInstrBuilder(MF, &" << RecycleVarName
1054 << ");\n";
1055
1056 for (auto Def : I->ImplicitDefs) {
1057 auto Namespace = Def->getValueAsString("Namespace");
1058 OS << " MIB.addDef(" << Namespace << "::" << Def->getName()
1059 << ", RegState::Implicit);\n";
1060 }
1061 for (auto Use : I->ImplicitUses) {
1062 auto Namespace = Use->getValueAsString("Namespace");
1063 OS << " MIB.addUse(" << Namespace << "::" << Use->getName()
1064 << ", RegState::Implicit);\n";
1065 }
1066 }
1067
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001068 OS << " MachineInstr &NewI = " << RecycleVarName << ";\n";
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001069 return;
1070 }
1071
1072 // TODO: Simple permutation looks like it could be almost as common as
1073 // mutation due to commutative operations.
1074
1075 OS << "MachineInstrBuilder MIB = BuildMI(*I.getParent(), I, "
1076 "I.getDebugLoc(), TII.get("
1077 << I->Namespace << "::" << I->TheDef->getName() << "));\n";
1078 for (const auto &Renderer : OperandRenderers)
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001079 Renderer->emitCxxRenderStmts(OS, Rule);
Daniel Sandersbee57392017-04-04 13:25:23 +00001080 OS << " for (const auto *FromMI : ";
1081 Rule.emitCxxCapturedInsnList(OS);
1082 OS << ")\n";
1083 OS << " for (const auto &MMO : FromMI->memoperands())\n";
1084 OS << " MIB.addMemOperand(MMO);\n";
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001085 OS << " " << RecycleVarName << ".eraseFromParent();\n";
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001086 OS << " MachineInstr &NewI = *MIB;\n";
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001087 }
1088};
1089
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001090InstructionMatcher &RuleMatcher::addInstructionMatcher() {
1091 Matchers.emplace_back(new InstructionMatcher());
1092 return *Matchers.back();
1093}
Ahmed Bougacha56ca3a92017-02-04 00:47:10 +00001094
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001095template <class Kind, class... Args>
1096Kind &RuleMatcher::addAction(Args &&... args) {
1097 Actions.emplace_back(llvm::make_unique<Kind>(std::forward<Args>(args)...));
1098 return *static_cast<Kind *>(Actions.back().get());
1099}
Daniel Sandersdc662ff2017-01-26 11:10:14 +00001100
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001101std::string RuleMatcher::defineInsnVar(raw_ostream &OS,
1102 const InstructionMatcher &Matcher,
1103 StringRef Value) {
1104 std::string InsnVarName = "MI" + llvm::to_string(NextInsnVarID++);
1105 OS << "MachineInstr &" << InsnVarName << " = " << Value << ";\n";
1106 InsnVariableNames[&Matcher] = InsnVarName;
1107 return InsnVarName;
1108}
1109
1110StringRef RuleMatcher::getInsnVarName(const InstructionMatcher &InsnMatcher) const {
1111 const auto &I = InsnVariableNames.find(&InsnMatcher);
1112 if (I != InsnVariableNames.end())
1113 return I->second;
1114 llvm_unreachable("Matched Insn was not captured in a local variable");
1115}
1116
Daniel Sandersbee57392017-04-04 13:25:23 +00001117/// Emit a C++ initializer_list containing references to every matched instruction.
1118void RuleMatcher::emitCxxCapturedInsnList(raw_ostream &OS) {
Daniel Sanders9e4817d2017-04-04 14:27:06 +00001119 SmallVector<StringRef, 2> Names;
Daniel Sandersbee57392017-04-04 13:25:23 +00001120 for (const auto &Pair : InsnVariableNames)
Daniel Sanders9e4817d2017-04-04 14:27:06 +00001121 Names.push_back(Pair.second);
1122 std::sort(Names.begin(), Names.end());
1123
1124 OS << "{";
1125 for (const auto &Name : Names)
1126 OS << "&" << Name << ", ";
Daniel Sandersbee57392017-04-04 13:25:23 +00001127 OS << "}";
1128}
1129
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001130/// Emit C++ statements to check the shape of the match and capture
1131/// instructions into local variables.
1132void RuleMatcher::emitCxxCaptureStmts(raw_ostream &OS, StringRef Expr) {
1133 assert(Matchers.size() == 1 && "Cannot handle multi-root matchers yet");
1134 std::string InsnVarName = defineInsnVar(OS, *Matchers.front(), Expr);
1135 Matchers.front()->emitCxxCaptureStmts(OS, *this, InsnVarName);
1136}
1137
1138void RuleMatcher::emit(raw_ostream &OS) {
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001139 if (Matchers.empty())
1140 llvm_unreachable("Unexpected empty matcher!");
Daniel Sandersdc662ff2017-01-26 11:10:14 +00001141
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001142 // The representation supports rules that require multiple roots such as:
1143 // %ptr(p0) = ...
1144 // %elt0(s32) = G_LOAD %ptr
1145 // %1(p0) = G_ADD %ptr, 4
1146 // %elt1(s32) = G_LOAD p0 %1
1147 // which could be usefully folded into:
1148 // %ptr(p0) = ...
1149 // %elt0(s32), %elt1(s32) = TGT_LOAD_PAIR %ptr
1150 // on some targets but we don't need to make use of that yet.
1151 assert(Matchers.size() == 1 && "Cannot handle multi-root matchers yet");
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001152 OS << "if ([&]() {\n";
1153
1154 emitCxxCaptureStmts(OS, "I");
1155
1156 OS << " if (";
1157 Matchers.front()->emitCxxPredicateExpr(OS, *this,
1158 getInsnVarName(*Matchers.front()));
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001159 OS << ") {\n";
1160
Daniel Sandersbee57392017-04-04 13:25:23 +00001161 // We must also check if it's safe to fold the matched instructions.
1162 if (InsnVariableNames.size() >= 2) {
1163 for (const auto &Pair : InsnVariableNames) {
1164 // Skip the root node since it isn't moving anywhere. Everything else is
1165 // sinking to meet it.
1166 if (Pair.first == Matchers.front().get())
1167 continue;
1168
1169 // Reject the difficult cases until we have a more accurate check.
1170 OS << " if (!isObviouslySafeToFold(" << Pair.second
1171 << ")) return false;\n";
1172
1173 // FIXME: Emit checks to determine it's _actually_ safe to fold and/or
1174 // account for unsafe cases.
1175 //
1176 // Example:
1177 // MI1--> %0 = ...
1178 // %1 = ... %0
1179 // MI0--> %2 = ... %0
1180 // It's not safe to erase MI1. We currently handle this by not
1181 // erasing %0 (even when it's dead).
1182 //
1183 // Example:
1184 // MI1--> %0 = load volatile @a
1185 // %1 = load volatile @a
1186 // MI0--> %2 = ... %0
1187 // It's not safe to sink %0's def past %1. We currently handle
1188 // this by rejecting all loads.
1189 //
1190 // Example:
1191 // MI1--> %0 = load @a
1192 // %1 = store @a
1193 // MI0--> %2 = ... %0
1194 // It's not safe to sink %0's def past %1. We currently handle
1195 // this by rejecting all loads.
1196 //
1197 // Example:
1198 // G_CONDBR %cond, @BB1
1199 // BB0:
1200 // MI1--> %0 = load @a
1201 // G_BR @BB1
1202 // BB1:
1203 // MI0--> %2 = ... %0
1204 // It's not always safe to sink %0 across control flow. In this
1205 // case it may introduce a memory fault. We currentl handle this
1206 // by rejecting all loads.
1207 }
1208 }
1209
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001210 for (const auto &MA : Actions) {
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001211 MA->emitCxxActionStmts(OS, *this, "I");
Daniel Sandersdc662ff2017-01-26 11:10:14 +00001212 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001213
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001214 OS << " constrainSelectedInstRegOperands(NewI, TII, TRI, RBI);\n";
1215 OS << " return true;\n";
1216 OS << " }\n";
1217 OS << " return false;\n";
1218 OS << " }()) { return true; }\n\n";
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001219}
Daniel Sanders43c882c2017-02-01 10:53:10 +00001220
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001221bool RuleMatcher::isHigherPriorityThan(const RuleMatcher &B) const {
1222 // Rules involving more match roots have higher priority.
1223 if (Matchers.size() > B.Matchers.size())
1224 return true;
1225 if (Matchers.size() < B.Matchers.size())
Daniel Sanders759ff412017-02-24 13:58:11 +00001226 return false;
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001227
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001228 for (const auto &Matcher : zip(Matchers, B.Matchers)) {
1229 if (std::get<0>(Matcher)->isHigherPriorityThan(*std::get<1>(Matcher)))
1230 return true;
1231 if (std::get<1>(Matcher)->isHigherPriorityThan(*std::get<0>(Matcher)))
1232 return false;
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001233 }
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001234
1235 return false;
Simon Pilgrima7d1da82017-03-15 22:50:47 +00001236}
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001237
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001238unsigned RuleMatcher::countTemporaryOperands() const {
1239 return std::accumulate(
1240 Matchers.begin(), Matchers.end(), 0,
1241 [](unsigned A, const std::unique_ptr<InstructionMatcher> &Matcher) {
1242 return A + Matcher->countTemporaryOperands();
1243 });
1244}
1245
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001246//===- GlobalISelEmitter class --------------------------------------------===//
1247
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001248class GlobalISelEmitter {
1249public:
1250 explicit GlobalISelEmitter(RecordKeeper &RK);
1251 void run(raw_ostream &OS);
1252
1253private:
1254 const RecordKeeper &RK;
1255 const CodeGenDAGPatterns CGP;
1256 const CodeGenTarget &Target;
1257
1258 /// Keep track of the equivalence between SDNodes and Instruction.
1259 /// This is defined using 'GINodeEquiv' in the target description.
1260 DenseMap<Record *, const CodeGenInstruction *> NodeEquivs;
1261
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001262 /// Keep track of the equivalence between ComplexPattern's and
1263 /// GIComplexOperandMatcher. Map entries are specified by subclassing
1264 /// GIComplexPatternEquiv.
1265 DenseMap<const Record *, const Record *> ComplexPatternEquivs;
1266
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001267 void gatherNodeEquivs();
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001268 const CodeGenInstruction *findNodeEquiv(Record *N) const;
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001269
Daniel Sandersc270c502017-03-30 09:36:33 +00001270 Error importRulePredicates(RuleMatcher &M, ArrayRef<Init *> Predicates) const;
Daniel Sandersffc7d582017-03-29 15:37:18 +00001271 Expected<InstructionMatcher &>
Daniel Sandersc270c502017-03-30 09:36:33 +00001272 createAndImportSelDAGMatcher(InstructionMatcher &InsnMatcher,
1273 const TreePatternNode *Src) const;
1274 Error importChildMatcher(InstructionMatcher &InsnMatcher,
1275 TreePatternNode *SrcChild, unsigned OpIdx,
1276 unsigned &TempOpIdx) const;
1277 Expected<BuildMIAction &> createAndImportInstructionRenderer(
1278 RuleMatcher &M, const TreePatternNode *Dst,
1279 const InstructionMatcher &InsnMatcher) const;
1280 Error importExplicitUseRenderer(BuildMIAction &DstMIBuilder,
1281 TreePatternNode *DstChild,
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001282 const InstructionMatcher &InsnMatcher) const;
Daniel Sandersc270c502017-03-30 09:36:33 +00001283 Error
Daniel Sandersffc7d582017-03-29 15:37:18 +00001284 importImplicitDefRenderers(BuildMIAction &DstMIBuilder,
1285 const std::vector<Record *> &ImplicitDefs) const;
1286
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001287 /// Analyze pattern \p P, returning a matcher for it if possible.
1288 /// Otherwise, return an Error explaining why we don't support it.
1289 Expected<RuleMatcher> runOnPattern(const PatternToMatch &P);
1290};
1291
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001292void GlobalISelEmitter::gatherNodeEquivs() {
1293 assert(NodeEquivs.empty());
1294 for (Record *Equiv : RK.getAllDerivedDefinitions("GINodeEquiv"))
1295 NodeEquivs[Equiv->getValueAsDef("Node")] =
1296 &Target.getInstruction(Equiv->getValueAsDef("I"));
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001297
1298 assert(ComplexPatternEquivs.empty());
1299 for (Record *Equiv : RK.getAllDerivedDefinitions("GIComplexPatternEquiv")) {
1300 Record *SelDAGEquiv = Equiv->getValueAsDef("SelDAGEquivalent");
1301 if (!SelDAGEquiv)
1302 continue;
1303 ComplexPatternEquivs[SelDAGEquiv] = Equiv;
1304 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001305}
1306
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001307const CodeGenInstruction *GlobalISelEmitter::findNodeEquiv(Record *N) const {
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001308 return NodeEquivs.lookup(N);
1309}
1310
1311GlobalISelEmitter::GlobalISelEmitter(RecordKeeper &RK)
1312 : RK(RK), CGP(RK), Target(CGP.getTargetInfo()) {}
1313
1314//===- Emitter ------------------------------------------------------------===//
1315
Daniel Sandersc270c502017-03-30 09:36:33 +00001316Error
Daniel Sandersffc7d582017-03-29 15:37:18 +00001317GlobalISelEmitter::importRulePredicates(RuleMatcher &M,
1318 ArrayRef<Init *> Predicates) const {
1319 if (!Predicates.empty())
Daniel Sandersd0656a32017-04-13 09:45:37 +00001320 return failedImport("Pattern has a rule predicate (" +
1321 explainRulePredicates(Predicates) + ")");
Daniel Sandersc270c502017-03-30 09:36:33 +00001322 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001323}
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001324
Daniel Sandersc270c502017-03-30 09:36:33 +00001325Expected<InstructionMatcher &> GlobalISelEmitter::createAndImportSelDAGMatcher(
1326 InstructionMatcher &InsnMatcher, const TreePatternNode *Src) const {
Daniel Sandersffc7d582017-03-29 15:37:18 +00001327 // Start with the defined operands (i.e., the results of the root operator).
1328 if (Src->getExtTypes().size() > 1)
1329 return failedImport("Src pattern has multiple results");
1330
1331 auto SrcGIOrNull = findNodeEquiv(Src->getOperator());
1332 if (!SrcGIOrNull)
Daniel Sandersd0656a32017-04-13 09:45:37 +00001333 return failedImport("Pattern operator lacks an equivalent Instruction" +
1334 explainOperator(Src->getOperator()));
Daniel Sandersffc7d582017-03-29 15:37:18 +00001335 auto &SrcGI = *SrcGIOrNull;
1336
1337 // The operators look good: match the opcode and mutate it to the new one.
1338 InsnMatcher.addPredicate<InstructionOpcodeMatcher>(&SrcGI);
1339
1340 unsigned OpIdx = 0;
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001341 unsigned TempOpIdx = 0;
Daniel Sandersffc7d582017-03-29 15:37:18 +00001342 for (const EEVT::TypeSet &Ty : Src->getExtTypes()) {
1343 auto OpTyOrNone = MVTToLLT(Ty.getConcrete());
1344
1345 if (!OpTyOrNone)
1346 return failedImport(
1347 "Result of Src pattern operator has an unsupported type");
1348
1349 // Results don't have a name unless they are the root node. The caller will
1350 // set the name if appropriate.
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001351 OperandMatcher &OM = InsnMatcher.addOperand(OpIdx++, "", TempOpIdx);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001352 OM.addPredicate<LLTOperandMatcher>(*OpTyOrNone);
1353 }
1354
Daniel Sandersffc7d582017-03-29 15:37:18 +00001355 // Match the used operands (i.e. the children of the operator).
1356 for (unsigned i = 0, e = Src->getNumChildren(); i != e; ++i) {
1357 if (auto Error = importChildMatcher(InsnMatcher, Src->getChild(i), OpIdx++,
Daniel Sandersc270c502017-03-30 09:36:33 +00001358 TempOpIdx))
Daniel Sandersffc7d582017-03-29 15:37:18 +00001359 return std::move(Error);
1360 }
1361
1362 return InsnMatcher;
1363}
1364
Daniel Sandersc270c502017-03-30 09:36:33 +00001365Error GlobalISelEmitter::importChildMatcher(InstructionMatcher &InsnMatcher,
1366 TreePatternNode *SrcChild,
1367 unsigned OpIdx,
1368 unsigned &TempOpIdx) const {
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001369 OperandMatcher &OM =
1370 InsnMatcher.addOperand(OpIdx, SrcChild->getName(), TempOpIdx);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001371
1372 if (SrcChild->hasAnyPredicate())
Daniel Sandersd0656a32017-04-13 09:45:37 +00001373 return failedImport("Src pattern child has predicate (" +
1374 explainPredicates(SrcChild) + ")");
Daniel Sandersffc7d582017-03-29 15:37:18 +00001375
1376 ArrayRef<EEVT::TypeSet> ChildTypes = SrcChild->getExtTypes();
1377 if (ChildTypes.size() != 1)
1378 return failedImport("Src pattern child has multiple results");
1379
1380 // Check MBB's before the type check since they are not a known type.
1381 if (!SrcChild->isLeaf()) {
1382 if (SrcChild->getOperator()->isSubClassOf("SDNode")) {
1383 auto &ChildSDNI = CGP.getSDNodeInfo(SrcChild->getOperator());
1384 if (ChildSDNI.getSDClassName() == "BasicBlockSDNode") {
1385 OM.addPredicate<MBBOperandMatcher>();
Daniel Sandersc270c502017-03-30 09:36:33 +00001386 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001387 }
1388 }
Daniel Sandersffc7d582017-03-29 15:37:18 +00001389 }
1390
1391 auto OpTyOrNone = MVTToLLT(ChildTypes.front().getConcrete());
1392 if (!OpTyOrNone)
1393 return failedImport("Src operand has an unsupported type");
1394 OM.addPredicate<LLTOperandMatcher>(*OpTyOrNone);
1395
Daniel Sandersbee57392017-04-04 13:25:23 +00001396 // Check for nested instructions.
1397 if (!SrcChild->isLeaf()) {
1398 // Map the node to a gMIR instruction.
1399 InstructionOperandMatcher &InsnOperand =
1400 OM.addPredicate<InstructionOperandMatcher>();
1401 auto InsnMatcherOrError =
1402 createAndImportSelDAGMatcher(InsnOperand.getInsnMatcher(), SrcChild);
1403 if (auto Error = InsnMatcherOrError.takeError())
1404 return Error;
1405
1406 return Error::success();
1407 }
1408
Daniel Sandersffc7d582017-03-29 15:37:18 +00001409 // Check for constant immediates.
1410 if (auto *ChildInt = dyn_cast<IntInit>(SrcChild->getLeafValue())) {
1411 OM.addPredicate<IntOperandMatcher>(ChildInt->getValue());
Daniel Sandersc270c502017-03-30 09:36:33 +00001412 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001413 }
1414
1415 // Check for def's like register classes or ComplexPattern's.
1416 if (auto *ChildDefInit = dyn_cast<DefInit>(SrcChild->getLeafValue())) {
1417 auto *ChildRec = ChildDefInit->getDef();
1418
1419 // Check for register classes.
1420 if (ChildRec->isSubClassOf("RegisterClass")) {
1421 OM.addPredicate<RegisterBankOperandMatcher>(
1422 Target.getRegisterClass(ChildRec));
Daniel Sandersc270c502017-03-30 09:36:33 +00001423 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001424 }
1425
1426 // Check for ComplexPattern's.
1427 if (ChildRec->isSubClassOf("ComplexPattern")) {
1428 const auto &ComplexPattern = ComplexPatternEquivs.find(ChildRec);
1429 if (ComplexPattern == ComplexPatternEquivs.end())
Daniel Sandersd0656a32017-04-13 09:45:37 +00001430 return failedImport("SelectionDAG ComplexPattern (" +
1431 ChildRec->getName() + ") not mapped to GlobalISel");
Daniel Sandersffc7d582017-03-29 15:37:18 +00001432
1433 const auto &Predicate = OM.addPredicate<ComplexPatternOperandMatcher>(
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001434 OM, *ComplexPattern->second);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001435 TempOpIdx += Predicate.countTemporaryOperands();
Daniel Sandersc270c502017-03-30 09:36:33 +00001436 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001437 }
1438
Daniel Sandersd0656a32017-04-13 09:45:37 +00001439 if (ChildRec->isSubClassOf("ImmLeaf")) {
1440 return failedImport(
1441 "Src pattern child def is an unsupported tablegen class (ImmLeaf)");
1442 }
1443
Daniel Sandersffc7d582017-03-29 15:37:18 +00001444 return failedImport(
1445 "Src pattern child def is an unsupported tablegen class");
1446 }
1447
1448 return failedImport("Src pattern child is an unsupported kind");
1449}
1450
Daniel Sandersc270c502017-03-30 09:36:33 +00001451Error GlobalISelEmitter::importExplicitUseRenderer(
Daniel Sandersffc7d582017-03-29 15:37:18 +00001452 BuildMIAction &DstMIBuilder, TreePatternNode *DstChild,
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001453 const InstructionMatcher &InsnMatcher) const {
Daniel Sandersffc7d582017-03-29 15:37:18 +00001454 // The only non-leaf child we accept is 'bb': it's an operator because
1455 // BasicBlockSDNode isn't inline, but in MI it's just another operand.
1456 if (!DstChild->isLeaf()) {
1457 if (DstChild->getOperator()->isSubClassOf("SDNode")) {
1458 auto &ChildSDNI = CGP.getSDNodeInfo(DstChild->getOperator());
1459 if (ChildSDNI.getSDClassName() == "BasicBlockSDNode") {
1460 DstMIBuilder.addRenderer<CopyRenderer>(InsnMatcher,
1461 DstChild->getName());
Daniel Sandersc270c502017-03-30 09:36:33 +00001462 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001463 }
1464 }
1465 return failedImport("Dst pattern child isn't a leaf node or an MBB");
1466 }
1467
1468 // Otherwise, we're looking for a bog-standard RegisterClass operand.
1469 if (DstChild->hasAnyPredicate())
Daniel Sandersd0656a32017-04-13 09:45:37 +00001470 return failedImport("Dst pattern child has predicate (" +
1471 explainPredicates(DstChild) + ")");
Daniel Sandersffc7d582017-03-29 15:37:18 +00001472
1473 if (auto *ChildDefInit = dyn_cast<DefInit>(DstChild->getLeafValue())) {
1474 auto *ChildRec = ChildDefInit->getDef();
1475
1476 ArrayRef<EEVT::TypeSet> ChildTypes = DstChild->getExtTypes();
1477 if (ChildTypes.size() != 1)
1478 return failedImport("Dst pattern child has multiple results");
1479
1480 auto OpTyOrNone = MVTToLLT(ChildTypes.front().getConcrete());
1481 if (!OpTyOrNone)
1482 return failedImport("Dst operand has an unsupported type");
1483
1484 if (ChildRec->isSubClassOf("Register")) {
1485 DstMIBuilder.addRenderer<AddRegisterRenderer>(ChildRec);
Daniel Sandersc270c502017-03-30 09:36:33 +00001486 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001487 }
1488
1489 if (ChildRec->isSubClassOf("RegisterClass")) {
1490 DstMIBuilder.addRenderer<CopyRenderer>(InsnMatcher, DstChild->getName());
Daniel Sandersc270c502017-03-30 09:36:33 +00001491 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001492 }
1493
1494 if (ChildRec->isSubClassOf("ComplexPattern")) {
1495 const auto &ComplexPattern = ComplexPatternEquivs.find(ChildRec);
1496 if (ComplexPattern == ComplexPatternEquivs.end())
1497 return failedImport(
1498 "SelectionDAG ComplexPattern not mapped to GlobalISel");
1499
1500 SmallVector<OperandPlaceholder, 2> RenderedOperands;
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001501 const OperandMatcher &OM = InsnMatcher.getOperand(DstChild->getName());
1502 for (unsigned I = 0; I < OM.countTemporaryOperands(); ++I)
1503 RenderedOperands.push_back(OperandPlaceholder::CreateTemporary(
1504 OM.getAllocatedTemporariesBaseID() + I));
Daniel Sandersffc7d582017-03-29 15:37:18 +00001505 DstMIBuilder.addRenderer<RenderComplexPatternOperand>(
1506 *ComplexPattern->second, RenderedOperands);
Daniel Sandersc270c502017-03-30 09:36:33 +00001507 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001508 }
1509
Daniel Sandersd0656a32017-04-13 09:45:37 +00001510 if (ChildRec->isSubClassOf("SDNodeXForm"))
1511 return failedImport("Dst pattern child def is an unsupported tablegen "
1512 "class (SDNodeXForm)");
1513
Daniel Sandersffc7d582017-03-29 15:37:18 +00001514 return failedImport(
1515 "Dst pattern child def is an unsupported tablegen class");
1516 }
1517
1518 return failedImport("Dst pattern child is an unsupported kind");
1519}
1520
Daniel Sandersc270c502017-03-30 09:36:33 +00001521Expected<BuildMIAction &> GlobalISelEmitter::createAndImportInstructionRenderer(
Daniel Sandersffc7d582017-03-29 15:37:18 +00001522 RuleMatcher &M, const TreePatternNode *Dst,
1523 const InstructionMatcher &InsnMatcher) const {
1524 Record *DstOp = Dst->getOperator();
Daniel Sandersd0656a32017-04-13 09:45:37 +00001525 if (!DstOp->isSubClassOf("Instruction")) {
1526 if (DstOp->isSubClassOf("ValueType"))
1527 return failedImport(
1528 "Pattern operator isn't an instruction (it's a ValueType)");
Daniel Sandersffc7d582017-03-29 15:37:18 +00001529 return failedImport("Pattern operator isn't an instruction");
Daniel Sandersd0656a32017-04-13 09:45:37 +00001530 }
Daniel Sandersffc7d582017-03-29 15:37:18 +00001531 auto &DstI = Target.getInstruction(DstOp);
1532
1533 auto &DstMIBuilder = M.addAction<BuildMIAction>(&DstI, InsnMatcher);
1534
1535 // Render the explicit defs.
1536 for (unsigned I = 0; I < DstI.Operands.NumDefs; ++I) {
1537 const auto &DstIOperand = DstI.Operands[I];
1538 DstMIBuilder.addRenderer<CopyRenderer>(InsnMatcher, DstIOperand.Name);
1539 }
1540
Daniel Sanders0ed28822017-04-12 08:23:08 +00001541 // Figure out which operands need defaults inserted. Operands that subclass
1542 // OperandWithDefaultOps are considered from left to right until we have
1543 // enough operands to render the instruction.
1544 SmallSet<unsigned, 2> DefaultOperands;
1545 unsigned DstINumUses = DstI.Operands.size() - DstI.Operands.NumDefs;
1546 unsigned NumDefaultOperands = 0;
1547 for (unsigned I = 0; I < DstINumUses &&
1548 DstINumUses > Dst->getNumChildren() + NumDefaultOperands;
1549 ++I) {
1550 const auto &DstIOperand = DstI.Operands[DstI.Operands.NumDefs + I];
1551 if (DstIOperand.Rec->isSubClassOf("OperandWithDefaultOps")) {
1552 DefaultOperands.insert(I);
1553 NumDefaultOperands +=
1554 DstIOperand.Rec->getValueAsDag("DefaultOps")->getNumArgs();
1555 }
1556 }
1557 if (DstINumUses > Dst->getNumChildren() + DefaultOperands.size())
1558 return failedImport("Insufficient operands supplied and default ops "
1559 "couldn't make up the shortfall");
1560 if (DstINumUses < Dst->getNumChildren() + DefaultOperands.size())
1561 return failedImport("Too many operands supplied");
1562
Daniel Sandersffc7d582017-03-29 15:37:18 +00001563 // Render the explicit uses.
Daniel Sanders0ed28822017-04-12 08:23:08 +00001564 unsigned Child = 0;
1565 for (unsigned I = 0; I != DstINumUses; ++I) {
1566 // If we need to insert default ops here, then do so.
1567 if (DefaultOperands.count(I)) {
1568 const auto &DstIOperand = DstI.Operands[DstI.Operands.NumDefs + I];
1569
1570 DagInit *DefaultOps = DstIOperand.Rec->getValueAsDag("DefaultOps");
1571 for (const auto *DefaultOp : DefaultOps->args()) {
1572 // Look through ValueType operators.
1573 if (const DagInit *DefaultDagOp = dyn_cast<DagInit>(DefaultOp)) {
1574 if (const DefInit *DefaultDagOperator =
1575 dyn_cast<DefInit>(DefaultDagOp->getOperator())) {
1576 if (DefaultDagOperator->getDef()->isSubClassOf("ValueType"))
1577 DefaultOp = DefaultDagOp->getArg(0);
1578 }
1579 }
1580
1581 if (const DefInit *DefaultDefOp = dyn_cast<DefInit>(DefaultOp)) {
1582 DstMIBuilder.addRenderer<AddRegisterRenderer>(DefaultDefOp->getDef());
1583 continue;
1584 }
1585
1586 if (const IntInit *DefaultIntOp = dyn_cast<IntInit>(DefaultOp)) {
1587 DstMIBuilder.addRenderer<ImmRenderer>(DefaultIntOp->getValue());
1588 continue;
1589 }
1590
1591 return failedImport("Could not add default op");
1592 }
1593
1594 continue;
1595 }
1596
1597 if (auto Error = importExplicitUseRenderer(
1598 DstMIBuilder, Dst->getChild(Child), InsnMatcher))
Daniel Sandersffc7d582017-03-29 15:37:18 +00001599 return std::move(Error);
Daniel Sanders0ed28822017-04-12 08:23:08 +00001600 ++Child;
Daniel Sandersffc7d582017-03-29 15:37:18 +00001601 }
1602
1603 return DstMIBuilder;
1604}
1605
Daniel Sandersc270c502017-03-30 09:36:33 +00001606Error GlobalISelEmitter::importImplicitDefRenderers(
Daniel Sandersffc7d582017-03-29 15:37:18 +00001607 BuildMIAction &DstMIBuilder,
1608 const std::vector<Record *> &ImplicitDefs) const {
1609 if (!ImplicitDefs.empty())
1610 return failedImport("Pattern defines a physical register");
Daniel Sandersc270c502017-03-30 09:36:33 +00001611 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001612}
1613
1614Expected<RuleMatcher> GlobalISelEmitter::runOnPattern(const PatternToMatch &P) {
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001615 // Keep track of the matchers and actions to emit.
Ahmed Bougacha9aa4c102017-02-04 00:47:08 +00001616 RuleMatcher M;
1617 M.addAction<DebugCommentAction>(P);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001618
Daniel Sandersc270c502017-03-30 09:36:33 +00001619 if (auto Error = importRulePredicates(M, P.getPredicates()->getValues()))
Daniel Sandersffc7d582017-03-29 15:37:18 +00001620 return std::move(Error);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001621
1622 // Next, analyze the pattern operators.
1623 TreePatternNode *Src = P.getSrcPattern();
1624 TreePatternNode *Dst = P.getDstPattern();
1625
1626 // If the root of either pattern isn't a simple operator, ignore it.
Daniel Sandersd0656a32017-04-13 09:45:37 +00001627 if (auto Err = isTrivialOperatorNode(Dst))
1628 return failedImport("Dst pattern root isn't a trivial operator (" +
1629 toString(std::move(Err)) + ")");
1630 if (auto Err = isTrivialOperatorNode(Src))
1631 return failedImport("Src pattern root isn't a trivial operator (" +
1632 toString(std::move(Err)) + ")");
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001633
Daniel Sandersbee57392017-04-04 13:25:23 +00001634 // Start with the defined operands (i.e., the results of the root operator).
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001635 Record *DstOp = Dst->getOperator();
1636 if (!DstOp->isSubClassOf("Instruction"))
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001637 return failedImport("Pattern operator isn't an instruction");
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001638
1639 auto &DstI = Target.getInstruction(DstOp);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001640 if (DstI.Operands.NumDefs != Src->getExtTypes().size())
Daniel Sandersd0656a32017-04-13 09:45:37 +00001641 return failedImport("Src pattern results and dst MI defs are different (" +
1642 to_string(Src->getExtTypes().size()) + " def(s) vs " +
1643 to_string(DstI.Operands.NumDefs) + " def(s))");
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001644
Daniel Sandersffc7d582017-03-29 15:37:18 +00001645 InstructionMatcher &InsnMatcherTemp = M.addInstructionMatcher();
Daniel Sandersc270c502017-03-30 09:36:33 +00001646 auto InsnMatcherOrError = createAndImportSelDAGMatcher(InsnMatcherTemp, Src);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001647 if (auto Error = InsnMatcherOrError.takeError())
1648 return std::move(Error);
1649 InstructionMatcher &InsnMatcher = InsnMatcherOrError.get();
1650
1651 // The root of the match also has constraints on the register bank so that it
1652 // matches the result instruction.
1653 unsigned OpIdx = 0;
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001654 for (const EEVT::TypeSet &Ty : Src->getExtTypes()) {
Daniel Sandersffc7d582017-03-29 15:37:18 +00001655 (void)Ty;
1656
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001657 const auto &DstIOperand = DstI.Operands[OpIdx];
1658 Record *DstIOpRec = DstIOperand.Rec;
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001659 if (!DstIOpRec->isSubClassOf("RegisterClass"))
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001660 return failedImport("Dst MI def isn't a register class");
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001661
Daniel Sandersffc7d582017-03-29 15:37:18 +00001662 OperandMatcher &OM = InsnMatcher.getOperand(OpIdx);
1663 OM.setSymbolicName(DstIOperand.Name);
Daniel Sandersdc662ff2017-01-26 11:10:14 +00001664 OM.addPredicate<RegisterBankOperandMatcher>(
1665 Target.getRegisterClass(DstIOpRec));
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001666 ++OpIdx;
1667 }
1668
Daniel Sandersc270c502017-03-30 09:36:33 +00001669 auto DstMIBuilderOrError =
1670 createAndImportInstructionRenderer(M, Dst, InsnMatcher);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001671 if (auto Error = DstMIBuilderOrError.takeError())
1672 return std::move(Error);
1673 BuildMIAction &DstMIBuilder = DstMIBuilderOrError.get();
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001674
Daniel Sandersffc7d582017-03-29 15:37:18 +00001675 // Render the implicit defs.
1676 // These are only added to the root of the result.
Daniel Sandersc270c502017-03-30 09:36:33 +00001677 if (auto Error = importImplicitDefRenderers(DstMIBuilder, P.getDstRegs()))
Daniel Sandersffc7d582017-03-29 15:37:18 +00001678 return std::move(Error);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001679
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001680 // We're done with this pattern! It's eligible for GISel emission; return it.
Daniel Sandersb41ce2b2017-02-20 14:31:27 +00001681 ++NumPatternImported;
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001682 return std::move(M);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001683}
1684
1685void GlobalISelEmitter::run(raw_ostream &OS) {
1686 // Track the GINodeEquiv definitions.
1687 gatherNodeEquivs();
1688
1689 emitSourceFileHeader(("Global Instruction Selector for the " +
1690 Target.getName() + " target").str(), OS);
Daniel Sandersb41ce2b2017-02-20 14:31:27 +00001691 std::vector<RuleMatcher> Rules;
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001692 // Look through the SelectionDAG patterns we found, possibly emitting some.
1693 for (const PatternToMatch &Pat : CGP.ptms()) {
1694 ++NumPatternTotal;
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001695 auto MatcherOrErr = runOnPattern(Pat);
1696
1697 // The pattern analysis can fail, indicating an unsupported pattern.
1698 // Report that if we've been asked to do so.
1699 if (auto Err = MatcherOrErr.takeError()) {
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001700 if (WarnOnSkippedPatterns) {
1701 PrintWarning(Pat.getSrcRecord()->getLoc(),
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001702 "Skipped pattern: " + toString(std::move(Err)));
1703 } else {
1704 consumeError(std::move(Err));
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001705 }
Daniel Sandersb41ce2b2017-02-20 14:31:27 +00001706 ++NumPatternImportsSkipped;
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001707 continue;
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001708 }
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001709
Daniel Sandersb41ce2b2017-02-20 14:31:27 +00001710 Rules.push_back(std::move(MatcherOrErr.get()));
1711 }
1712
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001713 std::stable_sort(Rules.begin(), Rules.end(),
1714 [&](const RuleMatcher &A, const RuleMatcher &B) {
Daniel Sanders759ff412017-02-24 13:58:11 +00001715 if (A.isHigherPriorityThan(B)) {
1716 assert(!B.isHigherPriorityThan(A) && "Cannot be more important "
1717 "and less important at "
1718 "the same time");
1719 return true;
1720 }
1721 return false;
1722 });
1723
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001724 unsigned MaxTemporaries = 0;
1725 for (const auto &Rule : Rules)
1726 MaxTemporaries = std::max(MaxTemporaries, Rule.countTemporaryOperands());
1727
1728 OS << "#ifdef GET_GLOBALISEL_TEMPORARIES_DECL\n";
1729 for (unsigned I = 0; I < MaxTemporaries; ++I)
1730 OS << " mutable MachineOperand TempOp" << I << ";\n";
1731 OS << "#endif // ifdef GET_GLOBALISEL_TEMPORARIES_DECL\n\n";
1732
1733 OS << "#ifdef GET_GLOBALISEL_TEMPORARIES_INIT\n";
1734 for (unsigned I = 0; I < MaxTemporaries; ++I)
1735 OS << ", TempOp" << I << "(MachineOperand::CreatePlaceholder())\n";
1736 OS << "#endif // ifdef GET_GLOBALISEL_TEMPORARIES_INIT\n\n";
1737
1738 OS << "#ifdef GET_GLOBALISEL_IMPL\n"
1739 << "bool " << Target.getName()
1740 << "InstructionSelector::selectImpl(MachineInstr &I) const {\n"
1741 << " MachineFunction &MF = *I.getParent()->getParent();\n"
1742 << " const MachineRegisterInfo &MRI = MF.getRegInfo();\n";
1743
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001744 for (auto &Rule : Rules) {
Daniel Sandersb41ce2b2017-02-20 14:31:27 +00001745 Rule.emit(OS);
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001746 ++NumPatternEmitted;
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001747 }
1748
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001749 OS << " return false;\n"
1750 << "}\n"
1751 << "#endif // ifdef GET_GLOBALISEL_IMPL\n";
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001752}
1753
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001754} // end anonymous namespace
1755
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001756//===----------------------------------------------------------------------===//
1757
1758namespace llvm {
1759void EmitGlobalISel(RecordKeeper &RK, raw_ostream &OS) {
1760 GlobalISelEmitter(RK).run(OS);
1761}
1762} // End llvm namespace