From: Ted Kremenek Date: Tue, 24 Feb 2009 02:23:11 +0000 (+0000) Subject: Fix : Add basic transfer function support in the static X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=265a305997c63a28d87ddd370958db083f98bc1a;p=clang Fix : Add basic transfer function support in the static analyzer for array subscript expressions involving bases that are vectors. This solution is probably a hack: it gets the lvalue of the vector instead of an rvalue like all other types. This should be reviewed (big FIXME in GRExprEngine). git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@65366 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/lib/Analysis/GRExprEngine.cpp b/lib/Analysis/GRExprEngine.cpp index 60bef6eb71..d34e8a3ed3 100644 --- a/lib/Analysis/GRExprEngine.cpp +++ b/lib/Analysis/GRExprEngine.cpp @@ -850,7 +850,16 @@ void GRExprEngine::VisitArraySubscriptExpr(ArraySubscriptExpr* A, NodeTy* Pred, Expr* Base = A->getBase()->IgnoreParens(); Expr* Idx = A->getIdx()->IgnoreParens(); NodeSet Tmp; - Visit(Base, Pred, Tmp); // Get Base's rvalue, which should be an LocVal. + + if (Base->getType()->isVectorType()) { + // For vector types get its lvalue. + // FIXME: This may not be correct. Is the rvalue of a vector its location? + // In fact, I think this is just a hack. We need to get the right + // semantics. + VisitLValue(Base, Pred, Tmp); + } + else + Visit(Base, Pred, Tmp); // Get Base's rvalue, which should be an LocVal. for (NodeSet::iterator I1=Tmp.begin(), E1=Tmp.end(); I1!=E1; ++I1) { NodeSet Tmp2; diff --git a/lib/Analysis/RegionStore.cpp b/lib/Analysis/RegionStore.cpp index e2a1d36182..c61094c20c 100644 --- a/lib/Analysis/RegionStore.cpp +++ b/lib/Analysis/RegionStore.cpp @@ -591,9 +591,14 @@ SVal RegionStoreManager::Retrieve(const GRState* St, Loc L, QualType T) { // // Such funny addressing will occur due to layering of regions. - if (const TypedRegion* TR = dyn_cast(R)) - if (TR->getRValueType(getContext())->isStructureType()) + if (const TypedRegion* TR = dyn_cast(R)) { + QualType T =TR->getRValueType(getContext()); + if (T->isStructureType()) return RetrieveStruct(St, TR); + // FIXME: handle Vector types. + if (T->isVectorType()) + return UnknownVal(); + } RegionBindingsTy B = GetRegionBindings(St->getStore()); RegionBindingsTy::data_type* V = B.lookup(R); @@ -636,6 +641,7 @@ SVal RegionStoreManager::Retrieve(const GRState* St, Loc L, QualType T) { return loc::MemRegionVal(getSelfRegion(0)); } + if (MRMgr.onStack(R) || MRMgr.onHeap(R)) { // All stack variables are considered to have undefined values // upon creation. All heap allocated blocks are considered to diff --git a/test/Analysis/misc-ps.m b/test/Analysis/misc-ps.m index f7e82be683..02a7d82020 100644 --- a/test/Analysis/misc-ps.m +++ b/test/Analysis/misc-ps.m @@ -115,4 +115,14 @@ int pr_3543(void) { ({}); } +// +// This test case test the use of a vector type within an array subscript +// expression. +typedef long long __a64vector __attribute__((__vector_size__(8))); +typedef long long __a128vector __attribute__((__vector_size__(16))); +static inline __a64vector __attribute__((__always_inline__, __nodebug__)) +my_test_mm_movepi64_pi64(__a128vector a) { + return (__a64vector)a[0]; +} +