TIPS:在编译过程汇中,汇编代码块是原封不动地送到汇编语言编译阶段的。一、为什么会用到汇编?6 v& }, T, m+ M( c# Q) y M: K1.为了提高速度和效率。不过这种情况很少了,现在C/C++编译器的优化很厉害了。" J* F' O' f0 y& i+ |* X0 Z2.为了实现某些C语言中不具备、但为不同的机器所特有的功能。这是主要原因。* N1 n, F0 L4 k% B+ x7 q! a3.为了利用通用的汇编语言例程。也常会遇到。3 `$ |4 J# D1 `6 g2 M# c8 t# a8 e二、何时使用汇编?第一种情况是,绝对没有其他方法可以使用。4 p) ?8 Y: p' b; v第二种情况出现在某个C语言程序的执行时间必须减少的时候。三、如何嵌入汇编?- |& `; P/ o% F0 d& W) d' v1 S2 KTurbo C :1.使用预处理程序的伪指令#asm和#endasm,#asm用来开始一个汇编程序块,而#endasm指令用于该块的结束。例:代码:
mul(a,b)int a,b;{ #asm mov ax,word ptr 8[bp] imul ax word ptr 10[bp] #endasm}
2.使用asm语句格式:asm<汇编语句>代码:
mul(a,b)int a,b;{ asm mov ax,word ptr 8[bp] asm imul ax word ptr 10[bp]}
注意:asm行后面没有分号 Visual C++ :* x3 M; I1 ?$ \3 r5 l/ g格式:__asm 汇编指令 [ ; ]__asm { 汇编指令 } [ ; ]6 k" W* t) s. V- ^: R8 O" wasm前面是两条下划线,后面的方括号内容表示分号可有可无。使用方法:一条一条地用代码:
__asm mov al, 2__asm mov dx, 0xD007__asm out dx, al
组成一块地用代码:
__asm { mov al, 2 mov dx, 0xD007 out dx, al}
还可以弄成一条代码:
__asm mov al, 2 __asm mov dx, 0xD007 __asm out dx, al
msdn里面的内容:代码:
/* POWER2.C */ #include <stdio.h> int power2( int num, int power ); void main( void ) { printf( "3 times 2 to the power of 5 is %d\n", power2( 3, 5) ); } int power2( int num, int power ) { __asm { mov eax, num ; Get first argument mov ecx, power ; Get second argument shl eax, cl ; EAX = EAX * ( 2 to the power of CL ) } /* Return with result in EAX */ }
GNU GCC :由于内容比较多,所以简单说一下。用到的关键字 “__asm__” 表示后面的代码为内嵌汇编,“asm”是“__asm__”的别名。 4 a% v4 ^. G$ F“__volatile__” 表示编译器不要优化代码,后面的指令保留原样,“volatile”是它的别名。 . f7 c' V) l% t" u; _! _括号里面是汇编指令。 内嵌汇编语法如下: . \( N5 b; o+ s& C; W8 c" v4 R6 V__asm__( 汇编语句模板: 输出部分: 输入部分: + _. M) L3 I% @ 破坏描述部分) & F2 G4 u' o( T0 m5 a- a" g2 \, @. z一个简单的汇编模板:代码:
int a=10,b;asm("movl %1, %%eax; movl %%eax, %0;" :"=r"(b) /*输出部*/ :"r"(a) /*输入部*/ :"%eax" /*毁坏部*/ );
表示C语言里的“b=a;”。* J$ R' N' m$ e* s* `里边r表示使用任意寄存器,%0、%1表示使用两个寄存器,一般只能%0~%9共十个操作数,按输入输出部变量出现顺序进行映射。寄存器用两个百分号,是因为使用了%0%1这些数字使百分号有了特殊意义,所以在操作数出现的寄存器必, C+ }+ z, d* a U须用双百分表示。毁坏部里边的%eax表示eax寄存器在汇编代码块执行过程中会被改写,在执行前要保护好,这是提交给编0 z4 a6 w9 g8 }( ]2 k0 |2 B译器决定的。更多内容见《AT&T汇编语言与GCC内嵌汇编简介》