forked from OSchip/llvm-project
				
			
		
			
				
	
	
		
			136 lines
		
	
	
		
			3.0 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			136 lines
		
	
	
		
			3.0 KiB
		
	
	
	
		
			C
		
	
	
	
| // RUN: %clang_analyze_cc1 -analyzer-checker=unix.Malloc -analyzer-inline-max-stack-depth=5 -verify %s
 | |
| 
 | |
| #include "Inputs/system-header-simulator.h"
 | |
| 
 | |
| void *malloc(size_t);
 | |
| void *valloc(size_t);
 | |
| void free(void *);
 | |
| void *realloc(void *ptr, size_t size);
 | |
| void *reallocf(void *ptr, size_t size);
 | |
| void *calloc(size_t nmemb, size_t size);
 | |
| 
 | |
| void exit(int) __attribute__ ((__noreturn__));
 | |
| void *memcpy(void * restrict s1, const void * restrict s2, size_t n);
 | |
| size_t strlen(const char *);
 | |
| 
 | |
| static void my_malloc1(void **d, size_t size) {
 | |
|   *d = malloc(size);
 | |
| }
 | |
| 
 | |
| static void *my_malloc2(int elevel, size_t size) {
 | |
|   void     *data;
 | |
|   data = malloc(size);
 | |
|   if (data == 0)
 | |
|     exit(0);
 | |
|   return data;
 | |
| }
 | |
| 
 | |
| static void my_free1(void *p) {
 | |
|   free(p);
 | |
| }
 | |
| 
 | |
| static void test1() {
 | |
|   void *data = 0;
 | |
|   my_malloc1(&data, 4);
 | |
| } // expected-warning {{Potential leak of memory pointed to by 'data'}}
 | |
| 
 | |
| static void test11() {
 | |
|   void *data = 0;
 | |
|   my_malloc1(&data, 4);
 | |
|   my_free1(data);
 | |
| }
 | |
| 
 | |
| static void testUniqueingByallocationSiteInTopLevelFunction() {
 | |
|   void *data = my_malloc2(1, 4);
 | |
|   data = 0;
 | |
|   int x = 5;// expected-warning {{Potential leak of memory pointed to by 'data'}}
 | |
|   data = my_malloc2(1, 4);
 | |
| } // expected-warning {{Potential leak of memory pointed to by 'data'}}
 | |
| 
 | |
| static void test3() {
 | |
|   void *data = my_malloc2(1, 4);
 | |
|   free(data);
 | |
|   data = my_malloc2(1, 4);
 | |
|   free(data);
 | |
| }
 | |
| 
 | |
| int test4() {
 | |
|   int *data = (int*)my_malloc2(1, 4);
 | |
|   my_free1(data);
 | |
|   data = (int *)my_malloc2(1, 4);
 | |
|   my_free1(data);
 | |
|   return *data; // expected-warning {{Use of memory after it is freed}}
 | |
| }
 | |
| 
 | |
| void test6() {
 | |
|   int *data = (int *)my_malloc2(1, 4);
 | |
|   my_free1((int*)data);
 | |
|   my_free1((int*)data); // expected-warning{{Use of memory after it is freed}}
 | |
| }
 | |
| 
 | |
| // TODO: We should warn here.
 | |
| void test5() {
 | |
|   int *data;
 | |
|   my_free1((int*)data);
 | |
| }
 | |
| 
 | |
| static char *reshape(char *in) {
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| void testThatRemoveDeadBindingsRunBeforeEachCall() {
 | |
|     char *v = malloc(12);
 | |
|     v = reshape(v);
 | |
|     v = reshape(v);// expected-warning {{Potential leak of memory pointed to by 'v'}}
 | |
| }
 | |
| 
 | |
| // Test that we keep processing after 'return;'
 | |
| void fooWithEmptyReturn(int x) {
 | |
|   if (x)
 | |
|     return;
 | |
|   x++;
 | |
|   return;
 | |
| }
 | |
| 
 | |
| int uafAndCallsFooWithEmptyReturn() {
 | |
|   int *x = (int*)malloc(12);
 | |
|   free(x);
 | |
|   fooWithEmptyReturn(12);
 | |
|   return *x; // expected-warning {{Use of memory after it is freed}}
 | |
| }
 | |
| 
 | |
| 
 | |
| // If we inline any of the malloc-family functions, the checker shouldn't also
 | |
| // try to do additional modeling. <rdar://problem/12317671>
 | |
| char *strndup(const char *str, size_t n) {
 | |
|   if (!str)
 | |
|     return 0;
 | |
|   
 | |
|   // DO NOT FIX. This is to test that we are actually using the inlined
 | |
|   // behavior!
 | |
|   if (n < 5)
 | |
|     return 0;
 | |
|   
 | |
|   size_t length = strlen(str);
 | |
|   if (length < n)
 | |
|     n = length;
 | |
|   
 | |
|   char *result = malloc(n + 1);
 | |
|   memcpy(result, str, n);
 | |
|   result[n] = '\0';
 | |
|   return result;
 | |
| }
 | |
| 
 | |
| void useStrndup(size_t n) {
 | |
|   if (n == 0) {
 | |
|     (void)strndup(0, 20); // no-warning
 | |
|     return;
 | |
|   } else if (n < 5) {
 | |
|     (void)strndup("hi there", n); // no-warning
 | |
|     return;
 | |
|   } else {
 | |
|     (void)strndup("hi there", n);
 | |
|     return; // expected-warning{{leak}}
 | |
|   }
 | |
| }
 |