blob: 1a6e1e467592d23df982b7d5ce5e1c0b2c9a8582 [file] [log] [blame]
Zhongxing Xua3a5a282009-06-23 06:22:22 +00001//== ValueManager.cpp - Aggregate manager of symbols and SVals --*- 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 ValueManager, a class that manages symbolic values
11// and SVals created for use by GRExprEngine and related classes. It
12// wraps and owns SymbolManager, MemRegionManager, and BasicValueFactory.
13//
14//===----------------------------------------------------------------------===//
15
16#include "clang/Analysis/PathSensitive/ValueManager.h"
17
18using namespace clang;
19using namespace llvm;
20
21//===----------------------------------------------------------------------===//
22// Utility methods for constructing SVals.
23//===----------------------------------------------------------------------===//
24
25SVal ValueManager::makeZeroVal(QualType T) {
26 if (Loc::IsLocType(T))
27 return Loc::MakeNull(BasicVals);
28
29 if (T->isIntegerType())
30 return NonLoc::MakeVal(BasicVals, 0, T);
31
32 // FIXME: Handle floats.
33 // FIXME: Handle structs.
34 return UnknownVal();
35}
36
37SVal ValueManager::makeZeroArrayIndex() {
38 return nonloc::ConcreteInt(BasicVals.getZeroWithPtrWidth(false));
39}
40
41//===----------------------------------------------------------------------===//
42// Utility methods for constructing Non-Locs.
43//===----------------------------------------------------------------------===//
44
45NonLoc ValueManager::makeNonLoc(SymbolRef sym) {
46 return nonloc::SymbolVal(sym);
47}
48
49NonLoc ValueManager::makeIntVal(const APSInt& V) {
50 return nonloc::ConcreteInt(BasicVals.getValue(V));
51}
52
53NonLoc ValueManager::makeNonLoc(const SymExpr *lhs, BinaryOperator::Opcode op,
54 const APSInt& v, QualType T) {
55 // The Environment ensures we always get a persistent APSInt in
56 // BasicValueFactory, so we don't need to get the APSInt from
57 // BasicValueFactory again.
58 assert(!Loc::IsLocType(T));
59 return nonloc::SymExprVal(SymMgr.getSymIntExpr(lhs, op, v, T));
60}
61
62NonLoc ValueManager::makeNonLoc(const SymExpr *lhs, BinaryOperator::Opcode op,
63 const SymExpr *rhs, QualType T) {
64 assert(SymMgr.getType(lhs) == SymMgr.getType(rhs));
65 assert(!Loc::IsLocType(T));
66 return nonloc::SymExprVal(SymMgr.getSymSymExpr(lhs, op, rhs, T));
67}
68
69NonLoc ValueManager::makeTruthVal(bool b, QualType T) {
70 return nonloc::ConcreteInt(BasicVals.getTruthValue(b, T));
71}
72
73SVal ValueManager::getRegionValueSymbolVal(const MemRegion* R, QualType T) {
74 SymbolRef sym = SymMgr.getRegionValueSymbol(R, T);
75
76 if (const TypedRegion* TR = dyn_cast<TypedRegion>(R)) {
77 if (T.isNull())
78 T = TR->getValueType(SymMgr.getContext());
79
80 // If T is of function pointer type, create a CodeTextRegion wrapping a
81 // symbol.
82 if (T->isFunctionPointerType()) {
83 return Loc::MakeVal(MemMgr.getCodeTextRegion(sym, T));
84 }
85
86 if (Loc::IsLocType(T))
87 return Loc::MakeVal(MemMgr.getSymbolicRegion(sym));
88
89 // Only handle integers for now.
90 if (T->isIntegerType() && T->isScalarType())
91 return makeNonLoc(sym);
92 }
93
94 return UnknownVal();
95}
96
97SVal ValueManager::getConjuredSymbolVal(const Expr* E, unsigned Count) {
98 QualType T = E->getType();
99 SymbolRef sym = SymMgr.getConjuredSymbol(E, Count);
100
101 // If T is of function pointer type, create a CodeTextRegion wrapping a
102 // symbol.
103 if (T->isFunctionPointerType()) {
104 return Loc::MakeVal(MemMgr.getCodeTextRegion(sym, T));
105 }
106
107 if (Loc::IsLocType(T))
108 return Loc::MakeVal(MemMgr.getSymbolicRegion(sym));
109
110 if (T->isIntegerType() && T->isScalarType())
111 return makeNonLoc(sym);
112
113 return UnknownVal();
114}
115
116SVal ValueManager::getConjuredSymbolVal(const Expr* E, QualType T,
117 unsigned Count) {
118
119 SymbolRef sym = SymMgr.getConjuredSymbol(E, T, Count);
120
121 // If T is of function pointer type, create a CodeTextRegion wrapping a
122 // symbol.
123 if (T->isFunctionPointerType()) {
124 return Loc::MakeVal(MemMgr.getCodeTextRegion(sym, T));
125 }
126
127 if (Loc::IsLocType(T))
128 return Loc::MakeVal(MemMgr.getSymbolicRegion(sym));
129
130 if (T->isIntegerType() && T->isScalarType())
131 return makeNonLoc(sym);
132
133 return UnknownVal();
134}
135
136SVal ValueManager::getFunctionPointer(const FunctionDecl* FD) {
137 CodeTextRegion* R
138 = MemMgr.getCodeTextRegion(FD, Context.getPointerType(FD->getType()));
139 return loc::MemRegionVal(R);
140}
141