Add prefix based function layout when profile is available.

Summary: If a function is hot, put it in text.hot section.

Reviewers: davidxl

Subscribers: llvm-commits

Differential Revision: http://reviews.llvm.org/D17532

git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@261607 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Dehao Chen 2016-02-23 03:39:24 +00:00
parent 5b9b80ea30
commit 5261d1373d
4 changed files with 74 additions and 0 deletions

View File

@ -21,6 +21,7 @@
#define LLVM_PROFILEDATA_PROFILE_COMMON_H
namespace llvm {
class Function;
namespace IndexedInstrProf {
struct Summary;
}
@ -28,6 +29,8 @@ namespace sampleprof {
class FunctionSamples;
}
struct InstrProfRecord;
inline const char *getHotSectionPrefix() { return ".hot"; }
inline const char *getUnlikelySectionPrefix() { return ".unlikely"; }
// The profile summary is one or more (Cutoff, MinCount, NumCounts) triplets.
// The semantics of counts depend on the type of profile. For instrumentation
// profile, counts are block counts and for sample profile, counts are
@ -66,6 +69,10 @@ protected:
public:
static const int Scale = 1000000;
// \brief Returns true if F is a hot function.
static bool isFunctionHot(const Function *F);
// \brief Returns true if F is unlikley executed.
static bool isFunctionUnlikely(const Function *F);
inline std::vector<ProfileSummaryEntry> &getDetailedSummary();
void computeDetailedSummary();
/// \brief A vector of useful cutoff values for detailed summary.

View File

@ -34,6 +34,7 @@
#include "llvm/MC/MCSymbolELF.h"
#include "llvm/MC/MCValue.h"
#include "llvm/ProfileData/InstrProf.h"
#include "llvm/ProfileData/ProfileCommon.h"
#include "llvm/Support/COFF.h"
#include "llvm/Support/Dwarf.h"
#include "llvm/Support/ELF.h"
@ -244,6 +245,11 @@ static StringRef getSectionPrefixForGlobal(SectionKind Kind) {
return ".data.rel.ro";
}
static cl::opt<bool> GroupFunctionsByHotness(
"group-functions-by-hotness",
llvm::cl::desc("Partition hot/cold functions by sections prefix"),
cl::init(false));
static MCSectionELF *
selectELFSectionForGlobal(MCContext &Ctx, const GlobalValue *GV,
SectionKind Kind, Mangler &Mang,
@ -296,6 +302,16 @@ selectELFSectionForGlobal(MCContext &Ctx, const GlobalValue *GV,
Name = getSectionPrefixForGlobal(Kind);
}
if (GroupFunctionsByHotness) {
if (const Function *F = dyn_cast<Function>(GV)) {
if (ProfileSummary::isFunctionHot(F)) {
Name += getHotSectionPrefix();
} else if (ProfileSummary::isFunctionUnlikely(F)) {
Name += getUnlikelySectionPrefix();
}
}
}
if (EmitUniqueSection && UniqueSectionNames) {
Name.push_back('.');
TM.getNameWithPrefix(Name, GV, Mang, true);

View File

@ -11,6 +11,8 @@
//
//===----------------------------------------------------------------------===//
#include "llvm/IR/Attributes.h"
#include "llvm/IR/Function.h"
#include "llvm/ProfileData/InstrProf.h"
#include "llvm/ProfileData/ProfileCommon.h"
#include "llvm/ProfileData/SampleProf.h"
@ -75,6 +77,24 @@ void ProfileSummary::computeDetailedSummary() {
}
}
// Returns true if the function is a hot function.
bool ProfileSummary::isFunctionHot(const Function *F) {
// FIXME: update when summary data is stored in module's metadata.
return false;
}
// Returns true if the function is a cold function.
bool ProfileSummary::isFunctionUnlikely(const Function *F) {
if (F->hasFnAttribute(Attribute::Cold)) {
return true;
}
if (!F->getEntryCount()) {
return false;
}
// FIXME: update when summary data is stored in module's metadata.
return (*F->getEntryCount()) == 0;
}
InstrProfSummary::InstrProfSummary(const IndexedInstrProf::Summary &S)
: ProfileSummary(), MaxInternalBlockCount(S.get(
IndexedInstrProf::Summary::MaxInternalBlockCount)),

View File

@ -0,0 +1,31 @@
; RUN: llc < %s -mtriple=x86_64-pc-linux -group-functions-by-hotness=true | FileCheck %s -check-prefix=PARTITION
; RUN: llc < %s -mtriple=x86_64-pc-linux -function-sections -group-functions-by-hotness=false | FileCheck %s -check-prefix=NO-PARTITION-FUNCTION-SECTION
; RUN: llc < %s -mtriple=x86_64-pc-linux -function-sections -group-functions-by-hotness=true | FileCheck %s -check-prefix=PARTITION-FUNCTION-SECTION
; PARTITION: .text.unlikely
; PARTITION: .globl _Z3foov
; NO-PARTITION-FUNCTION-SECTION: .text._Z3foov
; PARTITION-FUNCTION-SECTION: .text.unlikely._Z3foov
define i32 @_Z3foov() #0 {
ret i32 0
}
; PARTITION: .globl _Z3barv
; NO-PARTITION-FUNCTION-SECTION: .text._Z3barv
; PARTITION-FUNCTION-SECTION: .text.unlikely._Z3barv
define i32 @_Z3barv() #1 !prof !0 {
ret i32 1
}
; PARTITION: .text
; PARTITION: .globl _Z3bazv
; NO-PARTITION-FUNCTION-SECTION: .text._Z3bazv
; PARTITION-FUNCTION-SECTION: .text._Z3bazv
define i32 @_Z3bazv() #1 {
ret i32 2
}
attributes #0 = { nounwind uwtable cold }
attributes #1 = { nounwind uwtable }
!0 = !{!"function_entry_count", i64 0}