×
单片机 > 单片机程序设计 > 详情

51单片机asm与C混合编程

发布时间:2020-06-05 发布时间:
|

    一是源程序中直接混合嵌入,二是做成库函数调用,三是做成是中间文件在链接中加入。

  在单片机应用系统设计中,过去主要采用汇编语言开发程序。汇编语言编写的程序对单片机硬件操作很方便,编写的程序代码短,效率高,但系统设计的周期长,可读性和可移植性都很差。C语言程序开发是近年来单片机系统开发应用所采用的主要开发方式之一,C 语言功能丰富、表达能力强、使用灵活方便、开发周期短、可读性强、可移植性好。但是,采用C 语言编程还是存在着如对硬件没有汇编方便、效率没有汇编高、编写延时程序精确度不高等缺点,因而现在单片机系统开发中经常用到C 语言与汇编语言混合编程技术。混合编程技术可以把C 语言和汇编语言的优点结合起来,编写出性能优良的程序。单片机混合编程技术通常是,程序的框架或主体部分用C 语言编写,对那些使用频率高、要求执行效率高、延时精确的部分用汇编语言编写,这样既保证了整个程序的可读性,又保证了单片机应用系统的性能。

1、混合编程的基本方式
  C 语言与汇编语言混合编程通常有两种基本方法:在C 语言中嵌入汇编程序和在C 语言中调用汇编程序。
1.1 在C51 中嵌入汇编程序
  在C51 中嵌入汇编程序主要用于实现延时或中断处理,以便生成精练的代码,减少运行时间。嵌入式汇编通常用在当汇编函数不大,且内部没有复杂的跳转的时候。在单片机C 语言程序中嵌入汇编程序是通过C51 中的预处理指令# pragma asm/endasm 语句实现,格式如下:

# pragma ASM
;汇编程序代码
# pragma ENDASM

  通过# pragma asm 和# pragma endasm 告诉C51 编译器它们之间的语句行不用编译成汇编程序代码。

1.2  在C51 中调用汇编程序
  在C51 中调用汇编程序的方法应用较多,C 模块与汇编模块的接口较简单,分别用C51 与A51 对源程序进行编译,然后用L51 将obj 文件连接即可,关键问题在于C 函数与汇编函数之间的参数传递和得到正确返回值,以保证模块间的数据交换。
2、C51 与汇编程序的参数传递
  在C51 中嵌入汇编程序或调用汇编程序,其参数传递的过程是不一样的。
  2.1 在C51 中嵌入汇编程序的参数传递
  对于在C 语言程序中通过# pragma asm 和#pragma endasm 嵌入的汇编程序,C51 编译器在编译时只是将当中的汇编程序不编译,而不做其他任何处理,因此不存在函数调用时的参数传递和返回值问题。如果要在C 程序中和汇编程序中实现数据传递,可以通过变量或特殊功能寄存器来实现,例如,在C 程序的变量定义部分定义Z 变量,在C 语言程序和汇编程序中共同访问Z 变量,这样,C 语言程序可以通过Z 变量把参数传递给汇编程序,汇编程序可以通过Z 变量把参数返回给C语言程序。
2.2 在C51 中调用汇编程序的参数传递
  在C51 中调用汇编程序是通过函数调用的形式来实现的。由于C51 程序函数有明确的参数和返回值约定,因此在C51 中调用汇编程序进行参数传递时都必须严格遵守C51 函数的参数和返回值相关约定。
  在C51 中调用汇编程序进行参数传递关键在于要弄清C51 函数的参数传递规则。在C51 中调用汇编程序进行参数传递的方式有两种:一种是通过寄存器传递参数;一种是通过固定存储区传递。
2.2.1 通过寄存器传递参数。
  Franklin C51 规定调用函数最多可通过51 单片机的工作寄存器传递3 个参数,余下的通过固定存储区传递。可以用“NOREGPARMS”命令取消用寄存器传递参数,如果用寄存器传递参数取消或参数太多,参数通过固定存储区传递。用寄存器传递参数的函数,在生成代码时被Cx51 编译器在函数名前加上一个下划线“_”的前缀,在固定存储区传递参数的函数则没有下划线。不同的参数用到的寄存器不一样,不同的数据类型用到的寄存器也不同。通过寄存器传递的参数如表1 所示。



 表1 中,int 型和long 型数据传递时高位数据在低位寄存器中,低位数据在高位寄存器中;float型数据满足32 位的IEEE 格式,指数和符号位在R7 中;通用指针存储类型在R3 中,高位在R2 中。函数参数传递举例情况如表2 所示。


2.2.2  通过固定存储区传递。
  用固定存储区传递参数给汇编程序,参数段首地址用段名“ ? function-name ? BYTE”和“ ?function-name ? BIT”保存,function-name 为函数的名称,其中“, ? function-name ? BIT”保存位参数段首地址,“ ? function-name ? BYTE”保存别的参数段首地址,即使通过寄存器传递参数,参数也将在这些段中分配空间,参数按声明的先后在每个段中顺序保存。
  用做参数传递的固定存储区可在内部数据区或外部数据区,这由存储模式决定。 Small 模式的参数段用内部数据区,Compact 和Large 模式用外部数据区。
2.2.3  函数返回值。
  函数返回值通常用寄存器传递,表3 列出了可能的返回值和所用的寄存器。

 

3、C51 中嵌入汇编程序的实现方法
  通常,在C51 程序中嵌入汇编程序的处理方法如下:
第一步,在C 文件中以如下方式嵌入汇编程序。

# pragma ASM
;汇编程序
# pragma ENDASM

第二步,在keil C51 软件的Project 窗口右键单击嵌入汇编程序的C 文件,选择“Options for ?”,点击右边的“Generate Assembler SRC File”和“Assemble SRC File”,使检查框由灰色变成黑色(有效) 状态。


第三步,根据选择的编译模式,把相应的库文件(如Small 模式时, 是Keil \ C51 \ Lib \ C51S。Lib) 加入工程中, 该文件必须作为工程的最后文件。

库文件与编译模式的关系如下:
C51S.LIB - 没有浮点运算的Small model
C51C.LIB - 没有浮点运算的Compact model
C51L.LIB - 没有浮点运算的Large model
C51FPS.LIB - 带浮点运算的Small model
C51FPC.LIB - 带浮点运算的Compact model
C51FPL.LIB - 带浮点运算的Large model
第四步,编译,即可生成目标代码。

keil软件中c编程如何制作一个库函数并在其他地方随意调用?

在项目的Output设置中选择输出lib而不是可执行目标文件。
4、C51 中调用汇编程序的实现方法
  为了能够在C 语言中调用汇编程序,要求汇编程序的编写必须符合C 语言的相关命名规则。
  C51 程序在调用汇编程序时,除了前面参数传递的相关规则外,函数及其相关段也需要满足一定的规则。
  一个C51 源程序模块被编译后,其中的每一个函数以“ ? PR ? 函数名? 模块名”为名的命名规则被分配到一个独立的CODE 段。 例如,如果模块“FUNC51”内包含一个名为“func”的函数, 则其CODE 段的名字是“ ? PR ? FUNC ? FUNC51”,如果函数中还包含有data 和bit 对象的局部变量,编译器将按“ ? 函数名? BYTE 和? 函数名? BIT”命令规则建立一个data 和bit 段,它们代表所要传递参数的起始位置,其偏移值为零。段内代码与数据定义也遵循一定的规则。这些段是公开的,它们的地址可被其他模块访问。另外,这些段被编译器赋予“OVERLAYABLE”标志,其可被L51 连接P定位器做覆盖分析。
  下面是一个简单的C51 程序编译时形成的汇编程序。
  C 语言源程序如下:

#define uchar unsigned char
uchar max(uchar x ,uchar y) {
uchar z ;
z = (x > = y) ? x :y ;
return(z) ;
}

  汇编后形成的SRC 文件(只须扩展名改为.a51就变成汇编程序) 如下:

NAME  A1              ;定义模块名称
?PR?_max?  A1  SEGMENT CODE  ;定义程序代码
  PUBLIC  _max          ;定义公共符号
; #define uchar unsigned char
;uchar max(uchar x ,uchar y)
  RSEG   ?PR?_max?  A1 ;程序代码段
_max:                 ;起始地址
  USING  0
    ;SOURCE LINE # 2
; ??Variable ’y ? 041’assigned to Register ’R5’??
; ??Variable ’x ? 040’assigned to Register ’R7’??
; {
    ; SOURCE LINE # 3
; uchar z ;
; z = (x > = y) ? x :y ;
    ; SOURCE LINE # 5
  MOV  A, R7          ;R7 中为第二个字节参数
  CLR  C
  SUBB  A, R5          ;R5 中为第一个字节参数
  JC   ?C0001
  SJMP  ?C0002
?C0001:
  MOV  R7, AR5         ;R7 中为返回值
?C0002:
; ??Variable ’z ? 042’assigned to Register ’R7’??
; return(z) ;
    ; SOURCE LINE # 6
    ; SOURCE LINE # 7
; }
?C0003:
  RET
; END OF - max
  END

  可以看出,要编写为C51 调用的汇编程序,除了参数必须按前面规定的寄存器或存储器传送外,程序格式也有相应的规则。这些规则比较繁琐,在实际处理中往往按下面方式处理:
第一步,先用C 语言程序编写出程序框架,如文件名为a1.c (注意参数) 。
第二步,在keil C51 的Project 窗口中用右键单击该C 语言文件,在右键菜单中选择“Options for?”,点击右边的“Generate Assembler SRCFile”和“Assemble SRC File”,使检查框由灰色变成黑色(有效) 状态。
第三步,根据选择的编译模式,把相应的库文件(如Small 模式时,是Keil \ C51 \ Lib \ C51S.Lib) 加入工程中,该文件必须作为工程的最后文件。 库文件与编译模式的关系如前面所述。
第四步,编译后将会产生一个SRC 的文件,将这个文件扩展名改为ASM。这样就形成了可供C51程序调用的汇编程序。随后可在该文件的代码段中加入所需指令代码。
第五步,将该汇编程序与调用它的主程序一起加到工程文件中,这时工程文件中不再需要原来的C 语言文件和库文件,主程序只需要在程序开始处用EXTERN 对所调用的汇编程序中的函数作声明,主程序中可调用汇编程序中的函数。

5、总结
  在单片机C 语言与汇编语言混合编程中,主要应注意相应的使用方法和参数传递。在单片机应用系统设计中,采用单片机C 语言与汇编语言混合编程的方法,既可提高程序开发的效率,又可以很方便的操作单片机硬件;既能保证整个程序的可读性,又能保证对硬件访问的精确性。适当的用好C 语言与汇编语言混合编程技术,就可以开发出性能较好的单片机应用程序。

 

SEGMENT、RSEG与CSEG的解释

1) SEGMENT
       用于声明一个段名,属于汇编命令。
       其前面的符号表示段名,后面的符号为段的类型,如CODE为程序代码段,DATA代表内部RAM段。
RSEG DATA_SEG
声明以下内容应位于可重定位的DATA_SEG段
2) RSEG xxx
      声明以下的语句应位于可重定位的xxx段,在编译时不进行定位,只有在链接时才确定其固定地址。

CSEG AT 00000H
声明以下代码位于开始地址是0000H的CODE段

3) CSEG AT  yyyH
       声明以下代码应位于CODE的绝对地址yyyH,在编译时已进行定位,不可重定位。通常用于中断向量和复位向量声明。

(1) 伪指令org用来规定目标程序存放单元的偏移量。
比如,如果在源程序的第一条指令前用了如下指令:
org 200h 那么,汇编程序会把指令指针的ip的值设成200h,即目标程序的第一个字节放在200h处,后面的内容则顺序存放,除非遇上另一个org 语句。

(2) CSEG AT 定义绝对地址,而不是相对地址,
CSEG [AT absolute address] Define an absolute segment within the code address space.

当有C51与汇编混合编程时而要定位某段汇编代码,最好使用CSEG AT XXXH。

 

使用以上编译指令的好处:
       通常在编写一个较大的项目时,都会将不同功能的程序划分成不同的模块。以便于功能调试以及代码重用,从而提高工作效率和程序稳定性,减少程序bug。例如在一个项目中使用到AT24Cxx存储器,我们可以将AT24C相关的驱动代码写在一个单独的模块(文件)中,在模块程序起始的地方用RSEG CODE_SEG声明代码可重定位。那么此程序会由编译系统自动决定其链接地址,在其他模块程序修改后不会对其产生任何影响。并且此模块可以很方便地应用到其他使用AT24C的项目中去。
       假如我们没有使用RSEG CODE_SEG,而是用ORG来声明,那么在其他模块程序修改后我们可能需要再次修改其地址值,稍有不慎可能会造成地址重叠,或者是空间浪费。

 

段名的开头为PR,是为了和C51内部命名转换兼容,命名转换规律如下:
-------+--------------+------------------------------
段前缀 | 存储类型 | 说明
-------+--------------+------------------------------
?PR? | program | 可执行的程序代码
?CO? | code | 程序存储区的常数数据
?BI? | bit | 内部数据区的位数据
?BA? | bdata | 内部数据区的可位寻址数据
?DT? | data | 内部数据区
?FD? | far | FAR存储区(RAM空间)
?FC? | const far | FAR存储区(常数ROM空间)
?ID? | idata | 间接寻址内部数据区
?PD? | pdata | 外部数据区的分页数据
?XD? | xdata | XDATA存储区(RAM空间)
?XC? | const xdata | XDATA存储区(常数ROM空间)
-------+--------------+------------------------------





『本文转载自网络,版权归原作者所有,如有侵权请联系删除』

热门文章 更多
51单片机CO2检测显示程序解析