最近遇到这样一个场景:将dynamorio内置到QEMU模拟器中,能直接分析QEMU模拟出来的指令等信息。QEMU是用C语言写的,而dynamorio使用C++实现的,这就要涉及到用C程序(QEMU)调用C++(dynamorio)的函数。我们知道,C和C++的命名空间不同,函数编译出来的名字都不同,怎么去写程序和编译呢?
1. 编写demo
一共三个文件testFunc.h、testFunc.cpp和main.c,用main.c调用testFunc.cpp的函数。例子参考的某一篇博客
#ifndef _TEST_FUNC_H_
#define _TEST_FUNC_H_
#ifdef __cplusplus
extern "C" {
#endif
void func(int data);
#ifdef __cplusplus
}
#endif
#endif
#include <stdio.h>
#include "testFunc.h"
#ifdef __cplusplus
extern "C"{
#endif
void func(int data){
printf("data is %d \n", data);
}
#ifdef __cplusplus
}
#endif
#include <stdio.h>
#include "testFunc.h"
int main(void){
func(100);
return 0;
}
2. 编写Makefile
main : testFunc.o main.o
gcc testFunc.o main.o -lstdc++ -o main
testFunc.o : testFunc.cpp
g++ -c testFunc.cpp -o testFunc.o
main.o : main.c
gcc -c main.c -o main.o
clean :
rm *.o main
也可以直接使用cmake工具去自动生成Makefile,但是鉴于QEMU的Makefile是自己写好的,所以只能硬着头皮上了。
QEMU虽然是个C项目,但也支持编译C++程序,从configure的结果就可以看出,QEMU已经指定了C++编译器,并命名为“c++”,想要编译C++程序直接可以在Makefile中用"c++ xx.cpp -o xx.o"这样编译。 上面的Makefile之所以是g++是因为这个demo不是在QEMU项目下做的,我已经测试过,讲gcc改成cc,g++改成c++就可以在QEMU中使用了。
3. 编译执行
make
./main
成功编译并运行
4. 原理
- 首先是"__cplusplus"宏和extern "C"关键字。extern “C"是专门给C++编译器看的,告诉编译器,下面包括的代码要以C语言的规则去编译。而是用C++编译器的时候会声明”__cplusplus"宏,我们将extern "C"用这个宏判断包起来,是防止该程序被C编译器编译,C编译器是识别不出extern "C"的,如果去掉宏判断,gcc main.c会报错。
- 所以我们需要先用C++编译器编译C++程序,但是不链接;然后用C编译器编译调用C++函数的C程序,最后手动将C和C++程序连接起来即可。
思考一下:如果C++ extern"C"的部分包含了C++特有的虚函数或者类,C程序还能使用吗?
|