Memory leak detection in C++

来源:互联网 发布:许知远 陈嘉映 知乎 编辑:程序博客网 时间:2024/05/01 12:08

Introduction

Memory leakage has always been a part of bugs in C code where a programmer allocates memory in run time (in heap) and fails to deallocate it. Most programmers use some third party software to detect memory leakage in their code, but we can also write very simple code to detect memory leakage in our program ourselves. Usually we allocate memory in C using malloc() and calloc() in run time and then we deallocate the reserved memory using free(). Sometimes we do not free the reserved memory, though, which causes memory leakage. The method below is a very simple one that helps to detect memory leakage in your program.

Using the code

Let's assume that you have allocated some memory in your code using malloc() and calloc() and haven't deallocated it. Let's say your code looks something like below:

test.c#include<malloc.h>int main() {    char * ptr1 = (char *) malloc (10);     // allocating 10 bytes    int * ptr2 = (int *) calloc (10, sizeof(int));     // allocating 40 bytes let sizeof int =  4 bytes)    float * ptr3 = (float *) calloc (15, sizeof(float));     // allocating 60 bytes    ............    ............    ............    free(ptr2);    return 0;}

Steps for detecting memory leakage

I have tested the code in a Linux machine using GCC. You can test the same code in Windows, as well.

Step 1

To test for a memory leak, just add the leak_detector_c.h file to the test file and then add one line to the start of main function. Now the test code should look like below:

test.c#include<malloc.h>#include "leak_detector_c.h"int main() {    char * ptr1;     int * ptr2;     float * ptr3;    atexit(report_mem_leak);    ptr1 = (char *) malloc (10);     // allocating 10 bytes            ptr2 = (int *) calloc (10, sizeof(int));     // allocating 40 bytes let sizeof int =  4 bytes)    ptr3 = (float *) calloc (15, sizeof(float));     // allocating 60 bytes            ............    ............    ............    free(ptr2);    return 0;}

Step 2

Now compile the code and run the program.

# gcc -c leak_detector_.c# gcc -c test.c# gcc -o memtest leak_detctor_c.o test.o# ./memtest# cat /home/leak_info.txt
You will get output like below:
Memory Leak Summary-----------------------------------address : 140668936size    : 10 bytesfile    : test.cline    : 5-----------------------------------address : 140669560size    : 60 bytesfile    : test.cline    : 7-----------------------------------

The output shows the file name and line number that causes the memory leak. Now you can free the unallocated memory. If you have multiple source files, you can add the header file in all of the files where you want to detect possible memory leaks. Compile the program as above. Next, let's have a look into the code and see how it works.

The leak_detctor_c.h file contains some macros. The preprocessor replaces the call of malloc(), calloc() and free functions with xmalloc(), xcalloc() and xfree() respectively. While calling malloc(), our xmalloc() is called. We keep all information of the allocated memory -- such as the address, size, file name and line number -- in a linked list. When the code calls the free() function, it actually calls our xfree() and we manage to do the cleanup task. That is, we remove the entry of the allocated memory from the list and free up the allocated memory.

At the end of the program, we can get the unallocated memory references from the list. The lineatexit(report_mem_leak) registers the report_mem_leak() function to be called at the end of the program. This function writes the memory leak summary into the leak_info.txt file. You can also use #pragma exit directive instead of atexit().

原创粉丝点击