1 //===- ConstraintManager.h - Constraints on symbolic values. ----*- C++ -*-===//
3 // The LLVM Compiler Infrastructure
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
8 //===----------------------------------------------------------------------===//
10 // This file defined the interface to manage constraints on symbolic values.
12 //===----------------------------------------------------------------------===//
14 #ifndef LLVM_CLANG_STATICANALYZER_CORE_PATHSENSITIVE_CONSTRAINTMANAGER_H
15 #define LLVM_CLANG_STATICANALYZER_CORE_PATHSENSITIVE_CONSTRAINTMANAGER_H
17 #include "clang/Basic/LLVM.h"
18 #include "clang/StaticAnalyzer/Core/PathSensitive/ProgramState_Fwd.h"
19 #include "clang/StaticAnalyzer/Core/PathSensitive/SVals.h"
20 #include "clang/StaticAnalyzer/Core/PathSensitive/SymExpr.h"
21 #include "llvm/ADT/Optional.h"
22 #include "llvm/Support/SaveAndRestore.h"
35 class ProgramStateManager;
39 class ConditionTruthVal {
43 /// Construct a ConditionTruthVal indicating the constraint is constrained
44 /// to either true or false, depending on the boolean value provided.
45 ConditionTruthVal(bool constraint) : Val(constraint) {}
47 /// Construct a ConstraintVal indicating the constraint is underconstrained.
48 ConditionTruthVal() = default;
50 /// \return Stored value, assuming that the value is known.
51 /// Crashes otherwise.
52 bool getValue() const {
56 /// Return true if the constraint is perfectly constrained to 'true'.
57 bool isConstrainedTrue() const {
58 return Val.hasValue() && Val.getValue();
61 /// Return true if the constraint is perfectly constrained to 'false'.
62 bool isConstrainedFalse() const {
63 return Val.hasValue() && !Val.getValue();
66 /// Return true if the constrained is perfectly constrained.
67 bool isConstrained() const {
68 return Val.hasValue();
71 /// Return true if the constrained is underconstrained and we do not know
72 /// if the constraint is true of value.
73 bool isUnderconstrained() const {
74 return !Val.hasValue();
78 class ConstraintManager {
80 ConstraintManager() = default;
81 virtual ~ConstraintManager();
83 virtual ProgramStateRef assume(ProgramStateRef state,
87 using ProgramStatePair = std::pair<ProgramStateRef, ProgramStateRef>;
89 /// Returns a pair of states (StTrue, StFalse) where the given condition is
90 /// assumed to be true or false, respectively.
91 ProgramStatePair assumeDual(ProgramStateRef State, DefinedSVal Cond) {
92 ProgramStateRef StTrue = assume(State, Cond, true);
94 // If StTrue is infeasible, asserting the falseness of Cond is unnecessary
95 // because the existing constraints already establish this.
98 // This check is expensive and should be disabled even in Release+Asserts
100 // FIXME: __OPTIMIZE__ is a GNU extension that Clang implements but MSVC
101 // does not. Is there a good equivalent there?
102 assert(assume(State, Cond, false) && "System is over constrained.");
104 return ProgramStatePair((ProgramStateRef)nullptr, State);
107 ProgramStateRef StFalse = assume(State, Cond, false);
109 // We are careful to return the original state, /not/ StTrue,
110 // because we want to avoid having callers generate a new node
111 // in the ExplodedGraph.
112 return ProgramStatePair(State, (ProgramStateRef)nullptr);
115 return ProgramStatePair(StTrue, StFalse);
118 virtual ProgramStateRef assumeInclusiveRange(ProgramStateRef State,
120 const llvm::APSInt &From,
121 const llvm::APSInt &To,
124 virtual ProgramStatePair assumeInclusiveRangeDual(ProgramStateRef State,
126 const llvm::APSInt &From,
127 const llvm::APSInt &To) {
128 ProgramStateRef StInRange =
129 assumeInclusiveRange(State, Value, From, To, true);
131 // If StTrue is infeasible, asserting the falseness of Cond is unnecessary
132 // because the existing constraints already establish this.
134 return ProgramStatePair((ProgramStateRef)nullptr, State);
136 ProgramStateRef StOutOfRange =
137 assumeInclusiveRange(State, Value, From, To, false);
139 // We are careful to return the original state, /not/ StTrue,
140 // because we want to avoid having callers generate a new node
141 // in the ExplodedGraph.
142 return ProgramStatePair(State, (ProgramStateRef)nullptr);
145 return ProgramStatePair(StInRange, StOutOfRange);
148 /// If a symbol is perfectly constrained to a constant, attempt
149 /// to return the concrete value.
151 /// Note that a ConstraintManager is not obligated to return a concretized
152 /// value for a symbol, even if it is perfectly constrained.
153 virtual const llvm::APSInt* getSymVal(ProgramStateRef state,
154 SymbolRef sym) const {
158 /// Scan all symbols referenced by the constraints. If the symbol is not
159 /// alive, remove it.
160 virtual ProgramStateRef removeDeadBindings(ProgramStateRef state,
161 SymbolReaper& SymReaper) = 0;
163 virtual void print(ProgramStateRef state,
166 const char *sep) = 0;
168 virtual void EndPath(ProgramStateRef state) {}
170 /// Convenience method to query the state to see if a symbol is null or
171 /// not null, or if neither assumption can be made.
172 ConditionTruthVal isNull(ProgramStateRef State, SymbolRef Sym) {
173 SaveAndRestore<bool> DisableNotify(NotifyAssumeClients, false);
175 return checkNull(State, Sym);
179 /// A flag to indicate that clients should be notified of assumptions.
180 /// By default this is the case, but sometimes this needs to be restricted
181 /// to avoid infinite recursions within the ConstraintManager.
183 /// Note that this flag allows the ConstraintManager to be re-entrant,
184 /// but not thread-safe.
185 bool NotifyAssumeClients = true;
187 /// canReasonAbout - Not all ConstraintManagers can accurately reason about
188 /// all SVal values. This method returns true if the ConstraintManager can
189 /// reasonably handle a given SVal value. This is typically queried by
190 /// ExprEngine to determine if the value should be replaced with a
191 /// conjured symbolic value in order to recover some precision.
192 virtual bool canReasonAbout(SVal X) const = 0;
194 /// Returns whether or not a symbol is known to be null ("true"), known to be
195 /// non-null ("false"), or may be either ("underconstrained").
196 virtual ConditionTruthVal checkNull(ProgramStateRef State, SymbolRef Sym);
199 std::unique_ptr<ConstraintManager>
200 CreateRangeConstraintManager(ProgramStateManager &statemgr,
201 SubEngine *subengine);
203 std::unique_ptr<ConstraintManager>
204 CreateZ3ConstraintManager(ProgramStateManager &statemgr, SubEngine *subengine);
209 #endif // LLVM_CLANG_STATICANALYZER_CORE_PATHSENSITIVE_CONSTRAINTMANAGER_H