]> granicus.if.org Git - clang/commitdiff
Added basic psuedoconstant checking in IdempotentOperationChecker and fixed some...
authorTom Care <tom.care@uqconnect.edu.au>
Mon, 16 Aug 2010 21:43:52 +0000 (21:43 +0000)
committerTom Care <tom.care@uqconnect.edu.au>
Mon, 16 Aug 2010 21:43:52 +0000 (21:43 +0000)
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@111190 91177308-0d34-0410-b5e6-96231b3b80d8

lib/Checker/IdempotentOperationChecker.cpp
test/Analysis/idempotent-operations.c

index 74f4a62ccce63f3c9a4e4e66ec35c3ec0953198b..4f2810ea94ef188e8d1b5f0af3012b40e1eb6748 100644 (file)
@@ -81,6 +81,7 @@ class IdempotentOperationChecker
                                           const CFGBlock *CB,
                                           const GRCoreEngine &CE);
     static bool CanVary(const Expr *Ex, ASTContext &Ctx);
+    static bool isPseudoConstant(const DeclRefExpr *D);
 
     // Hash table
     typedef llvm::DenseMap<const BinaryOperator *,
@@ -530,8 +531,7 @@ bool IdempotentOperationChecker::CanVary(const Expr *Ex, ASTContext &Ctx) {
     return SE->getTypeOfArgument()->isVariableArrayType();
   }
   case Stmt::DeclRefExprClass:
-    //    return !IsPseudoConstant(cast<DeclRefExpr>(Ex));
-    return true;
+    return !isPseudoConstant(cast<DeclRefExpr>(Ex));
 
   // The next cases require recursion for subexpressions
   case Stmt::BinaryOperatorClass: {
@@ -555,3 +555,17 @@ bool IdempotentOperationChecker::CanVary(const Expr *Ex, ASTContext &Ctx) {
   }
 }
 
+// Returns true if a DeclRefExpr behaves like a constant.
+bool IdempotentOperationChecker::isPseudoConstant(const DeclRefExpr *DR) {
+  // Check for an enum
+  if (isa<EnumConstantDecl>(DR->getDecl()))
+    return true;
+
+  // Check for a static variable
+  // FIXME: Analysis should model static vars
+  if (const VarDecl *VD = dyn_cast<VarDecl>(DR->getDecl()))
+    if (VD->isStaticLocal())
+      return true;
+
+  return false;
+}
index 23401e8cdb805027b582be0f1d500ed5ca649b5a..a54a3aae081aa87a946ff3573dec61b35ac8bb70 100644 (file)
@@ -74,12 +74,14 @@ void bailout() {
 // False positive tests
 
 unsigned false1() {
-  return (5 - 2 - 3); // no-warning
+  int a = 10;
+  return a * (5 - 2 - 3); // no-warning
 }
 
 enum testenum { enum1 = 0, enum2 };
 unsigned false2() {
-  return enum1; // no-warning
+  int a = 1234;
+  return enum1 + a; // no-warning
 }
 
 extern unsigned foo();