(2)清除进位标志
(3)将低字求反,然后 +1
(4)将进位保存起来
(5)将高位字求反,加上刚才的进位
--------------
XXXX:
MOV AX, X + 1 取 X 的高位字
CMP AX, 32768
JB X_EXIT 是正数就跳转
MOV AX, X 取 X 的低位字
XOR AX, 0FFFFH 取反
ADD AX, 1
PUSHF
MOV X, AX
MOV AX, X + 1 取X 的高位字
XOR AX, 0FFFFH 取反
POPF
ADC AX, 0
MOV X + 1, AX
X_EXIT:
RET
伪 指 令伪指令是对汇编起某种控制作用的特殊命令,其格式与通常的 *** 作指令一样,并可加在汇编程序的任何地方,但它们并不产生机器指令。许多伪指令要求带参数,这在定义伪指令时由“表达式”域指出,任何数值与表达式匀可以作为参数。不同汇编程序允许的伪指令并不相同,以下所述的伪指令仅适用于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,001BH和0023H的位置,否则就会出错,如果我们编程时不作特殊处理,让机器代码一个接一个地生成,不能保证这些代码正好处于这些规定的位置,执行就会出错,这时就要用到ORG伪指令了。看如下例子:例:INT_0 EQU 1000HTIME_0 EQU 1010HINT_1 EQU 1020HTIME_1 EQU 1030HSERIAL EQU 1040HAJMP START 跳转到主程序起始点LJMP INT_0 外中断0处理程序LJMP TIME_0 定时中断0处理程序LJMP INT_1 外中断1处理程序LJMP TIME_1 定时中断1处理程序LJMP SERIAL 串行口中断程序START:NOPEND上面的程序经汇编后列表文件如下:The Cybernetic Micro Systems 8051 Family Assembler, Version 3.03 Page 108-26-96 1000 = INT_0 EQU 1000H1010 = TIME_0 EQU 1010H1020 = INT_1 EQU 1020H1030 = TIME_1 EQU 1030H1040 = SERIAL EQU 1040H0000 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 NOP0000 ENDThe Cybernetic Micro Systems 8051 Family Assembler, Version 3.03 Page 208-26-96%T Symbol Name Type ValueINT_0 . . . . . . . . . . . . . I 1000INT_1 . . . . . . . . . . . . . I 1020SERIAL. . . . . . . . . . . . . I 1040START . . . . . . . . . . . . . L 0011TIME_0. . . . . . . . . . . . . I 1010TIME_1. . . . . . . . . . . . . I 1030%Z00 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 1000HTIME_0 EQU 1010HINT_1 EQU 1020HTIME_1 EQU 1030HSERIAL EQU 1040HAJMP START 跳转到主程序起始点ORG 0003HLJMP INT_0 外中断0处理程序ORG 000BHLJMP TIME_0 定时中断0处理程序ORG 0013HLJMP INT_1 外中断1处理程序ORG 001BHLJMP TIME_1 定时中断1处理程序ORG 0023HLJMP SERIAL 串行口中断程序START:NOPEND上面的程序经过汇编后列表文件如下:The Cybernetic Micro Systems 8051 Family Assembler, Version 3.03 Page 108-26-961000 = INT_0 EQU 1000H1010 = TIME_0 EQU 1010H1020 = INT_1 EQU 1020H1030 = TIME_1 EQU 1030H1040 = SERIAL EQU 1040H0000 0126 AJMP START 跳转到主程序起始点0003 ORG 0003H0003 021000 LJMP INT_0 外中断0处理程序000B ORG 000BH000B 021010 LJMP TIME_0 定时中断0处理程序0013 ORG 0013H0013 021020 LJMP INT_1 外中断1处理程序001B ORG 001BH001B 021030 LJMP TIME_1 定时中断1处理程序0023 ORG 0023H0023 021040 LJMP SERIAL 串行口中断程序START:0026 00 NOP0000 ENDThe Cybernetic Micro Systems 8051 Family Assembler, Version 3.03 Page 208-26-96%T Symbol Name Type ValueINT_0 . . . . . . . . . . . . . I 1000INT_1 . . . . . . . . . . . . . I 1020SERIAL. . . . . . . . . . . . . I 1040START . . . . . . . . . . . . . L 0026TIME_0. . . . . . . . . . . . . I 1010TIME_1. . . . . . . . . . . . . I 1030%Z00 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,另一种用字符“=”即符号名 EQU 表达式符号名 = 表达式两种形式的效果是一样的。符号名在左边,其对应的值在右边。值可以是变元,其它的符号名或表达式。只要在两遍扫描中求出表达式的值就行,否则引用该符号名时将报错。当表达式的值是字符串时,只取后两个字符。若串长为1,高位字节被置0,符号名的值被打印在程序清单中。由等值伪指令定义的符号名不允许重名。如果经定义的符号名被重定义,则汇编将报出错,并且这个符号名按新定义的处理,最好不要在程序中出现重名。例:0469= ABC EQU 469H0464= XY EQU ABC-502F0= JK = 7520754 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伪指令与等值伪指令的唯一区别在于SET伪指令所定义的符号名右以在程序中多次定义,而不报错。例:002D= K57 SET 101101B8707= K57 SET 34567五、DATA与BYTE DATA与BYTE都是用来定义字节类型的存储单元,赋予字节类型的存储单元一个符号名,以便在程序中通过符号名来访问这个存储单元,以帮助对程序的理解。BYTE与DATE之间的区别类似于EQU和SET,BYTE伪指令不能定义重名。六、WORD WORD伪指令类似于DATE伪指令,只是WORD伪指令定义了一个字类型的符号名,其格式为:符号名 WORD 表达式0027= VAL31 WORD 390021= PAR7 WORD 21H一个字由2个字节组成。当然,因为8051汇编语言集没有字 *** 作,所以程序执行时,只处理字节。WROD伪指令仅仅允许用户定义一个认为是字的存储位置。七、BIT BIT伪指令定义了一个字位类型的符号名,其格式为:符号名 BIT 表达式这里表达式的值是一个位地址,这个伪指令有助于位的地址符号化。例:002F= LOG3 BIT 470014= Y731 BIT 14H八、ALTNAME 替换名(ALTNAME)伪指令提供用户一种手段,以定义一个符号名来替换一个保留字,此后这个答名与被替换的保留字均可等效地用于程序中。任何保留类型的答名均可被替换。替换名伪指令格式为:ALTNAME 保留字,新名例:0002= ALTNAME R2 COUNT013A EA MOV A,R2013B E502 MOV A,COUNT九、DB DB伪指令用于定义一个连续的存储区,给该存储区的存储单元赋值。该伪指令的参数即为存储单元的值,在表达式中对变元个数没有限制,只要此条伪指令能容纳在源程序的一行内,其格式为:标号: DB 表达式只要表达式不是字符串,每一表达式值都被赋给一个字节。计算表达式值时按16位处理,但其结果只取低8位,若多个表达式出现在一个DB伪指令中,它们必须以逗号分开。表达式中有字符串时,以单引号“'”作分隔符,每个字符占一个字节,字符串不加改变地被存在各字节中,并不将小写字母转换成大写字母。例如:DB 00H 01H 03H 46HDB 'This is a demo!'十、DW DW为以字节为单元(十六位二进制)来给一个的存储区赋值,其格式为:标号: DW 表达式例如:0000 3035 D46B DW 12341,54379,10110100101110B0004 2D2E0006 4344 4243 DW 'ABCD','BC','A'000A 0041000C 2868 02E8 DW 456*375h,83+295h,'YZ',72h-4560010 595A FEAA十一、DS DS为定义存储内容的伪指令,用它定义一个存储区,并用指定的参数填满该存储区。DS伪指令包含两个变元,第一个变元定义了存储区的长度的字节数,在汇编时,汇编程序将跳过这些单元把其它指令汇编在这些字节之后,因此在使用DS伪指令时第一个变元不可活力第二个变元表示在这些单元中真入什么值,第二个变元可以活力活力时这些字节将不处理。下例中0173处有一条DS 9,则空出9个字节,下一第指令被汇编到017C处;在017C处空出1BH个单元,在这些字节中被27H所填充。DS指令的格式如下:标号: DS 表达式1,表达式2表达式1定义了存储区的长度(以字节为单位)。这个变元不能省略。表达式2是可选择的,它的值低8位用以填入所定义的存储区。若省略则这部分存储单元不处理。例:0000 04 INC A0001 DS 9000A 04 INC A000B DS 1BH,27H0026 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.ASMORG 27HSTART:CLR AMOV R3,AINCLUDE MOD1INC R5INCLUDE MOD2.ASMDEC R3END START主程序为带有END伪指令的完整的源文件。程序中有两INCLUDE伪指令,分别将两附加的文件MOD1.ASM及MOD2.ASM链接到主程序中。以下是这两个文件。MOD1.ASMMOV R2,#31HMOV R5,#18HMOD2.ASMMOV R6,#47HANL A,#07HMOV 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 1MOD2.ASMMOV R6,#47HANL A,#07HMOV R1,A08-27-96MAIN.ASM0027 ORG 27HSTART:0027 E4 CLR A0028 FB MOV R3,AI INCLUDE MOD1I MOD1.ASMI0029 7A31 MOV R2,#31HI002B 7D18 MOV R5,#18HI002D 0D INC R5I INCLUDE MOD2.ASMI MOD2.ASMI002E 7E47 MOV R6,#47HI0030 5407 ANL A,#07HI0032 F9 MOV R1,A0033 1B DEC R30027 END STARTThe Cybernetic Micro Systems 8051 Family Assembler, Version 3.03 Page 208-27-96%T Symbol Name Type ValueSTART . . . . . . . . . . . . . L 0027%Z00 Errors (0000)十三、TITL TITLE伪指令用于在列表文件页头建立一个标题,其格式为:$TITLE 标题行这里标量行就是将出现在页头的标量与通常的字符串定义不同。这里标量行不加引号。汇编从$TITLE之后的第一个可打印字符开始,到回车符之间的字符串作为标量标量的最大长度是60个字符,基标量行省略,则标题行为空行。若TITLE伪指令在一页,它说明的标量行包含在本页,否则,标题将出现在下页页头。十四、PAGE PAGE伪指令用于形成新的一中定义一面的行数。其格式为:$PAGE 表达式若表达 式缺省则开始新的一页,若有表达式,则每页行数重新定义。汇编开始时页长为66行。一页中除出页外,剩余55行用于打印源程序,这一格式适用于标准打印纸。如果变元值小于66,页内可打印的源代码行将相应减少。页长最小值为12。若小于12时,每页内除页上只打印一行源程序。页长变元是16位字节,因而每页最长可定义到65535行,这时分页打印变为连续打印,在屏幕显示程序清单或在卷筒纸上打印程序清单时,常常使用连续打印,如果在启动汇编时用/N选项,页长就是65535。十五、LIST与NOLIST 它们的格式为:$LIST$NOLISTLIST伪指令使汇编时主生程序清单,但即使不用该指令,汇编也会自动产生清单。但如果使用了NOLIST伪指令后需要继续主生清单则必须使用LIST伪指令。NOLIST伪指令使汇编时不产生清单,所有包含此伪指令及在这条伪指令之后的语句都不进入列表文件。当不需要任何列表文件,并且不需要显示程序清单时,可以在启动汇编时不加.L附加项,且在源代码的第一行加上NOLIST指令。使用NOLIST伪指令与附加项/L不同之处是NOLIST伪指令可加在源程序中,与LIST伪指令配合使用,使源程序中某些部分不产生清单。而不加附加项/L则不产生任何程序清单。不过,不管有无$NOLIST伪指令,程序在汇编时检查到的错误都将在屏幕上显示出错的源代码行及错误信息十六、NOCODE 其格式为 $NOCODENOCODE伪指令使得在汇编时,条件汇编程序结构中那些真值为假的条件不产生清单。有关条件汇编结构在下面介绍。如果没有这条伪指令,汇编将主生所有条件下的清单,不论其真值是否为真。但是假的条件,不产生目标码。而NOCODE伪指令使汇编清单中只列出那些由汇编程序用到的部分,因此,当使用NOCODE伪指令时,程序清单与源程序并非逐行对应。 本文来自CSDN博客,转载请标明出处: http://blog.csdn.net/sunnf/archive/2008/10/23/3129370.aspx
一。从一数到十COUNT EQU 0x30003100 定义变量COUNT的基地址 AREA Example1,CODE,READONLY声明代码段Example1为只读 ENTRY 标识程序入口
CODE32 声明32位ARM指令 START LDR R1,=COUNT 将0X30003100赋给R1 MOV R0,#0 执行R0=0
STR R0,[R1] 存储R0寄存器的数据到R1指向的存储单元 LOOP LDR R1,=COUNT 将0X30003100赋给R1
LDR R0,[R1] 将R1中的数值作为地址,取出此地址中的数据保存到R0中 ADD R0,R0,#1 执行R0=R0+1
CMP R0,#10 将R0与10进行比较
MOVHS R0,#0 若R0大于等于10,则R0=0
STR R0,[R1] 存储R0寄存器的数据到R1指向的地址单元 B LOOP 跳转到LOOP
END 汇编文件结束
二,9的8次幂
X EQU 9 初始化X为9 n EQU 8 初始化N为8
AREA Example3,CODE,READONLY 生明代码段Example3为只读 ENTRY 标识程序入口路
CODE32 声明32位ARM指令
START LDR SP,=0x30003F00 把0x30003F00 赋给SP(R13) LDR R0,=X 把9赋给R0 LDR R1,=n 把8赋给R1
BL POW 跳转到POW,并把下一条指令地址存入到R14中 HALT B HALT 等待跳转
POW STMFD SP!,{R1-R12,LR} 将R1-R12入栈,满递减堆栈 MOVS R2,R1 将R1赋给R2,并影响标志位 MOVEQ R0,#1 若Z=1,则R0=1
BEQ POW_END 若Z=1,跳转到POW_END MOV R1,R0 将R0中值赋给R1 SUB R2,R2,#1 将R2-1的只赋给R2 POW_L1 BL DO_MUL 跳转到DO-MUL,并把下一条指令地址存入R14中 SUBS R2,R2,#1 将R2-1的值赋给R2,并影响标志位 BNE POW_L1 若Z=0,跳转到POW_L1
POW_END LDMFD SP!,{R1-R12,PC} 数据出栈,存入到R1-R12,PC中 DO_MUL MUL R0,R1,R0 把R1*R0的值赋给R0 MOV PC,LR LR中的值赋给PC END 汇编结束
三:从一一直加到一百
程序清单(一) C 语言实验参考程序
#define uint8 unsigned char 定义一个无符号字符常量uint8 #define uint32 unsigned int 定义一个无符号整型常量unint32
#define N 100 定义一个常量N=100(宏定义,100用N代替) uint32 sum定义sum为无符号整型常量(声明一个unsigned int型的变量sum) void Main(void) 主函数
{uint32 i定义无符号整型常量i(声明一个unsigned int型的变量i) sum=0sum初始值为0
for(i=0i<=Ni++) i在N内自增加1(i从0开始,i<=N时循环成立) {sum+=i} 把sum+i赋给sum while(1)为真循环 }
程序清单(二) 简单的启动代码
IMPORT |Image$$RO$$Limit | R0输出段存储区域界线 IMPORT |Image$$RW$$Base | RW输出段运行时起始地址 IMPORT |Image$$ZI$$Base | ZI输出段运行时起始地址 IMPORT |Image$$ZI$$Limit | ZI输出段存储区域界线 IMPORT Main 主函数
AREA Start,CODE,READONLY 声明代码段start,为只读 ENTRY 程序入口
CODE32 声明32位ARM指令 Reset LDR SP,=0x40003f00 将0x40003f00赋给SP
LDR R0,=|Image$$RO$$Limit| 将R0输出段存储区域界线赋给R0 LDR R1,=|Image$$RW$$Base | 将RW输出段运行时起始地址赋给R1 LDR R3,=|Image$$ZI$$Base | 将ZI输出段运行时起始地址赋给R3 CMP R0,R1 比较R0和R1,相等Z=1,反之Z=0 BEQ LOOP1 若Z=1,则跳到LOOP1
LOOP0 CMP R1,R3 比较R1和R3,若R1<r3,c=0
LDRCC R2,[R0],#4 若C=0,读取R0地址单元内容并且存入R2,且R0=R0+4 STRCC R2,[R1],#4 若C=0,读取R2中的数据存入R1,且R1=R1+4 BCC LOOP0 若C=0,跳转到LOOP0
LOOP1 LDR R1,=|Image$$ZI$$Limit| 将ZI输出段存储区域赋给R1 MOV R2,#0 把0赋给R2
LOOP2 CMP R3,R1 比较R1和R3,若R1<r3,c=0 strcc="" r2,[r3],#4="" 若c="0,将R2中数据保存到内存单元R3中,且R3=R3+4" bcc="" loop2="" b="" main="" 跳转到主程序="" end="" 汇编结束=""
四、程序清单(一) C 语言调用汇编的参考程序
#define uint8 unsigned char 定义一个无符号字符常量uint8 #define uint32 unsigned int 定义一个无符号整型常量.uint32
extern uint32 Add(uint32 x,uint32 y)//声明子程序Add为一个无符号整型常量,它为2个无符号整型常量x,y的和
uint32 sum定义sum为无符号整型常量 void Main(void) 无返回主程序
{sum=Add(555,168)sum等于555+168 while(1)为真循环 }
程序清单(二) 汇编加法函数程序
EXPORT Add 声明子程序Add方便调用 AREA Start,CODE,READONLY 声明代码段start,为只读 ENTRY 程序入口
CODE32 声明32位ARM指令
Add ADD R0,R0,R1 将R0+R1值赋给R0 MOV PC,LR 将LR值赋给PC
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)