MLIR  20.0.0git
Pass.cpp
Go to the documentation of this file.
1 //===- Pass.cpp - Pass infrastructure implementation ----------------------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 //
9 // This file implements common pass infrastructure.
10 //
11 //===----------------------------------------------------------------------===//
12 
13 #include "mlir/Pass/Pass.h"
14 #include "PassDetail.h"
15 #include "mlir/IR/Diagnostics.h"
16 #include "mlir/IR/Dialect.h"
17 #include "mlir/IR/OpDefinition.h"
18 #include "mlir/IR/Threading.h"
19 #include "mlir/IR/Verifier.h"
21 #include "llvm/ADT/Hashing.h"
22 #include "llvm/ADT/STLExtras.h"
23 #include "llvm/ADT/ScopeExit.h"
24 #include "llvm/Support/CommandLine.h"
25 #include "llvm/Support/CrashRecoveryContext.h"
26 #include "llvm/Support/Mutex.h"
27 #include "llvm/Support/Signals.h"
28 #include "llvm/Support/Threading.h"
29 #include "llvm/Support/ToolOutputFile.h"
30 #include <optional>
31 
32 using namespace mlir;
33 using namespace mlir::detail;
34 
35 //===----------------------------------------------------------------------===//
36 // PassExecutionAction
37 //===----------------------------------------------------------------------===//
38 
40  const Pass &pass)
41  : Base(irUnits), pass(pass) {}
42 
43 void PassExecutionAction::print(raw_ostream &os) const {
44  os << llvm::formatv("`{0}` running `{1}` on Operation `{2}`", tag,
45  pass.getName(), getOp()->getName());
46 }
47 
50  return irUnits.empty() ? nullptr
51  : llvm::dyn_cast_if_present<Operation *>(irUnits[0]);
52 }
53 
54 //===----------------------------------------------------------------------===//
55 // Pass
56 //===----------------------------------------------------------------------===//
57 
58 /// Out of line virtual method to ensure vtables and metadata are emitted to a
59 /// single .o file.
60 void Pass::anchor() {}
61 
62 /// Attempt to initialize the options of this pass from the given string.
64  StringRef options,
65  function_ref<LogicalResult(const Twine &)> errorHandler) {
66  std::string errStr;
67  llvm::raw_string_ostream os(errStr);
68  if (failed(passOptions.parseFromString(options, os))) {
69  os.flush();
70  return errorHandler(errStr);
71  }
72  return success();
73 }
74 
75 /// Copy the option values from 'other', which is another instance of this
76 /// pass.
77 void Pass::copyOptionValuesFrom(const Pass *other) {
78  passOptions.copyOptionValuesFrom(other->passOptions);
79 }
80 
81 /// Prints out the pass in the textual representation of pipelines. If this is
82 /// an adaptor pass, print its pass managers.
83 void Pass::printAsTextualPipeline(raw_ostream &os) {
84  // Special case for adaptors to print its pass managers.
85  if (auto *adaptor = dyn_cast<OpToOpPassAdaptor>(this)) {
86  llvm::interleave(
87  adaptor->getPassManagers(),
88  [&](OpPassManager &pm) { pm.printAsTextualPipeline(os); },
89  [&] { os << ","; });
90  return;
91  }
92  // Otherwise, print the pass argument followed by its options. If the pass
93  // doesn't have an argument, print the name of the pass to give some indicator
94  // of what pass was run.
95  StringRef argument = getArgument();
96  if (!argument.empty())
97  os << argument;
98  else
99  os << "unknown<" << getName() << ">";
100  passOptions.print(os);
101 }
102 
103 //===----------------------------------------------------------------------===//
104 // OpPassManagerImpl
105 //===----------------------------------------------------------------------===//
106 
107 namespace mlir {
108 namespace detail {
111  : name(opName.getStringRef().str()), opName(opName),
114  : name(name == OpPassManager::getAnyOpAnchorName() ? "" : name.str()),
119  : name(rhs.name), opName(rhs.opName),
121  nesting(rhs.nesting) {
122  for (const std::unique_ptr<Pass> &pass : rhs.passes) {
123  std::unique_ptr<Pass> newPass = pass->clone();
124  newPass->threadingSibling = pass.get();
125  passes.push_back(std::move(newPass));
126  }
127  }
128 
129  /// Merge the passes of this pass manager into the one provided.
130  void mergeInto(OpPassManagerImpl &rhs);
131 
132  /// Nest a new operation pass manager for the given operation kind under this
133  /// pass manager.
135  return nest(OpPassManager(nestedName, nesting));
136  }
137  OpPassManager &nest(StringRef nestedName) {
138  return nest(OpPassManager(nestedName, nesting));
139  }
141 
142  /// Nest the given pass manager under this pass manager.
143  OpPassManager &nest(OpPassManager &&nested);
144 
145  /// Add the given pass to this pass manager. If this pass has a concrete
146  /// operation type, it must be the same type as this pass manager.
147  void addPass(std::unique_ptr<Pass> pass);
148 
149  /// Clear the list of passes in this pass manager, other options are
150  /// preserved.
151  void clear();
152 
153  /// Finalize the pass list in preparation for execution. This includes
154  /// coalescing adjacent pass managers when possible, verifying scheduled
155  /// passes, etc.
156  LogicalResult finalizePassList(MLIRContext *ctx);
157 
158  /// Return the operation name of this pass manager.
159  std::optional<OperationName> getOpName(MLIRContext &context) {
160  if (!name.empty() && !opName)
161  opName = OperationName(name, &context);
162  return opName;
163  }
164  std::optional<StringRef> getOpName() const {
165  return name.empty() ? std::optional<StringRef>()
166  : std::optional<StringRef>(name);
167  }
168 
169  /// Return the name used to anchor this pass manager. This is either the name
170  /// of an operation, or the result of `getAnyOpAnchorName()` in the case of an
171  /// op-agnostic pass manager.
172  StringRef getOpAnchorName() const {
173  return getOpName().value_or(OpPassManager::getAnyOpAnchorName());
174  }
175 
176  /// Indicate if the current pass manager can be scheduled on the given
177  /// operation type.
179 
180  /// The name of the operation that passes of this pass manager operate on.
181  std::string name;
182 
183  /// The cached OperationName (internalized in the context) for the name of the
184  /// operation that passes of this pass manager operate on.
185  std::optional<OperationName> opName;
186 
187  /// The set of passes to run as part of this pass manager.
188  std::vector<std::unique_ptr<Pass>> passes;
189 
190  /// The current initialization generation of this pass manager. This is used
191  /// to indicate when a pass manager should be reinitialized.
193 
194  /// Control the implicit nesting of passes that mismatch the name set for this
195  /// OpPassManager.
197 };
198 } // namespace detail
199 } // namespace mlir
200 
202  assert(name == rhs.name && "merging unrelated pass managers");
203  for (auto &pass : passes)
204  rhs.passes.push_back(std::move(pass));
205  passes.clear();
206 }
207 
208 OpPassManager &OpPassManagerImpl::nest(OpPassManager &&nested) {
209  auto *adaptor = new OpToOpPassAdaptor(std::move(nested));
210  addPass(std::unique_ptr<Pass>(adaptor));
211  return adaptor->getPassManagers().front();
212 }
213 
214 void OpPassManagerImpl::addPass(std::unique_ptr<Pass> pass) {
215  // If this pass runs on a different operation than this pass manager, then
216  // implicitly nest a pass manager for this operation if enabled.
217  std::optional<StringRef> pmOpName = getOpName();
218  std::optional<StringRef> passOpName = pass->getOpName();
219  if (pmOpName && passOpName && *pmOpName != *passOpName) {
220  if (nesting == OpPassManager::Nesting::Implicit)
221  return nest(*passOpName).addPass(std::move(pass));
222  llvm::report_fatal_error(llvm::Twine("Can't add pass '") + pass->getName() +
223  "' restricted to '" + *passOpName +
224  "' on a PassManager intended to run on '" +
225  getOpAnchorName() + "', did you intend to nest?");
226  }
227 
228  passes.emplace_back(std::move(pass));
229 }
230 
231 void OpPassManagerImpl::clear() { passes.clear(); }
232 
233 LogicalResult OpPassManagerImpl::finalizePassList(MLIRContext *ctx) {
234  auto finalizeAdaptor = [ctx](OpToOpPassAdaptor *adaptor) {
235  for (auto &pm : adaptor->getPassManagers())
236  if (failed(pm.getImpl().finalizePassList(ctx)))
237  return failure();
238  return success();
239  };
240 
241  // Walk the pass list and merge adjacent adaptors.
242  OpToOpPassAdaptor *lastAdaptor = nullptr;
243  for (auto &pass : passes) {
244  // Check to see if this pass is an adaptor.
245  if (auto *currentAdaptor = dyn_cast<OpToOpPassAdaptor>(pass.get())) {
246  // If it is the first adaptor in a possible chain, remember it and
247  // continue.
248  if (!lastAdaptor) {
249  lastAdaptor = currentAdaptor;
250  continue;
251  }
252 
253  // Otherwise, try to merge into the existing adaptor and delete the
254  // current one. If merging fails, just remember this as the last adaptor.
255  if (succeeded(currentAdaptor->tryMergeInto(ctx, *lastAdaptor)))
256  pass.reset();
257  else
258  lastAdaptor = currentAdaptor;
259  } else if (lastAdaptor) {
260  // If this pass isn't an adaptor, finalize it and forget the last adaptor.
261  if (failed(finalizeAdaptor(lastAdaptor)))
262  return failure();
263  lastAdaptor = nullptr;
264  }
265  }
266 
267  // If there was an adaptor at the end of the manager, finalize it as well.
268  if (lastAdaptor && failed(finalizeAdaptor(lastAdaptor)))
269  return failure();
270 
271  // Now that the adaptors have been merged, erase any empty slots corresponding
272  // to the merged adaptors that were nulled-out in the loop above.
273  llvm::erase_if(passes, std::logical_not<std::unique_ptr<Pass>>());
274 
275  // If this is a op-agnostic pass manager, there is nothing left to do.
276  std::optional<OperationName> rawOpName = getOpName(*ctx);
277  if (!rawOpName)
278  return success();
279 
280  // Otherwise, verify that all of the passes are valid for the current
281  // operation anchor.
282  std::optional<RegisteredOperationName> opName =
283  rawOpName->getRegisteredInfo();
284  for (std::unique_ptr<Pass> &pass : passes) {
285  if (opName && !pass->canScheduleOn(*opName)) {
286  return emitError(UnknownLoc::get(ctx))
287  << "unable to schedule pass '" << pass->getName()
288  << "' on a PassManager intended to run on '" << getOpAnchorName()
289  << "'!";
290  }
291  }
292  return success();
293 }
294 
295 bool OpPassManagerImpl::canScheduleOn(MLIRContext &context,
296  OperationName opName) {
297  // If this pass manager is op-specific, we simply check if the provided
298  // operation name is the same as this one.
299  std::optional<OperationName> pmOpName = getOpName(context);
300  if (pmOpName)
301  return pmOpName == opName;
302 
303  // Otherwise, this is an op-agnostic pass manager. Check that the operation
304  // can be scheduled on all passes within the manager.
305  std::optional<RegisteredOperationName> registeredInfo =
306  opName.getRegisteredInfo();
307  if (!registeredInfo ||
308  !registeredInfo->hasTrait<OpTrait::IsIsolatedFromAbove>())
309  return false;
310  return llvm::all_of(passes, [&](const std::unique_ptr<Pass> &pass) {
311  return pass->canScheduleOn(*registeredInfo);
312  });
313 }
314 
315 //===----------------------------------------------------------------------===//
316 // OpPassManager
317 //===----------------------------------------------------------------------===//
318 
320  : impl(new OpPassManagerImpl(nesting)) {}
321 OpPassManager::OpPassManager(StringRef name, Nesting nesting)
322  : impl(new OpPassManagerImpl(name, nesting)) {}
324  : impl(new OpPassManagerImpl(name, nesting)) {}
325 OpPassManager::OpPassManager(OpPassManager &&rhs) { *this = std::move(rhs); }
326 OpPassManager::OpPassManager(const OpPassManager &rhs) { *this = rhs; }
328  impl = std::make_unique<OpPassManagerImpl>(*rhs.impl);
329  return *this;
330 }
332  impl = std::move(rhs.impl);
333  return *this;
334 }
335 
337 
339  return MutableArrayRef<std::unique_ptr<Pass>>{impl->passes}.begin();
340 }
342  return MutableArrayRef<std::unique_ptr<Pass>>{impl->passes}.end();
343 }
344 
346  return ArrayRef<std::unique_ptr<Pass>>{impl->passes}.begin();
347 }
349  return ArrayRef<std::unique_ptr<Pass>>{impl->passes}.end();
350 }
351 
352 /// Nest a new operation pass manager for the given operation kind under this
353 /// pass manager.
355  return impl->nest(nestedName);
356 }
357 OpPassManager &OpPassManager::nest(StringRef nestedName) {
358  return impl->nest(nestedName);
359 }
360 OpPassManager &OpPassManager::nestAny() { return impl->nestAny(); }
361 
362 /// Add the given pass to this pass manager. If this pass has a concrete
363 /// operation type, it must be the same type as this pass manager.
364 void OpPassManager::addPass(std::unique_ptr<Pass> pass) {
365  impl->addPass(std::move(pass));
366 }
367 
368 void OpPassManager::clear() { impl->clear(); }
369 
370 /// Returns the number of passes held by this manager.
371 size_t OpPassManager::size() const { return impl->passes.size(); }
372 
373 /// Returns the internal implementation instance.
375 
376 /// Return the operation name that this pass manager operates on.
377 std::optional<StringRef> OpPassManager::getOpName() const {
378  return impl->getOpName();
379 }
380 
381 /// Return the operation name that this pass manager operates on.
382 std::optional<OperationName>
384  return impl->getOpName(context);
385 }
386 
388  return impl->getOpAnchorName();
389 }
390 
391 /// Prints out the passes of the pass manager as the textual representation
392 /// of pipelines.
394  raw_ostream &os, StringRef anchorName,
396  os << anchorName << "(";
397  llvm::interleave(
398  passes, [&](mlir::Pass &pass) { pass.printAsTextualPipeline(os); },
399  [&]() { os << ","; });
400  os << ")";
401 }
402 void OpPassManager::printAsTextualPipeline(raw_ostream &os) const {
403  StringRef anchorName = getOpAnchorName();
405  os, anchorName,
406  {MutableArrayRef<std::unique_ptr<Pass>>{impl->passes}.begin(),
407  MutableArrayRef<std::unique_ptr<Pass>>{impl->passes}.end()});
408 }
409 
411  llvm::errs() << "Pass Manager with " << impl->passes.size() << " passes:\n";
412  printAsTextualPipeline(llvm::errs());
413  llvm::errs() << "\n";
414 }
415 
417  DialectRegistry &dialects) {
418  for (const Pass &pass : pm.getPasses())
419  pass.getDependentDialects(dialects);
420 }
421 
423  registerDialectsForPipeline(*this, dialects);
424 }
425 
426 void OpPassManager::setNesting(Nesting nesting) { impl->nesting = nesting; }
427 
429 
430 LogicalResult OpPassManager::initialize(MLIRContext *context,
431  unsigned newInitGeneration) {
432  if (impl->initializationGeneration == newInitGeneration)
433  return success();
434  impl->initializationGeneration = newInitGeneration;
435  for (Pass &pass : getPasses()) {
436  // If this pass isn't an adaptor, directly initialize it.
437  auto *adaptor = dyn_cast<OpToOpPassAdaptor>(&pass);
438  if (!adaptor) {
439  if (failed(pass.initialize(context)))
440  return failure();
441  continue;
442  }
443 
444  // Otherwise, initialize each of the adaptors pass managers.
445  for (OpPassManager &adaptorPM : adaptor->getPassManagers())
446  if (failed(adaptorPM.initialize(context, newInitGeneration)))
447  return failure();
448  }
449  return success();
450 }
451 
452 llvm::hash_code OpPassManager::hash() {
453  llvm::hash_code hashCode{};
454  for (Pass &pass : getPasses()) {
455  // If this pass isn't an adaptor, directly hash it.
456  auto *adaptor = dyn_cast<OpToOpPassAdaptor>(&pass);
457  if (!adaptor) {
458  hashCode = llvm::hash_combine(hashCode, &pass);
459  continue;
460  }
461  // Otherwise, hash recursively each of the adaptors pass managers.
462  for (OpPassManager &adaptorPM : adaptor->getPassManagers())
463  llvm::hash_combine(hashCode, adaptorPM.hash());
464  }
465  return hashCode;
466 }
467 
468 
469 //===----------------------------------------------------------------------===//
470 // OpToOpPassAdaptor
471 //===----------------------------------------------------------------------===//
472 
473 LogicalResult OpToOpPassAdaptor::run(Pass *pass, Operation *op,
474  AnalysisManager am, bool verifyPasses,
475  unsigned parentInitGeneration) {
476  std::optional<RegisteredOperationName> opInfo = op->getRegisteredInfo();
477  if (!opInfo)
478  return op->emitOpError()
479  << "trying to schedule a pass on an unregistered operation";
480  if (!opInfo->hasTrait<OpTrait::IsIsolatedFromAbove>())
481  return op->emitOpError() << "trying to schedule a pass on an operation not "
482  "marked as 'IsolatedFromAbove'";
483  if (!pass->canScheduleOn(*op->getName().getRegisteredInfo()))
484  return op->emitOpError()
485  << "trying to schedule a pass on an unsupported operation";
486 
487  // Initialize the pass state with a callback for the pass to dynamically
488  // execute a pipeline on the currently visited operation.
490  PassInstrumentation::PipelineParentInfo parentInfo = {llvm::get_threadid(),
491  pass};
492  auto dynamicPipelineCallback = [&](OpPassManager &pipeline,
493  Operation *root) -> LogicalResult {
494  if (!op->isAncestor(root))
495  return root->emitOpError()
496  << "Trying to schedule a dynamic pipeline on an "
497  "operation that isn't "
498  "nested under the current operation the pass is processing";
499  assert(
500  pipeline.getImpl().canScheduleOn(*op->getContext(), root->getName()));
501 
502  // Before running, finalize the passes held by the pipeline.
503  if (failed(pipeline.getImpl().finalizePassList(root->getContext())))
504  return failure();
505 
506  // Initialize the user provided pipeline and execute the pipeline.
507  if (failed(pipeline.initialize(root->getContext(), parentInitGeneration)))
508  return failure();
509  AnalysisManager nestedAm = root == op ? am : am.nest(root);
510  return OpToOpPassAdaptor::runPipeline(pipeline, root, nestedAm,
511  verifyPasses, parentInitGeneration,
512  pi, &parentInfo);
513  };
514  pass->passState.emplace(op, am, dynamicPipelineCallback);
515 
516  // Instrument before the pass has run.
517  if (pi)
518  pi->runBeforePass(pass, op);
519 
520  bool passFailed = false;
522  [&]() {
523  // Invoke the virtual runOnOperation method.
524  if (auto *adaptor = dyn_cast<OpToOpPassAdaptor>(pass))
525  adaptor->runOnOperation(verifyPasses);
526  else
527  pass->runOnOperation();
528  passFailed = pass->passState->irAndPassFailed.getInt();
529  },
530  {op}, *pass);
531 
532  // Invalidate any non preserved analyses.
533  am.invalidate(pass->passState->preservedAnalyses);
534 
535  // When verifyPasses is specified, we run the verifier (unless the pass
536  // failed).
537  if (!passFailed && verifyPasses) {
538  bool runVerifierNow = true;
539 
540  // If the pass is an adaptor pass, we don't run the verifier recursively
541  // because the nested operations should have already been verified after
542  // nested passes had run.
543  bool runVerifierRecursively = !isa<OpToOpPassAdaptor>(pass);
544 
545  // Reduce compile time by avoiding running the verifier if the pass didn't
546  // change the IR since the last time the verifier was run:
547  //
548  // 1) If the pass said that it preserved all analyses then it can't have
549  // permuted the IR.
550  //
551  // We run these checks in EXPENSIVE_CHECKS mode out of caution.
552 #ifndef EXPENSIVE_CHECKS
553  runVerifierNow = !pass->passState->preservedAnalyses.isAll();
554 #endif
555  if (runVerifierNow)
556  passFailed = failed(verify(op, runVerifierRecursively));
557  }
558 
559  // Instrument after the pass has run.
560  if (pi) {
561  if (passFailed)
562  pi->runAfterPassFailed(pass, op);
563  else
564  pi->runAfterPass(pass, op);
565  }
566 
567  // Return if the pass signaled a failure.
568  return failure(passFailed);
569 }
570 
571 /// Run the given operation and analysis manager on a provided op pass manager.
572 LogicalResult OpToOpPassAdaptor::runPipeline(
573  OpPassManager &pm, Operation *op, AnalysisManager am, bool verifyPasses,
574  unsigned parentInitGeneration, PassInstrumentor *instrumentor,
575  const PassInstrumentation::PipelineParentInfo *parentInfo) {
576  assert((!instrumentor || parentInfo) &&
577  "expected parent info if instrumentor is provided");
578  auto scopeExit = llvm::make_scope_exit([&] {
579  // Clear out any computed operation analyses. These analyses won't be used
580  // any more in this pipeline, and this helps reduce the current working set
581  // of memory. If preserving these analyses becomes important in the future
582  // we can re-evaluate this.
583  am.clear();
584  });
585 
586  // Run the pipeline over the provided operation.
587  if (instrumentor) {
588  instrumentor->runBeforePipeline(pm.getOpName(*op->getContext()),
589  *parentInfo);
590  }
591 
592  for (Pass &pass : pm.getPasses())
593  if (failed(run(&pass, op, am, verifyPasses, parentInitGeneration)))
594  return failure();
595 
596  if (instrumentor) {
597  instrumentor->runAfterPipeline(pm.getOpName(*op->getContext()),
598  *parentInfo);
599  }
600  return success();
601 }
602 
603 /// Find an operation pass manager with the given anchor name, or nullptr if one
604 /// does not exist.
605 static OpPassManager *
607  auto *it = llvm::find_if(
608  mgrs, [&](OpPassManager &mgr) { return mgr.getOpAnchorName() == name; });
609  return it == mgrs.end() ? nullptr : &*it;
610 }
611 
612 /// Find an operation pass manager that can operate on an operation of the given
613 /// type, or nullptr if one does not exist.
615  OperationName name,
616  MLIRContext &context) {
617  auto *it = llvm::find_if(mgrs, [&](OpPassManager &mgr) {
618  return mgr.getImpl().canScheduleOn(context, name);
619  });
620  return it == mgrs.end() ? nullptr : &*it;
621 }
622 
623 OpToOpPassAdaptor::OpToOpPassAdaptor(OpPassManager &&mgr) {
624  mgrs.emplace_back(std::move(mgr));
625 }
626 
627 void OpToOpPassAdaptor::getDependentDialects(DialectRegistry &dialects) const {
628  for (auto &pm : mgrs)
629  pm.getDependentDialects(dialects);
630 }
631 
632 LogicalResult OpToOpPassAdaptor::tryMergeInto(MLIRContext *ctx,
633  OpToOpPassAdaptor &rhs) {
634  // Functor used to check if a pass manager is generic, i.e. op-agnostic.
635  auto isGenericPM = [&](OpPassManager &pm) { return !pm.getOpName(); };
636 
637  // Functor used to detect if the given generic pass manager will have a
638  // potential schedule conflict with the given `otherPMs`.
639  auto hasScheduleConflictWith = [&](OpPassManager &genericPM,
641  return llvm::any_of(otherPMs, [&](OpPassManager &pm) {
642  // If this is a non-generic pass manager, a conflict will arise if a
643  // non-generic pass manager's operation name can be scheduled on the
644  // generic passmanager.
645  if (std::optional<OperationName> pmOpName = pm.getOpName(*ctx))
646  return genericPM.getImpl().canScheduleOn(*ctx, *pmOpName);
647  // Otherwise, this is a generic pass manager. We current can't determine
648  // when generic pass managers can be merged, so conservatively assume they
649  // conflict.
650  return true;
651  });
652  };
653 
654  // Check that if either adaptor has a generic pass manager, that pm is
655  // compatible within any non-generic pass managers.
656  //
657  // Check the current adaptor.
658  auto *lhsGenericPMIt = llvm::find_if(mgrs, isGenericPM);
659  if (lhsGenericPMIt != mgrs.end() &&
660  hasScheduleConflictWith(*lhsGenericPMIt, rhs.mgrs))
661  return failure();
662  // Check the rhs adaptor.
663  auto *rhsGenericPMIt = llvm::find_if(rhs.mgrs, isGenericPM);
664  if (rhsGenericPMIt != rhs.mgrs.end() &&
665  hasScheduleConflictWith(*rhsGenericPMIt, mgrs))
666  return failure();
667 
668  for (auto &pm : mgrs) {
669  // If an existing pass manager exists, then merge the given pass manager
670  // into it.
671  if (auto *existingPM =
672  findPassManagerWithAnchor(rhs.mgrs, pm.getOpAnchorName())) {
673  pm.getImpl().mergeInto(existingPM->getImpl());
674  } else {
675  // Otherwise, add the given pass manager to the list.
676  rhs.mgrs.emplace_back(std::move(pm));
677  }
678  }
679  mgrs.clear();
680 
681  // After coalescing, sort the pass managers within rhs by name.
682  auto compareFn = [](const OpPassManager *lhs, const OpPassManager *rhs) {
683  // Order op-specific pass managers first and op-agnostic pass managers last.
684  if (std::optional<StringRef> lhsName = lhs->getOpName()) {
685  if (std::optional<StringRef> rhsName = rhs->getOpName())
686  return lhsName->compare(*rhsName);
687  return -1; // lhs(op-specific) < rhs(op-agnostic)
688  }
689  return 1; // lhs(op-agnostic) > rhs(op-specific)
690  };
691  llvm::array_pod_sort(rhs.mgrs.begin(), rhs.mgrs.end(), compareFn);
692  return success();
693 }
694 
695 /// Returns the adaptor pass name.
696 std::string OpToOpPassAdaptor::getAdaptorName() {
697  std::string name = "Pipeline Collection : [";
698  llvm::raw_string_ostream os(name);
699  llvm::interleaveComma(getPassManagers(), os, [&](OpPassManager &pm) {
700  os << '\'' << pm.getOpAnchorName() << '\'';
701  });
702  os << ']';
703  return os.str();
704 }
705 
706 void OpToOpPassAdaptor::runOnOperation() {
707  llvm_unreachable(
708  "Unexpected call to Pass::runOnOperation() on OpToOpPassAdaptor");
709 }
710 
711 /// Run the held pipeline over all nested operations.
712 void OpToOpPassAdaptor::runOnOperation(bool verifyPasses) {
713  if (getContext().isMultithreadingEnabled())
714  runOnOperationAsyncImpl(verifyPasses);
715  else
716  runOnOperationImpl(verifyPasses);
717 }
718 
719 /// Run this pass adaptor synchronously.
720 void OpToOpPassAdaptor::runOnOperationImpl(bool verifyPasses) {
721  auto am = getAnalysisManager();
722  PassInstrumentation::PipelineParentInfo parentInfo = {llvm::get_threadid(),
723  this};
724  auto *instrumentor = am.getPassInstrumentor();
725  for (auto &region : getOperation()->getRegions()) {
726  for (auto &block : region) {
727  for (auto &op : block) {
728  auto *mgr = findPassManagerFor(mgrs, op.getName(), *op.getContext());
729  if (!mgr)
730  continue;
731 
732  // Run the held pipeline over the current operation.
733  unsigned initGeneration = mgr->impl->initializationGeneration;
734  if (failed(runPipeline(*mgr, &op, am.nest(&op), verifyPasses,
735  initGeneration, instrumentor, &parentInfo)))
736  return signalPassFailure();
737  }
738  }
739  }
740 }
741 
742 /// Utility functor that checks if the two ranges of pass managers have a size
743 /// mismatch.
746  return lhs.size() != rhs.size() ||
747  llvm::any_of(llvm::seq<size_t>(0, lhs.size()),
748  [&](size_t i) { return lhs[i].size() != rhs[i].size(); });
749 }
750 
751 /// Run this pass adaptor synchronously.
752 void OpToOpPassAdaptor::runOnOperationAsyncImpl(bool verifyPasses) {
753  AnalysisManager am = getAnalysisManager();
754  MLIRContext *context = &getContext();
755 
756  // Create the async executors if they haven't been created, or if the main
757  // pipeline has changed.
758  if (asyncExecutors.empty() || hasSizeMismatch(asyncExecutors.front(), mgrs))
759  asyncExecutors.assign(context->getThreadPool().getMaxConcurrency(), mgrs);
760 
761  // This struct represents the information for a single operation to be
762  // scheduled on a pass manager.
763  struct OpPMInfo {
764  OpPMInfo(unsigned passManagerIdx, Operation *op, AnalysisManager am)
765  : passManagerIdx(passManagerIdx), op(op), am(am) {}
766 
767  /// The index of the pass manager to schedule the operation on.
768  unsigned passManagerIdx;
769  /// The operation to schedule.
770  Operation *op;
771  /// The analysis manager for the operation.
772  AnalysisManager am;
773  };
774 
775  // Run a prepass over the operation to collect the nested operations to
776  // execute over. This ensures that an analysis manager exists for each
777  // operation, as well as providing a queue of operations to execute over.
778  std::vector<OpPMInfo> opInfos;
780  for (auto &region : getOperation()->getRegions()) {
781  for (Operation &op : region.getOps()) {
782  // Get the pass manager index for this operation type.
783  auto pmIdxIt = knownOpPMIdx.try_emplace(op.getName(), std::nullopt);
784  if (pmIdxIt.second) {
785  if (auto *mgr = findPassManagerFor(mgrs, op.getName(), *context))
786  pmIdxIt.first->second = std::distance(mgrs.begin(), mgr);
787  }
788 
789  // If this operation can be scheduled, add it to the list.
790  if (pmIdxIt.first->second)
791  opInfos.emplace_back(*pmIdxIt.first->second, &op, am.nest(&op));
792  }
793  }
794 
795  // Get the current thread for this adaptor.
796  PassInstrumentation::PipelineParentInfo parentInfo = {llvm::get_threadid(),
797  this};
798  auto *instrumentor = am.getPassInstrumentor();
799 
800  // An atomic failure variable for the async executors.
801  std::vector<std::atomic<bool>> activePMs(asyncExecutors.size());
802  std::fill(activePMs.begin(), activePMs.end(), false);
803  auto processFn = [&](OpPMInfo &opInfo) {
804  // Find an executor for this operation.
805  auto it = llvm::find_if(activePMs, [](std::atomic<bool> &isActive) {
806  bool expectedInactive = false;
807  return isActive.compare_exchange_strong(expectedInactive, true);
808  });
809  unsigned pmIndex = it - activePMs.begin();
810 
811  // Get the pass manager for this operation and execute it.
812  OpPassManager &pm = asyncExecutors[pmIndex][opInfo.passManagerIdx];
813  LogicalResult pipelineResult = runPipeline(
814  pm, opInfo.op, opInfo.am, verifyPasses,
815  pm.impl->initializationGeneration, instrumentor, &parentInfo);
816 
817  // Reset the active bit for this pass manager.
818  activePMs[pmIndex].store(false);
819  return pipelineResult;
820  };
821 
822  // Signal a failure if any of the executors failed.
823  if (failed(failableParallelForEach(context, opInfos, processFn)))
824  signalPassFailure();
825 }
826 
827 //===----------------------------------------------------------------------===//
828 // PassManager
829 //===----------------------------------------------------------------------===//
830 
831 PassManager::PassManager(MLIRContext *ctx, StringRef operationName,
832  Nesting nesting)
833  : OpPassManager(operationName, nesting), context(ctx), passTiming(false),
834  verifyPasses(true) {}
835 
837  : OpPassManager(operationName, nesting),
838  context(operationName.getContext()), passTiming(false),
839  verifyPasses(true) {}
840 
841 PassManager::~PassManager() = default;
842 
843 void PassManager::enableVerifier(bool enabled) { verifyPasses = enabled; }
844 
845 /// Run the passes within this manager on the provided operation.
846 LogicalResult PassManager::run(Operation *op) {
847  MLIRContext *context = getContext();
848  std::optional<OperationName> anchorOp = getOpName(*context);
849  if (anchorOp && anchorOp != op->getName())
850  return emitError(op->getLoc())
851  << "can't run '" << getOpAnchorName() << "' pass manager on '"
852  << op->getName() << "' op";
853 
854  // Register all dialects for the current pipeline.
855  DialectRegistry dependentDialects;
856  getDependentDialects(dependentDialects);
857  context->appendDialectRegistry(dependentDialects);
858  for (StringRef name : dependentDialects.getDialectNames())
859  context->getOrLoadDialect(name);
860 
861  // Before running, make sure to finalize the pipeline pass list.
862  if (failed(getImpl().finalizePassList(context)))
863  return failure();
864 
865  // Notify the context that we start running a pipeline for bookkeeping.
866  context->enterMultiThreadedExecution();
867 
868  // Initialize all of the passes within the pass manager with a new generation.
869  llvm::hash_code newInitKey = context->getRegistryHash();
870  llvm::hash_code pipelineKey = hash();
871  if (newInitKey != initializationKey || pipelineKey != pipelineInitializationKey) {
872  if (failed(initialize(context, impl->initializationGeneration + 1)))
873  return failure();
874  initializationKey = newInitKey;
875  pipelineKey = pipelineInitializationKey;
876  }
877 
878  // Construct a top level analysis manager for the pipeline.
879  ModuleAnalysisManager am(op, instrumentor.get());
880 
881  // If reproducer generation is enabled, run the pass manager with crash
882  // handling enabled.
883  LogicalResult result =
884  crashReproGenerator ? runWithCrashRecovery(op, am) : runPasses(op, am);
885 
886  // Notify the context that the run is done.
887  context->exitMultiThreadedExecution();
888 
889  // Dump all of the pass statistics if necessary.
890  if (passStatisticsMode)
891  dumpStatistics();
892  return result;
893 }
894 
895 /// Add the provided instrumentation to the pass manager.
896 void PassManager::addInstrumentation(std::unique_ptr<PassInstrumentation> pi) {
897  if (!instrumentor)
898  instrumentor = std::make_unique<PassInstrumentor>();
899 
900  instrumentor->addInstrumentation(std::move(pi));
901 }
902 
903 LogicalResult PassManager::runPasses(Operation *op, AnalysisManager am) {
904  return OpToOpPassAdaptor::runPipeline(*this, op, am, verifyPasses,
905  impl->initializationGeneration);
906 }
907 
908 //===----------------------------------------------------------------------===//
909 // AnalysisManager
910 //===----------------------------------------------------------------------===//
911 
912 /// Get an analysis manager for the given operation, which must be a proper
913 /// descendant of the current operation represented by this analysis manager.
915  Operation *currentOp = impl->getOperation();
916  assert(currentOp->isProperAncestor(op) &&
917  "expected valid descendant operation");
918 
919  // Check for the base case where the provided operation is immediately nested.
920  if (currentOp == op->getParentOp())
921  return nestImmediate(op);
922 
923  // Otherwise, we need to collect all ancestors up to the current operation.
924  SmallVector<Operation *, 4> opAncestors;
925  do {
926  opAncestors.push_back(op);
927  op = op->getParentOp();
928  } while (op != currentOp);
929 
930  AnalysisManager result = *this;
931  for (Operation *op : llvm::reverse(opAncestors))
932  result = result.nestImmediate(op);
933  return result;
934 }
935 
936 /// Get an analysis manager for the given immediately nested child operation.
937 AnalysisManager AnalysisManager::nestImmediate(Operation *op) {
938  assert(impl->getOperation() == op->getParentOp() &&
939  "expected immediate child operation");
940 
941  auto [it, inserted] = impl->childAnalyses.try_emplace(op);
942  if (inserted)
943  it->second = std::make_unique<NestedAnalysisMap>(op, impl);
944  return {it->second.get()};
945 }
946 
947 /// Invalidate any non preserved analyses.
949  const detail::PreservedAnalyses &pa) {
950  // If all analyses were preserved, then there is nothing to do here.
951  if (pa.isAll())
952  return;
953 
954  // Invalidate the analyses for the current operation directly.
955  analyses.invalidate(pa);
956 
957  // If no analyses were preserved, then just simply clear out the child
958  // analysis results.
959  if (pa.isNone()) {
960  childAnalyses.clear();
961  return;
962  }
963 
964  // Otherwise, invalidate each child analysis map.
965  SmallVector<NestedAnalysisMap *, 8> mapsToInvalidate(1, this);
966  while (!mapsToInvalidate.empty()) {
967  auto *map = mapsToInvalidate.pop_back_val();
968  for (auto &analysisPair : map->childAnalyses) {
969  analysisPair.second->invalidate(pa);
970  if (!analysisPair.second->childAnalyses.empty())
971  mapsToInvalidate.push_back(analysisPair.second.get());
972  }
973  }
974 }
975 
976 //===----------------------------------------------------------------------===//
977 // PassInstrumentation
978 //===----------------------------------------------------------------------===//
979 
981 
983  std::optional<OperationName> name, const PipelineParentInfo &parentInfo) {}
984 
986  std::optional<OperationName> name, const PipelineParentInfo &parentInfo) {}
987 
988 //===----------------------------------------------------------------------===//
989 // PassInstrumentor
990 //===----------------------------------------------------------------------===//
991 
992 namespace mlir {
993 namespace detail {
995  /// Mutex to keep instrumentation access thread-safe.
997 
998  /// Set of registered instrumentations.
999  std::vector<std::unique_ptr<PassInstrumentation>> instrumentations;
1000 };
1001 } // namespace detail
1002 } // namespace mlir
1003 
1006 
1007 /// See PassInstrumentation::runBeforePipeline for details.
1009  std::optional<OperationName> name,
1010  const PassInstrumentation::PipelineParentInfo &parentInfo) {
1011  llvm::sys::SmartScopedLock<true> instrumentationLock(impl->mutex);
1012  for (auto &instr : impl->instrumentations)
1013  instr->runBeforePipeline(name, parentInfo);
1014 }
1015 
1016 /// See PassInstrumentation::runAfterPipeline for details.
1018  std::optional<OperationName> name,
1019  const PassInstrumentation::PipelineParentInfo &parentInfo) {
1020  llvm::sys::SmartScopedLock<true> instrumentationLock(impl->mutex);
1021  for (auto &instr : llvm::reverse(impl->instrumentations))
1022  instr->runAfterPipeline(name, parentInfo);
1023 }
1024 
1025 /// See PassInstrumentation::runBeforePass for details.
1027  llvm::sys::SmartScopedLock<true> instrumentationLock(impl->mutex);
1028  for (auto &instr : impl->instrumentations)
1029  instr->runBeforePass(pass, op);
1030 }
1031 
1032 /// See PassInstrumentation::runAfterPass for details.
1034  llvm::sys::SmartScopedLock<true> instrumentationLock(impl->mutex);
1035  for (auto &instr : llvm::reverse(impl->instrumentations))
1036  instr->runAfterPass(pass, op);
1037 }
1038 
1039 /// See PassInstrumentation::runAfterPassFailed for details.
1041  llvm::sys::SmartScopedLock<true> instrumentationLock(impl->mutex);
1042  for (auto &instr : llvm::reverse(impl->instrumentations))
1043  instr->runAfterPassFailed(pass, op);
1044 }
1045 
1046 /// See PassInstrumentation::runBeforeAnalysis for details.
1048  Operation *op) {
1049  llvm::sys::SmartScopedLock<true> instrumentationLock(impl->mutex);
1050  for (auto &instr : impl->instrumentations)
1051  instr->runBeforeAnalysis(name, id, op);
1052 }
1053 
1054 /// See PassInstrumentation::runAfterAnalysis for details.
1056  Operation *op) {
1057  llvm::sys::SmartScopedLock<true> instrumentationLock(impl->mutex);
1058  for (auto &instr : llvm::reverse(impl->instrumentations))
1059  instr->runAfterAnalysis(name, id, op);
1060 }
1061 
1062 /// Add the given instrumentation to the collection.
1064  std::unique_ptr<PassInstrumentation> pi) {
1065  llvm::sys::SmartScopedLock<true> instrumentationLock(impl->mutex);
1066  impl->instrumentations.emplace_back(std::move(pi));
1067 }
static InFlightDiagnostic mergeInto(FunctionOpInterface func1, FunctionOpInterface func2)
Merge func1 into func2.
Definition: Utils.cpp:31
static MLIRContext * getContext(OpFoldResult val)
static llvm::ManagedStatic< PassManagerOptions > options
static void registerDialectsForPipeline(const OpPassManager &pm, DialectRegistry &dialects)
Definition: Pass.cpp:416
static bool hasSizeMismatch(ArrayRef< OpPassManager > lhs, ArrayRef< OpPassManager > rhs)
Utility functor that checks if the two ranges of pass managers have a size mismatch.
Definition: Pass.cpp:744
static OpPassManager * findPassManagerFor(MutableArrayRef< OpPassManager > mgrs, OperationName name, MLIRContext &context)
Find an operation pass manager that can operate on an operation of the given type,...
Definition: Pass.cpp:614
void printAsTextualPipeline(raw_ostream &os, StringRef anchorName, const llvm::iterator_range< OpPassManager::pass_iterator > &passes)
Prints out the passes of the pass manager as the textual representation of pipelines.
Definition: Pass.cpp:393
static OpPassManager * findPassManagerWithAnchor(MutableArrayRef< OpPassManager > mgrs, StringRef name)
Find an operation pass manager with the given anchor name, or nullptr if one does not exist.
Definition: Pass.cpp:606
This class represents an analysis manager for a particular operation instance.
void clear()
Clear any held analyses.
void invalidate(const PreservedAnalyses &pa)
Invalidate any non preserved analyses,.
AnalysisManager nest(Operation *op)
Get an analysis manager for the given operation, which must be a proper descendant of the current ope...
Definition: Pass.cpp:914
PassInstrumentor * getPassInstrumentor() const
Returns a pass instrumentation object for the current operation.
The DialectRegistry maps a dialect namespace to a constructor for the matching dialect.
auto getDialectNames() const
Return the names of dialects known to this registry.
MLIRContext is the top-level object for a collection of MLIR operations.
Definition: MLIRContext.h:60
void appendDialectRegistry(const DialectRegistry &registry)
Append the contents of the given dialect registry to the registry associated with this context.
void executeAction(function_ref< void()> actionFn, const tracing::Action &action)
Dispatch the provided action to the handler if any, or just execute it.
Definition: MLIRContext.h:259
llvm::hash_code getRegistryHash()
Returns a hash of the registry of the context that may be used to give a rough indicator of if the st...
void enterMultiThreadedExecution()
These APIs are tracking whether the context will be used in a multithreading environment: this has no...
llvm::ThreadPoolInterface & getThreadPool()
Return the thread pool used by this context.
T * getOrLoadDialect()
Get (or create) a dialect for the given derived dialect type.
Definition: MLIRContext.h:97
void exitMultiThreadedExecution()
An analysis manager class specifically for the top-level operation.
This class represents a pass manager that runs passes on either a specific operation type,...
Definition: PassManager.h:47
OpPassManager & operator=(const OpPassManager &rhs)
Definition: Pass.cpp:327
friend class PassManager
Allow access to the constructor.
Definition: PassManager.h:185
OpPassManager(Nesting nesting=Nesting::Explicit)
Construct a new op-agnostic ("any") pass manager with the given operation type and nesting behavior.
Definition: Pass.cpp:319
OpPassManager & nest()
Definition: PassManager.h:98
void printAsTextualPipeline(raw_ostream &os) const
Prints out the passes of the pass manager as the textual representation of pipelines.
Definition: Pass.cpp:402
std::optional< OperationName > getOpName(MLIRContext &context) const
Return the operation name that this pass manager operates on, or std::nullopt if this is an op-agnost...
Definition: Pass.cpp:383
void setNesting(Nesting nesting)
Enable or disable the implicit nesting on this particular PassManager.
Definition: Pass.cpp:426
size_t size() const
Returns the number of passes held by this manager.
Definition: Pass.cpp:371
detail::OpPassManagerImpl & getImpl()
Returns the internal implementation instance.
Definition: Pass.cpp:374
void addPass(std::unique_ptr< Pass > pass)
Add the given pass to this pass manager.
Definition: Pass.cpp:364
Nesting getNesting()
Return the current nesting mode.
Definition: Pass.cpp:428
Nesting
This enum represents the nesting behavior of the pass manager.
Definition: PassManager.h:50
@ Implicit
Implicit nesting behavior.
void dump()
Raw dump of the pass manager to llvm::errs().
Definition: Pass.cpp:410
llvm::pointee_iterator< MutableArrayRef< std::unique_ptr< Pass > >::iterator > pass_iterator
Iterator over the passes in this pass manager.
Definition: PassManager.h:77
llvm::pointee_iterator< ArrayRef< std::unique_ptr< Pass > >::const_iterator > const_pass_iterator
Definition: PassManager.h:83
void getDependentDialects(DialectRegistry &dialects) const
Register dependent dialects for the current pass manager.
Definition: Pass.cpp:422
StringRef getOpAnchorName() const
Return the name used to anchor this pass manager.
Definition: Pass.cpp:387
std::optional< StringRef > getOpName() const
Return the operation name that this pass manager operates on, or std::nullopt if this is an op-agnost...
Definition: Pass.cpp:377
OpPassManager & nestAny()
Nest a new op-agnostic ("any") pass manager under this pass manager.
Definition: Pass.cpp:360
iterator_range< pass_iterator > getPasses()
Definition: PassManager.h:80
static StringRef getAnyOpAnchorName()
Return the string name used to anchor op-agnostic pass managers that operate generically on any viabl...
Definition: PassManager.h:138
void clear()
Clear the pipeline, but not the other options set on this OpPassManager.
Definition: Pass.cpp:368
pass_iterator begin()
Definition: Pass.cpp:338
pass_iterator end()
Definition: Pass.cpp:341
This class provides the API for ops that are known to be isolated from above.
std::optional< RegisteredOperationName > getRegisteredInfo() const
If this operation is registered, returns the registered information, std::nullopt otherwise.
Operation is the basic unit of execution within MLIR.
Definition: Operation.h:88
MLIRContext * getContext()
Return the context this operation is associated with.
Definition: Operation.h:216
std::optional< RegisteredOperationName > getRegisteredInfo()
If this operation has a registered operation description, return it.
Definition: Operation.h:123
Location getLoc()
The source location the operation was defined or derived from.
Definition: Operation.h:223
Operation * getParentOp()
Returns the closest surrounding operation that contains this operation or nullptr if this is a top-le...
Definition: Operation.h:234
OperationName getName()
The name of an operation is the key identifier for it.
Definition: Operation.h:119
bool isAncestor(Operation *other)
Return true if this operation is an ancestor of the other operation.
Definition: Operation.h:263
bool isProperAncestor(Operation *other)
Return true if this operation is a proper ancestor of the other operation.
Definition: Operation.cpp:219
InFlightDiagnostic emitOpError(const Twine &message={})
Emit an error with the op name prefixed, like "'dim' op " which is convenient for verifiers.
Definition: Operation.cpp:671
This class encapsulates the "action" of executing a single pass.
Definition: Pass.h:492
void print(raw_ostream &os) const override
Print a textual version of this action to os.
Definition: Pass.cpp:43
PassExecutionAction(ArrayRef< IRUnit > irUnits, const Pass &pass)
Define a TypeID for this PassExecutionAction.
Definition: Pass.cpp:39
Operation * getOp() const
Get the operation that is the base of this pass.
Definition: Pass.cpp:48
static constexpr StringLiteral tag
The tag required by ActionImpl to identify this action.
Definition: Pass.h:503
const Pass & pass
Reference to the pass being run.
Definition: Pass.h:520
virtual void runAfterPipeline(std::optional< OperationName > name, const PipelineParentInfo &parentInfo)
A callback to run after a pass pipeline has executed.
Definition: Pass.cpp:985
virtual ~PassInstrumentation()=0
virtual void runBeforePipeline(std::optional< OperationName > name, const PipelineParentInfo &parentInfo)
A callback to run before a pass pipeline is executed.
Definition: Pass.cpp:982
This class holds a collection of PassInstrumentation objects, and invokes their respective call backs...
void runAfterPassFailed(Pass *pass, Operation *op)
See PassInstrumentation::runAfterPassFailed for details.
Definition: Pass.cpp:1040
void addInstrumentation(std::unique_ptr< PassInstrumentation > pi)
Add the given instrumentation to the collection.
Definition: Pass.cpp:1063
void runBeforeAnalysis(StringRef name, TypeID id, Operation *op)
See PassInstrumentation::runBeforeAnalysis for details.
Definition: Pass.cpp:1047
void runAfterPass(Pass *pass, Operation *op)
See PassInstrumentation::runAfterPass for details.
Definition: Pass.cpp:1033
void runAfterAnalysis(StringRef name, TypeID id, Operation *op)
See PassInstrumentation::runAfterAnalysis for details.
Definition: Pass.cpp:1055
void runBeforePass(Pass *pass, Operation *op)
See PassInstrumentation::runBeforePass for details.
Definition: Pass.cpp:1026
void runBeforePipeline(std::optional< OperationName > name, const PassInstrumentation::PipelineParentInfo &parentInfo)
See PassInstrumentation::runBeforePipeline for details.
Definition: Pass.cpp:1008
void runAfterPipeline(std::optional< OperationName > name, const PassInstrumentation::PipelineParentInfo &parentInfo)
See PassInstrumentation::runAfterPipeline for details.
Definition: Pass.cpp:1017
MLIRContext * getContext() const
Return an instance of the context.
Definition: PassManager.h:256
LogicalResult run(Operation *op)
Run the passes within this manager on the provided operation.
Definition: Pass.cpp:846
void addInstrumentation(std::unique_ptr< PassInstrumentation > pi)
Add the provided instrumentation to the pass manager.
Definition: Pass.cpp:896
void enableVerifier(bool enabled=true)
Runs the verifier after each individual pass.
Definition: Pass.cpp:843
The abstract base pass class.
Definition: Pass.h:51
void copyOptionValuesFrom(const Pass *other)
Copy the option values from 'other', which is another instance of this pass.
Definition: Pass.cpp:77
void printAsTextualPipeline(raw_ostream &os)
Prints out the pass in the textual representation of pipelines.
Definition: Pass.cpp:83
virtual LogicalResult initializeOptions(StringRef options, function_ref< LogicalResult(const Twine &)> errorHandler)
Attempt to initialize the options of this pass from the given string.
Definition: Pass.cpp:63
virtual bool canScheduleOn(RegisteredOperationName opName) const =0
Indicate if the current pass can be scheduled on the given operation type.
std::optional< StringRef > getOpName() const
Returns the name of the operation that this pass operates on, or std::nullopt if this is a generic Op...
Definition: Pass.h:83
virtual void runOnOperation()=0
The polymorphic API that runs the pass over the currently held operation.
virtual StringRef getName() const =0
Returns the derived pass name.
virtual StringRef getArgument() const
Return the command line argument used when registering this pass.
Definition: Pass.h:75
This class provides an efficient unique identifier for a specific C++ type.
Definition: TypeID.h:104
void invalidate(const PreservedAnalyses &pa)
Invalidate any cached analyses based upon the given set of preserved analyses.
An adaptor pass used to run operation passes over nested operations.
Definition: PassDetail.h:26
LogicalResult parseFromString(StringRef options, raw_ostream &errorStream=llvm::errs())
Parse options out as key=value pairs that can then be handed off to the llvm::cl command line passing...
void print(raw_ostream &os) const
Print the options held by this struct in a form that can be parsed via 'parseFromString'.
void copyOptionValuesFrom(const PassOptions &other)
Copy the option values from 'other' into 'this', where 'other' has the same options as 'this'.
A utility class to represent the analyses that are known to be preserved.
bool isAll() const
Returns true if all analyses were marked preserved.
bool isNone() const
Returns true if no analyses were marked preserved.
virtual ArrayRef< IRUnit > getContextIRUnits() const
Return the set of IR units that are associated with this action.
Definition: Action.h:55
ArrayRef< IRUnit > irUnits
Set of IR units (operations, regions, blocks, values) that are associated with this action.
Definition: Action.h:66
AttrTypeReplacer.
Include the generated interface declarations.
LogicalResult failableParallelForEach(MLIRContext *context, IteratorT begin, IteratorT end, FuncT &&func)
Invoke the given function on the elements between [begin, end) asynchronously.
Definition: Threading.h:36
InFlightDiagnostic emitError(Location loc)
Utility method to emit an error message using this location.
auto get(MLIRContext *context, Ts &&...params)
Helper method that injects context only if needed, this helps unify some of the attribute constructio...
LogicalResult verify(Operation *op, bool verifyRecursively=true)
Perform (potentially expensive) checks of invariants, used to detect compiler bugs,...
Definition: Verifier.cpp:426
This struct represents information related to the parent pass of pipeline.
detail::AnalysisMap analyses
The analyses for the owning operation.
DenseMap< Operation *, std::unique_ptr< NestedAnalysisMap > > childAnalyses
The cached analyses for nested operations.
void invalidate(const PreservedAnalyses &pa)
Invalidate any non preserved analyses.
Definition: Pass.cpp:948
void clear()
Clear the list of passes in this pass manager, other options are preserved.
Definition: Pass.cpp:231
OpPassManagerImpl(const OpPassManagerImpl &rhs)
Definition: Pass.cpp:118
std::string name
The name of the operation that passes of this pass manager operate on.
Definition: Pass.cpp:181
OpPassManager::Nesting nesting
Control the implicit nesting of passes that mismatch the name set for this OpPassManager.
Definition: Pass.cpp:196
std::optional< OperationName > getOpName(MLIRContext &context)
Return the operation name of this pass manager.
Definition: Pass.cpp:159
void addPass(std::unique_ptr< Pass > pass)
Add the given pass to this pass manager.
Definition: Pass.cpp:214
OpPassManager & nest(StringRef nestedName)
Definition: Pass.cpp:137
unsigned initializationGeneration
The current initialization generation of this pass manager.
Definition: Pass.cpp:192
OpPassManagerImpl(OperationName opName, OpPassManager::Nesting nesting)
Definition: Pass.cpp:110
bool canScheduleOn(MLIRContext &context, OperationName opName)
Indicate if the current pass manager can be scheduled on the given operation type.
Definition: Pass.cpp:295
OpPassManagerImpl(OpPassManager::Nesting nesting)
Definition: Pass.cpp:116
StringRef getOpAnchorName() const
Return the name used to anchor this pass manager.
Definition: Pass.cpp:172
OpPassManager & nestAny()
Definition: Pass.cpp:140
OpPassManager & nest(OperationName nestedName)
Nest a new operation pass manager for the given operation kind under this pass manager.
Definition: Pass.cpp:134
std::vector< std::unique_ptr< Pass > > passes
The set of passes to run as part of this pass manager.
Definition: Pass.cpp:188
std::optional< StringRef > getOpName() const
Definition: Pass.cpp:164
void mergeInto(OpPassManagerImpl &rhs)
Merge the passes of this pass manager into the one provided.
Definition: Pass.cpp:201
LogicalResult finalizePassList(MLIRContext *ctx)
Finalize the pass list in preparation for execution.
Definition: Pass.cpp:233
std::optional< OperationName > opName
The cached OperationName (internalized in the context) for the name of the operation that passes of t...
Definition: Pass.cpp:185
OpPassManagerImpl(StringRef name, OpPassManager::Nesting nesting)
Definition: Pass.cpp:113
std::vector< std::unique_ptr< PassInstrumentation > > instrumentations
Set of registered instrumentations.
Definition: Pass.cpp:999
llvm::sys::SmartMutex< true > mutex
Mutex to keep instrumentation access thread-safe.
Definition: Pass.cpp:996