Home | History | Annotate | Download | only in Analysis
      1 // RUN: %clang_cc1 -analyze -analyzer-checker=core,unix.Malloc,debug.ExprInspection -analyzer-store region -std=c++11 -verify %s
      2 #include "Inputs/system-header-simulator-cxx.h"
      3 
      4 void clang_analyzer_eval(bool);
      5 
      6 typedef __typeof__(sizeof(int)) size_t;
      7 extern "C" void *malloc(size_t);
      8 extern "C" void free(void *);
      9 
     10 int someGlobal;
     11 
     12 class SomeClass {
     13 public:
     14   void f(int *p);
     15 };
     16 
     17 void testImplicitlyDeclaredGlobalNew() {
     18   if (someGlobal != 0)
     19     return;
     20 
     21   // This used to crash because the global operator new is being implicitly
     22   // declared and it does not have a valid source location. (PR13090)
     23   void *x = ::operator new(0);
     24   ::operator delete(x);
     25 
     26   // Check that the new/delete did not invalidate someGlobal;
     27   clang_analyzer_eval(someGlobal == 0); // expected-warning{{TRUE}}
     28 }
     29 
     30 void *testPlacementNew() {
     31   int *x = (int *)malloc(sizeof(int));
     32   *x = 1;
     33   clang_analyzer_eval(*x == 1); // expected-warning{{TRUE}};
     34 
     35   void *y = new (x) int;
     36   clang_analyzer_eval(x == y); // expected-warning{{TRUE}};
     37   clang_analyzer_eval(*x == 1); // expected-warning{{UNKNOWN}};
     38 
     39   return y;
     40 }
     41 
     42 void *operator new(size_t, size_t, int *);
     43 void *testCustomNew() {
     44   int x[1] = {1};
     45   clang_analyzer_eval(*x == 1); // expected-warning{{TRUE}};
     46 
     47   void *y = new (0, x) int;
     48   clang_analyzer_eval(*x == 1); // expected-warning{{UNKNOWN}};
     49 
     50   return y; // no-warning
     51 }
     52 
     53 void *operator new(size_t, void *, void *);
     54 void *testCustomNewMalloc() {
     55   int *x = (int *)malloc(sizeof(int));
     56 
     57   // Should be no-warning (the custom allocator could have freed x).
     58   void *y = new (0, x) int; // no-warning
     59 
     60   return y;
     61 }
     62 
     63 void testScalarInitialization() {
     64   int *n = new int(3);
     65   clang_analyzer_eval(*n == 3); // expected-warning{{TRUE}}
     66 
     67   new (n) int();
     68   clang_analyzer_eval(*n == 0); // expected-warning{{TRUE}}
     69 
     70   new (n) int{3};
     71   clang_analyzer_eval(*n == 3); // expected-warning{{TRUE}}
     72 
     73   new (n) int{};
     74   clang_analyzer_eval(*n == 0); // expected-warning{{TRUE}}
     75 }
     76 
     77 struct PtrWrapper {
     78   int *x;
     79 
     80   PtrWrapper(int *input) : x(input) {}
     81 };
     82 
     83 PtrWrapper *testNewInvalidation() {
     84   // Ensure that we don't consider this a leak.
     85   return new PtrWrapper(static_cast<int *>(malloc(4))); // no-warning
     86 }
     87 
     88 void testNewInvalidationPlacement(PtrWrapper *w) {
     89   // Ensure that we don't consider this a leak.
     90   new (w) PtrWrapper(static_cast<int *>(malloc(4))); // no-warning
     91 }
     92 
     93 int **testNewInvalidationScalar() {
     94   // Ensure that we don't consider this a leak.
     95   return new (int *)(static_cast<int *>(malloc(4))); // no-warning
     96 }
     97 
     98 void testNewInvalidationScalarPlacement(int **p) {
     99   // Ensure that we don't consider this a leak.
    100   new (p) (int *)(static_cast<int *>(malloc(4))); // no-warning
    101 }
    102 
    103 void testCacheOut(PtrWrapper w) {
    104   extern bool coin();
    105   if (coin())
    106     w.x = 0;
    107   new (&w.x) (int*)(0); // we cache out here; don't crash
    108 }
    109 
    110 void testUseAfter(int *p) {
    111   SomeClass *c = new SomeClass;
    112   free(p);
    113   c->f(p); // expected-warning{{Use of memory after it is freed}}
    114   delete c;
    115 }
    116 
    117 //--------------------------------------------------------------------
    118 // Check for intersection with other checkers from MallocChecker.cpp
    119 // bounded with unix.Malloc
    120 //--------------------------------------------------------------------
    121 
    122 // new/delete oparators are subjects of cplusplus.NewDelete.
    123 void testNewDeleteNoWarn() {
    124   int i;
    125   delete &i; // no-warning
    126 
    127   int *p1 = new int;
    128   delete ++p1; // no-warning
    129 
    130   int *p2 = new int;
    131   delete p2;
    132   delete p2; // no-warning
    133 
    134   int *p3 = new int; // no-warning
    135 }
    136 
    137 // unix.Malloc does not know about operators new/delete.
    138 void testDeleteMallocked() {
    139   int *x = (int *)malloc(sizeof(int));
    140   delete x; // FIXME: Shoud detect pointer escape and keep silent after 'delete' is modeled properly.
    141 } // expected-warning{{Potential leak of memory pointed to by 'x'}}
    142 
    143 void testDeleteOpAfterFree() {
    144   int *p = (int *)malloc(sizeof(int));
    145   free(p);
    146   operator delete(p); // expected-warning{{Use of memory after it is freed}}
    147 }
    148 
    149 void testDeleteAfterFree() {
    150   int *p = (int *)malloc(sizeof(int));
    151   free(p);
    152   delete p; // expected-warning{{Use of memory after it is freed}}
    153 }
    154 
    155 void testStandardPlacementNewAfterFree() {
    156   int *p = (int *)malloc(sizeof(int));
    157   free(p);
    158   p = new(p) int; // expected-warning{{Use of memory after it is freed}}
    159 }
    160 
    161 void testCustomPlacementNewAfterFree() {
    162   int *p = (int *)malloc(sizeof(int));
    163   free(p);
    164   p = new(0, p) int; // expected-warning{{Use of memory after it is freed}}
    165 }
    166 
    167 void testUsingThisAfterDelete() {
    168   SomeClass *c = new SomeClass;
    169   delete c;
    170   c->f(0); // no-warning
    171 }
    172 
    173 void testAggregateNew() {
    174   struct Point { int x, y; };
    175   new Point{1, 2}; // no crash
    176 
    177   Point p;
    178   new (&p) Point{1, 2}; // no crash
    179   clang_analyzer_eval(p.x == 1); // expected-warning{{TRUE}}
    180   clang_analyzer_eval(p.y == 2); // expected-warning{{TRUE}}
    181 }
    182 
    183 //--------------------------------
    184 // Incorrectly-modelled behavior
    185 //--------------------------------
    186 
    187 int testNoInitialization() {
    188   int *n = new int;
    189 
    190   // Should warn that *n is uninitialized.
    191   if (*n) { // no-warning
    192     delete n;
    193     return 0;
    194   }
    195   delete n;
    196   return 1;
    197 }
    198 
    199 int testNoInitializationPlacement() {
    200   int n;
    201   new (&n) int;
    202 
    203   // Should warn that n is uninitialized.
    204   if (n) { // no-warning
    205     return 0;
    206   }
    207   return 1;
    208 }
    209 
    210 // Test modelling destructor call on call to delete
    211 class IntPair{
    212 public:
    213   int x;
    214   int y;
    215   IntPair() {};
    216   ~IntPair() {x = x/y;}; //expected-warning {{Division by zero}}
    217 };
    218 
    219 void testCallToDestructor() {
    220   IntPair *b = new IntPair();
    221   b->x = 1;
    222   b->y = 0;
    223   delete b; // This results in divide by zero in destructor
    224 }
    225 
    226 // Test Deleting a value that's passed as an argument.
    227 class DerefClass{
    228 public:
    229   int *x;
    230   DerefClass() {};
    231   ~DerefClass() {*x = 1;}; //expected-warning {{Dereference of null pointer (loaded from field 'x')}}
    232 };
    233 
    234 void testDestCall(DerefClass *arg) {
    235   delete arg;
    236 }
    237 
    238 void test_delete_dtor_Arg() {
    239   DerefClass *pair = new DerefClass();
    240   pair->x = 0;
    241   testDestCall(pair);
    242 }
    243 
    244 //Deleting the address of a local variable, null pointer
    245 void abort(void) __attribute__((noreturn));
    246 
    247 class NoReturnDtor {
    248 public:
    249   NoReturnDtor() {}
    250   ~NoReturnDtor() {abort();}
    251 };
    252 
    253 void test_delete_dtor_LocalVar() {
    254   NoReturnDtor test;
    255   delete &test; // no warn or crash
    256 }
    257 
    258 class DerivedNoReturn:public NoReturnDtor {
    259 public:
    260   DerivedNoReturn() {};
    261   ~DerivedNoReturn() {};
    262 };
    263 
    264 void testNullDtorDerived() {
    265   DerivedNoReturn *p = new DerivedNoReturn();
    266   delete p; // Calls the base destructor which aborts, checked below
    267   clang_analyzer_eval(true); // no warn
    268 }
    269 
    270 //Deleting a non-class pointer should not crash/warn
    271 void test_var_delete() {
    272   int *v = new int;
    273   delete v;  // no crash/warn
    274   clang_analyzer_eval(true); // expected-warning{{TRUE}}
    275 }
    276 
    277 void testDeleteNull() {
    278   NoReturnDtor *foo = 0;
    279   delete foo; // should not call destructor, checked below
    280   clang_analyzer_eval(true); // expected-warning{{TRUE}}
    281 }
    282 
    283 void testNullAssigneddtor() {
    284   NoReturnDtor *p = 0;
    285   NoReturnDtor *s = p;
    286   delete s; // should not call destructor, checked below
    287   clang_analyzer_eval(true); // expected-warning{{TRUE}}
    288 }
    289 
    290 void deleteArg(NoReturnDtor *test) {
    291   delete test;
    292 }
    293 
    294 void testNulldtorArg() {
    295   NoReturnDtor *p = 0;
    296   deleteArg(p);
    297   clang_analyzer_eval(true); // expected-warning{{TRUE}}
    298 }
    299 
    300 void testDeleteUnknown(NoReturnDtor *foo) {
    301   delete foo; // should assume non-null and call noreturn destructor
    302   clang_analyzer_eval(true); // no-warning
    303 }
    304 
    305 void testArrayNull() {
    306   NoReturnDtor *fooArray = 0;
    307   delete[] fooArray; // should not call destructor, checked below
    308   clang_analyzer_eval(true); // expected-warning{{TRUE}}
    309 }
    310 
    311 void testArrayDestr() {
    312   NoReturnDtor *p = new NoReturnDtor[2];
    313   delete[] p; // Calls the base destructor which aborts, checked below
    314   //TODO: clang_analyzer_eval should not be called
    315   clang_analyzer_eval(true); // expected-warning{{TRUE}}
    316 }
    317 
    318 // Invalidate Region even in case of default destructor
    319 class InvalidateDestTest {
    320 public:
    321   int x;
    322   int *y;
    323   ~InvalidateDestTest();
    324 };
    325 
    326 int test_member_invalidation() {
    327 
    328   //test invalidation of member variable
    329   InvalidateDestTest *test = new InvalidateDestTest();
    330   test->x = 5;
    331   int *k = &(test->x);
    332   clang_analyzer_eval(*k == 5); // expected-warning{{TRUE}}
    333   delete test;
    334   clang_analyzer_eval(*k == 5); // expected-warning{{UNKNOWN}}
    335 
    336   //test invalidation of member pointer
    337   int localVar = 5;
    338   test = new InvalidateDestTest();
    339   test->y = &localVar;
    340   delete test;
    341   clang_analyzer_eval(localVar == 5); // expected-warning{{UNKNOWN}}
    342 
    343   // Test aray elements are invalidated.
    344   int Var1 = 5;
    345   int Var2 = 5;
    346   InvalidateDestTest *a = new InvalidateDestTest[2];
    347   a[0].y = &Var1;
    348   a[1].y = &Var2;
    349   delete[] a;
    350   clang_analyzer_eval(Var1 == 5); // expected-warning{{UNKNOWN}}
    351   clang_analyzer_eval(Var2 == 5); // expected-warning{{UNKNOWN}}
    352   return 0;
    353 }
    354