blob: bd9d9910d4405e5fce4f1666b758ac553f95f174 [file] [log] [blame]
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001//===- GlobalISelEmitter.cpp - Generate an instruction selector -----------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10/// \file
11/// This tablegen backend emits code for use by the GlobalISel instruction
12/// selector. See include/llvm/CodeGen/TargetGlobalISel.td.
13///
14/// This file analyzes the patterns recognized by the SelectionDAGISel tablegen
15/// backend, filters out the ones that are unsupported, maps
16/// SelectionDAG-specific constructs to their GlobalISel counterpart
17/// (when applicable: MVT to LLT; SDNode to generic Instruction).
18///
19/// Not all patterns are supported: pass the tablegen invocation
20/// "-warn-on-skipped-patterns" to emit a warning when a pattern is skipped,
21/// as well as why.
22///
23/// The generated file defines a single method:
24/// bool <Target>InstructionSelector::selectImpl(MachineInstr &I) const;
25/// intended to be used in InstructionSelector::select as the first-step
26/// selector for the patterns that don't require complex C++.
27///
28/// FIXME: We'll probably want to eventually define a base
29/// "TargetGenInstructionSelector" class.
30///
31//===----------------------------------------------------------------------===//
32
33#include "CodeGenDAGPatterns.h"
34#include "llvm/ADT/Optional.h"
Daniel Sanders0ed28822017-04-12 08:23:08 +000035#include "llvm/ADT/SmallSet.h"
Ahmed Bougacha36f70352016-12-21 23:26:20 +000036#include "llvm/ADT/Statistic.h"
37#include "llvm/CodeGen/MachineValueType.h"
38#include "llvm/Support/CommandLine.h"
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +000039#include "llvm/Support/Error.h"
Daniel Sanders52b4ce72017-03-07 23:20:35 +000040#include "llvm/Support/LowLevelTypeImpl.h"
Pavel Labath52a82e22017-02-21 09:19:41 +000041#include "llvm/Support/ScopedPrinter.h"
Ahmed Bougacha36f70352016-12-21 23:26:20 +000042#include "llvm/TableGen/Error.h"
43#include "llvm/TableGen/Record.h"
44#include "llvm/TableGen/TableGenBackend.h"
45#include <string>
Daniel Sanders8a4bae92017-03-14 21:32:08 +000046#include <numeric>
Ahmed Bougacha36f70352016-12-21 23:26:20 +000047using namespace llvm;
48
49#define DEBUG_TYPE "gisel-emitter"
50
51STATISTIC(NumPatternTotal, "Total number of patterns");
Daniel Sandersb41ce2b2017-02-20 14:31:27 +000052STATISTIC(NumPatternImported, "Number of patterns imported from SelectionDAG");
53STATISTIC(NumPatternImportsSkipped, "Number of SelectionDAG imports skipped");
Ahmed Bougacha36f70352016-12-21 23:26:20 +000054STATISTIC(NumPatternEmitted, "Number of patterns emitted");
55
Daniel Sanders0848b232017-03-27 13:15:13 +000056cl::OptionCategory GlobalISelEmitterCat("Options for -gen-global-isel");
57
Ahmed Bougacha36f70352016-12-21 23:26:20 +000058static cl::opt<bool> WarnOnSkippedPatterns(
59 "warn-on-skipped-patterns",
60 cl::desc("Explain why a pattern was skipped for inclusion "
61 "in the GlobalISel selector"),
Daniel Sanders0848b232017-03-27 13:15:13 +000062 cl::init(false), cl::cat(GlobalISelEmitterCat));
Ahmed Bougacha36f70352016-12-21 23:26:20 +000063
Daniel Sandersbdfebb82017-03-15 20:18:38 +000064namespace {
Ahmed Bougacha36f70352016-12-21 23:26:20 +000065//===- Helper functions ---------------------------------------------------===//
66
Daniel Sanders52b4ce72017-03-07 23:20:35 +000067/// This class stands in for LLT wherever we want to tablegen-erate an
68/// equivalent at compiler run-time.
69class LLTCodeGen {
70private:
71 LLT Ty;
72
73public:
74 LLTCodeGen(const LLT &Ty) : Ty(Ty) {}
75
76 void emitCxxConstructorCall(raw_ostream &OS) const {
77 if (Ty.isScalar()) {
78 OS << "LLT::scalar(" << Ty.getSizeInBits() << ")";
79 return;
80 }
81 if (Ty.isVector()) {
82 OS << "LLT::vector(" << Ty.getNumElements() << ", " << Ty.getSizeInBits()
83 << ")";
84 return;
85 }
86 llvm_unreachable("Unhandled LLT");
87 }
Daniel Sanders8a4bae92017-03-14 21:32:08 +000088
89 const LLT &get() const { return Ty; }
90};
91
92class InstructionMatcher;
93class OperandPlaceholder {
94private:
95 enum PlaceholderKind {
96 OP_MatchReference,
97 OP_Temporary,
98 } Kind;
99
100 struct MatchReferenceData {
101 InstructionMatcher *InsnMatcher;
102 StringRef InsnVarName;
103 StringRef SymbolicName;
104 };
105
106 struct TemporaryData {
107 unsigned OpIdx;
108 };
109
110 union {
111 struct MatchReferenceData MatchReference;
112 struct TemporaryData Temporary;
113 };
114
115 OperandPlaceholder(PlaceholderKind Kind) : Kind(Kind) {}
116
117public:
118 ~OperandPlaceholder() {}
119
120 static OperandPlaceholder
121 CreateMatchReference(InstructionMatcher *InsnMatcher,
Daniel Sandersdb7ed372017-04-04 13:52:00 +0000122 StringRef InsnVarName, StringRef SymbolicName) {
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000123 OperandPlaceholder Result(OP_MatchReference);
124 Result.MatchReference.InsnMatcher = InsnMatcher;
125 Result.MatchReference.InsnVarName = InsnVarName;
126 Result.MatchReference.SymbolicName = SymbolicName;
127 return Result;
128 }
129
130 static OperandPlaceholder CreateTemporary(unsigned OpIdx) {
131 OperandPlaceholder Result(OP_Temporary);
132 Result.Temporary.OpIdx = OpIdx;
133 return Result;
134 }
135
136 void emitCxxValueExpr(raw_ostream &OS) const;
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000137};
138
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000139/// Convert an MVT to an equivalent LLT if possible, or the invalid LLT() for
140/// MVTs that don't map cleanly to an LLT (e.g., iPTR, *any, ...).
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000141static Optional<LLTCodeGen> MVTToLLT(MVT::SimpleValueType SVT) {
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000142 MVT VT(SVT);
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000143 if (VT.isVector() && VT.getVectorNumElements() != 1)
144 return LLTCodeGen(LLT::vector(VT.getVectorNumElements(), VT.getScalarSizeInBits()));
145 if (VT.isInteger() || VT.isFloatingPoint())
146 return LLTCodeGen(LLT::scalar(VT.getSizeInBits()));
147 return None;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000148}
149
150static bool isTrivialOperatorNode(const TreePatternNode *N) {
151 return !N->isLeaf() && !N->hasAnyPredicate() && !N->getTransformFn();
152}
153
154//===- Matchers -----------------------------------------------------------===//
155
Daniel Sandersbee57392017-04-04 13:25:23 +0000156class OperandMatcher;
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000157class MatchAction;
158
159/// Generates code to check that a match rule matches.
160class RuleMatcher {
161 /// A list of matchers that all need to succeed for the current rule to match.
162 /// FIXME: This currently supports a single match position but could be
163 /// extended to support multiple positions to support div/rem fusion or
164 /// load-multiple instructions.
165 std::vector<std::unique_ptr<InstructionMatcher>> Matchers;
166
167 /// A list of actions that need to be taken when all predicates in this rule
168 /// have succeeded.
169 std::vector<std::unique_ptr<MatchAction>> Actions;
170
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000171 /// A map of instruction matchers to the local variables created by
172 /// emitCxxCaptureStmts().
173 std::map<const InstructionMatcher *, std::string> InsnVariableNames;
174
175 /// ID for the next instruction variable defined with defineInsnVar()
176 unsigned NextInsnVarID;
177
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000178public:
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000179 RuleMatcher()
180 : Matchers(), Actions(), InsnVariableNames(), NextInsnVarID(0) {}
Zachary Turnerb7dbd872017-03-20 19:56:52 +0000181 RuleMatcher(RuleMatcher &&Other) = default;
182 RuleMatcher &operator=(RuleMatcher &&Other) = default;
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000183
184 InstructionMatcher &addInstructionMatcher();
185
186 template <class Kind, class... Args> Kind &addAction(Args &&... args);
187
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000188 std::string defineInsnVar(raw_ostream &OS, const InstructionMatcher &Matcher,
189 StringRef Value);
190 StringRef getInsnVarName(const InstructionMatcher &InsnMatcher) const;
191
Daniel Sandersbee57392017-04-04 13:25:23 +0000192 void emitCxxCapturedInsnList(raw_ostream &OS);
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000193 void emitCxxCaptureStmts(raw_ostream &OS, StringRef Expr);
194
195 void emit(raw_ostream &OS);
Daniel Sandersbdfebb82017-03-15 20:18:38 +0000196
197 /// Compare the priority of this object and B.
198 ///
199 /// Returns true if this object is more important than B.
200 bool isHigherPriorityThan(const RuleMatcher &B) const;
201
202 /// Report the maximum number of temporary operands needed by the rule
203 /// matcher.
204 unsigned countTemporaryOperands() const;
205};
206
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000207template <class PredicateTy> class PredicateListMatcher {
208private:
209 typedef std::vector<std::unique_ptr<PredicateTy>> PredicateVec;
210 PredicateVec Predicates;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000211
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000212public:
213 /// Construct a new operand predicate and add it to the matcher.
214 template <class Kind, class... Args>
215 Kind &addPredicate(Args&&... args) {
Ahmed Bougachab67a3ce2017-01-26 22:07:37 +0000216 Predicates.emplace_back(
217 llvm::make_unique<Kind>(std::forward<Args>(args)...));
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000218 return *static_cast<Kind *>(Predicates.back().get());
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000219 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000220
221 typename PredicateVec::const_iterator predicates_begin() const { return Predicates.begin(); }
222 typename PredicateVec::const_iterator predicates_end() const { return Predicates.end(); }
223 iterator_range<typename PredicateVec::const_iterator> predicates() const {
224 return make_range(predicates_begin(), predicates_end());
225 }
Daniel Sanders759ff412017-02-24 13:58:11 +0000226 typename PredicateVec::size_type predicates_size() const { return Predicates.size(); }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000227
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000228 /// Emit a C++ expression that tests whether all the predicates are met.
Ahmed Bougachab67a3ce2017-01-26 22:07:37 +0000229 template <class... Args>
Daniel Sandersf8c804f2017-01-28 11:10:42 +0000230 void emitCxxPredicateListExpr(raw_ostream &OS, Args &&... args) const {
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000231 if (Predicates.empty()) {
232 OS << "true";
233 return;
234 }
235
236 StringRef Separator = "";
237 for (const auto &Predicate : predicates()) {
238 OS << Separator << "(";
Ahmed Bougachab67a3ce2017-01-26 22:07:37 +0000239 Predicate->emitCxxPredicateExpr(OS, std::forward<Args>(args)...);
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000240 OS << ")";
Ahmed Bougacha905af9f2017-02-04 00:47:02 +0000241 Separator = " &&\n";
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000242 }
243 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000244};
245
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000246/// Generates code to check a predicate of an operand.
247///
248/// Typical predicates include:
249/// * Operand is a particular register.
250/// * Operand is assigned a particular register bank.
251/// * Operand is an MBB.
252class OperandPredicateMatcher {
253public:
Daniel Sanders759ff412017-02-24 13:58:11 +0000254 /// This enum is used for RTTI and also defines the priority that is given to
255 /// the predicate when generating the matcher code. Kinds with higher priority
256 /// must be tested first.
257 ///
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000258 /// The relative priority of OPM_LLT, OPM_RegBank, and OPM_MBB do not matter
259 /// but OPM_Int must have priority over OPM_RegBank since constant integers
260 /// are represented by a virtual register defined by a G_CONSTANT instruction.
Daniel Sanders759ff412017-02-24 13:58:11 +0000261 enum PredicateKind {
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000262 OPM_ComplexPattern,
Daniel Sandersbee57392017-04-04 13:25:23 +0000263 OPM_Instruction,
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000264 OPM_Int,
Daniel Sanders759ff412017-02-24 13:58:11 +0000265 OPM_LLT,
266 OPM_RegBank,
267 OPM_MBB,
268 };
269
270protected:
271 PredicateKind Kind;
272
273public:
274 OperandPredicateMatcher(PredicateKind Kind) : Kind(Kind) {}
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000275 virtual ~OperandPredicateMatcher() {}
276
Daniel Sanders759ff412017-02-24 13:58:11 +0000277 PredicateKind getKind() const { return Kind; }
278
Daniel Sandersbee57392017-04-04 13:25:23 +0000279 /// Return the OperandMatcher for the specified operand or nullptr if there
280 /// isn't one by that name in this operand predicate matcher.
281 ///
282 /// InstructionOperandMatcher is the only subclass that can return non-null
283 /// for this.
284 virtual Optional<const OperandMatcher *>
Daniel Sandersdb7ed372017-04-04 13:52:00 +0000285 getOptionalOperand(StringRef SymbolicName) const {
Daniel Sandersbee57392017-04-04 13:25:23 +0000286 assert(!SymbolicName.empty() && "Cannot lookup unnamed operand");
287 return None;
288 }
289
290 /// Emit C++ statements to capture instructions into local variables.
291 ///
292 /// Only InstructionOperandMatcher needs to do anything for this method.
293 virtual void emitCxxCaptureStmts(raw_ostream &OS, RuleMatcher &Rule,
294 StringRef Expr) const {}
295
Daniel Sanderse604ef52017-02-20 15:30:43 +0000296 /// Emit a C++ expression that checks the predicate for the given operand.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000297 virtual void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanderse604ef52017-02-20 15:30:43 +0000298 StringRef OperandExpr) const = 0;
Daniel Sanders759ff412017-02-24 13:58:11 +0000299
300 /// Compare the priority of this object and B.
301 ///
302 /// Returns true if this object is more important than B.
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000303 virtual bool isHigherPriorityThan(const OperandPredicateMatcher &B) const {
304 return Kind < B.Kind;
Daniel Sanders759ff412017-02-24 13:58:11 +0000305 };
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000306
307 /// Report the maximum number of temporary operands needed by the predicate
308 /// matcher.
309 virtual unsigned countTemporaryOperands() const { return 0; }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000310};
311
312/// Generates code to check that an operand is a particular LLT.
313class LLTOperandMatcher : public OperandPredicateMatcher {
314protected:
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000315 LLTCodeGen Ty;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000316
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000317public:
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000318 LLTOperandMatcher(const LLTCodeGen &Ty)
Daniel Sanders759ff412017-02-24 13:58:11 +0000319 : OperandPredicateMatcher(OPM_LLT), Ty(Ty) {}
320
321 static bool classof(const OperandPredicateMatcher *P) {
322 return P->getKind() == OPM_LLT;
323 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000324
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000325 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanderse604ef52017-02-20 15:30:43 +0000326 StringRef OperandExpr) const override {
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000327 OS << "MRI.getType(" << OperandExpr << ".getReg()) == (";
328 Ty.emitCxxConstructorCall(OS);
329 OS << ")";
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000330 }
331};
332
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000333/// Generates code to check that an operand is a particular target constant.
334class ComplexPatternOperandMatcher : public OperandPredicateMatcher {
335protected:
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000336 const OperandMatcher &Operand;
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000337 const Record &TheDef;
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000338
339 unsigned getNumOperands() const {
340 return TheDef.getValueAsDag("Operands")->getNumArgs();
341 }
342
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000343 unsigned getAllocatedTemporariesBaseID() const;
344
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000345public:
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000346 ComplexPatternOperandMatcher(const OperandMatcher &Operand,
347 const Record &TheDef)
348 : OperandPredicateMatcher(OPM_ComplexPattern), Operand(Operand),
349 TheDef(TheDef) {}
350
351 static bool classof(const OperandPredicateMatcher *P) {
352 return P->getKind() == OPM_ComplexPattern;
353 }
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000354
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000355 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000356 StringRef OperandExpr) const override {
357 OS << TheDef.getValueAsString("MatcherFn") << "(" << OperandExpr;
358 for (unsigned I = 0; I < getNumOperands(); ++I) {
359 OS << ", ";
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000360 OperandPlaceholder::CreateTemporary(getAllocatedTemporariesBaseID() + I)
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000361 .emitCxxValueExpr(OS);
362 }
363 OS << ")";
364 }
365
366 unsigned countTemporaryOperands() const override {
367 return getNumOperands();
368 }
369};
370
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000371/// Generates code to check that an operand is in a particular register bank.
372class RegisterBankOperandMatcher : public OperandPredicateMatcher {
373protected:
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000374 const CodeGenRegisterClass &RC;
375
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000376public:
Daniel Sanders759ff412017-02-24 13:58:11 +0000377 RegisterBankOperandMatcher(const CodeGenRegisterClass &RC)
378 : OperandPredicateMatcher(OPM_RegBank), RC(RC) {}
379
380 static bool classof(const OperandPredicateMatcher *P) {
381 return P->getKind() == OPM_RegBank;
382 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000383
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000384 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanderse604ef52017-02-20 15:30:43 +0000385 StringRef OperandExpr) const override {
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000386 OS << "(&RBI.getRegBankFromRegClass(" << RC.getQualifiedName()
Daniel Sanderse604ef52017-02-20 15:30:43 +0000387 << "RegClass) == RBI.getRegBank(" << OperandExpr
388 << ".getReg(), MRI, TRI))";
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000389 }
390};
391
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000392/// Generates code to check that an operand is a basic block.
393class MBBOperandMatcher : public OperandPredicateMatcher {
394public:
Daniel Sanders759ff412017-02-24 13:58:11 +0000395 MBBOperandMatcher() : OperandPredicateMatcher(OPM_MBB) {}
396
397 static bool classof(const OperandPredicateMatcher *P) {
398 return P->getKind() == OPM_MBB;
399 }
400
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000401 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sanderse604ef52017-02-20 15:30:43 +0000402 StringRef OperandExpr) const override {
403 OS << OperandExpr << ".isMBB()";
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000404 }
405};
406
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000407/// Generates code to check that an operand is a particular int.
408class IntOperandMatcher : public OperandPredicateMatcher {
409protected:
410 int64_t Value;
411
412public:
413 IntOperandMatcher(int64_t Value)
414 : OperandPredicateMatcher(OPM_Int), Value(Value) {}
415
416 static bool classof(const OperandPredicateMatcher *P) {
417 return P->getKind() == OPM_Int;
418 }
419
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000420 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Simon Pilgrimd0302912017-02-24 17:20:27 +0000421 StringRef OperandExpr) const override {
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000422 OS << "isOperandImmEqual(" << OperandExpr << ", " << Value << ", MRI)";
423 }
424};
425
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000426/// Generates code to check that a set of predicates match for a particular
427/// operand.
428class OperandMatcher : public PredicateListMatcher<OperandPredicateMatcher> {
429protected:
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000430 InstructionMatcher &Insn;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000431 unsigned OpIdx;
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000432 std::string SymbolicName;
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000433
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000434 /// The index of the first temporary variable allocated to this operand. The
435 /// number of allocated temporaries can be found with
436 /// countTemporaryOperands().
437 unsigned AllocatedTemporariesBaseID;
438
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000439public:
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000440 OperandMatcher(InstructionMatcher &Insn, unsigned OpIdx,
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000441 const std::string &SymbolicName,
442 unsigned AllocatedTemporariesBaseID)
443 : Insn(Insn), OpIdx(OpIdx), SymbolicName(SymbolicName),
444 AllocatedTemporariesBaseID(AllocatedTemporariesBaseID) {}
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000445
446 bool hasSymbolicName() const { return !SymbolicName.empty(); }
447 const StringRef getSymbolicName() const { return SymbolicName; }
Daniel Sandersffc7d582017-03-29 15:37:18 +0000448 void setSymbolicName(StringRef Name) {
449 assert(SymbolicName.empty() && "Operand already has a symbolic name");
450 SymbolicName = Name;
451 }
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000452 unsigned getOperandIndex() const { return OpIdx; }
453
Daniel Sandersdb7ed372017-04-04 13:52:00 +0000454 std::string getOperandExpr(StringRef InsnVarName) const {
Pavel Labath52a82e22017-02-21 09:19:41 +0000455 return (InsnVarName + ".getOperand(" + llvm::to_string(OpIdx) + ")").str();
Daniel Sanderse604ef52017-02-20 15:30:43 +0000456 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000457
Daniel Sandersbee57392017-04-04 13:25:23 +0000458 Optional<const OperandMatcher *>
459 getOptionalOperand(StringRef DesiredSymbolicName) const {
460 assert(!DesiredSymbolicName.empty() && "Cannot lookup unnamed operand");
461 if (DesiredSymbolicName == SymbolicName)
462 return this;
463 for (const auto &OP : predicates()) {
464 const auto &MaybeOperand = OP->getOptionalOperand(DesiredSymbolicName);
465 if (MaybeOperand.hasValue())
466 return MaybeOperand.getValue();
467 }
468 return None;
469 }
470
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000471 InstructionMatcher &getInstructionMatcher() const { return Insn; }
472
Daniel Sandersbee57392017-04-04 13:25:23 +0000473 /// Emit C++ statements to capture instructions into local variables.
474 void emitCxxCaptureStmts(raw_ostream &OS, RuleMatcher &Rule,
475 StringRef OperandExpr) const {
476 for (const auto &Predicate : predicates())
477 Predicate->emitCxxCaptureStmts(OS, Rule, OperandExpr);
478 }
479
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000480 /// Emit a C++ expression that tests whether the instruction named in
481 /// InsnVarName matches all the predicate and all the operands.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000482 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Daniel Sandersdb7ed372017-04-04 13:52:00 +0000483 StringRef InsnVarName) const {
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000484 OS << "(/* ";
485 if (SymbolicName.empty())
486 OS << "Operand " << OpIdx;
487 else
488 OS << SymbolicName;
489 OS << " */ ";
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000490 emitCxxPredicateListExpr(OS, Rule, getOperandExpr(InsnVarName));
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000491 OS << ")";
492 }
Daniel Sanders759ff412017-02-24 13:58:11 +0000493
494 /// Compare the priority of this object and B.
495 ///
496 /// Returns true if this object is more important than B.
497 bool isHigherPriorityThan(const OperandMatcher &B) const {
498 // Operand matchers involving more predicates have higher priority.
499 if (predicates_size() > B.predicates_size())
500 return true;
501 if (predicates_size() < B.predicates_size())
502 return false;
503
504 // This assumes that predicates are added in a consistent order.
505 for (const auto &Predicate : zip(predicates(), B.predicates())) {
506 if (std::get<0>(Predicate)->isHigherPriorityThan(*std::get<1>(Predicate)))
507 return true;
508 if (std::get<1>(Predicate)->isHigherPriorityThan(*std::get<0>(Predicate)))
509 return false;
510 }
511
512 return false;
513 };
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000514
515 /// Report the maximum number of temporary operands needed by the operand
516 /// matcher.
517 unsigned countTemporaryOperands() const {
518 return std::accumulate(
519 predicates().begin(), predicates().end(), 0,
520 [](unsigned A,
521 const std::unique_ptr<OperandPredicateMatcher> &Predicate) {
522 return A + Predicate->countTemporaryOperands();
523 });
524 }
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000525
526 unsigned getAllocatedTemporariesBaseID() const {
527 return AllocatedTemporariesBaseID;
528 }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000529};
530
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000531unsigned ComplexPatternOperandMatcher::getAllocatedTemporariesBaseID() const {
532 return Operand.getAllocatedTemporariesBaseID();
533}
534
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000535/// Generates code to check a predicate on an instruction.
536///
537/// Typical predicates include:
538/// * The opcode of the instruction is a particular value.
539/// * The nsw/nuw flag is/isn't set.
540class InstructionPredicateMatcher {
Daniel Sanders759ff412017-02-24 13:58:11 +0000541protected:
542 /// This enum is used for RTTI and also defines the priority that is given to
543 /// the predicate when generating the matcher code. Kinds with higher priority
544 /// must be tested first.
545 enum PredicateKind {
546 IPM_Opcode,
547 };
548
549 PredicateKind Kind;
550
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000551public:
Daniel Sanders8d4d72f2017-02-24 14:53:35 +0000552 InstructionPredicateMatcher(PredicateKind Kind) : Kind(Kind) {}
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000553 virtual ~InstructionPredicateMatcher() {}
554
Daniel Sanders759ff412017-02-24 13:58:11 +0000555 PredicateKind getKind() const { return Kind; }
556
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000557 /// Emit a C++ expression that tests whether the instruction named in
558 /// InsnVarName matches the predicate.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000559 virtual void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Ahmed Bougacha6a1ac5a2017-02-09 02:50:01 +0000560 StringRef InsnVarName) const = 0;
Daniel Sanders759ff412017-02-24 13:58:11 +0000561
562 /// Compare the priority of this object and B.
563 ///
564 /// Returns true if this object is more important than B.
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000565 virtual bool isHigherPriorityThan(const InstructionPredicateMatcher &B) const {
Daniel Sanders759ff412017-02-24 13:58:11 +0000566 return Kind < B.Kind;
567 };
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000568
569 /// Report the maximum number of temporary operands needed by the predicate
570 /// matcher.
571 virtual unsigned countTemporaryOperands() const { return 0; }
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000572};
573
574/// Generates code to check the opcode of an instruction.
575class InstructionOpcodeMatcher : public InstructionPredicateMatcher {
576protected:
577 const CodeGenInstruction *I;
578
579public:
Daniel Sanders8d4d72f2017-02-24 14:53:35 +0000580 InstructionOpcodeMatcher(const CodeGenInstruction *I)
581 : InstructionPredicateMatcher(IPM_Opcode), I(I) {}
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000582
Daniel Sanders759ff412017-02-24 13:58:11 +0000583 static bool classof(const InstructionPredicateMatcher *P) {
584 return P->getKind() == IPM_Opcode;
585 }
586
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000587 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
Ahmed Bougacha6a1ac5a2017-02-09 02:50:01 +0000588 StringRef InsnVarName) const override {
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000589 OS << InsnVarName << ".getOpcode() == " << I->Namespace
590 << "::" << I->TheDef->getName();
591 }
Daniel Sanders759ff412017-02-24 13:58:11 +0000592
593 /// Compare the priority of this object and B.
594 ///
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000595 /// Returns true if this object is more important than B.
596 bool isHigherPriorityThan(const InstructionPredicateMatcher &B) const override {
Daniel Sanders759ff412017-02-24 13:58:11 +0000597 if (InstructionPredicateMatcher::isHigherPriorityThan(B))
598 return true;
599 if (B.InstructionPredicateMatcher::isHigherPriorityThan(*this))
600 return false;
601
602 // Prioritize opcodes for cosmetic reasons in the generated source. Although
603 // this is cosmetic at the moment, we may want to drive a similar ordering
604 // using instruction frequency information to improve compile time.
605 if (const InstructionOpcodeMatcher *BO =
606 dyn_cast<InstructionOpcodeMatcher>(&B))
607 return I->TheDef->getName() < BO->I->TheDef->getName();
608
609 return false;
610 };
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000611};
612
613/// Generates code to check that a set of predicates and operands match for a
614/// particular instruction.
615///
616/// Typical predicates include:
617/// * Has a specific opcode.
618/// * Has an nsw/nuw flag or doesn't.
619class InstructionMatcher
620 : public PredicateListMatcher<InstructionPredicateMatcher> {
621protected:
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000622 typedef std::vector<std::unique_ptr<OperandMatcher>> OperandVec;
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000623
624 /// The operands to match. All rendered operands must be present even if the
625 /// condition is always true.
626 OperandVec Operands;
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000627
628public:
629 /// Add an operand to the matcher.
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000630 OperandMatcher &addOperand(unsigned OpIdx, const std::string &SymbolicName,
631 unsigned AllocatedTemporariesBaseID) {
632 Operands.emplace_back(new OperandMatcher(*this, OpIdx, SymbolicName,
633 AllocatedTemporariesBaseID));
634 return *Operands.back();
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000635 }
636
Daniel Sandersffc7d582017-03-29 15:37:18 +0000637 OperandMatcher &getOperand(unsigned OpIdx) {
638 auto I = std::find_if(Operands.begin(), Operands.end(),
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000639 [&OpIdx](const std::unique_ptr<OperandMatcher> &X) {
640 return X->getOperandIndex() == OpIdx;
Daniel Sandersffc7d582017-03-29 15:37:18 +0000641 });
642 if (I != Operands.end())
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000643 return **I;
Daniel Sandersffc7d582017-03-29 15:37:18 +0000644 llvm_unreachable("Failed to lookup operand");
645 }
646
Daniel Sandersbee57392017-04-04 13:25:23 +0000647 Optional<const OperandMatcher *>
648 getOptionalOperand(StringRef SymbolicName) const {
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000649 assert(!SymbolicName.empty() && "Cannot lookup unnamed operand");
Daniel Sandersbee57392017-04-04 13:25:23 +0000650 for (const auto &Operand : Operands) {
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000651 const auto &OM = Operand->getOptionalOperand(SymbolicName);
Daniel Sandersbee57392017-04-04 13:25:23 +0000652 if (OM.hasValue())
653 return OM.getValue();
654 }
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000655 return None;
656 }
657
Daniel Sandersdb7ed372017-04-04 13:52:00 +0000658 const OperandMatcher &getOperand(StringRef SymbolicName) const {
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000659 Optional<const OperandMatcher *>OM = getOptionalOperand(SymbolicName);
660 if (OM.hasValue())
661 return *OM.getValue();
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000662 llvm_unreachable("Failed to lookup operand");
663 }
664
665 unsigned getNumOperands() const { return Operands.size(); }
Daniel Sandersbee57392017-04-04 13:25:23 +0000666 OperandVec::iterator operands_begin() { return Operands.begin(); }
667 OperandVec::iterator operands_end() { return Operands.end(); }
668 iterator_range<OperandVec::iterator> operands() {
669 return make_range(operands_begin(), operands_end());
670 }
Daniel Sandersffc7d582017-03-29 15:37:18 +0000671 OperandVec::const_iterator operands_begin() const { return Operands.begin(); }
672 OperandVec::const_iterator operands_end() const { return Operands.end(); }
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000673 iterator_range<OperandVec::const_iterator> operands() const {
674 return make_range(operands_begin(), operands_end());
675 }
676
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000677 /// Emit C++ statements to check the shape of the match and capture
678 /// instructions into local variables.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000679 void emitCxxCaptureStmts(raw_ostream &OS, RuleMatcher &Rule, StringRef Expr) {
680 OS << "if (" << Expr << ".getNumOperands() < " << getNumOperands() << ")\n"
681 << " return false;\n";
Daniel Sandersbee57392017-04-04 13:25:23 +0000682 for (const auto &Operand : Operands) {
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000683 Operand->emitCxxCaptureStmts(OS, Rule, Operand->getOperandExpr(Expr));
Daniel Sandersbee57392017-04-04 13:25:23 +0000684 }
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000685 }
686
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000687 /// Emit a C++ expression that tests whether the instruction named in
688 /// InsnVarName matches all the predicates and all the operands.
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000689 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
690 StringRef InsnVarName) const {
691 emitCxxPredicateListExpr(OS, Rule, InsnVarName);
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000692 for (const auto &Operand : Operands) {
Ahmed Bougacha905af9f2017-02-04 00:47:02 +0000693 OS << " &&\n(";
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000694 Operand->emitCxxPredicateExpr(OS, Rule, InsnVarName);
Daniel Sandersdc662ff2017-01-26 11:10:14 +0000695 OS << ")";
696 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000697 }
Daniel Sanders759ff412017-02-24 13:58:11 +0000698
699 /// Compare the priority of this object and B.
700 ///
701 /// Returns true if this object is more important than B.
702 bool isHigherPriorityThan(const InstructionMatcher &B) const {
703 // Instruction matchers involving more operands have higher priority.
704 if (Operands.size() > B.Operands.size())
705 return true;
706 if (Operands.size() < B.Operands.size())
707 return false;
708
709 for (const auto &Predicate : zip(predicates(), B.predicates())) {
710 if (std::get<0>(Predicate)->isHigherPriorityThan(*std::get<1>(Predicate)))
711 return true;
712 if (std::get<1>(Predicate)->isHigherPriorityThan(*std::get<0>(Predicate)))
713 return false;
714 }
715
716 for (const auto &Operand : zip(Operands, B.Operands)) {
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000717 if (std::get<0>(Operand)->isHigherPriorityThan(*std::get<1>(Operand)))
Daniel Sanders759ff412017-02-24 13:58:11 +0000718 return true;
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000719 if (std::get<1>(Operand)->isHigherPriorityThan(*std::get<0>(Operand)))
Daniel Sanders759ff412017-02-24 13:58:11 +0000720 return false;
721 }
722
723 return false;
724 };
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000725
726 /// Report the maximum number of temporary operands needed by the instruction
727 /// matcher.
728 unsigned countTemporaryOperands() const {
729 return std::accumulate(predicates().begin(), predicates().end(), 0,
730 [](unsigned A,
731 const std::unique_ptr<InstructionPredicateMatcher>
732 &Predicate) {
733 return A + Predicate->countTemporaryOperands();
734 }) +
Daniel Sanders4f3eb242017-04-05 13:14:03 +0000735 std::accumulate(
736 Operands.begin(), Operands.end(), 0,
737 [](unsigned A, const std::unique_ptr<OperandMatcher> &Operand) {
738 return A + Operand->countTemporaryOperands();
739 });
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000740 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000741};
742
Daniel Sandersbee57392017-04-04 13:25:23 +0000743/// Generates code to check that the operand is a register defined by an
744/// instruction that matches the given instruction matcher.
745///
746/// For example, the pattern:
747/// (set $dst, (G_MUL (G_ADD $src1, $src2), $src3))
748/// would use an InstructionOperandMatcher for operand 1 of the G_MUL to match
749/// the:
750/// (G_ADD $src1, $src2)
751/// subpattern.
752class InstructionOperandMatcher : public OperandPredicateMatcher {
753protected:
754 std::unique_ptr<InstructionMatcher> InsnMatcher;
755
756public:
757 InstructionOperandMatcher()
758 : OperandPredicateMatcher(OPM_Instruction),
759 InsnMatcher(new InstructionMatcher()) {}
760
761 static bool classof(const OperandPredicateMatcher *P) {
762 return P->getKind() == OPM_Instruction;
763 }
764
765 InstructionMatcher &getInsnMatcher() const { return *InsnMatcher; }
766
767 Optional<const OperandMatcher *>
768 getOptionalOperand(StringRef SymbolicName) const override {
769 assert(!SymbolicName.empty() && "Cannot lookup unnamed operand");
770 return InsnMatcher->getOptionalOperand(SymbolicName);
771 }
772
773 void emitCxxCaptureStmts(raw_ostream &OS, RuleMatcher &Rule,
774 StringRef OperandExpr) const override {
775 OS << "if (!" << OperandExpr + ".isReg())\n"
776 << " return false;\n";
777 std::string InsnVarName = Rule.defineInsnVar(
778 OS, *InsnMatcher,
779 ("*MRI.getVRegDef(" + OperandExpr + ".getReg())").str());
780 InsnMatcher->emitCxxCaptureStmts(OS, Rule, InsnVarName);
781 }
782
783 void emitCxxPredicateExpr(raw_ostream &OS, RuleMatcher &Rule,
784 StringRef OperandExpr) const override {
785 OperandExpr = Rule.getInsnVarName(*InsnMatcher);
786 OS << "(";
787 InsnMatcher->emitCxxPredicateExpr(OS, Rule, OperandExpr);
788 OS << ")\n";
789 }
790};
791
Daniel Sanders43c882c2017-02-01 10:53:10 +0000792//===- Actions ------------------------------------------------------------===//
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000793void OperandPlaceholder::emitCxxValueExpr(raw_ostream &OS) const {
794 switch (Kind) {
795 case OP_MatchReference:
796 OS << MatchReference.InsnMatcher->getOperand(MatchReference.SymbolicName)
797 .getOperandExpr(MatchReference.InsnVarName);
798 break;
799 case OP_Temporary:
800 OS << "TempOp" << Temporary.OpIdx;
801 break;
802 }
803}
Daniel Sanders43c882c2017-02-01 10:53:10 +0000804
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000805class OperandRenderer {
806public:
Daniel Sanders0ed28822017-04-12 08:23:08 +0000807 enum RendererKind { OR_Copy, OR_Imm, OR_Register, OR_ComplexPattern };
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000808
809protected:
810 RendererKind Kind;
811
812public:
813 OperandRenderer(RendererKind Kind) : Kind(Kind) {}
814 virtual ~OperandRenderer() {}
815
816 RendererKind getKind() const { return Kind; }
817
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000818 virtual void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const = 0;
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000819};
820
821/// A CopyRenderer emits code to copy a single operand from an existing
822/// instruction to the one being built.
823class CopyRenderer : public OperandRenderer {
824protected:
825 /// The matcher for the instruction that this operand is copied from.
826 /// This provides the facility for looking up an a operand by it's name so
827 /// that it can be used as a source for the instruction being built.
828 const InstructionMatcher &Matched;
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000829 /// The name of the operand.
830 const StringRef SymbolicName;
831
832public:
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000833 CopyRenderer(const InstructionMatcher &Matched, StringRef SymbolicName)
834 : OperandRenderer(OR_Copy), Matched(Matched), SymbolicName(SymbolicName) {
835 }
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000836
837 static bool classof(const OperandRenderer *R) {
838 return R->getKind() == OR_Copy;
839 }
840
841 const StringRef getSymbolicName() const { return SymbolicName; }
842
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000843 void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const override {
844 const OperandMatcher &Operand = Matched.getOperand(SymbolicName);
845 StringRef InsnVarName =
846 Rule.getInsnVarName(Operand.getInstructionMatcher());
847 std::string OperandExpr = Operand.getOperandExpr(InsnVarName);
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000848 OS << " MIB.add(" << OperandExpr << "/*" << SymbolicName << "*/);\n";
849 }
850};
851
852/// Adds a specific physical register to the instruction being built.
853/// This is typically useful for WZR/XZR on AArch64.
854class AddRegisterRenderer : public OperandRenderer {
855protected:
856 const Record *RegisterDef;
857
858public:
859 AddRegisterRenderer(const Record *RegisterDef)
860 : OperandRenderer(OR_Register), RegisterDef(RegisterDef) {}
861
862 static bool classof(const OperandRenderer *R) {
863 return R->getKind() == OR_Register;
864 }
865
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000866 void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const override {
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000867 OS << " MIB.addReg(" << RegisterDef->getValueAsString("Namespace")
868 << "::" << RegisterDef->getName() << ");\n";
869 }
870};
871
Daniel Sanders0ed28822017-04-12 08:23:08 +0000872/// Adds a specific immediate to the instruction being built.
873class ImmRenderer : public OperandRenderer {
874protected:
875 int64_t Imm;
876
877public:
878 ImmRenderer(int64_t Imm)
879 : OperandRenderer(OR_Imm), Imm(Imm) {}
880
881 static bool classof(const OperandRenderer *R) {
882 return R->getKind() == OR_Imm;
883 }
884
885 void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const override {
886 OS << " MIB.addImm(" << Imm << ");\n";
887 }
888};
889
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000890class RenderComplexPatternOperand : public OperandRenderer {
891private:
892 const Record &TheDef;
893 std::vector<OperandPlaceholder> Sources;
894
895 unsigned getNumOperands() const {
896 return TheDef.getValueAsDag("Operands")->getNumArgs();
897 }
898
899public:
900 RenderComplexPatternOperand(const Record &TheDef,
901 const ArrayRef<OperandPlaceholder> Sources)
902 : OperandRenderer(OR_ComplexPattern), TheDef(TheDef), Sources(Sources) {}
903
904 static bool classof(const OperandRenderer *R) {
905 return R->getKind() == OR_ComplexPattern;
906 }
907
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000908 void emitCxxRenderStmts(raw_ostream &OS, RuleMatcher &Rule) const override {
Daniel Sanders8a4bae92017-03-14 21:32:08 +0000909 assert(Sources.size() == getNumOperands() && "Inconsistent number of operands");
910 for (const auto &Source : Sources) {
911 OS << "MIB.add(";
912 Source.emitCxxValueExpr(OS);
913 OS << ");\n";
914 }
915 }
916};
917
Ahmed Bougacha56ca3a92017-02-04 00:47:10 +0000918/// An action taken when all Matcher predicates succeeded for a parent rule.
919///
920/// Typical actions include:
921/// * Changing the opcode of an instruction.
922/// * Adding an operand to an instruction.
Daniel Sanders43c882c2017-02-01 10:53:10 +0000923class MatchAction {
924public:
925 virtual ~MatchAction() {}
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000926
927 /// Emit the C++ statements to implement the action.
928 ///
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000929 /// \param RecycleVarName If given, it's an instruction to recycle. The
930 /// requirements on the instruction vary from action to
931 /// action.
932 virtual void emitCxxActionStmts(raw_ostream &OS, RuleMatcher &Rule,
933 StringRef RecycleVarName) const = 0;
Daniel Sanders43c882c2017-02-01 10:53:10 +0000934};
935
Ahmed Bougacha9aa4c102017-02-04 00:47:08 +0000936/// Generates a comment describing the matched rule being acted upon.
937class DebugCommentAction : public MatchAction {
938private:
939 const PatternToMatch &P;
940
941public:
942 DebugCommentAction(const PatternToMatch &P) : P(P) {}
943
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000944 void emitCxxActionStmts(raw_ostream &OS, RuleMatcher &Rule,
945 StringRef RecycleVarName) const override {
946 OS << "// " << *P.getSrcPattern() << " => " << *P.getDstPattern() << "\n";
Ahmed Bougacha9aa4c102017-02-04 00:47:08 +0000947 }
948};
949
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000950/// Generates code to build an instruction or mutate an existing instruction
951/// into the desired instruction when this is possible.
952class BuildMIAction : public MatchAction {
Daniel Sanders43c882c2017-02-01 10:53:10 +0000953private:
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000954 const CodeGenInstruction *I;
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000955 const InstructionMatcher &Matched;
956 std::vector<std::unique_ptr<OperandRenderer>> OperandRenderers;
957
958 /// True if the instruction can be built solely by mutating the opcode.
959 bool canMutate() const {
960 for (const auto &Renderer : enumerate(OperandRenderers)) {
Zachary Turner309a0882017-03-13 16:24:10 +0000961 if (const auto *Copy = dyn_cast<CopyRenderer>(&*Renderer.value())) {
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000962 if (Matched.getOperand(Copy->getSymbolicName()).getOperandIndex() !=
Zachary Turner309a0882017-03-13 16:24:10 +0000963 Renderer.index())
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000964 return false;
965 } else
966 return false;
967 }
968
969 return true;
970 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +0000971
Daniel Sanders43c882c2017-02-01 10:53:10 +0000972public:
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000973 BuildMIAction(const CodeGenInstruction *I, const InstructionMatcher &Matched)
974 : I(I), Matched(Matched) {}
Daniel Sanders43c882c2017-02-01 10:53:10 +0000975
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000976 template <class Kind, class... Args>
977 Kind &addRenderer(Args&&... args) {
978 OperandRenderers.emplace_back(
979 llvm::make_unique<Kind>(std::forward<Args>(args)...));
980 return *static_cast<Kind *>(OperandRenderers.back().get());
981 }
982
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000983 void emitCxxActionStmts(raw_ostream &OS, RuleMatcher &Rule,
984 StringRef RecycleVarName) const override {
Daniel Sanders066ebbf2017-02-24 15:43:30 +0000985 if (canMutate()) {
Tim Northover4340d642017-03-20 21:58:23 +0000986 OS << " " << RecycleVarName << ".setDesc(TII.get(" << I->Namespace
Daniel Sandersb96f40d2017-03-20 15:20:42 +0000987 << "::" << I->TheDef->getName() << "));\n";
Tim Northover4340d642017-03-20 21:58:23 +0000988
989 if (!I->ImplicitDefs.empty() || !I->ImplicitUses.empty()) {
990 OS << " auto MIB = MachineInstrBuilder(MF, &" << RecycleVarName
991 << ");\n";
992
993 for (auto Def : I->ImplicitDefs) {
994 auto Namespace = Def->getValueAsString("Namespace");
995 OS << " MIB.addDef(" << Namespace << "::" << Def->getName()
996 << ", RegState::Implicit);\n";
997 }
998 for (auto Use : I->ImplicitUses) {
999 auto Namespace = Use->getValueAsString("Namespace");
1000 OS << " MIB.addUse(" << Namespace << "::" << Use->getName()
1001 << ", RegState::Implicit);\n";
1002 }
1003 }
1004
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001005 OS << " MachineInstr &NewI = " << RecycleVarName << ";\n";
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001006 return;
1007 }
1008
1009 // TODO: Simple permutation looks like it could be almost as common as
1010 // mutation due to commutative operations.
1011
1012 OS << "MachineInstrBuilder MIB = BuildMI(*I.getParent(), I, "
1013 "I.getDebugLoc(), TII.get("
1014 << I->Namespace << "::" << I->TheDef->getName() << "));\n";
1015 for (const auto &Renderer : OperandRenderers)
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001016 Renderer->emitCxxRenderStmts(OS, Rule);
Daniel Sandersbee57392017-04-04 13:25:23 +00001017 OS << " for (const auto *FromMI : ";
1018 Rule.emitCxxCapturedInsnList(OS);
1019 OS << ")\n";
1020 OS << " for (const auto &MMO : FromMI->memoperands())\n";
1021 OS << " MIB.addMemOperand(MMO);\n";
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001022 OS << " " << RecycleVarName << ".eraseFromParent();\n";
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001023 OS << " MachineInstr &NewI = *MIB;\n";
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001024 }
1025};
1026
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001027InstructionMatcher &RuleMatcher::addInstructionMatcher() {
1028 Matchers.emplace_back(new InstructionMatcher());
1029 return *Matchers.back();
1030}
Ahmed Bougacha56ca3a92017-02-04 00:47:10 +00001031
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001032template <class Kind, class... Args>
1033Kind &RuleMatcher::addAction(Args &&... args) {
1034 Actions.emplace_back(llvm::make_unique<Kind>(std::forward<Args>(args)...));
1035 return *static_cast<Kind *>(Actions.back().get());
1036}
Daniel Sandersdc662ff2017-01-26 11:10:14 +00001037
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001038std::string RuleMatcher::defineInsnVar(raw_ostream &OS,
1039 const InstructionMatcher &Matcher,
1040 StringRef Value) {
1041 std::string InsnVarName = "MI" + llvm::to_string(NextInsnVarID++);
1042 OS << "MachineInstr &" << InsnVarName << " = " << Value << ";\n";
1043 InsnVariableNames[&Matcher] = InsnVarName;
1044 return InsnVarName;
1045}
1046
1047StringRef RuleMatcher::getInsnVarName(const InstructionMatcher &InsnMatcher) const {
1048 const auto &I = InsnVariableNames.find(&InsnMatcher);
1049 if (I != InsnVariableNames.end())
1050 return I->second;
1051 llvm_unreachable("Matched Insn was not captured in a local variable");
1052}
1053
Daniel Sandersbee57392017-04-04 13:25:23 +00001054/// Emit a C++ initializer_list containing references to every matched instruction.
1055void RuleMatcher::emitCxxCapturedInsnList(raw_ostream &OS) {
Daniel Sanders9e4817d2017-04-04 14:27:06 +00001056 SmallVector<StringRef, 2> Names;
Daniel Sandersbee57392017-04-04 13:25:23 +00001057 for (const auto &Pair : InsnVariableNames)
Daniel Sanders9e4817d2017-04-04 14:27:06 +00001058 Names.push_back(Pair.second);
1059 std::sort(Names.begin(), Names.end());
1060
1061 OS << "{";
1062 for (const auto &Name : Names)
1063 OS << "&" << Name << ", ";
Daniel Sandersbee57392017-04-04 13:25:23 +00001064 OS << "}";
1065}
1066
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001067/// Emit C++ statements to check the shape of the match and capture
1068/// instructions into local variables.
1069void RuleMatcher::emitCxxCaptureStmts(raw_ostream &OS, StringRef Expr) {
1070 assert(Matchers.size() == 1 && "Cannot handle multi-root matchers yet");
1071 std::string InsnVarName = defineInsnVar(OS, *Matchers.front(), Expr);
1072 Matchers.front()->emitCxxCaptureStmts(OS, *this, InsnVarName);
1073}
1074
1075void RuleMatcher::emit(raw_ostream &OS) {
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001076 if (Matchers.empty())
1077 llvm_unreachable("Unexpected empty matcher!");
Daniel Sandersdc662ff2017-01-26 11:10:14 +00001078
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001079 // The representation supports rules that require multiple roots such as:
1080 // %ptr(p0) = ...
1081 // %elt0(s32) = G_LOAD %ptr
1082 // %1(p0) = G_ADD %ptr, 4
1083 // %elt1(s32) = G_LOAD p0 %1
1084 // which could be usefully folded into:
1085 // %ptr(p0) = ...
1086 // %elt0(s32), %elt1(s32) = TGT_LOAD_PAIR %ptr
1087 // on some targets but we don't need to make use of that yet.
1088 assert(Matchers.size() == 1 && "Cannot handle multi-root matchers yet");
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001089 OS << "if ([&]() {\n";
1090
1091 emitCxxCaptureStmts(OS, "I");
1092
1093 OS << " if (";
1094 Matchers.front()->emitCxxPredicateExpr(OS, *this,
1095 getInsnVarName(*Matchers.front()));
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001096 OS << ") {\n";
1097
Daniel Sandersbee57392017-04-04 13:25:23 +00001098 // We must also check if it's safe to fold the matched instructions.
1099 if (InsnVariableNames.size() >= 2) {
1100 for (const auto &Pair : InsnVariableNames) {
1101 // Skip the root node since it isn't moving anywhere. Everything else is
1102 // sinking to meet it.
1103 if (Pair.first == Matchers.front().get())
1104 continue;
1105
1106 // Reject the difficult cases until we have a more accurate check.
1107 OS << " if (!isObviouslySafeToFold(" << Pair.second
1108 << ")) return false;\n";
1109
1110 // FIXME: Emit checks to determine it's _actually_ safe to fold and/or
1111 // account for unsafe cases.
1112 //
1113 // Example:
1114 // MI1--> %0 = ...
1115 // %1 = ... %0
1116 // MI0--> %2 = ... %0
1117 // It's not safe to erase MI1. We currently handle this by not
1118 // erasing %0 (even when it's dead).
1119 //
1120 // Example:
1121 // MI1--> %0 = load volatile @a
1122 // %1 = load volatile @a
1123 // MI0--> %2 = ... %0
1124 // It's not safe to sink %0's def past %1. We currently handle
1125 // this by rejecting all loads.
1126 //
1127 // Example:
1128 // MI1--> %0 = load @a
1129 // %1 = store @a
1130 // MI0--> %2 = ... %0
1131 // It's not safe to sink %0's def past %1. We currently handle
1132 // this by rejecting all loads.
1133 //
1134 // Example:
1135 // G_CONDBR %cond, @BB1
1136 // BB0:
1137 // MI1--> %0 = load @a
1138 // G_BR @BB1
1139 // BB1:
1140 // MI0--> %2 = ... %0
1141 // It's not always safe to sink %0 across control flow. In this
1142 // case it may introduce a memory fault. We currentl handle this
1143 // by rejecting all loads.
1144 }
1145 }
1146
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001147 for (const auto &MA : Actions) {
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001148 MA->emitCxxActionStmts(OS, *this, "I");
Daniel Sandersdc662ff2017-01-26 11:10:14 +00001149 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001150
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001151 OS << " constrainSelectedInstRegOperands(NewI, TII, TRI, RBI);\n";
1152 OS << " return true;\n";
1153 OS << " }\n";
1154 OS << " return false;\n";
1155 OS << " }()) { return true; }\n\n";
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001156}
Daniel Sanders43c882c2017-02-01 10:53:10 +00001157
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001158bool RuleMatcher::isHigherPriorityThan(const RuleMatcher &B) const {
1159 // Rules involving more match roots have higher priority.
1160 if (Matchers.size() > B.Matchers.size())
1161 return true;
1162 if (Matchers.size() < B.Matchers.size())
Daniel Sanders759ff412017-02-24 13:58:11 +00001163 return false;
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001164
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001165 for (const auto &Matcher : zip(Matchers, B.Matchers)) {
1166 if (std::get<0>(Matcher)->isHigherPriorityThan(*std::get<1>(Matcher)))
1167 return true;
1168 if (std::get<1>(Matcher)->isHigherPriorityThan(*std::get<0>(Matcher)))
1169 return false;
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001170 }
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001171
1172 return false;
Simon Pilgrima7d1da82017-03-15 22:50:47 +00001173}
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001174
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001175unsigned RuleMatcher::countTemporaryOperands() const {
1176 return std::accumulate(
1177 Matchers.begin(), Matchers.end(), 0,
1178 [](unsigned A, const std::unique_ptr<InstructionMatcher> &Matcher) {
1179 return A + Matcher->countTemporaryOperands();
1180 });
1181}
1182
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001183//===- GlobalISelEmitter class --------------------------------------------===//
1184
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001185class GlobalISelEmitter {
1186public:
1187 explicit GlobalISelEmitter(RecordKeeper &RK);
1188 void run(raw_ostream &OS);
1189
1190private:
1191 const RecordKeeper &RK;
1192 const CodeGenDAGPatterns CGP;
1193 const CodeGenTarget &Target;
1194
1195 /// Keep track of the equivalence between SDNodes and Instruction.
1196 /// This is defined using 'GINodeEquiv' in the target description.
1197 DenseMap<Record *, const CodeGenInstruction *> NodeEquivs;
1198
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001199 /// Keep track of the equivalence between ComplexPattern's and
1200 /// GIComplexOperandMatcher. Map entries are specified by subclassing
1201 /// GIComplexPatternEquiv.
1202 DenseMap<const Record *, const Record *> ComplexPatternEquivs;
1203
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001204 void gatherNodeEquivs();
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001205 const CodeGenInstruction *findNodeEquiv(Record *N) const;
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001206
Daniel Sandersc270c502017-03-30 09:36:33 +00001207 Error importRulePredicates(RuleMatcher &M, ArrayRef<Init *> Predicates) const;
Daniel Sandersffc7d582017-03-29 15:37:18 +00001208 Expected<InstructionMatcher &>
Daniel Sandersc270c502017-03-30 09:36:33 +00001209 createAndImportSelDAGMatcher(InstructionMatcher &InsnMatcher,
1210 const TreePatternNode *Src) const;
1211 Error importChildMatcher(InstructionMatcher &InsnMatcher,
1212 TreePatternNode *SrcChild, unsigned OpIdx,
1213 unsigned &TempOpIdx) const;
1214 Expected<BuildMIAction &> createAndImportInstructionRenderer(
1215 RuleMatcher &M, const TreePatternNode *Dst,
1216 const InstructionMatcher &InsnMatcher) const;
1217 Error importExplicitUseRenderer(BuildMIAction &DstMIBuilder,
1218 TreePatternNode *DstChild,
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001219 const InstructionMatcher &InsnMatcher) const;
Daniel Sandersc270c502017-03-30 09:36:33 +00001220 Error
Daniel Sandersffc7d582017-03-29 15:37:18 +00001221 importImplicitDefRenderers(BuildMIAction &DstMIBuilder,
1222 const std::vector<Record *> &ImplicitDefs) const;
1223
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001224 /// Analyze pattern \p P, returning a matcher for it if possible.
1225 /// Otherwise, return an Error explaining why we don't support it.
1226 Expected<RuleMatcher> runOnPattern(const PatternToMatch &P);
1227};
1228
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001229void GlobalISelEmitter::gatherNodeEquivs() {
1230 assert(NodeEquivs.empty());
1231 for (Record *Equiv : RK.getAllDerivedDefinitions("GINodeEquiv"))
1232 NodeEquivs[Equiv->getValueAsDef("Node")] =
1233 &Target.getInstruction(Equiv->getValueAsDef("I"));
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001234
1235 assert(ComplexPatternEquivs.empty());
1236 for (Record *Equiv : RK.getAllDerivedDefinitions("GIComplexPatternEquiv")) {
1237 Record *SelDAGEquiv = Equiv->getValueAsDef("SelDAGEquivalent");
1238 if (!SelDAGEquiv)
1239 continue;
1240 ComplexPatternEquivs[SelDAGEquiv] = Equiv;
1241 }
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001242}
1243
Daniel Sandersbdfebb82017-03-15 20:18:38 +00001244const CodeGenInstruction *GlobalISelEmitter::findNodeEquiv(Record *N) const {
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001245 return NodeEquivs.lookup(N);
1246}
1247
1248GlobalISelEmitter::GlobalISelEmitter(RecordKeeper &RK)
1249 : RK(RK), CGP(RK), Target(CGP.getTargetInfo()) {}
1250
1251//===- Emitter ------------------------------------------------------------===//
1252
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001253/// Helper function to let the emitter report skip reason error messages.
1254static Error failedImport(const Twine &Reason) {
1255 return make_error<StringError>(Reason, inconvertibleErrorCode());
1256}
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001257
Daniel Sandersc270c502017-03-30 09:36:33 +00001258Error
Daniel Sandersffc7d582017-03-29 15:37:18 +00001259GlobalISelEmitter::importRulePredicates(RuleMatcher &M,
1260 ArrayRef<Init *> Predicates) const {
1261 if (!Predicates.empty())
1262 return failedImport("Pattern has a predicate");
Daniel Sandersc270c502017-03-30 09:36:33 +00001263 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001264}
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001265
Daniel Sandersc270c502017-03-30 09:36:33 +00001266Expected<InstructionMatcher &> GlobalISelEmitter::createAndImportSelDAGMatcher(
1267 InstructionMatcher &InsnMatcher, const TreePatternNode *Src) const {
Daniel Sandersffc7d582017-03-29 15:37:18 +00001268 // Start with the defined operands (i.e., the results of the root operator).
1269 if (Src->getExtTypes().size() > 1)
1270 return failedImport("Src pattern has multiple results");
1271
1272 auto SrcGIOrNull = findNodeEquiv(Src->getOperator());
1273 if (!SrcGIOrNull)
1274 return failedImport("Pattern operator lacks an equivalent Instruction");
1275 auto &SrcGI = *SrcGIOrNull;
1276
1277 // The operators look good: match the opcode and mutate it to the new one.
1278 InsnMatcher.addPredicate<InstructionOpcodeMatcher>(&SrcGI);
1279
1280 unsigned OpIdx = 0;
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001281 unsigned TempOpIdx = 0;
Daniel Sandersffc7d582017-03-29 15:37:18 +00001282 for (const EEVT::TypeSet &Ty : Src->getExtTypes()) {
1283 auto OpTyOrNone = MVTToLLT(Ty.getConcrete());
1284
1285 if (!OpTyOrNone)
1286 return failedImport(
1287 "Result of Src pattern operator has an unsupported type");
1288
1289 // Results don't have a name unless they are the root node. The caller will
1290 // set the name if appropriate.
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001291 OperandMatcher &OM = InsnMatcher.addOperand(OpIdx++, "", TempOpIdx);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001292 OM.addPredicate<LLTOperandMatcher>(*OpTyOrNone);
1293 }
1294
Daniel Sandersffc7d582017-03-29 15:37:18 +00001295 // Match the used operands (i.e. the children of the operator).
1296 for (unsigned i = 0, e = Src->getNumChildren(); i != e; ++i) {
1297 if (auto Error = importChildMatcher(InsnMatcher, Src->getChild(i), OpIdx++,
Daniel Sandersc270c502017-03-30 09:36:33 +00001298 TempOpIdx))
Daniel Sandersffc7d582017-03-29 15:37:18 +00001299 return std::move(Error);
1300 }
1301
1302 return InsnMatcher;
1303}
1304
Daniel Sandersc270c502017-03-30 09:36:33 +00001305Error GlobalISelEmitter::importChildMatcher(InstructionMatcher &InsnMatcher,
1306 TreePatternNode *SrcChild,
1307 unsigned OpIdx,
1308 unsigned &TempOpIdx) const {
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001309 OperandMatcher &OM =
1310 InsnMatcher.addOperand(OpIdx, SrcChild->getName(), TempOpIdx);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001311
1312 if (SrcChild->hasAnyPredicate())
1313 return failedImport("Src pattern child has predicate");
1314
1315 ArrayRef<EEVT::TypeSet> ChildTypes = SrcChild->getExtTypes();
1316 if (ChildTypes.size() != 1)
1317 return failedImport("Src pattern child has multiple results");
1318
1319 // Check MBB's before the type check since they are not a known type.
1320 if (!SrcChild->isLeaf()) {
1321 if (SrcChild->getOperator()->isSubClassOf("SDNode")) {
1322 auto &ChildSDNI = CGP.getSDNodeInfo(SrcChild->getOperator());
1323 if (ChildSDNI.getSDClassName() == "BasicBlockSDNode") {
1324 OM.addPredicate<MBBOperandMatcher>();
Daniel Sandersc270c502017-03-30 09:36:33 +00001325 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001326 }
1327 }
Daniel Sandersffc7d582017-03-29 15:37:18 +00001328 }
1329
1330 auto OpTyOrNone = MVTToLLT(ChildTypes.front().getConcrete());
1331 if (!OpTyOrNone)
1332 return failedImport("Src operand has an unsupported type");
1333 OM.addPredicate<LLTOperandMatcher>(*OpTyOrNone);
1334
Daniel Sandersbee57392017-04-04 13:25:23 +00001335 // Check for nested instructions.
1336 if (!SrcChild->isLeaf()) {
1337 // Map the node to a gMIR instruction.
1338 InstructionOperandMatcher &InsnOperand =
1339 OM.addPredicate<InstructionOperandMatcher>();
1340 auto InsnMatcherOrError =
1341 createAndImportSelDAGMatcher(InsnOperand.getInsnMatcher(), SrcChild);
1342 if (auto Error = InsnMatcherOrError.takeError())
1343 return Error;
1344
1345 return Error::success();
1346 }
1347
Daniel Sandersffc7d582017-03-29 15:37:18 +00001348 // Check for constant immediates.
1349 if (auto *ChildInt = dyn_cast<IntInit>(SrcChild->getLeafValue())) {
1350 OM.addPredicate<IntOperandMatcher>(ChildInt->getValue());
Daniel Sandersc270c502017-03-30 09:36:33 +00001351 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001352 }
1353
1354 // Check for def's like register classes or ComplexPattern's.
1355 if (auto *ChildDefInit = dyn_cast<DefInit>(SrcChild->getLeafValue())) {
1356 auto *ChildRec = ChildDefInit->getDef();
1357
1358 // Check for register classes.
1359 if (ChildRec->isSubClassOf("RegisterClass")) {
1360 OM.addPredicate<RegisterBankOperandMatcher>(
1361 Target.getRegisterClass(ChildRec));
Daniel Sandersc270c502017-03-30 09:36:33 +00001362 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001363 }
1364
1365 // Check for ComplexPattern's.
1366 if (ChildRec->isSubClassOf("ComplexPattern")) {
1367 const auto &ComplexPattern = ComplexPatternEquivs.find(ChildRec);
1368 if (ComplexPattern == ComplexPatternEquivs.end())
1369 return failedImport(
1370 "SelectionDAG ComplexPattern not mapped to GlobalISel");
1371
1372 const auto &Predicate = OM.addPredicate<ComplexPatternOperandMatcher>(
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001373 OM, *ComplexPattern->second);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001374 TempOpIdx += Predicate.countTemporaryOperands();
Daniel Sandersc270c502017-03-30 09:36:33 +00001375 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001376 }
1377
1378 return failedImport(
1379 "Src pattern child def is an unsupported tablegen class");
1380 }
1381
1382 return failedImport("Src pattern child is an unsupported kind");
1383}
1384
Daniel Sandersc270c502017-03-30 09:36:33 +00001385Error GlobalISelEmitter::importExplicitUseRenderer(
Daniel Sandersffc7d582017-03-29 15:37:18 +00001386 BuildMIAction &DstMIBuilder, TreePatternNode *DstChild,
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001387 const InstructionMatcher &InsnMatcher) const {
Daniel Sandersffc7d582017-03-29 15:37:18 +00001388 // The only non-leaf child we accept is 'bb': it's an operator because
1389 // BasicBlockSDNode isn't inline, but in MI it's just another operand.
1390 if (!DstChild->isLeaf()) {
1391 if (DstChild->getOperator()->isSubClassOf("SDNode")) {
1392 auto &ChildSDNI = CGP.getSDNodeInfo(DstChild->getOperator());
1393 if (ChildSDNI.getSDClassName() == "BasicBlockSDNode") {
1394 DstMIBuilder.addRenderer<CopyRenderer>(InsnMatcher,
1395 DstChild->getName());
Daniel Sandersc270c502017-03-30 09:36:33 +00001396 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001397 }
1398 }
1399 return failedImport("Dst pattern child isn't a leaf node or an MBB");
1400 }
1401
1402 // Otherwise, we're looking for a bog-standard RegisterClass operand.
1403 if (DstChild->hasAnyPredicate())
1404 return failedImport("Dst pattern child has predicate");
1405
1406 if (auto *ChildDefInit = dyn_cast<DefInit>(DstChild->getLeafValue())) {
1407 auto *ChildRec = ChildDefInit->getDef();
1408
1409 ArrayRef<EEVT::TypeSet> ChildTypes = DstChild->getExtTypes();
1410 if (ChildTypes.size() != 1)
1411 return failedImport("Dst pattern child has multiple results");
1412
1413 auto OpTyOrNone = MVTToLLT(ChildTypes.front().getConcrete());
1414 if (!OpTyOrNone)
1415 return failedImport("Dst operand has an unsupported type");
1416
1417 if (ChildRec->isSubClassOf("Register")) {
1418 DstMIBuilder.addRenderer<AddRegisterRenderer>(ChildRec);
Daniel Sandersc270c502017-03-30 09:36:33 +00001419 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001420 }
1421
1422 if (ChildRec->isSubClassOf("RegisterClass")) {
1423 DstMIBuilder.addRenderer<CopyRenderer>(InsnMatcher, DstChild->getName());
Daniel Sandersc270c502017-03-30 09:36:33 +00001424 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001425 }
1426
1427 if (ChildRec->isSubClassOf("ComplexPattern")) {
1428 const auto &ComplexPattern = ComplexPatternEquivs.find(ChildRec);
1429 if (ComplexPattern == ComplexPatternEquivs.end())
1430 return failedImport(
1431 "SelectionDAG ComplexPattern not mapped to GlobalISel");
1432
1433 SmallVector<OperandPlaceholder, 2> RenderedOperands;
Daniel Sanders4f3eb242017-04-05 13:14:03 +00001434 const OperandMatcher &OM = InsnMatcher.getOperand(DstChild->getName());
1435 for (unsigned I = 0; I < OM.countTemporaryOperands(); ++I)
1436 RenderedOperands.push_back(OperandPlaceholder::CreateTemporary(
1437 OM.getAllocatedTemporariesBaseID() + I));
Daniel Sandersffc7d582017-03-29 15:37:18 +00001438 DstMIBuilder.addRenderer<RenderComplexPatternOperand>(
1439 *ComplexPattern->second, RenderedOperands);
Daniel Sandersc270c502017-03-30 09:36:33 +00001440 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001441 }
1442
1443 return failedImport(
1444 "Dst pattern child def is an unsupported tablegen class");
1445 }
1446
1447 return failedImport("Dst pattern child is an unsupported kind");
1448}
1449
Daniel Sandersc270c502017-03-30 09:36:33 +00001450Expected<BuildMIAction &> GlobalISelEmitter::createAndImportInstructionRenderer(
Daniel Sandersffc7d582017-03-29 15:37:18 +00001451 RuleMatcher &M, const TreePatternNode *Dst,
1452 const InstructionMatcher &InsnMatcher) const {
1453 Record *DstOp = Dst->getOperator();
1454 if (!DstOp->isSubClassOf("Instruction"))
1455 return failedImport("Pattern operator isn't an instruction");
1456 auto &DstI = Target.getInstruction(DstOp);
1457
1458 auto &DstMIBuilder = M.addAction<BuildMIAction>(&DstI, InsnMatcher);
1459
1460 // Render the explicit defs.
1461 for (unsigned I = 0; I < DstI.Operands.NumDefs; ++I) {
1462 const auto &DstIOperand = DstI.Operands[I];
1463 DstMIBuilder.addRenderer<CopyRenderer>(InsnMatcher, DstIOperand.Name);
1464 }
1465
Daniel Sanders0ed28822017-04-12 08:23:08 +00001466 // Figure out which operands need defaults inserted. Operands that subclass
1467 // OperandWithDefaultOps are considered from left to right until we have
1468 // enough operands to render the instruction.
1469 SmallSet<unsigned, 2> DefaultOperands;
1470 unsigned DstINumUses = DstI.Operands.size() - DstI.Operands.NumDefs;
1471 unsigned NumDefaultOperands = 0;
1472 for (unsigned I = 0; I < DstINumUses &&
1473 DstINumUses > Dst->getNumChildren() + NumDefaultOperands;
1474 ++I) {
1475 const auto &DstIOperand = DstI.Operands[DstI.Operands.NumDefs + I];
1476 if (DstIOperand.Rec->isSubClassOf("OperandWithDefaultOps")) {
1477 DefaultOperands.insert(I);
1478 NumDefaultOperands +=
1479 DstIOperand.Rec->getValueAsDag("DefaultOps")->getNumArgs();
1480 }
1481 }
1482 if (DstINumUses > Dst->getNumChildren() + DefaultOperands.size())
1483 return failedImport("Insufficient operands supplied and default ops "
1484 "couldn't make up the shortfall");
1485 if (DstINumUses < Dst->getNumChildren() + DefaultOperands.size())
1486 return failedImport("Too many operands supplied");
1487
Daniel Sandersffc7d582017-03-29 15:37:18 +00001488 // Render the explicit uses.
Daniel Sanders0ed28822017-04-12 08:23:08 +00001489 unsigned Child = 0;
1490 for (unsigned I = 0; I != DstINumUses; ++I) {
1491 // If we need to insert default ops here, then do so.
1492 if (DefaultOperands.count(I)) {
1493 const auto &DstIOperand = DstI.Operands[DstI.Operands.NumDefs + I];
1494
1495 DagInit *DefaultOps = DstIOperand.Rec->getValueAsDag("DefaultOps");
1496 for (const auto *DefaultOp : DefaultOps->args()) {
1497 // Look through ValueType operators.
1498 if (const DagInit *DefaultDagOp = dyn_cast<DagInit>(DefaultOp)) {
1499 if (const DefInit *DefaultDagOperator =
1500 dyn_cast<DefInit>(DefaultDagOp->getOperator())) {
1501 if (DefaultDagOperator->getDef()->isSubClassOf("ValueType"))
1502 DefaultOp = DefaultDagOp->getArg(0);
1503 }
1504 }
1505
1506 if (const DefInit *DefaultDefOp = dyn_cast<DefInit>(DefaultOp)) {
1507 DstMIBuilder.addRenderer<AddRegisterRenderer>(DefaultDefOp->getDef());
1508 continue;
1509 }
1510
1511 if (const IntInit *DefaultIntOp = dyn_cast<IntInit>(DefaultOp)) {
1512 DstMIBuilder.addRenderer<ImmRenderer>(DefaultIntOp->getValue());
1513 continue;
1514 }
1515
1516 return failedImport("Could not add default op");
1517 }
1518
1519 continue;
1520 }
1521
1522 if (auto Error = importExplicitUseRenderer(
1523 DstMIBuilder, Dst->getChild(Child), InsnMatcher))
Daniel Sandersffc7d582017-03-29 15:37:18 +00001524 return std::move(Error);
Daniel Sanders0ed28822017-04-12 08:23:08 +00001525 ++Child;
Daniel Sandersffc7d582017-03-29 15:37:18 +00001526 }
1527
1528 return DstMIBuilder;
1529}
1530
Daniel Sandersc270c502017-03-30 09:36:33 +00001531Error GlobalISelEmitter::importImplicitDefRenderers(
Daniel Sandersffc7d582017-03-29 15:37:18 +00001532 BuildMIAction &DstMIBuilder,
1533 const std::vector<Record *> &ImplicitDefs) const {
1534 if (!ImplicitDefs.empty())
1535 return failedImport("Pattern defines a physical register");
Daniel Sandersc270c502017-03-30 09:36:33 +00001536 return Error::success();
Daniel Sandersffc7d582017-03-29 15:37:18 +00001537}
1538
1539Expected<RuleMatcher> GlobalISelEmitter::runOnPattern(const PatternToMatch &P) {
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001540 // Keep track of the matchers and actions to emit.
Ahmed Bougacha9aa4c102017-02-04 00:47:08 +00001541 RuleMatcher M;
1542 M.addAction<DebugCommentAction>(P);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001543
Daniel Sandersc270c502017-03-30 09:36:33 +00001544 if (auto Error = importRulePredicates(M, P.getPredicates()->getValues()))
Daniel Sandersffc7d582017-03-29 15:37:18 +00001545 return std::move(Error);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001546
1547 // Next, analyze the pattern operators.
1548 TreePatternNode *Src = P.getSrcPattern();
1549 TreePatternNode *Dst = P.getDstPattern();
1550
1551 // If the root of either pattern isn't a simple operator, ignore it.
1552 if (!isTrivialOperatorNode(Dst))
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001553 return failedImport("Dst pattern root isn't a trivial operator");
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001554 if (!isTrivialOperatorNode(Src))
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001555 return failedImport("Src pattern root isn't a trivial operator");
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001556
Daniel Sandersbee57392017-04-04 13:25:23 +00001557 // Start with the defined operands (i.e., the results of the root operator).
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001558 Record *DstOp = Dst->getOperator();
1559 if (!DstOp->isSubClassOf("Instruction"))
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001560 return failedImport("Pattern operator isn't an instruction");
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001561
1562 auto &DstI = Target.getInstruction(DstOp);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001563 if (DstI.Operands.NumDefs != Src->getExtTypes().size())
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001564 return failedImport("Src pattern results and dst MI defs are different");
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001565
Daniel Sandersffc7d582017-03-29 15:37:18 +00001566 InstructionMatcher &InsnMatcherTemp = M.addInstructionMatcher();
Daniel Sandersc270c502017-03-30 09:36:33 +00001567 auto InsnMatcherOrError = createAndImportSelDAGMatcher(InsnMatcherTemp, Src);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001568 if (auto Error = InsnMatcherOrError.takeError())
1569 return std::move(Error);
1570 InstructionMatcher &InsnMatcher = InsnMatcherOrError.get();
1571
1572 // The root of the match also has constraints on the register bank so that it
1573 // matches the result instruction.
1574 unsigned OpIdx = 0;
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001575 for (const EEVT::TypeSet &Ty : Src->getExtTypes()) {
Daniel Sandersffc7d582017-03-29 15:37:18 +00001576 (void)Ty;
1577
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001578 const auto &DstIOperand = DstI.Operands[OpIdx];
1579 Record *DstIOpRec = DstIOperand.Rec;
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001580 if (!DstIOpRec->isSubClassOf("RegisterClass"))
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001581 return failedImport("Dst MI def isn't a register class");
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001582
Daniel Sandersffc7d582017-03-29 15:37:18 +00001583 OperandMatcher &OM = InsnMatcher.getOperand(OpIdx);
1584 OM.setSymbolicName(DstIOperand.Name);
Daniel Sandersdc662ff2017-01-26 11:10:14 +00001585 OM.addPredicate<RegisterBankOperandMatcher>(
1586 Target.getRegisterClass(DstIOpRec));
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001587 ++OpIdx;
1588 }
1589
Daniel Sandersc270c502017-03-30 09:36:33 +00001590 auto DstMIBuilderOrError =
1591 createAndImportInstructionRenderer(M, Dst, InsnMatcher);
Daniel Sandersffc7d582017-03-29 15:37:18 +00001592 if (auto Error = DstMIBuilderOrError.takeError())
1593 return std::move(Error);
1594 BuildMIAction &DstMIBuilder = DstMIBuilderOrError.get();
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001595
Daniel Sandersffc7d582017-03-29 15:37:18 +00001596 // Render the implicit defs.
1597 // These are only added to the root of the result.
Daniel Sandersc270c502017-03-30 09:36:33 +00001598 if (auto Error = importImplicitDefRenderers(DstMIBuilder, P.getDstRegs()))
Daniel Sandersffc7d582017-03-29 15:37:18 +00001599 return std::move(Error);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001600
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001601 // We're done with this pattern! It's eligible for GISel emission; return it.
Daniel Sandersb41ce2b2017-02-20 14:31:27 +00001602 ++NumPatternImported;
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001603 return std::move(M);
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001604}
1605
1606void GlobalISelEmitter::run(raw_ostream &OS) {
1607 // Track the GINodeEquiv definitions.
1608 gatherNodeEquivs();
1609
1610 emitSourceFileHeader(("Global Instruction Selector for the " +
1611 Target.getName() + " target").str(), OS);
Daniel Sandersb41ce2b2017-02-20 14:31:27 +00001612 std::vector<RuleMatcher> Rules;
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001613 // Look through the SelectionDAG patterns we found, possibly emitting some.
1614 for (const PatternToMatch &Pat : CGP.ptms()) {
1615 ++NumPatternTotal;
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001616 auto MatcherOrErr = runOnPattern(Pat);
1617
1618 // The pattern analysis can fail, indicating an unsupported pattern.
1619 // Report that if we've been asked to do so.
1620 if (auto Err = MatcherOrErr.takeError()) {
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001621 if (WarnOnSkippedPatterns) {
1622 PrintWarning(Pat.getSrcRecord()->getLoc(),
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001623 "Skipped pattern: " + toString(std::move(Err)));
1624 } else {
1625 consumeError(std::move(Err));
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001626 }
Daniel Sandersb41ce2b2017-02-20 14:31:27 +00001627 ++NumPatternImportsSkipped;
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001628 continue;
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001629 }
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001630
Daniel Sandersb41ce2b2017-02-20 14:31:27 +00001631 Rules.push_back(std::move(MatcherOrErr.get()));
1632 }
1633
Daniel Sanders066ebbf2017-02-24 15:43:30 +00001634 std::stable_sort(Rules.begin(), Rules.end(),
1635 [&](const RuleMatcher &A, const RuleMatcher &B) {
Daniel Sanders759ff412017-02-24 13:58:11 +00001636 if (A.isHigherPriorityThan(B)) {
1637 assert(!B.isHigherPriorityThan(A) && "Cannot be more important "
1638 "and less important at "
1639 "the same time");
1640 return true;
1641 }
1642 return false;
1643 });
1644
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001645 unsigned MaxTemporaries = 0;
1646 for (const auto &Rule : Rules)
1647 MaxTemporaries = std::max(MaxTemporaries, Rule.countTemporaryOperands());
1648
1649 OS << "#ifdef GET_GLOBALISEL_TEMPORARIES_DECL\n";
1650 for (unsigned I = 0; I < MaxTemporaries; ++I)
1651 OS << " mutable MachineOperand TempOp" << I << ";\n";
1652 OS << "#endif // ifdef GET_GLOBALISEL_TEMPORARIES_DECL\n\n";
1653
1654 OS << "#ifdef GET_GLOBALISEL_TEMPORARIES_INIT\n";
1655 for (unsigned I = 0; I < MaxTemporaries; ++I)
1656 OS << ", TempOp" << I << "(MachineOperand::CreatePlaceholder())\n";
1657 OS << "#endif // ifdef GET_GLOBALISEL_TEMPORARIES_INIT\n\n";
1658
1659 OS << "#ifdef GET_GLOBALISEL_IMPL\n"
1660 << "bool " << Target.getName()
1661 << "InstructionSelector::selectImpl(MachineInstr &I) const {\n"
1662 << " MachineFunction &MF = *I.getParent()->getParent();\n"
1663 << " const MachineRegisterInfo &MRI = MF.getRegInfo();\n";
1664
Daniel Sandersb96f40d2017-03-20 15:20:42 +00001665 for (auto &Rule : Rules) {
Daniel Sandersb41ce2b2017-02-20 14:31:27 +00001666 Rule.emit(OS);
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001667 ++NumPatternEmitted;
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001668 }
1669
Daniel Sanders8a4bae92017-03-14 21:32:08 +00001670 OS << " return false;\n"
1671 << "}\n"
1672 << "#endif // ifdef GET_GLOBALISEL_IMPL\n";
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001673}
1674
Ahmed Bougacha982c5eb2017-02-10 04:00:17 +00001675} // end anonymous namespace
1676
Ahmed Bougacha36f70352016-12-21 23:26:20 +00001677//===----------------------------------------------------------------------===//
1678
1679namespace llvm {
1680void EmitGlobalISel(RecordKeeper &RK, raw_ostream &OS) {
1681 GlobalISelEmitter(RK).run(OS);
1682}
1683} // End llvm namespace