51汇编伪指令
伪指令是对汇编起某种控制作用的特殊命令,其格式与通常的操作指令一样,并可加在汇编程序的任何地方,但它们并不产生机器指令。
许多伪指令要求带参数,这在定义伪指令时由“表达式”域指出,任何数值与表达式匀可以作为参数。
不同汇编程序允许的伪指令并不相同,以下所述的伪指令仅适用于MASM51系统,但一些基本的伪指令在大部份汇编程序中都能使用,当使用其它的汇编程序版本时,只要注意一下它们之间的区别就可以了。MASM51中可用的伪指令有:
ORG 设置程序起始地址
END 标志源代码结束
EQU 定义常数
SET 定义整型数
DATA 给字节类型符号定值
BYTE 给字节类型符号定值
WROD 给字类型符号定值
BIT 给位地址取名
ALTNAME 用自定义名取代保留字
DB 给一块连续的存储区装载字节型数据
DW 给一块连续的存储区装载字型数据
DS 预留一个连续的存储区或装入指定字节。
INCLUDE 将一个源文件插入程序中
TITLE 列表文件中加入标题行
NOLIST 汇编时不产生列表文件
NOCODE 条件汇编时,条件为假的不产生清单
- 三、EQU
- 四、SET
- 九、DB
- 十三、TITL
- 十四、PAGE
- 十五、LIST与NOLIST
一、ORG
伪指令ORG用于为在它之后的程序设置地址值,它有一个参数,其格式为:
ORG 表达式
表达式可以是一个具体的数值,也可以包含变量名,如果包含变量名,则必须保证,当第一次遇到这条伪指令时,其中的变量必须已有定义(已有具体的数值),否则,无定义的值将由0替换,这将会造成错误。在列表文件中,由ORG定义的指令地址会被打印出来。
ORG指令有什么用途呢?指令被翻译成机器码后,将被存入系统的ROM中,一般情况下,机器码总是一个接一个地放在存储器中,但有一些代码,其位置有特殊要求,典型的是五个中断入口,它们必须被放在0003H,000BH,0013H,001BH和0023H的位置,否则就会出错,如果我们编程时不作特殊处理,让机器代码一个接一个地生成,不能保证这些代码正好处于这些规定的位置,执行就会出错,这时就要用到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),根据各编程器不同而不同,一般为FFH或00H。
二、END
END语句标志源代码的结束,汇编程序遇到END语句即停止运行。若没有END语句,汇编将报错。END语句有一个参数,可以是数值0,也可以是表达式,其格式是:
标号: END 表达式
它的值就是程序的地址并且作为一个特殊的记录写入HEX文件。若这个表达式省略,HEX文件中其值就是0。
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=464H,JK的值为752(即2F0H),XYJK的值XY+JK=464H+2F0H=754H
SET伪指令有些类似于等值伪指令,它定义了一个整数类型的符号名,它的格式为
符号名 SET 表达式
SET伪指令与等值伪指令的唯一区别在于SET伪指令所定义的符号名右以在程序中多次定义,而不报错。
例:002D= K57 SET 101101B
8707= K57 SET 34567
五、DATA与BYTE
DATA与BYTE都是用来定义字节类型的存储单元,赋予字节类型的存储单元一个符号名,以便在程序中通过符号名来访问这个存储单元,以帮助对程序的理解。
BYTE与DATE之间的区别类似于EQU和SET,BYTE伪指令不能定义重名。
六、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 表达式
只要表达式不是字符串,每一表达式值都被赋给一个字节。计算表达式值时按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)
TITLE伪指令用于在列表文件页头建立一个标题,其格式为:
$TITLE 标题行
这里标量行就是将出现在页头的标量与通常的字符串定义不同。这里标量行不加引号。汇编从$TITLE之后的第一个可打印字符开始,到回车符之间的字符串作为标量标量的最大长度是60个字符,基标量行省略,则标题行为空行。若TITLE伪指令在一页,它说明的标量行包含在本页,否则,标题将出现在下页页头。
PAGE伪指令用于形成新的一中定义一面的行数。其格式为:
$PAGE 表达式
若表达 式缺省则开始新的一页,若有表达式,则每页行数重新定义。汇编开始时页长为66行。一页中除出页外,剩余55行用于打印源程序,这一格式适用于标准打印纸。
如果变元值小于66,页内可打印的源代码行将相应减少。页长最小值为12。若小于12时,每页内除页上只打印一行源程序。
页长变元是16位字节,因而每页最长可定义到65535行,这时分页打印变为连续打印,在屏幕显示程序清单或在卷筒纸上打印程序清单时,常常使用连续打印,如果在启动汇编时用/N选项,页长就是65535。
它们的格式为:
$LIST
$NOLIST
LIST伪指令使汇编时主生程序清单,但即使不用该指令,汇编也会自动产生清单。但如果使用了NOLIST伪指令后需要继续主生清单则必须使用LIST伪指令。
NOLIST伪指令使汇编时不产生清单,所有包含此伪指令及在这条伪指令之后的语句都不进入列表文件。当不需要任何列表文件,并且不需要显示程序清单时,可以在启动汇编时不加.L附加项,且在源代码的第一行加上NOLIST指令。
使用NOLIST伪指令与附加项/L不同之处是NOLIST伪指令可加在源程序中,与LIST伪指令配合使用,使源程序中某些部分不产生清单。而不加附加项/L则不产生任何程序清单。不过,不管有无$NOLIST伪指令,程序在汇编时检查到的错误都将在屏幕上显示出错的源代码行及错误信息
十六、NOCODE
其格式为 $NOCODE
NOCODE伪指令使得在汇编时,条件汇编程序结构中那些真值为假的条件不产生清单。有关条件汇编结构在下面介绍。如果没有这条伪指令,汇编将主生所有条件下的清单,不论其真值是否为真。但是假的条件,不产生目标码。而NOCODE伪指令使汇编清单中只列出那些由汇编程序用到的部分,因此,当使用NOCODE伪指令时,程序清单与源程序并非逐行对应。
相关文章