mirror of
https://github.com/apple/swift.git
synced 2025-12-21 12:14:44 +01:00
The old call graph remains, but the new call graph can obtained by calling getCallGraph() on the analysis pass. I'll move the few other passes that use the call graph over soon and then rip out the old call graph. No diffs in the stdlib. Swift SVN r21565
256 lines
7.9 KiB
C++
256 lines
7.9 KiB
C++
//===----- CallGraphAnalysis.cpp - Call graph construction ----*- C++ -*---===//
|
|
//
|
|
// 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "swift/SILAnalysis/CallGraphAnalysis.h"
|
|
#include "llvm/ADT/DenseMap.h"
|
|
#include "llvm/ADT/SetVector.h"
|
|
#include "llvm/ADT/SmallVector.h"
|
|
#include "llvm/ADT/Statistic.h"
|
|
#include <algorithm>
|
|
#include <utility>
|
|
|
|
using namespace swift;
|
|
|
|
#define DEBUG_TYPE "call-graph"
|
|
|
|
STATISTIC(NumCallGraphNodes, "# of call graph nodes created");
|
|
STATISTIC(NumCallSitesWithEdges, "# of call sites with edges");
|
|
STATISTIC(NumCallSitesWithoutEdges,
|
|
"# of non-builtin call sites without call graph edges");
|
|
STATISTIC(NumCallSitesOfBuiltins, "# of call sites calling builtins");
|
|
|
|
CallGraph::CallGraph(SILModule *M, bool completeModule) {
|
|
// Build the initial call graph by creating a node for each
|
|
// function, and an edge for each direct call to a free function.
|
|
// TODO: Handle other kinds of applies.
|
|
|
|
unsigned NodeOrdinal = 0;
|
|
for (auto &F : *M)
|
|
addCallGraphNode(&F, NodeOrdinal++);
|
|
|
|
for (auto &F : *M)
|
|
if (F.isDefinition())
|
|
addEdges(&F);
|
|
}
|
|
|
|
void CallGraph::addCallGraphNode(SILFunction *F, unsigned NodeOrdinal) {
|
|
// TODO: Compute this from the call graph itself after stripping
|
|
// unreachable nodes from graph.
|
|
++NumCallGraphNodes;
|
|
auto *Node = new CallGraphNode(F, NodeOrdinal);
|
|
|
|
assert(!FunctionToNodeMap.count(F) &&
|
|
"Added function already has a call graph node!");
|
|
|
|
FunctionToNodeMap[F] = Node;
|
|
|
|
// TODO: Only add functions clearly visible from outside our
|
|
// compilation scope as roots.
|
|
if (F->isDefinition())
|
|
CallGraphRoots.push_back(Node);
|
|
}
|
|
|
|
void CallGraph::addEdgesForApply(ApplyInst *AI, CallGraphNode *CallerNode) {
|
|
auto Callee = AI->getCallee();
|
|
|
|
switch (Callee->getKind()) {
|
|
case ValueKind::FunctionRefInst: {
|
|
auto *CalleeFn = cast<FunctionRefInst>(Callee)->getReferencedFunction();
|
|
// TODO: Compute this from the call graph itself after stripping
|
|
// unreachable nodes from graph.
|
|
++NumCallSitesWithEdges;
|
|
auto *CalleeNode = getCallGraphNode(CalleeFn);
|
|
assert(CalleeNode &&
|
|
"Expected to have a call graph node for all functions!");
|
|
|
|
auto *CallSite = new CallGraphEdge(AI, CalleeNode);
|
|
CallerNode->addCallSite(CallSite);
|
|
CalleeNode->addCaller(CallSite);
|
|
break;
|
|
}
|
|
|
|
case ValueKind::DynamicMethodInst: {
|
|
// TODO: Decide how to handle these in graph construction and
|
|
// analysis passes. We might just leave them out of the
|
|
// graph.
|
|
++NumCallSitesWithoutEdges;
|
|
break;
|
|
}
|
|
|
|
case ValueKind::ThinToThickFunctionInst: {
|
|
auto ThinCallee = cast<ThinToThickFunctionInst>(Callee)->getOperand();
|
|
// TODO: We want to see through these to the underlying function.
|
|
assert(cast<FunctionRefInst>(ThinCallee) && "Expected function reference!");
|
|
++NumCallSitesWithoutEdges;
|
|
break;
|
|
}
|
|
|
|
case ValueKind::SILArgument: {
|
|
// First-pass call-graph construction will not do anything with
|
|
// these, but a second pass can potentially statically determine
|
|
// the called function in some cases.
|
|
++NumCallSitesWithoutEdges;
|
|
break;
|
|
}
|
|
|
|
case ValueKind::ApplyInst: {
|
|
// TODO: Probably not worth iterating invocation- then
|
|
// reverse-invocation order to catch this.
|
|
++NumCallSitesWithoutEdges;
|
|
break;
|
|
}
|
|
|
|
case ValueKind::TupleExtractInst: {
|
|
// TODO: It would be good to tunnel through extracts so that we
|
|
// can build a more accurate call graph prior to any
|
|
// optimizations.
|
|
++NumCallSitesWithoutEdges;
|
|
break;
|
|
}
|
|
|
|
case ValueKind::StructExtractInst: {
|
|
// TODO: It would be good to tunnel through extracts so that we
|
|
// can build a more accurate call graph prior to any
|
|
// optimizations.
|
|
++NumCallSitesWithoutEdges;
|
|
break;
|
|
}
|
|
|
|
case ValueKind::BuiltinFunctionRefInst: {
|
|
++NumCallSitesOfBuiltins;
|
|
break;
|
|
}
|
|
|
|
case ValueKind::PartialApplyInst:
|
|
case ValueKind::ClassMethodInst:
|
|
case ValueKind::ProtocolMethodInst:
|
|
case ValueKind::WitnessMethodInst:
|
|
case ValueKind::SuperMethodInst:
|
|
// TODO: Each of these requires specific handling.
|
|
++NumCallSitesWithoutEdges;
|
|
break;
|
|
|
|
default:
|
|
assert(!isa<MethodInst>(Callee)
|
|
&& "Unhandled method instruction in call graph construction!");
|
|
|
|
// There are cases where we will be very hard pressed to determine
|
|
// what we are calling.
|
|
++NumCallSitesWithoutEdges;
|
|
break;
|
|
}
|
|
}
|
|
|
|
void CallGraph::addEdges(SILFunction *F) {
|
|
auto *CallerNode = getCallGraphNode(F);
|
|
assert(CallerNode && "Expected call graph node for function!");
|
|
|
|
for (auto &BB : *F)
|
|
for (auto &I : BB)
|
|
if (auto *AI = dyn_cast<ApplyInst>(&I))
|
|
addEdgesForApply(AI, CallerNode);
|
|
}
|
|
|
|
static void orderCallees(const CallGraphEdge::CalleeSetType &Callees,
|
|
llvm::SmallVectorImpl<CallGraphNode *> &OrderedNodes) {
|
|
for (auto *Node : Callees)
|
|
OrderedNodes.push_back(Node);
|
|
|
|
std::sort(OrderedNodes.begin(), OrderedNodes.end(),
|
|
[](CallGraphNode *left, CallGraphNode *right) {
|
|
return left->getOrdinal() < right->getOrdinal();
|
|
});
|
|
}
|
|
|
|
/// Finds SCCs in the call graph. Our call graph has an unconventional
|
|
/// form where each edge of the graph is really a multi-edge that can
|
|
/// point to multiple call graph nodes in the case where we can call
|
|
/// one of several different functions.
|
|
class CallGraphSCCFinder {
|
|
public:
|
|
CallGraphSCCFinder(llvm::SmallVectorImpl<CallGraphSCC *> &TheSCCs)
|
|
: NextDFSNum(0), TheSCCs(TheSCCs) {}
|
|
|
|
void DFS(CallGraphNode *Node) {
|
|
// Set the DFSNum for this node if we haven't already, and if we
|
|
// have, which indicates it's already been visited, return.
|
|
if (!DFSNum.insert(std::make_pair(Node, NextDFSNum)).second)
|
|
return;
|
|
|
|
assert(MinDFSNum.find(Node) == MinDFSNum.end() &&
|
|
"Node should not already have a minimum DFS number!");
|
|
|
|
MinDFSNum[Node] = NextDFSNum;
|
|
++NextDFSNum;
|
|
|
|
DFSStack.insert(Node);
|
|
|
|
for (auto *CallSite : Node->getCallSites()) {
|
|
llvm::SmallVector<CallGraphNode *, 4> OrderedNodes;
|
|
orderCallees(CallSite->getCalleeSet(), OrderedNodes);
|
|
|
|
for (auto *CalleeNode : OrderedNodes) {
|
|
if (DFSNum.find(CalleeNode) == DFSNum.end()) {
|
|
DFS(CalleeNode);
|
|
MinDFSNum[Node] = std::min(MinDFSNum[Node], MinDFSNum[CalleeNode]);
|
|
} else if (DFSStack.count(CalleeNode)) {
|
|
MinDFSNum[Node] = std::min(MinDFSNum[Node], DFSNum[CalleeNode]);
|
|
}
|
|
}
|
|
}
|
|
|
|
// If this node is the root of an SCC (including SCCs with a
|
|
// single node), pop the SCC and push it on our SCC stack.
|
|
if (DFSNum[Node] == MinDFSNum[Node]) {
|
|
auto *SCC = new CallGraphSCC();
|
|
|
|
CallGraphNode *Popped;
|
|
do {
|
|
Popped = DFSStack.pop_back_val();
|
|
SCC->SCCNodes.push_back(Popped);
|
|
} while (Popped != Node);
|
|
|
|
TheSCCs.push_back(SCC);
|
|
}
|
|
}
|
|
|
|
private:
|
|
unsigned NextDFSNum;
|
|
llvm::SmallVectorImpl<CallGraphSCC *> &TheSCCs;
|
|
|
|
llvm::DenseMap<CallGraphNode *, unsigned> DFSNum;
|
|
llvm::DenseMap<CallGraphNode *, unsigned> MinDFSNum;
|
|
llvm::SetVector<CallGraphNode *> DFSStack;
|
|
};
|
|
|
|
void CallGraph::computeBottomUpSCCOrder() {
|
|
if (!BottomUpSCCOrder.empty()) {
|
|
for (auto *SCC : BottomUpSCCOrder)
|
|
delete SCC;
|
|
|
|
BottomUpSCCOrder.clear();
|
|
}
|
|
|
|
for (auto *Node : getCallGraphRoots())
|
|
CallGraphSCCFinder(BottomUpSCCOrder).DFS(Node);
|
|
}
|
|
|
|
void CallGraph::computeBottomUpFunctionOrder() {
|
|
BottomUpFunctionOrder.clear();
|
|
|
|
computeBottomUpSCCOrder();
|
|
|
|
for (auto *SCC : BottomUpSCCOrder)
|
|
for (auto *Node : SCC->SCCNodes)
|
|
BottomUpFunctionOrder.push_back(Node->getFunction());
|
|
}
|