78 lines
1.9 KiB
C++
78 lines
1.9 KiB
C++
|
// 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);
|
||
|
}
|