`
luotuoass
  • 浏览: 652509 次
文章分类
社区版块
存档分类
最新评论

防止内存泄露 Linux下用Valgrind做检查

 
阅读更多

用C/C++开发其中最令人头疼的一个问题就是内存管理,有时候为了查找一个内存泄漏或者一个内存访问越界,需要要花上好几天时间,如果有一款工具能够帮助我们做这件事情就好了,valgrind正好就是这样的一款工具。

Valgrind是一款基于模拟linux下的程序调试器和剖析器的软件套件,可以运行于x86, amd64和ppc32架构上。valgrind包含一个核心,它提供一个虚拟的CPU运行程序,还有一系列的工具,它们完成调试,剖析和一些类似的任务。valgrind是高度模块化的,所以开发人员或者用户可以给它添加新的工具而不会损坏己有的结构。

valgrind的官方网址是:http://valgrind.org

你可以在它的网站上下载到最新的valgrind,它是开放源码和免费的。

一、介绍

valgrind包含几个标准的工具,它们是:

1、memcheck

memcheck探测程序中内存管理存在的问题。它检查所有对内存的读/写操作,并截取所有的malloc/new/free/delete调用。因此memcheck工具能够探测到以下问题:

1)使用未初始化的内存

2)读/写已经被释放的内存

3)读/写内存越界

4)读/写不恰当的内存栈空间

5)内存泄漏

6)使用malloc/new/new[]和free/delete/delete[]不匹配。

2、cachegrind

cachegrind是一个cache剖析器。它模拟执行CPU中的L1, D1和L2 cache,因此它能很精确的指出代码中的cache未命中。如果你需要,它可以打印出cache未命中的次数,内存引用和发生cache未命中的每一行代码,每一个函数,每一个模块和整个程序的摘要。如果你要求更细致的信息,它可以打印出每一行机器码的未命中次数。在x86和amd64上,cachegrind通过CPUID自动探测机器的cache配置,所以在多数情况下它不再需要更多的配置信息了。

3、helgrind

helgrind查找多线程程序中的竞争数据。helgrind查找内存地址,那些被多于一条线程访问的内存地址,但是没有使用一致的锁就会被查出。这表示这些地址在多线程间访问的时候没有进行同步,很可能会引起很难查找的时序问题。

二、valgrind对你的程序都做了些什么

valgrind被设计成非侵入式的,它直接工作于可执行文件上,因此在检查前不需要重新编译、连接和修改你的程序。要检查一个程序很简单,只需要执行下面的命令就可以了

valgrind --tool=tool_name program_name

比如我们要对ls -l命令做内存检查,只需要执行下面的命令就可以了

valgrind --tool=memcheck ls -l

不管是使用哪个工具,valgrind在开始之前总会先取得对你的程序的控制权,从可执行关联库里读取调试信息。然后在valgrind核心提供的虚拟CPU上运行程序,valgrind会根据选择的工具来处理代码,该工具会向代码中加入检测代码,并把这些代码作为最终代码返回给valgrind核心,最后valgrind核心运行这些代码。

如果要检查内存泄漏,只需要增加--leak-check=yes就可以了,命令如下

valgrind --tool=memcheck --leak-check=yes ls -l

不同工具间加入的代码变化非常的大。在每个作用域的末尾,memcheck加入代码检查每一片内存的访问和进行值计算,代码大小至少增加12倍,运行速度要比平时慢25到50倍。

valgrind模拟程序中的每一条指令执行,因此,检查工具和剖析工具不仅仅是对你的应用程序,还有对共享库,GNU C库,X的客户端库都起作用。

三、现在开始

首先,在编译程序的时候打开调试模式(gcc编译器的-g选项)。如果没有调试信息,即使最好的valgrind工具也将中能够猜测特定的代码是属于哪一个函数。打开调试选项进行编译后再用valgrind检查,valgrind将会给你的个详细的报告,比如哪一行代码出现了内存泄漏。

当检查的是C++程序的时候,还应该考虑另一个选项 -fno-inline。它使得函数调用链很清晰,这样可以减少你在浏览大型C++程序时的混乱。比如在使用这个选项的时候,用memcheck检查openoffice就很容易。当然,你可能不会做这项工作,但是使用这一选项使得valgrind生成更精确的错误报告和减少混乱。

一些编译优化选项(比如-O2或者更高的优化选项),可能会使得memcheck提交错误的未初始化报告,因此,为了使得valgrind的报告更精确,在编译的时候最好不要使用优化选项。

如果程序是通过脚本启动的,可以修改脚本里启动程序的代码,或者使用--trace-children=yes选项来运行脚本。

下面是用memcheck检查ls -l命令的输出报告,在终端下执行下面的命令

valgrind --tool=memcheck ls -l

程序会打印出ls -l命令的结果,最后是valgrind的检查报告如下:

==4187==

==4187== ERROR SUMMARY: 0 errors from 0 contexts (suppressed: 19 from 2)

==4187== malloc/free: in use at exit: 15,154 bytes in 105 blocks.

==4187== malloc/free: 310 allocs, 205 frees, 60,093 bytes allocated.

==4187== For counts of detected errors, rerun with: -v

==4187== searching for pointers to 105 not-freed blocks.

==4187== checked 145,292 bytes.

==4187==

==4187== LEAK SUMMARY:

==4187== definitely lost: 0 bytes in 0 blocks.

==4187== possibly lost: 0 bytes in 0 blocks.

==4187== still reachable: 15,154 bytes in 105 blocks.

==4187== suppressed: 0 bytes in 0 blocks.

==4187== Reachable blocks (those to which a pointer was found) are not shown.

==4187== To see them, rerun with: --show-reachable=yes

这里的“4187”指的是执行ls -l的进程ID,这有利于区别不同进程的报告。memcheck会给出报告,分配置和释放了多少内存,有多少内存泄漏了,还有多少内存的访问是可达的,检查了多少字节的内存。

下面举两个用valgrind做内存检查的例子

例子一 (test.c):

<ccid_nobr><table cellspacing="0" bordercolordark="#ffffff" cellpadding="2" width="400" align="center" bordercolorlight="black" border="1"><tbody><tr><td class="code" style="FONT-SIZE: 9pt" bgcolor="#e6e6e6"><pre><ccid_code>#include &lt;string.h&gt; int main(int argc, char *argv[]) { char *ptr; ptr = (char*) malloc(10); strcpy(ptr, "01234567890"); return 0; }</ccid_code></pre></td></tr></tbody></table></ccid_nobr>

编译程序

gcc -g -o test test.c

用valgrind执行命令

valgrind --tool=memcheck --leak-check=yes ./test

报告如下

==4270== Memcheck, a memory error detector.

==4270== Copyright (C) 2002-2006, and GNU GPL'd, by Julian Seward et al.

==4270== Using LibVEX rev 1606, a library for dynamic binary translation.

==4270== Copyright (C) 2004-2006, and GNU GPL'd, by OpenWorks LLP.

==4270== Using valgrind-3.2.0, a dynamic binary instrumentation framework.

==4270== Copyright (C) 2000-2006, and GNU GPL'd, by Julian Seward et al.

==4270== For more details, rerun with: -v

==4270==

==4270== Invalid write of size 1

==4270== at 0x4006190: strcpy (mc_replace_strmem.c:271)

==4270== by 0x80483DB: main (test.c:8)

==4270== Address 0x4023032 is 0 bytes after a block of size 10 alloc'd

==4270== at 0x40044F6: malloc (vg_replace_malloc.c:149)

==4270== by 0x80483C5: main (test.c:7)

==4270==

==4270== Invalid write of size 1

==4270== at 0x400619C: strcpy (mc_replace_strmem.c:271)

==4270== by 0x80483DB: main (test.c:8)

==4270== Address 0x4023033 is 1 bytes after a block of size 10 alloc'd

==4270== at 0x40044F6: malloc (vg_replace_malloc.c:149)

==4270== by 0x80483C5: main (test.c:7)

==4270==

==4270== ERROR SUMMARY: 2 errors from 2 contexts (suppressed: 12 from 1)

==4270== malloc/free: in use at exit: 10 bytes in 1 blocks.

==4270== malloc/free: 1 allocs, 0 frees, 10 bytes allocated.

==4270== For counts of detected errors, rerun with: -v

==4270== searching for pointers to 1 not-freed blocks.

==4270== checked 51,496 bytes.

==4270==

==4270==

==4270== 10 bytes in 1 blocks are definitely lost in loss record 1 of 1

==4270== at 0x40044F6: malloc (vg_replace_malloc.c:149)

==4270== by 0x80483C5: main (test.c:7)

==4270==

==4270== LEAK SUMMARY:

==4270== definitely lost: 10 bytes in 1 blocks.

==4270== possibly lost: 0 bytes in 0 blocks.

==4270== still reachable: 0 bytes in 0 blocks.

==4270== suppressed: 0 bytes in 0 blocks.

==4270== Reachable blocks (those to which a pointer was found) are not shown.

==4270== To see them, rerun with: --show-reachable=yes

从这份报告可以看出,进程号是4270,test.c的第8行写内存越界了,引起写内存越界的是strcpy函数,

第7行泄漏了10个字节的内存,引起内存泄漏的是malloc函数。

例子二(test2.c)

<ccid_nobr><table cellspacing="0" bordercolordark="#ffffff" cellpadding="2" width="400" align="center" bordercolorlight="black" border="1"><tbody><tr><td class="code" style="FONT-SIZE: 9pt" bgcolor="#e6e6e6"><pre><ccid_code>#include &lt;stdio.h&gt; int foo(int x) { if (x &lt; 0) { printf("%d ", x); } return 0; } int main(int argc, char *argv[]) { int x; foo(x); return 0; }</ccid_code></pre></td></tr></tbody></table></ccid_nobr>

编译程序

gcc -g -o test2 test2.c

用valgrind做内存检查

valgrind --tool=memcheck ./test2

输出报告如下

==4285== Memcheck, a memory error detector.

==4285== Copyright (C) 2002-2006, and GNU GPL'd, by Julian Seward et al.

==4285== Using LibVEX rev 1606, a library for dynamic binary translation.

==4285== Copyright (C) 2004-2006, and GNU GPL'd, by OpenWorks LLP.

==4285== Using valgrind-3.2.0, a dynamic binary instrumentation framework.

==4285== Copyright (C) 2000-2006, and GNU GPL'd, by Julian Seward et al.

==4285== For more details, rerun with: -v

==4285==

==4285== Conditional jump or move depends on uninitialised value(s)

==4285== at 0x8048372: foo (test2.c:5)

==4285== by 0x80483B4: main (test2.c:16)

==4285==p p

==4285== ERROR SUMMARY: 1 errors from 1 contexts (suppressed: 12 from 1)

==4285== malloc/free: in use at exit: 0 bytes in 0 blocks.

==4285== malloc/free: 0 allocs, 0 frees, 0 bytes allocated.

==4285== For counts of detected errors, rerun with: -v

==4285== All heap blocks were freed -- no leaks are possible.

从这份报告可以看出进程PID是4285,test2.c文件的第16行调用了foo函数,在test2.c文件的第5行foo函数使用了一个未初始化的变量。

valgrind还有很多使用选项,具体可以查看valgrind的man手册页和valgrind官方网站的在线文档。

来自:http://linux.ccidnet.com/art/302/20071024/1252121_1.html

分享到:
评论

相关推荐

    linux下检查内存泄漏的工具+例子

    因此,在日常开发中,良好的编程习惯,如及时释放不再使用的内存,是防止内存泄漏的根本。 总之,掌握Valgrind的使用,可以帮助程序员在Linux环境下更有效地发现和修复内存泄漏问题,提高软件质量,保障系统的稳定...

    valgrind内存检查工具

    Valgrind是一款强大的开源内存分析工具,主要用于检测C和C++程序中的内存错误,包括内存泄漏、无效内存访问以及未初始化的...对于大型项目,定期进行Valgrind检查可以帮助保持代码健康,避免因内存问题导致的潜在灾难。

    Linux下C语言程序内存泄漏的研究.pdf

    本文将深入探讨内存泄漏的概念、分类、危害,以及如何在Linux环境下使用Valgrind工具来检测和解决内存泄漏。 内存泄漏是指程序动态分配的内存,在使用完毕后没有正确释放,导致这部分内存无法被再次使用或回收。...

    内存泄露检测工具及学习资料(valgrind)

    Valgrind是一款强大的动态分析工具,专用于帮助开发者检测和解决Linux环境下的内存泄漏问题。下面将详细介绍Valgrind的工作原理、使用方法以及如何利用它来定位和修复内存泄漏。 Valgrind是一个开源的模拟执行引擎...

    Linux 平台检测内存是否泄漏

    本文将深入探讨如何在Linux环境下检测和分析内存泄漏,主要针对C++程序。 首先,我们需要理解内存泄漏的基本概念。内存泄漏是指程序在申请内存后,无法释放已申请的内存空间,一次小的内存泄漏可能看似无害,但随着...

    嵌入式Linux内存与性能详解-史子旺

    最后,书中可能涵盖了内存泄漏检测和防止措施,比如使用Valgrind等工具进行内存泄漏检查,并教导读者如何编写无泄漏的C/C++代码。此外,可能会讲解如何通过配置内核参数、调整启动参数、使用内存限制工具等手段来...

    valgrind-3.17.0.tar.gz

    总之,Valgrind-3.17.0是Linux环境下开发C和C++程序的得力助手,它能帮助开发者发现和解决内存管理问题,提高代码质量,防止因内存泄漏和多线程错误导致的系统不稳定。对于任何在Linux系统上开发内存敏感应用的人来...

    C++内存泄露检测原理、源码及详解

    在C++编程中,内存管理是一项关键任务,而内存泄露是开发者经常面临的问题之一。...通过静态和动态分析方法,结合跨平台策略,我们可以有效地防止和解决内存泄露问题,确保程序在各种环境下都能高效、安全地运行。

    内存泄漏查找方法总结

    总的来说,防止内存泄漏的关键在于良好的编程习惯和适当的工具使用。通过理解和应用上述方法,开发者可以更有效地检测和解决内存泄漏问题,从而提高程序的稳定性和效率。在实际开发中,应始终关注内存管理,确保分配...

    测试系统内存泄露的工具

    内存泄露是计算机程序设计中一个严重的问题,尤其是在服务器环境下,长时间运行的应用程序如果存在内存泄露,会导致系统资源耗尽...通过使用合适的工具和遵循良好的编程习惯,我们可以有效地管理和防止内存泄露的发生。

    C/C++程序内存泄漏检测

    总的来说,防止C/C++程序内存泄漏的关键在于良好的编程习惯,比如使用智能指针管理动态内存,及时释放不再使用的内存,以及在开发阶段积极进行内存泄漏检测。通过上述方法,开发者可以更有效地定位和修复内存泄漏...

    valgrind-3.10.1

    valgrind是linux下非常好用的一个检查内存泄漏的好工具,其标准工具包括memcheck、cachegrind、halgrind等;该工具也可以在android中使用,可有效检查并防止内存泄漏。

    linux下段错误检查sigsegv

    - 使用`valgrind`工具检查内存错误,如内存泄漏和无效访问。 - `strace`可以追踪系统调用,查看是否有不当的文件或资源操作。 - `gdb`的`watch`命令可以监视特定变量的变化,帮助找出问题的触发点。 6. 防止段...

    检测内存泄露,让自己清楚哪里出问题了

    内存泄露是程序运行时常见的问题,它发生在程序分配了一些内存,但在不再需要这些内存时未能...同时,培养良好的编程习惯,如尽早释放不再使用的内存,避免全局变量和长生命周期的对象,也能有效防止内存泄露的发生。

    Visual Leak Detector-C++内存泄露检测工具

    然而,需要注意的是,由于VLD依赖于Windows调试API,因此它主要适用于Windows平台的C++应用,对于跨平台的项目可能需要寻找其他内存泄漏检测工具,如Valgrind(主要用于Linux环境)。 总的来说,Visual Leak ...

    【技术点之一】使用 CRT 调试功能来检测内存泄漏.rar_C++检测内存泄露的方法_ROO_YGR_泄漏

    6. **静态分析工具**:除了使用 CRT 调试功能,还可以借助静态代码分析工具,如Visual Studio的静态分析器,或者第三方工具如Valgrind(针对Linux环境),它们能够在编译阶段发现潜在的内存泄漏问题。 7. **RAII...

Global site tag (gtag.js) - Google Analytics