原创 C51和汇编的混合编程

2009-9-17 09:44 2919 8 8 分类: MCU/ 嵌入式

一、在C代码中嵌入汇编代码


#pragma asm …#pragma endasm


pragma只是用于C代码之间潜入asm代码,不是变相的混合编程技术,它不能直接调用其他文件(注意是文件)中的函数。
真正意义上的多模块编程,每个模块之间都可以相互调用、传递参数,这种编程就必须用寄存器、堆栈、内存区传递参数了。


二、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完成,我用这个例子,只是为了说明嵌入汇编的方法。

例1:

下面是C语言的主程序

#include <REG52.H>
#include <stdio.h>
extern char asm(char c,char b);
bit VAL;
void main (void)
{
  char out="0x49";
  char direct;
  char key;
  SCON = 0x50; /* SCON: mode 1, 8-bit UART, enable rcvr */
  TMOD |= 0x20; /* TMOD: timer 1, mode 2, 8-bit reload */
  TH1 = 0xfd; /* TH1: reload value for 9600 baud @ 11.0592MHz */
  TR1 = 1; /* TR1: timer 1 run */
  TI = 1; /* TI: set TI to send first char of UART */
  VAL=0;
  while (1)
  {
    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?ASMTEST SEGMENT CODE
?BI?_asm?ASMTEST SEGMENT BIT
PUBLIC ?_asm?BIT
PUBLIC _asm
RSEG ?BI?_asm?ASMTEST
?_asm?BIT:
VAL: DBIT 1
RSEG ?PR?_asm?ASMTEST
_asm:
MOV A,R7
MOV C,VAL
DJNZ R5,JP1
RRC A
JP1:
DJNZ R5,JP2
RLC A
JP2:
MOV 90h,A
MOV R7,A
MOV VAL,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调用汇编的用法,希望对大家有帮助。
PARTNER CONTENT

文章评论0条评论)

登录后参与讨论
EE直播间
更多
我要评论
0
8
关闭 站长推荐上一条 /3 下一条