C语言内存泄露检测--Memwatch

项目:迅雷下载库内存泄露检测

作者:曾金龙

供职:深圳迅雷网络技术股份有限公司

领域:迅雷下载库

时间:2014-07-26


迅雷的移动下载库是用C语言编写的,为了能够横跨欧亚非拉C语言内存泄露检测--Memwatch(ios,android,还有诺基亚的什么系统来着,对了symban)和性能,我们厂的下载库就选择了C,然后在C代码里面使用各种宏,预编译等来达到一 个下载库,多平台使用。不过平台相关的都已经约束在了一个叫common的库里面的一个子模块里面了。这极大的解放了我厂的码农朋友。

但是C毕竟是C,在聊起C的时候,我们组最喜欢说的是,我们权限最大,想干啥干啥。最悲痛的是,测试妹子过来了,然后说各种问题,但是logcat都捕获不到,或者有时候崩溃了但是没有tombstone,等。其实,C最大的危险就是内存泄露。。。因为C的内存泄露,我和测试妹子的感情最近变得更加的亲密而又微妙。。。


老大发话了,必须解决所有C内存泄露。。。

在我下写我负责的数据存储模块的时候,我就动了一个小技巧,就能够把本模块的内存泄露全部给检测出来,而我这朴素的思想其实也是其他内存泄露检测工具的灵魂,只是,老大嫌我的太朴素了。

我的做法很简单,每次malloc内存的时候不是调用系统的malloc,而是调用我的malloc,取名为fc_malloc(因为我的模块叫file_cache),free则调用fc_free。具体代码如下:

#define fc_malloc(size) fc_malloc_imp(size,__FILE__,__LINE__)

#define fc_free(ptr) fc_free_imp(ptr)

typedef struct mem_log{
void* addr;
char* file;
int   line;
}memlog;
#ifdef _LOGGER
static list g_mem_list
#endif // _LOGGER
void init_module()
{
    #ifdef _LOGGER
    list_init(&g_mem_list);
    #endif
}

void uninit_module()
{
    #ifdef _LOGGER
    list_for_each(g_mem_list,v)
    {
        memlog* iterm=(memlog*)LIST_VALUE(v);
        printf("%s,%d,%x\n",iterm->file,iterm->line,iterm->addr);//output the memory not free..
        free(iterm);
    }
    list_clear(&g_mem_list);

    #endif // _LOGGER

}

void* fc_malloc_imp(size_t size,char* file,int line)
{
    void* ptr=malloc(size);

    #ifdef _LOGGER

    if(ptr!=NULL)
    {
        memlog* iterm = (memlog*)malloc(sizeof(memlog));
        iterm->addr=ptr;
        iterm->file=file;
        iterm->line=line;
        list_insert(&g_mem_list,iterm);
    }
    #endif // _LOGGER
    return ptr;
}
void fc_free_imp(void* ptr)
{
    #ifdef _LOGGER
    if(ptr!=NULL)
    {
        list_erase_by_addr(&g_mem_list,ptr);//in this funciton we need free the memlog iterm memory.
    }

    #endif // _LOGGER
    free(ptr);
}

却是很简单,说白了就是用一个链表去等级下每次malloc的地址信息,当然为了好定位,包含文件和行信息,然后在模块卸载的时候,就可以打印出哪些内存还没有被释放。这样对于检查内存泄露,已经达到目的了,很简单吧。有用的东西都是很简单,但很巧妙。

在老大说我应该去看下现有的内存检测工具之前,我觉得我 的这个“创作”已经帮了我解决了所有的内存泄露问题,至少,我和测试妹子的关系从聊bug转移到聊哪件衣服好看了,恩,不错。



上一篇:把ElasticSearch当成是NoSQL数据库


下一篇:如何把其他服务商的服务器项目迁移到阿里云服务器上