Add support for binding and retrieving VarRegions in flat store.

llvm-svn: 95529
This commit is contained in:
Zhongxing Xu 2010-02-08 05:40:07 +00:00
parent c126311d3e
commit 000a859f05
2 changed files with 71 additions and 12 deletions

View File

@ -11,6 +11,7 @@
#include "llvm/ADT/ImmutableIntervalMap.h"
using namespace clang;
using llvm::Interval;
// The actual store type.
typedef llvm::ImmutableIntervalMap<SVal> BindingVal;
@ -27,8 +28,8 @@ public:
RBFactory(mgr.getAllocator()),
BVFactory(mgr.getAllocator()) {}
SVal Retrieve(Store store, Loc loc, QualType T);
Store Bind(Store store, Loc loc, SVal val);
SVal Retrieve(Store store, Loc L, QualType T);
Store Bind(Store store, Loc L, SVal val);
Store Remove(Store St, Loc L);
Store BindCompoundLiteral(Store store, const CompoundLiteralExpr* cl,
const LocationContext *LC, SVal v);
@ -41,7 +42,9 @@ public:
return 0;
}
SVal getLValueVar(const VarDecl *VD, const LocationContext *LC);
SVal getLValueVar(const VarDecl *VD, const LocationContext *LC) {
return loc::MemRegionVal(MRMgr.getVarRegion(VD, LC));
}
SVal getLValueString(const StringLiteral* sl);
SVal getLValueIvar(const ObjCIvarDecl* decl, SVal base);
@ -65,6 +68,15 @@ public:
void print(Store store, llvm::raw_ostream& Out, const char* nl,
const char *sep);
void iterBindings(Store store, BindingsHandler& f);
private:
static RegionBindings getRegionBindings(Store store) {
return RegionBindings(static_cast<const RegionBindings::TreeTy*>(store));
}
Interval RegionToInterval(const MemRegion *R);
SVal RetrieveRegionWithNoBinding(const MemRegion *R, QualType T);
};
} // end anonymous namespace
@ -72,12 +84,42 @@ StoreManager *clang::CreateFlatStoreManager(GRStateManager &StMgr) {
return new FlatStoreManager(StMgr);
}
SVal FlatStoreManager::Retrieve(Store store, Loc loc, QualType T) {
return UnknownVal();
SVal FlatStoreManager::Retrieve(Store store, Loc L, QualType T) {
const MemRegion *R = cast<loc::MemRegionVal>(L).getRegion();
Interval I = RegionToInterval(R);
RegionBindings B = getRegionBindings(store);
const BindingVal *BV = B.lookup(R);
if (BV) {
const SVal *V = BVFactory.Lookup(*BV, I);
if (V)
return *V;
else
return RetrieveRegionWithNoBinding(R, T);
}
return RetrieveRegionWithNoBinding(R, T);
}
Store FlatStoreManager::Bind(Store store, Loc loc, SVal val) {
return store;
SVal FlatStoreManager::RetrieveRegionWithNoBinding(const MemRegion *R,
QualType T) {
if (R->hasStackNonParametersStorage())
return UndefinedVal();
else
return ValMgr.getRegionValueSymbolVal(R, T);
}
Store FlatStoreManager::Bind(Store store, Loc L, SVal val) {
const MemRegion *R = cast<loc::MemRegionVal>(L).getRegion();
RegionBindings B = getRegionBindings(store);
const BindingVal *V = B.lookup(R);
BindingVal BV = BVFactory.GetEmptyMap();
if (V)
BV = *V;
Interval I = RegionToInterval(R);
BV = BVFactory.Add(BV, I, val);
B = RBFactory.Add(B, R, BV);
return B.getRoot();
}
Store FlatStoreManager::Remove(Store store, Loc L) {
@ -91,11 +133,6 @@ Store FlatStoreManager::BindCompoundLiteral(Store store,
return store;
}
SVal FlatStoreManager::getLValueVar(const VarDecl *VD,
const LocationContext *LC) {
return UnknownVal();
}
SVal FlatStoreManager::getLValueString(const StringLiteral* sl) {
return UnknownVal();
}
@ -138,3 +175,15 @@ void FlatStoreManager::print(Store store, llvm::raw_ostream& Out,
void FlatStoreManager::iterBindings(Store store, BindingsHandler& f) {
}
Interval FlatStoreManager::RegionToInterval(const MemRegion *R) {
switch (R->getKind()) {
case MemRegion::VarRegionKind: {
QualType T = cast<VarRegion>(R)->getValueType(StateMgr.getContext());
uint64_t Size = StateMgr.getContext().getTypeSize(T);
return Interval(0, Size-1);
}
default:
assert(0 && "Region kind unhandled.");
}
}

View File

@ -0,0 +1,10 @@
// RUN: %clang_cc1 -analyze -analyzer-check-objc-mem -analyzer-store=flat -verify %s
void f1() {
int x;
int *p;
x = 1;
p = 0;
if (x != 1)
*p = 1; // no-warning
}