blob: 7a500eaf4111e2d9566d73eb47249cd7b83ef579 [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()) {
Igor Breger1593a742017-04-26 15:59:05 +000083 OS << "LLT::vector(" << Ty.getNumElements() << ", " << Ty.getScalarSizeInBits()
Daniel Sanders52b4ce72017-03-07 23:20:35 +000084 << ")";
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;
Ahmed Bougacha36f70352016-12-21 23:26:20 +000094/// Convert an MVT to an equivalent LLT if possible, or the invalid LLT() for
95/// MVTs that don't map cleanly to an LLT (e.g., iPTR, *any, ...).
Daniel Sanders52b4ce72017-03-07 23:20:35 +000096static Optional<LLTCodeGen> MVTToLLT(MVT::SimpleValueType SVT) {
Ahmed Bougacha36f70352016-12-21 23:26:20 +000097 MVT VT(SVT);
Daniel Sanders52b4ce72017-03-07 23:20:35 +000098 if (VT.isVector() && VT.getVectorNumElements() != 1)
99 return LLTCodeGen(LLT::vector(VT.getVectorNumElements(), VT.getScalarSizeInBits()));
100 if (VT.isInteger() || VT.isFloatingPoint())
101 return LLTCodeGen(LLT::scalar(VT.getSizeInBits()));
102 return None;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000103}
104
Daniel Sandersd0656a32017-04-13 09:45:37 +0000105static std::string explainPredicates(const TreePatternNode *N) {
106 std::string Explanation = "";
107 StringRef Separator = "";
108 for (const auto &P : N->getPredicateFns()) {
109 Explanation +=
110 (Separator + P.getOrigPatFragRecord()->getRecord()->getName()).str();
111 if (P.isAlwaysTrue())
112 Explanation += " always-true";
113 if (P.isImmediatePattern())
114 Explanation += " immediate";
115 }
116 return Explanation;
117}
118
Daniel Sandersd0656a32017-04-13 09:45:37 +0000119std::string explainOperator(Record *Operator) {
120 if (Operator->isSubClassOf("SDNode"))
121 return " (" + Operator->getValueAsString("Opcode") + ")";
122
123 if (Operator->isSubClassOf("Intrinsic"))
124 return (" (Operator is an Intrinsic, " + Operator->getName() + ")").str();
125
126 return " (Operator not understood)";
127}
128
129/// Helper function to let the emitter report skip reason error messages.
130static Error failedImport(const Twine &Reason) {
131 return make_error<StringError>(Reason, inconvertibleErrorCode());
132}
133
134static Error isTrivialOperatorNode(const TreePatternNode *N) {
135 std::string Explanation = "";
136 std::string Separator = "";
137 if (N->isLeaf()) {
138 Explanation = "Is a leaf";
139 Separator = ", ";
140 }
141
142 if (N->hasAnyPredicate()) {
143 Explanation = Separator + "Has a predicate (" + explainPredicates(N) + ")";
144 Separator = ", ";
145 }
146
147 if (N->getTransformFn()) {
148 Explanation += Separator + "Has a transform function";
149 Separator = ", ";
150 }
151
152 if (!N->isLeaf() && !N->hasAnyPredicate() && !N->getTransformFn())
153 return Error::success();
154
155 return failedImport(Explanation);
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000156}
157
158//===- Matchers -----------------------------------------------------------===//
159
Daniel Sandersbee57392017-04-04 13:25:23 +0000160class OperandMatcher;
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000161class MatchAction;
162
163/// Generates code to check that a match rule matches.
164class RuleMatcher {
165 /// A list of matchers that all need to succeed for the current rule to match.
166 /// FIXME: This currently supports a single match position but could be
167 /// extended to support multiple positions to support div/rem fusion or
168 /// load-multiple instructions.
169 std::vector<std::unique_ptr<InstructionMatcher>> Matchers;
170
171 /// A list of actions that need to be taken when all predicates in this rule
172 /// have succeeded.
173 std::vector<std::unique_ptr<MatchAction>> Actions;
174
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000175 /// A map of instruction matchers to the local variables created by
176 /// emitCxxCaptureStmts().
177 std::map<const InstructionMatcher *, std::string> InsnVariableNames;
178
179 /// ID for the next instruction variable defined with defineInsnVar()
180 unsigned NextInsnVarID;
181
Daniel Sanderse7b0d662017-04-21 15:59:56 +0000182 std::vector<Record *> RequiredFeatures;
183
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000184public:
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000185 RuleMatcher()
186 : Matchers(), Actions(), InsnVariableNames(), NextInsnVarID(0) {}
Zachary Turnerb7dbd872017-03-20 19:56:52 +0000187 RuleMatcher(RuleMatcher &&Other) = default;
188 RuleMatcher &operator=(RuleMatcher &&Other) = default;
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000189
190 InstructionMatcher &addInstructionMatcher();
Daniel Sanderse7b0d662017-04-21 15:59:56 +0000191 void addRequiredFeature(Record *Feature);
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000192
193 template <class Kind, class... Args> Kind &addAction(Args &&... args);
194
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000195 std::string defineInsnVar(raw_ostream &OS, const InstructionMatcher &Matcher,
196 StringRef Value);
197 StringRef getInsnVarName(const InstructionMatcher &InsnMatcher) const;
198
Daniel Sandersbee57392017-04-04 13:25:23 +0000199 void emitCxxCapturedInsnList(raw_ostream &OS);
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000200 void emitCxxCaptureStmts(raw_ostream &OS, StringRef Expr);
201
Daniel Sanderse9fdba32017-04-29 17:30:09 +0000202void emit(raw_ostream &OS, SubtargetFeatureInfoMap SubtargetFeatures);
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000203
Daniel Sanderse9fdba32017-04-29 17:30:09 +0000204/// Compare the priority of this object and B.
205///
206/// Returns true if this object is more important than B.
207bool isHigherPriorityThan(const RuleMatcher &B) const;
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000208
Daniel Sanderse9fdba32017-04-29 17:30:09 +0000209/// Report the maximum number of temporary operands needed by the rule
210/// matcher.
211unsigned countRendererFns() const;
Daniel Sanders2deea182017-04-22 15:11:04 +0000212
Daniel Sanderse9fdba32017-04-29 17:30:09 +0000213// FIXME: Remove this as soon as possible
214InstructionMatcher &insnmatcher_front() const { return *Matchers.front(); }
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000215};
216
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000217template <class PredicateTy> class PredicateListMatcher {
218private:
219 typedef std::vector<std::unique_ptr<PredicateTy>> PredicateVec;
220 PredicateVec Predicates;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000221
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000222public:
223 /// Construct a new operand predicate and add it to the matcher.
224 template <class Kind, class... Args>
225 Kind &addPredicate(Args&&... args) {
Ahmed Bougachab67a3ce2017-01-26 22:07:37 +0000226 Predicates.emplace_back(
227 llvm::make_unique<Kind>(std::forward<Args>(args)...));
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000228 return *static_cast<Kind *>(Predicates.back().get());
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000229 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000230
231 typename PredicateVec::const_iterator predicates_begin() const { return Predicates.begin(); }
232 typename PredicateVec::const_iterator predicates_end() const { return Predicates.end(); }
233 iterator_range<typename PredicateVec::const_iterator> predicates() const {
234 return make_range(predicates_begin(), predicates_end());
235 }
Daniel Sanders759ff412017-02-24 13:58:11 +0000236 typename PredicateVec::size_type predicates_size() const { return Predicates.size(); }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000237
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000238 /// Emit a C++ expression that tests whether all the predicates are met.
Ahmed Bougachab67a3ce2017-01-26 22:07:37 +0000239 template <class... Args>
Daniel Sandersf8c804f2017-01-28 11:10:42 +0000240 void emitCxxPredicateListExpr(raw_ostream &OS, Args &&... args) const {
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000241 if (Predicates.empty()) {
242 OS << "true";
243 return;
244 }
245
246 StringRef Separator = "";
247 for (const auto &Predicate : predicates()) {
248 OS << Separator << "(";
Ahmed Bougachab67a3ce2017-01-26 22:07:37 +0000249 Predicate->emitCxxPredicateExpr(OS, std::forward<Args>(args)...);
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000250 OS << ")";
Ahmed Bougacha905af9f2017-02-04 00:47:02 +0000251 Separator = " &&\n";
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000252 }
253 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000254};
255
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000256/// Generates code to check a predicate of an operand.
257///
258/// Typical predicates include:
259/// * Operand is a particular register.
260/// * Operand is assigned a particular register bank.
261/// * Operand is an MBB.
262class OperandPredicateMatcher {
263public:
Daniel Sanders759ff412017-02-24 13:58:11 +0000264 /// This enum is used for RTTI and also defines the priority that is given to
265 /// the predicate when generating the matcher code. Kinds with higher priority
266 /// must be tested first.
267 ///
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000268 /// The relative priority of OPM_LLT, OPM_RegBank, and OPM_MBB do not matter
269 /// but OPM_Int must have priority over OPM_RegBank since constant integers
270 /// are represented by a virtual register defined by a G_CONSTANT instruction.
Daniel Sanders759ff412017-02-24 13:58:11 +0000271 enum PredicateKind {
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000272 OPM_ComplexPattern,
Daniel Sandersbee57392017-04-04 13:25:23 +0000273 OPM_Instruction,
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000274 OPM_Int,
Daniel Sanders759ff412017-02-24 13:58:11 +0000275 OPM_LLT,
276 OPM_RegBank,
277 OPM_MBB,
278 };
279
280protected:
281 PredicateKind Kind;
282
283public:
284 OperandPredicateMatcher(PredicateKind Kind) : Kind(Kind) {}
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000285 virtual ~OperandPredicateMatcher() {}
286
Daniel Sanders759ff412017-02-24 13:58:11 +0000287 PredicateKind getKind() const { return Kind; }
288
Daniel Sandersbee57392017-04-04 13:25:23 +0000289 /// Return the OperandMatcher for the specified operand or nullptr if there
290 /// isn't one by that name in this operand predicate matcher.
291 ///
292 /// InstructionOperandMatcher is the only subclass that can return non-null
293 /// for this.
294 virtual Optional<const OperandMatcher *>
Daniel Sandersdb7ed372017-04-04 13:52:00 +0000295 getOptionalOperand(StringRef SymbolicName) const {
Daniel Sandersbee57392017-04-04 13:25:23 +0000296 assert(!SymbolicName.empty() && "Cannot lookup unnamed operand");
297 return None;
298 }
299
300 /// Emit C++ statements to capture instructions into local variables.
301 ///
302 /// Only InstructionOperandMatcher needs to do anything for this method.
303 virtual void emitCxxCaptureStmts(raw_ostream &OS, RuleMatcher &Rule,
304 StringRef Expr) const {}
305
Daniel Sanderse604ef52017-02-20 15:30:43 +0000306 /// Emit a C++ expression that checks the predicate for the given operand.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000307 virtual void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanderse604ef52017-02-20 15:30:43 +0000308 StringRef OperandExpr) const = 0;
Daniel Sanders759ff412017-02-24 13:58:11 +0000309
310 /// Compare the priority of this object and B.
311 ///
312 /// Returns true if this object is more important than B.
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000313 virtual bool isHigherPriorityThan(const OperandPredicateMatcher &B) const {
314 return Kind < B.Kind;
Daniel Sanders759ff412017-02-24 13:58:11 +0000315 };
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000316
317 /// Report the maximum number of temporary operands needed by the predicate
318 /// matcher.
Daniel Sanders2deea182017-04-22 15:11:04 +0000319 virtual unsigned countRendererFns() const { return 0; }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000320};
321
322/// Generates code to check that an operand is a particular LLT.
323class LLTOperandMatcher : public OperandPredicateMatcher {
324protected:
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000325 LLTCodeGen Ty;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000326
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000327public:
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000328 LLTOperandMatcher(const LLTCodeGen &Ty)
Daniel Sanders759ff412017-02-24 13:58:11 +0000329 : OperandPredicateMatcher(OPM_LLT), Ty(Ty) {}
330
331 static bool classof(const OperandPredicateMatcher *P) {
332 return P->getKind() == OPM_LLT;
333 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000334
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000335 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanderse604ef52017-02-20 15:30:43 +0000336 StringRef OperandExpr) const override {
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000337 OS << "MRI.getType(" << OperandExpr << ".getReg()) == (";
338 Ty.emitCxxConstructorCall(OS);
339 OS << ")";
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000340 }
341};
342
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000343/// Generates code to check that an operand is a particular target constant.
344class ComplexPatternOperandMatcher : public OperandPredicateMatcher {
345protected:
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000346 const OperandMatcher &Operand;
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000347 const Record &TheDef;
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000348
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000349 unsigned getAllocatedTemporariesBaseID() const;
350
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000351public:
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000352 ComplexPatternOperandMatcher(const OperandMatcher &Operand,
353 const Record &TheDef)
354 : OperandPredicateMatcher(OPM_ComplexPattern), Operand(Operand),
355 TheDef(TheDef) {}
356
357 static bool classof(const OperandPredicateMatcher *P) {
358 return P->getKind() == OPM_ComplexPattern;
359 }
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000360
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000361 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000362 StringRef OperandExpr) const override {
Daniel Sanders2deea182017-04-22 15:11:04 +0000363 unsigned ID = getAllocatedTemporariesBaseID();
364 OS << "(Renderer" << ID << " = " << TheDef.getValueAsString("MatcherFn")
365 << "(" << OperandExpr << "))";
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000366 }
367
Daniel Sanders2deea182017-04-22 15:11:04 +0000368 unsigned countRendererFns() const override {
369 return 1;
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000370 }
371};
372
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000373/// Generates code to check that an operand is in a particular register bank.
374class RegisterBankOperandMatcher : public OperandPredicateMatcher {
375protected:
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000376 const CodeGenRegisterClass &RC;
377
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000378public:
Daniel Sanders759ff412017-02-24 13:58:11 +0000379 RegisterBankOperandMatcher(const CodeGenRegisterClass &RC)
380 : OperandPredicateMatcher(OPM_RegBank), RC(RC) {}
381
382 static bool classof(const OperandPredicateMatcher *P) {
383 return P->getKind() == OPM_RegBank;
384 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000385
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000386 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanderse604ef52017-02-20 15:30:43 +0000387 StringRef OperandExpr) const override {
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000388 OS << "(&RBI.getRegBankFromRegClass(" << RC.getQualifiedName()
Daniel Sanderse604ef52017-02-20 15:30:43 +0000389 << "RegClass) == RBI.getRegBank(" << OperandExpr
390 << ".getReg(), MRI, TRI))";
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000391 }
392};
393
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000394/// Generates code to check that an operand is a basic block.
395class MBBOperandMatcher : public OperandPredicateMatcher {
396public:
Daniel Sanders759ff412017-02-24 13:58:11 +0000397 MBBOperandMatcher() : OperandPredicateMatcher(OPM_MBB) {}
398
399 static bool classof(const OperandPredicateMatcher *P) {
400 return P->getKind() == OPM_MBB;
401 }
402
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000403 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanderse604ef52017-02-20 15:30:43 +0000404 StringRef OperandExpr) const override {
405 OS << OperandExpr << ".isMBB()";
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000406 }
407};
408
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000409/// Generates code to check that an operand is a particular int.
410class IntOperandMatcher : public OperandPredicateMatcher {
411protected:
412 int64_t Value;
413
414public:
415 IntOperandMatcher(int64_t Value)
416 : OperandPredicateMatcher(OPM_Int), Value(Value) {}
417
418 static bool classof(const OperandPredicateMatcher *P) {
419 return P->getKind() == OPM_Int;
420 }
421
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000422 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Simon Pilgrimd0302912017-02-24 17:20:27 +0000423 StringRef OperandExpr) const override {
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000424 OS << "isOperandImmEqual(" << OperandExpr << ", " << Value << ", MRI)";
425 }
426};
427
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000428/// Generates code to check that a set of predicates match for a particular
429/// operand.
430class OperandMatcher : public PredicateListMatcher<OperandPredicateMatcher> {
431protected:
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000432 InstructionMatcher &Insn;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000433 unsigned OpIdx;
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000434 std::string SymbolicName;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000435
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000436 /// The index of the first temporary variable allocated to this operand. The
437 /// number of allocated temporaries can be found with
Daniel Sanders2deea182017-04-22 15:11:04 +0000438 /// countRendererFns().
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000439 unsigned AllocatedTemporariesBaseID;
440
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000441public:
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000442 OperandMatcher(InstructionMatcher &Insn, unsigned OpIdx,
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000443 const std::string &SymbolicName,
444 unsigned AllocatedTemporariesBaseID)
445 : Insn(Insn), OpIdx(OpIdx), SymbolicName(SymbolicName),
446 AllocatedTemporariesBaseID(AllocatedTemporariesBaseID) {}
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000447
448 bool hasSymbolicName() const { return !SymbolicName.empty(); }
449 const StringRef getSymbolicName() const { return SymbolicName; }
Daniel Sandersffc7d582017-03-29 15:37:18 +0000450 void setSymbolicName(StringRef Name) {
451 assert(SymbolicName.empty() && "Operand already has a symbolic name");
452 SymbolicName = Name;
453 }
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000454 unsigned getOperandIndex() const { return OpIdx; }
455
Daniel Sandersdb7ed372017-04-04 13:52:00 +0000456 std::string getOperandExpr(StringRef InsnVarName) const {
Pavel Labath52a82e22017-02-21 09:19:41 +0000457 return (InsnVarName + ".getOperand(" + llvm::to_string(OpIdx) + ")").str();
Daniel Sanderse604ef52017-02-20 15:30:43 +0000458 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000459
Daniel Sandersbee57392017-04-04 13:25:23 +0000460 Optional<const OperandMatcher *>
461 getOptionalOperand(StringRef DesiredSymbolicName) const {
462 assert(!DesiredSymbolicName.empty() && "Cannot lookup unnamed operand");
463 if (DesiredSymbolicName == SymbolicName)
464 return this;
465 for (const auto &OP : predicates()) {
466 const auto &MaybeOperand = OP->getOptionalOperand(DesiredSymbolicName);
467 if (MaybeOperand.hasValue())
468 return MaybeOperand.getValue();
469 }
470 return None;
471 }
472
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000473 InstructionMatcher &getInstructionMatcher() const { return Insn; }
474
Daniel Sandersbee57392017-04-04 13:25:23 +0000475 /// Emit C++ statements to capture instructions into local variables.
476 void emitCxxCaptureStmts(raw_ostream &OS, RuleMatcher &Rule,
477 StringRef OperandExpr) const {
478 for (const auto &Predicate : predicates())
479 Predicate->emitCxxCaptureStmts(OS, Rule, OperandExpr);
480 }
481
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000482 /// Emit a C++ expression that tests whether the instruction named in
483 /// InsnVarName matches all the predicate and all the operands.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000484 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sandersdb7ed372017-04-04 13:52:00 +0000485 StringRef InsnVarName) const {
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000486 OS << "(/* ";
487 if (SymbolicName.empty())
488 OS << "Operand " << OpIdx;
489 else
490 OS << SymbolicName;
491 OS << " */ ";
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000492 emitCxxPredicateListExpr(OS, Rule, getOperandExpr(InsnVarName));
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000493 OS << ")";
494 }
Daniel Sanders759ff412017-02-24 13:58:11 +0000495
496 /// Compare the priority of this object and B.
497 ///
498 /// Returns true if this object is more important than B.
499 bool isHigherPriorityThan(const OperandMatcher &B) const {
500 // Operand matchers involving more predicates have higher priority.
501 if (predicates_size() > B.predicates_size())
502 return true;
503 if (predicates_size() < B.predicates_size())
504 return false;
505
506 // This assumes that predicates are added in a consistent order.
507 for (const auto &Predicate : zip(predicates(), B.predicates())) {
508 if (std::get<0>(Predicate)->isHigherPriorityThan(*std::get<1>(Predicate)))
509 return true;
510 if (std::get<1>(Predicate)->isHigherPriorityThan(*std::get<0>(Predicate)))
511 return false;
512 }
513
514 return false;
515 };
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000516
517 /// Report the maximum number of temporary operands needed by the operand
518 /// matcher.
Daniel Sanders2deea182017-04-22 15:11:04 +0000519 unsigned countRendererFns() const {
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000520 return std::accumulate(
521 predicates().begin(), predicates().end(), 0,
522 [](unsigned A,
523 const std::unique_ptr<OperandPredicateMatcher> &Predicate) {
Daniel Sanders2deea182017-04-22 15:11:04 +0000524 return A + Predicate->countRendererFns();
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000525 });
526 }
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000527
528 unsigned getAllocatedTemporariesBaseID() const {
529 return AllocatedTemporariesBaseID;
530 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000531};
532
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000533unsigned ComplexPatternOperandMatcher::getAllocatedTemporariesBaseID() const {
534 return Operand.getAllocatedTemporariesBaseID();
535}
536
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000537/// Generates code to check a predicate on an instruction.
538///
539/// Typical predicates include:
540/// * The opcode of the instruction is a particular value.
541/// * The nsw/nuw flag is/isn't set.
542class InstructionPredicateMatcher {
Daniel Sanders759ff412017-02-24 13:58:11 +0000543protected:
544 /// This enum is used for RTTI and also defines the priority that is given to
545 /// the predicate when generating the matcher code. Kinds with higher priority
546 /// must be tested first.
547 enum PredicateKind {
548 IPM_Opcode,
549 };
550
551 PredicateKind Kind;
552
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000553public:
Daniel Sanders8d4d72f2017-02-24 14:53:35 +0000554 InstructionPredicateMatcher(PredicateKind Kind) : Kind(Kind) {}
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000555 virtual ~InstructionPredicateMatcher() {}
556
Daniel Sanders759ff412017-02-24 13:58:11 +0000557 PredicateKind getKind() const { return Kind; }
558
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000559 /// Emit a C++ expression that tests whether the instruction named in
560 /// InsnVarName matches the predicate.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000561 virtual void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Ahmed Bougacha6a1ac5a2017-02-09 02:50:01 +0000562 StringRef InsnVarName) const = 0;
Daniel Sanders759ff412017-02-24 13:58:11 +0000563
564 /// Compare the priority of this object and B.
565 ///
566 /// Returns true if this object is more important than B.
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000567 virtual bool isHigherPriorityThan(const InstructionPredicateMatcher &B) const {
Daniel Sanders759ff412017-02-24 13:58:11 +0000568 return Kind < B.Kind;
569 };
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000570
571 /// Report the maximum number of temporary operands needed by the predicate
572 /// matcher.
Daniel Sanders2deea182017-04-22 15:11:04 +0000573 virtual unsigned countRendererFns() const { return 0; }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000574};
575
576/// Generates code to check the opcode of an instruction.
577class InstructionOpcodeMatcher : public InstructionPredicateMatcher {
578protected:
579 const CodeGenInstruction *I;
580
581public:
Daniel Sanders8d4d72f2017-02-24 14:53:35 +0000582 InstructionOpcodeMatcher(const CodeGenInstruction *I)
583 : InstructionPredicateMatcher(IPM_Opcode), I(I) {}
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000584
Daniel Sanders759ff412017-02-24 13:58:11 +0000585 static bool classof(const InstructionPredicateMatcher *P) {
586 return P->getKind() == IPM_Opcode;
587 }
588
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000589 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Ahmed Bougacha6a1ac5a2017-02-09 02:50:01 +0000590 StringRef InsnVarName) const override {
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000591 OS << InsnVarName << ".getOpcode() == " << I->Namespace
592 << "::" << I->TheDef->getName();
593 }
Daniel Sanders759ff412017-02-24 13:58:11 +0000594
595 /// Compare the priority of this object and B.
596 ///
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000597 /// Returns true if this object is more important than B.
598 bool isHigherPriorityThan(const InstructionPredicateMatcher &B) const override {
Daniel Sanders759ff412017-02-24 13:58:11 +0000599 if (InstructionPredicateMatcher::isHigherPriorityThan(B))
600 return true;
601 if (B.InstructionPredicateMatcher::isHigherPriorityThan(*this))
602 return false;
603
604 // Prioritize opcodes for cosmetic reasons in the generated source. Although
605 // this is cosmetic at the moment, we may want to drive a similar ordering
606 // using instruction frequency information to improve compile time.
607 if (const InstructionOpcodeMatcher *BO =
608 dyn_cast<InstructionOpcodeMatcher>(&B))
609 return I->TheDef->getName() < BO->I->TheDef->getName();
610
611 return false;
612 };
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000613};
614
615/// Generates code to check that a set of predicates and operands match for a
616/// particular instruction.
617///
618/// Typical predicates include:
619/// * Has a specific opcode.
620/// * Has an nsw/nuw flag or doesn't.
621class InstructionMatcher
622 : public PredicateListMatcher<InstructionPredicateMatcher> {
623protected:
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000624 typedef std::vector<std::unique_ptr<OperandMatcher>> OperandVec;
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000625
626 /// The operands to match. All rendered operands must be present even if the
627 /// condition is always true.
628 OperandVec Operands;
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000629
630public:
631 /// Add an operand to the matcher.
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000632 OperandMatcher &addOperand(unsigned OpIdx, const std::string &SymbolicName,
633 unsigned AllocatedTemporariesBaseID) {
634 Operands.emplace_back(new OperandMatcher(*this, OpIdx, SymbolicName,
635 AllocatedTemporariesBaseID));
636 return *Operands.back();
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000637 }
638
Daniel Sandersffc7d582017-03-29 15:37:18 +0000639 OperandMatcher &getOperand(unsigned OpIdx) {
640 auto I = std::find_if(Operands.begin(), Operands.end(),
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000641 [&OpIdx](const std::unique_ptr<OperandMatcher> &X) {
642 return X->getOperandIndex() == OpIdx;
Daniel Sandersffc7d582017-03-29 15:37:18 +0000643 });
644 if (I != Operands.end())
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000645 return **I;
Daniel Sandersffc7d582017-03-29 15:37:18 +0000646 llvm_unreachable("Failed to lookup operand");
647 }
648
Daniel Sandersbee57392017-04-04 13:25:23 +0000649 Optional<const OperandMatcher *>
650 getOptionalOperand(StringRef SymbolicName) const {
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000651 assert(!SymbolicName.empty() && "Cannot lookup unnamed operand");
Daniel Sandersbee57392017-04-04 13:25:23 +0000652 for (const auto &Operand : Operands) {
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000653 const auto &OM = Operand->getOptionalOperand(SymbolicName);
Daniel Sandersbee57392017-04-04 13:25:23 +0000654 if (OM.hasValue())
655 return OM.getValue();
656 }
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000657 return None;
658 }
659
Daniel Sandersdb7ed372017-04-04 13:52:00 +0000660 const OperandMatcher &getOperand(StringRef SymbolicName) const {
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000661 Optional<const OperandMatcher *>OM = getOptionalOperand(SymbolicName);
662 if (OM.hasValue())
663 return *OM.getValue();
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000664 llvm_unreachable("Failed to lookup operand");
665 }
666
667 unsigned getNumOperands() const { return Operands.size(); }
Daniel Sandersbee57392017-04-04 13:25:23 +0000668 OperandVec::iterator operands_begin() { return Operands.begin(); }
669 OperandVec::iterator operands_end() { return Operands.end(); }
670 iterator_range<OperandVec::iterator> operands() {
671 return make_range(operands_begin(), operands_end());
672 }
Daniel Sandersffc7d582017-03-29 15:37:18 +0000673 OperandVec::const_iterator operands_begin() const { return Operands.begin(); }
674 OperandVec::const_iterator operands_end() const { return Operands.end(); }
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000675 iterator_range<OperandVec::const_iterator> operands() const {
676 return make_range(operands_begin(), operands_end());
677 }
678
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000679 /// Emit C++ statements to check the shape of the match and capture
680 /// instructions into local variables.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000681 void emitCxxCaptureStmts(raw_ostream &OS, RuleMatcher &Rule, StringRef Expr) {
682 OS << "if (" << Expr << ".getNumOperands() < " << getNumOperands() << ")\n"
683 << " return false;\n";
Daniel Sandersbee57392017-04-04 13:25:23 +0000684 for (const auto &Operand : Operands) {
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000685 Operand->emitCxxCaptureStmts(OS, Rule, Operand->getOperandExpr(Expr));
Daniel Sandersbee57392017-04-04 13:25:23 +0000686 }
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000687 }
688
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000689 /// Emit a C++ expression that tests whether the instruction named in
690 /// InsnVarName matches all the predicates and all the operands.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000691 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
692 StringRef InsnVarName) const {
693 emitCxxPredicateListExpr(OS, Rule, InsnVarName);
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000694 for (const auto &Operand : Operands) {
Ahmed Bougacha905af9f2017-02-04 00:47:02 +0000695 OS << " &&\n(";
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000696 Operand->emitCxxPredicateExpr(OS, Rule, InsnVarName);
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000697 OS << ")";
698 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000699 }
Daniel Sanders759ff412017-02-24 13:58:11 +0000700
701 /// Compare the priority of this object and B.
702 ///
703 /// Returns true if this object is more important than B.
704 bool isHigherPriorityThan(const InstructionMatcher &B) const {
705 // Instruction matchers involving more operands have higher priority.
706 if (Operands.size() > B.Operands.size())
707 return true;
708 if (Operands.size() < B.Operands.size())
709 return false;
710
711 for (const auto &Predicate : zip(predicates(), B.predicates())) {
712 if (std::get<0>(Predicate)->isHigherPriorityThan(*std::get<1>(Predicate)))
713 return true;
714 if (std::get<1>(Predicate)->isHigherPriorityThan(*std::get<0>(Predicate)))
715 return false;
716 }
717
718 for (const auto &Operand : zip(Operands, B.Operands)) {
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000719 if (std::get<0>(Operand)->isHigherPriorityThan(*std::get<1>(Operand)))
Daniel Sanders759ff412017-02-24 13:58:11 +0000720 return true;
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000721 if (std::get<1>(Operand)->isHigherPriorityThan(*std::get<0>(Operand)))
Daniel Sanders759ff412017-02-24 13:58:11 +0000722 return false;
723 }
724
725 return false;
726 };
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000727
728 /// Report the maximum number of temporary operands needed by the instruction
729 /// matcher.
Daniel Sanders2deea182017-04-22 15:11:04 +0000730 unsigned countRendererFns() const {
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000731 return std::accumulate(predicates().begin(), predicates().end(), 0,
732 [](unsigned A,
733 const std::unique_ptr<InstructionPredicateMatcher>
734 &Predicate) {
Daniel Sanders2deea182017-04-22 15:11:04 +0000735 return A + Predicate->countRendererFns();
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000736 }) +
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000737 std::accumulate(
738 Operands.begin(), Operands.end(), 0,
739 [](unsigned A, const std::unique_ptr<OperandMatcher> &Operand) {
Daniel Sanders2deea182017-04-22 15:11:04 +0000740 return A + Operand->countRendererFns();
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000741 });
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000742 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000743};
744
Daniel Sandersbee57392017-04-04 13:25:23 +0000745/// Generates code to check that the operand is a register defined by an
746/// instruction that matches the given instruction matcher.
747///
748/// For example, the pattern:
749/// (set $dst, (G_MUL (G_ADD $src1, $src2), $src3))
750/// would use an InstructionOperandMatcher for operand 1 of the G_MUL to match
751/// the:
752/// (G_ADD $src1, $src2)
753/// subpattern.
754class InstructionOperandMatcher : public OperandPredicateMatcher {
755protected:
756 std::unique_ptr<InstructionMatcher> InsnMatcher;
757
758public:
759 InstructionOperandMatcher()
760 : OperandPredicateMatcher(OPM_Instruction),
761 InsnMatcher(new InstructionMatcher()) {}
762
763 static bool classof(const OperandPredicateMatcher *P) {
764 return P->getKind() == OPM_Instruction;
765 }
766
767 InstructionMatcher &getInsnMatcher() const { return *InsnMatcher; }
768
769 Optional<const OperandMatcher *>
770 getOptionalOperand(StringRef SymbolicName) const override {
771 assert(!SymbolicName.empty() && "Cannot lookup unnamed operand");
772 return InsnMatcher->getOptionalOperand(SymbolicName);
773 }
774
775 void emitCxxCaptureStmts(raw_ostream &OS, RuleMatcher &Rule,
776 StringRef OperandExpr) const override {
777 OS << "if (!" << OperandExpr + ".isReg())\n"
Daniel Sandersed205a02017-05-17 12:43:30 +0000778 << " return false;\n"
779 << "if (TRI.isPhysicalRegister(" << OperandExpr + ".getReg()))\n"
Daniel Sandersbee57392017-04-04 13:25:23 +0000780 << " return false;\n";
781 std::string InsnVarName = Rule.defineInsnVar(
782 OS, *InsnMatcher,
783 ("*MRI.getVRegDef(" + OperandExpr + ".getReg())").str());
784 InsnMatcher->emitCxxCaptureStmts(OS, Rule, InsnVarName);
785 }
786
787 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
788 StringRef OperandExpr) const override {
789 OperandExpr = Rule.getInsnVarName(*InsnMatcher);
790 OS << "(";
791 InsnMatcher->emitCxxPredicateExpr(OS, Rule, OperandExpr);
792 OS << ")\n";
793 }
794};
795
Daniel Sanders43c882c2017-02-01 10:53:10 +0000796//===- Actions ------------------------------------------------------------===//
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000797class OperandRenderer {
798public:
Daniel Sanders0ed28822017-04-12 08:23:08 +0000799 enum RendererKind { OR_Copy, OR_Imm, OR_Register, OR_ComplexPattern };
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000800
801protected:
802 RendererKind Kind;
803
804public:
805 OperandRenderer(RendererKind Kind) : Kind(Kind) {}
806 virtual ~OperandRenderer() {}
807
808 RendererKind getKind() const { return Kind; }
809
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000810 virtual void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const = 0;
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000811};
812
813/// A CopyRenderer emits code to copy a single operand from an existing
814/// instruction to the one being built.
815class CopyRenderer : public OperandRenderer {
816protected:
817 /// The matcher for the instruction that this operand is copied from.
818 /// This provides the facility for looking up an a operand by it's name so
819 /// that it can be used as a source for the instruction being built.
820 const InstructionMatcher &Matched;
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000821 /// The name of the operand.
822 const StringRef SymbolicName;
823
824public:
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000825 CopyRenderer(const InstructionMatcher &Matched, StringRef SymbolicName)
826 : OperandRenderer(OR_Copy), Matched(Matched), SymbolicName(SymbolicName) {
827 }
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000828
829 static bool classof(const OperandRenderer *R) {
830 return R->getKind() == OR_Copy;
831 }
832
833 const StringRef getSymbolicName() const { return SymbolicName; }
834
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000835 void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const override {
836 const OperandMatcher &Operand = Matched.getOperand(SymbolicName);
837 StringRef InsnVarName =
838 Rule.getInsnVarName(Operand.getInstructionMatcher());
839 std::string OperandExpr = Operand.getOperandExpr(InsnVarName);
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000840 OS << " MIB.add(" << OperandExpr << "/*" << SymbolicName << "*/);\n";
841 }
842};
843
844/// Adds a specific physical register to the instruction being built.
845/// This is typically useful for WZR/XZR on AArch64.
846class AddRegisterRenderer : public OperandRenderer {
847protected:
848 const Record *RegisterDef;
849
850public:
851 AddRegisterRenderer(const Record *RegisterDef)
852 : OperandRenderer(OR_Register), RegisterDef(RegisterDef) {}
853
854 static bool classof(const OperandRenderer *R) {
855 return R->getKind() == OR_Register;
856 }
857
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000858 void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const override {
Diana Picus8abcbbb2017-05-02 09:40:49 +0000859 OS << " MIB.addReg(" << (RegisterDef->getValue("Namespace")
860 ? RegisterDef->getValueAsString("Namespace")
861 : "")
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000862 << "::" << RegisterDef->getName() << ");\n";
863 }
864};
865
Daniel Sanders0ed28822017-04-12 08:23:08 +0000866/// Adds a specific immediate to the instruction being built.
867class ImmRenderer : public OperandRenderer {
868protected:
869 int64_t Imm;
870
871public:
872 ImmRenderer(int64_t Imm)
873 : OperandRenderer(OR_Imm), Imm(Imm) {}
874
875 static bool classof(const OperandRenderer *R) {
876 return R->getKind() == OR_Imm;
877 }
878
879 void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const override {
880 OS << " MIB.addImm(" << Imm << ");\n";
881 }
882};
883
Daniel Sanders2deea182017-04-22 15:11:04 +0000884/// Adds operands by calling a renderer function supplied by the ComplexPattern
885/// matcher function.
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000886class RenderComplexPatternOperand : public OperandRenderer {
887private:
888 const Record &TheDef;
Daniel Sanders2deea182017-04-22 15:11:04 +0000889 /// The name of the operand.
890 const StringRef SymbolicName;
891 /// The renderer number. This must be unique within a rule since it's used to
892 /// identify a temporary variable to hold the renderer function.
893 unsigned RendererID;
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000894
895 unsigned getNumOperands() const {
896 return TheDef.getValueAsDag("Operands")->getNumArgs();
897 }
898
899public:
Daniel Sanders2deea182017-04-22 15:11:04 +0000900 RenderComplexPatternOperand(const Record &TheDef, StringRef SymbolicName,
901 unsigned RendererID)
902 : OperandRenderer(OR_ComplexPattern), TheDef(TheDef),
903 SymbolicName(SymbolicName), RendererID(RendererID) {}
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000904
905 static bool classof(const OperandRenderer *R) {
906 return R->getKind() == OR_ComplexPattern;
907 }
908
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000909 void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const override {
Daniel Sanders2deea182017-04-22 15:11:04 +0000910 OS << "Renderer" << RendererID << "(MIB);\n";
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000911 }
912};
913
Ahmed Bougacha56ca3a92017-02-04 00:47:10 +0000914/// An action taken when all Matcher predicates succeeded for a parent rule.
915///
916/// Typical actions include:
917/// * Changing the opcode of an instruction.
918/// * Adding an operand to an instruction.
Daniel Sanders43c882c2017-02-01 10:53:10 +0000919class MatchAction {
920public:
921 virtual ~MatchAction() {}
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000922
923 /// Emit the C++ statements to implement the action.
924 ///
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000925 /// \param RecycleVarName If given, it's an instruction to recycle. The
926 /// requirements on the instruction vary from action to
927 /// action.
928 virtual void emitCxxActionStmts(raw_ostream &OS, RuleMatcher &Rule,
929 StringRef RecycleVarName) const = 0;
Daniel Sanders43c882c2017-02-01 10:53:10 +0000930};
931
Ahmed Bougacha9aa4c102017-02-04 00:47:08 +0000932/// Generates a comment describing the matched rule being acted upon.
933class DebugCommentAction : public MatchAction {
934private:
935 const PatternToMatch &P;
936
937public:
938 DebugCommentAction(const PatternToMatch &P) : P(P) {}
939
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000940 void emitCxxActionStmts(raw_ostream &OS, RuleMatcher &Rule,
941 StringRef RecycleVarName) const override {
942 OS << "// " << *P.getSrcPattern() << " => " << *P.getDstPattern() << "\n";
Ahmed Bougacha9aa4c102017-02-04 00:47:08 +0000943 }
944};
945
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000946/// Generates code to build an instruction or mutate an existing instruction
947/// into the desired instruction when this is possible.
948class BuildMIAction : public MatchAction {
Daniel Sanders43c882c2017-02-01 10:53:10 +0000949private:
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000950 const CodeGenInstruction *I;
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000951 const InstructionMatcher &Matched;
952 std::vector<std::unique_ptr<OperandRenderer>> OperandRenderers;
953
954 /// True if the instruction can be built solely by mutating the opcode.
955 bool canMutate() const {
Daniel Sanderse9fdba32017-04-29 17:30:09 +0000956 if (OperandRenderers.size() != Matched.getNumOperands())
957 return false;
958
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000959 for (const auto &Renderer : enumerate(OperandRenderers)) {
Zachary Turner309a0882017-03-13 16:24:10 +0000960 if (const auto *Copy = dyn_cast<CopyRenderer>(&*Renderer.value())) {
Daniel Sanders3016d3c2017-04-22 14:31:28 +0000961 const OperandMatcher &OM = Matched.getOperand(Copy->getSymbolicName());
962 if (&Matched != &OM.getInstructionMatcher() ||
963 OM.getOperandIndex() != Renderer.index())
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000964 return false;
965 } else
966 return false;
967 }
968
969 return true;
970 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000971
Daniel Sanders43c882c2017-02-01 10:53:10 +0000972public:
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000973 BuildMIAction(const CodeGenInstruction *I, const InstructionMatcher &Matched)
974 : I(I), Matched(Matched) {}
Daniel Sanders43c882c2017-02-01 10:53:10 +0000975
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000976 template <class Kind, class... Args>
977 Kind &addRenderer(Args&&... args) {
978 OperandRenderers.emplace_back(
979 llvm::make_unique<Kind>(std::forward<Args>(args)...));
980 return *static_cast<Kind *>(OperandRenderers.back().get());
981 }
982
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000983 void emitCxxActionStmts(raw_ostream &OS, RuleMatcher &Rule,
984 StringRef RecycleVarName) const override {
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000985 if (canMutate()) {
Tim Northover4340d642017-03-20 21:58:23 +0000986 OS << " " << RecycleVarName << ".setDesc(TII.get(" << I->Namespace
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000987 << "::" << I->TheDef->getName() << "));\n";
Tim Northover4340d642017-03-20 21:58:23 +0000988
989 if (!I->ImplicitDefs.empty() || !I->ImplicitUses.empty()) {
990 OS << " auto MIB = MachineInstrBuilder(MF, &" << RecycleVarName
991 << ");\n";
992
993 for (auto Def : I->ImplicitDefs) {
Diana Picus8abcbbb2017-05-02 09:40:49 +0000994 auto Namespace = Def->getValue("Namespace")
995 ? Def->getValueAsString("Namespace")
996 : "";
Tim Northover4340d642017-03-20 21:58:23 +0000997 OS << " MIB.addDef(" << Namespace << "::" << Def->getName()
998 << ", RegState::Implicit);\n";
999 }
1000 for (auto Use : I->ImplicitUses) {
Diana Picus8abcbbb2017-05-02 09:40:49 +00001001 auto Namespace = Use->getValue("Namespace")
1002 ? Use->getValueAsString("Namespace")
1003 : "";
Tim Northover4340d642017-03-20 21:58:23 +00001004 OS << " MIB.addUse(" << Namespace << "::" << Use->getName()
1005 << ", RegState::Implicit);\n";
1006 }
1007 }
1008
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001009 OS << " MachineInstr &NewI = " << RecycleVarName << ";\n";
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001010 return;
1011 }
1012
1013 // TODO: Simple permutation looks like it could be almost as common as
1014 // mutation due to commutative operations.
1015
1016 OS << "MachineInstrBuilder MIB = BuildMI(*I.getParent(), I, "
1017 "I.getDebugLoc(), TII.get("
1018 << I->Namespace << "::" << I->TheDef->getName() << "));\n";
1019 for (const auto &Renderer : OperandRenderers)
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001020 Renderer->emitCxxRenderStmts(OS, Rule);
Daniel Sandersbee57392017-04-04 13:25:23 +00001021 OS << " for (const auto *FromMI : ";
1022 Rule.emitCxxCapturedInsnList(OS);
1023 OS << ")\n";
1024 OS << " for (const auto &MMO : FromMI->memoperands())\n";
1025 OS << " MIB.addMemOperand(MMO);\n";
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001026 OS << " " << RecycleVarName << ".eraseFromParent();\n";
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001027 OS << " MachineInstr &NewI = *MIB;\n";
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001028 }
1029};
1030
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001031InstructionMatcher &RuleMatcher::addInstructionMatcher() {
1032 Matchers.emplace_back(new InstructionMatcher());
1033 return *Matchers.back();
1034}
Ahmed Bougacha56ca3a92017-02-04 00:47:10 +00001035
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001036void RuleMatcher::addRequiredFeature(Record *Feature) {
1037 RequiredFeatures.push_back(Feature);
1038}
1039
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001040template <class Kind, class... Args>
1041Kind &RuleMatcher::addAction(Args &&... args) {
1042 Actions.emplace_back(llvm::make_unique<Kind>(std::forward<Args>(args)...));
1043 return *static_cast<Kind *>(Actions.back().get());
1044}
Daniel Sandersdc662ff2017-01-26 11:10:14 +00001045
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001046std::string RuleMatcher::defineInsnVar(raw_ostream &OS,
1047 const InstructionMatcher &Matcher,
1048 StringRef Value) {
1049 std::string InsnVarName = "MI" + llvm::to_string(NextInsnVarID++);
1050 OS << "MachineInstr &" << InsnVarName << " = " << Value << ";\n";
1051 InsnVariableNames[&Matcher] = InsnVarName;
1052 return InsnVarName;
1053}
1054
1055StringRef RuleMatcher::getInsnVarName(const InstructionMatcher &InsnMatcher) const {
1056 const auto &I = InsnVariableNames.find(&InsnMatcher);
1057 if (I != InsnVariableNames.end())
1058 return I->second;
1059 llvm_unreachable("Matched Insn was not captured in a local variable");
1060}
1061
Daniel Sandersbee57392017-04-04 13:25:23 +00001062/// Emit a C++ initializer_list containing references to every matched instruction.
1063void RuleMatcher::emitCxxCapturedInsnList(raw_ostream &OS) {
Daniel Sanders9e4817d2017-04-04 14:27:06 +00001064 SmallVector<StringRef, 2> Names;
Daniel Sandersbee57392017-04-04 13:25:23 +00001065 for (const auto &Pair : InsnVariableNames)
Daniel Sanders9e4817d2017-04-04 14:27:06 +00001066 Names.push_back(Pair.second);
1067 std::sort(Names.begin(), Names.end());
1068
1069 OS << "{";
1070 for (const auto &Name : Names)
1071 OS << "&" << Name << ", ";
Daniel Sandersbee57392017-04-04 13:25:23 +00001072 OS << "}";
1073}
1074
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001075/// Emit C++ statements to check the shape of the match and capture
1076/// instructions into local variables.
1077void RuleMatcher::emitCxxCaptureStmts(raw_ostream &OS, StringRef Expr) {
1078 assert(Matchers.size() == 1 && "Cannot handle multi-root matchers yet");
1079 std::string InsnVarName = defineInsnVar(OS, *Matchers.front(), Expr);
1080 Matchers.front()->emitCxxCaptureStmts(OS, *this, InsnVarName);
1081}
1082
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001083void RuleMatcher::emit(raw_ostream &OS,
Daniel Sanderse9fdba32017-04-29 17:30:09 +00001084 SubtargetFeatureInfoMap SubtargetFeatures) {
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001085 if (Matchers.empty())
1086 llvm_unreachable("Unexpected empty matcher!");
Daniel Sandersdc662ff2017-01-26 11:10:14 +00001087
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001088 // The representation supports rules that require multiple roots such as:
1089 // %ptr(p0) = ...
1090 // %elt0(s32) = G_LOAD %ptr
1091 // %1(p0) = G_ADD %ptr, 4
1092 // %elt1(s32) = G_LOAD p0 %1
1093 // which could be usefully folded into:
1094 // %ptr(p0) = ...
1095 // %elt0(s32), %elt1(s32) = TGT_LOAD_PAIR %ptr
1096 // on some targets but we don't need to make use of that yet.
1097 assert(Matchers.size() == 1 && "Cannot handle multi-root matchers yet");
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001098
1099 OS << "if (";
1100 OS << "[&]() {\n";
1101 if (!RequiredFeatures.empty()) {
1102 OS << " PredicateBitset ExpectedFeatures = {";
1103 StringRef Separator = "";
1104 for (const auto &Predicate : RequiredFeatures) {
1105 const auto &I = SubtargetFeatures.find(Predicate);
1106 assert(I != SubtargetFeatures.end() && "Didn't import predicate?");
1107 OS << Separator << I->second.getEnumBitName();
1108 Separator = ", ";
1109 }
1110 OS << "};\n";
1111 OS << "if ((AvailableFeatures & ExpectedFeatures) != ExpectedFeatures)\n"
1112 << " return false;\n";
1113 }
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001114
1115 emitCxxCaptureStmts(OS, "I");
1116
1117 OS << " if (";
1118 Matchers.front()->emitCxxPredicateExpr(OS, *this,
1119 getInsnVarName(*Matchers.front()));
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001120 OS << ") {\n";
1121
Daniel Sandersbee57392017-04-04 13:25:23 +00001122 // We must also check if it's safe to fold the matched instructions.
1123 if (InsnVariableNames.size() >= 2) {
1124 for (const auto &Pair : InsnVariableNames) {
1125 // Skip the root node since it isn't moving anywhere. Everything else is
1126 // sinking to meet it.
1127 if (Pair.first == Matchers.front().get())
1128 continue;
1129
1130 // Reject the difficult cases until we have a more accurate check.
1131 OS << " if (!isObviouslySafeToFold(" << Pair.second
1132 << ")) return false;\n";
1133
1134 // FIXME: Emit checks to determine it's _actually_ safe to fold and/or
1135 // account for unsafe cases.
1136 //
1137 // Example:
1138 // MI1--> %0 = ...
1139 // %1 = ... %0
1140 // MI0--> %2 = ... %0
1141 // It's not safe to erase MI1. We currently handle this by not
1142 // erasing %0 (even when it's dead).
1143 //
1144 // Example:
1145 // MI1--> %0 = load volatile @a
1146 // %1 = load volatile @a
1147 // MI0--> %2 = ... %0
1148 // It's not safe to sink %0's def past %1. We currently handle
1149 // this by rejecting all loads.
1150 //
1151 // Example:
1152 // MI1--> %0 = load @a
1153 // %1 = store @a
1154 // MI0--> %2 = ... %0
1155 // It's not safe to sink %0's def past %1. We currently handle
1156 // this by rejecting all loads.
1157 //
1158 // Example:
1159 // G_CONDBR %cond, @BB1
1160 // BB0:
1161 // MI1--> %0 = load @a
1162 // G_BR @BB1
1163 // BB1:
1164 // MI0--> %2 = ... %0
1165 // It's not always safe to sink %0 across control flow. In this
1166 // case it may introduce a memory fault. We currentl handle this
1167 // by rejecting all loads.
1168 }
1169 }
1170
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001171 for (const auto &MA : Actions) {
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001172 MA->emitCxxActionStmts(OS, *this, "I");
Daniel Sandersdc662ff2017-01-26 11:10:14 +00001173 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001174
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001175 OS << " constrainSelectedInstRegOperands(NewI, TII, TRI, RBI);\n";
1176 OS << " return true;\n";
1177 OS << " }\n";
1178 OS << " return false;\n";
1179 OS << " }()) { return true; }\n\n";
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001180}
Daniel Sanders43c882c2017-02-01 10:53:10 +00001181
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001182bool RuleMatcher::isHigherPriorityThan(const RuleMatcher &B) const {
1183 // Rules involving more match roots have higher priority.
1184 if (Matchers.size() > B.Matchers.size())
1185 return true;
1186 if (Matchers.size() < B.Matchers.size())
Daniel Sanders759ff412017-02-24 13:58:11 +00001187 return false;
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001188
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001189 for (const auto &Matcher : zip(Matchers, B.Matchers)) {
1190 if (std::get<0>(Matcher)->isHigherPriorityThan(*std::get<1>(Matcher)))
1191 return true;
1192 if (std::get<1>(Matcher)->isHigherPriorityThan(*std::get<0>(Matcher)))
1193 return false;
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001194 }
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001195
1196 return false;
Simon Pilgrima7d1da82017-03-15 22:50:47 +00001197}
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001198
Daniel Sanders2deea182017-04-22 15:11:04 +00001199unsigned RuleMatcher::countRendererFns() const {
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001200 return std::accumulate(
1201 Matchers.begin(), Matchers.end(), 0,
1202 [](unsigned A, const std::unique_ptr<InstructionMatcher> &Matcher) {
Daniel Sanders2deea182017-04-22 15:11:04 +00001203 return A + Matcher->countRendererFns();
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001204 });
1205}
1206
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001207//===- GlobalISelEmitter class --------------------------------------------===//
1208
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001209class GlobalISelEmitter {
1210public:
1211 explicit GlobalISelEmitter(RecordKeeper &RK);
1212 void run(raw_ostream &OS);
1213
1214private:
1215 const RecordKeeper &RK;
1216 const CodeGenDAGPatterns CGP;
1217 const CodeGenTarget &Target;
1218
1219 /// Keep track of the equivalence between SDNodes and Instruction.
1220 /// This is defined using 'GINodeEquiv' in the target description.
1221 DenseMap<Record *, const CodeGenInstruction *> NodeEquivs;
1222
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001223 /// Keep track of the equivalence between ComplexPattern's and
1224 /// GIComplexOperandMatcher. Map entries are specified by subclassing
1225 /// GIComplexPatternEquiv.
1226 DenseMap<const Record *, const Record *> ComplexPatternEquivs;
1227
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001228 // Map of predicates to their subtarget features.
Daniel Sanderse9fdba32017-04-29 17:30:09 +00001229 SubtargetFeatureInfoMap SubtargetFeatures;
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001230
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001231 void gatherNodeEquivs();
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001232 const CodeGenInstruction *findNodeEquiv(Record *N) const;
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001233
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001234 Error importRulePredicates(RuleMatcher &M, ArrayRef<Init *> Predicates);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001235 Expected<InstructionMatcher &>
Daniel Sandersc270c502017-03-30 09:36:33 +00001236 createAndImportSelDAGMatcher(InstructionMatcher &InsnMatcher,
1237 const TreePatternNode *Src) const;
1238 Error importChildMatcher(InstructionMatcher &InsnMatcher,
1239 TreePatternNode *SrcChild, unsigned OpIdx,
1240 unsigned &TempOpIdx) const;
1241 Expected<BuildMIAction &> createAndImportInstructionRenderer(
1242 RuleMatcher &M, const TreePatternNode *Dst,
1243 const InstructionMatcher &InsnMatcher) const;
1244 Error importExplicitUseRenderer(BuildMIAction &DstMIBuilder,
1245 TreePatternNode *DstChild,
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001246 const InstructionMatcher &InsnMatcher) const;
Diana Picus382602f2017-05-17 08:57:28 +00001247 Error importDefaultOperandRenderers(BuildMIAction &DstMIBuilder,
1248 DagInit *DefaultOps) const;
Daniel Sandersc270c502017-03-30 09:36:33 +00001249 Error
Daniel Sandersffc7d582017-03-29 15:37:18 +00001250 importImplicitDefRenderers(BuildMIAction &DstMIBuilder,
1251 const std::vector<Record *> &ImplicitDefs) const;
1252
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001253 /// Analyze pattern \p P, returning a matcher for it if possible.
1254 /// Otherwise, return an Error explaining why we don't support it.
1255 Expected<RuleMatcher> runOnPattern(const PatternToMatch &P);
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001256
1257 void declareSubtargetFeature(Record *Predicate);
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001258};
1259
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001260void GlobalISelEmitter::gatherNodeEquivs() {
1261 assert(NodeEquivs.empty());
1262 for (Record *Equiv : RK.getAllDerivedDefinitions("GINodeEquiv"))
1263 NodeEquivs[Equiv->getValueAsDef("Node")] =
1264 &Target.getInstruction(Equiv->getValueAsDef("I"));
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001265
1266 assert(ComplexPatternEquivs.empty());
1267 for (Record *Equiv : RK.getAllDerivedDefinitions("GIComplexPatternEquiv")) {
1268 Record *SelDAGEquiv = Equiv->getValueAsDef("SelDAGEquivalent");
1269 if (!SelDAGEquiv)
1270 continue;
1271 ComplexPatternEquivs[SelDAGEquiv] = Equiv;
1272 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001273}
1274
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001275const CodeGenInstruction *GlobalISelEmitter::findNodeEquiv(Record *N) const {
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001276 return NodeEquivs.lookup(N);
1277}
1278
1279GlobalISelEmitter::GlobalISelEmitter(RecordKeeper &RK)
1280 : RK(RK), CGP(RK), Target(CGP.getTargetInfo()) {}
1281
1282//===- Emitter ------------------------------------------------------------===//
1283
Daniel Sandersc270c502017-03-30 09:36:33 +00001284Error
Daniel Sandersffc7d582017-03-29 15:37:18 +00001285GlobalISelEmitter::importRulePredicates(RuleMatcher &M,
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001286 ArrayRef<Init *> Predicates) {
1287 for (const Init *Predicate : Predicates) {
1288 const DefInit *PredicateDef = static_cast<const DefInit *>(Predicate);
1289 declareSubtargetFeature(PredicateDef->getDef());
1290 M.addRequiredFeature(PredicateDef->getDef());
1291 }
1292
Daniel Sandersc270c502017-03-30 09:36:33 +00001293 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001294}
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001295
Daniel Sandersc270c502017-03-30 09:36:33 +00001296Expected<InstructionMatcher &> GlobalISelEmitter::createAndImportSelDAGMatcher(
1297 InstructionMatcher &InsnMatcher, const TreePatternNode *Src) const {
Daniel Sandersffc7d582017-03-29 15:37:18 +00001298 // Start with the defined operands (i.e., the results of the root operator).
1299 if (Src->getExtTypes().size() > 1)
1300 return failedImport("Src pattern has multiple results");
1301
1302 auto SrcGIOrNull = findNodeEquiv(Src->getOperator());
1303 if (!SrcGIOrNull)
Daniel Sandersd0656a32017-04-13 09:45:37 +00001304 return failedImport("Pattern operator lacks an equivalent Instruction" +
1305 explainOperator(Src->getOperator()));
Daniel Sandersffc7d582017-03-29 15:37:18 +00001306 auto &SrcGI = *SrcGIOrNull;
1307
1308 // The operators look good: match the opcode and mutate it to the new one.
1309 InsnMatcher.addPredicate<InstructionOpcodeMatcher>(&SrcGI);
1310
1311 unsigned OpIdx = 0;
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001312 unsigned TempOpIdx = 0;
Daniel Sandersffc7d582017-03-29 15:37:18 +00001313 for (const EEVT::TypeSet &Ty : Src->getExtTypes()) {
1314 auto OpTyOrNone = MVTToLLT(Ty.getConcrete());
1315
1316 if (!OpTyOrNone)
1317 return failedImport(
1318 "Result of Src pattern operator has an unsupported type");
1319
1320 // Results don't have a name unless they are the root node. The caller will
1321 // set the name if appropriate.
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001322 OperandMatcher &OM = InsnMatcher.addOperand(OpIdx++, "", TempOpIdx);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001323 OM.addPredicate<LLTOperandMatcher>(*OpTyOrNone);
1324 }
1325
Daniel Sandersffc7d582017-03-29 15:37:18 +00001326 // Match the used operands (i.e. the children of the operator).
1327 for (unsigned i = 0, e = Src->getNumChildren(); i != e; ++i) {
1328 if (auto Error = importChildMatcher(InsnMatcher, Src->getChild(i), OpIdx++,
Daniel Sandersc270c502017-03-30 09:36:33 +00001329 TempOpIdx))
Daniel Sandersffc7d582017-03-29 15:37:18 +00001330 return std::move(Error);
1331 }
1332
1333 return InsnMatcher;
1334}
1335
Daniel Sandersc270c502017-03-30 09:36:33 +00001336Error GlobalISelEmitter::importChildMatcher(InstructionMatcher &InsnMatcher,
1337 TreePatternNode *SrcChild,
1338 unsigned OpIdx,
1339 unsigned &TempOpIdx) const {
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001340 OperandMatcher &OM =
1341 InsnMatcher.addOperand(OpIdx, SrcChild->getName(), TempOpIdx);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001342
1343 if (SrcChild->hasAnyPredicate())
Daniel Sandersd0656a32017-04-13 09:45:37 +00001344 return failedImport("Src pattern child has predicate (" +
1345 explainPredicates(SrcChild) + ")");
Daniel Sandersffc7d582017-03-29 15:37:18 +00001346
1347 ArrayRef<EEVT::TypeSet> ChildTypes = SrcChild->getExtTypes();
1348 if (ChildTypes.size() != 1)
1349 return failedImport("Src pattern child has multiple results");
1350
1351 // Check MBB's before the type check since they are not a known type.
1352 if (!SrcChild->isLeaf()) {
1353 if (SrcChild->getOperator()->isSubClassOf("SDNode")) {
1354 auto &ChildSDNI = CGP.getSDNodeInfo(SrcChild->getOperator());
1355 if (ChildSDNI.getSDClassName() == "BasicBlockSDNode") {
1356 OM.addPredicate<MBBOperandMatcher>();
Daniel Sandersc270c502017-03-30 09:36:33 +00001357 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001358 }
1359 }
Daniel Sandersffc7d582017-03-29 15:37:18 +00001360 }
1361
1362 auto OpTyOrNone = MVTToLLT(ChildTypes.front().getConcrete());
1363 if (!OpTyOrNone)
1364 return failedImport("Src operand has an unsupported type");
1365 OM.addPredicate<LLTOperandMatcher>(*OpTyOrNone);
1366
Daniel Sandersbee57392017-04-04 13:25:23 +00001367 // Check for nested instructions.
1368 if (!SrcChild->isLeaf()) {
1369 // Map the node to a gMIR instruction.
1370 InstructionOperandMatcher &InsnOperand =
1371 OM.addPredicate<InstructionOperandMatcher>();
1372 auto InsnMatcherOrError =
1373 createAndImportSelDAGMatcher(InsnOperand.getInsnMatcher(), SrcChild);
1374 if (auto Error = InsnMatcherOrError.takeError())
1375 return Error;
1376
1377 return Error::success();
1378 }
1379
Daniel Sandersffc7d582017-03-29 15:37:18 +00001380 // Check for constant immediates.
1381 if (auto *ChildInt = dyn_cast<IntInit>(SrcChild->getLeafValue())) {
1382 OM.addPredicate<IntOperandMatcher>(ChildInt->getValue());
Daniel Sandersc270c502017-03-30 09:36:33 +00001383 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001384 }
1385
1386 // Check for def's like register classes or ComplexPattern's.
1387 if (auto *ChildDefInit = dyn_cast<DefInit>(SrcChild->getLeafValue())) {
1388 auto *ChildRec = ChildDefInit->getDef();
1389
1390 // Check for register classes.
1391 if (ChildRec->isSubClassOf("RegisterClass")) {
1392 OM.addPredicate<RegisterBankOperandMatcher>(
1393 Target.getRegisterClass(ChildRec));
Daniel Sandersc270c502017-03-30 09:36:33 +00001394 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001395 }
1396
Daniel Sanders658541f2017-04-22 15:53:21 +00001397 if (ChildRec->isSubClassOf("RegisterOperand")) {
1398 OM.addPredicate<RegisterBankOperandMatcher>(
1399 Target.getRegisterClass(ChildRec->getValueAsDef("RegClass")));
1400 return Error::success();
1401 }
1402
Daniel Sandersffc7d582017-03-29 15:37:18 +00001403 // Check for ComplexPattern's.
1404 if (ChildRec->isSubClassOf("ComplexPattern")) {
1405 const auto &ComplexPattern = ComplexPatternEquivs.find(ChildRec);
1406 if (ComplexPattern == ComplexPatternEquivs.end())
Daniel Sandersd0656a32017-04-13 09:45:37 +00001407 return failedImport("SelectionDAG ComplexPattern (" +
1408 ChildRec->getName() + ") not mapped to GlobalISel");
Daniel Sandersffc7d582017-03-29 15:37:18 +00001409
Daniel Sanders2deea182017-04-22 15:11:04 +00001410 OM.addPredicate<ComplexPatternOperandMatcher>(OM,
1411 *ComplexPattern->second);
1412 TempOpIdx++;
Daniel Sandersc270c502017-03-30 09:36:33 +00001413 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001414 }
1415
Daniel Sandersd0656a32017-04-13 09:45:37 +00001416 if (ChildRec->isSubClassOf("ImmLeaf")) {
1417 return failedImport(
1418 "Src pattern child def is an unsupported tablegen class (ImmLeaf)");
1419 }
1420
Daniel Sandersffc7d582017-03-29 15:37:18 +00001421 return failedImport(
1422 "Src pattern child def is an unsupported tablegen class");
1423 }
1424
1425 return failedImport("Src pattern child is an unsupported kind");
1426}
1427
Daniel Sandersc270c502017-03-30 09:36:33 +00001428Error GlobalISelEmitter::importExplicitUseRenderer(
Daniel Sandersffc7d582017-03-29 15:37:18 +00001429 BuildMIAction &DstMIBuilder, TreePatternNode *DstChild,
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001430 const InstructionMatcher &InsnMatcher) const {
Daniel Sandersffc7d582017-03-29 15:37:18 +00001431 // The only non-leaf child we accept is 'bb': it's an operator because
1432 // BasicBlockSDNode isn't inline, but in MI it's just another operand.
1433 if (!DstChild->isLeaf()) {
1434 if (DstChild->getOperator()->isSubClassOf("SDNode")) {
1435 auto &ChildSDNI = CGP.getSDNodeInfo(DstChild->getOperator());
1436 if (ChildSDNI.getSDClassName() == "BasicBlockSDNode") {
1437 DstMIBuilder.addRenderer<CopyRenderer>(InsnMatcher,
1438 DstChild->getName());
Daniel Sandersc270c502017-03-30 09:36:33 +00001439 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001440 }
1441 }
1442 return failedImport("Dst pattern child isn't a leaf node or an MBB");
1443 }
1444
1445 // Otherwise, we're looking for a bog-standard RegisterClass operand.
1446 if (DstChild->hasAnyPredicate())
Daniel Sandersd0656a32017-04-13 09:45:37 +00001447 return failedImport("Dst pattern child has predicate (" +
1448 explainPredicates(DstChild) + ")");
Daniel Sandersffc7d582017-03-29 15:37:18 +00001449
1450 if (auto *ChildDefInit = dyn_cast<DefInit>(DstChild->getLeafValue())) {
1451 auto *ChildRec = ChildDefInit->getDef();
1452
1453 ArrayRef<EEVT::TypeSet> ChildTypes = DstChild->getExtTypes();
1454 if (ChildTypes.size() != 1)
1455 return failedImport("Dst pattern child has multiple results");
1456
1457 auto OpTyOrNone = MVTToLLT(ChildTypes.front().getConcrete());
1458 if (!OpTyOrNone)
1459 return failedImport("Dst operand has an unsupported type");
1460
1461 if (ChildRec->isSubClassOf("Register")) {
1462 DstMIBuilder.addRenderer<AddRegisterRenderer>(ChildRec);
Daniel Sandersc270c502017-03-30 09:36:33 +00001463 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001464 }
1465
Daniel Sanders658541f2017-04-22 15:53:21 +00001466 if (ChildRec->isSubClassOf("RegisterClass") ||
1467 ChildRec->isSubClassOf("RegisterOperand")) {
Daniel Sandersffc7d582017-03-29 15:37:18 +00001468 DstMIBuilder.addRenderer<CopyRenderer>(InsnMatcher, DstChild->getName());
Daniel Sandersc270c502017-03-30 09:36:33 +00001469 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001470 }
1471
1472 if (ChildRec->isSubClassOf("ComplexPattern")) {
1473 const auto &ComplexPattern = ComplexPatternEquivs.find(ChildRec);
1474 if (ComplexPattern == ComplexPatternEquivs.end())
1475 return failedImport(
1476 "SelectionDAG ComplexPattern not mapped to GlobalISel");
1477
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001478 const OperandMatcher &OM = InsnMatcher.getOperand(DstChild->getName());
Daniel Sandersffc7d582017-03-29 15:37:18 +00001479 DstMIBuilder.addRenderer<RenderComplexPatternOperand>(
Daniel Sanders2deea182017-04-22 15:11:04 +00001480 *ComplexPattern->second, DstChild->getName(),
1481 OM.getAllocatedTemporariesBaseID());
Daniel Sandersc270c502017-03-30 09:36:33 +00001482 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001483 }
1484
Daniel Sandersd0656a32017-04-13 09:45:37 +00001485 if (ChildRec->isSubClassOf("SDNodeXForm"))
1486 return failedImport("Dst pattern child def is an unsupported tablegen "
1487 "class (SDNodeXForm)");
1488
Daniel Sandersffc7d582017-03-29 15:37:18 +00001489 return failedImport(
1490 "Dst pattern child def is an unsupported tablegen class");
1491 }
1492
1493 return failedImport("Dst pattern child is an unsupported kind");
1494}
1495
Daniel Sandersc270c502017-03-30 09:36:33 +00001496Expected<BuildMIAction &> GlobalISelEmitter::createAndImportInstructionRenderer(
Daniel Sandersffc7d582017-03-29 15:37:18 +00001497 RuleMatcher &M, const TreePatternNode *Dst,
1498 const InstructionMatcher &InsnMatcher) const {
1499 Record *DstOp = Dst->getOperator();
Daniel Sandersd0656a32017-04-13 09:45:37 +00001500 if (!DstOp->isSubClassOf("Instruction")) {
1501 if (DstOp->isSubClassOf("ValueType"))
1502 return failedImport(
1503 "Pattern operator isn't an instruction (it's a ValueType)");
Daniel Sandersffc7d582017-03-29 15:37:18 +00001504 return failedImport("Pattern operator isn't an instruction");
Daniel Sandersd0656a32017-04-13 09:45:37 +00001505 }
Daniel Sandersffc7d582017-03-29 15:37:18 +00001506 auto &DstI = Target.getInstruction(DstOp);
1507
1508 auto &DstMIBuilder = M.addAction<BuildMIAction>(&DstI, InsnMatcher);
1509
1510 // Render the explicit defs.
1511 for (unsigned I = 0; I < DstI.Operands.NumDefs; ++I) {
1512 const auto &DstIOperand = DstI.Operands[I];
1513 DstMIBuilder.addRenderer<CopyRenderer>(InsnMatcher, DstIOperand.Name);
1514 }
1515
1516 // Render the explicit uses.
Daniel Sanders0ed28822017-04-12 08:23:08 +00001517 unsigned Child = 0;
Diana Picus382602f2017-05-17 08:57:28 +00001518 unsigned DstINumUses = DstI.Operands.size() - DstI.Operands.NumDefs;
1519 unsigned NumDefaultOps = 0;
Daniel Sanders0ed28822017-04-12 08:23:08 +00001520 for (unsigned I = 0; I != DstINumUses; ++I) {
Diana Picus382602f2017-05-17 08:57:28 +00001521 const auto &DstIOperand = DstI.Operands[DstI.Operands.NumDefs + I];
Daniel Sanders0ed28822017-04-12 08:23:08 +00001522
Diana Picus382602f2017-05-17 08:57:28 +00001523 // If the operand has default values, introduce them now.
1524 // FIXME: Until we have a decent test case that dictates we should do
1525 // otherwise, we're going to assume that operands with default values cannot
1526 // be specified in the patterns. Therefore, adding them will not cause us to
1527 // end up with too many rendered operands.
1528 if (DstIOperand.Rec->isSubClassOf("OperandWithDefaultOps")) {
Daniel Sanders0ed28822017-04-12 08:23:08 +00001529 DagInit *DefaultOps = DstIOperand.Rec->getValueAsDag("DefaultOps");
Diana Picus382602f2017-05-17 08:57:28 +00001530 if (auto Error = importDefaultOperandRenderers(DstMIBuilder, DefaultOps))
1531 return std::move(Error);
1532 ++NumDefaultOps;
Daniel Sanders0ed28822017-04-12 08:23:08 +00001533 continue;
1534 }
1535
1536 if (auto Error = importExplicitUseRenderer(
1537 DstMIBuilder, Dst->getChild(Child), InsnMatcher))
Daniel Sandersffc7d582017-03-29 15:37:18 +00001538 return std::move(Error);
Daniel Sanders0ed28822017-04-12 08:23:08 +00001539 ++Child;
Daniel Sandersffc7d582017-03-29 15:37:18 +00001540 }
1541
Diana Picus382602f2017-05-17 08:57:28 +00001542 if (NumDefaultOps + Dst->getNumChildren() != DstINumUses)
Diana Picuseb2057c2017-05-17 09:25:08 +00001543 return failedImport("Expected " + llvm::to_string(DstINumUses) +
Diana Picus382602f2017-05-17 08:57:28 +00001544 " used operands but found " +
Diana Picuseb2057c2017-05-17 09:25:08 +00001545 llvm::to_string(Dst->getNumChildren()) +
1546 " explicit ones and " + llvm::to_string(NumDefaultOps) +
Diana Picus382602f2017-05-17 08:57:28 +00001547 " default ones");
1548
Daniel Sandersffc7d582017-03-29 15:37:18 +00001549 return DstMIBuilder;
1550}
1551
Diana Picus382602f2017-05-17 08:57:28 +00001552Error GlobalISelEmitter::importDefaultOperandRenderers(
1553 BuildMIAction &DstMIBuilder, DagInit *DefaultOps) const {
1554 for (const auto *DefaultOp : DefaultOps->args()) {
1555 // Look through ValueType operators.
1556 if (const DagInit *DefaultDagOp = dyn_cast<DagInit>(DefaultOp)) {
1557 if (const DefInit *DefaultDagOperator =
1558 dyn_cast<DefInit>(DefaultDagOp->getOperator())) {
1559 if (DefaultDagOperator->getDef()->isSubClassOf("ValueType"))
1560 DefaultOp = DefaultDagOp->getArg(0);
1561 }
1562 }
1563
1564 if (const DefInit *DefaultDefOp = dyn_cast<DefInit>(DefaultOp)) {
1565 DstMIBuilder.addRenderer<AddRegisterRenderer>(DefaultDefOp->getDef());
1566 continue;
1567 }
1568
1569 if (const IntInit *DefaultIntOp = dyn_cast<IntInit>(DefaultOp)) {
1570 DstMIBuilder.addRenderer<ImmRenderer>(DefaultIntOp->getValue());
1571 continue;
1572 }
1573
1574 return failedImport("Could not add default op");
1575 }
1576
1577 return Error::success();
1578}
1579
Daniel Sandersc270c502017-03-30 09:36:33 +00001580Error GlobalISelEmitter::importImplicitDefRenderers(
Daniel Sandersffc7d582017-03-29 15:37:18 +00001581 BuildMIAction &DstMIBuilder,
1582 const std::vector<Record *> &ImplicitDefs) const {
1583 if (!ImplicitDefs.empty())
1584 return failedImport("Pattern defines a physical register");
Daniel Sandersc270c502017-03-30 09:36:33 +00001585 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001586}
1587
1588Expected<RuleMatcher> GlobalISelEmitter::runOnPattern(const PatternToMatch &P) {
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001589 // Keep track of the matchers and actions to emit.
Ahmed Bougacha9aa4c102017-02-04 00:47:08 +00001590 RuleMatcher M;
1591 M.addAction<DebugCommentAction>(P);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001592
Daniel Sandersc270c502017-03-30 09:36:33 +00001593 if (auto Error = importRulePredicates(M, P.getPredicates()->getValues()))
Daniel Sandersffc7d582017-03-29 15:37:18 +00001594 return std::move(Error);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001595
1596 // Next, analyze the pattern operators.
1597 TreePatternNode *Src = P.getSrcPattern();
1598 TreePatternNode *Dst = P.getDstPattern();
1599
1600 // If the root of either pattern isn't a simple operator, ignore it.
Daniel Sandersd0656a32017-04-13 09:45:37 +00001601 if (auto Err = isTrivialOperatorNode(Dst))
1602 return failedImport("Dst pattern root isn't a trivial operator (" +
1603 toString(std::move(Err)) + ")");
1604 if (auto Err = isTrivialOperatorNode(Src))
1605 return failedImport("Src pattern root isn't a trivial operator (" +
1606 toString(std::move(Err)) + ")");
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001607
Daniel Sandersbee57392017-04-04 13:25:23 +00001608 // Start with the defined operands (i.e., the results of the root operator).
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001609 Record *DstOp = Dst->getOperator();
1610 if (!DstOp->isSubClassOf("Instruction"))
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001611 return failedImport("Pattern operator isn't an instruction");
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001612
1613 auto &DstI = Target.getInstruction(DstOp);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001614 if (DstI.Operands.NumDefs != Src->getExtTypes().size())
Daniel Sandersd0656a32017-04-13 09:45:37 +00001615 return failedImport("Src pattern results and dst MI defs are different (" +
1616 to_string(Src->getExtTypes().size()) + " def(s) vs " +
1617 to_string(DstI.Operands.NumDefs) + " def(s))");
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001618
Daniel Sandersffc7d582017-03-29 15:37:18 +00001619 InstructionMatcher &InsnMatcherTemp = M.addInstructionMatcher();
Daniel Sandersc270c502017-03-30 09:36:33 +00001620 auto InsnMatcherOrError = createAndImportSelDAGMatcher(InsnMatcherTemp, Src);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001621 if (auto Error = InsnMatcherOrError.takeError())
1622 return std::move(Error);
1623 InstructionMatcher &InsnMatcher = InsnMatcherOrError.get();
1624
1625 // The root of the match also has constraints on the register bank so that it
1626 // matches the result instruction.
1627 unsigned OpIdx = 0;
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001628 for (const EEVT::TypeSet &Ty : Src->getExtTypes()) {
Daniel Sandersffc7d582017-03-29 15:37:18 +00001629 (void)Ty;
1630
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001631 const auto &DstIOperand = DstI.Operands[OpIdx];
1632 Record *DstIOpRec = DstIOperand.Rec;
Daniel Sanders658541f2017-04-22 15:53:21 +00001633 if (DstIOpRec->isSubClassOf("RegisterOperand"))
1634 DstIOpRec = DstIOpRec->getValueAsDef("RegClass");
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001635 if (!DstIOpRec->isSubClassOf("RegisterClass"))
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001636 return failedImport("Dst MI def isn't a register class");
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001637
Daniel Sandersffc7d582017-03-29 15:37:18 +00001638 OperandMatcher &OM = InsnMatcher.getOperand(OpIdx);
1639 OM.setSymbolicName(DstIOperand.Name);
Daniel Sandersdc662ff2017-01-26 11:10:14 +00001640 OM.addPredicate<RegisterBankOperandMatcher>(
1641 Target.getRegisterClass(DstIOpRec));
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001642 ++OpIdx;
1643 }
1644
Daniel Sandersc270c502017-03-30 09:36:33 +00001645 auto DstMIBuilderOrError =
1646 createAndImportInstructionRenderer(M, Dst, InsnMatcher);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001647 if (auto Error = DstMIBuilderOrError.takeError())
1648 return std::move(Error);
1649 BuildMIAction &DstMIBuilder = DstMIBuilderOrError.get();
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001650
Daniel Sandersffc7d582017-03-29 15:37:18 +00001651 // Render the implicit defs.
1652 // These are only added to the root of the result.
Daniel Sandersc270c502017-03-30 09:36:33 +00001653 if (auto Error = importImplicitDefRenderers(DstMIBuilder, P.getDstRegs()))
Daniel Sandersffc7d582017-03-29 15:37:18 +00001654 return std::move(Error);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001655
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001656 // We're done with this pattern! It's eligible for GISel emission; return it.
Daniel Sandersb41ce2b2017-02-20 14:31:27 +00001657 ++NumPatternImported;
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001658 return std::move(M);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001659}
1660
1661void GlobalISelEmitter::run(raw_ostream &OS) {
1662 // Track the GINodeEquiv definitions.
1663 gatherNodeEquivs();
1664
1665 emitSourceFileHeader(("Global Instruction Selector for the " +
1666 Target.getName() + " target").str(), OS);
Daniel Sandersb41ce2b2017-02-20 14:31:27 +00001667 std::vector<RuleMatcher> Rules;
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001668 // Look through the SelectionDAG patterns we found, possibly emitting some.
1669 for (const PatternToMatch &Pat : CGP.ptms()) {
1670 ++NumPatternTotal;
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001671 auto MatcherOrErr = runOnPattern(Pat);
1672
1673 // The pattern analysis can fail, indicating an unsupported pattern.
1674 // Report that if we've been asked to do so.
1675 if (auto Err = MatcherOrErr.takeError()) {
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001676 if (WarnOnSkippedPatterns) {
1677 PrintWarning(Pat.getSrcRecord()->getLoc(),
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001678 "Skipped pattern: " + toString(std::move(Err)));
1679 } else {
1680 consumeError(std::move(Err));
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001681 }
Daniel Sandersb41ce2b2017-02-20 14:31:27 +00001682 ++NumPatternImportsSkipped;
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001683 continue;
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001684 }
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001685
Daniel Sandersb41ce2b2017-02-20 14:31:27 +00001686 Rules.push_back(std::move(MatcherOrErr.get()));
1687 }
1688
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001689 std::stable_sort(Rules.begin(), Rules.end(),
1690 [&](const RuleMatcher &A, const RuleMatcher &B) {
Daniel Sanders759ff412017-02-24 13:58:11 +00001691 if (A.isHigherPriorityThan(B)) {
1692 assert(!B.isHigherPriorityThan(A) && "Cannot be more important "
1693 "and less important at "
1694 "the same time");
1695 return true;
1696 }
1697 return false;
1698 });
1699
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001700 unsigned MaxTemporaries = 0;
1701 for (const auto &Rule : Rules)
Daniel Sanders2deea182017-04-22 15:11:04 +00001702 MaxTemporaries = std::max(MaxTemporaries, Rule.countRendererFns());
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001703
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001704 OS << "#ifdef GET_GLOBALISEL_PREDICATE_BITSET\n"
1705 << "const unsigned MAX_SUBTARGET_PREDICATES = " << SubtargetFeatures.size()
1706 << ";\n"
1707 << "using PredicateBitset = "
1708 "llvm::PredicateBitsetImpl<MAX_SUBTARGET_PREDICATES>;\n"
1709 << "#endif // ifdef GET_GLOBALISEL_PREDICATE_BITSET\n\n";
1710
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001711 OS << "#ifdef GET_GLOBALISEL_TEMPORARIES_DECL\n";
1712 for (unsigned I = 0; I < MaxTemporaries; ++I)
Daniel Sanders2deea182017-04-22 15:11:04 +00001713 OS << " mutable ComplexRendererFn Renderer" << I << ";\n";
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001714 OS << "#endif // ifdef GET_GLOBALISEL_TEMPORARIES_DECL\n\n";
1715
1716 OS << "#ifdef GET_GLOBALISEL_TEMPORARIES_INIT\n";
1717 for (unsigned I = 0; I < MaxTemporaries; ++I)
Daniel Sanders2deea182017-04-22 15:11:04 +00001718 OS << ", Renderer" << I << "(nullptr)\n";
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001719 OS << "#endif // ifdef GET_GLOBALISEL_TEMPORARIES_INIT\n\n";
1720
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001721 OS << "#ifdef GET_GLOBALISEL_IMPL\n";
1722 SubtargetFeatureInfo::emitSubtargetFeatureBitEnumeration(SubtargetFeatures,
1723 OS);
Daniel Sanderse9fdba32017-04-29 17:30:09 +00001724
1725 // Separate subtarget features by how often they must be recomputed.
1726 SubtargetFeatureInfoMap ModuleFeatures;
1727 std::copy_if(SubtargetFeatures.begin(), SubtargetFeatures.end(),
1728 std::inserter(ModuleFeatures, ModuleFeatures.end()),
1729 [](const SubtargetFeatureInfoMap::value_type &X) {
1730 return !X.second.mustRecomputePerFunction();
1731 });
1732 SubtargetFeatureInfoMap FunctionFeatures;
1733 std::copy_if(SubtargetFeatures.begin(), SubtargetFeatures.end(),
1734 std::inserter(FunctionFeatures, FunctionFeatures.end()),
1735 [](const SubtargetFeatureInfoMap::value_type &X) {
1736 return X.second.mustRecomputePerFunction();
1737 });
1738
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001739 SubtargetFeatureInfo::emitComputeAvailableFeatures(
Daniel Sanderse9fdba32017-04-29 17:30:09 +00001740 Target.getName(), "InstructionSelector", "computeAvailableModuleFeatures",
1741 ModuleFeatures, OS);
1742 SubtargetFeatureInfo::emitComputeAvailableFeatures(
1743 Target.getName(), "InstructionSelector",
1744 "computeAvailableFunctionFeatures", FunctionFeatures, OS,
1745 "const MachineFunction *MF");
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001746
1747 OS << "bool " << Target.getName()
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001748 << "InstructionSelector::selectImpl(MachineInstr &I) const {\n"
1749 << " MachineFunction &MF = *I.getParent()->getParent();\n"
Daniel Sanderse9fdba32017-04-29 17:30:09 +00001750 << " const MachineRegisterInfo &MRI = MF.getRegInfo();\n"
1751 << " // FIXME: This should be computed on a per-function basis rather than per-insn.\n"
1752 << " AvailableFunctionFeatures = computeAvailableFunctionFeatures(&STI, &MF);\n"
1753 << " const PredicateBitset AvailableFeatures = getAvailableFeatures();\n";
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001754
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001755 for (auto &Rule : Rules) {
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001756 Rule.emit(OS, SubtargetFeatures);
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001757 ++NumPatternEmitted;
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001758 }
1759
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001760 OS << " return false;\n"
1761 << "}\n"
1762 << "#endif // ifdef GET_GLOBALISEL_IMPL\n";
Daniel Sanderse9fdba32017-04-29 17:30:09 +00001763
1764 OS << "#ifdef GET_GLOBALISEL_PREDICATES_DECL\n"
1765 << "PredicateBitset AvailableModuleFeatures;\n"
1766 << "mutable PredicateBitset AvailableFunctionFeatures;\n"
1767 << "PredicateBitset getAvailableFeatures() const {\n"
1768 << " return AvailableModuleFeatures | AvailableFunctionFeatures;\n"
1769 << "}\n"
1770 << "PredicateBitset\n"
1771 << "computeAvailableModuleFeatures(const " << Target.getName()
1772 << "Subtarget *Subtarget) const;\n"
1773 << "PredicateBitset\n"
1774 << "computeAvailableFunctionFeatures(const " << Target.getName()
1775 << "Subtarget *Subtarget,\n"
1776 << " const MachineFunction *MF) const;\n"
1777 << "#endif // ifdef GET_GLOBALISEL_PREDICATES_DECL\n";
1778
1779 OS << "#ifdef GET_GLOBALISEL_PREDICATES_INIT\n"
1780 << "AvailableModuleFeatures(computeAvailableModuleFeatures(&STI)),\n"
1781 << "AvailableFunctionFeatures()\n"
1782 << "#endif // ifdef GET_GLOBALISEL_PREDICATES_INIT\n";
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001783}
1784
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001785void GlobalISelEmitter::declareSubtargetFeature(Record *Predicate) {
1786 if (SubtargetFeatures.count(Predicate) == 0)
1787 SubtargetFeatures.emplace(
1788 Predicate, SubtargetFeatureInfo(Predicate, SubtargetFeatures.size()));
1789}
1790
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001791} // end anonymous namespace
1792
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001793//===----------------------------------------------------------------------===//
1794
1795namespace llvm {
1796void EmitGlobalISel(RecordKeeper &RK, raw_ostream &OS) {
1797 GlobalISelEmitter(RK).run(OS);
1798}
1799} // End llvm namespace