如何在C语⾔中嵌⼊汇编
C语⾔怎样嵌⼊汇编:
TIPS:在编译过程汇中,汇编代码块是原封不动地送到汇编语⾔编译阶段的。
⼀、为什么会⽤到汇编?
1.为了提⾼速度和效率。不过这种情况很少了,现在C/C++编译器的优化很厉害了。
2.为了实现某些C语⾔中不具备、但为不同的机器所特有的功能。这是主要原因。
3.为了利⽤通⽤的汇编语⾔例程。也常会遇到。
⼆、何时使⽤汇编?
第⼀种情况是,绝对没有其他⽅法可以使⽤。
第⼆种情况出现在某个C语⾔程序的执⾏时间必须减少的时候。
三、如何嵌⼊汇编?
【Turbo 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++】:
格式:
__asm 汇编指令 [ ; ]
__asm { 汇编指令 } [ ; ]
asm前⾯是两条下划线,后⾯的⽅括号内容表⽰分号可有可⽆。
<⽰例1>
__asm mov al, 2
__asm mov dx, 0xD007
__asm out dx, al
<⽰例2>
__asm {
mov al, 2
mov dx, 0xD007
out dx, al
}
<⽰例3>
__asm mov al, 2  __asm mov dx, 0xD007  __asm out dx, al
<⽰例4>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__”的别名。
“__volatile__”  表⽰编译器不要优化代码,后⾯的指令保留原样,“volatile”是它的别名。  括号⾥⾯是汇编指令。
内嵌汇编语法如下:
__asm__(
汇编语句模板:
输出部分:
输⼊部分:
破坏描述部分)
⼀个简单的汇编模板:
<;代码⽰例>
int a=10,b;
asm("movl %1, %%eax;
movl %%eax, %0;"
:"=r"(b)          /*输出部*/
:"r"(a)          /*输⼊部*/
:"%eax"          /*修正部*/
);
表⽰C语⾔⾥的“b=a;”
⾥边r表⽰使⽤任意寄存器,%0、%1表⽰使⽤两个寄存器,⼀般只能%0~%9共⼗个操作数,按输⼊输出部变量出现顺序进⾏映射。      寄存器⽤两个百分号,是因为使⽤了%0%1这些数字使百分号有了特殊意义,所以在操作数出现的寄存器必须⽤双百分表⽰。
修正部⾥边的%eax表⽰eax寄存器在汇编代码块执⾏过程中会被改写,在执⾏前要保护好,这是提交给编译器决定的。
更多内容见
四:问题分析
void main()
{
__asm__("
jmp forward
backward:
popl %esi # Get the address of
# hello world string
movl $4, %eax # Do write system call
movl $2, %ebx
movl %esi, %ecx
movl $12, %edx
int $0x80
int3 # Breakpoint. Here the
# program will stop and
# give control back to
# the parent
forward:
call backward
汇编指令有多少个.string "Hello World\n""
)
;
}
使⽤ gcc –o hello hello.c来编译它。编译不过,提⽰双引号没匹配!!
解答:我编译运⾏过了,下⾯修改过的代码是可以的,内联汇编有要求:
1、  指令必须包括在引号⾥。
2、  如果包含的指令超过⼀条,那么必须使⽤新⾏字符分割汇编语⾔代码的每⼀⾏。通常还包含制表符帮助缩进汇编语⾔代码,使代码更容易阅读。
需要第⼆个规格是因为编译器逐字的取得asm段中的汇编代码,并且把他们放在为程序⽣成的汇编代码中。每条汇编语⾔指令都必须在单独的⼀⾏中--因此需要包含新⾏字符。
【上机实练】※
⽤C写的程序效率可能不如汇编,⽽且有些平台相关的指令必须⼿写,例如x86是端⼝I/O,⽽c语⾔就没有这个概念,所以in/out指令必须⽤汇编来写。
①gcc提供了⼀种扩展写法可以在C代码中使⽤内联汇编,最简单的格式是__asm__("assembly  code"); , 例如__asm__("nop"); ,nop这条指令让CPU空转⼀个周期,如果需要执⾏多条指令则⽤\n\t将各条指令隔开,例如:
__asm__("movl $1, %eax\n\t"
"movl $4, %ebx\n\t"
"int $0x80");
或者
__asm__("movl $1, %eax\n"
"movl $4, %ebx\n"
"int $0x80");
/
/visual code 环境测试可⽤
通常c代码中的内联汇编需要和c变量建⽴关联,需要⽤到完整的内联汇编样式:
__asm__(assembler template
: output operands              /* optional */
: input operands                /* optional */
: list of clobbered registers  /* optional */
);
这种格式由四部分组成,第⼀部分是汇编指令,和上⾯的例⼦⼀样,第⼆部分和第三部分是约束条件,第⼆部分指⽰汇编指令的运算结果要输出到那些c操作数中,c操作数应该是左值表达式,第三部分指⽰汇编指令需要从那些c操作数获取输⼊,第四部分是在汇编指令中被修改过的寄存器列表,指⽰编译器哪些寄存器的值在执⾏这条__asm__语句时会改变。后三个部分都是可选的,如果没有就空着值写个":"号。

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。