我们在第一章中已经详细介绍了中档 PIC 单片机的 35 条指令,源程序的编写主要就是用这些基本的指令实现你的控制任务。但为了增加源程序的可读性和可维护性,我们引入了伪指令的概念。伪指令本身不会产生可执行的汇编指令,但它们可以帮组“管理”你编写的程序,其实用性和必要性绝不亚于 35 条正真的汇编指令。我们在此着重介绍最常用的几种伪指令。
#include 或 include
#include 伪指令的作用是把另外一个文件的内容全部包含复制到本伪指令所在的位置。被包含复制的文件可以是任何形式的文本文件,当然文件中的内容和语法结构必须是MPASM 能够识别的。最经常被“include”的是针对 PIC 单片机内部特殊功能寄存器定义的包含头文件,在 MPLAB 安装后它们全部放在路径“ C:\Program Files\MPLAB IDE\MCHIP_Tools”下,每一个型号的 PIC 单片机都有一个对应的预定义包含头文件,扩展名是“.inc”。除了一些符号预定义文件,你也可以把现有的其它程序文件作为一个代码模块直接“包含”进来作为自己程序的一部分。见例 3-01。
#include<p16f877a.inc> ;把预定义的 PIC16F877A寄存器符号包含到此处
#include”math.asm” ;把现有的程序文件包含进来作为自己代码的一部分
例 3-01
请注意被包含文件的引用方式。一种是<>尖括号引用,这种引用意味着让编译器去默认的路径下寻找该文件,MPASM 默认的寄存器预定义文件存放路径即为上面提及的MPLAB 安装后的目录;另一种是””双引号引用,这种引用方式的意思是指示编译器从引号中指定的全程文件路径下寻找该文件。例 3-01 中”math.asm”没有指定路径,即意味着在当前项目路径下寻找 math.asm 文件。如果编译器找不到被包含的文件,将会有错误信息告知。
请在你的源程序中尽量用 MPLAB标准头文件定义的寄存器符号。一来这些被定义的寄存器符号和芯片数据手册上的描述一一对应,理解起来即直观又容易;二来如果用你自己定义符号就缺乏一个大家能一起交流的标准平台,其他人要解读你的代码时将费时费力。故例
3-01 中的首行#include 包含引用伪指令可以说是 PIC 单片机程序编写时的标准必备。
list
list 伪指令可以设定程序编译时的一些信息,例如所选单片机的型号,编译时选择的缺省数制等。例如:
listp=16f877a, r=DEC ;单片机型号为PIC16F877A,无特别指明的数字为十进制数
例 3-02
如果程序开发时使用项目管理的模式,则所有 list 伪指令可以描述的参数项都可以在项目的设定选项中通过对话框的形式设定并保存。在此只需对 list 伪指令稍作了解即可。
__config
此伪指令的重要作用是把芯片的配置字设定在源程序中,请参阅 2.5 节的详细说明。建议大家尽量用此伪指令把芯片的配置字写在程序中。
__idlocs
PIC 单片机中有一处非常特殊的标记单元。它独立于任何其它存储器,唯一的作用就是作为一个标记。此标记值无法用软件读到,读取和写入的方法只有通过编程器实现。此标记值没有读保护,你可以利用它存放程序的版本或日期等信息。如果需要,则可以用伪指令__idloc 在程序中定义具体的值。
__idloc0x1234 ;设定芯片的标记值为0x1234,注意前面有两个下划线符
例 3-03
和__config 伪指令定义的配置字一样,用__idloc 定义的芯片标记值在最后也会存放在HEX文件中,这就要求编程器能够解析它。
errorlevel
errorlevel 的用途是控制编译信息的输出显示。编译器在编译你的源程序时会提供很多信息,有些信息是你必须要处理的,例如错误信息(Error),只要有错误信息存在,你的程序将永远无法完成编译;有些可能只需要关注,例如警告信息(Warning);也有一些可能你根本就不感兴趣,它们只是一些提示信息(Message)而已。注意出现警告和提示信息时将不会中止编译器的编译工作,你的程序将被编译并最终产生 HEX文件。图 3-14 中显示了一个程序编译后的各种信息实例,其中既有错误信息,也有警告和提示信息。我们可以用errorlevel 伪指令来控制输出信息的级别,或刻意关闭/打开一些提示信息。
图 3-14
编译信息的输出显示级别有三种,分别是 0、1 和 2。级别 0 代表显示所有信息,包括各种错误、警告和提示信息,如图 3-14 所示;级别 1 代表显示错误和警告信息,忽略提示信息;级别 3代表只显示错误信息而忽略警告和提示信息。在任何一个大的级别上还可以对
某些信息单独设定显示或关闭。每个信息都有一个识别标号,见图 3-14 中信息项“[]”中的数字,打开或关闭某类信息只需在 errorlevel 伪指令中引用信息识别标号,并在其前面用“+”或“-”号,即代表打开或关闭这一类信息,例如:
errorlevel0, -302, -305 ;显示所有信息,但不需要302 和 305 这两类提示信息
errorlevel1, +305 ;显示错误和警告信息,但同时还要关注 305类的提示信息
例 3-04
#define / #undefine
#define 的作用是定义常数符号,即用一个符号变量替换另一个符号串或变量。被替换的可以是任意字母数字组成的符号但替换者本身不能是一个纯数字。例如:
#defineDELAY_TIME 1000 ;定义常数符号,即用DELAY_TIME符号代替 1000
#defineKEY1 PORTB,7 ;用KEY1符号代替端口 PORTB 的第7 引脚
例 3-05
用#define 伪指令定义符号后,可使程序中的变量或指令变得更具实际意义,也使程序变得更易维护。指令“btfss PORTB,7”和“btfss KEY1”在事先用了例 3-05 中的#define 后编译的结果是一样的,但明显地后者看起来更容易理解,一看就知道这是在测试编号为KEY1 的一个按键。而且如果你的硬件设计改动了 KEY1 所接的单片机引脚,只要改动这一处#define 重新定义引脚位置,程序的其它部分无需任何修改,再编译一次即可得到更新后的软件代码。一个好的编程习惯是事先把一些代表实际意义的变量、单片机的输入输出引脚在硬件电路中的实际功能等用#define 伪指令定义成简单直观的符号名字,然后在程序中直接用其符号名字而不用简单机械的数字形式。替换的工作由编译器在编译时自动完成。它会先扫描你的源程序代码,把事先#define 的符号名改回成被替换的字符串,然后再继续编译生产机器码。
equ
equ 顾名思义是“等于”的意思,其作用和#define伪指令有点类似,也是用一个符号名字替换其它数字变量,但它只能替换立即数。如果要替换一个符号名字,则此符号名必须事先用#define或 equ 伪指令已经定义替换了一个立即数。例如:
#define MyCount 0x70 ;定义 MyCount 符号替换立即数 0x70
w_tempequ 0x20 ;符号名w_temp等于0x20
count1equ MyCount ;符号名count1等同于MyCount
;如果 MyCount 没有事先定义则会产生一个错误
例 3-06
在绝对定位的编程模式中 equ 被经常用于定义用户自己的变量,即用一个符号名代替一个固定的存储单元地址,上例 3-06 中的 w_temp 定义即属于此类。用 equ 方式定义的符号在汇编后可以生成相关的调试信息,可以通过各种变量观察的方式显示此符号所代表的内存地址处的数据内容,但用#define 方式定义的符号则不能产生调试信息。要注意 equ 伪指令本身并没有限定所定义的一定是一个变量地址,它只是一个简单的符号和数字替换而已,其意义必须和具体的指令结合才能确定,如下例 3-07 中对符号 w_temp 的理解。
w_tempequ 0x20 ;符号名w_temp等于0x20
movlw 0x55 ;W=0x55
movwf w_temp ;把 W的值送给变量w_temp, (0x20单元内容=0x55)
movf w_temp, w ;把w_temp 单元内容送W, (W=0x55)
movwf FSR ;把 W的内容送 FSR, (FSR=0x55)
movlw w_temp ;把w_temp 所代表的立即数即地址值送给W, (W=0x20)
movwf FSR ;让FSR 指针指向w_temp, (FSR=0x20而不是0x55)
例 3-07
cblock / endc
用 equ 伪指令可以给一个符号变量分配一个地址。但在一个程序设计过程中往往需要定义很多变量,你当然可以给每一个变量逐个用 equ 的方法分配一个地址空间。但如果变量很多,这样做就显得非常麻烦,你必须自己安排每个变量的地址,小心不能出现地址重叠;若要在已定义分配好的变量间插入新的变量,那就必须重新逐个安排随后变量的地址等等。cblock/endc 伪指令可以轻松解决有很多变量定义的场合出现的这些问题,我们把它叫作变量块连续定义。具体用法如下:
cblock 伪指令声明变量块的起始地址,endc 伪指令声明变量块定义结束,cblock/endc中间可以插入任意多的变量声明。其地址编排由编译器自动计算:第一个变量地址分配从起始地址开始,然后按所声明变量保留的字节数自动分配后面变量的地址,变量所需保留的字
节数用“:”加后面的数字表示,如果只有一个字节“:1”可以省略不写。以例 3-08来说明:
cblock0x20 ;变量定义起始地址为0x20
w_temp ;w_temp地址为0x20,占一个字节
status_temp ;status_temp地址为 0x21,占一个字节
buffer:8 ;buffer的起始地址为 0x22,并保留 8个字节单元
var1 ;var1的地址为 0x2a,占一个字节
var2 ;var2的地址为 0x2b,占一个字节
endc;结束变量连续定义
例 3-08
用 cblock 方式定义的变量和用 equ 方式定义的变量一样在汇编后可以生成相关的调试信息,可以通过各种变量观察的方式显示此符号所代表的内存地址和其中的数据内容,所以实际编程时一般无需关心计算每个变量的具体地址。程序员要注意的用这种方式连续定义很多变量时不要让变量块跨越所处 bank 的边界。你可以在 cblock 中随意插入新定义的变量,或通过改变起始地址的方式使变量块整个挪到其它内存地址处,地址的更新由编译器代劳。
org
org 用以定义程序代码的起始地址,通过此伪指令你可以把程序定位到任何可用的程序空间,它实现的是程序代码绝对定位,如例 3-09:
org0x0000 ;定义复位入口地址,以下指令从地址0x0000 开始
goto main ;
org0x0004 ;定义中断入口地址,以下指令从地址0x0004 开始
movwf w_temp ;保存 w
;... ;其它中断服务代码
org0x0800 ;定义 page1 的起始地址,以下指令代码放在 page1
Sub1 return
例 3-09
只要你认为代码需要确定放在某一特定地址处,在程序的任何地方都可以用 org 伪指令重新定义存放的起始地址,且地址顺序可以任意编排。但要注意的是若干个确定起始地址的代码块不能相互重叠,否则编译器会报错,无法得到正确结果。若用可重定位方式开发指令代码时一般不能用 org 伪指令绝对定位代码。
dt
dt的作用是定义表格数据。在第一章介绍基本汇编指令时已经提到,PIC 单片机实现表格定义的最基本指令是“retlw xx”,表格中的每一个字节数据都以指令“retlw”的形式出现。若表格较大,就需要很多“retlw”指令,比较麻烦,可读性也差。这时我们可以用此“dt”伪指令替代“retlw”实现很多数据的表格定义。如例 3-10:
Table addwf PCL,f ;PC 相对寻址查表
dt0 ;retlw 0
dt 1, 2, ’3’ ;retlw 1
;retlw 2
;retlw 0x33 (’3’的ASCII 码)
dt”ABC” ;retlw ’A’
;retlw ’B’
;retlw ’C’
例 3-10
de
de 伪指令可以让你在源程序中定义片内 EEPROM 的初值。毫无疑问,该条伪指令只适用于那些内含 EEPROM数据存储器的单片机,例如:PIC16F87x、PIC16F62x 等等。在中档PIC 单片机中,除了 PIC16F7x 系列外,其它 Flash 型的单片机都有片上 EEPROM,只是字节数多少的问题。你可以编写代码在程序运行时来设定片内 EEPROM数据区的初值,但此EEPROM 区还可以在芯片编程烧写时通过编程器对其设定初值。对编程器而言 EEPROM 数据区是程序空间的延伸,它有个特别的编程起始地址 0x2100。基于这一前提,我们可以在源程序中利用“org”和“de”伪指令定义片内 EEPROM 数据的初值,这样最后得到的 HEX文件被烧入到单片机内后,EEPROM 区就同时被特定数据所初始化。看例 3-11 的实例
org 0x2100 ;特殊的程序空间起始地址
;编程器能识别此地址作为EEPROM数据区的起始地址
de 0, 1, 2, 3 ;EEPROM地址单元[0]=0, [1]=1, [2]=2, [3]=3
de”ABCD” ;[4]=0x41, [5]=0x42, [6]=0x43, [7]=0x44
例 3-11
按例 3-11 所示的定义,芯片完成编程烧入后,其内部 EEPROM 区从 0x00 单元开始被分别初始化成 0x00、0x01、0x02、0x03、0x41、0x42、0x43、0x44。其它未被初始化的 EEPROM单元全部是 0xff。
要注意并不是所有的编程工具都能支持此法定义的 EEPROM 初始值烧入。能直接挂接在 MPLAB 环境下的 Microchip 原厂或兼容的编程工具都可以支持“de”伪指令定义的EEPROM 初值烧入,但其它第三方生产的编程工具就不一定,使用前请咨询编程器的生产厂商。
fill
fill 伪指令可以实现对程序空间连续自动填充某一特定的指令数据,被填充的可以是一个立即数(实际肯定代表某一条指令),也可以是一条形象的汇编指令。基本上在一个设计中都有一些程序空间没有写上具体的指令编码(空白处),在单片机正常运行时这些地方的指令是不会被执行到的。但在有干扰的情况下程序跑飞正好落在这些非法指令处时,就有必要设置软件陷阱捕捉这些非法跳转,让程序恢复正常运行。如果要程序员一个一个地址去分析哪里有空的指令单元然后又用特殊指令一条一条填入,这是根本行不通的。fill 伪指令在这时就派上用场了。
fill 0x0000, 5 ;从当前地址处连续5 个程序字填成0x0000(NOP指令)
fill (goto $), NEXT_BLOCK-$ ;从当前地址开始到标号 NEXT_BLOCK前所有程序空间填上
;goto $ (死循环)指令
org 0x0800
NEXT_BLOCK
例 3-12
请大家特别注意上例 3-12 中第二行 fill 伪指令的用法。在你自己的程序中也可以用同样的方法把所有未用到的程序空间填上“goto $”这样一条死循环的指令。一旦单片机执行过程中非法跳到这些指令处时指令运行就将被“俘获”,停在那里直到看门狗复位,然后程序从头开始。这是软件陷阱的最基本处理方法。若填充指令“goto 0x0000”直接跳转到复位地址处可能会有问题,因为 goto 指令执行时必须和 PCLATH寄存器配合(跨页跳转的问题),若 PCLATH[4:3]不为00就不能跳到复位地址 0x0000 处。在程序跑飞非法跳转到设定的陷阱处时你又怎能保证 PCLATH中的页面设定为正好指向第 0 页?
end
end伪指令告诉汇编编译器编译工作到此为止,end 后面所有的信息,不管正确与否,一概不管。绝大多数情形下你的程序的最后一行应该是“end”。无论如何,end 必须出现在程序中,不然编译器会报错,无法进行编译工作。