Skip to content

Commit 2ca72e0

Browse files
committedAug 22, 2018
[analyzer] Improve CallDescription to handle c++ method.
Summary: `CallDecription` can only handle function for the time being. If we want to match c++ method, we can only use method name to match and can't improve the matching accuracy through the qualifiers. This patch add the support for `QualifiedName` matching to improve the matching accuracy. Reviewers: xazax.hun, NoQ, george.karpenkov, rnkovacs Reviewed By: xazax.hun, NoQ, rnkovacs Subscribers: Szelethus, szepet, rnkovacs, a.sidorin, mikhail.ramalho, cfe-commits, MTC Differential Revision: https://reviews.llvm.org/D48027 llvm-svn: 340407
1 parent 0353308 commit 2ca72e0

File tree

3 files changed

+85
-52
lines changed

3 files changed

+85
-52
lines changed
 

‎clang/include/clang/StaticAnalyzer/Core/PathSensitive/CallEvent.h

+20-3
Original file line numberDiff line numberDiff line change
@@ -80,12 +80,28 @@ class CallDescription {
8080

8181
mutable IdentifierInfo *II = nullptr;
8282
mutable bool IsLookupDone = false;
83-
StringRef FuncName;
83+
// The list of the qualified names used to identify the specified CallEvent,
84+
// e.g. "{a, b}" represent the qualified names, like "a::b".
85+
std::vector<StringRef> QualifiedName;
8486
unsigned RequiredArgs;
8587

8688
public:
8789
const static unsigned NoArgRequirement = std::numeric_limits<unsigned>::max();
8890

91+
/// Constructs a CallDescription object.
92+
///
93+
/// @param QualifiedName The list of the qualified names of the function that
94+
/// will be matched. It does not require the user to provide the full list of
95+
/// the qualified name. The more details provided, the more accurate the
96+
/// matching.
97+
///
98+
/// @param RequiredArgs The number of arguments that is expected to match a
99+
/// call. Omit this parameter to match every occurrence of call with a given
100+
/// name regardless the number of arguments.
101+
CallDescription(std::vector<StringRef> QualifiedName,
102+
unsigned RequiredArgs = NoArgRequirement)
103+
: QualifiedName(QualifiedName), RequiredArgs(RequiredArgs) {}
104+
89105
/// Constructs a CallDescription object.
90106
///
91107
/// @param FuncName The name of the function that will be matched.
@@ -94,10 +110,11 @@ class CallDescription {
94110
/// call. Omit this parameter to match every occurrence of call with a given
95111
/// name regardless the number of arguments.
96112
CallDescription(StringRef FuncName, unsigned RequiredArgs = NoArgRequirement)
97-
: FuncName(FuncName), RequiredArgs(RequiredArgs) {}
113+
: CallDescription(std::vector<StringRef>({FuncName}), NoArgRequirement) {
114+
}
98115

99116
/// Get the name of the function that this object matches.
100-
StringRef getFunctionName() const { return FuncName; }
117+
StringRef getFunctionName() const { return QualifiedName.back(); }
101118
};
102119

103120
template<typename T = CallEvent>

‎clang/lib/StaticAnalyzer/Checkers/InnerPointerChecker.cpp

+37-48
Original file line numberDiff line numberDiff line change
@@ -86,14 +86,20 @@ class InnerPointerChecker
8686
};
8787

8888
InnerPointerChecker()
89-
: AppendFn("append"), AssignFn("assign"), ClearFn("clear"),
90-
CStrFn("c_str"), DataFn("data"), EraseFn("erase"), InsertFn("insert"),
91-
PopBackFn("pop_back"), PushBackFn("push_back"), ReplaceFn("replace"),
92-
ReserveFn("reserve"), ResizeFn("resize"),
93-
ShrinkToFitFn("shrink_to_fit"), SwapFn("swap") {}
94-
95-
/// Check if the object of this member function call is a `basic_string`.
96-
bool isCalledOnStringObject(const CXXInstanceCall *ICall) const;
89+
: AppendFn({"std", "basic_string", "append"}),
90+
AssignFn({"std", "basic_string", "assign"}),
91+
ClearFn({"std", "basic_string", "clear"}),
92+
CStrFn({"std", "basic_string", "c_str"}),
93+
DataFn({"std", "basic_string", "data"}),
94+
EraseFn({"std", "basic_string", "erase"}),
95+
InsertFn({"std", "basic_string", "insert"}),
96+
PopBackFn({"std", "basic_string", "pop_back"}),
97+
PushBackFn({"std", "basic_string", "push_back"}),
98+
ReplaceFn({"std", "basic_string", "replace"}),
99+
ReserveFn({"std", "basic_string", "reserve"}),
100+
ResizeFn({"std", "basic_string", "resize"}),
101+
ShrinkToFitFn({"std", "basic_string", "shrink_to_fit"}),
102+
SwapFn({"std", "basic_string", "swap"}) {}
97103

98104
/// Check whether the called member function potentially invalidates
99105
/// pointers referring to the container object's inner buffer.
@@ -122,21 +128,6 @@ class InnerPointerChecker
122128

123129
} // end anonymous namespace
124130

125-
bool InnerPointerChecker::isCalledOnStringObject(
126-
const CXXInstanceCall *ICall) const {
127-
const auto *ObjRegion =
128-
dyn_cast_or_null<TypedValueRegion>(ICall->getCXXThisVal().getAsRegion());
129-
if (!ObjRegion)
130-
return false;
131-
132-
QualType ObjTy = ObjRegion->getValueType();
133-
if (ObjTy.isNull())
134-
return false;
135-
136-
CXXRecordDecl *Decl = ObjTy->getAsCXXRecordDecl();
137-
return Decl && Decl->getName() == "basic_string";
138-
}
139-
140131
bool InnerPointerChecker::isInvalidatingMemberFunction(
141132
const CallEvent &Call) const {
142133
if (const auto *MemOpCall = dyn_cast<CXXMemberOperatorCall>(&Call)) {
@@ -220,33 +211,31 @@ void InnerPointerChecker::checkPostCall(const CallEvent &Call,
220211
ProgramStateRef State = C.getState();
221212

222213
if (const auto *ICall = dyn_cast<CXXInstanceCall>(&Call)) {
223-
if (isCalledOnStringObject(ICall)) {
224-
const auto *ObjRegion = dyn_cast_or_null<TypedValueRegion>(
225-
ICall->getCXXThisVal().getAsRegion());
226-
227-
if (Call.isCalled(CStrFn) || Call.isCalled(DataFn)) {
228-
SVal RawPtr = Call.getReturnValue();
229-
if (SymbolRef Sym = RawPtr.getAsSymbol(/*IncludeBaseRegions=*/true)) {
230-
// Start tracking this raw pointer by adding it to the set of symbols
231-
// associated with this container object in the program state map.
232-
233-
PtrSet::Factory &F = State->getStateManager().get_context<PtrSet>();
234-
const PtrSet *SetPtr = State->get<RawPtrMap>(ObjRegion);
235-
PtrSet Set = SetPtr ? *SetPtr : F.getEmptySet();
236-
assert(C.wasInlined || !Set.contains(Sym));
237-
Set = F.add(Set, Sym);
238-
239-
State = State->set<RawPtrMap>(ObjRegion, Set);
240-
C.addTransition(State);
241-
}
242-
return;
214+
const auto *ObjRegion = dyn_cast_or_null<TypedValueRegion>(
215+
ICall->getCXXThisVal().getAsRegion());
216+
217+
if (Call.isCalled(CStrFn) || Call.isCalled(DataFn)) {
218+
SVal RawPtr = Call.getReturnValue();
219+
if (SymbolRef Sym = RawPtr.getAsSymbol(/*IncludeBaseRegions=*/true)) {
220+
// Start tracking this raw pointer by adding it to the set of symbols
221+
// associated with this container object in the program state map.
222+
223+
PtrSet::Factory &F = State->getStateManager().get_context<PtrSet>();
224+
const PtrSet *SetPtr = State->get<RawPtrMap>(ObjRegion);
225+
PtrSet Set = SetPtr ? *SetPtr : F.getEmptySet();
226+
assert(C.wasInlined || !Set.contains(Sym));
227+
Set = F.add(Set, Sym);
228+
229+
State = State->set<RawPtrMap>(ObjRegion, Set);
230+
C.addTransition(State);
243231
}
232+
return;
233+
}
244234

245-
// Check [string.require] / second point.
246-
if (isInvalidatingMemberFunction(Call)) {
247-
markPtrSymbolsReleased(Call, State, ObjRegion, C);
248-
return;
249-
}
235+
// Check [string.require] / second point.
236+
if (isInvalidatingMemberFunction(Call)) {
237+
markPtrSymbolsReleased(Call, State, ObjRegion, C);
238+
return;
250239
}
251240
}
252241

‎clang/lib/StaticAnalyzer/Core/CallEvent.cpp

+28-1
Original file line numberDiff line numberDiff line change
@@ -359,11 +359,38 @@ bool CallEvent::isCalled(const CallDescription &CD) const {
359359
return false;
360360
if (!CD.IsLookupDone) {
361361
CD.IsLookupDone = true;
362-
CD.II = &getState()->getStateManager().getContext().Idents.get(CD.FuncName);
362+
CD.II = &getState()->getStateManager().getContext().Idents.get(
363+
CD.getFunctionName());
363364
}
364365
const IdentifierInfo *II = getCalleeIdentifier();
365366
if (!II || II != CD.II)
366367
return false;
368+
369+
const Decl *D = getDecl();
370+
// If CallDescription provides prefix names, use them to improve matching
371+
// accuracy.
372+
if (CD.QualifiedName.size() > 1 && D) {
373+
const DeclContext *Ctx = D->getDeclContext();
374+
std::vector<StringRef> QualifiedName = CD.QualifiedName;
375+
QualifiedName.pop_back();
376+
for (; Ctx && isa<NamedDecl>(Ctx); Ctx = Ctx->getParent()) {
377+
if (const auto *ND = dyn_cast<NamespaceDecl>(Ctx)) {
378+
if (!QualifiedName.empty() && ND->getName() == QualifiedName.back())
379+
QualifiedName.pop_back();
380+
continue;
381+
}
382+
383+
if (const auto *RD = dyn_cast<RecordDecl>(Ctx)) {
384+
if (!QualifiedName.empty() && RD->getName() == QualifiedName.back())
385+
QualifiedName.pop_back();
386+
continue;
387+
}
388+
}
389+
390+
if (!QualifiedName.empty())
391+
return false;
392+
}
393+
367394
return (CD.RequiredArgs == CallDescription::NoArgRequirement ||
368395
CD.RequiredArgs == getNumArgs());
369396
}

0 commit comments

Comments
 (0)
Please sign in to comment.