mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2024-11-22 10:42:39 +01:00
46d7a90a64
This change leverages the work done in D83743 to replay in the SampleProfile inliner to also be used in the CGSCC inliner. NOTE: currently restricted to non-ML advisors only. The added switch `-cgscc-inline-replay=<remarks file>` will replay the inlining decisions in that file where the remarks file is generated via `-Rpass=inline`. The aim here is to make it easier to analyze changes that would modify inlining heuristics to be separated from this behavior. Doing so allows easier examination of assembly and runtime behavior compared to the baseline rather than trying to dig through the large churn caused by inlining. In LTO compilation, since inlining is done twice you can separately specify replay by passing the flag to the FE (`-cgscc-inline-replay=`) and to the linker (`-Wl,cgscc-inline-replay=`) with the remarks generated from their respective places. Testing on mysqld by comparing the inline decisions between base (generates remarks.txt) and diff (replay using identical input/tools with remarks.txt) and examining the inlining sites with `diff` shows 14,000 mismatches out of 247,341 for a ~94% replay accuracy. I believe this gap can be narrowed further though for the general case we may never achieve full accuracy. For my personal use, this is close enough to be representative: I set the baseline as the one generated by the replay on identical input/toolset and compare that to my modified input/toolset using the same replay. Testing: ninja check-llvm newly added test correctly replays CGSCC inlining decisions Reviewed By: mtrofin, wenlei Differential Revision: https://reviews.llvm.org/D94334
83 lines
3.0 KiB
C++
83 lines
3.0 KiB
C++
//===- ReplayInlineAdvisor.cpp - Replay InlineAdvisor ---------------------===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file implements ReplayInlineAdvisor that replays inline decisions based
|
|
// on previous inline remarks from optimization remark log. This is a best
|
|
// effort approach useful for testing compiler/source changes while holding
|
|
// inlining steady.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "llvm/Analysis/ReplayInlineAdvisor.h"
|
|
#include "llvm/IR/DebugInfoMetadata.h"
|
|
#include "llvm/IR/Instructions.h"
|
|
#include "llvm/Support/LineIterator.h"
|
|
|
|
using namespace llvm;
|
|
|
|
#define DEBUG_TYPE "inline-replay"
|
|
|
|
ReplayInlineAdvisor::ReplayInlineAdvisor(
|
|
Module &M, FunctionAnalysisManager &FAM, LLVMContext &Context,
|
|
std::unique_ptr<InlineAdvisor> OriginalAdvisor, StringRef RemarksFile,
|
|
bool EmitRemarks)
|
|
: InlineAdvisor(M, FAM), OriginalAdvisor(std::move(OriginalAdvisor)),
|
|
HasReplayRemarks(false), EmitRemarks(EmitRemarks) {
|
|
auto BufferOrErr = MemoryBuffer::getFileOrSTDIN(RemarksFile);
|
|
std::error_code EC = BufferOrErr.getError();
|
|
if (EC) {
|
|
Context.emitError("Could not open remarks file: " + EC.message());
|
|
return;
|
|
}
|
|
|
|
// Example for inline remarks to parse:
|
|
// main:3:1.1: _Z3subii inlined into main at callsite sum:1 @ main:3:1.1
|
|
// We use the callsite string after `at callsite` to replay inlining.
|
|
line_iterator LineIt(*BufferOrErr.get(), /*SkipBlanks=*/true);
|
|
for (; !LineIt.is_at_eof(); ++LineIt) {
|
|
StringRef Line = *LineIt;
|
|
auto Pair = Line.split(" at callsite ");
|
|
|
|
auto Callee = Pair.first.split(" inlined into").first.rsplit(": ").second;
|
|
|
|
auto CallSite = Pair.second.split(";").first;
|
|
|
|
if (Callee.empty() || CallSite.empty())
|
|
continue;
|
|
|
|
std::string Combined = (Callee + CallSite).str();
|
|
InlineSitesFromRemarks.insert(Combined);
|
|
}
|
|
|
|
HasReplayRemarks = true;
|
|
}
|
|
|
|
std::unique_ptr<InlineAdvice> ReplayInlineAdvisor::getAdviceImpl(CallBase &CB) {
|
|
assert(HasReplayRemarks);
|
|
|
|
Function &Caller = *CB.getCaller();
|
|
auto &ORE = FAM.getResult<OptimizationRemarkEmitterAnalysis>(Caller);
|
|
|
|
if (InlineSitesFromRemarks.empty())
|
|
return std::make_unique<DefaultInlineAdvice>(this, CB, None, ORE,
|
|
EmitRemarks);
|
|
|
|
std::string CallSiteLoc = getCallSiteLocation(CB.getDebugLoc());
|
|
StringRef Callee = CB.getCalledFunction()->getName();
|
|
std::string Combined = (Callee + CallSiteLoc).str();
|
|
auto Iter = InlineSitesFromRemarks.find(Combined);
|
|
|
|
Optional<InlineCost> InlineRecommended = None;
|
|
if (Iter != InlineSitesFromRemarks.end()) {
|
|
InlineRecommended = llvm::InlineCost::getAlways("found in replay");
|
|
}
|
|
|
|
return std::make_unique<DefaultInlineAdvice>(this, CB, InlineRecommended, ORE,
|
|
EmitRemarks);
|
|
}
|