Files
swift-mirror/lib/SILPasses/PerformanceInliner.cpp
2014-02-07 23:07:11 +00:00

175 lines
5.6 KiB
C++

//===- PerformanceInliner.cpp - Basic cost based inlining for performance -===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2015 Apple Inc. and the Swift project authors
// Licensed under Apache License v2.0 with Runtime Library Exception
//
// See http://swift.org/LICENSE.txt for license information
// See http://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
//
//===----------------------------------------------------------------------===//
#define DEBUG_TYPE "sil-inliner"
#include "swift/SIL/SILInstruction.h"
#include "swift/SIL/CallGraph.h"
#include "swift/SIL/SILModule.h"
#include "swift/SILPasses/Passes.h"
#include "swift/SILPasses/Transforms.h"
#include "swift/SILPasses/Utils/Local.h"
#include "swift/SILPasses/Utils/SILInliner.h"
#include "llvm/ADT/Statistic.h"
#include "llvm/Support/Debug.h"
#include "llvm/Support/CommandLine.h"
#include "llvm/ADT/MapVector.h"
using namespace swift;
STATISTIC(NumFunctionsInlined, "Number of functions inlined");
namespace {
class SILPerformanceInliner {
const unsigned InlineCostThreshold;
public:
explicit SILPerformanceInliner(unsigned threshold)
: InlineCostThreshold(threshold) {}
bool inlineCallsIntoFunction(SILFunction *F);
};
}
//===----------------------------------------------------------------------===//
// Call Graph Creation
//===----------------------------------------------------------------------===//
/// \brief Returns a SILFunction if this ApplyInst calls a recognizable function
/// that is legal to inline.
static SILFunction *getInlinableFunction(ApplyInst *AI) {
// Avoid substituion lists, we don't support them.
if (AI->hasSubstitutions())
return nullptr;
auto *FRI = dyn_cast<FunctionRefInst>(AI->getCallee().getDef());
if (!FRI)
return nullptr;
SILFunction *F = FRI->getReferencedFunction();
if (F->empty() || F->isExternalDeclaration()) {
DEBUG(llvm::dbgs() << " Can't inline " << F->getName() << ".\n");
return nullptr;
}
DEBUG(llvm::dbgs() << " Can inline " << F->getName() << ".\n");
return F;
}
//===----------------------------------------------------------------------===//
// Inliner
//===----------------------------------------------------------------------===//
/// \brief Attempt to inline all calls smaller than our threshold into F until.
/// returns True if a function was inlined.
bool SILPerformanceInliner::inlineCallsIntoFunction(SILFunction *Caller) {
bool Changed = false;
SILInliner Inliner(*Caller, SILInliner::InlineKind::PerformanceInline);
DEBUG(llvm::dbgs() << "Visiting Function: " << Caller->getName() << "\n");
llvm::SmallVector<ApplyInst*, 8> CallSites;
// Collect all of the ApplyInsts in this function. We will be changing the
// control flow and collecting the AIs simplifies the scan.
for (auto &BB : *Caller) {
auto I = BB.begin(), E = BB.end();
while (I != E) {
// Check if this is a call site.
ApplyInst *AI = dyn_cast<ApplyInst>(I++);
if (AI)
CallSites.push_back(AI);
}
}
for (auto AI : CallSites) {
DEBUG(llvm::dbgs() << " Found call site:" << *AI);
// Get the callee.
SILFunction *Callee = getInlinableFunction(AI);
if (!Callee)
continue;
DEBUG(llvm::dbgs() << " Found callee:" << Callee->getName() << ".\n");
// Prevent circular inlining.
if (Callee == Caller) {
DEBUG(llvm::dbgs() << " Skipping recursive calls.\n");
continue;
}
// Calculate the inlining cost of the callee.
unsigned CalleeCost = getFunctionCost(Callee, Caller, InlineCostThreshold);
if (CalleeCost > InlineCostThreshold) {
DEBUG(llvm::dbgs() << " Function too big to inline. Skipping.\n");
continue;
}
// Add the arguments from AI into a SILValue list.
SmallVector<SILValue, 8> Args;
for (const auto &Arg : AI->getArguments())
Args.push_back(Arg);
// Ok, we are within budget. Attempt to inline.
DEBUG(llvm::dbgs() << " Inlining " << Callee->getName() << " Into " <<
Caller->getName() << "\n");
// We already moved the iterator to the next instruction because the AI
// will be erased by the inliner. Notice that we will skip all of the
// newly inlined ApplyInsts. That's okay because we will visit them in
// our next invocation of the inliner.
Inliner.inlineFunction(AI, Callee, ArrayRef<Substitution>(), Args);
NumFunctionsInlined++;
Changed = true;
}
return Changed;
}
class SILPerformanceInlinerPass : public SILModuleTransform {
public:
SILPerformanceInlinerPass() {}
void run() {
CallGraphAnalysis* CGA = PM->getAnalysis<CallGraphAnalysis>();
if (getOptions().InlineThreshold == 0) {
DEBUG(llvm::dbgs() << "*** The Performance Inliner is disabled ***\n");
return;
}
// Initialize the worklist with a bottom-up call-graph order list of
// functions.
const std::vector<SILFunction *> &Order = CGA->bottomUpCallGraphOrder();
std::vector<SILFunction *> Worklist(Order);
std::reverse(Worklist.begin(), Worklist.end());
SILPerformanceInliner inliner(getOptions().InlineThreshold);
bool Changed = false;
while (!Worklist.empty()) {
SILFunction *F = Worklist.back();
Worklist.pop_back();
Changed |= inliner.inlineCallsIntoFunction(F);
}
// Invalidate the call graph.
if (Changed)
invalidateAnalysis(SILAnalysis::InvalidationKind::CallGraph);
}
StringRef getName() override { return "Performance Inlining"; }
};
SILTransform *swift::createPerfInliner() {
return new SILPerformanceInlinerPass();
}