mirror of
https://github.com/intel/llvm.git
synced 2026-01-24 17:01:00 +08:00
Prevent assert in ASTMatchFinder.
If nodes without memoization data (e.g. TypeLocs) are bound to specific names, that effectively prevents memoization as those elements cannot be compared effectively. If it is tried anyway, this can lead to an assert as demonstrated in the new test. In the long term, the better solution will be to enable DynTypedNodes without memoization data. For now, simply skip memoization instead. llvm-svn: 213751
This commit is contained in:
@@ -103,6 +103,16 @@ public:
|
||||
return NodeMap;
|
||||
}
|
||||
|
||||
/// \brief Returns \c true if this \c BoundNodesMap can be compared, i.e. all
|
||||
/// stored nodes have memoization data.
|
||||
bool isComparable() const {
|
||||
for (const auto &IDAndNode : NodeMap) {
|
||||
if (!IDAndNode.second.getMemoizationData())
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
private:
|
||||
IDToNodeMap NodeMap;
|
||||
};
|
||||
@@ -153,6 +163,16 @@ public:
|
||||
return Bindings < Other.Bindings;
|
||||
}
|
||||
|
||||
/// \brief Returns \c true if this \c BoundNodesTreeBuilder can be compared,
|
||||
/// i.e. all stored node maps have memoization data.
|
||||
bool isComparable() const {
|
||||
for (const BoundNodesMap &NodesMap : Bindings) {
|
||||
if (!NodesMap.isComparable())
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
private:
|
||||
SmallVector<BoundNodesMap, 16> Bindings;
|
||||
};
|
||||
|
||||
@@ -372,7 +372,7 @@ public:
|
||||
BoundNodesTreeBuilder *Builder, int MaxDepth,
|
||||
TraversalKind Traversal, BindKind Bind) {
|
||||
// For AST-nodes that don't have an identity, we can't memoize.
|
||||
if (!Node.getMemoizationData())
|
||||
if (!Node.getMemoizationData() || !Builder->isComparable())
|
||||
return matchesRecursively(Node, Matcher, Builder, MaxDepth, Traversal,
|
||||
Bind);
|
||||
|
||||
|
||||
@@ -643,6 +643,12 @@ TEST(DeclarationMatcher, HasDescendant) {
|
||||
"};", ZDescendantClassXDescendantClassY));
|
||||
}
|
||||
|
||||
TEST(DeclarationMatcher, HasDescendantMemoization) {
|
||||
DeclarationMatcher CannotMemoize =
|
||||
decl(hasDescendant(typeLoc().bind("x")), has(decl()));
|
||||
EXPECT_TRUE(matches("void f() { int i; }", CannotMemoize));
|
||||
}
|
||||
|
||||
// Implements a run method that returns whether BoundNodes contains a
|
||||
// Decl bound to Id that can be dynamically cast to T.
|
||||
// Optionally checks that the check succeeded a specific number of times.
|
||||
|
||||
Reference in New Issue
Block a user