You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

217 lines
4.4 KiB

/*
* Copyright (c) Facebook, Inc. and its affiliates.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*/
#include <stdlib.h>
void malloc_no_free_bad() { int* p = malloc(sizeof(p)); }
int* malloc_returned_ok() {
int* p = malloc(sizeof(p));
return p;
}
void malloc_out_parameter_ok(int** x) { *x = (int*)malloc(sizeof(int)); }
void malloc_out_parameter_local_mutation_ok(int** x) {
*x = (int*)malloc(sizeof(int));
x = NULL; // not visible from the outside
}
void malloc_out_parameter_local_mutation_bad(int** x) {
*x = (int*)malloc(sizeof(int));
*x = NULL;
}
void malloc_then_free_ok() {
int* p = malloc(sizeof(p));
if (p) {
*p = 5;
free(p);
}
}
int* create_p() {
int* p = malloc(sizeof(p));
return p;
}
void malloc_interproc_no_free_bad() { int* p = create_p(); }
void malloc_interproc_no_free_bad2() {
int* p = malloc(sizeof(p));
int z = 3;
int y = 4;
int* q = p;
}
void malloc_formal_leak_bad(int* x) { x = (int*)malloc(sizeof(int*)); }
static void* (*const malloc_func)(size_t) = malloc;
static void (*const free_func)(void*) = free;
void* malloc_via_ptr(size_t size) {
void* ret = NULL;
if (size <= 0) {
return NULL;
}
ret = malloc_func(size);
return ret;
}
void free_via_ptr(void* x) { free_func(x); }
void malloc_ptr_leak_bad() { int* p = (int*)malloc_via_ptr(sizeof(int)); }
void malloc_ptr_no_check_leak_bad() {
int* p = (int*)malloc_via_ptr(sizeof(int));
*p = 42;
}
void malloc_ptr_free_ok() {
int* p = (int*)malloc_via_ptr(sizeof(int));
free(p);
}
void malloc_ptr_free_ptr_ok() {
int* p = (int*)malloc_via_ptr(sizeof(int));
free_via_ptr(p);
}
void alias_ptr_free_ok(int* out, int flag) {
int* y;
if (flag) {
y = (int*)malloc(sizeof(int));
} else {
y = out;
}
if (y && y != out) {
free(y);
}
}
void report_leak_in_correct_line_bad(int* x) {
x = (int*)malloc(sizeof(int));
if (x != NULL) {
return; // should report leak at this line
}
free(x);
}
void* realloc_wrapper(void* p, size_t size) { return realloc(p, size); }
void realloc_free_ok() {
int* p = (int*)malloc(sizeof(int));
int* q = realloc_wrapper(p, sizeof(int));
free(q);
}
void realloc_no_free_bad() {
int* p = (int*)malloc(sizeof(int));
int* q = realloc_wrapper(p, sizeof(int));
}
void realloc_no_check_bad() {
int* p = (int*)malloc(sizeof(int));
int* q = realloc_wrapper(p, sizeof(int));
*q = 42;
free(q);
}
void* my_malloc(size_t size);
void* a_malloc(size_t size);
void my_free(void* p);
void* my_realloc(void* p, size_t size);
void user_malloc_leak_bad() { int* x = (int*)a_malloc(sizeof(int)); }
void test_config_options_1_ok() {
int* p = (int*)malloc(sizeof(int));
int* q = my_realloc(p, sizeof(int));
my_free(q);
}
void test_config_options_2_ok() {
int* p = (int*)my_malloc(sizeof(int));
int* q = realloc(p, sizeof(int));
my_free(q);
}
void test_config_options_no_free_bad() {
int* p = (int*)my_malloc(sizeof(int));
int* q = my_realloc(p, sizeof(int));
}
struct ref_counted {
size_t count;
int data;
};
// the address of the malloc()'d pointer can be retrieved from the
// return value using pointer arithmetic
int* alloc_ref_counted_ok() {
struct ref_counted* p =
(struct ref_counted*)malloc(sizeof(struct ref_counted));
if (p) {
p->count = 1;
return &(p->data);
} else {
return NULL;
}
}
// returning the value of the field loses the address of p
int alloc_ref_counted_bad() {
struct ref_counted* p =
(struct ref_counted*)malloc(sizeof(struct ref_counted));
if (p) {
p->count = 1;
p->data = 42;
return p->data;
} else {
return NULL;
}
}
// the address of the malloc()'d pointer can be retrieved from the
// return value using pointer arithmetic
void* alloc_ref_counted_arith_ok(size_t size) {
int* p = (int*)malloc(size + sizeof(int));
if (p) {
// register count = 1 and point past the ref count
*p++ = 1;
}
return p;
}
int return_malloc_deref_bad() {
int* p = (int*)malloc(sizeof(int));
if (p) {
*p = 42;
return *p;
}
return 10;
}
typedef struct node_st {
int* data;
} NODE;
void mutual_recursion_2(NODE* x);
void mutual_recursion(NODE* x) { mutual_recursion_2(x); }
void mutual_recursion_2(NODE* x) { mutual_recursion(x); }
void FP_interproc_mutual_recusion_leak(NODE* x) {
int* d;
if (x->data == NULL) {
x->data = (int*)malloc(sizeof(int));
}
mutual_recursion(x);
}