mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2024-12-02 08:26:29 +00:00
[AutoFDO] Top-down Inlining for specialization with context-sensitive profile
Summary: AutoFDO's sample profile loader processes function in arbitrary source code order, so if I change the order of two functions in source code, the inline decision can change. This also prevented the use of context-sensitive profile to do specialization while inlining. This commit enforces SCC top-down order for sample profile loader. With this change, we can now do specialization, as illustrated by the added test case: Say if we have A->B->C and D->B->C call path, we want to inline C into B when root inliner is B, but not when root inliner is A or D, this is not possible without enforcing top-down order. E.g. Once C is inlined into B, A and D can only choose to inline (B->C) as a whole or nothing, but what we want is only inline B into A and D, not its recursive callee C. If we process functions in top-down order, this is no longer a problem, which is what this commit is doing. This change is guarded with a new switch "-sample-profile-top-down-load" for tuning, and it depends on D70653. Eventually, top-down can be the default order for sample profile loader. Reviewers: wmi, davidxl Subscribers: hiraditya, llvm-commits, tejohnson Tags: #llvm Differential Revision: https://reviews.llvm.org/D70655
This commit is contained in:
parent
bb9a3aaf60
commit
c0449913cc
@ -26,6 +26,7 @@
|
||||
#include "llvm/ADT/DenseMap.h"
|
||||
#include "llvm/ADT/DenseSet.h"
|
||||
#include "llvm/ADT/None.h"
|
||||
#include "llvm/ADT/SCCIterator.h"
|
||||
#include "llvm/ADT/SmallPtrSet.h"
|
||||
#include "llvm/ADT/SmallSet.h"
|
||||
#include "llvm/ADT/SmallVector.h"
|
||||
@ -33,6 +34,8 @@
|
||||
#include "llvm/ADT/StringRef.h"
|
||||
#include "llvm/ADT/Twine.h"
|
||||
#include "llvm/Analysis/AssumptionCache.h"
|
||||
#include "llvm/Analysis/CallGraph.h"
|
||||
#include "llvm/Analysis/CallGraphSCCPass.h"
|
||||
#include "llvm/Analysis/InlineCost.h"
|
||||
#include "llvm/Analysis/LoopInfo.h"
|
||||
#include "llvm/Analysis/OptimizationRemarkEmitter.h"
|
||||
@ -142,6 +145,11 @@ static cl::opt<bool> ProfileMergeInlinee(
|
||||
cl::desc("Merge past inlinee's profile to outline version if sample "
|
||||
"profile loader decided not to inline a call site."));
|
||||
|
||||
static cl::opt<bool> ProfileTopDownLoad(
|
||||
"sample-profile-top-down-load", cl::Hidden, cl::init(false),
|
||||
cl::desc("Do profile annotation and inlining for functions in top-down "
|
||||
"order of call graph during sample profile loading."));
|
||||
|
||||
namespace {
|
||||
|
||||
using BlockWeightMap = DenseMap<const BasicBlock *, uint64_t>;
|
||||
@ -291,7 +299,7 @@ public:
|
||||
|
||||
bool doInitialization(Module &M);
|
||||
bool runOnModule(Module &M, ModuleAnalysisManager *AM,
|
||||
ProfileSummaryInfo *_PSI);
|
||||
ProfileSummaryInfo *_PSI, CallGraph *CG);
|
||||
|
||||
void dump() { Reader->dump(); }
|
||||
|
||||
@ -323,6 +331,7 @@ protected:
|
||||
void propagateWeights(Function &F);
|
||||
uint64_t visitEdge(Edge E, unsigned *NumUnknownEdges, Edge *UnknownEdge);
|
||||
void buildEdges(Function &F);
|
||||
std::vector<Function *> buildFunctionOrder(Module &M, CallGraph *CG);
|
||||
bool propagateThroughEdges(Function &F, bool UpdateBlockCount);
|
||||
void computeDominanceAndLoopInfo(Function &F);
|
||||
void clearFunctionData();
|
||||
@ -1696,6 +1705,33 @@ INITIALIZE_PASS_DEPENDENCY(ProfileSummaryInfoWrapperPass)
|
||||
INITIALIZE_PASS_END(SampleProfileLoaderLegacyPass, "sample-profile",
|
||||
"Sample Profile loader", false, false)
|
||||
|
||||
std::vector<Function *>
|
||||
SampleProfileLoader::buildFunctionOrder(Module &M, CallGraph *CG) {
|
||||
std::vector<Function *> FunctionOrderList;
|
||||
FunctionOrderList.reserve(M.size());
|
||||
|
||||
if (!ProfileTopDownLoad || CG == nullptr) {
|
||||
for (Function &F : M)
|
||||
if (!F.isDeclaration())
|
||||
FunctionOrderList.push_back(&F);
|
||||
return FunctionOrderList;
|
||||
}
|
||||
|
||||
assert(&CG->getModule() == &M);
|
||||
scc_iterator<CallGraph *> CGI = scc_begin(CG);
|
||||
while (!CGI.isAtEnd()) {
|
||||
for (CallGraphNode *node : *CGI) {
|
||||
auto F = node->getFunction();
|
||||
if (F && !F->isDeclaration())
|
||||
FunctionOrderList.push_back(F);
|
||||
}
|
||||
++CGI;
|
||||
}
|
||||
|
||||
std::reverse(FunctionOrderList.begin(), FunctionOrderList.end());
|
||||
return FunctionOrderList;
|
||||
}
|
||||
|
||||
bool SampleProfileLoader::doInitialization(Module &M) {
|
||||
auto &Ctx = M.getContext();
|
||||
|
||||
@ -1733,7 +1769,7 @@ ModulePass *llvm::createSampleProfileLoaderPass(StringRef Name) {
|
||||
}
|
||||
|
||||
bool SampleProfileLoader::runOnModule(Module &M, ModuleAnalysisManager *AM,
|
||||
ProfileSummaryInfo *_PSI) {
|
||||
ProfileSummaryInfo *_PSI, CallGraph *CG) {
|
||||
GUIDToFuncNameMapper Mapper(M, *Reader, GUIDToFuncNameMap);
|
||||
if (!ProfileIsValid)
|
||||
return false;
|
||||
@ -1768,11 +1804,11 @@ bool SampleProfileLoader::runOnModule(Module &M, ModuleAnalysisManager *AM,
|
||||
}
|
||||
|
||||
bool retval = false;
|
||||
for (auto &F : M)
|
||||
if (!F.isDeclaration()) {
|
||||
clearFunctionData();
|
||||
retval |= runOnFunction(F, AM);
|
||||
}
|
||||
for (auto F : buildFunctionOrder(M, CG)) {
|
||||
assert(!F->isDeclaration());
|
||||
clearFunctionData();
|
||||
retval |= runOnFunction(*F, AM);
|
||||
}
|
||||
|
||||
// Account for cold calls not inlined....
|
||||
for (const std::pair<Function *, NotInlinedProfileInfo> &pair :
|
||||
@ -1787,7 +1823,7 @@ bool SampleProfileLoaderLegacyPass::runOnModule(Module &M) {
|
||||
TTIWP = &getAnalysis<TargetTransformInfoWrapperPass>();
|
||||
ProfileSummaryInfo *PSI =
|
||||
&getAnalysis<ProfileSummaryInfoWrapperPass>().getPSI();
|
||||
return SampleLoader.runOnModule(M, nullptr, PSI);
|
||||
return SampleLoader.runOnModule(M, nullptr, PSI, nullptr);
|
||||
}
|
||||
|
||||
bool SampleProfileLoader::runOnFunction(Function &F, ModuleAnalysisManager *AM) {
|
||||
@ -1871,7 +1907,8 @@ PreservedAnalyses SampleProfileLoaderPass::run(Module &M,
|
||||
SampleLoader.doInitialization(M);
|
||||
|
||||
ProfileSummaryInfo *PSI = &AM.getResult<ProfileSummaryAnalysis>(M);
|
||||
if (!SampleLoader.runOnModule(M, &AM, PSI))
|
||||
CallGraph &CG = AM.getResult<CallGraphAnalysis>(M);
|
||||
if (!SampleLoader.runOnModule(M, &AM, PSI, &CG))
|
||||
return PreservedAnalyses::all();
|
||||
|
||||
return PreservedAnalyses::none();
|
||||
|
10
test/Transforms/SampleProfile/Inputs/inline-topdown.prof
Normal file
10
test/Transforms/SampleProfile/Inputs/inline-topdown.prof
Normal file
@ -0,0 +1,10 @@
|
||||
main:225715:0
|
||||
2.1: 5553
|
||||
3: 5391
|
||||
3.1: _Z3sumii:50000
|
||||
1: _Z3subii:0
|
||||
1: 0
|
||||
|
||||
_Z3sumii:6010:50000
|
||||
1: _Z3subii:60000
|
||||
1: 9
|
123
test/Transforms/SampleProfile/inline-topdown.ll
Normal file
123
test/Transforms/SampleProfile/inline-topdown.ll
Normal file
@ -0,0 +1,123 @@
|
||||
; Note that this needs new pass manager for now. Passing `-sample-profile-top-down-load` to legacy pass manager is a no-op.
|
||||
|
||||
; Test we aren't doing specialization for inlining with default source order
|
||||
; RUN: opt < %s -passes=sample-profile -sample-profile-file=%S/Inputs/inline-topdown.prof -S | FileCheck -check-prefix=DEFAULT %s
|
||||
|
||||
; Test we specialize based on call path with context-sensitive profile while inlining with '-sample-profile-top-down-load'
|
||||
; RUN: opt < %s -passes=sample-profile -sample-profile-file=%S/Inputs/inline-topdown.prof -sample-profile-merge-inlinee -sample-profile-top-down-load -S | FileCheck -check-prefix=TOPDOWN %s
|
||||
|
||||
|
||||
@.str = private unnamed_addr constant [11 x i8] c"sum is %d\0A\00", align 1
|
||||
|
||||
define i32 @_Z3sumii(i32 %x, i32 %y) !dbg !6 {
|
||||
entry:
|
||||
%x.addr = alloca i32, align 4
|
||||
%y.addr = alloca i32, align 4
|
||||
store i32 %x, i32* %x.addr, align 4
|
||||
store i32 %y, i32* %y.addr, align 4
|
||||
%tmp = load i32, i32* %x.addr, align 4, !dbg !8
|
||||
%tmp1 = load i32, i32* %y.addr, align 4, !dbg !8
|
||||
%add = add nsw i32 %tmp, %tmp1, !dbg !8
|
||||
%tmp2 = load i32, i32* %x.addr, align 4, !dbg !8
|
||||
%tmp3 = load i32, i32* %y.addr, align 4, !dbg !8
|
||||
%call = call i32 @_Z3subii(i32 %tmp2, i32 %tmp3), !dbg !8
|
||||
ret i32 %add, !dbg !8
|
||||
}
|
||||
|
||||
define i32 @_Z3subii(i32 %x, i32 %y) !dbg !9 {
|
||||
entry:
|
||||
%x.addr = alloca i32, align 4
|
||||
%y.addr = alloca i32, align 4
|
||||
store i32 %x, i32* %x.addr, align 4
|
||||
store i32 %y, i32* %y.addr, align 4
|
||||
%tmp = load i32, i32* %x.addr, align 4, !dbg !10
|
||||
%tmp1 = load i32, i32* %y.addr, align 4, !dbg !10
|
||||
%add = sub nsw i32 %tmp, %tmp1, !dbg !10
|
||||
ret i32 %add, !dbg !11
|
||||
}
|
||||
|
||||
define i32 @main() !dbg !12 {
|
||||
entry:
|
||||
%retval = alloca i32, align 4
|
||||
%s = alloca i32, align 4
|
||||
%i = alloca i32, align 4
|
||||
store i32 0, i32* %retval
|
||||
store i32 0, i32* %i, align 4, !dbg !13
|
||||
br label %while.cond, !dbg !14
|
||||
|
||||
while.cond: ; preds = %if.end, %entry
|
||||
%tmp = load i32, i32* %i, align 4, !dbg !15
|
||||
%inc = add nsw i32 %tmp, 1, !dbg !15
|
||||
store i32 %inc, i32* %i, align 4, !dbg !15
|
||||
%cmp = icmp slt i32 %tmp, 400000000, !dbg !15
|
||||
br i1 %cmp, label %while.body, label %while.end, !dbg !15
|
||||
|
||||
while.body: ; preds = %while.cond
|
||||
%tmp1 = load i32, i32* %i, align 4, !dbg !17
|
||||
%cmp1 = icmp ne i32 %tmp1, 100, !dbg !17
|
||||
br i1 %cmp1, label %if.then, label %if.else, !dbg !17
|
||||
|
||||
if.then: ; preds = %while.body
|
||||
%tmp2 = load i32, i32* %i, align 4, !dbg !19
|
||||
%tmp3 = load i32, i32* %s, align 4, !dbg !19
|
||||
%call = call i32 @_Z3sumii(i32 %tmp2, i32 %tmp3), !dbg !19
|
||||
store i32 %call, i32* %s, align 4, !dbg !19
|
||||
br label %if.end, !dbg !19
|
||||
|
||||
if.else: ; preds = %while.body
|
||||
store i32 30, i32* %s, align 4, !dbg !21
|
||||
br label %if.end
|
||||
|
||||
if.end: ; preds = %if.else, %if.then
|
||||
br label %while.cond, !dbg !23
|
||||
|
||||
while.end: ; preds = %while.cond
|
||||
%tmp4 = load i32, i32* %s, align 4, !dbg !25
|
||||
%call2 = call i32 (i8*, ...) @printf(i8* getelementptr inbounds ([11 x i8], [11 x i8]* @.str, i32 0, i32 0), i32 %tmp4), !dbg !25
|
||||
ret i32 0, !dbg !26
|
||||
}
|
||||
|
||||
declare i32 @printf(i8*, ...)
|
||||
|
||||
!llvm.dbg.cu = !{!0}
|
||||
!llvm.module.flags = !{!3, !4}
|
||||
!llvm.ident = !{!5}
|
||||
|
||||
!0 = distinct !DICompileUnit(language: DW_LANG_C_plus_plus, file: !1, producer: "clang version 3.5 ", isOptimized: false, runtimeVersion: 0, emissionKind: NoDebug, enums: !2, retainedTypes: !2, globals: !2, imports: !2)
|
||||
!1 = !DIFile(filename: "calls.cc", directory: ".")
|
||||
!2 = !{}
|
||||
!3 = !{i32 2, !"Dwarf Version", i32 4}
|
||||
!4 = !{i32 1, !"Debug Info Version", i32 3}
|
||||
!5 = !{!"clang version 3.5 "}
|
||||
!6 = distinct !DISubprogram(name: "sum", scope: !1, file: !1, line: 3, type: !7, scopeLine: 3, virtualIndex: 6, flags: DIFlagPrototyped, spFlags: DISPFlagDefinition, unit: !0, retainedNodes: !2)
|
||||
!7 = !DISubroutineType(types: !2)
|
||||
!8 = !DILocation(line: 4, scope: !6)
|
||||
!9 = distinct !DISubprogram(name: "sub", scope: !1, file: !1, line: 20, type: !7, scopeLine: 20, virtualIndex: 6, flags: DIFlagPrototyped, spFlags: DISPFlagDefinition, unit: !0, retainedNodes: !2)
|
||||
!10 = !DILocation(line: 20, scope: !9)
|
||||
!11 = !DILocation(line: 21, scope: !9)
|
||||
!12 = distinct !DISubprogram(name: "main", scope: !1, file: !1, line: 7, type: !7, scopeLine: 7, virtualIndex: 6, flags: DIFlagPrototyped, spFlags: DISPFlagDefinition, unit: !0, retainedNodes: !2)
|
||||
!13 = !DILocation(line: 8, scope: !12)
|
||||
!14 = !DILocation(line: 9, scope: !12)
|
||||
!15 = !DILocation(line: 9, scope: !16)
|
||||
!16 = !DILexicalBlockFile(scope: !12, file: !1, discriminator: 2)
|
||||
!17 = !DILocation(line: 10, scope: !18)
|
||||
!18 = distinct !DILexicalBlock(scope: !12, file: !1, line: 10)
|
||||
!19 = !DILocation(line: 10, scope: !20)
|
||||
!20 = !DILexicalBlockFile(scope: !18, file: !1, discriminator: 2)
|
||||
!21 = !DILocation(line: 10, scope: !22)
|
||||
!22 = !DILexicalBlockFile(scope: !18, file: !1, discriminator: 4)
|
||||
!23 = !DILocation(line: 10, scope: !24)
|
||||
!24 = !DILexicalBlockFile(scope: !18, file: !1, discriminator: 6)
|
||||
!25 = !DILocation(line: 11, scope: !12)
|
||||
!26 = !DILocation(line: 12, scope: !12)
|
||||
|
||||
|
||||
; DEFAULT: @_Z3sumii
|
||||
; DEFAULT-NOT: call i32 @_Z3subii
|
||||
; DEFAULT: @main()
|
||||
; DEFAULT-NOT: call i32 @_Z3subii
|
||||
|
||||
; TOPDOWN: @_Z3sumii
|
||||
; TOPDOWN-NOT: call i32 @_Z3subii
|
||||
; TOPDOWN: @main()
|
||||
; TOPDOWN: call i32 @_Z3subii
|
Loading…
Reference in New Issue
Block a user