xref: /src/contrib/llvm-project/llvm/lib/CodeGen/MachinePassManager.cpp (revision 0fca6ea1d4eea4c934cfff25ac9ee8ad6fe95583)
1b60736ecSDimitry Andric //===---------- MachinePassManager.cpp ------------------------------------===//
2b60736ecSDimitry Andric //
3b60736ecSDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4b60736ecSDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
5b60736ecSDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6b60736ecSDimitry Andric //
7b60736ecSDimitry Andric //===----------------------------------------------------------------------===//
8b60736ecSDimitry Andric //
9b60736ecSDimitry Andric // This file contains the pass management machinery for machine functions.
10b60736ecSDimitry Andric //
11b60736ecSDimitry Andric //===----------------------------------------------------------------------===//
12b60736ecSDimitry Andric 
13b60736ecSDimitry Andric #include "llvm/CodeGen/MachinePassManager.h"
14b60736ecSDimitry Andric #include "llvm/CodeGen/MachineFunction.h"
15ac9a064cSDimitry Andric #include "llvm/CodeGen/MachineFunctionAnalysis.h"
16b60736ecSDimitry Andric #include "llvm/CodeGen/MachineModuleInfo.h"
17b60736ecSDimitry Andric #include "llvm/IR/PassManagerImpl.h"
18b60736ecSDimitry Andric 
19b60736ecSDimitry Andric using namespace llvm;
20b60736ecSDimitry Andric 
21ac9a064cSDimitry Andric AnalysisKey FunctionAnalysisManagerMachineFunctionProxy::Key;
22ac9a064cSDimitry Andric 
23b60736ecSDimitry Andric namespace llvm {
24b60736ecSDimitry Andric template class AnalysisManager<MachineFunction>;
25b60736ecSDimitry Andric template class PassManager<MachineFunction>;
26ac9a064cSDimitry Andric template class InnerAnalysisManagerProxy<MachineFunctionAnalysisManager,
27ac9a064cSDimitry Andric                                          Module>;
28ac9a064cSDimitry Andric template class InnerAnalysisManagerProxy<MachineFunctionAnalysisManager,
29ac9a064cSDimitry Andric                                          Function>;
30ac9a064cSDimitry Andric template class OuterAnalysisManagerProxy<ModuleAnalysisManager,
31ac9a064cSDimitry Andric                                          MachineFunction>;
32ac9a064cSDimitry Andric } // namespace llvm
33b60736ecSDimitry Andric 
invalidate(MachineFunction & IR,const PreservedAnalyses & PA,MachineFunctionAnalysisManager::Invalidator & Inv)34ac9a064cSDimitry Andric bool FunctionAnalysisManagerMachineFunctionProxy::Result::invalidate(
35ac9a064cSDimitry Andric     MachineFunction &IR, const PreservedAnalyses &PA,
36ac9a064cSDimitry Andric     MachineFunctionAnalysisManager::Invalidator &Inv) {
37ac9a064cSDimitry Andric   // MachineFunction passes should not invalidate Function analyses.
38ac9a064cSDimitry Andric   // TODO: verify that PA doesn't invalidate Function analyses.
39ac9a064cSDimitry Andric   return false;
40b60736ecSDimitry Andric }
41b60736ecSDimitry Andric 
42ac9a064cSDimitry Andric template <>
invalidate(Module & M,const PreservedAnalyses & PA,ModuleAnalysisManager::Invalidator & Inv)43ac9a064cSDimitry Andric bool MachineFunctionAnalysisManagerModuleProxy::Result::invalidate(
44ac9a064cSDimitry Andric     Module &M, const PreservedAnalyses &PA,
45ac9a064cSDimitry Andric     ModuleAnalysisManager::Invalidator &Inv) {
46ac9a064cSDimitry Andric   // If literally everything is preserved, we're done.
47ac9a064cSDimitry Andric   if (PA.areAllPreserved())
48ac9a064cSDimitry Andric     return false; // This is still a valid proxy.
49ac9a064cSDimitry Andric 
50ac9a064cSDimitry Andric   // If this proxy isn't marked as preserved, then even if the result remains
51ac9a064cSDimitry Andric   // valid, the key itself may no longer be valid, so we clear everything.
52ac9a064cSDimitry Andric   //
53ac9a064cSDimitry Andric   // Note that in order to preserve this proxy, a module pass must ensure that
54ac9a064cSDimitry Andric   // the MFAM has been completely updated to handle the deletion of functions.
55ac9a064cSDimitry Andric   // Specifically, any MFAM-cached results for those functions need to have been
56ac9a064cSDimitry Andric   // forcibly cleared. When preserved, this proxy will only invalidate results
57ac9a064cSDimitry Andric   // cached on functions *still in the module* at the end of the module pass.
58ac9a064cSDimitry Andric   auto PAC = PA.getChecker<MachineFunctionAnalysisManagerModuleProxy>();
59ac9a064cSDimitry Andric   if (!PAC.preserved() && !PAC.preservedSet<AllAnalysesOn<Module>>()) {
60ac9a064cSDimitry Andric     InnerAM->clear();
61ac9a064cSDimitry Andric     return true;
62b60736ecSDimitry Andric   }
63b60736ecSDimitry Andric 
64ac9a064cSDimitry Andric   // FIXME: be more precise, see
65ac9a064cSDimitry Andric   // FunctionAnalysisManagerModuleProxy::Result::invalidate.
66ac9a064cSDimitry Andric   if (!PA.allAnalysesInSetPreserved<AllAnalysesOn<MachineFunction>>()) {
67ac9a064cSDimitry Andric     InnerAM->clear();
68ac9a064cSDimitry Andric     return true;
69b60736ecSDimitry Andric   }
70b60736ecSDimitry Andric 
71ac9a064cSDimitry Andric   // Return false to indicate that this result is still a valid proxy.
72ac9a064cSDimitry Andric   return false;
73ac9a064cSDimitry Andric }
74b60736ecSDimitry Andric 
75ac9a064cSDimitry Andric template <>
invalidate(Function & F,const PreservedAnalyses & PA,FunctionAnalysisManager::Invalidator & Inv)76ac9a064cSDimitry Andric bool MachineFunctionAnalysisManagerFunctionProxy::Result::invalidate(
77ac9a064cSDimitry Andric     Function &F, const PreservedAnalyses &PA,
78ac9a064cSDimitry Andric     FunctionAnalysisManager::Invalidator &Inv) {
79ac9a064cSDimitry Andric   // If literally everything is preserved, we're done.
80ac9a064cSDimitry Andric   if (PA.areAllPreserved())
81ac9a064cSDimitry Andric     return false; // This is still a valid proxy.
82b60736ecSDimitry Andric 
83ac9a064cSDimitry Andric   // If this proxy isn't marked as preserved, then even if the result remains
84ac9a064cSDimitry Andric   // valid, the key itself may no longer be valid, so we clear everything.
85ac9a064cSDimitry Andric   //
86ac9a064cSDimitry Andric   // Note that in order to preserve this proxy, a module pass must ensure that
87ac9a064cSDimitry Andric   // the MFAM has been completely updated to handle the deletion of functions.
88ac9a064cSDimitry Andric   // Specifically, any MFAM-cached results for those functions need to have been
89ac9a064cSDimitry Andric   // forcibly cleared. When preserved, this proxy will only invalidate results
90ac9a064cSDimitry Andric   // cached on functions *still in the module* at the end of the module pass.
91ac9a064cSDimitry Andric   auto PAC = PA.getChecker<MachineFunctionAnalysisManagerFunctionProxy>();
92ac9a064cSDimitry Andric   if (!PAC.preserved() && !PAC.preservedSet<AllAnalysesOn<Function>>()) {
93ac9a064cSDimitry Andric     InnerAM->clear();
94ac9a064cSDimitry Andric     return true;
95ac9a064cSDimitry Andric   }
96b60736ecSDimitry Andric 
97ac9a064cSDimitry Andric   // FIXME: be more precise, see
98ac9a064cSDimitry Andric   // FunctionAnalysisManagerModuleProxy::Result::invalidate.
99ac9a064cSDimitry Andric   if (!PA.allAnalysesInSetPreserved<AllAnalysesOn<MachineFunction>>()) {
100ac9a064cSDimitry Andric     InnerAM->clear();
101ac9a064cSDimitry Andric     return true;
102ac9a064cSDimitry Andric   }
103ac9a064cSDimitry Andric 
104ac9a064cSDimitry Andric   // Return false to indicate that this result is still a valid proxy.
105ac9a064cSDimitry Andric   return false;
106ac9a064cSDimitry Andric }
107ac9a064cSDimitry Andric 
108ac9a064cSDimitry Andric PreservedAnalyses
run(Function & F,FunctionAnalysisManager & FAM)109ac9a064cSDimitry Andric FunctionToMachineFunctionPassAdaptor::run(Function &F,
110ac9a064cSDimitry Andric                                           FunctionAnalysisManager &FAM) {
111ac9a064cSDimitry Andric   MachineFunctionAnalysisManager &MFAM =
112ac9a064cSDimitry Andric       FAM.getResult<MachineFunctionAnalysisManagerFunctionProxy>(F)
113ac9a064cSDimitry Andric           .getManager();
114ac9a064cSDimitry Andric   PassInstrumentation PI = FAM.getResult<PassInstrumentationAnalysis>(F);
115ac9a064cSDimitry Andric   PreservedAnalyses PA = PreservedAnalyses::all();
116b60736ecSDimitry Andric   // Do not codegen any 'available_externally' functions at all, they have
117b60736ecSDimitry Andric   // definitions outside the translation unit.
118ac9a064cSDimitry Andric   if (F.isDeclaration() || F.hasAvailableExternallyLinkage())
119ac9a064cSDimitry Andric     return PreservedAnalyses::all();
120b60736ecSDimitry Andric 
121ac9a064cSDimitry Andric   MachineFunction &MF = FAM.getResult<MachineFunctionAnalysis>(F).getMF();
122b60736ecSDimitry Andric 
123ac9a064cSDimitry Andric   if (!PI.runBeforePass<MachineFunction>(*Pass, MF))
124ac9a064cSDimitry Andric     return PreservedAnalyses::all();
125ac9a064cSDimitry Andric   PreservedAnalyses PassPA = Pass->run(MF, MFAM);
126b60736ecSDimitry Andric   MFAM.invalidate(MF, PassPA);
127ac9a064cSDimitry Andric   PI.runAfterPass(*Pass, MF, PassPA);
128ac9a064cSDimitry Andric   PA.intersect(std::move(PassPA));
129b60736ecSDimitry Andric 
130ac9a064cSDimitry Andric   return PA;
131b60736ecSDimitry Andric }
132b60736ecSDimitry Andric 
printPipeline(raw_ostream & OS,function_ref<StringRef (StringRef)> MapClassName2PassName)133ac9a064cSDimitry Andric void FunctionToMachineFunctionPassAdaptor::printPipeline(
134ac9a064cSDimitry Andric     raw_ostream &OS, function_ref<StringRef(StringRef)> MapClassName2PassName) {
135ac9a064cSDimitry Andric   OS << "machine-function(";
136ac9a064cSDimitry Andric   Pass->printPipeline(OS, MapClassName2PassName);
137ac9a064cSDimitry Andric   OS << ')';
138b60736ecSDimitry Andric }
139b60736ecSDimitry Andric 
140ac9a064cSDimitry Andric template <>
141ac9a064cSDimitry Andric PreservedAnalyses
run(MachineFunction & MF,AnalysisManager<MachineFunction> & MFAM)142ac9a064cSDimitry Andric PassManager<MachineFunction>::run(MachineFunction &MF,
143ac9a064cSDimitry Andric                                   AnalysisManager<MachineFunction> &MFAM) {
144ac9a064cSDimitry Andric   PassInstrumentation PI = MFAM.getResult<PassInstrumentationAnalysis>(MF);
145ac9a064cSDimitry Andric   PreservedAnalyses PA = PreservedAnalyses::all();
146ac9a064cSDimitry Andric   for (auto &Pass : Passes) {
147ac9a064cSDimitry Andric     if (!PI.runBeforePass<MachineFunction>(*Pass, MF))
148ac9a064cSDimitry Andric       continue;
149ac9a064cSDimitry Andric 
150ac9a064cSDimitry Andric     PreservedAnalyses PassPA = Pass->run(MF, MFAM);
151ac9a064cSDimitry Andric     MFAM.invalidate(MF, PassPA);
152ac9a064cSDimitry Andric     PI.runAfterPass(*Pass, MF, PassPA);
153ac9a064cSDimitry Andric     PA.intersect(std::move(PassPA));
154ac9a064cSDimitry Andric   }
155ac9a064cSDimitry Andric   return PA;
156ac9a064cSDimitry Andric }
157ac9a064cSDimitry Andric 
getMachineFunctionPassPreservedAnalyses()158ac9a064cSDimitry Andric PreservedAnalyses llvm::getMachineFunctionPassPreservedAnalyses() {
159ac9a064cSDimitry Andric   PreservedAnalyses PA;
160ac9a064cSDimitry Andric   // Machine function passes are not allowed to modify the LLVM
161ac9a064cSDimitry Andric   // representation, therefore we should preserve all IR analyses.
162ac9a064cSDimitry Andric   PA.template preserveSet<AllAnalysesOn<Module>>();
163ac9a064cSDimitry Andric   PA.template preserveSet<AllAnalysesOn<Function>>();
164ac9a064cSDimitry Andric   return PA;
165ac9a064cSDimitry Andric }
166