LLVM 22.0.0git
LowerAllowCheckPass.cpp
Go to the documentation of this file.
1//===- LowerAllowCheckPass.cpp ----------------------------------*- C++ -*-===//
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
10
12#include "llvm/ADT/Statistic.h"
14#include "llvm/ADT/StringRef.h"
17#include "llvm/IR/Constants.h"
22#include "llvm/IR/Intrinsics.h"
23#include "llvm/IR/Metadata.h"
24#include "llvm/IR/Module.h"
25#include "llvm/Support/Debug.h"
27#include <memory>
28#include <random>
29
30using namespace llvm;
31
32#define DEBUG_TYPE "lower-allow-check"
33
34static cl::opt<int>
35 HotPercentileCutoff("lower-allow-check-percentile-cutoff-hot",
36 cl::desc("Hot percentile cutoff."));
37
38static cl::opt<float>
39 RandomRate("lower-allow-check-random-rate",
40 cl::desc("Probability value in the range [0.0, 1.0] of "
41 "unconditional pseudo-random checks."));
42
43STATISTIC(NumChecksTotal, "Number of checks");
44STATISTIC(NumChecksRemoved, "Number of removed checks");
45
46struct RemarkInfo {
51 : Kind("Kind", II->getArgOperand(0)),
52 F("Function", II->getParent()->getParent()),
53 BB("Block", II->getParent()->getName()) {}
54};
55
57 bool Removed) {
58 if (Removed) {
59 ORE.emit([&]() {
61 return OptimizationRemark(DEBUG_TYPE, "Removed", II)
62 << "Removed check: Kind=" << Info.Kind << " F=" << Info.F
63 << " BB=" << Info.BB;
64 });
65 } else {
66 ORE.emit([&]() {
68 return OptimizationRemarkMissed(DEBUG_TYPE, "Allowed", II)
69 << "Allowed check: Kind=" << Info.Kind << " F=" << Info.F
70 << " BB=" << Info.BB;
71 });
72 }
73}
74
76 const ProfileSummaryInfo *PSI,
78 const LowerAllowCheckPass::Options &Opts) {
80 std::unique_ptr<RandomNumberGenerator> Rng;
81
82 auto GetRng = [&]() -> RandomNumberGenerator & {
83 if (!Rng)
84 Rng = F.getParent()->createRNG(F.getName());
85 return *Rng;
86 };
87
88 auto GetCutoff = [&](const IntrinsicInst *II) -> unsigned {
89 if (HotPercentileCutoff.getNumOccurrences())
91 else if (II->getIntrinsicID() == Intrinsic::allow_ubsan_check) {
92 auto *Kind = cast<ConstantInt>(II->getArgOperand(0));
93 if (Kind->getZExtValue() < Opts.cutoffs.size())
94 return Opts.cutoffs[Kind->getZExtValue()];
95 } else if (II->getIntrinsicID() == Intrinsic::allow_runtime_check) {
96 return Opts.runtime_check;
97 }
98
99 return 0;
100 };
101
102 auto ShouldRemoveHot = [&](const BasicBlock &BB, unsigned int cutoff) {
103 return (cutoff == 1000000) ||
104 (PSI && PSI->isHotCountNthPercentile(
105 cutoff, BFI.getBlockProfileCount(&BB).value_or(0)));
106 };
107
108 auto ShouldRemoveRandom = [&]() {
109 return RandomRate.getNumOccurrences() &&
110 !std::bernoulli_distribution(RandomRate)(GetRng());
111 };
112
113 auto ShouldRemove = [&](const IntrinsicInst *II) {
114 unsigned int cutoff = GetCutoff(II);
115 return ShouldRemoveRandom() || ShouldRemoveHot(*(II->getParent()), cutoff);
116 };
117
118 for (Instruction &I : instructions(F)) {
120 if (!II)
121 continue;
122 auto ID = II->getIntrinsicID();
123 switch (ID) {
124 case Intrinsic::allow_ubsan_check:
125 case Intrinsic::allow_runtime_check: {
126 ++NumChecksTotal;
127
128 bool ToRemove = ShouldRemove(II);
129
130 ReplaceWithValue.push_back({
131 II,
132 ToRemove,
133 });
134 if (ToRemove)
135 ++NumChecksRemoved;
136 emitRemark(II, ORE, ToRemove);
137 break;
138 }
139 default:
140 break;
141 }
142 }
143
144 for (auto [I, V] : ReplaceWithValue) {
145 I->replaceAllUsesWith(ConstantInt::getBool(I->getType(), !V));
146 I->eraseFromParent();
147 }
148
149 return !ReplaceWithValue.empty();
150}
151
154 if (F.isDeclaration())
155 return PreservedAnalyses::all();
156 auto &MAMProxy = AM.getResult<ModuleAnalysisManagerFunctionProxy>(F);
157 ProfileSummaryInfo *PSI =
158 MAMProxy.getCachedResult<ProfileSummaryAnalysis>(*F.getParent());
162
163 return lowerAllowChecks(F, BFI, PSI, ORE, Opts)
164 // We do not change the CFG, we only replace the intrinsics with
165 // true or false.
168}
169
171 return RandomRate.getNumOccurrences() ||
172 HotPercentileCutoff.getNumOccurrences();
173}
174
176 raw_ostream &OS, function_ref<StringRef(StringRef)> MapClassName2PassName) {
178 OS, MapClassName2PassName);
179 OS << "<";
180
181 // Format is <cutoffs[0,1,2]=70000;cutoffs[5,6,8]=90000>
182 // but it's equally valid to specify
183 // cutoffs[0]=70000;cutoffs[1]=70000;cutoffs[2]=70000;cutoffs[5]=90000;...
184 // and that's what we do here. It is verbose but valid and easy to verify
185 // correctness.
186 // TODO: print shorter output by combining adjacent runs, etc.
187 int i = 0;
188 ListSeparator LS(";");
189 for (unsigned int cutoff : Opts.cutoffs) {
190 if (cutoff > 0)
191 OS << LS << "cutoffs[" << i << "]=" << cutoff;
192 i++;
193 }
194 if (Opts.runtime_check)
195 OS << LS << "runtime_check=" << Opts.runtime_check;
196
197 OS << '>';
198}
ReachingDefAnalysis InstSet & ToRemove
Expand Atomic instructions
static const Function * getParent(const Value *V)
Analysis containing CSE Info
Definition CSEInfo.cpp:27
This file contains the declarations for the subclasses of Constant, which represent the different fla...
#define DEBUG_TYPE
Module.h This file contains the declarations for the Module class.
static cl::opt< float > RandomRate("lower-allow-check-random-rate", cl::desc("Probability value in the range [0.0, 1.0] of " "unconditional pseudo-random checks."))
static cl::opt< int > HotPercentileCutoff("lower-allow-check-percentile-cutoff-hot", cl::desc("Hot percentile cutoff."))
static void emitRemark(IntrinsicInst *II, OptimizationRemarkEmitter &ORE, bool Removed)
static bool lowerAllowChecks(Function &F, const BlockFrequencyInfo &BFI, const ProfileSummaryInfo *PSI, OptimizationRemarkEmitter &ORE, const LowerAllowCheckPass::Options &Opts)
This file provides the interface for the pass responsible for removing expensive ubsan checks.
#define F(x, y, z)
Definition MD5.cpp:55
#define I(x, y, z)
Definition MD5.cpp:58
This file contains the declarations for metadata subclasses.
uint64_t IntrinsicInst * II
static StringRef getName(Value *V)
This file defines the SmallVector class.
This file defines the 'Statistic' class, which is designed to be an easy way to expose various metric...
#define STATISTIC(VARNAME, DESC)
Definition Statistic.h:167
This file contains some functions that are useful when dealing with strings.
PassT::Result & getResult(IRUnitT &IR, ExtraArgTs... ExtraArgs)
Get the result of an analysis pass for a given IR unit.
LLVM Basic Block Representation.
Definition BasicBlock.h:62
Analysis pass which computes BlockFrequencyInfo.
BlockFrequencyInfo pass uses BlockFrequencyInfoImpl implementation to estimate IR basic block frequen...
Represents analyses that only rely on functions' control flow.
Definition Analysis.h:73
static LLVM_ABI ConstantInt * getBool(LLVMContext &Context, bool V)
A wrapper class for inspecting calls to intrinsic functions.
A helper class to return the specified delimiter string after the first invocation of operator String...
LLVM_ABI PreservedAnalyses run(Function &F, FunctionAnalysisManager &AM)
LLVM_ABI void printPipeline(raw_ostream &OS, function_ref< StringRef(StringRef)> MapClassName2PassName)
static LLVM_ABI bool IsRequested()
The optimization diagnostic interface.
LLVM_ABI void emit(DiagnosticInfoOptimizationBase &OptDiag)
Output the remark via the diagnostic handler and to the optimization record file.
Diagnostic information for missed-optimization remarks.
Diagnostic information for applied optimization remarks.
A set of analyses that are preserved following a run of a transformation pass.
Definition Analysis.h:112
static PreservedAnalyses none()
Convenience factory function for the empty preserved set.
Definition Analysis.h:115
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
An analysis pass based on the new PM to deliver ProfileSummaryInfo.
Analysis providing profile information.
LLVM_ABI bool isHotCountNthPercentile(int PercentileCutoff, uint64_t C) const
Returns true if count C is considered hot with regard to a given hot percentile cutoff value.
A random number generator.
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
StringRef - Represent a constant reference to a string, i.e.
Definition StringRef.h:55
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.
Definition raw_ostream.h:53
unsigned ID
LLVM IR allows to use arbitrary numbers as calling convention identifiers.
Definition CallingConv.h:24
DiagnosticInfoOptimizationBase::Argument NV
This is an optimization pass for GlobalISel generic memory operations.
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
Definition Casting.h:649
OuterAnalysisManagerProxy< ModuleAnalysisManager, Function > ModuleAnalysisManagerFunctionProxy
Provide the ModuleAnalysisManager to Function proxy.
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
Definition Casting.h:565
AnalysisManager< Function > FunctionAnalysisManager
Convenience typedef for the Function analysis manager.
RemarkInfo(IntrinsicInst *II)
std::vector< unsigned int > cutoffs
A CRTP mix-in to automatically provide informational APIs needed for passes.
Definition PassManager.h:70