@@ -86,14 +86,20 @@ class InnerPointerChecker
86
86
};
87
87
88
88
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" }) {}
97
103
98
104
// / Check whether the called member function potentially invalidates
99
105
// / pointers referring to the container object's inner buffer.
@@ -122,21 +128,6 @@ class InnerPointerChecker
122
128
123
129
} // end anonymous namespace
124
130
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
-
140
131
bool InnerPointerChecker::isInvalidatingMemberFunction (
141
132
const CallEvent &Call) const {
142
133
if (const auto *MemOpCall = dyn_cast<CXXMemberOperatorCall>(&Call)) {
@@ -220,33 +211,31 @@ void InnerPointerChecker::checkPostCall(const CallEvent &Call,
220
211
ProgramStateRef State = C.getState ();
221
212
222
213
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);
243
231
}
232
+ return ;
233
+ }
244
234
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 ;
250
239
}
251
240
}
252
241
0 commit comments