forked from OSchip/llvm-project
				
			
		
			
				
	
	
		
			304 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			304 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			C++
		
	
	
	
| // RUN: %clang_cc1 -analyze -analyzer-checker=core,cplusplus.NewDelete -std=c++11 -fblocks -verify %s
 | |
| // RUN: %clang_cc1 -analyze -analyzer-checker=core,alpha.cplusplus.NewDeleteLeaks -DLEAKS -std=c++11 -fblocks -verify %s
 | |
| #include "Inputs/system-header-simulator-cxx.h"
 | |
| 
 | |
| typedef __typeof__(sizeof(int)) size_t;
 | |
| extern "C" void *malloc(size_t);
 | |
| int *global;
 | |
| 
 | |
| //------------------
 | |
| // check for leaks
 | |
| //------------------
 | |
| 
 | |
| //----- Standard non-placement operators
 | |
| void testGlobalOpNew() {
 | |
|   void *p = operator new(0);
 | |
| }
 | |
| #ifdef LEAKS
 | |
| // expected-warning@-2{{Potential leak of memory pointed to by 'p'}}
 | |
| #endif
 | |
| 
 | |
| void testGlobalOpNewArray() {
 | |
|   void *p = operator new[](0);
 | |
| }
 | |
| #ifdef LEAKS
 | |
| // expected-warning@-2{{Potential leak of memory pointed to by 'p'}}
 | |
| #endif
 | |
| 
 | |
| void testGlobalNewExpr() {
 | |
|   int *p = new int;
 | |
| }
 | |
| #ifdef LEAKS
 | |
| // expected-warning@-2{{Potential leak of memory pointed to by 'p'}}
 | |
| #endif
 | |
| 
 | |
| void testGlobalNewExprArray() {
 | |
|   int *p = new int[0];
 | |
| }
 | |
| #ifdef LEAKS
 | |
| // expected-warning@-2{{Potential leak of memory pointed to by 'p'}}
 | |
| #endif
 | |
| 
 | |
| //----- Standard nothrow placement operators
 | |
| void testGlobalNoThrowPlacementOpNewBeforeOverload() {
 | |
|   void *p = operator new(0, std::nothrow);
 | |
| }
 | |
| #ifdef LEAKS
 | |
| // expected-warning@-2{{Potential leak of memory pointed to by 'p'}}
 | |
| #endif
 | |
| 
 | |
| void testGlobalNoThrowPlacementExprNewBeforeOverload() {
 | |
|   int *p = new(std::nothrow) int;
 | |
| }
 | |
| #ifdef LEAKS
 | |
| // expected-warning@-2{{Potential leak of memory pointed to by 'p'}}
 | |
| #endif
 | |
| 
 | |
| //----- Standard pointer placement operators
 | |
| void testGlobalPointerPlacementNew() {
 | |
|   int i;
 | |
| 
 | |
|   void *p1 = operator new(0, &i); // no warn
 | |
| 
 | |
|   void *p2 = operator new[](0, &i); // no warn
 | |
| 
 | |
|   int *p3 = new(&i) int; // no warn
 | |
| 
 | |
|   int *p4 = new(&i) int[0]; // no warn
 | |
| }
 | |
| 
 | |
| //----- Other cases
 | |
| void testNewMemoryIsInHeap() {
 | |
|   int *p = new int;
 | |
|   if (global != p) // condition is always true as 'p' wraps a heap region that 
 | |
|                    // is different from a region wrapped by 'global'
 | |
|     global = p; // pointer escapes
 | |
| }
 | |
| 
 | |
| struct PtrWrapper {
 | |
|   int *x;
 | |
| 
 | |
|   PtrWrapper(int *input) : x(input) {}
 | |
| };
 | |
| 
 | |
| void testNewInvalidationPlacement(PtrWrapper *w) {
 | |
|   // Ensure that we don't consider this a leak.
 | |
|   new (w) PtrWrapper(new int); // no warn
 | |
| }
 | |
| 
 | |
| //---------------
 | |
| // other checks
 | |
| //---------------
 | |
| 
 | |
| class SomeClass {
 | |
| public:
 | |
|   void f(int *p);
 | |
| };
 | |
| 
 | |
| void f(int *p1, int *p2 = 0, int *p3 = 0);
 | |
| void g(SomeClass &c, ...);
 | |
| 
 | |
| void testUseFirstArgAfterDelete() {
 | |
|   int *p = new int;
 | |
|   delete p;
 | |
|   f(p); // expected-warning{{Use of memory after it is freed}}
 | |
| }
 | |
| 
 | |
| void testUseMiddleArgAfterDelete(int *p) {
 | |
|   delete p;
 | |
|   f(0, p); // expected-warning{{Use of memory after it is freed}}
 | |
| }
 | |
| 
 | |
| void testUseLastArgAfterDelete(int *p) {
 | |
|   delete p;
 | |
|   f(0, 0, p); // expected-warning{{Use of memory after it is freed}}
 | |
| }
 | |
| 
 | |
| void testUseSeveralArgsAfterDelete(int *p) {
 | |
|   delete p;
 | |
|   f(p, p, p); // expected-warning{{Use of memory after it is freed}}
 | |
| }
 | |
| 
 | |
| void testUseRefArgAfterDelete(SomeClass &c) {
 | |
|   delete &c;
 | |
|   g(c); // expected-warning{{Use of memory after it is freed}}
 | |
| }
 | |
| 
 | |
| void testVariadicArgAfterDelete() {
 | |
|   SomeClass c;
 | |
|   int *p = new int;
 | |
|   delete p;
 | |
|   g(c, 0, p); // expected-warning{{Use of memory after it is freed}}
 | |
| }
 | |
| 
 | |
| void testUseMethodArgAfterDelete(int *p) {
 | |
|   SomeClass *c = new SomeClass;
 | |
|   delete p;
 | |
|   c->f(p); // expected-warning{{Use of memory after it is freed}}
 | |
| }
 | |
| 
 | |
| void testUseThisAfterDelete() {
 | |
|   SomeClass *c = new SomeClass;
 | |
|   delete c;
 | |
|   c->f(0); // expected-warning{{Use of memory after it is freed}}
 | |
| }
 | |
| 
 | |
| void testDeleteAlloca() {
 | |
|   int *p = (int *)__builtin_alloca(sizeof(int));
 | |
|   delete p; // expected-warning{{Memory allocated by alloca() should not be deallocated}}
 | |
| }
 | |
| 
 | |
| void testDoubleDelete() {
 | |
|   int *p = new int;
 | |
|   delete p;
 | |
|   delete p; // expected-warning{{Attempt to free released memory}}
 | |
| }
 | |
| 
 | |
| void testExprDeleteArg() {
 | |
|   int i;
 | |
|   delete &i; // expected-warning{{Argument to 'delete' is the address of the local variable 'i', which is not memory allocated by 'new'}}
 | |
| }
 | |
| 
 | |
| void testExprDeleteArrArg() {
 | |
|   int i;
 | |
|   delete[] &i; // expected-warning{{Argument to 'delete[]' is the address of the local variable 'i', which is not memory allocated by 'new[]'}}
 | |
| }
 | |
| 
 | |
| void testAllocDeallocNames() {
 | |
|   int *p = new(std::nothrow) int[1];
 | |
|   delete[] (++p); // expected-warning{{Argument to 'delete[]' is offset by 4 bytes from the start of memory allocated by 'new[]'}}
 | |
| }
 | |
| 
 | |
| //--------------------------------
 | |
| // Test escape of newed const pointer. Note, a const pointer can be deleted.
 | |
| //--------------------------------
 | |
| struct StWithConstPtr {
 | |
|   const int *memp;
 | |
| };
 | |
| void escape(const int &x);
 | |
| void escapeStruct(const StWithConstPtr &x);
 | |
| void escapePtr(const StWithConstPtr *x);
 | |
| void escapeVoidPtr(const void *x);
 | |
| 
 | |
| void testConstEscape() {
 | |
|   int *p = new int(1);
 | |
|   escape(*p);
 | |
| } // no-warning
 | |
| 
 | |
| void testConstEscapeStruct() {
 | |
|   StWithConstPtr *St = new StWithConstPtr();
 | |
|   escapeStruct(*St);
 | |
| } // no-warning
 | |
| 
 | |
| void testConstEscapeStructPtr() {
 | |
|   StWithConstPtr *St = new StWithConstPtr();
 | |
|   escapePtr(St);
 | |
| } // no-warning
 | |
| 
 | |
| void testConstEscapeMember() {
 | |
|   StWithConstPtr St;
 | |
|   St.memp = new int(2);
 | |
|   escapeVoidPtr(St.memp);
 | |
| } // no-warning
 | |
| 
 | |
| void testConstEscapePlacementNew() {
 | |
|   int *x = (int *)malloc(sizeof(int));
 | |
|   void *y = new (x) int;
 | |
|   escapeVoidPtr(y);
 | |
| } // no-warning
 | |
| 
 | |
| 
 | |
| namespace reference_count {
 | |
|   class control_block {
 | |
|     unsigned count;
 | |
|   public:
 | |
|     control_block() : count(0) {}
 | |
|     void retain() { ++count; }
 | |
|     int release() { return --count; }
 | |
|   };
 | |
| 
 | |
|   template <typename T>
 | |
|   class shared_ptr {
 | |
|     T *p;
 | |
|     control_block *control;
 | |
| 
 | |
|   public:
 | |
|     shared_ptr() : p(0), control(0) {}
 | |
|     explicit shared_ptr(T *p) : p(p), control(new control_block) {
 | |
|       control->retain();
 | |
|     }
 | |
|     shared_ptr(shared_ptr &other) : p(other.p), control(other.control) {
 | |
|       if (control)
 | |
|           control->retain();
 | |
|     }
 | |
|     ~shared_ptr() {
 | |
|       if (control && control->release() == 0) {
 | |
|         delete p;
 | |
|         delete control;
 | |
|       }
 | |
|     };
 | |
| 
 | |
|     T &operator *() {
 | |
|       return *p;
 | |
|     };
 | |
| 
 | |
|     void swap(shared_ptr &other) {
 | |
|       T *tmp = p;
 | |
|       p = other.p;
 | |
|       other.p = tmp;
 | |
| 
 | |
|       control_block *ctrlTmp = control;
 | |
|       control = other.control;
 | |
|       other.control = ctrlTmp;
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   void testSingle() {
 | |
|     shared_ptr<int> a(new int);
 | |
|     *a = 1;
 | |
|   }
 | |
| 
 | |
|   void testDouble() {
 | |
|     shared_ptr<int> a(new int);
 | |
|     shared_ptr<int> b = a;
 | |
|     *a = 1;
 | |
|   }
 | |
| 
 | |
|   void testInvalidated() {
 | |
|     shared_ptr<int> a(new int);
 | |
|     shared_ptr<int> b = a;
 | |
|     *a = 1;
 | |
| 
 | |
|     extern void use(shared_ptr<int> &);
 | |
|     use(b);
 | |
|   }
 | |
| 
 | |
|   void testNestedScope() {
 | |
|     shared_ptr<int> a(new int);
 | |
|     {
 | |
|       shared_ptr<int> b = a;
 | |
|     }
 | |
|     *a = 1;
 | |
|   }
 | |
| 
 | |
|   void testSwap() {
 | |
|     shared_ptr<int> a(new int);
 | |
|     shared_ptr<int> b;
 | |
|     shared_ptr<int> c = a;
 | |
|     shared_ptr<int>(c).swap(b);
 | |
|   }
 | |
| 
 | |
|   void testUseAfterFree() {
 | |
|     int *p = new int;
 | |
|     {
 | |
|       shared_ptr<int> a(p);
 | |
|       shared_ptr<int> b = a;
 | |
|     }
 | |
| 
 | |
|     // FIXME: We should get a warning here, but we don't because we've
 | |
|     // conservatively modeled ~shared_ptr.
 | |
|     *p = 1;
 | |
|   }
 | |
| }
 | |
| 
 |