TIPS:在编译过程汇中,汇编代码块是原封不动地送到汇编语言编译阶段的。
, O8 Q: y3 /# C( S# r" [
一、为什么会用到汇编? ! N3 M2 ?7 D* {
1.为了提高速度和效率。不过这种情况很少了,现在C/C++编译器的优化很厉害了。 ) I5 N2 {$ f0 J- K
2.为了实现某些C语言中不具备、但为不同的机器所特有的功能。这是主要原因。
为了利用通用的汇编语言例程。也常会遇到。 7 S2 J$ _+ ] }8 L
二、何时使用汇编? : b, K* q0 B- |
第一种情况是,绝对没有其他方法可以使用。 ( l" D8 T6 M6 v& c% t: E: s
第二种情况出现在某个C语言程序的执行时间必须减少的时候。
三、如何嵌入汇编? - ~2 I+ ]- ? s( N# o u% E
Turbo C : * I: k$ S! a. W
1.使用预处理程序的伪指令#asm和#endasm,#asm用来开始一个汇编程序块,而#endasm指令用于该块的结
束。 : ~' `' a7 W2 w
例:
一、为什么会用到汇编? ! N3 M2 ?7 D* {
1.为了提高速度和效率。不过这种情况很少了,现在C/C++编译器的优化很厉害了。 ) I5 N2 {$ f0 J- K
2.为了实现某些C语言中不具备、但为不同的机器所特有的功能。这是主要原因。
为了利用通用的汇编语言例程。也常会遇到。 7 S2 J$ _+ ] }8 L
二、何时使用汇编? : b, K* q0 B- |
第一种情况是,绝对没有其他方法可以使用。 ( l" D8 T6 M6 v& c% t: E: s
第二种情况出现在某个C语言程序的执行时间必须减少的时候。
三、如何嵌入汇编? - ~2 I+ ]- ? s( N# o u% E
Turbo C : * I: k$ S! a. W
1.使用预处理程序的伪指令#asm和#endasm,#asm用来开始一个汇编程序块,而#endasm指令用于该块的结
束。 : ~' `' a7 W2 w
例:
代码:
mul(a,b)
int a,b;
{
#asm
mov ax,word ptr 8[bp]
imul ax word ptr 10[bp]
#endasm
}
2.使用asm语句
格式:asm<汇编语句>
格式:asm<汇编语句>
代码:
mul(a,b)
int a,b;
{
asm mov ax,word ptr 8[bp]
asm imul ax word ptr 10[bp]
}
注意:asm行后面没有分号
+ P5 j/ g/ b* I0 [- Y; h7 o
Visual C++ :
格式: 2 K6 Z4 C% R; K7 J( d, y
__asm 汇编指令 [ ; ]
__asm { 汇编指令 } [ ; ]
asm前面是两条下划线,后面的方括号内容表示分号可有可无。 7 j0 |5 |% ]( Y% I5 p
使用方法: 8 /- t$ [& n& d+ Z6 g: l
一条一条地用
Visual C++ :
格式: 2 K6 Z4 C% R; K7 J( d, y
__asm 汇编指令 [ ; ]
__asm { 汇编指令 } [ ; ]
asm前面是两条下划线,后面的方括号内容表示分号可有可无。 7 j0 |5 |% ]( Y% I5 p
使用方法: 8 /- t$ [& n& d+ Z6 g: l
一条一条地用
代码:
__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 :
1 ~& R' X8 N) ?; z$ }
由于内容比较多,所以简单说一下,具体都在后面的附件里。 $ p3 S0 H! }6 a* I" N; J+ O
用到的关键字 / L7 N6 `: ^5 }
“__asm__” 表示后面的代码为内嵌汇编,“asm”是“__asm__”的别名。 6 G/ B* K8 H. ]+ C, y# z& a, k
“__volatile__” 表示编译器不要优化代码,后面的指令保留原样,“volatile”是它的别名。
括号里面是汇编指令。
内嵌汇编语法如下:
__asm__( 1 Y7 Y- n* ]8 o' a) c: a8 M5 {* C
汇编语句模板: * q7 a. ^* Y2 b6 l% I& M$ h; v
输出部分: 6 l* ?! I0 P2 x# p- R% G( ?
输入部分:
破坏描述部分) 3 w9 Y; }: R8 ]# x8 b9 w
一个简单的汇编模板:
由于内容比较多,所以简单说一下,具体都在后面的附件里。 $ p3 S0 H! }6 a* I" N; J+ O
用到的关键字 / L7 N6 `: ^5 }
“__asm__” 表示后面的代码为内嵌汇编,“asm”是“__asm__”的别名。 6 G/ B* K8 H. ]+ C, y# z& a, k
“__volatile__” 表示编译器不要优化代码,后面的指令保留原样,“volatile”是它的别名。
括号里面是汇编指令。
内嵌汇编语法如下:
__asm__( 1 Y7 Y- n* ]8 o' a) c: a8 M5 {* C
汇编语句模板: * q7 a. ^* Y2 b6 l% I& M$ h; v
输出部分: 6 l* ?! I0 P2 x# p- R% G( ?
输入部分:
破坏描述部分) 3 w9 Y; }: R8 ]# x8 b9 w
一个简单的汇编模板:
代码:
int a=10,b;
asm("movl %1, %%eax;
movl %%eax, %0;"
:"=r"(b) /*输出部*/
:"r"(a) /*输入部*/
:"%eax" /*毁坏部*/
);
表示C语言里的“b=a;”。
9 |1 ]1 m2 d. A$ o
里边r表示使用任意寄存器,%0、%1表示使用两个寄存器,一般只能%0~%9共十个操作数,按输入输出部变量出现顺序进行映射。 ' V# C! @3 m! l* R' X
寄存器用两个百分号,是因为使用了%0%1这些数字使百分号有了特殊意义,所以在操作数出现的寄存器必
须用双百分表示。
毁坏部里边的%eax表示eax寄存器在汇编代码块执行过程中会被改写,在执行前要保护好,这是提交给编 4 z+ @) J# U7 o$ /6 T( }* z
译器决定的。
里边r表示使用任意寄存器,%0、%1表示使用两个寄存器,一般只能%0~%9共十个操作数,按输入输出部变量出现顺序进行映射。 ' V# C! @3 m! l* R' X
寄存器用两个百分号,是因为使用了%0%1这些数字使百分号有了特殊意义,所以在操作数出现的寄存器必
须用双百分表示。
毁坏部里边的%eax表示eax寄存器在汇编代码块执行过程中会被改写,在执行前要保护好,这是提交给编 4 z+ @) J# U7 o$ /6 T( }* z
译器决定的。