xref: /freebsd/contrib/llvm-project/clang/lib/StaticAnalyzer/Checkers/STLAlgorithmModeling.cpp (revision 3d36053ca6d6a17d408c8f92c504e6135dc9d8df)
1  //===-- STLAlgorithmModeling.cpp -----------------------------------*- C++ -*--//
2  //
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
6  //
7  //===----------------------------------------------------------------------===//
8  //
9  // Models STL algorithms.
10  //
11  //===----------------------------------------------------------------------===//
12  
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"
18  
19  #include "Iterator.h"
20  
21  using namespace clang;
22  using namespace ento;
23  using namespace iterator;
24  
25  namespace {
26  
27  class STLAlgorithmModeling : public Checker<eval::Call> {
28    bool evalFind(CheckerContext &C, const CallExpr *CE) const;
29  
30    void Find(CheckerContext &C, const CallExpr *CE, unsigned paramNum) const;
31  
32    using FnCheck = bool (STLAlgorithmModeling::*)(CheckerContext &,
33                                                  const CallExpr *) const;
34  
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},
59    };
60  
61  public:
62    STLAlgorithmModeling() = default;
63  
64    bool AggressiveStdFindModeling = false;
65  
66    bool evalCall(const CallEvent &Call, CheckerContext &C) const;
67  }; //
68  
69  bool STLAlgorithmModeling::evalCall(const CallEvent &Call,
70                                      CheckerContext &C) const {
71    const auto *CE = dyn_cast_or_null<CallExpr>(Call.getOriginExpr());
72    if (!CE)
73      return false;
74  
75    const FnCheck *Handler = Callbacks.lookup(Call);
76    if (!Handler)
77      return false;
78  
79    return (this->**Handler)(C, CE);
80  }
81  
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
87    // an iterator.
88    if (!isIteratorType(CE->getArg(1)->getType()))
89      return false;
90  
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())) {
94      Find(C, CE, 0);
95      return true;
96    }
97  
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())) {
101      Find(C, CE, 1);
102      return true;
103    }
104  
105    return false;
106  }
107  
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();
113  
114    SVal RetVal = SVB.conjureSymbolVal(nullptr, CE, LCtx, C.blockCount());
115    SVal Param = State->getSVal(CE->getArg(paramNum), LCtx);
116  
117    auto StateFound = State->BindExpr(CE, LCtx, RetVal);
118  
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);
124    if (Pos) {
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.");
129  
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);
137    }
138  
139    Param = State->getSVal(CE->getArg(paramNum + 1), LCtx);
140  
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
143    // is ahead of it.
144    // FIXME: Reverse iterators
145    Pos = getIteratorPosition(State, Param);
146    if (Pos) {
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.");
151  
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);
159    }
160  
161    C.addTransition(StateFound);
162  
163    if (AggressiveStdFindModeling) {
164      auto StateNotFound = State->BindExpr(CE, LCtx, Param);
165      C.addTransition(StateNotFound);
166    }
167  }
168  
169  } // namespace
170  
171  void ento::registerSTLAlgorithmModeling(CheckerManager &Mgr) {
172    auto *Checker = Mgr.registerChecker<STLAlgorithmModeling>();
173    Checker->AggressiveStdFindModeling =
174        Mgr.getAnalyzerOptions().getCheckerBooleanOption(Checker,
175                                                    "AggressiveStdFindModeling");
176  }
177  
178  bool ento::shouldRegisterSTLAlgorithmModeling(const CheckerManager &mgr) {
179    return true;
180  }
181  
182