在C中嵌入汇编

时间:2023-03-08 19:04:46
在C中嵌入汇编

早前公布了C和汇编混编的温度控制器程序,收到一些朋友的询问,他们无法在自己程序中使用我的18B20的汇编子程序或无法正常通过混编后的程序编译。

其实在KEIL中嵌入汇编的方法很简单。如图一,在C文件中要嵌入汇编的地方用#pragma asm和#pragma endasm分隔开来,这样编译时KEIL就知道这中间的一段是汇编了。

在C中嵌入汇编

在有加入汇编的文件中,还要设置编译该文件时的选项

在C中嵌入汇编

 Generate Assembler SRC File 生成汇编SRC文件

 Assemble SRC File 封装汇编文件

(如图三的状态为选中)

选上这两项就可以在C中嵌人汇编了,设置后在文件图示中多了三个红色的小方块。

在C中嵌入汇编

图3

  为了能对汇编进行封装还要在项目中加入相应的封装库文件, 在笔者的项目中编译模式是小模式所以选用C51S.LIB。这也是最常用的。这些库文件是中KEIL安装目录下的LIB目录中。 加好后就可以顺利编译了。(注:我只在7.0以上版本使用过)

在C中嵌入汇编  在C中嵌入汇编

C51编译器能对C语言程序进行高效率的编译,生成高效简洁的代码,在大多数的应用场合,采用C语言编程即可完成预期的任务,但是,在有些场合还是会用到汇编,例如在下面的几种情况下,采用汇编可能会有很多好处:

1、已有程序的移植:在单片机领域工作很久的工程人员可能会保留有很多的早期用汇编语言编制的程序模块,并且这些模块已经经过实际应用的验证,如果重新用C编程,可能工作量很大,这时就可以用嵌入汇编的方式把以前的汇编模块植入新的应用,可以明显的加快开发的进度。

2、局部功能需要足够短的执行时间:在有些应用中,部分的功能模块需要有很高的执行效率,而有些汇编的指令在C中没有对应的指令,这给我们对单片机的高效操作带来困难,嵌入汇编可是我们的程序执行更有效率。

3、对一些特定地址进行操作:在C中我们要对特定地址进行读写,一般用以下两种方式:用_AT_指令定义变量;定义指向外部端口或数据地址的指针;在汇编中只需要使用MOVX A,@DPTR或MOVX @DPTR,A就可以了,这样可以增强程序的可读性。

4、其他的需要汇编的应用:在这里我们不可能举出所有可能要用汇编的例子,在你的应用中,你可能在一个或多个应用中感到C语言的不足,而需要用到汇编指令,请你记住,可以在C中嵌入汇编子程序,这对你的程序非常有用。

首先介绍一下调用汇编的参数传递规则,见下表

传递的参数

char、1字节指针

int、2字节指针

long、float

一般指针

第一个参数

R7

R6,R7

R4~R7

R1,R2,R3

第二个参数

R5

R4,R5

 

R1,R2,R3

第三个参数

R3

R2,R3

R1,R2,R3

在下面的例子中我们首先给出一个C程序调用汇编的例子,先说明一下,在这个例子中,你完全可以用C完成,我用这个例子,只是为了说明嵌入汇编的方法。电子园51单片机学习网q\b7h3V

例1:下面是C语言的主程序

#include <REG52.H>
#include <stdio.h>
extern char asm(char c,char b);
bit VAL;
void main (void)
{
  charout=0x49;
  chardirect;
  charkey;
  SCON = 0x50;/* SCON: mode 1, 8-bit UART, enable rcvr */
//10位异步接收器,可变。REN=1,允许接收数据
TMOD |= 0x20;/* TMOD: timer 1, mode 2, 8-bit reload */
//定时器1工作于模式2
TH1 = 0xfd;/* TH1: reload value for 9600 baud @ 11.0592MHz */
  TR1 = ;/* TR1: timer 1 run */
  TI = ;/* TI: set TI to send first char of UART */
  VAL=;
  )
  {
    key=getchar();
    if(key=='R')
    {
      direct=0X01;
      out=asm(out,direct);/*汇编子程序调用*/
      printf ("Right rotate\n");
    }
    if(key=='L')
    {
      direct=0X02;
      out=asm(out,direct);/*汇编子程序调用*/
      printf ("Left rotate\n");
    }
    printf("%bx\n",out);
  }
}

下面是汇编的子程序(文件名称asmtest.asm)

NAME ASM
?PR?_asm?ASMTESTSEGMENT CODE
?BI?_asm?ASMTESTSEGMENT BIT
PUBLIC?_asm?BIT
PUBLIC_asm
RSEG?BI?_asm?ASMTEST
?_asm?BIT:
VAL:DBIT1
RSEG?PR?_asm?ASMTEST
_asm:MOV A,R7
MOV C,VAL
DJNZ R5,JP1
RRC A
JP1:DJNZ R5,JP2
RLC A
JP2:MOV90h,A
MOV R7,A
MOVVAL,C
RET
END

现在对这个例子简短的说明,这个例子是用来驱动一个不带细分的三相步进电机,用的是循环移位的指令来实现的,由于一个字节是8位,如果算上进位位,共9位,通过付值,可以得到这样的数100100100,大家可以看到任意取中间的3位,相邻的每3位与它都一样,这样我们就可以去中间的3位输出到端口去驱动一个三相的步进电机,通过对这个9位的二进制数进行循环移位,可以实现电机的步进,在51系列单片机中有两条指令可以帮助我们进行循环移位操作,这就是RRC A和RLC A指令,我们只要把第九位保存好,在需要移位操作时,把它付给PSW中C,再执行循环移位就可以了。

在这个例子中,我发现了几个需要注意的地方:

1、在C程序中,我不能把VAL变量设为extern类型,否则在连接时会有警告,导致数值不能传递

2、在汇编模块中,不能把SEGMENT BIT段置为OVERLAYABLE即可覆盖段,如果置为可覆盖段,那么在进入汇编模块时VAL变量值丢失

3、如果把VAL变量作为函数的参数传递,出现在返回后在执行printf函数后变量值丢失;

上面是本人写的一个C程序调用汇编的小例子,在本例中的几个printf函数是为了便于在Keil C51中模拟调试时观察运行结果的,在实际应用中可以将这几条去掉,这只是一个简短的示例,目的在于介绍一下C调用汇编的用法,希望对大家有帮助,同时由于本人水平有限,在程序中有些地方可能仍有不周到之处,欢迎广大单片机高手不吝指出。