什么是内联汇编(Inline assembly)?
- 这是GCC对C语言的扩张
- 可直接在C语句中插入汇编指令
有何用处?
- 调用C语言不支持的指令。C语言不足以完成所有的CPU指令,特别是特权指令,C语言没有相应的对照,比如加载全局描述符表lgdt。
- 用汇编在C语言中手动优化。
如何工作?
- 用给定的模板和约束来生成汇编指令
- 在C函数内形成汇编源码
先来一个例子
Example 1
汇编代码:
movl $0xffff,%eax
等价于:
asm ("movl $0xffff, %%eax\n")
asm
是内联汇编关键字,- movl 是汇编指令名,l表示mov操作的是长字(4字节,w是双字,b是一个字节;l,w,b是AT&T汇编格式,它是unix下惯用的汇编格式,更多的可以看AT&T汇编)。
%
号需要变成%%
。
CC内联汇编–Syntax
asm(assembler template
:output operands 输出操作数 (optional)
:input operands 输入操作数 (optional)
:clobbers (optional)
)
Example 2
汇编代码:
movl %cr0,%ebx
movl %ebx,12(%esp)
orl $-2147483648,12(%esp)
movl 12(%esp),%eax
movl %eax,%cr0
等价于:
uint32_t cr0;
asm volatile ("movl %%cr0,%0\n" :"=r"(cr0));
cr0 |= 0x8000000;
asm volatile ("movl %0, %%cr0\n" ::"r"(cr0));
在C中嵌入汇编的最大问题是如何将C语言变量与指令操作数相关联。当然,gcc都帮我们想好了。
第一句:movl %%cr0,%0\n
表示把cr0寄存器读到%0寄存器里去,=r
表示这个%0寄存器的内容最终会赋给cr0这个变量。
第二句:把cr0这个变量的某一位置成1(或运算)。
第三句:把处理后的cr0变量赋给%0寄存器,%0寄存器再赋给cr0寄存器。
volatile关键字告知编译器不需要去调整代码顺序。
%0表示第一个用到的寄存器。
r表示任意寄存器。
Example 3
long__res,arg1=2,arg2=22,arg3=222,arg4=233;
__asm__ volatile ("int $0x80"
:"=a" (__res)
:"0" (11),"b"(arg1),"c"(arg2),"d"(arg3),"S"(arg4));
)
赋值的变量满足约束(限定符)
汇编代码:
Constraints
movl $11,%eax a=%eax
movl-28(%ebp),%ebx b=%ebx
movl-24(%ebp),%ecx c=%ecx
movl-20(%ebp),%edx d=%edx
movl-16(%ebp),%esi S=%esi
int $0x80 D=%edi
movl%edi,-12(%ebp) 0=same as the first
将局部变量赋给相应的寄存器,int $0x80
产生软中断,软中断返回的值再赋给ebp