// RUN: %clang_analyze_cc1 -analyzer-checker=core,unix.Malloc -verify %s typedef __typeof(sizeof(int)) size_t; void* malloc(size_t size); void *calloc(size_t num, size_t size); void free(void * ptr); void escape(void *); void next_statement(); void conditional_malloc(bool coin) { static int *p; if (coin) { p = (int *)malloc(sizeof(int)); } p = 0; // Pointee of 'p' dies, which is recognized at the next statement. next_statement(); // expected-warning {{Potential memory leak}} } void malloc_twice() { static int *p; p = (int *)malloc(sizeof(int)); next_statement(); p = (int *)malloc(sizeof(int)); next_statement(); // expected-warning {{Potential memory leak}} p = 0; next_statement(); // expected-warning {{Potential memory leak}} } void malloc_escape() { static int *p; p = (int *)malloc(sizeof(int)); escape(p); // no-leak p = 0; // no-leak } void free_whatever_escaped(); void malloc_escape_reversed() { static int *p; escape(&p); p = (int *)malloc(sizeof(int)); free_whatever_escaped(); p = 0; // FIXME: We should not report a leak here. next_statement(); // expected-warning {{Potential memory leak}} } int *malloc_return_static() { static int *p = (int *)malloc(sizeof(int)); return p; // no-leak } int malloc_unreachable(int rng) { // 'p' does not escape and never freed :( static int *p; // For the second invocation of this function, we leak the previous pointer. // FIXME: We should catch this at some point. p = (int *)malloc(sizeof(int)); *p = 0; if (rng > 0) *p = rng; return *p; // FIXME: We just leaked 'p'. We should warn about this. } void malloc_cond(bool cond) { static int *p; if (cond) { p = (int*)malloc(sizeof(int)); free_whatever_escaped(); p = 0; // FIXME: We should not report a leak here. next_statement(); // expected-warning {{Potential memory leak}} } escape(&p); }