如何在64位的Linux系统上使用汇编和C语言混

如何在64位的Linux系统上使用汇编和C语言混,第1张

编译链接的时候使用的指令:(AMD处理器,64位 *** 作系统)

编译链接指令

1 nasm -f elf foo.s -o foo.o

2 gcc -c bar.c -o bar.o

3 ld -s -o foobar bar.o foo.o

汇编语言用nasm编写并用nasm编译器编译,而C语言用的是gcc编译,这些都没有问题,但是在链接的时候出错了,提示如下:

ld: i386 architecture of input file `foo.o' is incompatible with i386:x86-64 output

google了一下,意思就是nasm 编译产生的是32位的目标代码,gcc 在64位平台上默认产生的是64位的目标代码,这两者在链接的时候出错,gcc在64位平台上默认以64位的方式链接。

这样在解决的时候就会有两种解决方案:

<1>让gcc 产生32位的代码,并在链接的时候以32位的方式进行链接

在这种情况下只需要修改编译和链接指令即可,具体如下:

32位的编译链接指令

1 nasm -f elf foo.s -o foo.o

2 gcc -m32 -c bar.c -o bar.o

3 ld -m elf_i386 -s -o foobar foo.o bar.o

具体的-m32 和 -m elf_i386 请自行查阅gcc (man gcc)

如果你是高版本的gcc(可能是由于更新内核造成的),可能简单的使用-m32 的时候会提示以下错误(使用别人的历程,自己未曾遇到):

>In file included from /usr/include/stdio.h:28:0,

>from test.c:1:

>/usr/include/features.h:323:26: fatal error: bits/predefs.h: No such file or directory

>compilation terminated.

这应该是缺少构建32 位可执行程序缺少的包,使用以下指令安装:

sudo apt-get install libc6-dev-i386

此时应该就没有什么问题了。

(1) 参数个数少于7个:

f (a, b, c, d, e, f)

a->%rdi, b->%rsi, c->%rdx, d->%rcx, e->%r8, f->%r9

g (a, b)

a->%rdi, b->%rsi

有趣的是, 实际上将参数放入寄存器的语句是从右到左处理参数表的, 这点与32位的时候一致.

CODE

2) 参数个数大于 7 个的时候

H(a, b, c, d, e, f, g)

a->%rdi, b->%rsi, c->%rdx, d->%rcx, e->%rax

g->8(%esp)

f->(%esp)

call H

易失寄存器:

%rax, %rcx, %rdx, %rsi, %rdi, %r8, %r9 为易失寄存器, 被调用者不必恢复它们的值。

显然,这里出现的寄存器大多用于参数传递了, 值被改掉也无妨。而 %rax, %rdx 常用于

数值计算, %rcx 常用于循环计数,它们的值是经常改变的。其它的寄存器为非易失的,也

就是 rbp, rbx, rsp, r10~r15 的值如果在汇编模块中被改变了,在退出该模块时,必须将

其恢复。

教训:

用汇编写模块, 然后与 c 整合, 一定要搞清楚编译器的行为, 特别是参数传递的方式. 此外, 我现在比较担心的一点是, 将来如果要把程序移植 到 WIN/VC 环境怎么办? 以前我用cygwin的gcc来处理汇编模块, 用vc来处理c模块, 只需要很少改动. 现在的问题是, 如果VC用 不同的参数传递方式, 那我不就麻烦了?


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/yw/7460722.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-04-05
下一篇 2023-04-05

发表评论

登录后才能评论

评论列表(0条)

保存