blob: 488bf8d312bcf29cbbfb902f85286cec46387a81 [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"
Daniel Sanderse7b0d662017-04-21 15:59:56 +000034#include "SubtargetFeatureInfo.h"
Ahmed Bougacha36f70352016-12-21 23:26:20 +000035#include "llvm/ADT/Optional.h"
Daniel Sanders0ed28822017-04-12 08:23:08 +000036#include "llvm/ADT/SmallSet.h"
Ahmed Bougacha36f70352016-12-21 23:26:20 +000037#include "llvm/ADT/Statistic.h"
38#include "llvm/CodeGen/MachineValueType.h"
39#include "llvm/Support/CommandLine.h"
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +000040#include "llvm/Support/Error.h"
Daniel Sanders52b4ce72017-03-07 23:20:35 +000041#include "llvm/Support/LowLevelTypeImpl.h"
Pavel Labath52a82e22017-02-21 09:19:41 +000042#include "llvm/Support/ScopedPrinter.h"
Ahmed Bougacha36f70352016-12-21 23:26:20 +000043#include "llvm/TableGen/Error.h"
44#include "llvm/TableGen/Record.h"
45#include "llvm/TableGen/TableGenBackend.h"
46#include <string>
Daniel Sanders8a4bae92017-03-14 21:32:08 +000047#include <numeric>
Ahmed Bougacha36f70352016-12-21 23:26:20 +000048using namespace llvm;
49
50#define DEBUG_TYPE "gisel-emitter"
51
52STATISTIC(NumPatternTotal, "Total number of patterns");
Daniel Sandersb41ce2b2017-02-20 14:31:27 +000053STATISTIC(NumPatternImported, "Number of patterns imported from SelectionDAG");
54STATISTIC(NumPatternImportsSkipped, "Number of SelectionDAG imports skipped");
Ahmed Bougacha36f70352016-12-21 23:26:20 +000055STATISTIC(NumPatternEmitted, "Number of patterns emitted");
56
Daniel Sanders0848b232017-03-27 13:15:13 +000057cl::OptionCategory GlobalISelEmitterCat("Options for -gen-global-isel");
58
Ahmed Bougacha36f70352016-12-21 23:26:20 +000059static cl::opt<bool> WarnOnSkippedPatterns(
60 "warn-on-skipped-patterns",
61 cl::desc("Explain why a pattern was skipped for inclusion "
62 "in the GlobalISel selector"),
Daniel Sanders0848b232017-03-27 13:15:13 +000063 cl::init(false), cl::cat(GlobalISelEmitterCat));
Ahmed Bougacha36f70352016-12-21 23:26:20 +000064
Daniel Sandersbdfebb82017-03-15 20:18:38 +000065namespace {
Ahmed Bougacha36f70352016-12-21 23:26:20 +000066//===- Helper functions ---------------------------------------------------===//
67
Daniel Sanders52b4ce72017-03-07 23:20:35 +000068/// This class stands in for LLT wherever we want to tablegen-erate an
69/// equivalent at compiler run-time.
70class LLTCodeGen {
71private:
72 LLT Ty;
73
74public:
75 LLTCodeGen(const LLT &Ty) : Ty(Ty) {}
76
77 void emitCxxConstructorCall(raw_ostream &OS) const {
78 if (Ty.isScalar()) {
79 OS << "LLT::scalar(" << Ty.getSizeInBits() << ")";
80 return;
81 }
82 if (Ty.isVector()) {
83 OS << "LLT::vector(" << Ty.getNumElements() << ", " << Ty.getSizeInBits()
84 << ")";
85 return;
86 }
87 llvm_unreachable("Unhandled LLT");
88 }
Daniel Sanders8a4bae92017-03-14 21:32:08 +000089
90 const LLT &get() const { return Ty; }
91};
92
93class InstructionMatcher;
94class OperandPlaceholder {
95private:
96 enum PlaceholderKind {
97 OP_MatchReference,
98 OP_Temporary,
99 } Kind;
100
101 struct MatchReferenceData {
102 InstructionMatcher *InsnMatcher;
103 StringRef InsnVarName;
104 StringRef SymbolicName;
105 };
106
107 struct TemporaryData {
108 unsigned OpIdx;
109 };
110
111 union {
112 struct MatchReferenceData MatchReference;
113 struct TemporaryData Temporary;
114 };
115
116 OperandPlaceholder(PlaceholderKind Kind) : Kind(Kind) {}
117
118public:
119 ~OperandPlaceholder() {}
120
121 static OperandPlaceholder
122 CreateMatchReference(InstructionMatcher *InsnMatcher,
Daniel Sandersdb7ed372017-04-04 13:52:00 +0000123 StringRef InsnVarName, StringRef SymbolicName) {
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000124 OperandPlaceholder Result(OP_MatchReference);
125 Result.MatchReference.InsnMatcher = InsnMatcher;
126 Result.MatchReference.InsnVarName = InsnVarName;
127 Result.MatchReference.SymbolicName = SymbolicName;
128 return Result;
129 }
130
131 static OperandPlaceholder CreateTemporary(unsigned OpIdx) {
132 OperandPlaceholder Result(OP_Temporary);
133 Result.Temporary.OpIdx = OpIdx;
134 return Result;
135 }
136
137 void emitCxxValueExpr(raw_ostream &OS) const;
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000138};
139
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000140/// Convert an MVT to an equivalent LLT if possible, or the invalid LLT() for
141/// MVTs that don't map cleanly to an LLT (e.g., iPTR, *any, ...).
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000142static Optional<LLTCodeGen> MVTToLLT(MVT::SimpleValueType SVT) {
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000143 MVT VT(SVT);
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000144 if (VT.isVector() && VT.getVectorNumElements() != 1)
145 return LLTCodeGen(LLT::vector(VT.getVectorNumElements(), VT.getScalarSizeInBits()));
146 if (VT.isInteger() || VT.isFloatingPoint())
147 return LLTCodeGen(LLT::scalar(VT.getSizeInBits()));
148 return None;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000149}
150
Daniel Sandersd0656a32017-04-13 09:45:37 +0000151static std::string explainPredicates(const TreePatternNode *N) {
152 std::string Explanation = "";
153 StringRef Separator = "";
154 for (const auto &P : N->getPredicateFns()) {
155 Explanation +=
156 (Separator + P.getOrigPatFragRecord()->getRecord()->getName()).str();
157 if (P.isAlwaysTrue())
158 Explanation += " always-true";
159 if (P.isImmediatePattern())
160 Explanation += " immediate";
161 }
162 return Explanation;
163}
164
Daniel Sandersd0656a32017-04-13 09:45:37 +0000165std::string explainOperator(Record *Operator) {
166 if (Operator->isSubClassOf("SDNode"))
167 return " (" + Operator->getValueAsString("Opcode") + ")";
168
169 if (Operator->isSubClassOf("Intrinsic"))
170 return (" (Operator is an Intrinsic, " + Operator->getName() + ")").str();
171
172 return " (Operator not understood)";
173}
174
175/// Helper function to let the emitter report skip reason error messages.
176static Error failedImport(const Twine &Reason) {
177 return make_error<StringError>(Reason, inconvertibleErrorCode());
178}
179
180static Error isTrivialOperatorNode(const TreePatternNode *N) {
181 std::string Explanation = "";
182 std::string Separator = "";
183 if (N->isLeaf()) {
184 Explanation = "Is a leaf";
185 Separator = ", ";
186 }
187
188 if (N->hasAnyPredicate()) {
189 Explanation = Separator + "Has a predicate (" + explainPredicates(N) + ")";
190 Separator = ", ";
191 }
192
193 if (N->getTransformFn()) {
194 Explanation += Separator + "Has a transform function";
195 Separator = ", ";
196 }
197
198 if (!N->isLeaf() && !N->hasAnyPredicate() && !N->getTransformFn())
199 return Error::success();
200
201 return failedImport(Explanation);
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000202}
203
204//===- Matchers -----------------------------------------------------------===//
205
Daniel Sandersbee57392017-04-04 13:25:23 +0000206class OperandMatcher;
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000207class MatchAction;
208
209/// Generates code to check that a match rule matches.
210class RuleMatcher {
211 /// A list of matchers that all need to succeed for the current rule to match.
212 /// FIXME: This currently supports a single match position but could be
213 /// extended to support multiple positions to support div/rem fusion or
214 /// load-multiple instructions.
215 std::vector<std::unique_ptr<InstructionMatcher>> Matchers;
216
217 /// A list of actions that need to be taken when all predicates in this rule
218 /// have succeeded.
219 std::vector<std::unique_ptr<MatchAction>> Actions;
220
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000221 /// A map of instruction matchers to the local variables created by
222 /// emitCxxCaptureStmts().
223 std::map<const InstructionMatcher *, std::string> InsnVariableNames;
224
225 /// ID for the next instruction variable defined with defineInsnVar()
226 unsigned NextInsnVarID;
227
Daniel Sanderse7b0d662017-04-21 15:59:56 +0000228 std::vector<Record *> RequiredFeatures;
229
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000230public:
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000231 RuleMatcher()
232 : Matchers(), Actions(), InsnVariableNames(), NextInsnVarID(0) {}
Zachary Turnerb7dbd872017-03-20 19:56:52 +0000233 RuleMatcher(RuleMatcher &&Other) = default;
234 RuleMatcher &operator=(RuleMatcher &&Other) = default;
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000235
236 InstructionMatcher &addInstructionMatcher();
Daniel Sanderse7b0d662017-04-21 15:59:56 +0000237 void addRequiredFeature(Record *Feature);
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000238
239 template <class Kind, class... Args> Kind &addAction(Args &&... args);
240
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000241 std::string defineInsnVar(raw_ostream &OS, const InstructionMatcher &Matcher,
242 StringRef Value);
243 StringRef getInsnVarName(const InstructionMatcher &InsnMatcher) const;
244
Daniel Sandersbee57392017-04-04 13:25:23 +0000245 void emitCxxCapturedInsnList(raw_ostream &OS);
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000246 void emitCxxCaptureStmts(raw_ostream &OS, StringRef Expr);
247
Daniel Sanderse7b0d662017-04-21 15:59:56 +0000248 void emit(raw_ostream &OS,
249 std::map<Record *, SubtargetFeatureInfo, LessRecordByID>
250 SubtargetFeatures);
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000251
252 /// Compare the priority of this object and B.
253 ///
254 /// Returns true if this object is more important than B.
255 bool isHigherPriorityThan(const RuleMatcher &B) const;
256
257 /// Report the maximum number of temporary operands needed by the rule
258 /// matcher.
259 unsigned countTemporaryOperands() const;
260};
261
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000262template <class PredicateTy> class PredicateListMatcher {
263private:
264 typedef std::vector<std::unique_ptr<PredicateTy>> PredicateVec;
265 PredicateVec Predicates;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000266
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000267public:
268 /// Construct a new operand predicate and add it to the matcher.
269 template <class Kind, class... Args>
270 Kind &addPredicate(Args&&... args) {
Ahmed Bougachab67a3ce2017-01-26 22:07:37 +0000271 Predicates.emplace_back(
272 llvm::make_unique<Kind>(std::forward<Args>(args)...));
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000273 return *static_cast<Kind *>(Predicates.back().get());
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000274 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000275
276 typename PredicateVec::const_iterator predicates_begin() const { return Predicates.begin(); }
277 typename PredicateVec::const_iterator predicates_end() const { return Predicates.end(); }
278 iterator_range<typename PredicateVec::const_iterator> predicates() const {
279 return make_range(predicates_begin(), predicates_end());
280 }
Daniel Sanders759ff412017-02-24 13:58:11 +0000281 typename PredicateVec::size_type predicates_size() const { return Predicates.size(); }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000282
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000283 /// Emit a C++ expression that tests whether all the predicates are met.
Ahmed Bougachab67a3ce2017-01-26 22:07:37 +0000284 template <class... Args>
Daniel Sandersf8c804f2017-01-28 11:10:42 +0000285 void emitCxxPredicateListExpr(raw_ostream &OS, Args &&... args) const {
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000286 if (Predicates.empty()) {
287 OS << "true";
288 return;
289 }
290
291 StringRef Separator = "";
292 for (const auto &Predicate : predicates()) {
293 OS << Separator << "(";
Ahmed Bougachab67a3ce2017-01-26 22:07:37 +0000294 Predicate->emitCxxPredicateExpr(OS, std::forward<Args>(args)...);
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000295 OS << ")";
Ahmed Bougacha905af9f2017-02-04 00:47:02 +0000296 Separator = " &&\n";
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000297 }
298 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000299};
300
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000301/// Generates code to check a predicate of an operand.
302///
303/// Typical predicates include:
304/// * Operand is a particular register.
305/// * Operand is assigned a particular register bank.
306/// * Operand is an MBB.
307class OperandPredicateMatcher {
308public:
Daniel Sanders759ff412017-02-24 13:58:11 +0000309 /// This enum is used for RTTI and also defines the priority that is given to
310 /// the predicate when generating the matcher code. Kinds with higher priority
311 /// must be tested first.
312 ///
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000313 /// The relative priority of OPM_LLT, OPM_RegBank, and OPM_MBB do not matter
314 /// but OPM_Int must have priority over OPM_RegBank since constant integers
315 /// are represented by a virtual register defined by a G_CONSTANT instruction.
Daniel Sanders759ff412017-02-24 13:58:11 +0000316 enum PredicateKind {
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000317 OPM_ComplexPattern,
Daniel Sandersbee57392017-04-04 13:25:23 +0000318 OPM_Instruction,
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000319 OPM_Int,
Daniel Sanders759ff412017-02-24 13:58:11 +0000320 OPM_LLT,
321 OPM_RegBank,
322 OPM_MBB,
323 };
324
325protected:
326 PredicateKind Kind;
327
328public:
329 OperandPredicateMatcher(PredicateKind Kind) : Kind(Kind) {}
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000330 virtual ~OperandPredicateMatcher() {}
331
Daniel Sanders759ff412017-02-24 13:58:11 +0000332 PredicateKind getKind() const { return Kind; }
333
Daniel Sandersbee57392017-04-04 13:25:23 +0000334 /// Return the OperandMatcher for the specified operand or nullptr if there
335 /// isn't one by that name in this operand predicate matcher.
336 ///
337 /// InstructionOperandMatcher is the only subclass that can return non-null
338 /// for this.
339 virtual Optional<const OperandMatcher *>
Daniel Sandersdb7ed372017-04-04 13:52:00 +0000340 getOptionalOperand(StringRef SymbolicName) const {
Daniel Sandersbee57392017-04-04 13:25:23 +0000341 assert(!SymbolicName.empty() && "Cannot lookup unnamed operand");
342 return None;
343 }
344
345 /// Emit C++ statements to capture instructions into local variables.
346 ///
347 /// Only InstructionOperandMatcher needs to do anything for this method.
348 virtual void emitCxxCaptureStmts(raw_ostream &OS, RuleMatcher &Rule,
349 StringRef Expr) const {}
350
Daniel Sanderse604ef52017-02-20 15:30:43 +0000351 /// Emit a C++ expression that checks the predicate for the given operand.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000352 virtual void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanderse604ef52017-02-20 15:30:43 +0000353 StringRef OperandExpr) const = 0;
Daniel Sanders759ff412017-02-24 13:58:11 +0000354
355 /// Compare the priority of this object and B.
356 ///
357 /// Returns true if this object is more important than B.
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000358 virtual bool isHigherPriorityThan(const OperandPredicateMatcher &B) const {
359 return Kind < B.Kind;
Daniel Sanders759ff412017-02-24 13:58:11 +0000360 };
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000361
362 /// Report the maximum number of temporary operands needed by the predicate
363 /// matcher.
364 virtual unsigned countTemporaryOperands() const { return 0; }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000365};
366
367/// Generates code to check that an operand is a particular LLT.
368class LLTOperandMatcher : public OperandPredicateMatcher {
369protected:
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000370 LLTCodeGen Ty;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000371
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000372public:
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000373 LLTOperandMatcher(const LLTCodeGen &Ty)
Daniel Sanders759ff412017-02-24 13:58:11 +0000374 : OperandPredicateMatcher(OPM_LLT), Ty(Ty) {}
375
376 static bool classof(const OperandPredicateMatcher *P) {
377 return P->getKind() == OPM_LLT;
378 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000379
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000380 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanderse604ef52017-02-20 15:30:43 +0000381 StringRef OperandExpr) const override {
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000382 OS << "MRI.getType(" << OperandExpr << ".getReg()) == (";
383 Ty.emitCxxConstructorCall(OS);
384 OS << ")";
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000385 }
386};
387
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000388/// Generates code to check that an operand is a particular target constant.
389class ComplexPatternOperandMatcher : public OperandPredicateMatcher {
390protected:
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000391 const OperandMatcher &Operand;
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000392 const Record &TheDef;
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000393
394 unsigned getNumOperands() const {
395 return TheDef.getValueAsDag("Operands")->getNumArgs();
396 }
397
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000398 unsigned getAllocatedTemporariesBaseID() const;
399
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000400public:
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000401 ComplexPatternOperandMatcher(const OperandMatcher &Operand,
402 const Record &TheDef)
403 : OperandPredicateMatcher(OPM_ComplexPattern), Operand(Operand),
404 TheDef(TheDef) {}
405
406 static bool classof(const OperandPredicateMatcher *P) {
407 return P->getKind() == OPM_ComplexPattern;
408 }
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000409
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000410 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000411 StringRef OperandExpr) const override {
412 OS << TheDef.getValueAsString("MatcherFn") << "(" << OperandExpr;
413 for (unsigned I = 0; I < getNumOperands(); ++I) {
414 OS << ", ";
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000415 OperandPlaceholder::CreateTemporary(getAllocatedTemporariesBaseID() + I)
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000416 .emitCxxValueExpr(OS);
417 }
418 OS << ")";
419 }
420
421 unsigned countTemporaryOperands() const override {
422 return getNumOperands();
423 }
424};
425
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000426/// Generates code to check that an operand is in a particular register bank.
427class RegisterBankOperandMatcher : public OperandPredicateMatcher {
428protected:
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000429 const CodeGenRegisterClass &RC;
430
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000431public:
Daniel Sanders759ff412017-02-24 13:58:11 +0000432 RegisterBankOperandMatcher(const CodeGenRegisterClass &RC)
433 : OperandPredicateMatcher(OPM_RegBank), RC(RC) {}
434
435 static bool classof(const OperandPredicateMatcher *P) {
436 return P->getKind() == OPM_RegBank;
437 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000438
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000439 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanderse604ef52017-02-20 15:30:43 +0000440 StringRef OperandExpr) const override {
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000441 OS << "(&RBI.getRegBankFromRegClass(" << RC.getQualifiedName()
Daniel Sanderse604ef52017-02-20 15:30:43 +0000442 << "RegClass) == RBI.getRegBank(" << OperandExpr
443 << ".getReg(), MRI, TRI))";
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000444 }
445};
446
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000447/// Generates code to check that an operand is a basic block.
448class MBBOperandMatcher : public OperandPredicateMatcher {
449public:
Daniel Sanders759ff412017-02-24 13:58:11 +0000450 MBBOperandMatcher() : OperandPredicateMatcher(OPM_MBB) {}
451
452 static bool classof(const OperandPredicateMatcher *P) {
453 return P->getKind() == OPM_MBB;
454 }
455
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000456 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanderse604ef52017-02-20 15:30:43 +0000457 StringRef OperandExpr) const override {
458 OS << OperandExpr << ".isMBB()";
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000459 }
460};
461
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000462/// Generates code to check that an operand is a particular int.
463class IntOperandMatcher : public OperandPredicateMatcher {
464protected:
465 int64_t Value;
466
467public:
468 IntOperandMatcher(int64_t Value)
469 : OperandPredicateMatcher(OPM_Int), Value(Value) {}
470
471 static bool classof(const OperandPredicateMatcher *P) {
472 return P->getKind() == OPM_Int;
473 }
474
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000475 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Simon Pilgrimd0302912017-02-24 17:20:27 +0000476 StringRef OperandExpr) const override {
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000477 OS << "isOperandImmEqual(" << OperandExpr << ", " << Value << ", MRI)";
478 }
479};
480
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000481/// Generates code to check that a set of predicates match for a particular
482/// operand.
483class OperandMatcher : public PredicateListMatcher<OperandPredicateMatcher> {
484protected:
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000485 InstructionMatcher &Insn;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000486 unsigned OpIdx;
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000487 std::string SymbolicName;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000488
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000489 /// The index of the first temporary variable allocated to this operand. The
490 /// number of allocated temporaries can be found with
491 /// countTemporaryOperands().
492 unsigned AllocatedTemporariesBaseID;
493
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000494public:
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000495 OperandMatcher(InstructionMatcher &Insn, unsigned OpIdx,
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000496 const std::string &SymbolicName,
497 unsigned AllocatedTemporariesBaseID)
498 : Insn(Insn), OpIdx(OpIdx), SymbolicName(SymbolicName),
499 AllocatedTemporariesBaseID(AllocatedTemporariesBaseID) {}
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000500
501 bool hasSymbolicName() const { return !SymbolicName.empty(); }
502 const StringRef getSymbolicName() const { return SymbolicName; }
Daniel Sandersffc7d582017-03-29 15:37:18 +0000503 void setSymbolicName(StringRef Name) {
504 assert(SymbolicName.empty() && "Operand already has a symbolic name");
505 SymbolicName = Name;
506 }
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000507 unsigned getOperandIndex() const { return OpIdx; }
508
Daniel Sandersdb7ed372017-04-04 13:52:00 +0000509 std::string getOperandExpr(StringRef InsnVarName) const {
Pavel Labath52a82e22017-02-21 09:19:41 +0000510 return (InsnVarName + ".getOperand(" + llvm::to_string(OpIdx) + ")").str();
Daniel Sanderse604ef52017-02-20 15:30:43 +0000511 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000512
Daniel Sandersbee57392017-04-04 13:25:23 +0000513 Optional<const OperandMatcher *>
514 getOptionalOperand(StringRef DesiredSymbolicName) const {
515 assert(!DesiredSymbolicName.empty() && "Cannot lookup unnamed operand");
516 if (DesiredSymbolicName == SymbolicName)
517 return this;
518 for (const auto &OP : predicates()) {
519 const auto &MaybeOperand = OP->getOptionalOperand(DesiredSymbolicName);
520 if (MaybeOperand.hasValue())
521 return MaybeOperand.getValue();
522 }
523 return None;
524 }
525
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000526 InstructionMatcher &getInstructionMatcher() const { return Insn; }
527
Daniel Sandersbee57392017-04-04 13:25:23 +0000528 /// Emit C++ statements to capture instructions into local variables.
529 void emitCxxCaptureStmts(raw_ostream &OS, RuleMatcher &Rule,
530 StringRef OperandExpr) const {
531 for (const auto &Predicate : predicates())
532 Predicate->emitCxxCaptureStmts(OS, Rule, OperandExpr);
533 }
534
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000535 /// Emit a C++ expression that tests whether the instruction named in
536 /// InsnVarName matches all the predicate and all the operands.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000537 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sandersdb7ed372017-04-04 13:52:00 +0000538 StringRef InsnVarName) const {
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000539 OS << "(/* ";
540 if (SymbolicName.empty())
541 OS << "Operand " << OpIdx;
542 else
543 OS << SymbolicName;
544 OS << " */ ";
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000545 emitCxxPredicateListExpr(OS, Rule, getOperandExpr(InsnVarName));
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000546 OS << ")";
547 }
Daniel Sanders759ff412017-02-24 13:58:11 +0000548
549 /// Compare the priority of this object and B.
550 ///
551 /// Returns true if this object is more important than B.
552 bool isHigherPriorityThan(const OperandMatcher &B) const {
553 // Operand matchers involving more predicates have higher priority.
554 if (predicates_size() > B.predicates_size())
555 return true;
556 if (predicates_size() < B.predicates_size())
557 return false;
558
559 // This assumes that predicates are added in a consistent order.
560 for (const auto &Predicate : zip(predicates(), B.predicates())) {
561 if (std::get<0>(Predicate)->isHigherPriorityThan(*std::get<1>(Predicate)))
562 return true;
563 if (std::get<1>(Predicate)->isHigherPriorityThan(*std::get<0>(Predicate)))
564 return false;
565 }
566
567 return false;
568 };
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000569
570 /// Report the maximum number of temporary operands needed by the operand
571 /// matcher.
572 unsigned countTemporaryOperands() const {
573 return std::accumulate(
574 predicates().begin(), predicates().end(), 0,
575 [](unsigned A,
576 const std::unique_ptr<OperandPredicateMatcher> &Predicate) {
577 return A + Predicate->countTemporaryOperands();
578 });
579 }
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000580
581 unsigned getAllocatedTemporariesBaseID() const {
582 return AllocatedTemporariesBaseID;
583 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000584};
585
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000586unsigned ComplexPatternOperandMatcher::getAllocatedTemporariesBaseID() const {
587 return Operand.getAllocatedTemporariesBaseID();
588}
589
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000590/// Generates code to check a predicate on an instruction.
591///
592/// Typical predicates include:
593/// * The opcode of the instruction is a particular value.
594/// * The nsw/nuw flag is/isn't set.
595class InstructionPredicateMatcher {
Daniel Sanders759ff412017-02-24 13:58:11 +0000596protected:
597 /// This enum is used for RTTI and also defines the priority that is given to
598 /// the predicate when generating the matcher code. Kinds with higher priority
599 /// must be tested first.
600 enum PredicateKind {
601 IPM_Opcode,
602 };
603
604 PredicateKind Kind;
605
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000606public:
Daniel Sanders8d4d72f2017-02-24 14:53:35 +0000607 InstructionPredicateMatcher(PredicateKind Kind) : Kind(Kind) {}
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000608 virtual ~InstructionPredicateMatcher() {}
609
Daniel Sanders759ff412017-02-24 13:58:11 +0000610 PredicateKind getKind() const { return Kind; }
611
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000612 /// Emit a C++ expression that tests whether the instruction named in
613 /// InsnVarName matches the predicate.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000614 virtual void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Ahmed Bougacha6a1ac5a2017-02-09 02:50:01 +0000615 StringRef InsnVarName) const = 0;
Daniel Sanders759ff412017-02-24 13:58:11 +0000616
617 /// Compare the priority of this object and B.
618 ///
619 /// Returns true if this object is more important than B.
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000620 virtual bool isHigherPriorityThan(const InstructionPredicateMatcher &B) const {
Daniel Sanders759ff412017-02-24 13:58:11 +0000621 return Kind < B.Kind;
622 };
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000623
624 /// Report the maximum number of temporary operands needed by the predicate
625 /// matcher.
626 virtual unsigned countTemporaryOperands() const { return 0; }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000627};
628
629/// Generates code to check the opcode of an instruction.
630class InstructionOpcodeMatcher : public InstructionPredicateMatcher {
631protected:
632 const CodeGenInstruction *I;
633
634public:
Daniel Sanders8d4d72f2017-02-24 14:53:35 +0000635 InstructionOpcodeMatcher(const CodeGenInstruction *I)
636 : InstructionPredicateMatcher(IPM_Opcode), I(I) {}
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000637
Daniel Sanders759ff412017-02-24 13:58:11 +0000638 static bool classof(const InstructionPredicateMatcher *P) {
639 return P->getKind() == IPM_Opcode;
640 }
641
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000642 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Ahmed Bougacha6a1ac5a2017-02-09 02:50:01 +0000643 StringRef InsnVarName) const override {
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000644 OS << InsnVarName << ".getOpcode() == " << I->Namespace
645 << "::" << I->TheDef->getName();
646 }
Daniel Sanders759ff412017-02-24 13:58:11 +0000647
648 /// Compare the priority of this object and B.
649 ///
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000650 /// Returns true if this object is more important than B.
651 bool isHigherPriorityThan(const InstructionPredicateMatcher &B) const override {
Daniel Sanders759ff412017-02-24 13:58:11 +0000652 if (InstructionPredicateMatcher::isHigherPriorityThan(B))
653 return true;
654 if (B.InstructionPredicateMatcher::isHigherPriorityThan(*this))
655 return false;
656
657 // Prioritize opcodes for cosmetic reasons in the generated source. Although
658 // this is cosmetic at the moment, we may want to drive a similar ordering
659 // using instruction frequency information to improve compile time.
660 if (const InstructionOpcodeMatcher *BO =
661 dyn_cast<InstructionOpcodeMatcher>(&B))
662 return I->TheDef->getName() < BO->I->TheDef->getName();
663
664 return false;
665 };
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000666};
667
668/// Generates code to check that a set of predicates and operands match for a
669/// particular instruction.
670///
671/// Typical predicates include:
672/// * Has a specific opcode.
673/// * Has an nsw/nuw flag or doesn't.
674class InstructionMatcher
675 : public PredicateListMatcher<InstructionPredicateMatcher> {
676protected:
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000677 typedef std::vector<std::unique_ptr<OperandMatcher>> OperandVec;
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000678
679 /// The operands to match. All rendered operands must be present even if the
680 /// condition is always true.
681 OperandVec Operands;
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000682
683public:
684 /// Add an operand to the matcher.
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000685 OperandMatcher &addOperand(unsigned OpIdx, const std::string &SymbolicName,
686 unsigned AllocatedTemporariesBaseID) {
687 Operands.emplace_back(new OperandMatcher(*this, OpIdx, SymbolicName,
688 AllocatedTemporariesBaseID));
689 return *Operands.back();
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000690 }
691
Daniel Sandersffc7d582017-03-29 15:37:18 +0000692 OperandMatcher &getOperand(unsigned OpIdx) {
693 auto I = std::find_if(Operands.begin(), Operands.end(),
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000694 [&OpIdx](const std::unique_ptr<OperandMatcher> &X) {
695 return X->getOperandIndex() == OpIdx;
Daniel Sandersffc7d582017-03-29 15:37:18 +0000696 });
697 if (I != Operands.end())
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000698 return **I;
Daniel Sandersffc7d582017-03-29 15:37:18 +0000699 llvm_unreachable("Failed to lookup operand");
700 }
701
Daniel Sandersbee57392017-04-04 13:25:23 +0000702 Optional<const OperandMatcher *>
703 getOptionalOperand(StringRef SymbolicName) const {
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000704 assert(!SymbolicName.empty() && "Cannot lookup unnamed operand");
Daniel Sandersbee57392017-04-04 13:25:23 +0000705 for (const auto &Operand : Operands) {
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000706 const auto &OM = Operand->getOptionalOperand(SymbolicName);
Daniel Sandersbee57392017-04-04 13:25:23 +0000707 if (OM.hasValue())
708 return OM.getValue();
709 }
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000710 return None;
711 }
712
Daniel Sandersdb7ed372017-04-04 13:52:00 +0000713 const OperandMatcher &getOperand(StringRef SymbolicName) const {
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000714 Optional<const OperandMatcher *>OM = getOptionalOperand(SymbolicName);
715 if (OM.hasValue())
716 return *OM.getValue();
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000717 llvm_unreachable("Failed to lookup operand");
718 }
719
720 unsigned getNumOperands() const { return Operands.size(); }
Daniel Sandersbee57392017-04-04 13:25:23 +0000721 OperandVec::iterator operands_begin() { return Operands.begin(); }
722 OperandVec::iterator operands_end() { return Operands.end(); }
723 iterator_range<OperandVec::iterator> operands() {
724 return make_range(operands_begin(), operands_end());
725 }
Daniel Sandersffc7d582017-03-29 15:37:18 +0000726 OperandVec::const_iterator operands_begin() const { return Operands.begin(); }
727 OperandVec::const_iterator operands_end() const { return Operands.end(); }
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000728 iterator_range<OperandVec::const_iterator> operands() const {
729 return make_range(operands_begin(), operands_end());
730 }
731
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000732 /// Emit C++ statements to check the shape of the match and capture
733 /// instructions into local variables.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000734 void emitCxxCaptureStmts(raw_ostream &OS, RuleMatcher &Rule, StringRef Expr) {
735 OS << "if (" << Expr << ".getNumOperands() < " << getNumOperands() << ")\n"
736 << " return false;\n";
Daniel Sandersbee57392017-04-04 13:25:23 +0000737 for (const auto &Operand : Operands) {
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000738 Operand->emitCxxCaptureStmts(OS, Rule, Operand->getOperandExpr(Expr));
Daniel Sandersbee57392017-04-04 13:25:23 +0000739 }
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000740 }
741
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000742 /// Emit a C++ expression that tests whether the instruction named in
743 /// InsnVarName matches all the predicates and all the operands.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000744 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
745 StringRef InsnVarName) const {
746 emitCxxPredicateListExpr(OS, Rule, InsnVarName);
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000747 for (const auto &Operand : Operands) {
Ahmed Bougacha905af9f2017-02-04 00:47:02 +0000748 OS << " &&\n(";
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000749 Operand->emitCxxPredicateExpr(OS, Rule, InsnVarName);
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000750 OS << ")";
751 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000752 }
Daniel Sanders759ff412017-02-24 13:58:11 +0000753
754 /// Compare the priority of this object and B.
755 ///
756 /// Returns true if this object is more important than B.
757 bool isHigherPriorityThan(const InstructionMatcher &B) const {
758 // Instruction matchers involving more operands have higher priority.
759 if (Operands.size() > B.Operands.size())
760 return true;
761 if (Operands.size() < B.Operands.size())
762 return false;
763
764 for (const auto &Predicate : zip(predicates(), B.predicates())) {
765 if (std::get<0>(Predicate)->isHigherPriorityThan(*std::get<1>(Predicate)))
766 return true;
767 if (std::get<1>(Predicate)->isHigherPriorityThan(*std::get<0>(Predicate)))
768 return false;
769 }
770
771 for (const auto &Operand : zip(Operands, B.Operands)) {
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000772 if (std::get<0>(Operand)->isHigherPriorityThan(*std::get<1>(Operand)))
Daniel Sanders759ff412017-02-24 13:58:11 +0000773 return true;
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000774 if (std::get<1>(Operand)->isHigherPriorityThan(*std::get<0>(Operand)))
Daniel Sanders759ff412017-02-24 13:58:11 +0000775 return false;
776 }
777
778 return false;
779 };
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000780
781 /// Report the maximum number of temporary operands needed by the instruction
782 /// matcher.
783 unsigned countTemporaryOperands() const {
784 return std::accumulate(predicates().begin(), predicates().end(), 0,
785 [](unsigned A,
786 const std::unique_ptr<InstructionPredicateMatcher>
787 &Predicate) {
788 return A + Predicate->countTemporaryOperands();
789 }) +
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000790 std::accumulate(
791 Operands.begin(), Operands.end(), 0,
792 [](unsigned A, const std::unique_ptr<OperandMatcher> &Operand) {
793 return A + Operand->countTemporaryOperands();
794 });
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000795 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000796};
797
Daniel Sandersbee57392017-04-04 13:25:23 +0000798/// Generates code to check that the operand is a register defined by an
799/// instruction that matches the given instruction matcher.
800///
801/// For example, the pattern:
802/// (set $dst, (G_MUL (G_ADD $src1, $src2), $src3))
803/// would use an InstructionOperandMatcher for operand 1 of the G_MUL to match
804/// the:
805/// (G_ADD $src1, $src2)
806/// subpattern.
807class InstructionOperandMatcher : public OperandPredicateMatcher {
808protected:
809 std::unique_ptr<InstructionMatcher> InsnMatcher;
810
811public:
812 InstructionOperandMatcher()
813 : OperandPredicateMatcher(OPM_Instruction),
814 InsnMatcher(new InstructionMatcher()) {}
815
816 static bool classof(const OperandPredicateMatcher *P) {
817 return P->getKind() == OPM_Instruction;
818 }
819
820 InstructionMatcher &getInsnMatcher() const { return *InsnMatcher; }
821
822 Optional<const OperandMatcher *>
823 getOptionalOperand(StringRef SymbolicName) const override {
824 assert(!SymbolicName.empty() && "Cannot lookup unnamed operand");
825 return InsnMatcher->getOptionalOperand(SymbolicName);
826 }
827
828 void emitCxxCaptureStmts(raw_ostream &OS, RuleMatcher &Rule,
829 StringRef OperandExpr) const override {
830 OS << "if (!" << OperandExpr + ".isReg())\n"
831 << " return false;\n";
832 std::string InsnVarName = Rule.defineInsnVar(
833 OS, *InsnMatcher,
834 ("*MRI.getVRegDef(" + OperandExpr + ".getReg())").str());
835 InsnMatcher->emitCxxCaptureStmts(OS, Rule, InsnVarName);
836 }
837
838 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
839 StringRef OperandExpr) const override {
840 OperandExpr = Rule.getInsnVarName(*InsnMatcher);
841 OS << "(";
842 InsnMatcher->emitCxxPredicateExpr(OS, Rule, OperandExpr);
843 OS << ")\n";
844 }
845};
846
Daniel Sanders43c882c2017-02-01 10:53:10 +0000847//===- Actions ------------------------------------------------------------===//
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000848void OperandPlaceholder::emitCxxValueExpr(raw_ostream &OS) const {
849 switch (Kind) {
850 case OP_MatchReference:
851 OS << MatchReference.InsnMatcher->getOperand(MatchReference.SymbolicName)
852 .getOperandExpr(MatchReference.InsnVarName);
853 break;
854 case OP_Temporary:
855 OS << "TempOp" << Temporary.OpIdx;
856 break;
857 }
858}
Daniel Sanders43c882c2017-02-01 10:53:10 +0000859
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000860class OperandRenderer {
861public:
Daniel Sanders0ed28822017-04-12 08:23:08 +0000862 enum RendererKind { OR_Copy, OR_Imm, OR_Register, OR_ComplexPattern };
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000863
864protected:
865 RendererKind Kind;
866
867public:
868 OperandRenderer(RendererKind Kind) : Kind(Kind) {}
869 virtual ~OperandRenderer() {}
870
871 RendererKind getKind() const { return Kind; }
872
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000873 virtual void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const = 0;
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000874};
875
876/// A CopyRenderer emits code to copy a single operand from an existing
877/// instruction to the one being built.
878class CopyRenderer : public OperandRenderer {
879protected:
880 /// The matcher for the instruction that this operand is copied from.
881 /// This provides the facility for looking up an a operand by it's name so
882 /// that it can be used as a source for the instruction being built.
883 const InstructionMatcher &Matched;
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000884 /// The name of the operand.
885 const StringRef SymbolicName;
886
887public:
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000888 CopyRenderer(const InstructionMatcher &Matched, StringRef SymbolicName)
889 : OperandRenderer(OR_Copy), Matched(Matched), SymbolicName(SymbolicName) {
890 }
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000891
892 static bool classof(const OperandRenderer *R) {
893 return R->getKind() == OR_Copy;
894 }
895
896 const StringRef getSymbolicName() const { return SymbolicName; }
897
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000898 void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const override {
899 const OperandMatcher &Operand = Matched.getOperand(SymbolicName);
900 StringRef InsnVarName =
901 Rule.getInsnVarName(Operand.getInstructionMatcher());
902 std::string OperandExpr = Operand.getOperandExpr(InsnVarName);
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000903 OS << " MIB.add(" << OperandExpr << "/*" << SymbolicName << "*/);\n";
904 }
905};
906
907/// Adds a specific physical register to the instruction being built.
908/// This is typically useful for WZR/XZR on AArch64.
909class AddRegisterRenderer : public OperandRenderer {
910protected:
911 const Record *RegisterDef;
912
913public:
914 AddRegisterRenderer(const Record *RegisterDef)
915 : OperandRenderer(OR_Register), RegisterDef(RegisterDef) {}
916
917 static bool classof(const OperandRenderer *R) {
918 return R->getKind() == OR_Register;
919 }
920
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000921 void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const override {
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000922 OS << " MIB.addReg(" << RegisterDef->getValueAsString("Namespace")
923 << "::" << RegisterDef->getName() << ");\n";
924 }
925};
926
Daniel Sanders0ed28822017-04-12 08:23:08 +0000927/// Adds a specific immediate to the instruction being built.
928class ImmRenderer : public OperandRenderer {
929protected:
930 int64_t Imm;
931
932public:
933 ImmRenderer(int64_t Imm)
934 : OperandRenderer(OR_Imm), Imm(Imm) {}
935
936 static bool classof(const OperandRenderer *R) {
937 return R->getKind() == OR_Imm;
938 }
939
940 void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const override {
941 OS << " MIB.addImm(" << Imm << ");\n";
942 }
943};
944
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000945class RenderComplexPatternOperand : public OperandRenderer {
946private:
947 const Record &TheDef;
948 std::vector<OperandPlaceholder> Sources;
949
950 unsigned getNumOperands() const {
951 return TheDef.getValueAsDag("Operands")->getNumArgs();
952 }
953
954public:
955 RenderComplexPatternOperand(const Record &TheDef,
956 const ArrayRef<OperandPlaceholder> Sources)
957 : OperandRenderer(OR_ComplexPattern), TheDef(TheDef), Sources(Sources) {}
958
959 static bool classof(const OperandRenderer *R) {
960 return R->getKind() == OR_ComplexPattern;
961 }
962
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000963 void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const override {
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000964 assert(Sources.size() == getNumOperands() && "Inconsistent number of operands");
965 for (const auto &Source : Sources) {
966 OS << "MIB.add(";
967 Source.emitCxxValueExpr(OS);
968 OS << ");\n";
969 }
970 }
971};
972
Ahmed Bougacha56ca3a92017-02-04 00:47:10 +0000973/// An action taken when all Matcher predicates succeeded for a parent rule.
974///
975/// Typical actions include:
976/// * Changing the opcode of an instruction.
977/// * Adding an operand to an instruction.
Daniel Sanders43c882c2017-02-01 10:53:10 +0000978class MatchAction {
979public:
980 virtual ~MatchAction() {}
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000981
982 /// Emit the C++ statements to implement the action.
983 ///
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000984 /// \param RecycleVarName If given, it's an instruction to recycle. The
985 /// requirements on the instruction vary from action to
986 /// action.
987 virtual void emitCxxActionStmts(raw_ostream &OS, RuleMatcher &Rule,
988 StringRef RecycleVarName) const = 0;
Daniel Sanders43c882c2017-02-01 10:53:10 +0000989};
990
Ahmed Bougacha9aa4c102017-02-04 00:47:08 +0000991/// Generates a comment describing the matched rule being acted upon.
992class DebugCommentAction : public MatchAction {
993private:
994 const PatternToMatch &P;
995
996public:
997 DebugCommentAction(const PatternToMatch &P) : P(P) {}
998
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000999 void emitCxxActionStmts(raw_ostream &OS, RuleMatcher &Rule,
1000 StringRef RecycleVarName) const override {
1001 OS << "// " << *P.getSrcPattern() << " => " << *P.getDstPattern() << "\n";
Ahmed Bougacha9aa4c102017-02-04 00:47:08 +00001002 }
1003};
1004
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001005/// Generates code to build an instruction or mutate an existing instruction
1006/// into the desired instruction when this is possible.
1007class BuildMIAction : public MatchAction {
Daniel Sanders43c882c2017-02-01 10:53:10 +00001008private:
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001009 const CodeGenInstruction *I;
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001010 const InstructionMatcher &Matched;
1011 std::vector<std::unique_ptr<OperandRenderer>> OperandRenderers;
1012
1013 /// True if the instruction can be built solely by mutating the opcode.
1014 bool canMutate() const {
1015 for (const auto &Renderer : enumerate(OperandRenderers)) {
Zachary Turner309a0882017-03-13 16:24:10 +00001016 if (const auto *Copy = dyn_cast<CopyRenderer>(&*Renderer.value())) {
Daniel Sanders3016d3c2017-04-22 14:31:28 +00001017 const OperandMatcher &OM = Matched.getOperand(Copy->getSymbolicName());
1018 if (&Matched != &OM.getInstructionMatcher() ||
1019 OM.getOperandIndex() != Renderer.index())
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001020 return false;
1021 } else
1022 return false;
1023 }
1024
1025 return true;
1026 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001027
Daniel Sanders43c882c2017-02-01 10:53:10 +00001028public:
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001029 BuildMIAction(const CodeGenInstruction *I, const InstructionMatcher &Matched)
1030 : I(I), Matched(Matched) {}
Daniel Sanders43c882c2017-02-01 10:53:10 +00001031
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001032 template <class Kind, class... Args>
1033 Kind &addRenderer(Args&&... args) {
1034 OperandRenderers.emplace_back(
1035 llvm::make_unique<Kind>(std::forward<Args>(args)...));
1036 return *static_cast<Kind *>(OperandRenderers.back().get());
1037 }
1038
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001039 void emitCxxActionStmts(raw_ostream &OS, RuleMatcher &Rule,
1040 StringRef RecycleVarName) const override {
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001041 if (canMutate()) {
Tim Northover4340d642017-03-20 21:58:23 +00001042 OS << " " << RecycleVarName << ".setDesc(TII.get(" << I->Namespace
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001043 << "::" << I->TheDef->getName() << "));\n";
Tim Northover4340d642017-03-20 21:58:23 +00001044
1045 if (!I->ImplicitDefs.empty() || !I->ImplicitUses.empty()) {
1046 OS << " auto MIB = MachineInstrBuilder(MF, &" << RecycleVarName
1047 << ");\n";
1048
1049 for (auto Def : I->ImplicitDefs) {
1050 auto Namespace = Def->getValueAsString("Namespace");
1051 OS << " MIB.addDef(" << Namespace << "::" << Def->getName()
1052 << ", RegState::Implicit);\n";
1053 }
1054 for (auto Use : I->ImplicitUses) {
1055 auto Namespace = Use->getValueAsString("Namespace");
1056 OS << " MIB.addUse(" << Namespace << "::" << Use->getName()
1057 << ", RegState::Implicit);\n";
1058 }
1059 }
1060
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001061 OS << " MachineInstr &NewI = " << RecycleVarName << ";\n";
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001062 return;
1063 }
1064
1065 // TODO: Simple permutation looks like it could be almost as common as
1066 // mutation due to commutative operations.
1067
1068 OS << "MachineInstrBuilder MIB = BuildMI(*I.getParent(), I, "
1069 "I.getDebugLoc(), TII.get("
1070 << I->Namespace << "::" << I->TheDef->getName() << "));\n";
1071 for (const auto &Renderer : OperandRenderers)
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001072 Renderer->emitCxxRenderStmts(OS, Rule);
Daniel Sandersbee57392017-04-04 13:25:23 +00001073 OS << " for (const auto *FromMI : ";
1074 Rule.emitCxxCapturedInsnList(OS);
1075 OS << ")\n";
1076 OS << " for (const auto &MMO : FromMI->memoperands())\n";
1077 OS << " MIB.addMemOperand(MMO);\n";
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001078 OS << " " << RecycleVarName << ".eraseFromParent();\n";
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001079 OS << " MachineInstr &NewI = *MIB;\n";
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001080 }
1081};
1082
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001083InstructionMatcher &RuleMatcher::addInstructionMatcher() {
1084 Matchers.emplace_back(new InstructionMatcher());
1085 return *Matchers.back();
1086}
Ahmed Bougacha56ca3a92017-02-04 00:47:10 +00001087
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001088void RuleMatcher::addRequiredFeature(Record *Feature) {
1089 RequiredFeatures.push_back(Feature);
1090}
1091
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001092template <class Kind, class... Args>
1093Kind &RuleMatcher::addAction(Args &&... args) {
1094 Actions.emplace_back(llvm::make_unique<Kind>(std::forward<Args>(args)...));
1095 return *static_cast<Kind *>(Actions.back().get());
1096}
Daniel Sandersdc662ff2017-01-26 11:10:14 +00001097
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001098std::string RuleMatcher::defineInsnVar(raw_ostream &OS,
1099 const InstructionMatcher &Matcher,
1100 StringRef Value) {
1101 std::string InsnVarName = "MI" + llvm::to_string(NextInsnVarID++);
1102 OS << "MachineInstr &" << InsnVarName << " = " << Value << ";\n";
1103 InsnVariableNames[&Matcher] = InsnVarName;
1104 return InsnVarName;
1105}
1106
1107StringRef RuleMatcher::getInsnVarName(const InstructionMatcher &InsnMatcher) const {
1108 const auto &I = InsnVariableNames.find(&InsnMatcher);
1109 if (I != InsnVariableNames.end())
1110 return I->second;
1111 llvm_unreachable("Matched Insn was not captured in a local variable");
1112}
1113
Daniel Sandersbee57392017-04-04 13:25:23 +00001114/// Emit a C++ initializer_list containing references to every matched instruction.
1115void RuleMatcher::emitCxxCapturedInsnList(raw_ostream &OS) {
Daniel Sanders9e4817d2017-04-04 14:27:06 +00001116 SmallVector<StringRef, 2> Names;
Daniel Sandersbee57392017-04-04 13:25:23 +00001117 for (const auto &Pair : InsnVariableNames)
Daniel Sanders9e4817d2017-04-04 14:27:06 +00001118 Names.push_back(Pair.second);
1119 std::sort(Names.begin(), Names.end());
1120
1121 OS << "{";
1122 for (const auto &Name : Names)
1123 OS << "&" << Name << ", ";
Daniel Sandersbee57392017-04-04 13:25:23 +00001124 OS << "}";
1125}
1126
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001127/// Emit C++ statements to check the shape of the match and capture
1128/// instructions into local variables.
1129void RuleMatcher::emitCxxCaptureStmts(raw_ostream &OS, StringRef Expr) {
1130 assert(Matchers.size() == 1 && "Cannot handle multi-root matchers yet");
1131 std::string InsnVarName = defineInsnVar(OS, *Matchers.front(), Expr);
1132 Matchers.front()->emitCxxCaptureStmts(OS, *this, InsnVarName);
1133}
1134
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001135void RuleMatcher::emit(raw_ostream &OS,
1136 std::map<Record *, SubtargetFeatureInfo, LessRecordByID>
1137 SubtargetFeatures) {
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001138 if (Matchers.empty())
1139 llvm_unreachable("Unexpected empty matcher!");
Daniel Sandersdc662ff2017-01-26 11:10:14 +00001140
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001141 // The representation supports rules that require multiple roots such as:
1142 // %ptr(p0) = ...
1143 // %elt0(s32) = G_LOAD %ptr
1144 // %1(p0) = G_ADD %ptr, 4
1145 // %elt1(s32) = G_LOAD p0 %1
1146 // which could be usefully folded into:
1147 // %ptr(p0) = ...
1148 // %elt0(s32), %elt1(s32) = TGT_LOAD_PAIR %ptr
1149 // on some targets but we don't need to make use of that yet.
1150 assert(Matchers.size() == 1 && "Cannot handle multi-root matchers yet");
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001151
1152 OS << "if (";
1153 OS << "[&]() {\n";
1154 if (!RequiredFeatures.empty()) {
1155 OS << " PredicateBitset ExpectedFeatures = {";
1156 StringRef Separator = "";
1157 for (const auto &Predicate : RequiredFeatures) {
1158 const auto &I = SubtargetFeatures.find(Predicate);
1159 assert(I != SubtargetFeatures.end() && "Didn't import predicate?");
1160 OS << Separator << I->second.getEnumBitName();
1161 Separator = ", ";
1162 }
1163 OS << "};\n";
1164 OS << "if ((AvailableFeatures & ExpectedFeatures) != ExpectedFeatures)\n"
1165 << " return false;\n";
1166 }
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001167
1168 emitCxxCaptureStmts(OS, "I");
1169
1170 OS << " if (";
1171 Matchers.front()->emitCxxPredicateExpr(OS, *this,
1172 getInsnVarName(*Matchers.front()));
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001173 OS << ") {\n";
1174
Daniel Sandersbee57392017-04-04 13:25:23 +00001175 // We must also check if it's safe to fold the matched instructions.
1176 if (InsnVariableNames.size() >= 2) {
1177 for (const auto &Pair : InsnVariableNames) {
1178 // Skip the root node since it isn't moving anywhere. Everything else is
1179 // sinking to meet it.
1180 if (Pair.first == Matchers.front().get())
1181 continue;
1182
1183 // Reject the difficult cases until we have a more accurate check.
1184 OS << " if (!isObviouslySafeToFold(" << Pair.second
1185 << ")) return false;\n";
1186
1187 // FIXME: Emit checks to determine it's _actually_ safe to fold and/or
1188 // account for unsafe cases.
1189 //
1190 // Example:
1191 // MI1--> %0 = ...
1192 // %1 = ... %0
1193 // MI0--> %2 = ... %0
1194 // It's not safe to erase MI1. We currently handle this by not
1195 // erasing %0 (even when it's dead).
1196 //
1197 // Example:
1198 // MI1--> %0 = load volatile @a
1199 // %1 = load volatile @a
1200 // MI0--> %2 = ... %0
1201 // It's not safe to sink %0's def past %1. We currently handle
1202 // this by rejecting all loads.
1203 //
1204 // Example:
1205 // MI1--> %0 = load @a
1206 // %1 = store @a
1207 // MI0--> %2 = ... %0
1208 // It's not safe to sink %0's def past %1. We currently handle
1209 // this by rejecting all loads.
1210 //
1211 // Example:
1212 // G_CONDBR %cond, @BB1
1213 // BB0:
1214 // MI1--> %0 = load @a
1215 // G_BR @BB1
1216 // BB1:
1217 // MI0--> %2 = ... %0
1218 // It's not always safe to sink %0 across control flow. In this
1219 // case it may introduce a memory fault. We currentl handle this
1220 // by rejecting all loads.
1221 }
1222 }
1223
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001224 for (const auto &MA : Actions) {
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001225 MA->emitCxxActionStmts(OS, *this, "I");
Daniel Sandersdc662ff2017-01-26 11:10:14 +00001226 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001227
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001228 OS << " constrainSelectedInstRegOperands(NewI, TII, TRI, RBI);\n";
1229 OS << " return true;\n";
1230 OS << " }\n";
1231 OS << " return false;\n";
1232 OS << " }()) { return true; }\n\n";
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001233}
Daniel Sanders43c882c2017-02-01 10:53:10 +00001234
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001235bool RuleMatcher::isHigherPriorityThan(const RuleMatcher &B) const {
1236 // Rules involving more match roots have higher priority.
1237 if (Matchers.size() > B.Matchers.size())
1238 return true;
1239 if (Matchers.size() < B.Matchers.size())
Daniel Sanders759ff412017-02-24 13:58:11 +00001240 return false;
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001241
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001242 for (const auto &Matcher : zip(Matchers, B.Matchers)) {
1243 if (std::get<0>(Matcher)->isHigherPriorityThan(*std::get<1>(Matcher)))
1244 return true;
1245 if (std::get<1>(Matcher)->isHigherPriorityThan(*std::get<0>(Matcher)))
1246 return false;
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001247 }
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001248
1249 return false;
Simon Pilgrima7d1da82017-03-15 22:50:47 +00001250}
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001251
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001252unsigned RuleMatcher::countTemporaryOperands() const {
1253 return std::accumulate(
1254 Matchers.begin(), Matchers.end(), 0,
1255 [](unsigned A, const std::unique_ptr<InstructionMatcher> &Matcher) {
1256 return A + Matcher->countTemporaryOperands();
1257 });
1258}
1259
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001260//===- GlobalISelEmitter class --------------------------------------------===//
1261
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001262class GlobalISelEmitter {
1263public:
1264 explicit GlobalISelEmitter(RecordKeeper &RK);
1265 void run(raw_ostream &OS);
1266
1267private:
1268 const RecordKeeper &RK;
1269 const CodeGenDAGPatterns CGP;
1270 const CodeGenTarget &Target;
1271
1272 /// Keep track of the equivalence between SDNodes and Instruction.
1273 /// This is defined using 'GINodeEquiv' in the target description.
1274 DenseMap<Record *, const CodeGenInstruction *> NodeEquivs;
1275
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001276 /// Keep track of the equivalence between ComplexPattern's and
1277 /// GIComplexOperandMatcher. Map entries are specified by subclassing
1278 /// GIComplexPatternEquiv.
1279 DenseMap<const Record *, const Record *> ComplexPatternEquivs;
1280
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001281 // Map of predicates to their subtarget features.
1282 std::map<Record *, SubtargetFeatureInfo, LessRecordByID> SubtargetFeatures;
1283
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001284 void gatherNodeEquivs();
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001285 const CodeGenInstruction *findNodeEquiv(Record *N) const;
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001286
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001287 Error importRulePredicates(RuleMatcher &M, ArrayRef<Init *> Predicates);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001288 Expected<InstructionMatcher &>
Daniel Sandersc270c502017-03-30 09:36:33 +00001289 createAndImportSelDAGMatcher(InstructionMatcher &InsnMatcher,
1290 const TreePatternNode *Src) const;
1291 Error importChildMatcher(InstructionMatcher &InsnMatcher,
1292 TreePatternNode *SrcChild, unsigned OpIdx,
1293 unsigned &TempOpIdx) const;
1294 Expected<BuildMIAction &> createAndImportInstructionRenderer(
1295 RuleMatcher &M, const TreePatternNode *Dst,
1296 const InstructionMatcher &InsnMatcher) const;
1297 Error importExplicitUseRenderer(BuildMIAction &DstMIBuilder,
1298 TreePatternNode *DstChild,
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001299 const InstructionMatcher &InsnMatcher) const;
Daniel Sandersc270c502017-03-30 09:36:33 +00001300 Error
Daniel Sandersffc7d582017-03-29 15:37:18 +00001301 importImplicitDefRenderers(BuildMIAction &DstMIBuilder,
1302 const std::vector<Record *> &ImplicitDefs) const;
1303
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001304 /// Analyze pattern \p P, returning a matcher for it if possible.
1305 /// Otherwise, return an Error explaining why we don't support it.
1306 Expected<RuleMatcher> runOnPattern(const PatternToMatch &P);
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001307
1308 void declareSubtargetFeature(Record *Predicate);
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001309};
1310
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001311void GlobalISelEmitter::gatherNodeEquivs() {
1312 assert(NodeEquivs.empty());
1313 for (Record *Equiv : RK.getAllDerivedDefinitions("GINodeEquiv"))
1314 NodeEquivs[Equiv->getValueAsDef("Node")] =
1315 &Target.getInstruction(Equiv->getValueAsDef("I"));
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001316
1317 assert(ComplexPatternEquivs.empty());
1318 for (Record *Equiv : RK.getAllDerivedDefinitions("GIComplexPatternEquiv")) {
1319 Record *SelDAGEquiv = Equiv->getValueAsDef("SelDAGEquivalent");
1320 if (!SelDAGEquiv)
1321 continue;
1322 ComplexPatternEquivs[SelDAGEquiv] = Equiv;
1323 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001324}
1325
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001326const CodeGenInstruction *GlobalISelEmitter::findNodeEquiv(Record *N) const {
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001327 return NodeEquivs.lookup(N);
1328}
1329
1330GlobalISelEmitter::GlobalISelEmitter(RecordKeeper &RK)
1331 : RK(RK), CGP(RK), Target(CGP.getTargetInfo()) {}
1332
1333//===- Emitter ------------------------------------------------------------===//
1334
Daniel Sandersc270c502017-03-30 09:36:33 +00001335Error
Daniel Sandersffc7d582017-03-29 15:37:18 +00001336GlobalISelEmitter::importRulePredicates(RuleMatcher &M,
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001337 ArrayRef<Init *> Predicates) {
1338 for (const Init *Predicate : Predicates) {
1339 const DefInit *PredicateDef = static_cast<const DefInit *>(Predicate);
1340 declareSubtargetFeature(PredicateDef->getDef());
1341 M.addRequiredFeature(PredicateDef->getDef());
1342 }
1343
Daniel Sandersc270c502017-03-30 09:36:33 +00001344 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001345}
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001346
Daniel Sandersc270c502017-03-30 09:36:33 +00001347Expected<InstructionMatcher &> GlobalISelEmitter::createAndImportSelDAGMatcher(
1348 InstructionMatcher &InsnMatcher, const TreePatternNode *Src) const {
Daniel Sandersffc7d582017-03-29 15:37:18 +00001349 // Start with the defined operands (i.e., the results of the root operator).
1350 if (Src->getExtTypes().size() > 1)
1351 return failedImport("Src pattern has multiple results");
1352
1353 auto SrcGIOrNull = findNodeEquiv(Src->getOperator());
1354 if (!SrcGIOrNull)
Daniel Sandersd0656a32017-04-13 09:45:37 +00001355 return failedImport("Pattern operator lacks an equivalent Instruction" +
1356 explainOperator(Src->getOperator()));
Daniel Sandersffc7d582017-03-29 15:37:18 +00001357 auto &SrcGI = *SrcGIOrNull;
1358
1359 // The operators look good: match the opcode and mutate it to the new one.
1360 InsnMatcher.addPredicate<InstructionOpcodeMatcher>(&SrcGI);
1361
1362 unsigned OpIdx = 0;
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001363 unsigned TempOpIdx = 0;
Daniel Sandersffc7d582017-03-29 15:37:18 +00001364 for (const EEVT::TypeSet &Ty : Src->getExtTypes()) {
1365 auto OpTyOrNone = MVTToLLT(Ty.getConcrete());
1366
1367 if (!OpTyOrNone)
1368 return failedImport(
1369 "Result of Src pattern operator has an unsupported type");
1370
1371 // Results don't have a name unless they are the root node. The caller will
1372 // set the name if appropriate.
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001373 OperandMatcher &OM = InsnMatcher.addOperand(OpIdx++, "", TempOpIdx);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001374 OM.addPredicate<LLTOperandMatcher>(*OpTyOrNone);
1375 }
1376
Daniel Sandersffc7d582017-03-29 15:37:18 +00001377 // Match the used operands (i.e. the children of the operator).
1378 for (unsigned i = 0, e = Src->getNumChildren(); i != e; ++i) {
1379 if (auto Error = importChildMatcher(InsnMatcher, Src->getChild(i), OpIdx++,
Daniel Sandersc270c502017-03-30 09:36:33 +00001380 TempOpIdx))
Daniel Sandersffc7d582017-03-29 15:37:18 +00001381 return std::move(Error);
1382 }
1383
1384 return InsnMatcher;
1385}
1386
Daniel Sandersc270c502017-03-30 09:36:33 +00001387Error GlobalISelEmitter::importChildMatcher(InstructionMatcher &InsnMatcher,
1388 TreePatternNode *SrcChild,
1389 unsigned OpIdx,
1390 unsigned &TempOpIdx) const {
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001391 OperandMatcher &OM =
1392 InsnMatcher.addOperand(OpIdx, SrcChild->getName(), TempOpIdx);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001393
1394 if (SrcChild->hasAnyPredicate())
Daniel Sandersd0656a32017-04-13 09:45:37 +00001395 return failedImport("Src pattern child has predicate (" +
1396 explainPredicates(SrcChild) + ")");
Daniel Sandersffc7d582017-03-29 15:37:18 +00001397
1398 ArrayRef<EEVT::TypeSet> ChildTypes = SrcChild->getExtTypes();
1399 if (ChildTypes.size() != 1)
1400 return failedImport("Src pattern child has multiple results");
1401
1402 // Check MBB's before the type check since they are not a known type.
1403 if (!SrcChild->isLeaf()) {
1404 if (SrcChild->getOperator()->isSubClassOf("SDNode")) {
1405 auto &ChildSDNI = CGP.getSDNodeInfo(SrcChild->getOperator());
1406 if (ChildSDNI.getSDClassName() == "BasicBlockSDNode") {
1407 OM.addPredicate<MBBOperandMatcher>();
Daniel Sandersc270c502017-03-30 09:36:33 +00001408 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001409 }
1410 }
Daniel Sandersffc7d582017-03-29 15:37:18 +00001411 }
1412
1413 auto OpTyOrNone = MVTToLLT(ChildTypes.front().getConcrete());
1414 if (!OpTyOrNone)
1415 return failedImport("Src operand has an unsupported type");
1416 OM.addPredicate<LLTOperandMatcher>(*OpTyOrNone);
1417
Daniel Sandersbee57392017-04-04 13:25:23 +00001418 // Check for nested instructions.
1419 if (!SrcChild->isLeaf()) {
1420 // Map the node to a gMIR instruction.
1421 InstructionOperandMatcher &InsnOperand =
1422 OM.addPredicate<InstructionOperandMatcher>();
1423 auto InsnMatcherOrError =
1424 createAndImportSelDAGMatcher(InsnOperand.getInsnMatcher(), SrcChild);
1425 if (auto Error = InsnMatcherOrError.takeError())
1426 return Error;
1427
1428 return Error::success();
1429 }
1430
Daniel Sandersffc7d582017-03-29 15:37:18 +00001431 // Check for constant immediates.
1432 if (auto *ChildInt = dyn_cast<IntInit>(SrcChild->getLeafValue())) {
1433 OM.addPredicate<IntOperandMatcher>(ChildInt->getValue());
Daniel Sandersc270c502017-03-30 09:36:33 +00001434 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001435 }
1436
1437 // Check for def's like register classes or ComplexPattern's.
1438 if (auto *ChildDefInit = dyn_cast<DefInit>(SrcChild->getLeafValue())) {
1439 auto *ChildRec = ChildDefInit->getDef();
1440
1441 // Check for register classes.
1442 if (ChildRec->isSubClassOf("RegisterClass")) {
1443 OM.addPredicate<RegisterBankOperandMatcher>(
1444 Target.getRegisterClass(ChildRec));
Daniel Sandersc270c502017-03-30 09:36:33 +00001445 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001446 }
1447
1448 // Check for ComplexPattern's.
1449 if (ChildRec->isSubClassOf("ComplexPattern")) {
1450 const auto &ComplexPattern = ComplexPatternEquivs.find(ChildRec);
1451 if (ComplexPattern == ComplexPatternEquivs.end())
Daniel Sandersd0656a32017-04-13 09:45:37 +00001452 return failedImport("SelectionDAG ComplexPattern (" +
1453 ChildRec->getName() + ") not mapped to GlobalISel");
Daniel Sandersffc7d582017-03-29 15:37:18 +00001454
1455 const auto &Predicate = OM.addPredicate<ComplexPatternOperandMatcher>(
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001456 OM, *ComplexPattern->second);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001457 TempOpIdx += Predicate.countTemporaryOperands();
Daniel Sandersc270c502017-03-30 09:36:33 +00001458 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001459 }
1460
Daniel Sandersd0656a32017-04-13 09:45:37 +00001461 if (ChildRec->isSubClassOf("ImmLeaf")) {
1462 return failedImport(
1463 "Src pattern child def is an unsupported tablegen class (ImmLeaf)");
1464 }
1465
Daniel Sandersffc7d582017-03-29 15:37:18 +00001466 return failedImport(
1467 "Src pattern child def is an unsupported tablegen class");
1468 }
1469
1470 return failedImport("Src pattern child is an unsupported kind");
1471}
1472
Daniel Sandersc270c502017-03-30 09:36:33 +00001473Error GlobalISelEmitter::importExplicitUseRenderer(
Daniel Sandersffc7d582017-03-29 15:37:18 +00001474 BuildMIAction &DstMIBuilder, TreePatternNode *DstChild,
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001475 const InstructionMatcher &InsnMatcher) const {
Daniel Sandersffc7d582017-03-29 15:37:18 +00001476 // The only non-leaf child we accept is 'bb': it's an operator because
1477 // BasicBlockSDNode isn't inline, but in MI it's just another operand.
1478 if (!DstChild->isLeaf()) {
1479 if (DstChild->getOperator()->isSubClassOf("SDNode")) {
1480 auto &ChildSDNI = CGP.getSDNodeInfo(DstChild->getOperator());
1481 if (ChildSDNI.getSDClassName() == "BasicBlockSDNode") {
1482 DstMIBuilder.addRenderer<CopyRenderer>(InsnMatcher,
1483 DstChild->getName());
Daniel Sandersc270c502017-03-30 09:36:33 +00001484 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001485 }
1486 }
1487 return failedImport("Dst pattern child isn't a leaf node or an MBB");
1488 }
1489
1490 // Otherwise, we're looking for a bog-standard RegisterClass operand.
1491 if (DstChild->hasAnyPredicate())
Daniel Sandersd0656a32017-04-13 09:45:37 +00001492 return failedImport("Dst pattern child has predicate (" +
1493 explainPredicates(DstChild) + ")");
Daniel Sandersffc7d582017-03-29 15:37:18 +00001494
1495 if (auto *ChildDefInit = dyn_cast<DefInit>(DstChild->getLeafValue())) {
1496 auto *ChildRec = ChildDefInit->getDef();
1497
1498 ArrayRef<EEVT::TypeSet> ChildTypes = DstChild->getExtTypes();
1499 if (ChildTypes.size() != 1)
1500 return failedImport("Dst pattern child has multiple results");
1501
1502 auto OpTyOrNone = MVTToLLT(ChildTypes.front().getConcrete());
1503 if (!OpTyOrNone)
1504 return failedImport("Dst operand has an unsupported type");
1505
1506 if (ChildRec->isSubClassOf("Register")) {
1507 DstMIBuilder.addRenderer<AddRegisterRenderer>(ChildRec);
Daniel Sandersc270c502017-03-30 09:36:33 +00001508 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001509 }
1510
1511 if (ChildRec->isSubClassOf("RegisterClass")) {
1512 DstMIBuilder.addRenderer<CopyRenderer>(InsnMatcher, DstChild->getName());
Daniel Sandersc270c502017-03-30 09:36:33 +00001513 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001514 }
1515
1516 if (ChildRec->isSubClassOf("ComplexPattern")) {
1517 const auto &ComplexPattern = ComplexPatternEquivs.find(ChildRec);
1518 if (ComplexPattern == ComplexPatternEquivs.end())
1519 return failedImport(
1520 "SelectionDAG ComplexPattern not mapped to GlobalISel");
1521
1522 SmallVector<OperandPlaceholder, 2> RenderedOperands;
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001523 const OperandMatcher &OM = InsnMatcher.getOperand(DstChild->getName());
1524 for (unsigned I = 0; I < OM.countTemporaryOperands(); ++I)
1525 RenderedOperands.push_back(OperandPlaceholder::CreateTemporary(
1526 OM.getAllocatedTemporariesBaseID() + I));
Daniel Sandersffc7d582017-03-29 15:37:18 +00001527 DstMIBuilder.addRenderer<RenderComplexPatternOperand>(
1528 *ComplexPattern->second, RenderedOperands);
Daniel Sandersc270c502017-03-30 09:36:33 +00001529 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001530 }
1531
Daniel Sandersd0656a32017-04-13 09:45:37 +00001532 if (ChildRec->isSubClassOf("SDNodeXForm"))
1533 return failedImport("Dst pattern child def is an unsupported tablegen "
1534 "class (SDNodeXForm)");
1535
Daniel Sandersffc7d582017-03-29 15:37:18 +00001536 return failedImport(
1537 "Dst pattern child def is an unsupported tablegen class");
1538 }
1539
1540 return failedImport("Dst pattern child is an unsupported kind");
1541}
1542
Daniel Sandersc270c502017-03-30 09:36:33 +00001543Expected<BuildMIAction &> GlobalISelEmitter::createAndImportInstructionRenderer(
Daniel Sandersffc7d582017-03-29 15:37:18 +00001544 RuleMatcher &M, const TreePatternNode *Dst,
1545 const InstructionMatcher &InsnMatcher) const {
1546 Record *DstOp = Dst->getOperator();
Daniel Sandersd0656a32017-04-13 09:45:37 +00001547 if (!DstOp->isSubClassOf("Instruction")) {
1548 if (DstOp->isSubClassOf("ValueType"))
1549 return failedImport(
1550 "Pattern operator isn't an instruction (it's a ValueType)");
Daniel Sandersffc7d582017-03-29 15:37:18 +00001551 return failedImport("Pattern operator isn't an instruction");
Daniel Sandersd0656a32017-04-13 09:45:37 +00001552 }
Daniel Sandersffc7d582017-03-29 15:37:18 +00001553 auto &DstI = Target.getInstruction(DstOp);
1554
1555 auto &DstMIBuilder = M.addAction<BuildMIAction>(&DstI, InsnMatcher);
1556
1557 // Render the explicit defs.
1558 for (unsigned I = 0; I < DstI.Operands.NumDefs; ++I) {
1559 const auto &DstIOperand = DstI.Operands[I];
1560 DstMIBuilder.addRenderer<CopyRenderer>(InsnMatcher, DstIOperand.Name);
1561 }
1562
Daniel Sanders0ed28822017-04-12 08:23:08 +00001563 // Figure out which operands need defaults inserted. Operands that subclass
1564 // OperandWithDefaultOps are considered from left to right until we have
1565 // enough operands to render the instruction.
1566 SmallSet<unsigned, 2> DefaultOperands;
1567 unsigned DstINumUses = DstI.Operands.size() - DstI.Operands.NumDefs;
1568 unsigned NumDefaultOperands = 0;
1569 for (unsigned I = 0; I < DstINumUses &&
1570 DstINumUses > Dst->getNumChildren() + NumDefaultOperands;
1571 ++I) {
1572 const auto &DstIOperand = DstI.Operands[DstI.Operands.NumDefs + I];
1573 if (DstIOperand.Rec->isSubClassOf("OperandWithDefaultOps")) {
1574 DefaultOperands.insert(I);
1575 NumDefaultOperands +=
1576 DstIOperand.Rec->getValueAsDag("DefaultOps")->getNumArgs();
1577 }
1578 }
1579 if (DstINumUses > Dst->getNumChildren() + DefaultOperands.size())
1580 return failedImport("Insufficient operands supplied and default ops "
1581 "couldn't make up the shortfall");
1582 if (DstINumUses < Dst->getNumChildren() + DefaultOperands.size())
1583 return failedImport("Too many operands supplied");
1584
Daniel Sandersffc7d582017-03-29 15:37:18 +00001585 // Render the explicit uses.
Daniel Sanders0ed28822017-04-12 08:23:08 +00001586 unsigned Child = 0;
1587 for (unsigned I = 0; I != DstINumUses; ++I) {
1588 // If we need to insert default ops here, then do so.
1589 if (DefaultOperands.count(I)) {
1590 const auto &DstIOperand = DstI.Operands[DstI.Operands.NumDefs + I];
1591
1592 DagInit *DefaultOps = DstIOperand.Rec->getValueAsDag("DefaultOps");
1593 for (const auto *DefaultOp : DefaultOps->args()) {
1594 // Look through ValueType operators.
1595 if (const DagInit *DefaultDagOp = dyn_cast<DagInit>(DefaultOp)) {
1596 if (const DefInit *DefaultDagOperator =
1597 dyn_cast<DefInit>(DefaultDagOp->getOperator())) {
1598 if (DefaultDagOperator->getDef()->isSubClassOf("ValueType"))
1599 DefaultOp = DefaultDagOp->getArg(0);
1600 }
1601 }
1602
1603 if (const DefInit *DefaultDefOp = dyn_cast<DefInit>(DefaultOp)) {
1604 DstMIBuilder.addRenderer<AddRegisterRenderer>(DefaultDefOp->getDef());
1605 continue;
1606 }
1607
1608 if (const IntInit *DefaultIntOp = dyn_cast<IntInit>(DefaultOp)) {
1609 DstMIBuilder.addRenderer<ImmRenderer>(DefaultIntOp->getValue());
1610 continue;
1611 }
1612
1613 return failedImport("Could not add default op");
1614 }
1615
1616 continue;
1617 }
1618
1619 if (auto Error = importExplicitUseRenderer(
1620 DstMIBuilder, Dst->getChild(Child), InsnMatcher))
Daniel Sandersffc7d582017-03-29 15:37:18 +00001621 return std::move(Error);
Daniel Sanders0ed28822017-04-12 08:23:08 +00001622 ++Child;
Daniel Sandersffc7d582017-03-29 15:37:18 +00001623 }
1624
1625 return DstMIBuilder;
1626}
1627
Daniel Sandersc270c502017-03-30 09:36:33 +00001628Error GlobalISelEmitter::importImplicitDefRenderers(
Daniel Sandersffc7d582017-03-29 15:37:18 +00001629 BuildMIAction &DstMIBuilder,
1630 const std::vector<Record *> &ImplicitDefs) const {
1631 if (!ImplicitDefs.empty())
1632 return failedImport("Pattern defines a physical register");
Daniel Sandersc270c502017-03-30 09:36:33 +00001633 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001634}
1635
1636Expected<RuleMatcher> GlobalISelEmitter::runOnPattern(const PatternToMatch &P) {
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001637 // Keep track of the matchers and actions to emit.
Ahmed Bougacha9aa4c102017-02-04 00:47:08 +00001638 RuleMatcher M;
1639 M.addAction<DebugCommentAction>(P);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001640
Daniel Sandersc270c502017-03-30 09:36:33 +00001641 if (auto Error = importRulePredicates(M, P.getPredicates()->getValues()))
Daniel Sandersffc7d582017-03-29 15:37:18 +00001642 return std::move(Error);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001643
1644 // Next, analyze the pattern operators.
1645 TreePatternNode *Src = P.getSrcPattern();
1646 TreePatternNode *Dst = P.getDstPattern();
1647
1648 // If the root of either pattern isn't a simple operator, ignore it.
Daniel Sandersd0656a32017-04-13 09:45:37 +00001649 if (auto Err = isTrivialOperatorNode(Dst))
1650 return failedImport("Dst pattern root isn't a trivial operator (" +
1651 toString(std::move(Err)) + ")");
1652 if (auto Err = isTrivialOperatorNode(Src))
1653 return failedImport("Src pattern root isn't a trivial operator (" +
1654 toString(std::move(Err)) + ")");
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001655
Daniel Sandersbee57392017-04-04 13:25:23 +00001656 // Start with the defined operands (i.e., the results of the root operator).
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001657 Record *DstOp = Dst->getOperator();
1658 if (!DstOp->isSubClassOf("Instruction"))
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001659 return failedImport("Pattern operator isn't an instruction");
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001660
1661 auto &DstI = Target.getInstruction(DstOp);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001662 if (DstI.Operands.NumDefs != Src->getExtTypes().size())
Daniel Sandersd0656a32017-04-13 09:45:37 +00001663 return failedImport("Src pattern results and dst MI defs are different (" +
1664 to_string(Src->getExtTypes().size()) + " def(s) vs " +
1665 to_string(DstI.Operands.NumDefs) + " def(s))");
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001666
Daniel Sandersffc7d582017-03-29 15:37:18 +00001667 InstructionMatcher &InsnMatcherTemp = M.addInstructionMatcher();
Daniel Sandersc270c502017-03-30 09:36:33 +00001668 auto InsnMatcherOrError = createAndImportSelDAGMatcher(InsnMatcherTemp, Src);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001669 if (auto Error = InsnMatcherOrError.takeError())
1670 return std::move(Error);
1671 InstructionMatcher &InsnMatcher = InsnMatcherOrError.get();
1672
1673 // The root of the match also has constraints on the register bank so that it
1674 // matches the result instruction.
1675 unsigned OpIdx = 0;
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001676 for (const EEVT::TypeSet &Ty : Src->getExtTypes()) {
Daniel Sandersffc7d582017-03-29 15:37:18 +00001677 (void)Ty;
1678
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001679 const auto &DstIOperand = DstI.Operands[OpIdx];
1680 Record *DstIOpRec = DstIOperand.Rec;
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001681 if (!DstIOpRec->isSubClassOf("RegisterClass"))
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001682 return failedImport("Dst MI def isn't a register class");
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001683
Daniel Sandersffc7d582017-03-29 15:37:18 +00001684 OperandMatcher &OM = InsnMatcher.getOperand(OpIdx);
1685 OM.setSymbolicName(DstIOperand.Name);
Daniel Sandersdc662ff2017-01-26 11:10:14 +00001686 OM.addPredicate<RegisterBankOperandMatcher>(
1687 Target.getRegisterClass(DstIOpRec));
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001688 ++OpIdx;
1689 }
1690
Daniel Sandersc270c502017-03-30 09:36:33 +00001691 auto DstMIBuilderOrError =
1692 createAndImportInstructionRenderer(M, Dst, InsnMatcher);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001693 if (auto Error = DstMIBuilderOrError.takeError())
1694 return std::move(Error);
1695 BuildMIAction &DstMIBuilder = DstMIBuilderOrError.get();
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001696
Daniel Sandersffc7d582017-03-29 15:37:18 +00001697 // Render the implicit defs.
1698 // These are only added to the root of the result.
Daniel Sandersc270c502017-03-30 09:36:33 +00001699 if (auto Error = importImplicitDefRenderers(DstMIBuilder, P.getDstRegs()))
Daniel Sandersffc7d582017-03-29 15:37:18 +00001700 return std::move(Error);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001701
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001702 // We're done with this pattern! It's eligible for GISel emission; return it.
Daniel Sandersb41ce2b2017-02-20 14:31:27 +00001703 ++NumPatternImported;
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001704 return std::move(M);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001705}
1706
1707void GlobalISelEmitter::run(raw_ostream &OS) {
1708 // Track the GINodeEquiv definitions.
1709 gatherNodeEquivs();
1710
1711 emitSourceFileHeader(("Global Instruction Selector for the " +
1712 Target.getName() + " target").str(), OS);
Daniel Sandersb41ce2b2017-02-20 14:31:27 +00001713 std::vector<RuleMatcher> Rules;
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001714 // Look through the SelectionDAG patterns we found, possibly emitting some.
1715 for (const PatternToMatch &Pat : CGP.ptms()) {
1716 ++NumPatternTotal;
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001717 auto MatcherOrErr = runOnPattern(Pat);
1718
1719 // The pattern analysis can fail, indicating an unsupported pattern.
1720 // Report that if we've been asked to do so.
1721 if (auto Err = MatcherOrErr.takeError()) {
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001722 if (WarnOnSkippedPatterns) {
1723 PrintWarning(Pat.getSrcRecord()->getLoc(),
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001724 "Skipped pattern: " + toString(std::move(Err)));
1725 } else {
1726 consumeError(std::move(Err));
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001727 }
Daniel Sandersb41ce2b2017-02-20 14:31:27 +00001728 ++NumPatternImportsSkipped;
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001729 continue;
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001730 }
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001731
Daniel Sandersb41ce2b2017-02-20 14:31:27 +00001732 Rules.push_back(std::move(MatcherOrErr.get()));
1733 }
1734
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001735 std::stable_sort(Rules.begin(), Rules.end(),
1736 [&](const RuleMatcher &A, const RuleMatcher &B) {
Daniel Sanders759ff412017-02-24 13:58:11 +00001737 if (A.isHigherPriorityThan(B)) {
1738 assert(!B.isHigherPriorityThan(A) && "Cannot be more important "
1739 "and less important at "
1740 "the same time");
1741 return true;
1742 }
1743 return false;
1744 });
1745
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001746 unsigned MaxTemporaries = 0;
1747 for (const auto &Rule : Rules)
1748 MaxTemporaries = std::max(MaxTemporaries, Rule.countTemporaryOperands());
1749
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001750 OS << "#ifdef GET_GLOBALISEL_PREDICATE_BITSET\n"
1751 << "const unsigned MAX_SUBTARGET_PREDICATES = " << SubtargetFeatures.size()
1752 << ";\n"
1753 << "using PredicateBitset = "
1754 "llvm::PredicateBitsetImpl<MAX_SUBTARGET_PREDICATES>;\n"
1755 << "#endif // ifdef GET_GLOBALISEL_PREDICATE_BITSET\n\n";
1756
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001757 OS << "#ifdef GET_GLOBALISEL_TEMPORARIES_DECL\n";
1758 for (unsigned I = 0; I < MaxTemporaries; ++I)
1759 OS << " mutable MachineOperand TempOp" << I << ";\n";
1760 OS << "#endif // ifdef GET_GLOBALISEL_TEMPORARIES_DECL\n\n";
1761
1762 OS << "#ifdef GET_GLOBALISEL_TEMPORARIES_INIT\n";
1763 for (unsigned I = 0; I < MaxTemporaries; ++I)
1764 OS << ", TempOp" << I << "(MachineOperand::CreatePlaceholder())\n";
1765 OS << "#endif // ifdef GET_GLOBALISEL_TEMPORARIES_INIT\n\n";
1766
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001767 OS << "#ifdef GET_GLOBALISEL_IMPL\n";
1768 SubtargetFeatureInfo::emitSubtargetFeatureBitEnumeration(SubtargetFeatures,
1769 OS);
1770 SubtargetFeatureInfo::emitNameTable(SubtargetFeatures, OS);
1771 SubtargetFeatureInfo::emitComputeAvailableFeatures(
1772 Target.getName(), "InstructionSelector", "computeAvailableFeatures",
1773 SubtargetFeatures, OS);
1774
1775 OS << "bool " << Target.getName()
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001776 << "InstructionSelector::selectImpl(MachineInstr &I) const {\n"
1777 << " MachineFunction &MF = *I.getParent()->getParent();\n"
1778 << " const MachineRegisterInfo &MRI = MF.getRegInfo();\n";
1779
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001780 for (auto &Rule : Rules) {
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001781 Rule.emit(OS, SubtargetFeatures);
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001782 ++NumPatternEmitted;
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001783 }
1784
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001785 OS << " return false;\n"
1786 << "}\n"
1787 << "#endif // ifdef GET_GLOBALISEL_IMPL\n";
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001788}
1789
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001790void GlobalISelEmitter::declareSubtargetFeature(Record *Predicate) {
1791 if (SubtargetFeatures.count(Predicate) == 0)
1792 SubtargetFeatures.emplace(
1793 Predicate, SubtargetFeatureInfo(Predicate, SubtargetFeatures.size()));
1794}
1795
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001796} // end anonymous namespace
1797
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001798//===----------------------------------------------------------------------===//
1799
1800namespace llvm {
1801void EmitGlobalISel(RecordKeeper &RK, raw_ostream &OS) {
1802 GlobalISelEmitter(RK).run(OS);
1803}
1804} // End llvm namespace