原创 伪 指 令×××

2007-10-24 15:46 1634 9 9 分类: MCU/ 嵌入式
 




伪指令是对汇编起某种控制作用的特殊命令,其格式与通常的操作指令一样,并可加在汇编程序的任何地方,但它们并不产生机器指令。


许多伪指令要求带参数,这在定义伪指令时由“表达式”域指出,任何数值与表达式匀可以作为参数。


不同汇编程序允许的伪指令并不相同,以下所述的伪指令仅适用于MASM51系统,但一些基本的伪指令在大部份汇编程序中都能使用,当使用其它的汇编程序版本时,只要注意一下它们之间的区别就可以了。MASM51中可用的伪指令有:


ORG 设置程序起始地址


END 标志源代码结束


EQU 定义常数


SET 定义整型数


DATA 给字节类型符号定值


BYTE 给字节类型符号定值


WROD 给字类型符号定值


BIT 给位地址取名


ALTNAME 用自定义名取代保留字


DB 给一块连续的存储区装载字节型数据


DW 给一块连续的存储区装载字型数据


DS 预留一个连续的存储区或装入指定字节。


INCLUDE 将一个源文件插入程序中


TITLE 列表文件中加入标题行


NOLIST 汇编时不产生列表文件


NOCODE 条件汇编时,条件为假的不产生清单



一、ORG


伪指令ORG用于为在它之后的程序设置地址值,它有一个参数,其格式为:


ORG 表达式


表达式可以是一个具体的数值,也可以包含变量名,如果包含变量名,则必须保证,当第一次遇到这条伪指令时,其中的变量必须已有定义(已有具体的数值),否则,无定义的值将由0替换,这将会造成错误。在列表文件中,由ORG定义的指令地址会被打印出来。


ORG指令有什么用途呢?指令被翻译成机器码后,将被存入系统的ROM中,一般情况下,机器码总是一个接一个地放在存储器中,但有一些代码,其位置有特殊要求,典型的是五个中断入口,它们必须被放在0003H,000BH,0013H,001BH0023H的位置,否则就会出错,如果我们编程时不作特殊处理,让机器代码一个接一个地生成,不能保证这些代码正好处于这些规定的位置,执行就会出错,这时就要用到ORG伪指令了。看如下例子:


例:


INT_0 EQU 1000H


TIME_0 EQU 1010H


INT_1 EQU 1020H


TIME_1 EQU 1030H


SERIAL EQU 1040H


AJMP START ;跳转到主程序起始点


LJMP INT_0 ;外中断0处理程序


LJMP TIME_0 ;定时中断0处理程序


LJMP INT_1 ;外中断1处理程序


LJMP TIME_1 ;定时中断1处理程序


LJMP SERIAL ;串行口中断程序


START:


NOP


END


上面的程序经汇编后列表文件如下:



The Cybernetic Micro Systems 8051 Family Assembler, Version 3.03 Page 1


08-26-96



 


1000 = INT_0 EQU 1000H


1010 = TIME_0 EQU 1010H


1020 = INT_1 EQU 1020H


1030 = TIME_1 EQU 1030H


1040 = SERIAL EQU 1040H


0000 0111 AJMP START ;跳转到主程序起始点


0002 021000 LJMP INT_0 ;外中断0处理程序


0005 021010 LJMP TIME_0 ;定时中断0处理程序


0008 021020 LJMP INT_1 ;外中断1处理程序


000B 021030 LJMP TIME_1 ;定时中断1处理程序


000E 021040 LJMP SERIAL ;串行口中断程序


START:


0011 00 NOP


0000 END



The Cybernetic Micro Systems 8051 Family Assembler, Version 3.03 Page 2


08-26-96


;%T Symbol Name Type Value


INT_0 . . . . . . . . . . . . . I 1000


INT_1 . . . . . . . . . . . . . I 1020


SERIAL. . . . . . . . . . . . . I 1040


START . . . . . . . . . . . . . L 0011


TIME_0. . . . . . . . . . . . . I 1010


TIME_1. . . . . . . . . . . . . I 1030


;%Z


00 Errors (0000)


由列表文件,可以绘出代码在ROM中的映象图如下:



代码


01H


11H


02H


10H


00H


02H


10H


10H


02H


10H


20H


地址


00H


01H


02H


03H


04H


05H


06H


07H


08H


09H


0AH


代码


02H


10H


30H


02H


10H


40H


00H

    

地址


0BH


0CH


0DH


0EH


0FH


10H


11H


12H


13H


14H


15


由上面的映象图可知,在03H处的代码为10H,而不是我们要的02H,所以外断程序INT_0不能被正确执行,其它各中断程序的情况同样如此,如在0BH处,本来存放的应当是定时器0中断程序,但按上述的映象图,0BH处开始的3个代码是:02H,10H,30H,这是定时器1的入口地址,所以,如果定时器0发生中断,所执行的其实是定时器1的中断程序,这当然不对。


例2:


INT_0 EQU 1000H


TIME_0 EQU 1010H


INT_1 EQU 1020H


TIME_1 EQU 1030H


SERIAL EQU 1040H


AJMP START ;跳转到主程序起始点


ORG 0003H


LJMP INT_0 ;外中断0处理程序


ORG 000BH


LJMP TIME_0 ;定时中断0处理程序


ORG 0013H


LJMP INT_1 ;外中断1处理程序


ORG 001BH


LJMP TIME_1 ;定时中断1处理程序


ORG 0023H


LJMP SERIAL ;串行口中断程序


START:


NOP


END


上面的程序经过汇编后列表文件如下:


The Cybernetic Micro Systems 8051 Family Assembler, Version 3.03 Page 1


08-26-96


1000 = INT_0 EQU 1000H


1010 = TIME_0 EQU 1010H


1020 = INT_1 EQU 1020H


1030 = TIME_1 EQU 1030H


1040 = SERIAL EQU 1040H


0000 0126 AJMP START ;跳转到主程序起始点


0003 ORG 0003H


0003 021000 LJMP INT_0 ;外中断0处理程序


000B ORG 000BH


000B 021010 LJMP TIME_0 ;定时中断0处理程序


0013 ORG 0013H


0013 021020 LJMP INT_1 ;外中断1处理程序


001B ORG 001BH


001B 021030 LJMP TIME_1 ;定时中断1处理程序


0023 ORG 0023H


0023 021040 LJMP SERIAL ;串行口中断程序


START:


0026 00 NOP


0000 END


The Cybernetic Micro Systems 8051 Family Assembler, Version 3.03 Page 2


08-26-96


;%T Symbol Name Type Value


INT_0 . . . . . . . . . . . . . I 1000


INT_1 . . . . . . . . . . . . . I 1020


SERIAL. . . . . . . . . . . . . I 1040


START . . . . . . . . . . . . . L 0026


TIME_0. . . . . . . . . . . . . I 1010


TIME_1. . . . . . . . . . . . . I 1030


;%Z


00 Errors (0000)


由列表文件,可以绘出代码在ROM中的映象图如下:



代码


01H


11H

 

02H


10H


00H

     

地址


00H


01H


02H


03H


04H


05H


06H


07H


08H


09H


0AH


代码


02H


10H


10H

     

02H


01H


20H


地址


0BH


0CH


0DH


0EH


0FH


10H


11H


12H


13H


14H


15H


代码

   

 

02H


10H


30H

   

地址


16H


17H


18H


19H


1AH


1BH


1CH


1DH


1EH


1FH


20H


代码

  

02H


10H


40H


00H

     

地址


21H


22H


23H


24H


25H


26H


27H


28H


29H


2AH


2BH


由映象图可知,各中断程序的代码都在其规定地址处,一旦产生中断即可执行相应的程序。至于图中未填的部分(如02H),根据各编程器不同而不同,一般为FFH00H



二、END


END语句标志源代码的结束,汇编程序遇到END语句即停止运行。若没有END语句,汇编将报错。END语句有一个参数,可以是数值0,也可以是表达式,其格式是:


标号: END 表达式


它的值就是程序的地址并且作为一个特殊的记录写入HEX文件。若这个表达式省略,HEX文件中其值就是0

三、EQU


EQU以及其它一些符号定义伪指令用来给程序中出现的一些符号赋值。对这些符号名的要求与其它符号相同,即长度不限,大小写字母可互换并且必须以字母开头。


由等值指令定义的符号是汇编符号表的一部分。等值伪指令有两种形式。一种用EQU,另一种用字符“=”即


符号名 EQU 表达式


符号名 = 表达式


两种形式的效果是一样的。符号名在左边,其对应的值在右边。值可以是变元,其它的符号名或表达式。只要在两遍扫描中求出表达式的值就行,否则引用该符号名时将报错。当表达式的值是字符串时,只取后两个字符。若串长为1,高位字节被置0,符号名的值被打印在程序清单中。由等值伪指令定义的符号名不允许重名。如果经定义的符号名被重定义,则汇编将报出错,并且这个符号名按新定义的处理,最好不要在程序中出现重名。


例:0469= ABC EQU 469H


0464= XY EQU ABC-5


02F0= JK = 752


0754 XYJK = XY+JK


在列表文件中最左边的数字不是这些伪指令所在的地址而是通过汇编后赋给符号名的值。第一条符号名ABC被起来469H,第二条XY被赋于ABC-5,因此XY的值为469H-5=464HJK的值为752(即2F0H)XYJK的值XY+JK=464H+2F0H=754H

四、SET


SET伪指令有些类似于等值伪指令,它定义了一个整数类型的符号名,它的格式为


符号名 SET 表达式


SET伪指令与等值伪指令的唯一区别在于SET伪指令所定义的符号名右以在程序中多次定义,而不报错。


例:002D= K57 SET 101101B


8707= K57 SET 34567


五、DATABYTE


DATABYTE都是用来定义字节类型的存储单元,赋予字节类型的存储单元一个符号名,以便在程序中通过符号名来访问这个存储单元,以帮助对程序的理解。


BYTEDATE之间的区别类似于EQUSETBYTE伪指令不能定义重名。


六、WORD


WORD伪指令类似于DATE伪指令,只是WORD伪指令定义了一个字类型的符号名,其格式为:


符号名 WORD 表达式


0027= VAL31 WORD 39


0021= PAR7 WORD 21H


一个字由2个字节组成。当然,因为8051汇编语言集没有字操作,所以程序执行时,只处理字节。WROD伪指令仅仅允许用户定义一个认为是字的存储位置。


七、BIT


BIT伪指令定义了一个字位类型的符号名,其格式为:


符号名 BIT 表达式


这里表达式的值是一个位地址,这个伪指令有助于位的地址符号化。


例:


002F= LOG3 BIT 47


0014= Y731 BIT 14H


八、ALTNAME


替换名(ALTNAME)伪指令提供用户一种手段,以定义一个符号名来替换一个保留字,此后这个答名与被替换的保留字均可等效地用于程序中。任何保留类型的答名均可被替换。替换名伪指令格式为:


ALTNAME 保留字,新名


例:


0002= ALTNAME R2 COUNT


013A EA MOV A,R2


013B E502 MOV A,COUNT


九、DB


DB伪指令用于定义一个连续的存储区,给该存储区的存储单元赋值。该伪指令的参数即为存储单元的值,在表达式中对变元个数没有限制,只要此条伪指令能容纳在源程序的一行内,其格式为:


标号: DB 表达式


只要表达式不是字符串,每一表达式值都被赋给一个字节。计算表达式值时按16位处理,但其结果只取低8位,若多个表达式出现在一个DB伪指令中,它们必须以逗号分开。


表达式中有字符串时,以单引号“'”作分隔符,每个字符占一个字节,字符串不加改变地被存在各字节中,并不将小写字母转换成大写字母。


例如:


DB 00H 01H 03H 46H


DB 'This is a demo!'


十、DW


DW为以字节为单元(十六位二进制)来给一个的存储区赋值,其格式为:


标号: DW 表达式


例如:


0000 3035 D46B DW 12341,54379,10110100101110B


0004 2D2E


0006 4344 4243 DW 'ABCD','BC','A'


000A 0041


000C 2868 02E8 DW 456*375h,83+295h,'YZ',72h-456


0010 595A FEAA


十一、DS


DS为定义存储内容的伪指令,用它定义一个存储区,并用指定的参数填满该存储区。


DS伪指令包含两个变元,第一个变元定义了存储区的长度的字节数,在汇编时,汇编程序将跳过这些单元把其它指令汇编在这些字节之后,因此在使用DS伪指令时第一个变元不可活力第二个变元表示在这些单元中真入什么值,第二个变元可以活力活力时这些字节将不处理。下例中0173处有一条DS 9,则空出9个字节,下一第指令被汇编到017C处;在017C处空出1BH个单元,在这些字节中被27H所填充。


DS指令的格式如下:


标号: DS 表达式1,表达式2


表达式1定义了存储区的长度(以字节为单位)。这个变元不能省略。表达式2是可选择的,它的值低8位用以填入所定义的存储区。


若省略则这部分存储单元不处理。


例:


0000 04 INC A


0001 DS 9


000A 04 INC A


000B DS 1BH,27H


0026 04 INC A


十二、INCLUDE


INCLUDE伪指令用于链接源文件,即将一个源文件插入到另一个源文件中。它有一个参数,指出将要插入的文件名,该文件名中可包括驱动器名和路径名。若文件没有扩展名,则默认为是ASM。但待插入的文件必须是可以打开的。若文件打开操作失败,则产生致命错误,汇编将停止运行。反之,汇编程序将文件内容读入并按源代码处理。当遇到文件结速符时,汇编程序返回到INCLUDE伪指令处继续身下处理源程序。被插入的文件在程序清单中以“I”开头。


本宏汇编版本支持级嵌套,可在程序中用INCLUDE伪指令插入任意多个文件,但是,在一般情况下DOS允许打开的文件数量是有限的,如果用户需要打开较多的文件,则必须在CONFIG.SYS文件中加入FILES=40或更多的值,若超过8级嵌套或打开的文件太多,则产生致命错误,汇编中止运行。


INCLUDE伪指令提供了模块化程序设计手段,在汇编程序处理主程序时,模块被插入,尽管这不等价于链接和装配可重定位的目标模块,但它具有类似的功能,被插入的源文件中不应该包含END伪指令,否则,汇编就会提前停止运行,END伪指令只能出现在主程序中。此外,在主程序进行汇编前所有附加的源文件必须通过汇编,产生相应的HEX及LST文件,由于附加的文件没有END伪指令,因此,附加文件汇编时,汇编程序将显示:“没有结束语句”的错误,但并不影响与主程序的链接。


下面是一个使用INCLUDE伪指令的例子,其主程序的源文件MAIN.ASM为:


;MAIN.ASM


ORG 27H


START:


CLR A


MOV R3,A


INCLUDE MOD1


INC R5


INCLUDE MOD2.ASM


DEC R3


END START


主程序为带有END伪指令的完整的源文件。程序中有两INCLUDE伪指令,分别将两附加的文件MOD1.ASM及MOD2.ASM链接到主程序中。以下是这两个文件。


;MOD1.ASM


MOV R2,#31H


MOV R5,#18H



;MOD2.ASM


MOV R6,#47H


ANL A,#07H


MOV R1,A


注意MOD1.ASM及MOD2.ASM均没有END指令。


在进行汇编时必须先对MOD1.ASM和MOD2.ASM进行汇编,然后在汇编MAIN.ASM,由于上两个文件没有END伪指令,所以在汇编时会出现错误提示,不用管它,继续下面的工作,就可以得到正确的结果。


以下是形成的列表文件:


The Cybernetic Micro Systems 8051 Family Assembler, Version 3.03 Page 1


;MOD2.ASM


MOV R6,#47H


ANL A,#07H


MOV R1,A


08-27-96


;MAIN.ASM


0027 ORG 27H


START:


0027 E4 CLR A


0028 FB MOV R3,A


I INCLUDE MOD1


I ;MOD1.ASM


I0029 7A31 MOV R2,#31H


I002B 7D18 MOV R5,#18H


I


002D 0D INC R5


I INCLUDE MOD2.ASM


I ;MOD2.ASM


I002E 7E47 MOV R6,#47H


I0030 5407 ANL A,#07H


I0032 F9 MOV R1,A


0033 1B DEC R3


0027 END START


The Cybernetic Micro Systems 8051 Family Assembler, Version 3.03 Page 2


08-27-96


;%T Symbol Name Type Value


START . . . . . . . . . . . . . L 0027


;%Z


00 Errors (0000)

十三、TITL


TITLE伪指令用于在列表文件页头建立一个标题,其格式为:


$TITLE 标题行


这里标量行就是将出现在页头的标量与通常的字符串定义不同。这里标量行不加引号。汇编从$TITLE之后的第一个可打印字符开始,到回车符之间的字符串作为标量标量的最大长度是60个字符,基标量行省略,则标题行为空行。若TITLE伪指令在一页,它说明的标量行包含在本页,否则,标题将出现在下页页头。

十四、PAGE


PAGE伪指令用于形成新的一中定义一面的行数。其格式为:


$PAGE 表达式


若表达 式缺省则开始新的一页,若有表达式,则每页行数重新定义。汇编开始时页长为66行。一页中除出页外,剩余55行用于打印源程序,这一格式适用于标准打印纸。


如果变元值小于66,页内可打印的源代码行将相应减少。页长最小值为12。若小于12时,每页内除页上只打印一行源程序。


页长变元是16位字节,因而每页最长可定义到65535行,这时分页打印变为连续打印,在屏幕显示程序清单或在卷筒纸上打印程序清单时,常常使用连续打印,如果在启动汇编时用/N选项,页长就是65535。

十五、LIST与NOLIST


它们的格式为:


$LIST


$NOLIST


LIST伪指令使汇编时主生程序清单,但即使不用该指令,汇编也会自动产生清单。但如果使用了NOLIST伪指令后需要继续主生清单则必须使用LIST伪指令。


NOLIST伪指令使汇编时不产生清单,所有包含此伪指令及在这条伪指令之后的语句都不进入列表文件。当不需要任何列表文件,并且不需要显示程序清单时,可以在启动汇编时不加.L附加项,且在源代码的第一行加上NOLIST指令。


使用NOLIST伪指令与附加项/L不同之处是NOLIST伪指令可加在源程序中,与LIST伪指令配合使用,使源程序中某些部分不产生清单。而不加附加项/L则不产生任何程序清单。不过,不管有无$NOLIST伪指令,程序在汇编时检查到的错误都将在屏幕上显示出错的源代码行及错误信息


十六、NOCODE



其格式为 $NOCODE


NOCODE伪指令使得在汇编时,条件汇编程序结构中那些真值为假的条件不产生清单。有关条件汇编结构在下面介绍。如果没有这条伪指令,汇编将主生所有条件下的清单,不论其真值是否为真。但是假的条件,不产生目标码。而NOCODE伪指令使汇编清单中只列出那些由汇编程序用到的部分,因此,当使用NOCODE伪指令时,程序清单与源程序并非逐行对应。


 
PARTNER CONTENT

文章评论0条评论)

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