37#ifndef LLVM_IR_PASSMANAGER_H
38#define LLVM_IR_PASSMANAGER_H
64 static_assert(std::is_base_of<InfoMixin, DerivedT>::value,
65 "Must pass the derived type as the template argument!");
67 Name.consume_front(
"llvm::");
77template <
typename IRUnitT,
typename... ExtraArgTs>
class AnalysisManager;
88template <
typename DerivedT>
93 auto PassName = MapClassName2PassName(ClassName);
102template <
typename DerivedT>
108template <
typename DerivedT>
117template <
typename DerivedT>
135 static_assert(std::is_base_of<AnalysisInfoMixin, DerivedT>::value,
136 "Must pass the derived type as the template argument!");
137 return &DerivedT::Key;
145template <
typename PassT,
typename IRUnitT,
typename AnalysisManagerT,
146 typename... ArgTs,
size_t... Ns>
147typename PassT::Result
149 std::tuple<ArgTs...> Args,
150 std::index_sequence<Ns...>) {
152 return AM.template getResult<PassT>(
IR, std::get<Ns>(Args)...);
160template <
typename PassT,
typename IRUnitT,
typename... AnalysisArgTs,
161 typename... MainArgTs>
162typename PassT::Result
164 std::tuple<MainArgTs...> Args) {
166 PassT, IRUnitT>)(AM,
IR, Args,
167 std::index_sequence_for<AnalysisArgTs...>{});
184template <
typename IRUnitT,
185 typename AnalysisManagerT = AnalysisManager<IRUnitT>,
186 typename... ExtraArgTs>
188 PassManager<IRUnitT, AnalysisManagerT, ExtraArgTs...>> {
209 P->printPipeline(OS, MapClassName2PassName);
216 ExtraArgTs... ExtraArgs);
218 template <
typename PassT>
225 Passes.push_back(std::unique_ptr<PassConceptT>(
226 new PassModelT(std::forward<PassT>(
Pass))));
234 template <
typename PassT>
237 for (
auto &
P :
Pass.Passes)
238 Passes.push_back(std::move(
P));
248 std::vector<std::unique_ptr<PassConceptT>>
Passes;
251template <
typename IRUnitT>
293 using AnalysisResultListT =
294 std::list<std::pair<AnalysisKey *, std::unique_ptr<ResultConceptT>>>;
302 using AnalysisResultMapT =
304 typename AnalysisResultListT::iterator>;
333 template <
typename PassT>
339 return invalidateImpl<ResultModelT>(PassT::ID(),
IR, PA);
349 return invalidateImpl<>(
ID,
IR, PA);
355 template <
typename ResultT = ResultConceptT>
360 auto IMapI = IsResultInvalidated.find(
ID);
361 if (IMapI != IsResultInvalidated.end())
362 return IMapI->second;
365 auto RI = Results.find({
ID, &
IR});
367 "Trying to invalidate a dependent result that isn't in the "
368 "manager's cache is always an error, likely due to a stale result "
371 auto &Result =
static_cast<ResultT &
>(*RI->second->second);
378 std::tie(IMapI, Inserted) =
379 IsResultInvalidated.insert({
ID, Result.invalidate(
IR, PA, *
this)});
381 assert(Inserted &&
"Should not have already inserted this ID, likely "
382 "indicates a dependency cycle!");
383 return IMapI->second;
386 Invalidator(SmallDenseMap<AnalysisKey *, bool, 8> &IsResultInvalidated,
387 const AnalysisResultMapT &
Results)
390 SmallDenseMap<AnalysisKey *, bool, 8> &IsResultInvalidated;
391 const AnalysisResultMapT &Results;
401 assert(AnalysisResults.empty() == AnalysisResultLists.empty() &&
402 "The storage and index of analysis results disagree on how many "
404 return AnalysisResults.empty();
421 AnalysisResults.clear();
422 AnalysisResultLists.clear();
427 return AnalysisPasses.count(PassT::ID());
433 template <
typename PassT>
434 typename PassT::Result &
getResult(IRUnitT &
IR, ExtraArgTs... ExtraArgs) {
435 assert(AnalysisPasses.count(PassT::ID()) &&
436 "This analysis pass was not registered prior to being queried");
437 ResultConceptT &ResultConcept =
438 getResultImpl(PassT::ID(),
IR, ExtraArgs...);
444 return static_cast<ResultModelT &
>(ResultConcept).Result;
452 template <
typename PassT>
454 assert(AnalysisPasses.count(PassT::ID()) &&
455 "This analysis pass was not registered prior to being queried");
457 ResultConceptT *ResultConcept = getCachedResultImpl(PassT::ID(),
IR);
465 return &
static_cast<ResultModelT *
>(ResultConcept)->Result;
469 template <
typename PassT>
473 Invalidator Inv(IsResultInvalidated, AnalysisResults);
474 assert(!Result->invalidate(
IR, PA, Inv) &&
475 "Cached result cannot be invalidated");
494 template <
typename PassBuilderT>
500 auto &PassPtr = AnalysisPasses[PassT::ID()];
522 AnalysisResultListT &ResultsList = AnalysisResultLists[&
IR];
527 assert(
I != ResultsList.end() &&
"Analysis must be available");
528 ResultsList.erase(
I);
529 AnalysisResults.erase({
ID, &
IR});
535 typename AnalysisPassMapT::iterator PI = AnalysisPasses.find(
ID);
536 assert(PI != AnalysisPasses.end() &&
537 "Analysis passes must be registered prior to being queried!");
542 const PassConceptT &lookUpPass(AnalysisKey *
ID)
const {
543 typename AnalysisPassMapT::const_iterator PI = AnalysisPasses.find(
ID);
544 assert(PI != AnalysisPasses.end() &&
545 "Analysis passes must be registered prior to being queried!");
550 ResultConceptT &getResultImpl(AnalysisKey *
ID, IRUnitT &
IR,
551 ExtraArgTs... ExtraArgs);
554 ResultConceptT *getCachedResultImpl(AnalysisKey *
ID, IRUnitT &
IR)
const {
555 typename AnalysisResultMapT::const_iterator RI =
556 AnalysisResults.find({
ID, &
IR});
557 return RI == AnalysisResults.end() ? nullptr : &*RI->second->second;
561 using AnalysisPassMapT =
562 DenseMap<AnalysisKey *, std::unique_ptr<PassConceptT>>;
565 AnalysisPassMapT AnalysisPasses;
571 AnalysisResultListMapT AnalysisResultLists;
575 AnalysisResultMapT AnalysisResults;
604template <
typename AnalysisManagerT,
typename IRUnitT,
typename... ExtraArgTs>
607 InnerAnalysisManagerProxy<AnalysisManagerT, IRUnitT>> {
611 explicit Result(AnalysisManagerT &InnerAM) : InnerAM(&InnerAM) {}
617 Arg.InnerAM =
nullptr;
631 InnerAM =
RHS.InnerAM;
635 RHS.InnerAM =
nullptr;
657 AnalysisManagerT *InnerAM;
661 : InnerAM(&InnerAM) {}
670 return Result(*InnerAM);
679 AnalysisManagerT *InnerAM;
688template <
typename AnalysisManagerT,
typename IRUnitT,
typename... ExtraArgTs>
699LLVM_ABI bool FunctionAnalysisManagerModuleProxy::Result::invalidate(
701 ModuleAnalysisManager::Invalidator &Inv);
735template <
typename AnalysisManagerT,
typename IRUnitT,
typename... ExtraArgTs>
738 OuterAnalysisManagerProxy<AnalysisManagerT, IRUnitT, ExtraArgTs...>> {
743 explicit Result(
const AnalysisManagerT &OuterAM) : OuterAM(&OuterAM) {}
747 template <
typename PassT,
typename IRUnitTParam>
749 typename PassT::Result *Res =
752 OuterAM->template verifyNotInvalidated<PassT>(
IR, Res);
757 template <
typename PassT,
typename IRUnitTParam>
759 typename PassT::Result *Res =
761 return Res !=
nullptr;
771 for (
auto &KeyValuePair : OuterAnalysisInvalidationMap) {
773 auto &InnerIDs = KeyValuePair.second;
777 if (InnerIDs.empty())
781 for (
auto *OuterID : DeadKeys)
782 OuterAnalysisInvalidationMap.
erase(OuterID);
790 template <
typename OuterAnalysisT,
typename Inval
idatedAnalysisT>
793 AnalysisKey *InvalidatedID = InvalidatedAnalysisT::ID();
795 auto &InvalidatedIDList = OuterAnalysisInvalidationMap[OuterID];
801 InvalidatedIDList.push_back(InvalidatedID);
808 return OuterAnalysisInvalidationMap;
812 const AnalysisManagerT *OuterAM;
817 OuterAnalysisInvalidationMap;
821 : OuterAM(&OuterAM) {}
828 return Result(*OuterAM);
837 const AnalysisManagerT *OuterAM;
840template <
typename AnalysisManagerT,
typename IRUnitT,
typename... ExtraArgTs>
879 bool EagerlyInvalidate)
880 : Pass(
std::
move(Pass)), EagerlyInvalidate(EagerlyInvalidate) {}
889 std::unique_ptr<PassConceptT>
Pass;
890 bool EagerlyInvalidate;
895template <
typename FunctionPassT>
896ModuleToFunctionPassAdaptor
898 bool EagerlyInvalidate =
false) {
904 std::unique_ptr<ModuleToFunctionPassAdaptor::PassConceptT>(
905 new PassModelT(std::forward<FunctionPassT>(
Pass))),
918template <
typename AnalysisT,
typename IRUnitT,
919 typename AnalysisManagerT = AnalysisManager<IRUnitT>,
920 typename... ExtraArgTs>
923 AnalysisT, IRUnitT, AnalysisManagerT, ExtraArgTs...>> {
931 ExtraArgTs &&... Args) {
932 (void)AM.template getResult<AnalysisT>(Arg,
933 std::forward<ExtraArgTs>(Args)...);
939 auto ClassName = AnalysisT::name();
940 auto PassName = MapClassName2PassName(ClassName);
941 OS <<
"require<" <<
PassName <<
'>';
947template <
typename AnalysisT>
956 template <
typename IRUnitT,
typename AnalysisManagerT,
typename... ExtraArgTs>
964 auto ClassName = AnalysisT::name();
965 auto PassName = MapClassName2PassName(ClassName);
966 OS <<
"invalidate<" <<
PassName <<
'>';
977 template <
typename IRUnitT,
typename AnalysisManagerT,
typename... ExtraArgTs>
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
Function Alias Analysis Results
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
#define LLVM_TEMPLATE_ABI
#define LLVM_ATTRIBUTE_MINSIZE
#define LLVM_ATTRIBUTE_VISIBILITY_DEFAULT
This file defines the DenseMap class.
Legalize the Machine IR a function s Machine IR
Machine Check Debug Module
This header provides internal APIs and implementation details used by the pass management interfaces ...
static const char PassName[]
API to communicate dependencies between analyses during invalidation.
bool invalidate(AnalysisKey *ID, IRUnitT &IR, const PreservedAnalyses &PA)
A type-erased variant of the above invalidate method with the same core API other than passing an ana...
bool invalidate(IRUnitT &IR, const PreservedAnalyses &PA)
Trigger the invalidation of some other analysis pass if not already handled and return whether it was...
friend class AnalysisManager
A container for analyses that lazily runs them and caches their results.
bool isPassRegistered() const
Returns true if the specified analysis pass is registered.
AnalysisManager()
Construct an empty analysis manager.
void clear()
Clear all analysis results cached by this AnalysisManager.
AnalysisManager(AnalysisManager &&)
void verifyNotInvalidated(IRUnitT &IR, typename PassT::Result *Result) const
Verify that the given Result cannot be invalidated, assert otherwise.
AnalysisManager & operator=(AnalysisManager &&)
void clear(IRUnitT &IR, llvm::StringRef Name)
Clear any cached analysis results for a single unit of IR.
void invalidate(IRUnitT &IR, const PreservedAnalyses &PA)
Invalidate cached analyses for an IR unit.
void clearAnalysis(IRUnitT &IR)
Directly clear a cached analysis for an IR unit.
PassT::Result * getCachedResult(IRUnitT &IR) const
Get the cached result of an analysis pass for a given IR unit.
bool registerPass(PassBuilderT &&PassBuilder)
Register an analysis pass with the manager.
bool empty() const
Returns true if the analysis manager has an empty results cache.
PassT::Result & getResult(IRUnitT &IR, ExtraArgTs... ExtraArgs)
Get the result of an analysis pass for a given IR unit.
bool invalidate(IRUnitT &IR, const PreservedAnalyses &PA, typename AnalysisManager< IRUnitT, ExtraArgTs... >::Invalidator &Inv)
Handler for invalidation of the outer IR unit, IRUnitT.
Result(AnalysisManagerT &InnerAM)
Result & operator=(Result &&RHS)
AnalysisManagerT & getManager()
Accessor for the analysis manager.
An analysis over an "outer" IR unit that provides access to an analysis manager over an "inner" IR un...
Result run(IRUnitT &IR, AnalysisManager< IRUnitT, ExtraArgTs... > &AM, ExtraArgTs...)
Run the analysis pass and create our proxy result object.
InnerAnalysisManagerProxy(AnalysisManagerT &InnerAM)
A helper class to return the specified delimiter string after the first invocation of operator String...
Trivial adaptor that maps from a module to its functions.
detail::PassConcept< Function, FunctionAnalysisManager > PassConceptT
ModuleToFunctionPassAdaptor(std::unique_ptr< PassConceptT > Pass, bool EagerlyInvalidate)
A Module instance is used to store all the information related to an LLVM module.
Result(const AnalysisManagerT &OuterAM)
PassT::Result * getCachedResult(IRUnitTParam &IR) const
Get a cached analysis.
bool invalidate(IRUnitT &IRUnit, const PreservedAnalyses &PA, typename AnalysisManager< IRUnitT, ExtraArgTs... >::Invalidator &Inv)
When invalidation occurs, remove any registered invalidation events.
bool cachedResultExists(IRUnitTParam &IR) const
Method provided for unit testing, not intended for general use.
const SmallDenseMap< AnalysisKey *, TinyPtrVector< AnalysisKey * >, 2 > & getOuterInvalidations() const
Access the map from outer analyses to deferred invalidation requiring analyses.
void registerOuterAnalysisInvalidation()
Register a deferred invalidation event for when the outer analysis manager processes its invalidation...
An analysis over an "inner" IR unit that provides access to an analysis manager over a "outer" IR uni...
Result run(IRUnitT &, AnalysisManager< IRUnitT, ExtraArgTs... > &, ExtraArgTs...)
Run the analysis pass and create our proxy result object.
OuterAnalysisManagerProxy(const AnalysisManagerT &OuterAM)
This class provides access to building LLVM's passes.
Manages a sequence of passes over a particular unit of IR.
PassManager(PassManager &&Arg)
PassManager & operator=(PassManager &&RHS)
LLVM_ATTRIBUTE_MINSIZE std::enable_if_t<!std::is_same_v< PassT, PassManager > > addPass(PassT &&Pass)
void printPipeline(raw_ostream &OS, function_ref< StringRef(StringRef)> MapClassName2PassName)
std::vector< std::unique_ptr< PassConceptT > > Passes
PassManager()=default
Construct a pass manager.
detail::PassConcept< LazyCallGraph::SCC, CGSCCAnalysisManager, ExtraArgTs... > PassConceptT
LLVM_ATTRIBUTE_MINSIZE std::enable_if_t< std::is_same_v< PassT, PassManager > > addPass(PassT &&Pass)
When adding a pass manager pass that has the same type as this pass manager, simply move the passes o...
PreservedAnalyses run(IRUnitT &IR, AnalysisManagerT &AM, ExtraArgTs... ExtraArgs)
Run all of the passes in this manager over the given unit of IR.
bool isEmpty() const
Returns if the pass manager contains any passes.
Pass interface - Implemented by all 'passes'.
A set of analyses that are preserved following a run of a transformation pass.
static PreservedAnalyses none()
Convenience factory function for the empty preserved set.
static PreservedAnalyses all()
Construct a special preserved set that preserves all passes.
PreservedAnalyses & abandon()
Mark an analysis as abandoned.
iterator erase(const_iterator CI)
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
Represent a constant reference to a string, i.e.
An efficient, type-erasing, non-owning reference to a callable.
This class implements an extremely fast bulk output stream that can only output to a stream.
Pass manager infrastructure for declaring and invalidating analyses.
unsigned ID
LLVM IR allows to use arbitrary numbers as calling convention identifiers.
A self-contained host- and target-independent arbitrary-precision floating-point software implementat...
PassT::Result getAnalysisResultUnpackTuple(AnalysisManagerT &AM, IRUnitT &IR, std::tuple< ArgTs... > Args, std::index_sequence< Ns... >)
Actual unpacker of extra arguments in getAnalysisResult, passes only those tuple arguments that are m...
PassT::Result getAnalysisResult(AnalysisManager< IRUnitT, AnalysisArgTs... > &AM, IRUnitT &IR, std::tuple< MainArgTs... > Args)
Helper for partial unpacking of extra arguments in getAnalysisResult.
This is an optimization pass for GlobalISel generic memory operations.
ModuleToFunctionPassAdaptor createModuleToFunctionPassAdaptor(FunctionPassT &&Pass, bool EagerlyInvalidate=false)
A function to deduce a function pass type and wrap it in the templated adaptor.
OuterAnalysisManagerProxy< ModuleAnalysisManager, Function > ModuleAnalysisManagerFunctionProxy
Provide the ModuleAnalysisManager to Function proxy.
LLVM_GET_TYPE_NAME_CONSTEXPR StringRef getTypeName()
We provide a function which tries to compute the (demangled) name of a type statically.
LLVM_ABI void printIRUnitNameForStackTrace< Function >(raw_ostream &OS, const Function &IR)
void printIRUnitNameForStackTrace(raw_ostream &OS, const IRUnitT &IR)
InnerAnalysisManagerProxy< FunctionAnalysisManager, Module > FunctionAnalysisManagerModuleProxy
Provide the FunctionAnalysisManager to Module proxy.
LLVM_ABI void printIRUnitNameForStackTrace< Module >(raw_ostream &OS, const Module &IR)
PassManager< Module > ModulePassManager
Convenience typedef for a pass manager over modules.
LLVM_ATTRIBUTE_VISIBILITY_DEFAULT AnalysisKey InnerAnalysisManagerProxy< AnalysisManagerT, IRUnitT, ExtraArgTs... >::Key
PassManager< Function > FunctionPassManager
Convenience typedef for a pass manager over functions.
OutputIt move(R &&Range, OutputIt Out)
Provide wrappers to std::move which take ranges instead of having to pass begin/end explicitly.
auto find_if(R &&Range, UnaryPredicate P)
Provide wrappers to std::find_if which take ranges instead of having to pass begin/end explicitly.
void erase_if(Container &C, UnaryPredicate P)
Provide a container algorithm similar to C++ Library Fundamentals v2's erase_if which is equivalent t...
bool is_contained(R &&Range, const E &Element)
Returns true if Element is found in Range.
AnalysisManager< Function > FunctionAnalysisManager
Convenience typedef for the Function analysis manager.
AnalysisManager< Module > ModuleAnalysisManager
Convenience typedef for the Module analysis manager.
Implement std::hash so that hash_code can be used in STL containers.
A CRTP mix-in that provides informational APIs needed for analysis passes.
static AnalysisKey * ID()
Returns an opaque, unique ID for this analysis type.
A special type used by analysis passes to provide an address that identifies that particular analysis...
A utility pass that does nothing, but preserves no analyses.
PreservedAnalyses run(IRUnitT &, AnalysisManagerT &, ExtraArgTs &&...)
Run this pass over some unit of IR.
A no-op pass template which simply forces a specific analysis result to be invalidated.
void printPipeline(raw_ostream &OS, function_ref< StringRef(StringRef)> MapClassName2PassName)
PreservedAnalyses run(IRUnitT &Arg, AnalysisManagerT &AM, ExtraArgTs &&...)
Run this pass over some unit of IR.
A CRTP mix-in for passes that can be skipped.
A CRTP mix-in to automatically provide informational APIs needed for passes.
void printPipeline(raw_ostream &OS, function_ref< StringRef(StringRef)> MapClassName2PassName)
A utility pass template to force an analysis result to be available.
void printPipeline(raw_ostream &OS, function_ref< StringRef(StringRef)> MapClassName2PassName)
PreservedAnalyses run(IRUnitT &Arg, AnalysisManagerT &AM, ExtraArgTs &&... Args)
Run this pass over some unit of IR.
A CRTP mix-in for passes that should not be skipped.
Abstract concept of an analysis pass.
Wrapper to model the analysis pass concept.
Abstract concept of an analysis result.
Wrapper to model the analysis result concept.
static StringRef name()
Gets the name of the pass we are mixed into.
Template for the abstract base class used to dispatch polymorphically over pass objects.
A template wrapper used to implement the polymorphic API.