1 //===-- STLAlgorithmModeling.cpp -----------------------------------*- C++ -*--//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 // Models STL algorithms.
11 //===----------------------------------------------------------------------===//
13 #include "clang/StaticAnalyzer/Checkers/BuiltinCheckerRegistration.h"
14 #include "clang/StaticAnalyzer/Core/Checker.h"
15 #include "clang/StaticAnalyzer/Core/PathSensitive/CallDescription.h"
16 #include "clang/StaticAnalyzer/Core/PathSensitive/CallEvent.h"
17 #include "clang/StaticAnalyzer/Core/PathSensitive/CheckerContext.h"
21 using namespace clang
;
23 using namespace iterator
;
27 class STLAlgorithmModeling
: public Checker
<eval::Call
> {
28 bool evalFind(CheckerContext
&C
, const CallExpr
*CE
) const;
30 void Find(CheckerContext
&C
, const CallExpr
*CE
, unsigned paramNum
) const;
32 using FnCheck
= bool (STLAlgorithmModeling::*)(CheckerContext
&,
33 const CallExpr
*) const;
35 const CallDescriptionMap
<FnCheck
> Callbacks
= {
36 {{{"std", "find"}, 3}, &STLAlgorithmModeling::evalFind
},
37 {{{"std", "find"}, 4}, &STLAlgorithmModeling::evalFind
},
38 {{{"std", "find_if"}, 3}, &STLAlgorithmModeling::evalFind
},
39 {{{"std", "find_if"}, 4}, &STLAlgorithmModeling::evalFind
},
40 {{{"std", "find_if_not"}, 3}, &STLAlgorithmModeling::evalFind
},
41 {{{"std", "find_if_not"}, 4}, &STLAlgorithmModeling::evalFind
},
42 {{{"std", "find_first_of"}, 4}, &STLAlgorithmModeling::evalFind
},
43 {{{"std", "find_first_of"}, 5}, &STLAlgorithmModeling::evalFind
},
44 {{{"std", "find_first_of"}, 6}, &STLAlgorithmModeling::evalFind
},
45 {{{"std", "find_end"}, 4}, &STLAlgorithmModeling::evalFind
},
46 {{{"std", "find_end"}, 5}, &STLAlgorithmModeling::evalFind
},
47 {{{"std", "find_end"}, 6}, &STLAlgorithmModeling::evalFind
},
48 {{{"std", "lower_bound"}, 3}, &STLAlgorithmModeling::evalFind
},
49 {{{"std", "lower_bound"}, 4}, &STLAlgorithmModeling::evalFind
},
50 {{{"std", "upper_bound"}, 3}, &STLAlgorithmModeling::evalFind
},
51 {{{"std", "upper_bound"}, 4}, &STLAlgorithmModeling::evalFind
},
52 {{{"std", "search"}, 3}, &STLAlgorithmModeling::evalFind
},
53 {{{"std", "search"}, 4}, &STLAlgorithmModeling::evalFind
},
54 {{{"std", "search"}, 5}, &STLAlgorithmModeling::evalFind
},
55 {{{"std", "search"}, 6}, &STLAlgorithmModeling::evalFind
},
56 {{{"std", "search_n"}, 4}, &STLAlgorithmModeling::evalFind
},
57 {{{"std", "search_n"}, 5}, &STLAlgorithmModeling::evalFind
},
58 {{{"std", "search_n"}, 6}, &STLAlgorithmModeling::evalFind
},
62 STLAlgorithmModeling() = default;
64 bool AggressiveStdFindModeling
= false;
66 bool evalCall(const CallEvent
&Call
, CheckerContext
&C
) const;
69 bool STLAlgorithmModeling::evalCall(const CallEvent
&Call
,
70 CheckerContext
&C
) const {
71 const auto *CE
= dyn_cast_or_null
<CallExpr
>(Call
.getOriginExpr());
75 const FnCheck
*Handler
= Callbacks
.lookup(Call
);
79 return (this->**Handler
)(C
, CE
);
82 bool STLAlgorithmModeling::evalFind(CheckerContext
&C
,
83 const CallExpr
*CE
) const {
84 // std::find()-like functions either take their primary range in the first
85 // two parameters, or if the first parameter is "execution policy" then in
86 // the second and third. This means that the second parameter must always be
88 if (!isIteratorType(CE
->getArg(1)->getType()))
91 // If no "execution policy" parameter is used then the first argument is the
92 // beginning of the range.
93 if (isIteratorType(CE
->getArg(0)->getType())) {
98 // If "execution policy" parameter is used then the second argument is the
99 // beginning of the range.
100 if (isIteratorType(CE
->getArg(2)->getType())) {
108 void STLAlgorithmModeling::Find(CheckerContext
&C
, const CallExpr
*CE
,
109 unsigned paramNum
) const {
110 auto State
= C
.getState();
111 auto &SVB
= C
.getSValBuilder();
112 const auto *LCtx
= C
.getLocationContext();
114 SVal RetVal
= SVB
.conjureSymbolVal(nullptr, CE
, LCtx
, C
.blockCount());
115 SVal Param
= State
->getSVal(CE
->getArg(paramNum
), LCtx
);
117 auto StateFound
= State
->BindExpr(CE
, LCtx
, RetVal
);
119 // If we have an iterator position for the range-begin argument then we can
120 // assume that in case of successful search the position of the found element
121 // is not ahead of it.
122 // FIXME: Reverse iterators
123 const auto *Pos
= getIteratorPosition(State
, Param
);
125 StateFound
= createIteratorPosition(StateFound
, RetVal
, Pos
->getContainer(),
126 CE
, LCtx
, C
.blockCount());
127 const auto *NewPos
= getIteratorPosition(StateFound
, RetVal
);
128 assert(NewPos
&& "Failed to create new iterator position.");
130 SVal GreaterOrEqual
= SVB
.evalBinOp(StateFound
, BO_GE
,
131 nonloc::SymbolVal(NewPos
->getOffset()),
132 nonloc::SymbolVal(Pos
->getOffset()),
133 SVB
.getConditionType());
134 assert(isa
<DefinedSVal
>(GreaterOrEqual
) &&
135 "Symbol comparison must be a `DefinedSVal`");
136 StateFound
= StateFound
->assume(GreaterOrEqual
.castAs
<DefinedSVal
>(), true);
139 Param
= State
->getSVal(CE
->getArg(paramNum
+ 1), LCtx
);
141 // If we have an iterator position for the range-end argument then we can
142 // assume that in case of successful search the position of the found element
144 // FIXME: Reverse iterators
145 Pos
= getIteratorPosition(State
, Param
);
147 StateFound
= createIteratorPosition(StateFound
, RetVal
, Pos
->getContainer(),
148 CE
, LCtx
, C
.blockCount());
149 const auto *NewPos
= getIteratorPosition(StateFound
, RetVal
);
150 assert(NewPos
&& "Failed to create new iterator position.");
152 SVal Less
= SVB
.evalBinOp(StateFound
, BO_LT
,
153 nonloc::SymbolVal(NewPos
->getOffset()),
154 nonloc::SymbolVal(Pos
->getOffset()),
155 SVB
.getConditionType());
156 assert(isa
<DefinedSVal
>(Less
) &&
157 "Symbol comparison must be a `DefinedSVal`");
158 StateFound
= StateFound
->assume(Less
.castAs
<DefinedSVal
>(), true);
161 C
.addTransition(StateFound
);
163 if (AggressiveStdFindModeling
) {
164 auto StateNotFound
= State
->BindExpr(CE
, LCtx
, Param
);
165 C
.addTransition(StateNotFound
);
171 void ento::registerSTLAlgorithmModeling(CheckerManager
&Mgr
) {
172 auto *Checker
= Mgr
.registerChecker
<STLAlgorithmModeling
>();
173 Checker
->AggressiveStdFindModeling
=
174 Mgr
.getAnalyzerOptions().getCheckerBooleanOption(Checker
,
175 "AggressiveStdFindModeling");
178 bool ento::shouldRegisterSTLAlgorithmModeling(const CheckerManager
&mgr
) {