From: Ted Kremenek Date: Thu, 19 Jun 2008 17:55:38 +0000 (+0000) Subject: Introduce initial transfer function support for __imag__ and __real__. We don't X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=b8e26e63d9dcc09351d75677721c6c9ff7045b54;p=clang Introduce initial transfer function support for __imag__ and __real__. We don't have complex RValues yet, so this logic is only fully implemented when __imag__ and __real__ are used on non-complex types. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@52501 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/lib/Analysis/GRExprEngine.cpp b/lib/Analysis/GRExprEngine.cpp index 2580172d53..e8c1ec57a9 100644 --- a/lib/Analysis/GRExprEngine.cpp +++ b/lib/Analysis/GRExprEngine.cpp @@ -1596,7 +1596,7 @@ void GRExprEngine::VisitUnaryOperator(UnaryOperator* U, NodeTy* Pred, default: break; - + case UnaryOperator::Deref: { Expr* Ex = U->getSubExpr()->IgnoreParens(); @@ -1616,10 +1616,57 @@ void GRExprEngine::VisitUnaryOperator(UnaryOperator* U, NodeTy* Pred, return; } - + case UnaryOperator::Real: { + + Expr* Ex = U->getSubExpr()->IgnoreParens(); + NodeSet Tmp; + Visit(Ex, Pred, Tmp); + + for (NodeSet::iterator I=Tmp.begin(), E=Tmp.end(); I!=E; ++I) { + + // FIXME: We don't have complex RValues yet. + if (Ex->getType()->isAnyComplexType()) { + // Just report "Unknown." + Dst.Add(*I); + continue; + } + + // For all other types, UnaryOperator::Real is an identity operation. + assert (U->getType() == Ex->getType()); + ValueState* St = GetState(*I); + MakeNode(Dst, U, *I, SetRVal(St, U, GetRVal(St, Ex))); + } + + return; + } + + case UnaryOperator::Imag: { + + Expr* Ex = U->getSubExpr()->IgnoreParens(); + NodeSet Tmp; + Visit(Ex, Pred, Tmp); + + for (NodeSet::iterator I=Tmp.begin(), E=Tmp.end(); I!=E; ++I) { + // FIXME: We don't have complex RValues yet. + if (Ex->getType()->isAnyComplexType()) { + // Just report "Unknown." + Dst.Add(*I); + continue; + } + + // For all other types, UnaryOperator::Float returns 0. + assert (Ex->getType()->isIntegerType()); + ValueState* St = GetState(*I); + RVal X = NonLVal::MakeVal(BasicVals, 0, Ex->getType()); + MakeNode(Dst, U, *I, SetRVal(St, U, X)); + } + + return; + } + + // FIXME: Just report "Unknown" for OffsetOf. case UnaryOperator::OffsetOf: - // FIXME: Just report "Unknown" known for OffsetOf. Dst.Add(Pred); return; diff --git a/test/Analysis/complex.c b/test/Analysis/complex.c new file mode 100644 index 0000000000..7561293171 --- /dev/null +++ b/test/Analysis/complex.c @@ -0,0 +1,17 @@ +// RUN: clang -checker-simple -verify %s + +#include + +int f1(int * p) { + + // This branch should be infeasible + // because __imag__ p is 0. + if (!p && __imag__ (intptr_t) p) + *p = 1; // no-warning + + // If p != 0 then this branch is feasible; otherwise it is not. + if (__real__ (intptr_t) p) + *p = 1; // no-warning + + *p = 2; // expected-warning{{Dereference of null pointer}} +}