| // RUN: %clang_cc1 -analyze -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 {{Memory is never released; 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 {{Memory is never released; potential leak of memory pointed to by 'data'}} |
| data = my_malloc2(1, 4); |
| } // expected-warning {{Memory is never released; 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 {{Memory is never released; 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}} |
| } |
| } |