如何在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寄存器在汇编代码块执行过程中会被改写,在执行前要保护好,这是提交给编译器决定的。
       更多内容见《AT&T汇编语言与GCC内嵌汇编简介》

四:问题分析

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__语句时会改变。后三个部分都是可选的,如果没有就空着值写个":"号。

  • 7
    点赞
  • 45
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值