blob: 03d231a153dc30b47df57e0e3c1f4416d68e60d7 [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"))
Craig Topper2b8419a2017-05-31 19:01:11 +0000121 return (" (" + Operator->getValueAsString("Opcode") + ")").str();
Daniel Sandersd0656a32017-04-13 09:45:37 +0000122
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()) {
Daniel Sanders3334cc02017-05-23 20:02:48 +0000138 if (isa<IntInit>(N->getLeafValue()))
Daniel Sanders452c8ae2017-05-23 19:33:16 +0000139 return Error::success();
140
Daniel Sandersd0656a32017-04-13 09:45:37 +0000141 Explanation = "Is a leaf";
142 Separator = ", ";
143 }
144
145 if (N->hasAnyPredicate()) {
146 Explanation = Separator + "Has a predicate (" + explainPredicates(N) + ")";
147 Separator = ", ";
148 }
149
150 if (N->getTransformFn()) {
151 Explanation += Separator + "Has a transform function";
152 Separator = ", ";
153 }
154
155 if (!N->isLeaf() && !N->hasAnyPredicate() && !N->getTransformFn())
156 return Error::success();
157
158 return failedImport(Explanation);
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000159}
160
Daniel Sandersa6e2ceb2017-06-20 12:36:34 +0000161static Record *getInitValueAsRegClass(Init *V) {
162 if (DefInit *VDefInit = dyn_cast<DefInit>(V)) {
163 if (VDefInit->getDef()->isSubClassOf("RegisterOperand"))
164 return VDefInit->getDef()->getValueAsDef("RegClass");
165 if (VDefInit->getDef()->isSubClassOf("RegisterClass"))
166 return VDefInit->getDef();
167 }
168 return nullptr;
169}
170
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000171//===- Matchers -----------------------------------------------------------===//
172
Daniel Sandersbee57392017-04-04 13:25:23 +0000173class OperandMatcher;
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000174class MatchAction;
175
176/// Generates code to check that a match rule matches.
177class RuleMatcher {
178 /// A list of matchers that all need to succeed for the current rule to match.
179 /// FIXME: This currently supports a single match position but could be
180 /// extended to support multiple positions to support div/rem fusion or
181 /// load-multiple instructions.
182 std::vector<std::unique_ptr<InstructionMatcher>> Matchers;
183
184 /// A list of actions that need to be taken when all predicates in this rule
185 /// have succeeded.
186 std::vector<std::unique_ptr<MatchAction>> Actions;
187
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000188 /// A map of instruction matchers to the local variables created by
189 /// emitCxxCaptureStmts().
190 std::map<const InstructionMatcher *, std::string> InsnVariableNames;
191
192 /// ID for the next instruction variable defined with defineInsnVar()
193 unsigned NextInsnVarID;
194
Daniel Sanderse7b0d662017-04-21 15:59:56 +0000195 std::vector<Record *> RequiredFeatures;
196
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000197public:
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000198 RuleMatcher()
199 : Matchers(), Actions(), InsnVariableNames(), NextInsnVarID(0) {}
Zachary Turnerb7dbd872017-03-20 19:56:52 +0000200 RuleMatcher(RuleMatcher &&Other) = default;
201 RuleMatcher &operator=(RuleMatcher &&Other) = default;
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000202
203 InstructionMatcher &addInstructionMatcher();
Daniel Sanderse7b0d662017-04-21 15:59:56 +0000204 void addRequiredFeature(Record *Feature);
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000205
206 template <class Kind, class... Args> Kind &addAction(Args &&... args);
207
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000208 std::string defineInsnVar(raw_ostream &OS, const InstructionMatcher &Matcher,
209 StringRef Value);
210 StringRef getInsnVarName(const InstructionMatcher &InsnMatcher) const;
211
Daniel Sandersbee57392017-04-04 13:25:23 +0000212 void emitCxxCapturedInsnList(raw_ostream &OS);
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000213 void emitCxxCaptureStmts(raw_ostream &OS, StringRef Expr);
214
Daniel Sanderse9fdba32017-04-29 17:30:09 +0000215void emit(raw_ostream &OS, SubtargetFeatureInfoMap SubtargetFeatures);
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000216
Daniel Sanderse9fdba32017-04-29 17:30:09 +0000217/// Compare the priority of this object and B.
218///
219/// Returns true if this object is more important than B.
220bool isHigherPriorityThan(const RuleMatcher &B) const;
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000221
Daniel Sanderse9fdba32017-04-29 17:30:09 +0000222/// Report the maximum number of temporary operands needed by the rule
223/// matcher.
224unsigned countRendererFns() const;
Daniel Sanders2deea182017-04-22 15:11:04 +0000225
Daniel Sanderse9fdba32017-04-29 17:30:09 +0000226// FIXME: Remove this as soon as possible
227InstructionMatcher &insnmatcher_front() const { return *Matchers.front(); }
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000228};
229
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000230template <class PredicateTy> class PredicateListMatcher {
231private:
232 typedef std::vector<std::unique_ptr<PredicateTy>> PredicateVec;
233 PredicateVec Predicates;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000234
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000235public:
236 /// Construct a new operand predicate and add it to the matcher.
237 template <class Kind, class... Args>
238 Kind &addPredicate(Args&&... args) {
Ahmed Bougachab67a3ce2017-01-26 22:07:37 +0000239 Predicates.emplace_back(
240 llvm::make_unique<Kind>(std::forward<Args>(args)...));
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000241 return *static_cast<Kind *>(Predicates.back().get());
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000242 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000243
244 typename PredicateVec::const_iterator predicates_begin() const { return Predicates.begin(); }
245 typename PredicateVec::const_iterator predicates_end() const { return Predicates.end(); }
246 iterator_range<typename PredicateVec::const_iterator> predicates() const {
247 return make_range(predicates_begin(), predicates_end());
248 }
Daniel Sanders759ff412017-02-24 13:58:11 +0000249 typename PredicateVec::size_type predicates_size() const { return Predicates.size(); }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000250
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000251 /// Emit a C++ expression that tests whether all the predicates are met.
Ahmed Bougachab67a3ce2017-01-26 22:07:37 +0000252 template <class... Args>
Daniel Sandersf8c804f2017-01-28 11:10:42 +0000253 void emitCxxPredicateListExpr(raw_ostream &OS, Args &&... args) const {
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000254 if (Predicates.empty()) {
255 OS << "true";
256 return;
257 }
258
259 StringRef Separator = "";
260 for (const auto &Predicate : predicates()) {
261 OS << Separator << "(";
Ahmed Bougachab67a3ce2017-01-26 22:07:37 +0000262 Predicate->emitCxxPredicateExpr(OS, std::forward<Args>(args)...);
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000263 OS << ")";
Ahmed Bougacha905af9f2017-02-04 00:47:02 +0000264 Separator = " &&\n";
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000265 }
266 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000267};
268
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000269/// Generates code to check a predicate of an operand.
270///
271/// Typical predicates include:
272/// * Operand is a particular register.
273/// * Operand is assigned a particular register bank.
274/// * Operand is an MBB.
275class OperandPredicateMatcher {
276public:
Daniel Sanders759ff412017-02-24 13:58:11 +0000277 /// This enum is used for RTTI and also defines the priority that is given to
278 /// the predicate when generating the matcher code. Kinds with higher priority
279 /// must be tested first.
280 ///
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000281 /// The relative priority of OPM_LLT, OPM_RegBank, and OPM_MBB do not matter
282 /// but OPM_Int must have priority over OPM_RegBank since constant integers
283 /// are represented by a virtual register defined by a G_CONSTANT instruction.
Daniel Sanders759ff412017-02-24 13:58:11 +0000284 enum PredicateKind {
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000285 OPM_ComplexPattern,
Daniel Sandersbee57392017-04-04 13:25:23 +0000286 OPM_Instruction,
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000287 OPM_Int,
Daniel Sanders452c8ae2017-05-23 19:33:16 +0000288 OPM_LiteralInt,
Daniel Sanders759ff412017-02-24 13:58:11 +0000289 OPM_LLT,
290 OPM_RegBank,
291 OPM_MBB,
292 };
293
294protected:
295 PredicateKind Kind;
296
297public:
298 OperandPredicateMatcher(PredicateKind Kind) : Kind(Kind) {}
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000299 virtual ~OperandPredicateMatcher() {}
300
Daniel Sanders759ff412017-02-24 13:58:11 +0000301 PredicateKind getKind() const { return Kind; }
302
Daniel Sandersbee57392017-04-04 13:25:23 +0000303 /// Return the OperandMatcher for the specified operand or nullptr if there
304 /// isn't one by that name in this operand predicate matcher.
305 ///
306 /// InstructionOperandMatcher is the only subclass that can return non-null
307 /// for this.
308 virtual Optional<const OperandMatcher *>
Daniel Sandersdb7ed372017-04-04 13:52:00 +0000309 getOptionalOperand(StringRef SymbolicName) const {
Daniel Sandersbee57392017-04-04 13:25:23 +0000310 assert(!SymbolicName.empty() && "Cannot lookup unnamed operand");
311 return None;
312 }
313
314 /// Emit C++ statements to capture instructions into local variables.
315 ///
316 /// Only InstructionOperandMatcher needs to do anything for this method.
317 virtual void emitCxxCaptureStmts(raw_ostream &OS, RuleMatcher &Rule,
318 StringRef Expr) const {}
319
Daniel Sanderse604ef52017-02-20 15:30:43 +0000320 /// Emit a C++ expression that checks the predicate for the given operand.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000321 virtual void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanderse604ef52017-02-20 15:30:43 +0000322 StringRef OperandExpr) const = 0;
Daniel Sanders759ff412017-02-24 13:58:11 +0000323
324 /// Compare the priority of this object and B.
325 ///
326 /// Returns true if this object is more important than B.
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000327 virtual bool isHigherPriorityThan(const OperandPredicateMatcher &B) const {
328 return Kind < B.Kind;
Daniel Sanders759ff412017-02-24 13:58:11 +0000329 };
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000330
331 /// Report the maximum number of temporary operands needed by the predicate
332 /// matcher.
Daniel Sanders2deea182017-04-22 15:11:04 +0000333 virtual unsigned countRendererFns() const { return 0; }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000334};
335
336/// Generates code to check that an operand is a particular LLT.
337class LLTOperandMatcher : public OperandPredicateMatcher {
338protected:
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000339 LLTCodeGen Ty;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000340
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000341public:
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000342 LLTOperandMatcher(const LLTCodeGen &Ty)
Daniel Sanders759ff412017-02-24 13:58:11 +0000343 : OperandPredicateMatcher(OPM_LLT), Ty(Ty) {}
344
345 static bool classof(const OperandPredicateMatcher *P) {
346 return P->getKind() == OPM_LLT;
347 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000348
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000349 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanderse604ef52017-02-20 15:30:43 +0000350 StringRef OperandExpr) const override {
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000351 OS << "MRI.getType(" << OperandExpr << ".getReg()) == (";
352 Ty.emitCxxConstructorCall(OS);
353 OS << ")";
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000354 }
355};
356
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000357/// Generates code to check that an operand is a particular target constant.
358class ComplexPatternOperandMatcher : public OperandPredicateMatcher {
359protected:
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000360 const OperandMatcher &Operand;
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000361 const Record &TheDef;
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000362
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000363 unsigned getAllocatedTemporariesBaseID() const;
364
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000365public:
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000366 ComplexPatternOperandMatcher(const OperandMatcher &Operand,
367 const Record &TheDef)
368 : OperandPredicateMatcher(OPM_ComplexPattern), Operand(Operand),
369 TheDef(TheDef) {}
370
371 static bool classof(const OperandPredicateMatcher *P) {
372 return P->getKind() == OPM_ComplexPattern;
373 }
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000374
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000375 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000376 StringRef OperandExpr) const override {
Daniel Sanders2deea182017-04-22 15:11:04 +0000377 unsigned ID = getAllocatedTemporariesBaseID();
378 OS << "(Renderer" << ID << " = " << TheDef.getValueAsString("MatcherFn")
379 << "(" << OperandExpr << "))";
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000380 }
381
Daniel Sanders2deea182017-04-22 15:11:04 +0000382 unsigned countRendererFns() const override {
383 return 1;
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000384 }
385};
386
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000387/// Generates code to check that an operand is in a particular register bank.
388class RegisterBankOperandMatcher : public OperandPredicateMatcher {
389protected:
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000390 const CodeGenRegisterClass &RC;
391
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000392public:
Daniel Sanders759ff412017-02-24 13:58:11 +0000393 RegisterBankOperandMatcher(const CodeGenRegisterClass &RC)
394 : OperandPredicateMatcher(OPM_RegBank), RC(RC) {}
395
396 static bool classof(const OperandPredicateMatcher *P) {
397 return P->getKind() == OPM_RegBank;
398 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000399
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000400 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanderse604ef52017-02-20 15:30:43 +0000401 StringRef OperandExpr) const override {
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000402 OS << "(&RBI.getRegBankFromRegClass(" << RC.getQualifiedName()
Daniel Sanderse604ef52017-02-20 15:30:43 +0000403 << "RegClass) == RBI.getRegBank(" << OperandExpr
404 << ".getReg(), MRI, TRI))";
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000405 }
406};
407
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000408/// Generates code to check that an operand is a basic block.
409class MBBOperandMatcher : public OperandPredicateMatcher {
410public:
Daniel Sanders759ff412017-02-24 13:58:11 +0000411 MBBOperandMatcher() : OperandPredicateMatcher(OPM_MBB) {}
412
413 static bool classof(const OperandPredicateMatcher *P) {
414 return P->getKind() == OPM_MBB;
415 }
416
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000417 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanderse604ef52017-02-20 15:30:43 +0000418 StringRef OperandExpr) const override {
419 OS << OperandExpr << ".isMBB()";
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000420 }
421};
422
Daniel Sanders452c8ae2017-05-23 19:33:16 +0000423/// Generates code to check that an operand is a G_CONSTANT with a particular
424/// int.
425class ConstantIntOperandMatcher : public OperandPredicateMatcher {
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000426protected:
427 int64_t Value;
428
429public:
Daniel Sanders452c8ae2017-05-23 19:33:16 +0000430 ConstantIntOperandMatcher(int64_t Value)
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000431 : OperandPredicateMatcher(OPM_Int), Value(Value) {}
432
433 static bool classof(const OperandPredicateMatcher *P) {
434 return P->getKind() == OPM_Int;
435 }
436
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000437 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Simon Pilgrimd0302912017-02-24 17:20:27 +0000438 StringRef OperandExpr) const override {
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000439 OS << "isOperandImmEqual(" << OperandExpr << ", " << Value << ", MRI)";
440 }
441};
442
Daniel Sanders452c8ae2017-05-23 19:33:16 +0000443/// Generates code to check that an operand is a raw int (where MO.isImm() or
444/// MO.isCImm() is true).
445class LiteralIntOperandMatcher : public OperandPredicateMatcher {
446protected:
447 int64_t Value;
448
449public:
450 LiteralIntOperandMatcher(int64_t Value)
451 : OperandPredicateMatcher(OPM_LiteralInt), Value(Value) {}
452
453 static bool classof(const OperandPredicateMatcher *P) {
454 return P->getKind() == OPM_LiteralInt;
455 }
456
457 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
458 StringRef OperandExpr) const override {
459 OS << OperandExpr << ".isCImm() && " << OperandExpr
460 << ".getCImm()->equalsInt(" << Value << ")";
461 }
462};
463
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000464/// Generates code to check that a set of predicates match for a particular
465/// operand.
466class OperandMatcher : public PredicateListMatcher<OperandPredicateMatcher> {
467protected:
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000468 InstructionMatcher &Insn;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000469 unsigned OpIdx;
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000470 std::string SymbolicName;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000471
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000472 /// The index of the first temporary variable allocated to this operand. The
473 /// number of allocated temporaries can be found with
Daniel Sanders2deea182017-04-22 15:11:04 +0000474 /// countRendererFns().
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000475 unsigned AllocatedTemporariesBaseID;
476
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000477public:
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000478 OperandMatcher(InstructionMatcher &Insn, unsigned OpIdx,
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000479 const std::string &SymbolicName,
480 unsigned AllocatedTemporariesBaseID)
481 : Insn(Insn), OpIdx(OpIdx), SymbolicName(SymbolicName),
482 AllocatedTemporariesBaseID(AllocatedTemporariesBaseID) {}
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000483
484 bool hasSymbolicName() const { return !SymbolicName.empty(); }
485 const StringRef getSymbolicName() const { return SymbolicName; }
Daniel Sandersffc7d582017-03-29 15:37:18 +0000486 void setSymbolicName(StringRef Name) {
487 assert(SymbolicName.empty() && "Operand already has a symbolic name");
488 SymbolicName = Name;
489 }
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000490 unsigned getOperandIndex() const { return OpIdx; }
491
Daniel Sandersdb7ed372017-04-04 13:52:00 +0000492 std::string getOperandExpr(StringRef InsnVarName) const {
Pavel Labath52a82e22017-02-21 09:19:41 +0000493 return (InsnVarName + ".getOperand(" + llvm::to_string(OpIdx) + ")").str();
Daniel Sanderse604ef52017-02-20 15:30:43 +0000494 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000495
Daniel Sandersbee57392017-04-04 13:25:23 +0000496 Optional<const OperandMatcher *>
497 getOptionalOperand(StringRef DesiredSymbolicName) const {
498 assert(!DesiredSymbolicName.empty() && "Cannot lookup unnamed operand");
499 if (DesiredSymbolicName == SymbolicName)
500 return this;
501 for (const auto &OP : predicates()) {
502 const auto &MaybeOperand = OP->getOptionalOperand(DesiredSymbolicName);
503 if (MaybeOperand.hasValue())
504 return MaybeOperand.getValue();
505 }
506 return None;
507 }
508
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000509 InstructionMatcher &getInstructionMatcher() const { return Insn; }
510
Daniel Sandersbee57392017-04-04 13:25:23 +0000511 /// Emit C++ statements to capture instructions into local variables.
512 void emitCxxCaptureStmts(raw_ostream &OS, RuleMatcher &Rule,
513 StringRef OperandExpr) const {
514 for (const auto &Predicate : predicates())
515 Predicate->emitCxxCaptureStmts(OS, Rule, OperandExpr);
516 }
517
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000518 /// Emit a C++ expression that tests whether the instruction named in
519 /// InsnVarName matches all the predicate and all the operands.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000520 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sandersdb7ed372017-04-04 13:52:00 +0000521 StringRef InsnVarName) const {
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000522 OS << "(/* ";
523 if (SymbolicName.empty())
524 OS << "Operand " << OpIdx;
525 else
526 OS << SymbolicName;
527 OS << " */ ";
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000528 emitCxxPredicateListExpr(OS, Rule, getOperandExpr(InsnVarName));
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000529 OS << ")";
530 }
Daniel Sanders759ff412017-02-24 13:58:11 +0000531
532 /// Compare the priority of this object and B.
533 ///
534 /// Returns true if this object is more important than B.
535 bool isHigherPriorityThan(const OperandMatcher &B) const {
536 // Operand matchers involving more predicates have higher priority.
537 if (predicates_size() > B.predicates_size())
538 return true;
539 if (predicates_size() < B.predicates_size())
540 return false;
541
542 // This assumes that predicates are added in a consistent order.
543 for (const auto &Predicate : zip(predicates(), B.predicates())) {
544 if (std::get<0>(Predicate)->isHigherPriorityThan(*std::get<1>(Predicate)))
545 return true;
546 if (std::get<1>(Predicate)->isHigherPriorityThan(*std::get<0>(Predicate)))
547 return false;
548 }
549
550 return false;
551 };
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000552
553 /// Report the maximum number of temporary operands needed by the operand
554 /// matcher.
Daniel Sanders2deea182017-04-22 15:11:04 +0000555 unsigned countRendererFns() const {
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000556 return std::accumulate(
557 predicates().begin(), predicates().end(), 0,
558 [](unsigned A,
559 const std::unique_ptr<OperandPredicateMatcher> &Predicate) {
Daniel Sanders2deea182017-04-22 15:11:04 +0000560 return A + Predicate->countRendererFns();
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000561 });
562 }
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000563
564 unsigned getAllocatedTemporariesBaseID() const {
565 return AllocatedTemporariesBaseID;
566 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000567};
568
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000569unsigned ComplexPatternOperandMatcher::getAllocatedTemporariesBaseID() const {
570 return Operand.getAllocatedTemporariesBaseID();
571}
572
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000573/// Generates code to check a predicate on an instruction.
574///
575/// Typical predicates include:
576/// * The opcode of the instruction is a particular value.
577/// * The nsw/nuw flag is/isn't set.
578class InstructionPredicateMatcher {
Daniel Sanders759ff412017-02-24 13:58:11 +0000579protected:
580 /// This enum is used for RTTI and also defines the priority that is given to
581 /// the predicate when generating the matcher code. Kinds with higher priority
582 /// must be tested first.
583 enum PredicateKind {
584 IPM_Opcode,
585 };
586
587 PredicateKind Kind;
588
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000589public:
Daniel Sanders8d4d72f2017-02-24 14:53:35 +0000590 InstructionPredicateMatcher(PredicateKind Kind) : Kind(Kind) {}
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000591 virtual ~InstructionPredicateMatcher() {}
592
Daniel Sanders759ff412017-02-24 13:58:11 +0000593 PredicateKind getKind() const { return Kind; }
594
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000595 /// Emit a C++ expression that tests whether the instruction named in
596 /// InsnVarName matches the predicate.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000597 virtual void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Ahmed Bougacha6a1ac5a2017-02-09 02:50:01 +0000598 StringRef InsnVarName) const = 0;
Daniel Sanders759ff412017-02-24 13:58:11 +0000599
600 /// Compare the priority of this object and B.
601 ///
602 /// Returns true if this object is more important than B.
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000603 virtual bool isHigherPriorityThan(const InstructionPredicateMatcher &B) const {
Daniel Sanders759ff412017-02-24 13:58:11 +0000604 return Kind < B.Kind;
605 };
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000606
607 /// Report the maximum number of temporary operands needed by the predicate
608 /// matcher.
Daniel Sanders2deea182017-04-22 15:11:04 +0000609 virtual unsigned countRendererFns() const { return 0; }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000610};
611
612/// Generates code to check the opcode of an instruction.
613class InstructionOpcodeMatcher : public InstructionPredicateMatcher {
614protected:
615 const CodeGenInstruction *I;
616
617public:
Daniel Sanders8d4d72f2017-02-24 14:53:35 +0000618 InstructionOpcodeMatcher(const CodeGenInstruction *I)
619 : InstructionPredicateMatcher(IPM_Opcode), I(I) {}
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000620
Daniel Sanders759ff412017-02-24 13:58:11 +0000621 static bool classof(const InstructionPredicateMatcher *P) {
622 return P->getKind() == IPM_Opcode;
623 }
624
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000625 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Ahmed Bougacha6a1ac5a2017-02-09 02:50:01 +0000626 StringRef InsnVarName) const override {
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000627 OS << InsnVarName << ".getOpcode() == " << I->Namespace
628 << "::" << I->TheDef->getName();
629 }
Daniel Sanders759ff412017-02-24 13:58:11 +0000630
631 /// Compare the priority of this object and B.
632 ///
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000633 /// Returns true if this object is more important than B.
634 bool isHigherPriorityThan(const InstructionPredicateMatcher &B) const override {
Daniel Sanders759ff412017-02-24 13:58:11 +0000635 if (InstructionPredicateMatcher::isHigherPriorityThan(B))
636 return true;
637 if (B.InstructionPredicateMatcher::isHigherPriorityThan(*this))
638 return false;
639
640 // Prioritize opcodes for cosmetic reasons in the generated source. Although
641 // this is cosmetic at the moment, we may want to drive a similar ordering
642 // using instruction frequency information to improve compile time.
643 if (const InstructionOpcodeMatcher *BO =
644 dyn_cast<InstructionOpcodeMatcher>(&B))
645 return I->TheDef->getName() < BO->I->TheDef->getName();
646
647 return false;
648 };
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000649};
650
651/// Generates code to check that a set of predicates and operands match for a
652/// particular instruction.
653///
654/// Typical predicates include:
655/// * Has a specific opcode.
656/// * Has an nsw/nuw flag or doesn't.
657class InstructionMatcher
658 : public PredicateListMatcher<InstructionPredicateMatcher> {
659protected:
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000660 typedef std::vector<std::unique_ptr<OperandMatcher>> OperandVec;
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000661
662 /// The operands to match. All rendered operands must be present even if the
663 /// condition is always true.
664 OperandVec Operands;
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000665
666public:
667 /// Add an operand to the matcher.
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000668 OperandMatcher &addOperand(unsigned OpIdx, const std::string &SymbolicName,
669 unsigned AllocatedTemporariesBaseID) {
670 Operands.emplace_back(new OperandMatcher(*this, OpIdx, SymbolicName,
671 AllocatedTemporariesBaseID));
672 return *Operands.back();
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000673 }
674
Daniel Sandersffc7d582017-03-29 15:37:18 +0000675 OperandMatcher &getOperand(unsigned OpIdx) {
676 auto I = std::find_if(Operands.begin(), Operands.end(),
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000677 [&OpIdx](const std::unique_ptr<OperandMatcher> &X) {
678 return X->getOperandIndex() == OpIdx;
Daniel Sandersffc7d582017-03-29 15:37:18 +0000679 });
680 if (I != Operands.end())
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000681 return **I;
Daniel Sandersffc7d582017-03-29 15:37:18 +0000682 llvm_unreachable("Failed to lookup operand");
683 }
684
Daniel Sandersbee57392017-04-04 13:25:23 +0000685 Optional<const OperandMatcher *>
686 getOptionalOperand(StringRef SymbolicName) const {
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000687 assert(!SymbolicName.empty() && "Cannot lookup unnamed operand");
Daniel Sandersbee57392017-04-04 13:25:23 +0000688 for (const auto &Operand : Operands) {
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000689 const auto &OM = Operand->getOptionalOperand(SymbolicName);
Daniel Sandersbee57392017-04-04 13:25:23 +0000690 if (OM.hasValue())
691 return OM.getValue();
692 }
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000693 return None;
694 }
695
Daniel Sandersdb7ed372017-04-04 13:52:00 +0000696 const OperandMatcher &getOperand(StringRef SymbolicName) const {
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000697 Optional<const OperandMatcher *>OM = getOptionalOperand(SymbolicName);
698 if (OM.hasValue())
699 return *OM.getValue();
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000700 llvm_unreachable("Failed to lookup operand");
701 }
702
703 unsigned getNumOperands() const { return Operands.size(); }
Daniel Sandersbee57392017-04-04 13:25:23 +0000704 OperandVec::iterator operands_begin() { return Operands.begin(); }
705 OperandVec::iterator operands_end() { return Operands.end(); }
706 iterator_range<OperandVec::iterator> operands() {
707 return make_range(operands_begin(), operands_end());
708 }
Daniel Sandersffc7d582017-03-29 15:37:18 +0000709 OperandVec::const_iterator operands_begin() const { return Operands.begin(); }
710 OperandVec::const_iterator operands_end() const { return Operands.end(); }
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000711 iterator_range<OperandVec::const_iterator> operands() const {
712 return make_range(operands_begin(), operands_end());
713 }
714
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000715 /// Emit C++ statements to check the shape of the match and capture
716 /// instructions into local variables.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000717 void emitCxxCaptureStmts(raw_ostream &OS, RuleMatcher &Rule, StringRef Expr) {
718 OS << "if (" << Expr << ".getNumOperands() < " << getNumOperands() << ")\n"
719 << " return false;\n";
Daniel Sandersbee57392017-04-04 13:25:23 +0000720 for (const auto &Operand : Operands) {
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000721 Operand->emitCxxCaptureStmts(OS, Rule, Operand->getOperandExpr(Expr));
Daniel Sandersbee57392017-04-04 13:25:23 +0000722 }
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000723 }
724
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000725 /// Emit a C++ expression that tests whether the instruction named in
726 /// InsnVarName matches all the predicates and all the operands.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000727 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
728 StringRef InsnVarName) const {
729 emitCxxPredicateListExpr(OS, Rule, InsnVarName);
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000730 for (const auto &Operand : Operands) {
Ahmed Bougacha905af9f2017-02-04 00:47:02 +0000731 OS << " &&\n(";
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000732 Operand->emitCxxPredicateExpr(OS, Rule, InsnVarName);
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000733 OS << ")";
734 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000735 }
Daniel Sanders759ff412017-02-24 13:58:11 +0000736
737 /// Compare the priority of this object and B.
738 ///
739 /// Returns true if this object is more important than B.
740 bool isHigherPriorityThan(const InstructionMatcher &B) const {
741 // Instruction matchers involving more operands have higher priority.
742 if (Operands.size() > B.Operands.size())
743 return true;
744 if (Operands.size() < B.Operands.size())
745 return false;
746
747 for (const auto &Predicate : zip(predicates(), B.predicates())) {
748 if (std::get<0>(Predicate)->isHigherPriorityThan(*std::get<1>(Predicate)))
749 return true;
750 if (std::get<1>(Predicate)->isHigherPriorityThan(*std::get<0>(Predicate)))
751 return false;
752 }
753
754 for (const auto &Operand : zip(Operands, B.Operands)) {
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000755 if (std::get<0>(Operand)->isHigherPriorityThan(*std::get<1>(Operand)))
Daniel Sanders759ff412017-02-24 13:58:11 +0000756 return true;
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000757 if (std::get<1>(Operand)->isHigherPriorityThan(*std::get<0>(Operand)))
Daniel Sanders759ff412017-02-24 13:58:11 +0000758 return false;
759 }
760
761 return false;
762 };
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000763
764 /// Report the maximum number of temporary operands needed by the instruction
765 /// matcher.
Daniel Sanders2deea182017-04-22 15:11:04 +0000766 unsigned countRendererFns() const {
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000767 return std::accumulate(predicates().begin(), predicates().end(), 0,
768 [](unsigned A,
769 const std::unique_ptr<InstructionPredicateMatcher>
770 &Predicate) {
Daniel Sanders2deea182017-04-22 15:11:04 +0000771 return A + Predicate->countRendererFns();
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000772 }) +
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000773 std::accumulate(
774 Operands.begin(), Operands.end(), 0,
775 [](unsigned A, const std::unique_ptr<OperandMatcher> &Operand) {
Daniel Sanders2deea182017-04-22 15:11:04 +0000776 return A + Operand->countRendererFns();
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000777 });
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000778 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000779};
780
Daniel Sandersbee57392017-04-04 13:25:23 +0000781/// Generates code to check that the operand is a register defined by an
782/// instruction that matches the given instruction matcher.
783///
784/// For example, the pattern:
785/// (set $dst, (G_MUL (G_ADD $src1, $src2), $src3))
786/// would use an InstructionOperandMatcher for operand 1 of the G_MUL to match
787/// the:
788/// (G_ADD $src1, $src2)
789/// subpattern.
790class InstructionOperandMatcher : public OperandPredicateMatcher {
791protected:
792 std::unique_ptr<InstructionMatcher> InsnMatcher;
793
794public:
795 InstructionOperandMatcher()
796 : OperandPredicateMatcher(OPM_Instruction),
797 InsnMatcher(new InstructionMatcher()) {}
798
799 static bool classof(const OperandPredicateMatcher *P) {
800 return P->getKind() == OPM_Instruction;
801 }
802
803 InstructionMatcher &getInsnMatcher() const { return *InsnMatcher; }
804
805 Optional<const OperandMatcher *>
806 getOptionalOperand(StringRef SymbolicName) const override {
807 assert(!SymbolicName.empty() && "Cannot lookup unnamed operand");
808 return InsnMatcher->getOptionalOperand(SymbolicName);
809 }
810
811 void emitCxxCaptureStmts(raw_ostream &OS, RuleMatcher &Rule,
812 StringRef OperandExpr) const override {
813 OS << "if (!" << OperandExpr + ".isReg())\n"
Daniel Sandersed205a02017-05-17 12:43:30 +0000814 << " return false;\n"
815 << "if (TRI.isPhysicalRegister(" << OperandExpr + ".getReg()))\n"
Daniel Sandersbee57392017-04-04 13:25:23 +0000816 << " return false;\n";
817 std::string InsnVarName = Rule.defineInsnVar(
818 OS, *InsnMatcher,
819 ("*MRI.getVRegDef(" + OperandExpr + ".getReg())").str());
820 InsnMatcher->emitCxxCaptureStmts(OS, Rule, InsnVarName);
821 }
822
823 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
824 StringRef OperandExpr) const override {
825 OperandExpr = Rule.getInsnVarName(*InsnMatcher);
826 OS << "(";
827 InsnMatcher->emitCxxPredicateExpr(OS, Rule, OperandExpr);
828 OS << ")\n";
829 }
830};
831
Daniel Sanders43c882c2017-02-01 10:53:10 +0000832//===- Actions ------------------------------------------------------------===//
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000833class OperandRenderer {
834public:
Daniel Sanders0ed28822017-04-12 08:23:08 +0000835 enum RendererKind { OR_Copy, OR_Imm, OR_Register, OR_ComplexPattern };
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000836
837protected:
838 RendererKind Kind;
839
840public:
841 OperandRenderer(RendererKind Kind) : Kind(Kind) {}
842 virtual ~OperandRenderer() {}
843
844 RendererKind getKind() const { return Kind; }
845
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000846 virtual void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const = 0;
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000847};
848
849/// A CopyRenderer emits code to copy a single operand from an existing
850/// instruction to the one being built.
851class CopyRenderer : public OperandRenderer {
852protected:
853 /// The matcher for the instruction that this operand is copied from.
854 /// This provides the facility for looking up an a operand by it's name so
855 /// that it can be used as a source for the instruction being built.
856 const InstructionMatcher &Matched;
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000857 /// The name of the operand.
858 const StringRef SymbolicName;
859
860public:
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000861 CopyRenderer(const InstructionMatcher &Matched, StringRef SymbolicName)
862 : OperandRenderer(OR_Copy), Matched(Matched), SymbolicName(SymbolicName) {
863 }
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000864
865 static bool classof(const OperandRenderer *R) {
866 return R->getKind() == OR_Copy;
867 }
868
869 const StringRef getSymbolicName() const { return SymbolicName; }
870
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000871 void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const override {
872 const OperandMatcher &Operand = Matched.getOperand(SymbolicName);
873 StringRef InsnVarName =
874 Rule.getInsnVarName(Operand.getInstructionMatcher());
875 std::string OperandExpr = Operand.getOperandExpr(InsnVarName);
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000876 OS << " MIB.add(" << OperandExpr << "/*" << SymbolicName << "*/);\n";
877 }
878};
879
880/// Adds a specific physical register to the instruction being built.
881/// This is typically useful for WZR/XZR on AArch64.
882class AddRegisterRenderer : public OperandRenderer {
883protected:
884 const Record *RegisterDef;
885
886public:
887 AddRegisterRenderer(const Record *RegisterDef)
888 : OperandRenderer(OR_Register), RegisterDef(RegisterDef) {}
889
890 static bool classof(const OperandRenderer *R) {
891 return R->getKind() == OR_Register;
892 }
893
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000894 void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const override {
Diana Picus8abcbbb2017-05-02 09:40:49 +0000895 OS << " MIB.addReg(" << (RegisterDef->getValue("Namespace")
896 ? RegisterDef->getValueAsString("Namespace")
897 : "")
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000898 << "::" << RegisterDef->getName() << ");\n";
899 }
900};
901
Daniel Sanders0ed28822017-04-12 08:23:08 +0000902/// Adds a specific immediate to the instruction being built.
903class ImmRenderer : public OperandRenderer {
904protected:
905 int64_t Imm;
906
907public:
908 ImmRenderer(int64_t Imm)
909 : OperandRenderer(OR_Imm), Imm(Imm) {}
910
911 static bool classof(const OperandRenderer *R) {
912 return R->getKind() == OR_Imm;
913 }
914
915 void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const override {
916 OS << " MIB.addImm(" << Imm << ");\n";
917 }
918};
919
Daniel Sanders2deea182017-04-22 15:11:04 +0000920/// Adds operands by calling a renderer function supplied by the ComplexPattern
921/// matcher function.
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000922class RenderComplexPatternOperand : public OperandRenderer {
923private:
924 const Record &TheDef;
Daniel Sanders2deea182017-04-22 15:11:04 +0000925 /// The name of the operand.
926 const StringRef SymbolicName;
927 /// The renderer number. This must be unique within a rule since it's used to
928 /// identify a temporary variable to hold the renderer function.
929 unsigned RendererID;
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000930
931 unsigned getNumOperands() const {
932 return TheDef.getValueAsDag("Operands")->getNumArgs();
933 }
934
935public:
Daniel Sanders2deea182017-04-22 15:11:04 +0000936 RenderComplexPatternOperand(const Record &TheDef, StringRef SymbolicName,
937 unsigned RendererID)
938 : OperandRenderer(OR_ComplexPattern), TheDef(TheDef),
939 SymbolicName(SymbolicName), RendererID(RendererID) {}
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000940
941 static bool classof(const OperandRenderer *R) {
942 return R->getKind() == OR_ComplexPattern;
943 }
944
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000945 void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const override {
Daniel Sanders2deea182017-04-22 15:11:04 +0000946 OS << "Renderer" << RendererID << "(MIB);\n";
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000947 }
948};
949
Ahmed Bougacha56ca3a92017-02-04 00:47:10 +0000950/// An action taken when all Matcher predicates succeeded for a parent rule.
951///
952/// Typical actions include:
953/// * Changing the opcode of an instruction.
954/// * Adding an operand to an instruction.
Daniel Sanders43c882c2017-02-01 10:53:10 +0000955class MatchAction {
956public:
957 virtual ~MatchAction() {}
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000958
959 /// Emit the C++ statements to implement the action.
960 ///
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000961 /// \param RecycleVarName If given, it's an instruction to recycle. The
962 /// requirements on the instruction vary from action to
963 /// action.
964 virtual void emitCxxActionStmts(raw_ostream &OS, RuleMatcher &Rule,
965 StringRef RecycleVarName) const = 0;
Daniel Sanders43c882c2017-02-01 10:53:10 +0000966};
967
Ahmed Bougacha9aa4c102017-02-04 00:47:08 +0000968/// Generates a comment describing the matched rule being acted upon.
969class DebugCommentAction : public MatchAction {
970private:
971 const PatternToMatch &P;
972
973public:
974 DebugCommentAction(const PatternToMatch &P) : P(P) {}
975
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000976 void emitCxxActionStmts(raw_ostream &OS, RuleMatcher &Rule,
977 StringRef RecycleVarName) const override {
978 OS << "// " << *P.getSrcPattern() << " => " << *P.getDstPattern() << "\n";
Ahmed Bougacha9aa4c102017-02-04 00:47:08 +0000979 }
980};
981
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000982/// Generates code to build an instruction or mutate an existing instruction
983/// into the desired instruction when this is possible.
984class BuildMIAction : public MatchAction {
Daniel Sanders43c882c2017-02-01 10:53:10 +0000985private:
Daniel Sandersa6e2ceb2017-06-20 12:36:34 +0000986 std::string Name;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000987 const CodeGenInstruction *I;
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000988 const InstructionMatcher &Matched;
989 std::vector<std::unique_ptr<OperandRenderer>> OperandRenderers;
990
991 /// True if the instruction can be built solely by mutating the opcode.
992 bool canMutate() const {
Daniel Sanderse9fdba32017-04-29 17:30:09 +0000993 if (OperandRenderers.size() != Matched.getNumOperands())
994 return false;
995
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000996 for (const auto &Renderer : enumerate(OperandRenderers)) {
Zachary Turner309a0882017-03-13 16:24:10 +0000997 if (const auto *Copy = dyn_cast<CopyRenderer>(&*Renderer.value())) {
Daniel Sanders3016d3c2017-04-22 14:31:28 +0000998 const OperandMatcher &OM = Matched.getOperand(Copy->getSymbolicName());
999 if (&Matched != &OM.getInstructionMatcher() ||
1000 OM.getOperandIndex() != Renderer.index())
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001001 return false;
1002 } else
1003 return false;
1004 }
1005
1006 return true;
1007 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001008
Daniel Sanders43c882c2017-02-01 10:53:10 +00001009public:
Daniel Sandersa6e2ceb2017-06-20 12:36:34 +00001010 BuildMIAction(const StringRef Name, const CodeGenInstruction *I,
1011 const InstructionMatcher &Matched)
1012 : Name(Name), I(I), Matched(Matched) {}
Daniel Sanders43c882c2017-02-01 10:53:10 +00001013
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001014 template <class Kind, class... Args>
1015 Kind &addRenderer(Args&&... args) {
1016 OperandRenderers.emplace_back(
1017 llvm::make_unique<Kind>(std::forward<Args>(args)...));
1018 return *static_cast<Kind *>(OperandRenderers.back().get());
1019 }
1020
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001021 void emitCxxActionStmts(raw_ostream &OS, RuleMatcher &Rule,
1022 StringRef RecycleVarName) const override {
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001023 if (canMutate()) {
Tim Northover4340d642017-03-20 21:58:23 +00001024 OS << " " << RecycleVarName << ".setDesc(TII.get(" << I->Namespace
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001025 << "::" << I->TheDef->getName() << "));\n";
Tim Northover4340d642017-03-20 21:58:23 +00001026
1027 if (!I->ImplicitDefs.empty() || !I->ImplicitUses.empty()) {
1028 OS << " auto MIB = MachineInstrBuilder(MF, &" << RecycleVarName
1029 << ");\n";
1030
1031 for (auto Def : I->ImplicitDefs) {
Diana Picus8abcbbb2017-05-02 09:40:49 +00001032 auto Namespace = Def->getValue("Namespace")
1033 ? Def->getValueAsString("Namespace")
1034 : "";
Tim Northover4340d642017-03-20 21:58:23 +00001035 OS << " MIB.addDef(" << Namespace << "::" << Def->getName()
1036 << ", RegState::Implicit);\n";
1037 }
1038 for (auto Use : I->ImplicitUses) {
Diana Picus8abcbbb2017-05-02 09:40:49 +00001039 auto Namespace = Use->getValue("Namespace")
1040 ? Use->getValueAsString("Namespace")
1041 : "";
Tim Northover4340d642017-03-20 21:58:23 +00001042 OS << " MIB.addUse(" << Namespace << "::" << Use->getName()
1043 << ", RegState::Implicit);\n";
1044 }
1045 }
1046
Daniel Sandersa6e2ceb2017-06-20 12:36:34 +00001047 OS << " MachineInstr &" << Name << " = " << RecycleVarName << ";\n";
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001048 return;
1049 }
1050
1051 // TODO: Simple permutation looks like it could be almost as common as
1052 // mutation due to commutative operations.
1053
1054 OS << "MachineInstrBuilder MIB = BuildMI(*I.getParent(), I, "
1055 "I.getDebugLoc(), TII.get("
1056 << I->Namespace << "::" << I->TheDef->getName() << "));\n";
1057 for (const auto &Renderer : OperandRenderers)
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001058 Renderer->emitCxxRenderStmts(OS, Rule);
Daniel Sandersbee57392017-04-04 13:25:23 +00001059 OS << " for (const auto *FromMI : ";
1060 Rule.emitCxxCapturedInsnList(OS);
1061 OS << ")\n";
1062 OS << " for (const auto &MMO : FromMI->memoperands())\n";
1063 OS << " MIB.addMemOperand(MMO);\n";
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001064 OS << " " << RecycleVarName << ".eraseFromParent();\n";
Daniel Sandersa6e2ceb2017-06-20 12:36:34 +00001065 OS << " MachineInstr &" << Name << " = *MIB;\n";
1066 }
1067};
1068
1069/// Generates code to constrain the operands of an output instruction to the
1070/// register classes specified by the definition of that instruction.
1071class ConstrainOperandsToDefinitionAction : public MatchAction {
1072 std::string Name;
1073
1074public:
1075 ConstrainOperandsToDefinitionAction(const StringRef Name) : Name(Name) {}
1076
1077 void emitCxxActionStmts(raw_ostream &OS, RuleMatcher &Rule,
1078 StringRef RecycleVarName) const override {
1079 OS << " constrainSelectedInstRegOperands(" << Name << ", TII, TRI, RBI);\n";
1080 }
1081};
1082
1083/// Generates code to constrain the specified operand of an output instruction
1084/// to the specified register class.
1085class ConstrainOperandToRegClassAction : public MatchAction {
1086 std::string Name;
1087 unsigned OpIdx;
1088 const CodeGenRegisterClass &RC;
1089
1090public:
1091 ConstrainOperandToRegClassAction(const StringRef Name, unsigned OpIdx,
1092 const CodeGenRegisterClass &RC)
1093 : Name(Name), OpIdx(OpIdx), RC(RC) {}
1094
1095 void emitCxxActionStmts(raw_ostream &OS, RuleMatcher &Rule,
1096 StringRef RecycleVarName) const override {
1097 OS << " constrainOperandRegToRegClass(" << Name << ", " << OpIdx
1098 << ", " << RC.getQualifiedName() << "RegClass, TII, TRI, RBI);\n";
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001099 }
1100};
1101
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001102InstructionMatcher &RuleMatcher::addInstructionMatcher() {
1103 Matchers.emplace_back(new InstructionMatcher());
1104 return *Matchers.back();
1105}
Ahmed Bougacha56ca3a92017-02-04 00:47:10 +00001106
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001107void RuleMatcher::addRequiredFeature(Record *Feature) {
1108 RequiredFeatures.push_back(Feature);
1109}
1110
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001111template <class Kind, class... Args>
1112Kind &RuleMatcher::addAction(Args &&... args) {
1113 Actions.emplace_back(llvm::make_unique<Kind>(std::forward<Args>(args)...));
1114 return *static_cast<Kind *>(Actions.back().get());
1115}
Daniel Sandersdc662ff2017-01-26 11:10:14 +00001116
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001117std::string RuleMatcher::defineInsnVar(raw_ostream &OS,
1118 const InstructionMatcher &Matcher,
1119 StringRef Value) {
1120 std::string InsnVarName = "MI" + llvm::to_string(NextInsnVarID++);
1121 OS << "MachineInstr &" << InsnVarName << " = " << Value << ";\n";
1122 InsnVariableNames[&Matcher] = InsnVarName;
1123 return InsnVarName;
1124}
1125
1126StringRef RuleMatcher::getInsnVarName(const InstructionMatcher &InsnMatcher) const {
1127 const auto &I = InsnVariableNames.find(&InsnMatcher);
1128 if (I != InsnVariableNames.end())
1129 return I->second;
1130 llvm_unreachable("Matched Insn was not captured in a local variable");
1131}
1132
Daniel Sandersbee57392017-04-04 13:25:23 +00001133/// Emit a C++ initializer_list containing references to every matched instruction.
1134void RuleMatcher::emitCxxCapturedInsnList(raw_ostream &OS) {
Daniel Sanders9e4817d2017-04-04 14:27:06 +00001135 SmallVector<StringRef, 2> Names;
Daniel Sandersbee57392017-04-04 13:25:23 +00001136 for (const auto &Pair : InsnVariableNames)
Daniel Sanders9e4817d2017-04-04 14:27:06 +00001137 Names.push_back(Pair.second);
1138 std::sort(Names.begin(), Names.end());
1139
1140 OS << "{";
1141 for (const auto &Name : Names)
1142 OS << "&" << Name << ", ";
Daniel Sandersbee57392017-04-04 13:25:23 +00001143 OS << "}";
1144}
1145
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001146/// Emit C++ statements to check the shape of the match and capture
1147/// instructions into local variables.
1148void RuleMatcher::emitCxxCaptureStmts(raw_ostream &OS, StringRef Expr) {
1149 assert(Matchers.size() == 1 && "Cannot handle multi-root matchers yet");
1150 std::string InsnVarName = defineInsnVar(OS, *Matchers.front(), Expr);
1151 Matchers.front()->emitCxxCaptureStmts(OS, *this, InsnVarName);
1152}
1153
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001154void RuleMatcher::emit(raw_ostream &OS,
Daniel Sanderse9fdba32017-04-29 17:30:09 +00001155 SubtargetFeatureInfoMap SubtargetFeatures) {
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001156 if (Matchers.empty())
1157 llvm_unreachable("Unexpected empty matcher!");
Daniel Sandersdc662ff2017-01-26 11:10:14 +00001158
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001159 // The representation supports rules that require multiple roots such as:
1160 // %ptr(p0) = ...
1161 // %elt0(s32) = G_LOAD %ptr
1162 // %1(p0) = G_ADD %ptr, 4
1163 // %elt1(s32) = G_LOAD p0 %1
1164 // which could be usefully folded into:
1165 // %ptr(p0) = ...
1166 // %elt0(s32), %elt1(s32) = TGT_LOAD_PAIR %ptr
1167 // on some targets but we don't need to make use of that yet.
1168 assert(Matchers.size() == 1 && "Cannot handle multi-root matchers yet");
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001169
1170 OS << "if (";
1171 OS << "[&]() {\n";
1172 if (!RequiredFeatures.empty()) {
1173 OS << " PredicateBitset ExpectedFeatures = {";
1174 StringRef Separator = "";
1175 for (const auto &Predicate : RequiredFeatures) {
1176 const auto &I = SubtargetFeatures.find(Predicate);
1177 assert(I != SubtargetFeatures.end() && "Didn't import predicate?");
1178 OS << Separator << I->second.getEnumBitName();
1179 Separator = ", ";
1180 }
1181 OS << "};\n";
1182 OS << "if ((AvailableFeatures & ExpectedFeatures) != ExpectedFeatures)\n"
1183 << " return false;\n";
1184 }
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001185
1186 emitCxxCaptureStmts(OS, "I");
1187
1188 OS << " if (";
1189 Matchers.front()->emitCxxPredicateExpr(OS, *this,
1190 getInsnVarName(*Matchers.front()));
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001191 OS << ") {\n";
1192
Daniel Sandersbee57392017-04-04 13:25:23 +00001193 // We must also check if it's safe to fold the matched instructions.
1194 if (InsnVariableNames.size() >= 2) {
Galina Kistanova1754fee2017-05-25 01:51:53 +00001195 // Invert the map to create stable ordering (by var names)
1196 SmallVector<StringRef, 2> Names;
Daniel Sandersbee57392017-04-04 13:25:23 +00001197 for (const auto &Pair : InsnVariableNames) {
1198 // Skip the root node since it isn't moving anywhere. Everything else is
1199 // sinking to meet it.
1200 if (Pair.first == Matchers.front().get())
1201 continue;
1202
Galina Kistanova1754fee2017-05-25 01:51:53 +00001203 Names.push_back(Pair.second);
1204 }
1205 std::sort(Names.begin(), Names.end());
1206
1207 for (const auto &Name : Names) {
Daniel Sandersbee57392017-04-04 13:25:23 +00001208 // Reject the difficult cases until we have a more accurate check.
Galina Kistanova1754fee2017-05-25 01:51:53 +00001209 OS << " if (!isObviouslySafeToFold(" << Name
Daniel Sandersbee57392017-04-04 13:25:23 +00001210 << ")) return false;\n";
1211
1212 // FIXME: Emit checks to determine it's _actually_ safe to fold and/or
1213 // account for unsafe cases.
1214 //
1215 // Example:
1216 // MI1--> %0 = ...
1217 // %1 = ... %0
1218 // MI0--> %2 = ... %0
1219 // It's not safe to erase MI1. We currently handle this by not
1220 // erasing %0 (even when it's dead).
1221 //
1222 // Example:
1223 // MI1--> %0 = load volatile @a
1224 // %1 = load volatile @a
1225 // MI0--> %2 = ... %0
1226 // It's not safe to sink %0's def past %1. We currently handle
1227 // this by rejecting all loads.
1228 //
1229 // Example:
1230 // MI1--> %0 = load @a
1231 // %1 = store @a
1232 // MI0--> %2 = ... %0
1233 // It's not safe to sink %0's def past %1. We currently handle
1234 // this by rejecting all loads.
1235 //
1236 // Example:
1237 // G_CONDBR %cond, @BB1
1238 // BB0:
1239 // MI1--> %0 = load @a
1240 // G_BR @BB1
1241 // BB1:
1242 // MI0--> %2 = ... %0
1243 // It's not always safe to sink %0 across control flow. In this
1244 // case it may introduce a memory fault. We currentl handle this
1245 // by rejecting all loads.
1246 }
1247 }
1248
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001249 for (const auto &MA : Actions) {
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001250 MA->emitCxxActionStmts(OS, *this, "I");
Daniel Sandersdc662ff2017-01-26 11:10:14 +00001251 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001252
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001253 OS << " return true;\n";
1254 OS << " }\n";
1255 OS << " return false;\n";
1256 OS << " }()) { return true; }\n\n";
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001257}
Daniel Sanders43c882c2017-02-01 10:53:10 +00001258
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001259bool RuleMatcher::isHigherPriorityThan(const RuleMatcher &B) const {
1260 // Rules involving more match roots have higher priority.
1261 if (Matchers.size() > B.Matchers.size())
1262 return true;
1263 if (Matchers.size() < B.Matchers.size())
Daniel Sanders759ff412017-02-24 13:58:11 +00001264 return false;
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001265
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001266 for (const auto &Matcher : zip(Matchers, B.Matchers)) {
1267 if (std::get<0>(Matcher)->isHigherPriorityThan(*std::get<1>(Matcher)))
1268 return true;
1269 if (std::get<1>(Matcher)->isHigherPriorityThan(*std::get<0>(Matcher)))
1270 return false;
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001271 }
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001272
1273 return false;
Simon Pilgrima7d1da82017-03-15 22:50:47 +00001274}
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001275
Daniel Sanders2deea182017-04-22 15:11:04 +00001276unsigned RuleMatcher::countRendererFns() const {
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001277 return std::accumulate(
1278 Matchers.begin(), Matchers.end(), 0,
1279 [](unsigned A, const std::unique_ptr<InstructionMatcher> &Matcher) {
Daniel Sanders2deea182017-04-22 15:11:04 +00001280 return A + Matcher->countRendererFns();
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001281 });
1282}
1283
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001284//===- GlobalISelEmitter class --------------------------------------------===//
1285
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001286class GlobalISelEmitter {
1287public:
1288 explicit GlobalISelEmitter(RecordKeeper &RK);
1289 void run(raw_ostream &OS);
1290
1291private:
1292 const RecordKeeper &RK;
1293 const CodeGenDAGPatterns CGP;
1294 const CodeGenTarget &Target;
1295
1296 /// Keep track of the equivalence between SDNodes and Instruction.
1297 /// This is defined using 'GINodeEquiv' in the target description.
1298 DenseMap<Record *, const CodeGenInstruction *> NodeEquivs;
1299
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001300 /// Keep track of the equivalence between ComplexPattern's and
1301 /// GIComplexOperandMatcher. Map entries are specified by subclassing
1302 /// GIComplexPatternEquiv.
1303 DenseMap<const Record *, const Record *> ComplexPatternEquivs;
1304
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001305 // Map of predicates to their subtarget features.
Daniel Sanderse9fdba32017-04-29 17:30:09 +00001306 SubtargetFeatureInfoMap SubtargetFeatures;
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001307
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001308 void gatherNodeEquivs();
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001309 const CodeGenInstruction *findNodeEquiv(Record *N) const;
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001310
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001311 Error importRulePredicates(RuleMatcher &M, ArrayRef<Init *> Predicates);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001312 Expected<InstructionMatcher &>
Daniel Sandersc270c502017-03-30 09:36:33 +00001313 createAndImportSelDAGMatcher(InstructionMatcher &InsnMatcher,
1314 const TreePatternNode *Src) const;
1315 Error importChildMatcher(InstructionMatcher &InsnMatcher,
Daniel Sanders452c8ae2017-05-23 19:33:16 +00001316 const TreePatternNode *SrcChild, unsigned OpIdx,
Daniel Sandersc270c502017-03-30 09:36:33 +00001317 unsigned &TempOpIdx) const;
1318 Expected<BuildMIAction &> createAndImportInstructionRenderer(
1319 RuleMatcher &M, const TreePatternNode *Dst,
1320 const InstructionMatcher &InsnMatcher) const;
1321 Error importExplicitUseRenderer(BuildMIAction &DstMIBuilder,
1322 TreePatternNode *DstChild,
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001323 const InstructionMatcher &InsnMatcher) const;
Diana Picus382602f2017-05-17 08:57:28 +00001324 Error importDefaultOperandRenderers(BuildMIAction &DstMIBuilder,
1325 DagInit *DefaultOps) const;
Daniel Sandersc270c502017-03-30 09:36:33 +00001326 Error
Daniel Sandersffc7d582017-03-29 15:37:18 +00001327 importImplicitDefRenderers(BuildMIAction &DstMIBuilder,
1328 const std::vector<Record *> &ImplicitDefs) const;
1329
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001330 /// Analyze pattern \p P, returning a matcher for it if possible.
1331 /// Otherwise, return an Error explaining why we don't support it.
1332 Expected<RuleMatcher> runOnPattern(const PatternToMatch &P);
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001333
1334 void declareSubtargetFeature(Record *Predicate);
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001335};
1336
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001337void GlobalISelEmitter::gatherNodeEquivs() {
1338 assert(NodeEquivs.empty());
1339 for (Record *Equiv : RK.getAllDerivedDefinitions("GINodeEquiv"))
1340 NodeEquivs[Equiv->getValueAsDef("Node")] =
1341 &Target.getInstruction(Equiv->getValueAsDef("I"));
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001342
1343 assert(ComplexPatternEquivs.empty());
1344 for (Record *Equiv : RK.getAllDerivedDefinitions("GIComplexPatternEquiv")) {
1345 Record *SelDAGEquiv = Equiv->getValueAsDef("SelDAGEquivalent");
1346 if (!SelDAGEquiv)
1347 continue;
1348 ComplexPatternEquivs[SelDAGEquiv] = Equiv;
1349 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001350}
1351
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001352const CodeGenInstruction *GlobalISelEmitter::findNodeEquiv(Record *N) const {
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001353 return NodeEquivs.lookup(N);
1354}
1355
1356GlobalISelEmitter::GlobalISelEmitter(RecordKeeper &RK)
1357 : RK(RK), CGP(RK), Target(CGP.getTargetInfo()) {}
1358
1359//===- Emitter ------------------------------------------------------------===//
1360
Daniel Sandersc270c502017-03-30 09:36:33 +00001361Error
Daniel Sandersffc7d582017-03-29 15:37:18 +00001362GlobalISelEmitter::importRulePredicates(RuleMatcher &M,
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001363 ArrayRef<Init *> Predicates) {
1364 for (const Init *Predicate : Predicates) {
1365 const DefInit *PredicateDef = static_cast<const DefInit *>(Predicate);
1366 declareSubtargetFeature(PredicateDef->getDef());
1367 M.addRequiredFeature(PredicateDef->getDef());
1368 }
1369
Daniel Sandersc270c502017-03-30 09:36:33 +00001370 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001371}
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001372
Daniel Sandersc270c502017-03-30 09:36:33 +00001373Expected<InstructionMatcher &> GlobalISelEmitter::createAndImportSelDAGMatcher(
1374 InstructionMatcher &InsnMatcher, const TreePatternNode *Src) const {
Daniel Sandersffc7d582017-03-29 15:37:18 +00001375 // Start with the defined operands (i.e., the results of the root operator).
1376 if (Src->getExtTypes().size() > 1)
1377 return failedImport("Src pattern has multiple results");
1378
Daniel Sanders452c8ae2017-05-23 19:33:16 +00001379 if (Src->isLeaf()) {
1380 Init *SrcInit = Src->getLeafValue();
Daniel Sanders3334cc02017-05-23 20:02:48 +00001381 if (isa<IntInit>(SrcInit)) {
Daniel Sanders452c8ae2017-05-23 19:33:16 +00001382 InsnMatcher.addPredicate<InstructionOpcodeMatcher>(
1383 &Target.getInstruction(RK.getDef("G_CONSTANT")));
1384 } else
1385 return failedImport("Unable to deduce gMIR opcode to handle Src (which is a leaf)");
1386 } else {
1387 auto SrcGIOrNull = findNodeEquiv(Src->getOperator());
1388 if (!SrcGIOrNull)
1389 return failedImport("Pattern operator lacks an equivalent Instruction" +
1390 explainOperator(Src->getOperator()));
1391 auto &SrcGI = *SrcGIOrNull;
Daniel Sandersffc7d582017-03-29 15:37:18 +00001392
Daniel Sanders452c8ae2017-05-23 19:33:16 +00001393 // The operators look good: match the opcode
1394 InsnMatcher.addPredicate<InstructionOpcodeMatcher>(&SrcGI);
1395 }
Daniel Sandersffc7d582017-03-29 15:37:18 +00001396
1397 unsigned OpIdx = 0;
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001398 unsigned TempOpIdx = 0;
Daniel Sandersffc7d582017-03-29 15:37:18 +00001399 for (const EEVT::TypeSet &Ty : Src->getExtTypes()) {
1400 auto OpTyOrNone = MVTToLLT(Ty.getConcrete());
1401
1402 if (!OpTyOrNone)
1403 return failedImport(
1404 "Result of Src pattern operator has an unsupported type");
1405
1406 // Results don't have a name unless they are the root node. The caller will
1407 // set the name if appropriate.
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001408 OperandMatcher &OM = InsnMatcher.addOperand(OpIdx++, "", TempOpIdx);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001409 OM.addPredicate<LLTOperandMatcher>(*OpTyOrNone);
1410 }
1411
Daniel Sanders452c8ae2017-05-23 19:33:16 +00001412 if (Src->isLeaf()) {
1413 Init *SrcInit = Src->getLeafValue();
1414 if (IntInit *SrcIntInit = dyn_cast<IntInit>(SrcInit)) {
1415 OperandMatcher &OM = InsnMatcher.addOperand(OpIdx++, "", TempOpIdx);
1416 OM.addPredicate<LiteralIntOperandMatcher>(SrcIntInit->getValue());
1417 } else
1418 return failedImport("Unable to deduce gMIR opcode to handle Src (which is a leaf)");
1419 } else {
1420 // Match the used operands (i.e. the children of the operator).
1421 for (unsigned i = 0, e = Src->getNumChildren(); i != e; ++i) {
1422 if (auto Error = importChildMatcher(InsnMatcher, Src->getChild(i),
1423 OpIdx++, TempOpIdx))
1424 return std::move(Error);
1425 }
Daniel Sandersffc7d582017-03-29 15:37:18 +00001426 }
1427
1428 return InsnMatcher;
1429}
1430
Daniel Sandersc270c502017-03-30 09:36:33 +00001431Error GlobalISelEmitter::importChildMatcher(InstructionMatcher &InsnMatcher,
Daniel Sanders452c8ae2017-05-23 19:33:16 +00001432 const TreePatternNode *SrcChild,
Daniel Sandersc270c502017-03-30 09:36:33 +00001433 unsigned OpIdx,
1434 unsigned &TempOpIdx) const {
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001435 OperandMatcher &OM =
1436 InsnMatcher.addOperand(OpIdx, SrcChild->getName(), TempOpIdx);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001437
1438 if (SrcChild->hasAnyPredicate())
Daniel Sandersd0656a32017-04-13 09:45:37 +00001439 return failedImport("Src pattern child has predicate (" +
1440 explainPredicates(SrcChild) + ")");
Daniel Sandersffc7d582017-03-29 15:37:18 +00001441
1442 ArrayRef<EEVT::TypeSet> ChildTypes = SrcChild->getExtTypes();
1443 if (ChildTypes.size() != 1)
1444 return failedImport("Src pattern child has multiple results");
1445
1446 // Check MBB's before the type check since they are not a known type.
1447 if (!SrcChild->isLeaf()) {
1448 if (SrcChild->getOperator()->isSubClassOf("SDNode")) {
1449 auto &ChildSDNI = CGP.getSDNodeInfo(SrcChild->getOperator());
1450 if (ChildSDNI.getSDClassName() == "BasicBlockSDNode") {
1451 OM.addPredicate<MBBOperandMatcher>();
Daniel Sandersc270c502017-03-30 09:36:33 +00001452 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001453 }
1454 }
Daniel Sandersffc7d582017-03-29 15:37:18 +00001455 }
1456
1457 auto OpTyOrNone = MVTToLLT(ChildTypes.front().getConcrete());
1458 if (!OpTyOrNone)
Daniel Sandersc244ff62017-05-22 10:14:33 +00001459 return failedImport("Src operand has an unsupported type");
Daniel Sandersffc7d582017-03-29 15:37:18 +00001460 OM.addPredicate<LLTOperandMatcher>(*OpTyOrNone);
1461
Daniel Sandersbee57392017-04-04 13:25:23 +00001462 // Check for nested instructions.
1463 if (!SrcChild->isLeaf()) {
1464 // Map the node to a gMIR instruction.
1465 InstructionOperandMatcher &InsnOperand =
1466 OM.addPredicate<InstructionOperandMatcher>();
1467 auto InsnMatcherOrError =
1468 createAndImportSelDAGMatcher(InsnOperand.getInsnMatcher(), SrcChild);
1469 if (auto Error = InsnMatcherOrError.takeError())
1470 return Error;
1471
1472 return Error::success();
1473 }
1474
Daniel Sandersffc7d582017-03-29 15:37:18 +00001475 // Check for constant immediates.
1476 if (auto *ChildInt = dyn_cast<IntInit>(SrcChild->getLeafValue())) {
Daniel Sanders452c8ae2017-05-23 19:33:16 +00001477 OM.addPredicate<ConstantIntOperandMatcher>(ChildInt->getValue());
Daniel Sandersc270c502017-03-30 09:36:33 +00001478 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001479 }
1480
1481 // Check for def's like register classes or ComplexPattern's.
1482 if (auto *ChildDefInit = dyn_cast<DefInit>(SrcChild->getLeafValue())) {
1483 auto *ChildRec = ChildDefInit->getDef();
1484
1485 // Check for register classes.
Daniel Sandersa6e2ceb2017-06-20 12:36:34 +00001486 if (ChildRec->isSubClassOf("RegisterClass") ||
1487 ChildRec->isSubClassOf("RegisterOperand")) {
Daniel Sandersffc7d582017-03-29 15:37:18 +00001488 OM.addPredicate<RegisterBankOperandMatcher>(
Daniel Sandersa6e2ceb2017-06-20 12:36:34 +00001489 Target.getRegisterClass(getInitValueAsRegClass(ChildDefInit)));
Daniel Sanders658541f2017-04-22 15:53:21 +00001490 return Error::success();
1491 }
1492
Daniel Sandersffc7d582017-03-29 15:37:18 +00001493 // Check for ComplexPattern's.
1494 if (ChildRec->isSubClassOf("ComplexPattern")) {
1495 const auto &ComplexPattern = ComplexPatternEquivs.find(ChildRec);
1496 if (ComplexPattern == ComplexPatternEquivs.end())
Daniel Sandersd0656a32017-04-13 09:45:37 +00001497 return failedImport("SelectionDAG ComplexPattern (" +
1498 ChildRec->getName() + ") not mapped to GlobalISel");
Daniel Sandersffc7d582017-03-29 15:37:18 +00001499
Daniel Sanders2deea182017-04-22 15:11:04 +00001500 OM.addPredicate<ComplexPatternOperandMatcher>(OM,
1501 *ComplexPattern->second);
1502 TempOpIdx++;
Daniel Sandersc270c502017-03-30 09:36:33 +00001503 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001504 }
1505
Daniel Sandersd0656a32017-04-13 09:45:37 +00001506 if (ChildRec->isSubClassOf("ImmLeaf")) {
1507 return failedImport(
1508 "Src pattern child def is an unsupported tablegen class (ImmLeaf)");
1509 }
1510
Daniel Sandersffc7d582017-03-29 15:37:18 +00001511 return failedImport(
1512 "Src pattern child def is an unsupported tablegen class");
1513 }
1514
1515 return failedImport("Src pattern child is an unsupported kind");
1516}
1517
Daniel Sandersc270c502017-03-30 09:36:33 +00001518Error GlobalISelEmitter::importExplicitUseRenderer(
Daniel Sandersffc7d582017-03-29 15:37:18 +00001519 BuildMIAction &DstMIBuilder, TreePatternNode *DstChild,
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001520 const InstructionMatcher &InsnMatcher) const {
Daniel Sandersffc7d582017-03-29 15:37:18 +00001521 // The only non-leaf child we accept is 'bb': it's an operator because
1522 // BasicBlockSDNode isn't inline, but in MI it's just another operand.
1523 if (!DstChild->isLeaf()) {
1524 if (DstChild->getOperator()->isSubClassOf("SDNode")) {
1525 auto &ChildSDNI = CGP.getSDNodeInfo(DstChild->getOperator());
1526 if (ChildSDNI.getSDClassName() == "BasicBlockSDNode") {
1527 DstMIBuilder.addRenderer<CopyRenderer>(InsnMatcher,
1528 DstChild->getName());
Daniel Sandersc270c502017-03-30 09:36:33 +00001529 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001530 }
1531 }
1532 return failedImport("Dst pattern child isn't a leaf node or an MBB");
1533 }
1534
1535 // Otherwise, we're looking for a bog-standard RegisterClass operand.
1536 if (DstChild->hasAnyPredicate())
Daniel Sandersd0656a32017-04-13 09:45:37 +00001537 return failedImport("Dst pattern child has predicate (" +
1538 explainPredicates(DstChild) + ")");
Daniel Sandersffc7d582017-03-29 15:37:18 +00001539
1540 if (auto *ChildDefInit = dyn_cast<DefInit>(DstChild->getLeafValue())) {
1541 auto *ChildRec = ChildDefInit->getDef();
1542
1543 ArrayRef<EEVT::TypeSet> ChildTypes = DstChild->getExtTypes();
1544 if (ChildTypes.size() != 1)
1545 return failedImport("Dst pattern child has multiple results");
1546
1547 auto OpTyOrNone = MVTToLLT(ChildTypes.front().getConcrete());
1548 if (!OpTyOrNone)
1549 return failedImport("Dst operand has an unsupported type");
1550
1551 if (ChildRec->isSubClassOf("Register")) {
1552 DstMIBuilder.addRenderer<AddRegisterRenderer>(ChildRec);
Daniel Sandersc270c502017-03-30 09:36:33 +00001553 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001554 }
1555
Daniel Sanders658541f2017-04-22 15:53:21 +00001556 if (ChildRec->isSubClassOf("RegisterClass") ||
1557 ChildRec->isSubClassOf("RegisterOperand")) {
Daniel Sandersffc7d582017-03-29 15:37:18 +00001558 DstMIBuilder.addRenderer<CopyRenderer>(InsnMatcher, DstChild->getName());
Daniel Sandersc270c502017-03-30 09:36:33 +00001559 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001560 }
1561
1562 if (ChildRec->isSubClassOf("ComplexPattern")) {
1563 const auto &ComplexPattern = ComplexPatternEquivs.find(ChildRec);
1564 if (ComplexPattern == ComplexPatternEquivs.end())
1565 return failedImport(
1566 "SelectionDAG ComplexPattern not mapped to GlobalISel");
1567
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001568 const OperandMatcher &OM = InsnMatcher.getOperand(DstChild->getName());
Daniel Sandersffc7d582017-03-29 15:37:18 +00001569 DstMIBuilder.addRenderer<RenderComplexPatternOperand>(
Daniel Sanders2deea182017-04-22 15:11:04 +00001570 *ComplexPattern->second, DstChild->getName(),
1571 OM.getAllocatedTemporariesBaseID());
Daniel Sandersc270c502017-03-30 09:36:33 +00001572 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001573 }
1574
Daniel Sandersd0656a32017-04-13 09:45:37 +00001575 if (ChildRec->isSubClassOf("SDNodeXForm"))
1576 return failedImport("Dst pattern child def is an unsupported tablegen "
1577 "class (SDNodeXForm)");
1578
Daniel Sandersffc7d582017-03-29 15:37:18 +00001579 return failedImport(
1580 "Dst pattern child def is an unsupported tablegen class");
1581 }
1582
1583 return failedImport("Dst pattern child is an unsupported kind");
1584}
1585
Daniel Sandersc270c502017-03-30 09:36:33 +00001586Expected<BuildMIAction &> GlobalISelEmitter::createAndImportInstructionRenderer(
Daniel Sandersffc7d582017-03-29 15:37:18 +00001587 RuleMatcher &M, const TreePatternNode *Dst,
1588 const InstructionMatcher &InsnMatcher) const {
1589 Record *DstOp = Dst->getOperator();
Daniel Sandersd0656a32017-04-13 09:45:37 +00001590 if (!DstOp->isSubClassOf("Instruction")) {
1591 if (DstOp->isSubClassOf("ValueType"))
1592 return failedImport(
1593 "Pattern operator isn't an instruction (it's a ValueType)");
Daniel Sandersffc7d582017-03-29 15:37:18 +00001594 return failedImport("Pattern operator isn't an instruction");
Daniel Sandersd0656a32017-04-13 09:45:37 +00001595 }
Daniel Sandersa6e2ceb2017-06-20 12:36:34 +00001596 CodeGenInstruction *DstI = &Target.getInstruction(DstOp);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001597
Daniel Sandersa6e2ceb2017-06-20 12:36:34 +00001598 unsigned DstINumUses = DstI->Operands.size() - DstI->Operands.NumDefs;
1599 unsigned ExpectedDstINumUses = Dst->getNumChildren();
1600
1601 // COPY_TO_REGCLASS is just a copy with a ConstrainOperandToRegClassAction
1602 // attached.
1603 if (DstI->TheDef->getName() == "COPY_TO_REGCLASS") {
1604 DstI = &Target.getInstruction(RK.getDef("COPY"));
1605 DstINumUses--; // Ignore the class constraint.
1606 ExpectedDstINumUses--;
1607 }
1608
1609 auto &DstMIBuilder = M.addAction<BuildMIAction>("NewI", DstI, InsnMatcher);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001610
1611 // Render the explicit defs.
Daniel Sandersa6e2ceb2017-06-20 12:36:34 +00001612 for (unsigned I = 0; I < DstI->Operands.NumDefs; ++I) {
1613 const CGIOperandList::OperandInfo &DstIOperand = DstI->Operands[I];
Daniel Sandersffc7d582017-03-29 15:37:18 +00001614 DstMIBuilder.addRenderer<CopyRenderer>(InsnMatcher, DstIOperand.Name);
1615 }
1616
1617 // Render the explicit uses.
Daniel Sanders0ed28822017-04-12 08:23:08 +00001618 unsigned Child = 0;
Diana Picus382602f2017-05-17 08:57:28 +00001619 unsigned NumDefaultOps = 0;
Daniel Sanders0ed28822017-04-12 08:23:08 +00001620 for (unsigned I = 0; I != DstINumUses; ++I) {
Daniel Sandersa6e2ceb2017-06-20 12:36:34 +00001621 const CGIOperandList::OperandInfo &DstIOperand =
1622 DstI->Operands[DstI->Operands.NumDefs + I];
Daniel Sanders0ed28822017-04-12 08:23:08 +00001623
Diana Picus382602f2017-05-17 08:57:28 +00001624 // If the operand has default values, introduce them now.
1625 // FIXME: Until we have a decent test case that dictates we should do
1626 // otherwise, we're going to assume that operands with default values cannot
1627 // be specified in the patterns. Therefore, adding them will not cause us to
1628 // end up with too many rendered operands.
1629 if (DstIOperand.Rec->isSubClassOf("OperandWithDefaultOps")) {
Daniel Sanders0ed28822017-04-12 08:23:08 +00001630 DagInit *DefaultOps = DstIOperand.Rec->getValueAsDag("DefaultOps");
Diana Picus382602f2017-05-17 08:57:28 +00001631 if (auto Error = importDefaultOperandRenderers(DstMIBuilder, DefaultOps))
1632 return std::move(Error);
1633 ++NumDefaultOps;
Daniel Sanders0ed28822017-04-12 08:23:08 +00001634 continue;
1635 }
1636
1637 if (auto Error = importExplicitUseRenderer(
1638 DstMIBuilder, Dst->getChild(Child), InsnMatcher))
Daniel Sandersffc7d582017-03-29 15:37:18 +00001639 return std::move(Error);
Daniel Sanders0ed28822017-04-12 08:23:08 +00001640 ++Child;
Daniel Sandersffc7d582017-03-29 15:37:18 +00001641 }
1642
Daniel Sandersa6e2ceb2017-06-20 12:36:34 +00001643 if (NumDefaultOps + ExpectedDstINumUses != DstINumUses)
Diana Picuseb2057c2017-05-17 09:25:08 +00001644 return failedImport("Expected " + llvm::to_string(DstINumUses) +
Diana Picus382602f2017-05-17 08:57:28 +00001645 " used operands but found " +
Daniel Sandersa6e2ceb2017-06-20 12:36:34 +00001646 llvm::to_string(ExpectedDstINumUses) +
Diana Picuseb2057c2017-05-17 09:25:08 +00001647 " explicit ones and " + llvm::to_string(NumDefaultOps) +
Diana Picus382602f2017-05-17 08:57:28 +00001648 " default ones");
1649
Daniel Sandersffc7d582017-03-29 15:37:18 +00001650 return DstMIBuilder;
1651}
1652
Diana Picus382602f2017-05-17 08:57:28 +00001653Error GlobalISelEmitter::importDefaultOperandRenderers(
1654 BuildMIAction &DstMIBuilder, DagInit *DefaultOps) const {
Craig Topper481ff702017-05-29 21:49:34 +00001655 for (const auto *DefaultOp : DefaultOps->getArgs()) {
Diana Picus382602f2017-05-17 08:57:28 +00001656 // Look through ValueType operators.
1657 if (const DagInit *DefaultDagOp = dyn_cast<DagInit>(DefaultOp)) {
1658 if (const DefInit *DefaultDagOperator =
1659 dyn_cast<DefInit>(DefaultDagOp->getOperator())) {
1660 if (DefaultDagOperator->getDef()->isSubClassOf("ValueType"))
1661 DefaultOp = DefaultDagOp->getArg(0);
1662 }
1663 }
1664
1665 if (const DefInit *DefaultDefOp = dyn_cast<DefInit>(DefaultOp)) {
1666 DstMIBuilder.addRenderer<AddRegisterRenderer>(DefaultDefOp->getDef());
1667 continue;
1668 }
1669
1670 if (const IntInit *DefaultIntOp = dyn_cast<IntInit>(DefaultOp)) {
1671 DstMIBuilder.addRenderer<ImmRenderer>(DefaultIntOp->getValue());
1672 continue;
1673 }
1674
1675 return failedImport("Could not add default op");
1676 }
1677
1678 return Error::success();
1679}
1680
Daniel Sandersc270c502017-03-30 09:36:33 +00001681Error GlobalISelEmitter::importImplicitDefRenderers(
Daniel Sandersffc7d582017-03-29 15:37:18 +00001682 BuildMIAction &DstMIBuilder,
1683 const std::vector<Record *> &ImplicitDefs) const {
1684 if (!ImplicitDefs.empty())
1685 return failedImport("Pattern defines a physical register");
Daniel Sandersc270c502017-03-30 09:36:33 +00001686 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001687}
1688
1689Expected<RuleMatcher> GlobalISelEmitter::runOnPattern(const PatternToMatch &P) {
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001690 // Keep track of the matchers and actions to emit.
Ahmed Bougacha9aa4c102017-02-04 00:47:08 +00001691 RuleMatcher M;
1692 M.addAction<DebugCommentAction>(P);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001693
Daniel Sandersc270c502017-03-30 09:36:33 +00001694 if (auto Error = importRulePredicates(M, P.getPredicates()->getValues()))
Daniel Sandersffc7d582017-03-29 15:37:18 +00001695 return std::move(Error);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001696
1697 // Next, analyze the pattern operators.
1698 TreePatternNode *Src = P.getSrcPattern();
1699 TreePatternNode *Dst = P.getDstPattern();
1700
1701 // If the root of either pattern isn't a simple operator, ignore it.
Daniel Sandersd0656a32017-04-13 09:45:37 +00001702 if (auto Err = isTrivialOperatorNode(Dst))
1703 return failedImport("Dst pattern root isn't a trivial operator (" +
1704 toString(std::move(Err)) + ")");
1705 if (auto Err = isTrivialOperatorNode(Src))
1706 return failedImport("Src pattern root isn't a trivial operator (" +
1707 toString(std::move(Err)) + ")");
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001708
Daniel Sanders452c8ae2017-05-23 19:33:16 +00001709 if (Dst->isLeaf())
1710 return failedImport("Dst pattern root isn't a known leaf");
1711
Daniel Sandersbee57392017-04-04 13:25:23 +00001712 // Start with the defined operands (i.e., the results of the root operator).
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001713 Record *DstOp = Dst->getOperator();
1714 if (!DstOp->isSubClassOf("Instruction"))
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001715 return failedImport("Pattern operator isn't an instruction");
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001716
1717 auto &DstI = Target.getInstruction(DstOp);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001718 if (DstI.Operands.NumDefs != Src->getExtTypes().size())
Daniel Sandersd0656a32017-04-13 09:45:37 +00001719 return failedImport("Src pattern results and dst MI defs are different (" +
1720 to_string(Src->getExtTypes().size()) + " def(s) vs " +
1721 to_string(DstI.Operands.NumDefs) + " def(s))");
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001722
Daniel Sandersffc7d582017-03-29 15:37:18 +00001723 InstructionMatcher &InsnMatcherTemp = M.addInstructionMatcher();
Daniel Sandersc270c502017-03-30 09:36:33 +00001724 auto InsnMatcherOrError = createAndImportSelDAGMatcher(InsnMatcherTemp, Src);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001725 if (auto Error = InsnMatcherOrError.takeError())
1726 return std::move(Error);
1727 InstructionMatcher &InsnMatcher = InsnMatcherOrError.get();
1728
1729 // The root of the match also has constraints on the register bank so that it
1730 // matches the result instruction.
1731 unsigned OpIdx = 0;
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001732 for (const EEVT::TypeSet &Ty : Src->getExtTypes()) {
Daniel Sandersffc7d582017-03-29 15:37:18 +00001733 (void)Ty;
1734
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001735 const auto &DstIOperand = DstI.Operands[OpIdx];
1736 Record *DstIOpRec = DstIOperand.Rec;
Daniel Sandersa6e2ceb2017-06-20 12:36:34 +00001737 if (DstI.TheDef->getName() == "COPY_TO_REGCLASS") {
1738 DstIOpRec = getInitValueAsRegClass(Dst->getChild(1)->getLeafValue());
1739
1740 if (DstIOpRec == nullptr)
1741 return failedImport(
1742 "COPY_TO_REGCLASS operand #1 isn't a register class");
1743 } else if (DstIOpRec->isSubClassOf("RegisterOperand"))
Daniel Sanders658541f2017-04-22 15:53:21 +00001744 DstIOpRec = DstIOpRec->getValueAsDef("RegClass");
Daniel Sandersa6e2ceb2017-06-20 12:36:34 +00001745 else if (!DstIOpRec->isSubClassOf("RegisterClass"))
1746 return failedImport("Dst MI def isn't a register class" + to_string(*Dst));
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001747
Daniel Sandersffc7d582017-03-29 15:37:18 +00001748 OperandMatcher &OM = InsnMatcher.getOperand(OpIdx);
1749 OM.setSymbolicName(DstIOperand.Name);
Daniel Sandersdc662ff2017-01-26 11:10:14 +00001750 OM.addPredicate<RegisterBankOperandMatcher>(
1751 Target.getRegisterClass(DstIOpRec));
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001752 ++OpIdx;
1753 }
1754
Daniel Sandersc270c502017-03-30 09:36:33 +00001755 auto DstMIBuilderOrError =
1756 createAndImportInstructionRenderer(M, Dst, InsnMatcher);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001757 if (auto Error = DstMIBuilderOrError.takeError())
1758 return std::move(Error);
1759 BuildMIAction &DstMIBuilder = DstMIBuilderOrError.get();
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001760
Daniel Sandersffc7d582017-03-29 15:37:18 +00001761 // Render the implicit defs.
1762 // These are only added to the root of the result.
Daniel Sandersc270c502017-03-30 09:36:33 +00001763 if (auto Error = importImplicitDefRenderers(DstMIBuilder, P.getDstRegs()))
Daniel Sandersffc7d582017-03-29 15:37:18 +00001764 return std::move(Error);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001765
Daniel Sandersa6e2ceb2017-06-20 12:36:34 +00001766 // Constrain the registers to classes. This is normally derived from the
1767 // emitted instruction but a few instructions require special handling.
1768 if (DstI.TheDef->getName() == "COPY_TO_REGCLASS") {
1769 // COPY_TO_REGCLASS does not provide operand constraints itself but the
1770 // result is constrained to the class given by the second child.
1771 Record *DstIOpRec =
1772 getInitValueAsRegClass(Dst->getChild(1)->getLeafValue());
1773
1774 if (DstIOpRec == nullptr)
1775 return failedImport("COPY_TO_REGCLASS operand #1 isn't a register class");
1776
1777 M.addAction<ConstrainOperandToRegClassAction>(
1778 "NewI", 0, Target.getRegisterClass(DstIOpRec));
1779 } else
1780 M.addAction<ConstrainOperandsToDefinitionAction>("NewI");
1781
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001782 // We're done with this pattern! It's eligible for GISel emission; return it.
Daniel Sandersb41ce2b2017-02-20 14:31:27 +00001783 ++NumPatternImported;
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001784 return std::move(M);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001785}
1786
1787void GlobalISelEmitter::run(raw_ostream &OS) {
1788 // Track the GINodeEquiv definitions.
1789 gatherNodeEquivs();
1790
1791 emitSourceFileHeader(("Global Instruction Selector for the " +
1792 Target.getName() + " target").str(), OS);
Daniel Sandersb41ce2b2017-02-20 14:31:27 +00001793 std::vector<RuleMatcher> Rules;
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001794 // Look through the SelectionDAG patterns we found, possibly emitting some.
1795 for (const PatternToMatch &Pat : CGP.ptms()) {
1796 ++NumPatternTotal;
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001797 auto MatcherOrErr = runOnPattern(Pat);
1798
1799 // The pattern analysis can fail, indicating an unsupported pattern.
1800 // Report that if we've been asked to do so.
1801 if (auto Err = MatcherOrErr.takeError()) {
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001802 if (WarnOnSkippedPatterns) {
1803 PrintWarning(Pat.getSrcRecord()->getLoc(),
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001804 "Skipped pattern: " + toString(std::move(Err)));
1805 } else {
1806 consumeError(std::move(Err));
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001807 }
Daniel Sandersb41ce2b2017-02-20 14:31:27 +00001808 ++NumPatternImportsSkipped;
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001809 continue;
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001810 }
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001811
Daniel Sandersb41ce2b2017-02-20 14:31:27 +00001812 Rules.push_back(std::move(MatcherOrErr.get()));
1813 }
1814
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001815 std::stable_sort(Rules.begin(), Rules.end(),
1816 [&](const RuleMatcher &A, const RuleMatcher &B) {
Daniel Sanders759ff412017-02-24 13:58:11 +00001817 if (A.isHigherPriorityThan(B)) {
1818 assert(!B.isHigherPriorityThan(A) && "Cannot be more important "
1819 "and less important at "
1820 "the same time");
1821 return true;
1822 }
1823 return false;
1824 });
1825
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001826 unsigned MaxTemporaries = 0;
1827 for (const auto &Rule : Rules)
Daniel Sanders2deea182017-04-22 15:11:04 +00001828 MaxTemporaries = std::max(MaxTemporaries, Rule.countRendererFns());
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001829
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001830 OS << "#ifdef GET_GLOBALISEL_PREDICATE_BITSET\n"
1831 << "const unsigned MAX_SUBTARGET_PREDICATES = " << SubtargetFeatures.size()
1832 << ";\n"
1833 << "using PredicateBitset = "
1834 "llvm::PredicateBitsetImpl<MAX_SUBTARGET_PREDICATES>;\n"
1835 << "#endif // ifdef GET_GLOBALISEL_PREDICATE_BITSET\n\n";
1836
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001837 OS << "#ifdef GET_GLOBALISEL_TEMPORARIES_DECL\n";
1838 for (unsigned I = 0; I < MaxTemporaries; ++I)
Daniel Sanders2deea182017-04-22 15:11:04 +00001839 OS << " mutable ComplexRendererFn Renderer" << I << ";\n";
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001840 OS << "#endif // ifdef GET_GLOBALISEL_TEMPORARIES_DECL\n\n";
1841
1842 OS << "#ifdef GET_GLOBALISEL_TEMPORARIES_INIT\n";
1843 for (unsigned I = 0; I < MaxTemporaries; ++I)
Daniel Sanders2deea182017-04-22 15:11:04 +00001844 OS << ", Renderer" << I << "(nullptr)\n";
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001845 OS << "#endif // ifdef GET_GLOBALISEL_TEMPORARIES_INIT\n\n";
1846
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001847 OS << "#ifdef GET_GLOBALISEL_IMPL\n";
1848 SubtargetFeatureInfo::emitSubtargetFeatureBitEnumeration(SubtargetFeatures,
1849 OS);
Daniel Sanderse9fdba32017-04-29 17:30:09 +00001850
1851 // Separate subtarget features by how often they must be recomputed.
1852 SubtargetFeatureInfoMap ModuleFeatures;
1853 std::copy_if(SubtargetFeatures.begin(), SubtargetFeatures.end(),
1854 std::inserter(ModuleFeatures, ModuleFeatures.end()),
1855 [](const SubtargetFeatureInfoMap::value_type &X) {
1856 return !X.second.mustRecomputePerFunction();
1857 });
1858 SubtargetFeatureInfoMap FunctionFeatures;
1859 std::copy_if(SubtargetFeatures.begin(), SubtargetFeatures.end(),
1860 std::inserter(FunctionFeatures, FunctionFeatures.end()),
1861 [](const SubtargetFeatureInfoMap::value_type &X) {
1862 return X.second.mustRecomputePerFunction();
1863 });
1864
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001865 SubtargetFeatureInfo::emitComputeAvailableFeatures(
Daniel Sanderse9fdba32017-04-29 17:30:09 +00001866 Target.getName(), "InstructionSelector", "computeAvailableModuleFeatures",
1867 ModuleFeatures, OS);
1868 SubtargetFeatureInfo::emitComputeAvailableFeatures(
1869 Target.getName(), "InstructionSelector",
1870 "computeAvailableFunctionFeatures", FunctionFeatures, OS,
1871 "const MachineFunction *MF");
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001872
1873 OS << "bool " << Target.getName()
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001874 << "InstructionSelector::selectImpl(MachineInstr &I) const {\n"
1875 << " MachineFunction &MF = *I.getParent()->getParent();\n"
Daniel Sanderse9fdba32017-04-29 17:30:09 +00001876 << " const MachineRegisterInfo &MRI = MF.getRegInfo();\n"
1877 << " // FIXME: This should be computed on a per-function basis rather than per-insn.\n"
1878 << " AvailableFunctionFeatures = computeAvailableFunctionFeatures(&STI, &MF);\n"
1879 << " const PredicateBitset AvailableFeatures = getAvailableFeatures();\n";
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001880
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001881 for (auto &Rule : Rules) {
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001882 Rule.emit(OS, SubtargetFeatures);
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001883 ++NumPatternEmitted;
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001884 }
1885
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001886 OS << " return false;\n"
1887 << "}\n"
1888 << "#endif // ifdef GET_GLOBALISEL_IMPL\n";
Daniel Sanderse9fdba32017-04-29 17:30:09 +00001889
1890 OS << "#ifdef GET_GLOBALISEL_PREDICATES_DECL\n"
1891 << "PredicateBitset AvailableModuleFeatures;\n"
1892 << "mutable PredicateBitset AvailableFunctionFeatures;\n"
1893 << "PredicateBitset getAvailableFeatures() const {\n"
1894 << " return AvailableModuleFeatures | AvailableFunctionFeatures;\n"
1895 << "}\n"
1896 << "PredicateBitset\n"
1897 << "computeAvailableModuleFeatures(const " << Target.getName()
1898 << "Subtarget *Subtarget) const;\n"
1899 << "PredicateBitset\n"
1900 << "computeAvailableFunctionFeatures(const " << Target.getName()
1901 << "Subtarget *Subtarget,\n"
1902 << " const MachineFunction *MF) const;\n"
1903 << "#endif // ifdef GET_GLOBALISEL_PREDICATES_DECL\n";
1904
1905 OS << "#ifdef GET_GLOBALISEL_PREDICATES_INIT\n"
1906 << "AvailableModuleFeatures(computeAvailableModuleFeatures(&STI)),\n"
1907 << "AvailableFunctionFeatures()\n"
1908 << "#endif // ifdef GET_GLOBALISEL_PREDICATES_INIT\n";
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001909}
1910
Daniel Sanderse7b0d662017-04-21 15:59:56 +00001911void GlobalISelEmitter::declareSubtargetFeature(Record *Predicate) {
1912 if (SubtargetFeatures.count(Predicate) == 0)
1913 SubtargetFeatures.emplace(
1914 Predicate, SubtargetFeatureInfo(Predicate, SubtargetFeatures.size()));
1915}
1916
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001917} // end anonymous namespace
1918
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001919//===----------------------------------------------------------------------===//
1920
1921namespace llvm {
1922void EmitGlobalISel(RecordKeeper &RK, raw_ostream &OS) {
1923 GlobalISelEmitter(RK).run(OS);
1924}
1925} // End llvm namespace