LLVM 23.0.0git
AArch64O0PreLegalizerCombiner.cpp
Go to the documentation of this file.
1//=== lib/CodeGen/GlobalISel/AArch64O0PreLegalizerCombiner.cpp ------------===//
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 pass does combining of machine instructions at the generic MI level,
10// before the legalizer.
11//
12//===----------------------------------------------------------------------===//
13
14#include "AArch64.h"
32#include <memory>
33
34#define GET_GICOMBINER_DEPS
35#include "AArch64GenO0PreLegalizeGICombiner.inc"
36#undef GET_GICOMBINER_DEPS
37
38#define DEBUG_TYPE "aarch64-O0-prelegalizer-combiner"
39
40using namespace llvm;
41using namespace MIPatternMatch;
42
43#define GET_GICOMBINER_TYPES
44#include "AArch64GenO0PreLegalizeGICombiner.inc"
45#undef GET_GICOMBINER_TYPES
46
47namespace {
48
49class AArch64O0PreLegalizerCombinerImpl : public Combiner {
50protected:
51 const CombinerHelper Helper;
52 const AArch64O0PreLegalizerCombinerImplRuleConfig &RuleConfig;
53 const AArch64Subtarget &STI;
54 const LibcallLoweringInfo &Libcalls;
55
56public:
57 AArch64O0PreLegalizerCombinerImpl(
59 GISelCSEInfo *CSEInfo,
60 const AArch64O0PreLegalizerCombinerImplRuleConfig &RuleConfig,
61 const AArch64Subtarget &STI, const LibcallLoweringInfo &Libcalls);
62
63 static const char *getName() { return "AArch64O0PreLegalizerCombiner"; }
64
65 bool tryCombineAll(MachineInstr &I) const override;
66
67 bool tryCombineAllImpl(MachineInstr &I) const;
68
69private:
70#define GET_GICOMBINER_CLASS_MEMBERS
71#include "AArch64GenO0PreLegalizeGICombiner.inc"
72#undef GET_GICOMBINER_CLASS_MEMBERS
73};
74
75#define GET_GICOMBINER_IMPL
76#include "AArch64GenO0PreLegalizeGICombiner.inc"
77#undef GET_GICOMBINER_IMPL
78
79AArch64O0PreLegalizerCombinerImpl::AArch64O0PreLegalizerCombinerImpl(
81 GISelCSEInfo *CSEInfo,
82 const AArch64O0PreLegalizerCombinerImplRuleConfig &RuleConfig,
83 const AArch64Subtarget &STI, const LibcallLoweringInfo &Libcalls)
84 : Combiner(MF, CInfo, &VT, CSEInfo),
85 Helper(Observer, B, /*IsPreLegalize*/ true, &VT), RuleConfig(RuleConfig),
86 STI(STI), Libcalls(Libcalls),
88#include "AArch64GenO0PreLegalizeGICombiner.inc"
90{
91}
92
93bool AArch64O0PreLegalizerCombinerImpl::tryCombineAll(MachineInstr &MI) const {
94 if (tryCombineAllImpl(MI))
95 return true;
96
97 unsigned Opc = MI.getOpcode();
98 switch (Opc) {
99 case TargetOpcode::G_SHUFFLE_VECTOR:
100 return Helper.tryCombineShuffleVector(MI);
101 case TargetOpcode::G_MEMCPY_INLINE:
102 return Helper.tryEmitMemcpyInline(MI);
103 case TargetOpcode::G_MEMCPY:
104 case TargetOpcode::G_MEMMOVE:
105 case TargetOpcode::G_MEMSET: {
106 // At -O0 set a maxlen of 32 to inline;
107 unsigned MaxLen = 32;
108 // Try to inline memcpy type calls if optimizations are enabled.
109 if (Helper.tryCombineMemCpyFamily(MI, MaxLen))
110 return true;
111 if (Opc == TargetOpcode::G_MEMSET)
113 CInfo.EnableMinSize);
114 return false;
115 }
116 }
117
118 return false;
119}
120
121bool runCombiner(
122 MachineFunction &MF, GISelValueTracking *VT,
123 const LibcallLoweringInfo &Libcalls,
124 const AArch64O0PreLegalizerCombinerImplRuleConfig &RuleConfig) {
125 const Function &F = MF.getFunction();
126 const AArch64Subtarget &ST = MF.getSubtarget<AArch64Subtarget>();
127
128 CombinerInfo CInfo(/*AllowIllegalOps=*/true, /*ShouldLegalizeIllegal=*/false,
129 /*LegalizerInfo=*/nullptr, /*EnableOpt=*/false,
130 F.hasOptSize(), F.hasMinSize());
131 // Disable fixed-point iteration in the Combiner. This improves compile-time
132 // at the cost of possibly missing optimizations. See PR#94291 for details.
133 CInfo.MaxIterations = 1;
134
135 AArch64O0PreLegalizerCombinerImpl Impl(MF, CInfo, *VT,
136 /*CSEInfo*/ nullptr, RuleConfig, ST,
137 Libcalls);
138 return Impl.combineMachineInstrs();
139}
140
141// Pass boilerplate
142// ================
143
144class AArch64O0PreLegalizerCombinerLegacy : public MachineFunctionPass {
145public:
146 static char ID;
147
148 AArch64O0PreLegalizerCombinerLegacy();
149
150 StringRef getPassName() const override {
151 return "AArch64O0PreLegalizerCombiner";
152 }
153
154 bool runOnMachineFunction(MachineFunction &MF) override;
155
156 void getAnalysisUsage(AnalysisUsage &AU) const override;
157
158private:
159 AArch64O0PreLegalizerCombinerImplRuleConfig RuleConfig;
160};
161} // end anonymous namespace
162
163void AArch64O0PreLegalizerCombinerLegacy::getAnalysisUsage(
164 AnalysisUsage &AU) const {
165 AU.setPreservesCFG();
167 AU.addRequired<GISelValueTrackingAnalysisLegacy>();
168 AU.addPreserved<GISelValueTrackingAnalysisLegacy>();
169 AU.addRequired<LibcallLoweringInfoWrapper>();
171}
172
173AArch64O0PreLegalizerCombinerLegacy::AArch64O0PreLegalizerCombinerLegacy()
174 : MachineFunctionPass(ID) {
175 if (!RuleConfig.parseCommandLineOption())
176 report_fatal_error("Invalid rule identifier");
177}
178
179bool AArch64O0PreLegalizerCombinerLegacy::runOnMachineFunction(
180 MachineFunction &MF) {
181 if (MF.getProperties().hasFailedISel())
182 return false;
183
184 const Function &F = MF.getFunction();
185 GISelValueTracking *VT =
186 &getAnalysis<GISelValueTrackingAnalysisLegacy>().get(MF);
187
188 const AArch64Subtarget &ST = MF.getSubtarget<AArch64Subtarget>();
189 const LibcallLoweringInfo &Libcalls =
190 getAnalysis<LibcallLoweringInfoWrapper>().getLibcallLowering(
191 *F.getParent(), ST);
192
193 return runCombiner(MF, VT, Libcalls, RuleConfig);
194}
195
196char AArch64O0PreLegalizerCombinerLegacy::ID = 0;
197INITIALIZE_PASS_BEGIN(AArch64O0PreLegalizerCombinerLegacy, DEBUG_TYPE,
198 "Combine AArch64 machine instrs before legalization",
199 false, false)
203INITIALIZE_PASS_END(AArch64O0PreLegalizerCombinerLegacy, DEBUG_TYPE,
204 "Combine AArch64 machine instrs before legalization", false,
205 false)
206
208 : RuleConfig(
209 std::make_unique<AArch64O0PreLegalizerCombinerImplRuleConfig>()) {
210 if (!RuleConfig->parseCommandLineOption())
211 report_fatal_error("Invalid rule identifier");
212}
213
216
218 default;
219
223 if (MF.getProperties().hasFailedISel())
224 return PreservedAnalyses::all();
225
227
229 auto &MAMProxy =
231 const LibcallLoweringModuleAnalysisResult *LibcallResult =
232 MAMProxy.getCachedResult<LibcallLoweringModuleAnalysis>(
233 *MF.getFunction().getParent());
234 if (!LibcallResult)
235 reportFatalUsageError("LibcallLoweringModuleAnalysis result not available");
236
237 const LibcallLoweringInfo &Libcalls = LibcallResult->getLibcallLowering(ST);
238
239 if (!runCombiner(MF, &VT, Libcalls, *RuleConfig))
240 return PreservedAnalyses::all();
241
245 return PA;
246}
247
248namespace llvm {
250 return new AArch64O0PreLegalizerCombinerLegacy();
251}
252} // end namespace llvm
#define GET_GICOMBINER_CONSTRUCTOR_INITS
static const Function * getParent(const Value *V)
static GCRegistry::Add< OcamlGC > B("ocaml", "ocaml 3.10-compatible GC")
This contains common combine transformations that may be used in a combine pass,or by the target else...
Option class for Targets to specify which operations are combined how and when.
This contains the base class for all Combiners generated by TableGen.
Provides analysis for querying information about KnownBits during GISel passes.
#define DEBUG_TYPE
IRTranslator LLVM IR MI
#define F(x, y, z)
Definition MD5.cpp:54
#define I(x, y, z)
Definition MD5.cpp:57
Contains matchers for matching SSA Machine Instructions.
This file declares the MachineIRBuilder class.
#define INITIALIZE_PASS_DEPENDENCY(depName)
Definition PassSupport.h:42
#define INITIALIZE_PASS_END(passName, arg, name, cfg, analysis)
Definition PassSupport.h:44
#define INITIALIZE_PASS_BEGIN(passName, arg, name, cfg, analysis)
Definition PassSupport.h:39
static StringRef getName(Value *V)
Target-Independent Code Generator Pass Configuration Options pass.
PreservedAnalyses run(MachineFunction &MF, MachineFunctionAnalysisManager &MFAM)
PassT::Result & getResult(IRUnitT &IR, ExtraArgTs... ExtraArgs)
Get the result of an analysis pass for a given IR unit.
AnalysisUsage & addRequired()
AnalysisUsage & addPreserved()
Add the specified Pass class to the set of analyses preserved by this pass.
LLVM_ABI void setPreservesCFG()
This function should be called by the pass, iff they do not:
Definition Pass.cpp:270
Represents analyses that only rely on functions' control flow.
Definition Analysis.h:73
bool tryEmitMemcpyInline(MachineInstr &MI) const
Emit loads and stores that perform the given memcpy.
bool tryCombineShuffleVector(MachineInstr &MI) const
Try to combine G_SHUFFLE_VECTOR into G_CONCAT_VECTORS.
bool tryCombineMemCpyFamily(MachineInstr &MI, unsigned MaxLen=0) const
Optimize memcpy intrinsics et al, e.g.
Combiner implementation.
Definition Combiner.h:33
FunctionPass class - This class is used to implement most global optimizations.
Definition Pass.h:314
The actual analysis pass wrapper.
Definition CSEInfo.h:242
The CSE Analysis object.
Definition CSEInfo.h:72
To use KnownBitsInfo analysis in a pass, KnownBitsInfo &Info = getAnalysis<GISelValueTrackingInfoAnal...
Tracks which library functions to use for a particular subtarget.
Record a mapping from subtarget to LibcallLoweringInfo.
void getAnalysisUsage(AnalysisUsage &AU) const override
getAnalysisUsage - Subclasses that override getAnalysisUsage must call this.
const TargetSubtargetInfo & getSubtarget() const
getSubtarget - Return the subtarget for which this machine code is being compiled.
Function & getFunction()
Return the LLVM function that this machine code represents.
const MachineFunctionProperties & getProperties() const
Get the function properties.
Representation of each machine instruction.
A set of analyses that are preserved following a run of a transformation pass.
Definition Analysis.h:112
static PreservedAnalyses all()
Construct a special preserved set that preserves all passes.
Definition Analysis.h:118
PreservedAnalyses & preserveSet()
Mark an analysis set as preserved.
Definition Analysis.h:151
PreservedAnalyses & preserve()
Mark an analysis as preserved.
Definition Analysis.h:132
bool tryEmitBZero(MachineInstr &MI, MachineIRBuilder &MIRBuilder, const LibcallLoweringInfo &Libcalls, bool MinSize)
Replace a G_MEMSET with a value of 0 with a G_BZERO instruction if it is supported and beneficial to ...
unsigned ID
LLVM IR allows to use arbitrary numbers as calling convention identifiers.
Definition CallingConv.h:24
This is an optimization pass for GlobalISel generic memory operations.
OuterAnalysisManagerProxy< ModuleAnalysisManager, MachineFunction > ModuleAnalysisManagerMachineFunctionProxy
Provide the ModuleAnalysisManager to Function proxy.
FunctionPass * createAArch64O0PreLegalizerCombiner()
AnalysisManager< MachineFunction > MachineFunctionAnalysisManager
LLVM_ABI PreservedAnalyses getMachineFunctionPassPreservedAnalyses()
Returns the minimum set of Analyses that all machine function passes must preserve.
LLVM_ABI void report_fatal_error(Error Err, bool gen_crash_diag=true)
Definition Error.cpp:163
LLVM_ABI void getSelectionDAGFallbackAnalysisUsage(AnalysisUsage &AU)
Modify analysis usage so it preserves passes required for the SelectionDAG fallback.
Definition Utils.cpp:1126
LLVM_ABI void reportFatalUsageError(Error Err)
Report a fatal error that does not indicate a bug in LLVM.
Definition Error.cpp:177
Implement std::hash so that hash_code can be used in STL containers.
Definition BitVector.h:870