昨天晚上下了ccmalloc,准备研究研究。
下载地址:http://download.chinaunix.net/download.php?id=2359&ResourceID=1244 (如果该链接失效可以在该网站搜索,这是个非常不错的网站)
下载版本: 0.4.0 (ccmalloc-0.4.0.tar.gz )
ccmalloc的编译安装:
cd ccmalloc-0.4.0/
./configure #默认会设置安装目录为/usr/local
./configure --prefix=/some/path/to/install/position/of/ccmalloc #也可以通过该命令进行指定安装位置
make
make install #所谓的安装就是将一些库和一个脚本文件拷贝到指定的安装目录,在安装目录bin下面有一个shell脚本ccmalloc
#链接的时候使用该脚本就可以了
应用ccmalloc:
比如我们测试 正常的build过程如下:gcc -o ccm.elf test1.c
要使用ccmalloc,则使用命令:ccmalloc gcc -o ccm.elf test1.c #ccmalloc就是上面提到的安装目录的脚本,如果没在path内,您可能需要指定绝对路径
链接问题:
mayer@mayer-ubuntu:ccmalloc$ ccmalloc gcc -o ccm.elf test1.c [ccmalloc] version .. 0.4.0 [ccmalloc] prefix .. /home/mayer/all/project/ccmalloc/ccmalloc [ccmalloc] compilers .. g++ gcc [ccmalloc] wrapper .. /home/mayer/all/project/ccmalloc/ccmalloc/lib/ccmalloc-gcc.o gcc -o ccm.elf test1.c /home/mayer/all/project/ccmalloc/ccmalloc/lib/ccmalloc-gcc.o -L/home/mayer/all/project/ccmalloc/ccmalloc/lib -lccmalloc -ldl test1.c: In function ‘Leak’: test1.c:5: warning: incompatible implicit declaration of built-in function ‘malloc’ test1.c:5: warning: incompatible implicit declaration of built-in function ‘strlen’ test1.c:6: warning: incompatible implicit declaration of built-in function ‘memcpy’ test1.c: In function ‘AvoidLeak’: test1.c:11: warning: incompatible implicit declaration of built-in function ‘malloc’ test1.c:11: warning: incompatible implicit declaration of built-in function ‘strlen’ test1.c:12: warning: incompatible implicit declaration of built-in function ‘memcpy’ test1.c: In function ‘main’: test1.c:22: warning: incompatible implicit declaration of built-in function ‘free’ test1.c:24: warning: incompatible implicit declaration of built-in function ‘exit’ /home/mayer/all/project/ccmalloc/ccmalloc/lib/ccmalloc-gcc.o:(.eh_frame+0x12): undefined reference to `__gxx_personality_v0' collect2: ld returned 1 exit status
ccmalloc-gcc.o:(.eh_frame+0x12): undefined reference to `__gxx_personality_v0'
collect2: ld returned 1 exit status
从上面可以知道,链接的时候ccmalloc-gcc.o文件中没有找到对应的符号,google N久没找到答案,郁闷。
研究了ccmalloc编译过程中生成ccmalloc-gcc.o的过程:
obj/ccmalloc-gcc.o: src/ccmalloc.cc src/config.h src/ccmalloc.h
gcc -DCTORDTOR -g -c -o $@ src/ccmalloc.cc
而文件ccmalloc.cc的内容如下:
/*---------------------------------------------------------------------------- * (C) 1997-2003 Armin Biere * * $Id: ccmalloc.cc,v 1.11 2003/02/03 08:03:56 biere Exp $ *---------------------------------------------------------------------------- */ extern "C" { #include "ccmalloc.h" }; class CCMalloc_InitAndReport { public: CCMalloc_InitAndReport () { ccmalloc_static_initialization (); } ~CCMalloc_InitAndReport () { ccmalloc_report (); } }; static CCMalloc_InitAndReport ccmalloc_initAndReport;
该文件的内容中使用了类,是C++的东东。而文件名不是以.c结尾的,猜想编译的函数命名方式应该是c++的。并且该文件如果改名.c的话,那么用gcc也是不能编过的,原因见下面的转载内容。
网上的资源很是丰富啊,找到以下内容。
===================转载内容开始======================
gcc和g++都是GNU(组织)的一个编译器。
误区一:gcc只能编译c代码,g++只能编译c++代码
两者都可以,但是请注意:
1.后缀为.c的,gcc把它当作是C程序,而g++当作是c++程序;后缀为.cpp的,两者都会认为是c++程序,注意,虽然c++是c的超集,但是两者对语法的要求是有区别的。C++的语法规则更加严谨一些。
2.编译阶段,g++会调用gcc,对于c++代码,两者是等价的,但是因为gcc命令不能自动和C++程序使用的库联接,所以通常用g++来完成链接,为了统一起见,干脆编译/链接统统用g++了,这就给人一种错觉,好像cpp程序只能用g++似的。
误区二:gcc不会定义__cplusplus宏,而g++会
实际上,这个宏只是标志着编译器将会把代码按C还是C++语法来解释,如上所述,如果后缀为.c,并且采用gcc编译器,则该宏就是未定义的,否则,就是已定义。
误区三:编译只能用gcc,链接只能用g++
严格来说,这句话不算错误,但是它混淆了概念,应该这样说:编译可以用gcc/g++,而链接可以用g++或者gcc -lstdc++。因为gcc命令不能自动和C++程序使用的库联接,所以通常使用g++来完成联接。但在编译阶段,g++会自动调用gcc,二者等价。
误区四:extern "C"与gcc/g++有关系
实际上并无关系,无论是gcc还是g++,用extern "c"时,都是以C的命名方式来为symbol命名的,否则,都以c++方式命名。
===================转载内容结束======================
由上面的内容,我们知道了问题的根源,ccmalloc-gcc.o的函数命名方式为c++方式,当我们链接的时候,由于使用gcc不会自动去找对应的库,所以需要我们手动指定,所以链接的时候加上 -lstdc++ 即可解决问题,即使用
ccmalloc gcc -o ccm.elf test1.c -lstdc++