1//== SVals.h - Abstract Values for Static Analysis ---------*- C++ -*--==//
2//
3//                     The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10//  This file defines SVal, Loc, and NonLoc, classes that represent
11//  abstract r-values for use with path-sensitive value tracking.
12//
13//===----------------------------------------------------------------------===//
14
15#ifndef LLVM_CLANG_GR_RVALUE_H
16#define LLVM_CLANG_GR_RVALUE_H
17
18#include "clang/Basic/LLVM.h"
19#include "clang/StaticAnalyzer/Core/PathSensitive/ProgramState_Fwd.h"
20#include "clang/StaticAnalyzer/Core/PathSensitive/SymbolManager.h"
21#include "llvm/ADT/ImmutableList.h"
22
23//==------------------------------------------------------------------------==//
24//  Base SVal types.
25//==------------------------------------------------------------------------==//
26
27namespace clang {
28
29namespace ento {
30
31class CompoundValData;
32class LazyCompoundValData;
33class ProgramState;
34class BasicValueFactory;
35class MemRegion;
36class TypedValueRegion;
37class MemRegionManager;
38class ProgramStateManager;
39class SValBuilder;
40
41/// SVal - This represents a symbolic expression, which can be either
42///  an L-value or an R-value.
43///
44class SVal {
45public:
46  enum BaseKind {
47    // The enumerators must be representable using 2 bits.
48    UndefinedKind = 0,  // for subclass UndefinedVal (an uninitialized value)
49    UnknownKind = 1,    // for subclass UnknownVal (a void value)
50    LocKind = 2,        // for subclass Loc (an L-value)
51    NonLocKind = 3      // for subclass NonLoc (an R-value that's not
52                        //   an L-value)
53  };
54  enum { BaseBits = 2, BaseMask = 0x3 };
55
56protected:
57  const void *Data;
58
59  /// The lowest 2 bits are a BaseKind (0 -- 3).
60  ///  The higher bits are an unsigned "kind" value.
61  unsigned Kind;
62
63  explicit SVal(const void *d, bool isLoc, unsigned ValKind)
64  : Data(d), Kind((isLoc ? LocKind : NonLocKind) | (ValKind << BaseBits)) {}
65
66  explicit SVal(BaseKind k, const void *D = nullptr)
67    : Data(D), Kind(k) {}
68
69public:
70  explicit SVal() : Data(nullptr), Kind(0) {}
71
72  /// \brief Convert to the specified SVal type, asserting that this SVal is of
73  /// the desired type.
74  template<typename T>
75  T castAs() const {
76    assert(T::isKind(*this));
77    T t;
78    SVal& sv = t;
79    sv = *this;
80    return t;
81  }
82
83  /// \brief Convert to the specified SVal type, returning None if this SVal is
84  /// not of the desired type.
85  template<typename T>
86  Optional<T> getAs() const {
87    if (!T::isKind(*this))
88      return None;
89    T t;
90    SVal& sv = t;
91    sv = *this;
92    return t;
93  }
94
95  /// BufferTy - A temporary buffer to hold a set of SVals.
96  typedef SmallVector<SVal,5> BufferTy;
97
98  inline unsigned getRawKind() const { return Kind; }
99  inline BaseKind getBaseKind() const { return (BaseKind) (Kind & BaseMask); }
100  inline unsigned getSubKind() const { return (Kind & ~BaseMask) >> BaseBits; }
101
102  // This method is required for using SVal in a FoldingSetNode.  It
103  // extracts a unique signature for this SVal object.
104  inline void Profile(llvm::FoldingSetNodeID& ID) const {
105    ID.AddInteger((unsigned) getRawKind());
106    ID.AddPointer(Data);
107  }
108
109  inline bool operator==(const SVal& R) const {
110    return getRawKind() == R.getRawKind() && Data == R.Data;
111  }
112
113  inline bool operator!=(const SVal& R) const {
114    return !(*this == R);
115  }
116
117  inline bool isUnknown() const {
118    return getRawKind() == UnknownKind;
119  }
120
121  inline bool isUndef() const {
122    return getRawKind() == UndefinedKind;
123  }
124
125  inline bool isUnknownOrUndef() const {
126    return getRawKind() <= UnknownKind;
127  }
128
129  inline bool isValid() const {
130    return getRawKind() > UnknownKind;
131  }
132
133  bool isConstant() const;
134
135  bool isConstant(int I) const;
136
137  bool isZeroConstant() const;
138
139  /// hasConjuredSymbol - If this SVal wraps a conjured symbol, return true;
140  bool hasConjuredSymbol() const;
141
142  /// getAsFunctionDecl - If this SVal is a MemRegionVal and wraps a
143  /// CodeTextRegion wrapping a FunctionDecl, return that FunctionDecl.
144  /// Otherwise return 0.
145  const FunctionDecl *getAsFunctionDecl() const;
146
147  /// \brief If this SVal is a location and wraps a symbol, return that
148  ///  SymbolRef. Otherwise return 0.
149  ///
150  /// Casts are ignored during lookup.
151  /// \param IncludeBaseRegions The boolean that controls whether the search
152  /// should continue to the base regions if the region is not symbolic.
153  SymbolRef getAsLocSymbol(bool IncludeBaseRegions = false) const;
154
155  /// Get the symbol in the SVal or its base region.
156  SymbolRef getLocSymbolInBase() const;
157
158  /// \brief If this SVal wraps a symbol return that SymbolRef.
159  /// Otherwise, return 0.
160  ///
161  /// Casts are ignored during lookup.
162  /// \param IncludeBaseRegions The boolean that controls whether the search
163  /// should continue to the base regions if the region is not symbolic.
164  SymbolRef getAsSymbol(bool IncludeBaseRegions = false) const;
165
166  /// getAsSymbolicExpression - If this Sval wraps a symbolic expression then
167  ///  return that expression.  Otherwise return NULL.
168  const SymExpr *getAsSymbolicExpression() const;
169
170  const SymExpr* getAsSymExpr() const;
171
172  const MemRegion *getAsRegion() const;
173
174  void dumpToStream(raw_ostream &OS) const;
175  void dump() const;
176
177  SymExpr::symbol_iterator symbol_begin() const {
178    const SymExpr *SE = getAsSymbolicExpression();
179    if (SE)
180      return SE->symbol_begin();
181    else
182      return SymExpr::symbol_iterator();
183  }
184
185  SymExpr::symbol_iterator symbol_end() const {
186    return SymExpr::symbol_end();
187  }
188};
189
190
191class UndefinedVal : public SVal {
192public:
193  UndefinedVal() : SVal(UndefinedKind) {}
194
195private:
196  friend class SVal;
197  static bool isKind(const SVal& V) {
198    return V.getBaseKind() == UndefinedKind;
199  }
200};
201
202class DefinedOrUnknownSVal : public SVal {
203private:
204  // We want calling these methods to be a compiler error since they are
205  // tautologically false.
206  bool isUndef() const LLVM_DELETED_FUNCTION;
207  bool isValid() const LLVM_DELETED_FUNCTION;
208
209protected:
210  DefinedOrUnknownSVal() {}
211  explicit DefinedOrUnknownSVal(const void *d, bool isLoc, unsigned ValKind)
212    : SVal(d, isLoc, ValKind) {}
213
214  explicit DefinedOrUnknownSVal(BaseKind k, void *D = nullptr)
215    : SVal(k, D) {}
216
217private:
218  friend class SVal;
219  static bool isKind(const SVal& V) {
220    return !V.isUndef();
221  }
222};
223
224class UnknownVal : public DefinedOrUnknownSVal {
225public:
226  explicit UnknownVal() : DefinedOrUnknownSVal(UnknownKind) {}
227
228private:
229  friend class SVal;
230  static bool isKind(const SVal &V) {
231    return V.getBaseKind() == UnknownKind;
232  }
233};
234
235class DefinedSVal : public DefinedOrUnknownSVal {
236private:
237  // We want calling these methods to be a compiler error since they are
238  // tautologically true/false.
239  bool isUnknown() const LLVM_DELETED_FUNCTION;
240  bool isUnknownOrUndef() const LLVM_DELETED_FUNCTION;
241  bool isValid() const LLVM_DELETED_FUNCTION;
242protected:
243  DefinedSVal() {}
244  explicit DefinedSVal(const void *d, bool isLoc, unsigned ValKind)
245    : DefinedOrUnknownSVal(d, isLoc, ValKind) {}
246private:
247  friend class SVal;
248  static bool isKind(const SVal& V) {
249    return !V.isUnknownOrUndef();
250  }
251};
252
253
254/// \brief Represents an SVal that is guaranteed to not be UnknownVal.
255class KnownSVal : public SVal {
256  KnownSVal() {}
257  friend class SVal;
258  static bool isKind(const SVal &V) {
259    return !V.isUnknown();
260  }
261public:
262  KnownSVal(const DefinedSVal &V) : SVal(V) {}
263  KnownSVal(const UndefinedVal &V) : SVal(V) {}
264};
265
266class NonLoc : public DefinedSVal {
267protected:
268  NonLoc() {}
269  explicit NonLoc(unsigned SubKind, const void *d)
270    : DefinedSVal(d, false, SubKind) {}
271
272public:
273  void dumpToStream(raw_ostream &Out) const;
274
275private:
276  friend class SVal;
277  static bool isKind(const SVal& V) {
278    return V.getBaseKind() == NonLocKind;
279  }
280};
281
282class Loc : public DefinedSVal {
283protected:
284  Loc() {}
285  explicit Loc(unsigned SubKind, const void *D)
286  : DefinedSVal(const_cast<void*>(D), true, SubKind) {}
287
288public:
289  void dumpToStream(raw_ostream &Out) const;
290
291  static inline bool isLocType(QualType T) {
292    return T->isAnyPointerType() || T->isBlockPointerType() ||
293           T->isReferenceType() || T->isNullPtrType();
294  }
295
296private:
297  friend class SVal;
298  static bool isKind(const SVal& V) {
299    return V.getBaseKind() == LocKind;
300  }
301};
302
303//==------------------------------------------------------------------------==//
304//  Subclasses of NonLoc.
305//==------------------------------------------------------------------------==//
306
307namespace nonloc {
308
309enum Kind { ConcreteIntKind, SymbolValKind,
310            LocAsIntegerKind, CompoundValKind, LazyCompoundValKind };
311
312/// \brief Represents symbolic expression.
313class SymbolVal : public NonLoc {
314public:
315  SymbolVal(SymbolRef sym) : NonLoc(SymbolValKind, sym) {}
316
317  SymbolRef getSymbol() const {
318    return (const SymExpr*) Data;
319  }
320
321  bool isExpression() const {
322    return !isa<SymbolData>(getSymbol());
323  }
324
325private:
326  friend class SVal;
327  SymbolVal() {}
328  static bool isKind(const SVal& V) {
329    return V.getBaseKind() == NonLocKind &&
330           V.getSubKind() == SymbolValKind;
331  }
332
333  static bool isKind(const NonLoc& V) {
334    return V.getSubKind() == SymbolValKind;
335  }
336};
337
338/// \brief Value representing integer constant.
339class ConcreteInt : public NonLoc {
340public:
341  explicit ConcreteInt(const llvm::APSInt& V) : NonLoc(ConcreteIntKind, &V) {}
342
343  const llvm::APSInt& getValue() const {
344    return *static_cast<const llvm::APSInt*>(Data);
345  }
346
347  // Transfer functions for binary/unary operations on ConcreteInts.
348  SVal evalBinOp(SValBuilder &svalBuilder, BinaryOperator::Opcode Op,
349                 const ConcreteInt& R) const;
350
351  ConcreteInt evalComplement(SValBuilder &svalBuilder) const;
352
353  ConcreteInt evalMinus(SValBuilder &svalBuilder) const;
354
355private:
356  friend class SVal;
357  ConcreteInt() {}
358  static bool isKind(const SVal& V) {
359    return V.getBaseKind() == NonLocKind &&
360           V.getSubKind() == ConcreteIntKind;
361  }
362
363  static bool isKind(const NonLoc& V) {
364    return V.getSubKind() == ConcreteIntKind;
365  }
366};
367
368class LocAsInteger : public NonLoc {
369  friend class ento::SValBuilder;
370
371  explicit LocAsInteger(const std::pair<SVal, uintptr_t> &data)
372      : NonLoc(LocAsIntegerKind, &data) {
373    assert (data.first.getAs<Loc>());
374  }
375
376public:
377
378  Loc getLoc() const {
379    const std::pair<SVal, uintptr_t> *D =
380      static_cast<const std::pair<SVal, uintptr_t> *>(Data);
381    return D->first.castAs<Loc>();
382  }
383
384  Loc getPersistentLoc() const {
385    const std::pair<SVal, uintptr_t> *D =
386      static_cast<const std::pair<SVal, uintptr_t> *>(Data);
387    const SVal& V = D->first;
388    return V.castAs<Loc>();
389  }
390
391  unsigned getNumBits() const {
392    const std::pair<SVal, uintptr_t> *D =
393      static_cast<const std::pair<SVal, uintptr_t> *>(Data);
394    return D->second;
395  }
396
397private:
398  friend class SVal;
399  LocAsInteger() {}
400  static bool isKind(const SVal& V) {
401    return V.getBaseKind() == NonLocKind &&
402           V.getSubKind() == LocAsIntegerKind;
403  }
404
405  static bool isKind(const NonLoc& V) {
406    return V.getSubKind() == LocAsIntegerKind;
407  }
408};
409
410class CompoundVal : public NonLoc {
411  friend class ento::SValBuilder;
412
413  explicit CompoundVal(const CompoundValData* D) : NonLoc(CompoundValKind, D) {}
414
415public:
416  const CompoundValData* getValue() const {
417    return static_cast<const CompoundValData*>(Data);
418  }
419
420  typedef llvm::ImmutableList<SVal>::iterator iterator;
421  iterator begin() const;
422  iterator end() const;
423
424private:
425  friend class SVal;
426  CompoundVal() {}
427  static bool isKind(const SVal& V) {
428    return V.getBaseKind() == NonLocKind && V.getSubKind() == CompoundValKind;
429  }
430
431  static bool isKind(const NonLoc& V) {
432    return V.getSubKind() == CompoundValKind;
433  }
434};
435
436class LazyCompoundVal : public NonLoc {
437  friend class ento::SValBuilder;
438
439  explicit LazyCompoundVal(const LazyCompoundValData *D)
440    : NonLoc(LazyCompoundValKind, D) {}
441public:
442  const LazyCompoundValData *getCVData() const {
443    return static_cast<const LazyCompoundValData*>(Data);
444  }
445  const void *getStore() const;
446  const TypedValueRegion *getRegion() const;
447
448private:
449  friend class SVal;
450  LazyCompoundVal() {}
451  static bool isKind(const SVal& V) {
452    return V.getBaseKind() == NonLocKind &&
453           V.getSubKind() == LazyCompoundValKind;
454  }
455  static bool isKind(const NonLoc& V) {
456    return V.getSubKind() == LazyCompoundValKind;
457  }
458};
459
460} // end namespace ento::nonloc
461
462//==------------------------------------------------------------------------==//
463//  Subclasses of Loc.
464//==------------------------------------------------------------------------==//
465
466namespace loc {
467
468enum Kind { GotoLabelKind, MemRegionKind, ConcreteIntKind };
469
470class GotoLabel : public Loc {
471public:
472  explicit GotoLabel(LabelDecl *Label) : Loc(GotoLabelKind, Label) {}
473
474  const LabelDecl *getLabel() const {
475    return static_cast<const LabelDecl*>(Data);
476  }
477
478private:
479  friend class SVal;
480  GotoLabel() {}
481  static bool isKind(const SVal& V) {
482    return V.getBaseKind() == LocKind && V.getSubKind() == GotoLabelKind;
483  }
484
485  static bool isKind(const Loc& V) {
486    return V.getSubKind() == GotoLabelKind;
487  }
488};
489
490
491class MemRegionVal : public Loc {
492public:
493  explicit MemRegionVal(const MemRegion* r) : Loc(MemRegionKind, r) {}
494
495  /// \brief Get the underlining region.
496  const MemRegion* getRegion() const {
497    return static_cast<const MemRegion*>(Data);
498  }
499
500  /// \brief Get the underlining region and strip casts.
501  const MemRegion* stripCasts(bool StripBaseCasts = true) const;
502
503  template <typename REGION>
504  const REGION* getRegionAs() const {
505    return dyn_cast<REGION>(getRegion());
506  }
507
508  inline bool operator==(const MemRegionVal& R) const {
509    return getRegion() == R.getRegion();
510  }
511
512  inline bool operator!=(const MemRegionVal& R) const {
513    return getRegion() != R.getRegion();
514  }
515
516private:
517  friend class SVal;
518  MemRegionVal() {}
519  static bool isKind(const SVal& V) {
520    return V.getBaseKind() == LocKind &&
521           V.getSubKind() == MemRegionKind;
522  }
523
524  static bool isKind(const Loc& V) {
525    return V.getSubKind() == MemRegionKind;
526  }
527};
528
529class ConcreteInt : public Loc {
530public:
531  explicit ConcreteInt(const llvm::APSInt& V) : Loc(ConcreteIntKind, &V) {}
532
533  const llvm::APSInt& getValue() const {
534    return *static_cast<const llvm::APSInt*>(Data);
535  }
536
537  // Transfer functions for binary/unary operations on ConcreteInts.
538  SVal evalBinOp(BasicValueFactory& BasicVals, BinaryOperator::Opcode Op,
539                 const ConcreteInt& R) const;
540
541private:
542  friend class SVal;
543  ConcreteInt() {}
544  static bool isKind(const SVal& V) {
545    return V.getBaseKind() == LocKind &&
546           V.getSubKind() == ConcreteIntKind;
547  }
548
549  static bool isKind(const Loc& V) {
550    return V.getSubKind() == ConcreteIntKind;
551  }
552};
553
554} // end ento::loc namespace
555
556} // end ento namespace
557
558} // end clang namespace
559
560namespace llvm {
561static inline raw_ostream &operator<<(raw_ostream &os,
562                                            clang::ento::SVal V) {
563  V.dumpToStream(os);
564  return os;
565}
566
567template <typename T> struct isPodLike;
568template <> struct isPodLike<clang::ento::SVal> {
569  static const bool value = true;
570};
571
572} // end llvm namespace
573
574#endif
575