最近有台湾的网友写EMAIL给我说从汇编语言调用C程序的时候遇到了一些麻烦,我想了一下,确实,可能介绍C程序调用汇编程序的文章相对要多一些,而反过来的情况可能用到的人不多,所以决定以此为题写一个短篇。 首先,如果想从MASM编译的汇编程序调用DJGPP的GCC编译的C程序,这个可能性不大,两个编译出的差异恐怕太大了,本文使用MASM 6.11和TURBO C 3.0作为例子,完成一个从汇编调用C的小范例,这样的调用比较靠谱。 使用过TURBO C的人都知道,TC把编译出来的程序结构分成6种模式:Tiny、Small、Medium、Compact、Large和Huge,不管是汇编调C还是 C调用汇编,都首先要搞清楚使用什么模式,必须要保证汇编编译出的模式和TC编译出的模式一致,否则调用是要出问题;其次要搞清楚参数调用规则,比如 f(a,b),a、b这两个参数是如何传到C程序中f函数下的,解决了这两个问题,剩下的就是如何链接的问题了。 先说第一个问题,TC中的编译模式。 汇编程序的大致模式是这样的: <code> segment byte public 'code' <dseg> group _data, _bss _bss segment word public 'bss' end 这些段在TC下各种模式的对应名称如下: 存储模式 在TC下的名称 所以在编写汇编程序时,一定要使用TC中的名称来确定各个段,否则无法链接到一起 第二个问题,参数的传递,TC使用所谓的“C协议”来传递参数,即:将要传递的参数的值,按照逆序(从右向左)压入堆栈,后接返回地址。 比如函数f(a,b),为一个近(near)调用,a和b均为整数(int),调用f后,堆栈中的内容依次为:返回地址、a和b,所以当我们在汇编中使用call f时,要先运行push b和push a两条指令,否则,必然出现混乱。 下面我们实现一个范例。 程序实现了2 + 6 = 8的计算,汇编语言中,向C程序中的函数add_c传递两个参数:a和b,在C程序中完成运算并返回结果给汇编程序。使用Small模式进行编译,下面是汇编部分的程序: 文件名:test.asm DGROUP GROUP _DATA _TEXT SEGMENT BYTE PUBLIC 'CODE' 其中的_main是对应C语言里的main()函数,这样和C程序链接到一起时才有启动位置(后面可以看到,我们要使用TC下的启动代码)。 下面是C程序。 文件名:add_c.c add_c(int x, int y) { 这段程序实在没有什么好解释的。 下面我们要编译这两个程序,汇编程序的编译没有什么特别的,使用masm编译即可。 C程序其实也没什么特别的,只是要注意设成Small模式即可,只编译不要链接。 下面我们来把这两个编译好的OBJ文件链接到一起。 tlink \tc\lib\c0s.obj test add_c, test, test, \tc\lib\emu \tc\lib\cs 这个是我们前面提到的要解决的三个问题的最后一个,我们来解释一下: tlink:TC下的链接程序 test add_c:指两个已经编译好的OBJ文件test.obj和add_c.obj \tc\lib\emu \tc\lib\cs:是我们这个程序需要的TC下的库函数,这两个库在\tc\lib目录下可以找到。 执行结果: 希望此文能给文章开头提到的那位台湾朋友一些帮助。 |