xref: /freebsd/contrib/llvm-project/llvm/include/llvm/Transforms/IPO/SampleProfileMatcher.h (revision b64c5a0ace59af62eff52bfe110a521dc73c937b)
1 //===- Transforms/IPO/SampleProfileMatcher.h ----------*- 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 //
9 /// \file
10 /// This file provides the interface for SampleProfileMatcher.
11 //
12 //===----------------------------------------------------------------------===//
13 
14 #ifndef LLVM_TRANSFORMS_IPO_SAMPLEPROFILEMATCHER_H
15 #define LLVM_TRANSFORMS_IPO_SAMPLEPROFILEMATCHER_H
16 
17 #include "llvm/ADT/StringSet.h"
18 #include "llvm/Transforms/Utils/SampleProfileLoaderBaseImpl.h"
19 
20 namespace llvm {
21 
22 using AnchorList = std::vector<std::pair<LineLocation, FunctionId>>;
23 using AnchorMap = std::map<LineLocation, FunctionId>;
24 
25 // Sample profile matching - fuzzy match.
26 class SampleProfileMatcher {
27   Module &M;
28   SampleProfileReader &Reader;
29   LazyCallGraph &CG;
30   const PseudoProbeManager *ProbeManager;
31   const ThinOrFullLTOPhase LTOPhase;
32   SampleProfileMap FlattenedProfiles;
33   // For each function, the matcher generates a map, of which each entry is a
34   // mapping from the source location of current build to the source location
35   // in the profile.
36   StringMap<LocToLocMap> FuncMappings;
37 
38   // Match state for an anchor/callsite.
39   enum class MatchState {
40     Unknown = 0,
41     // Initial match between input profile and current IR.
42     InitialMatch = 1,
43     // Initial mismatch between input profile and current IR.
44     InitialMismatch = 2,
45     // InitialMatch stays matched after fuzzy profile matching.
46     UnchangedMatch = 3,
47     // InitialMismatch stays mismatched after fuzzy profile matching.
48     UnchangedMismatch = 4,
49     // InitialMismatch is recovered after fuzzy profile matching.
50     RecoveredMismatch = 5,
51     // InitialMatch is removed and becomes mismatched after fuzzy profile
52     // matching.
53     RemovedMatch = 6,
54   };
55 
56   // For each function, store every callsite and its matching state into this
57   // map, of which each entry is a pair of callsite location and MatchState.
58   // This is used for profile staleness computation and report.
59   StringMap<std::unordered_map<LineLocation, MatchState, LineLocationHash>>
60       FuncCallsiteMatchStates;
61 
62   struct FuncToProfileNameMapHash {
63     uint64_t
64     operator()(const std::pair<const Function *, FunctionId> &P) const {
65       return hash_combine(P.first, P.second);
66     }
67   };
68   // A map from a pair of function and profile name to a boolean value
69   // indicating whether they are matched. This is used as a cache for the
70   // matching result.
71   std::unordered_map<std::pair<const Function *, FunctionId>, bool,
72                      FuncToProfileNameMapHash>
73       FuncProfileMatchCache;
74   // The new functions found by the call graph matching. The map's key is the
75   // the new(renamed) function pointer and the value is old(unused) profile
76   // name.
77   std::unordered_map<Function *, FunctionId> FuncToProfileNameMap;
78 
79   // A map pointer to the FuncNameToProfNameMap in SampleProfileLoader,
80   // which maps the function name to the matched profile name. This is used
81   // for sample loader to look up profile using the new name.
82   HashKeyMap<std::unordered_map, FunctionId, FunctionId> *FuncNameToProfNameMap;
83 
84   // A map pointer to the SymbolMap in SampleProfileLoader, which stores all
85   // the original matched symbols before the matching. this is to determine if
86   // the profile is unused(to be matched) or not.
87   HashKeyMap<std::unordered_map, FunctionId, Function *> *SymbolMap;
88 
89   // The new functions from IR.
90   HashKeyMap<std::unordered_map, FunctionId, Function *>
91       FunctionsWithoutProfile;
92 
93   // Pointer to the Profile Symbol List in the reader.
94   std::shared_ptr<ProfileSymbolList> PSL;
95 
96   // Profile mismatch statstics:
97   uint64_t TotalProfiledFunc = 0;
98   // Num of checksum-mismatched function.
99   uint64_t NumStaleProfileFunc = 0;
100   uint64_t TotalProfiledCallsites = 0;
101   uint64_t NumMismatchedCallsites = 0;
102   uint64_t NumRecoveredCallsites = 0;
103   // Total samples for all profiled functions.
104   uint64_t TotalFunctionSamples = 0;
105   // Total samples for all checksum-mismatched functions.
106   uint64_t MismatchedFunctionSamples = 0;
107   uint64_t MismatchedCallsiteSamples = 0;
108   uint64_t RecoveredCallsiteSamples = 0;
109 
110   // Profile call-graph matching statstics:
111   uint64_t NumCallGraphRecoveredProfiledFunc = 0;
112   uint64_t NumCallGraphRecoveredFuncSamples = 0;
113 
114   // A dummy name for unknown indirect callee, used to differentiate from a
115   // non-call instruction that also has an empty callee name.
116   static constexpr const char *UnknownIndirectCallee =
117       "unknown.indirect.callee";
118 
119 public:
120   SampleProfileMatcher(
121       Module &M, SampleProfileReader &Reader, LazyCallGraph &CG,
122       const PseudoProbeManager *ProbeManager, ThinOrFullLTOPhase LTOPhase,
123       HashKeyMap<std::unordered_map, FunctionId, Function *> &SymMap,
124       std::shared_ptr<ProfileSymbolList> PSL,
125       HashKeyMap<std::unordered_map, FunctionId, FunctionId>
126           &FuncNameToProfNameMap)
127       : M(M), Reader(Reader), CG(CG), ProbeManager(ProbeManager),
128         LTOPhase(LTOPhase), FuncNameToProfNameMap(&FuncNameToProfNameMap),
129         SymbolMap(&SymMap), PSL(PSL) {};
130   void runOnModule();
131   void clearMatchingData() {
132     // Do not clear FuncMappings, it stores IRLoc to ProfLoc remappings which
133     // will be used for sample loader.
134     // Do not clear FlattenedProfiles as it contains function names referenced
135     // by FuncNameToProfNameMap. Clearing this memory could lead to a
136     // use-after-free error.
137     freeContainer(FuncCallsiteMatchStates);
138     freeContainer(FunctionsWithoutProfile);
139     freeContainer(FuncToProfileNameMap);
140   }
141 
142 private:
143   FunctionSamples *getFlattenedSamplesFor(const FunctionId &Fname) {
144     auto It = FlattenedProfiles.find(Fname);
145     if (It != FlattenedProfiles.end())
146       return &It->second;
147     return nullptr;
148   }
149   FunctionSamples *getFlattenedSamplesFor(const Function &F) {
150     StringRef CanonFName = FunctionSamples::getCanonicalFnName(F);
151     return getFlattenedSamplesFor(FunctionId(CanonFName));
152   }
153   template <typename T> inline void freeContainer(T &C) {
154     T Empty;
155     std::swap(C, Empty);
156   }
157   void getFilteredAnchorList(const AnchorMap &IRAnchors,
158                              const AnchorMap &ProfileAnchors,
159                              AnchorList &FilteredIRAnchorsList,
160                              AnchorList &FilteredProfileAnchorList);
161   void runOnFunction(Function &F);
162   void findIRAnchors(const Function &F, AnchorMap &IRAnchors) const;
163   void findProfileAnchors(const FunctionSamples &FS,
164                           AnchorMap &ProfileAnchors) const;
165   // Record the callsite match states for profile staleness report, the result
166   // is saved in FuncCallsiteMatchStates.
167   void recordCallsiteMatchStates(const Function &F, const AnchorMap &IRAnchors,
168                                  const AnchorMap &ProfileAnchors,
169                                  const LocToLocMap *IRToProfileLocationMap);
170 
171   bool isMismatchState(const enum MatchState &State) {
172     return State == MatchState::InitialMismatch ||
173            State == MatchState::UnchangedMismatch ||
174            State == MatchState::RemovedMatch;
175   };
176 
177   bool isInitialState(const enum MatchState &State) {
178     return State == MatchState::InitialMatch ||
179            State == MatchState::InitialMismatch;
180   };
181 
182   bool isFinalState(const enum MatchState &State) {
183     return State == MatchState::UnchangedMatch ||
184            State == MatchState::UnchangedMismatch ||
185            State == MatchState::RecoveredMismatch ||
186            State == MatchState::RemovedMatch;
187   };
188 
189   void countCallGraphRecoveredSamples(
190       const FunctionSamples &FS,
191       std::unordered_set<FunctionId> &MatchedUnusedProfile);
192   // Count the samples of checksum mismatched function for the top-level
193   // function and all inlinees.
194   void countMismatchedFuncSamples(const FunctionSamples &FS, bool IsTopLevel);
195   // Count the number of mismatched or recovered callsites.
196   void countMismatchCallsites(const FunctionSamples &FS);
197   // Count the samples of mismatched or recovered callsites for top-level
198   // function and all inlinees.
199   void countMismatchedCallsiteSamples(const FunctionSamples &FS);
200   void computeAndReportProfileStaleness();
201 
202   LocToLocMap &getIRToProfileLocationMap(const Function &F) {
203     auto Ret = FuncMappings.try_emplace(
204         FunctionSamples::getCanonicalFnName(F.getName()), LocToLocMap());
205     return Ret.first->second;
206   }
207   void distributeIRToProfileLocationMap();
208   void distributeIRToProfileLocationMap(FunctionSamples &FS);
209   // This function implements the Myers diff algorithm used for stale profile
210   // matching. The algorithm provides a simple and efficient way to find the
211   // Longest Common Subsequence(LCS) or the Shortest Edit Script(SES) of two
212   // sequences. For more details, refer to the paper 'An O(ND) Difference
213   // Algorithm and Its Variations' by Eugene W. Myers.
214   // In the scenario of profile fuzzy matching, the two sequences are the IR
215   // callsite anchors and profile callsite anchors. The subsequence equivalent
216   // parts from the resulting SES are used to remap the IR locations to the
217   // profile locations. As the number of function callsite is usually not big,
218   // we currently just implements the basic greedy version(page 6 of the paper).
219   LocToLocMap longestCommonSequence(const AnchorList &IRCallsiteAnchors,
220                                     const AnchorList &ProfileCallsiteAnchors,
221                                     bool MatchUnusedFunction);
222   void matchNonCallsiteLocs(const LocToLocMap &AnchorMatchings,
223                             const AnchorMap &IRAnchors,
224                             LocToLocMap &IRToProfileLocationMap);
225   void runStaleProfileMatching(const Function &F, const AnchorMap &IRAnchors,
226                                const AnchorMap &ProfileAnchors,
227                                LocToLocMap &IRToProfileLocationMap,
228                                bool RunCFGMatching, bool RunCGMatching);
229   // If the function doesn't have profile, return the pointer to the function.
230   bool functionHasProfile(const FunctionId &IRFuncName,
231                           Function *&FuncWithoutProfile);
232   bool isProfileUnused(const FunctionId &ProfileFuncName);
233   bool functionMatchesProfileHelper(const Function &IRFunc,
234                                     const FunctionId &ProfFunc);
235   // Determine if the function matches profile. If FindMatchedProfileOnly is
236   // set, only search the existing matched function. Otherwise, try matching the
237   // two functions.
238   bool functionMatchesProfile(const FunctionId &IRFuncName,
239                               const FunctionId &ProfileFuncName,
240                               bool FindMatchedProfileOnly);
241   // Determine if the function matches profile by computing a similarity ratio
242   // between two sequences of callsite anchors extracted from function and
243   // profile. If it's above the threshold, the function matches the profile.
244   bool functionMatchesProfile(Function &IRFunc, const FunctionId &ProfFunc,
245                               bool FindMatchedProfileOnly);
246   // Find functions that don't show in the profile or profile symbol list,
247   // which are supposed to be new functions. We use them as the targets for
248   // call graph matching.
249   void findFunctionsWithoutProfile();
250   void reportOrPersistProfileStats();
251 };
252 } // end namespace llvm
253 #endif // LLVM_TRANSFORMS_IPO_SAMPLEPROFILEMATCHER_H
254