forked from OSchip/llvm-project
[analyzer] Re-apply "Do part of the work to find shortest bug paths up front".
With the assurance that the trimmed graph does not contain cycles, this patch is safe (with a few tweaks), and provides the performance boost it was intended to. Part of performance work for <rdar://problem/13433687>. llvm-svn: 177469
This commit is contained in:
parent
34e19a1d1d
commit
86e04ceaad
|
@ -1888,23 +1888,79 @@ public:
|
|||
class TrimmedGraph {
|
||||
InterExplodedGraphMap ForwardMap;
|
||||
InterExplodedGraphMap InverseMap;
|
||||
|
||||
typedef llvm::DenseMap<const ExplodedNode *, unsigned> PriorityMapTy;
|
||||
PriorityMapTy PriorityMap;
|
||||
|
||||
OwningPtr<ExplodedGraph> G;
|
||||
const ExplodedNode *Root;
|
||||
public:
|
||||
///
|
||||
TrimmedGraph(const ExplodedGraph *OriginalGraph,
|
||||
ArrayRef<const ExplodedNode *> Nodes) {
|
||||
// The trimmed graph is created in the body of the constructor to ensure
|
||||
// that the DenseMaps have been initialized already.
|
||||
G.reset(OriginalGraph->trim(Nodes, /*BreakCycles=*/true,
|
||||
&ForwardMap, &InverseMap));
|
||||
}
|
||||
ArrayRef<const ExplodedNode *> Nodes);
|
||||
|
||||
void createBestReportGraph(ArrayRef<const ExplodedNode *> Nodes,
|
||||
ReportGraph &GraphWrapper) const;
|
||||
|
||||
void removeErrorNode(const ExplodedNode *Node);
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
TrimmedGraph::TrimmedGraph(const ExplodedGraph *OriginalGraph,
|
||||
ArrayRef<const ExplodedNode *> Nodes) {
|
||||
// The trimmed graph is created in the body of the constructor to ensure
|
||||
// that the DenseMaps have been initialized already.
|
||||
G.reset(OriginalGraph->trim(Nodes, /*BreakCycles=*/true,
|
||||
&ForwardMap, &InverseMap));
|
||||
|
||||
// Find the (first) error node in the trimmed graph. We just need to consult
|
||||
// the node map which maps from nodes in the original graph to nodes
|
||||
// in the new graph.
|
||||
std::queue<std::pair<const ExplodedNode *, unsigned> > WS;
|
||||
typedef llvm::SmallDenseMap<const ExplodedNode *, size_t, 32> IndexMapTy;
|
||||
IndexMapTy IndexMap(llvm::NextPowerOf2(Nodes.size() + 1));
|
||||
|
||||
for (unsigned i = 0, count = Nodes.size(); i < count; ++i) {
|
||||
const ExplodedNode *OriginalNode = Nodes[i];
|
||||
if (const ExplodedNode *N = ForwardMap.lookup(OriginalNode)) {
|
||||
WS.push(std::make_pair(N, 0));
|
||||
IndexMap[OriginalNode] = i;
|
||||
}
|
||||
}
|
||||
|
||||
assert(!WS.empty() && "No error node found in the trimmed graph.");
|
||||
|
||||
// Perform a reverse BFS to find all the shortest paths.
|
||||
Root = 0;
|
||||
while (!WS.empty()) {
|
||||
const ExplodedNode *Node;
|
||||
unsigned Priority;
|
||||
llvm::tie(Node, Priority) = WS.front();
|
||||
WS.pop();
|
||||
|
||||
PriorityMapTy::iterator PriorityEntry;
|
||||
bool IsNew;
|
||||
llvm::tie(PriorityEntry, IsNew) =
|
||||
PriorityMap.insert(std::make_pair(Node, Priority));
|
||||
|
||||
if (!IsNew) {
|
||||
assert(PriorityEntry->second <= Priority);
|
||||
continue;
|
||||
}
|
||||
|
||||
if (Node->pred_empty()) {
|
||||
assert(!Root && "more than one root");
|
||||
Root = Node;
|
||||
}
|
||||
|
||||
for (ExplodedNode::const_pred_iterator I = Node->pred_begin(),
|
||||
E = Node->pred_end();
|
||||
I != E; ++I)
|
||||
WS.push(std::make_pair(*I, Priority + 1));
|
||||
}
|
||||
|
||||
assert(Root);
|
||||
}
|
||||
|
||||
void TrimmedGraph::createBestReportGraph(ArrayRef<const ExplodedNode *> Nodes,
|
||||
ReportGraph &GraphWrapper) const {
|
||||
|
@ -1914,14 +1970,14 @@ void TrimmedGraph::createBestReportGraph(ArrayRef<const ExplodedNode *> Nodes,
|
|||
// Find the (first) error node in the trimmed graph. We just need to consult
|
||||
// the node map which maps from nodes in the original graph to nodes
|
||||
// in the new graph.
|
||||
std::queue<const ExplodedNode *> WS;
|
||||
std::queue<std::pair<const ExplodedNode *, unsigned> > WS;
|
||||
typedef llvm::SmallDenseMap<const ExplodedNode *, size_t, 32> IndexMapTy;
|
||||
IndexMapTy IndexMap;
|
||||
IndexMapTy IndexMap(llvm::NextPowerOf2(Nodes.size() + 1));
|
||||
|
||||
for (unsigned i = 0, count = Nodes.size(); i < count; ++i) {
|
||||
const ExplodedNode *OriginalNode = Nodes[i];
|
||||
if (const ExplodedNode *N = ForwardMap.lookup(OriginalNode)) {
|
||||
WS.push(N);
|
||||
WS.push(std::make_pair(N, 0));
|
||||
IndexMap[OriginalNode] = i;
|
||||
}
|
||||
}
|
||||
|
@ -1933,42 +1989,13 @@ void TrimmedGraph::createBestReportGraph(ArrayRef<const ExplodedNode *> Nodes,
|
|||
ExplodedGraph *GNew = new ExplodedGraph();
|
||||
GraphWrapper.Graph.reset(GNew);
|
||||
|
||||
// Sometimes the trimmed graph can contain a cycle. Perform a reverse BFS
|
||||
// to the root node, and then construct a new graph that contains only
|
||||
// a single path.
|
||||
llvm::DenseMap<const ExplodedNode *, unsigned> Visited;
|
||||
|
||||
unsigned cnt = 0;
|
||||
const ExplodedNode *Root = 0;
|
||||
|
||||
while (!WS.empty()) {
|
||||
const ExplodedNode *Node = WS.front();
|
||||
WS.pop();
|
||||
|
||||
if (Visited.find(Node) != Visited.end())
|
||||
continue;
|
||||
|
||||
Visited[Node] = cnt++;
|
||||
|
||||
if (Node->pred_empty()) {
|
||||
Root = Node;
|
||||
break;
|
||||
}
|
||||
|
||||
for (ExplodedNode::const_pred_iterator I=Node->pred_begin(),
|
||||
E=Node->pred_end(); I!=E; ++I)
|
||||
WS.push(*I);
|
||||
}
|
||||
|
||||
assert(Root);
|
||||
|
||||
// Now walk from the root down the BFS path, always taking the successor
|
||||
// with the lowest number.
|
||||
ExplodedNode *Last = 0;
|
||||
for ( const ExplodedNode *N = Root ;;) {
|
||||
// Lookup the number associated with the current node.
|
||||
llvm::DenseMap<const ExplodedNode *,unsigned>::iterator I = Visited.find(N);
|
||||
assert(I != Visited.end());
|
||||
PriorityMapTy::const_iterator I = PriorityMap.find(N);
|
||||
assert(I != PriorityMap.end());
|
||||
|
||||
// Create the equivalent node in the new graph with the same state
|
||||
// and location.
|
||||
|
@ -1994,14 +2021,14 @@ void TrimmedGraph::createBestReportGraph(ArrayRef<const ExplodedNode *> Nodes,
|
|||
}
|
||||
|
||||
// Find the next successor node. We choose the node that is marked
|
||||
// with the lowest DFS number.
|
||||
// with the lowest BFS number.
|
||||
unsigned MinVal = -1U;
|
||||
for (ExplodedNode::const_succ_iterator SI = N->succ_begin(),
|
||||
SE = N->succ_end();
|
||||
SI != SE; ++SI) {
|
||||
I = Visited.find(*SI);
|
||||
I = PriorityMap.find(*SI);
|
||||
|
||||
if (I == Visited.end())
|
||||
if (I == PriorityMap.end())
|
||||
continue;
|
||||
|
||||
if (I->second < MinVal) {
|
||||
|
@ -2014,6 +2041,60 @@ void TrimmedGraph::createBestReportGraph(ArrayRef<const ExplodedNode *> Nodes,
|
|||
}
|
||||
}
|
||||
|
||||
void TrimmedGraph::removeErrorNode(const ExplodedNode *ErrorNode) {
|
||||
ErrorNode = ForwardMap[ErrorNode];
|
||||
assert(ErrorNode && "not an error node");
|
||||
|
||||
PriorityMapTy::iterator PriorityEntry = PriorityMap.find(ErrorNode);
|
||||
assert(PriorityEntry != PriorityMap.end() && "error node already removed");
|
||||
PriorityMap.erase(PriorityEntry);
|
||||
|
||||
std::queue<const ExplodedNode *> WS;
|
||||
for (ExplodedNode::const_pred_iterator PI = ErrorNode->pred_begin(),
|
||||
PE = ErrorNode->pred_end();
|
||||
PI != PE; ++PI) {
|
||||
assert(PriorityMap.find(*PI) != PriorityMap.end() && "predecessor removed");
|
||||
WS.push(*PI);
|
||||
}
|
||||
|
||||
// Update all nodes possibly affected by this change.
|
||||
while (!WS.empty()) {
|
||||
const ExplodedNode *N = WS.front();
|
||||
WS.pop();
|
||||
|
||||
PriorityEntry = PriorityMap.find(N);
|
||||
|
||||
// Did we process this node already and find it unreachable?
|
||||
if (PriorityEntry == PriorityMap.end())
|
||||
continue;
|
||||
|
||||
unsigned MinPriority = -1U;
|
||||
for (ExplodedNode::const_succ_iterator SI = N->succ_begin(),
|
||||
SE = N->succ_end();
|
||||
SI != SE; ++SI) {
|
||||
PriorityMapTy::iterator SuccEntry = PriorityMap.find(*SI);
|
||||
if (SuccEntry == PriorityMap.end())
|
||||
continue;
|
||||
MinPriority = std::min(SuccEntry->second, MinPriority);
|
||||
}
|
||||
|
||||
if (MinPriority == -1U)
|
||||
PriorityMap.erase(N);
|
||||
else if (PriorityMap[N] == MinPriority + 1)
|
||||
continue;
|
||||
else
|
||||
PriorityMap[N] = MinPriority + 1;
|
||||
|
||||
for (ExplodedNode::const_pred_iterator PI = N->pred_begin(),
|
||||
PE = N->pred_end();
|
||||
PI != PE; ++PI) {
|
||||
assert(PriorityMap.find(*PI) != PriorityMap.end() && "premature removal");
|
||||
WS.push(*PI);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/// CompactPathDiagnostic - This function postprocesses a PathDiagnostic object
|
||||
/// and collapses PathDiagosticPieces that are expanded by macros.
|
||||
static void CompactPathDiagnostic(PathPieces &path, const SourceManager& SM) {
|
||||
|
@ -2216,8 +2297,10 @@ bool GRBugReporter::generatePathDiagnostic(PathDiagnostic& PD,
|
|||
finalReportConfigToken = R->getConfigurationChangeToken();
|
||||
} while (finalReportConfigToken != origReportConfigToken);
|
||||
|
||||
if (!R->isValid())
|
||||
if (!R->isValid()) {
|
||||
TrimG.removeErrorNode(R->getErrorNode());
|
||||
continue;
|
||||
}
|
||||
|
||||
// Finally, prune the diagnostic path of uninteresting stuff.
|
||||
if (!PD.path.empty()) {
|
||||
|
|
Loading…
Reference in New Issue