第4章 MCS-51系列单片机的程序设计

整理文档很辛苦,赏杯茶钱您下走!

免费阅读已结束,点击下载阅读编辑剩下 ...

阅读已结束,您可以下载文档离线阅读编辑

资源描述

MCS-51系列单片机的程序设计第四章交叉汇编语言MASM51单片机汇编语言程序设计方法及技巧本章内容了解交叉汇编语言MASM51的特点掌握顺序程序、分支程序、循环程序、查表程序、子程序的基本设计方法和技巧学习目的掌握汇编语言程序设计的一般步骤4.1交叉汇编语言MASM51汇编语言源程序是由汇编指令和伪指令组成的文件,其扩展名为“ASM”。源程序编辑完后只有转换为机器码表示的目标程序计算机才能执行,这个转换的过程称为汇编。1)人工汇编人工汇编是通过人工查出源程序中所有汇编指令的机器码,并依次将机器码输入单片机中,使之成为单片机能直接执行的目标程序。2)机器汇编机器汇编是指在计算机上通过汇编程序对源程序进行汇编,即从源程序到目标程序的转换过程是由计算机上的汇编程序自动完成的。3)交叉汇编是指用一种计算机上的汇编程序去汇编另一种计算机的源程序,为另一种计算机产生目标程序。单片机MCS-51汇编语言程序设计步骤:1分析问题2确定算法3设计程序流程图4分配内存单元5编写汇编语言源程序6调试程序4.2程序设计方法及技巧4.2.1顺序程序设计例4.1请用MCS-51汇编指令编写程序,将片内RAM104、105单元中的内容分别传送到片外数据存储器104H、105H单元中。分析:应注意片内和片外的寻址指令不同ORG1000HMOVR0,#68HMOVA,@R0;片内RAM104单元内容送累加器MOVDPTR,#0104HMOVX@DPTR,A;完成片内RAM104单元内容送片;外104H单元中INCR0MOVA,@R0;片内RAM105单元内容送累加器INCDPTRMOVX@DPTR,A;完成片内RAM105单元内容送片;外105H单元中SJMP$END1.简单分支程序YN顺序执行条件满足?转向目标程序4.2.2分支程序设计图4.1简单分支程序转移分析:只要将两数进行比较,用比较的结果作为分支转移的条件就很容易地找出其中的大数。已知片内RAMFIRST开始的两个单元中各存放有一个8位无符号二进制数,要求找出其中的大数并存入片内RAM的SECOND单元中。例4.2ORG1000HCLRCMOVR0,#FIRST;第一个数地址送R0中MOVA,@R0;取第一个数INCR0;R0指向第二个数地址SUBBA,@R0;两数比较JCBIG;第二个数大转BIGDECR0;R0指向第一个数地址BIG:MOVSECOND,@R0;存大数SJMP$END参考程序2)多路分支程序(散转程序)K=nK=n-1K=2转向分支n-1转向分支2转向分支1转向分支nK=?K=1…图4.2多路分支程序转移1)把AJMP或LJMP指令组成一个分支程序的转移表,表的首地址放入DPTR中;2)调整后的出口信息放入累加器A中;3)执行JMP@A+DPTR转向对应的分支程序。可通过间接转移指令JMP@A+DPTR和无条件转移指令中的AJMP或LJMP指令来实现多路分支转移:例4.3已知R0中存放有一数字关键码,要求根据码值的不同,分别转向对应的码值程序段去执行。即当R0=K时,便执行CODEK程序。ORG1000HMOVDPTR,#TAB;表首址送DPTRMOVA,R0;取数字关键码ADDA,R0;每条AJMP指令占用两个字;节,将关键码乘以2JNCLESS;关键码小于128转LESSINCDPHLESS:JMP@A+DPTRTAB:AJMPCODE0;关键码为0,转CODE0执行AJMPCODE1;关键码为1,转CODE1执行……AJMPCODE255;关键码为255,转CODE255;执行AJMP指令的转移范围不超出所在的2KB字节区间,如各小段程序较长,在2KB字节内无法全部容纳,上面的程序应该怎样修改?思考1)循环初始化设置循环的初始状态,位于循环程序的开头位置2)循环处理循环程序的主体部分,是通过反复执行来完成数据的具体处理,它位于循环体内。4.2.3循环程序设计循环程序由4部分组成:3)循环控制也在循环体内,是用于控制循环的继续与否。4)循环结束通常位于循环体后,是用来存放循环处理的最终结果及恢复各寄存器与工作单元的原始值。设MCS-51单片机内部RAM从40H开始的连续16个单元中,每个单元中分别存放有单字节的带符号数。要求将其中的正数存入50H开始的连续单元中,负数存入60H开始的连续单元中,并将正数、负数和零的个数分别存入70H、71H、72H单元中。例4.41)带符号数的最高位为符号位,可通过最高位来判断正负。2)程序中需要分别从40H、50H和60H单元开始取源数、存放正数和负数,故需要三个地址指针来指向对应的内存单元;设对应的地址指针分别为R0、R1和R2,由于R2不能用间接寻址方式,故R2需要同R1或R0进行交换后,才能进行负数的存放。分析图4.3程序流程图参考程序:ORG1000HCLRAMOV70H,A;存放正数的个数单元清0MOV71H,A;存放负数的个数单元清0MOV72H,A;存放零的个数单元清0MOVR3,#10H;设循环计数器初值MOVR0,#40H;设源数地址指针MOVR1,#50H;设正数地址指针MOVR2,#60H;设负数地址指针LOOP:MOVA,@R0;取源数JZZERO;源数为0,转ZER0JNBACC.7,POSI;源数为正,转POSIINC71H;负数个数加1MOVR4,A;暂存源数MOVA,R2XCHA,R1;R1与R2中的内容交换XCHA,R4MOV@R1,A;存负数XCHA,R4XCHA,R1;R1与R2中的内容交换INCR2;调整负数地址指针SJMPLOOP1;转循坏控制POSI:INC70H;正数个数加1MOV@R1,A;存正数INCR1;调整正数地址指针SJMPLOOP1ZERO:INC72H;零的个数加1LOOP1:INCR0;调整源数地址指针DJNZR3,LOOP;未送完,继续SJMP$END查表指令有:MOVCA,@A+DPTRMOVCA,@A+PC查表就是把事先计算或测得的数据按照一定的顺序编制成表格存放在存储器中,然后根据输入的数据,从表格中查出所需的结果。4.2.4查表程序设计片内RAM的EGA单元中有一个大写的英文字母,要求将此字母在英文字母表中的位置送入片内的COUNT单元中,如EGA单元中字母为I时,则COUNT单元中需送入数字9。例4.6如果将26个英文字母按照顺序编成一个表,通过查表法得到一个字母与EGA单元中的字母比较,若不相等再查,同时记录查表的次数。查完后,将记录次数送COUNT单元。分析ORG1000HEGAEQU50HCOUNTEQU60HMOVCOUNT,#00HMOVA,#16;调整地址LOOP:PUSHAMOVCA,@A+PC;查表CJNZA,EGA,NF;所查字母与EGA中字母比较SJMPLAST;相等则转到LASTNF:POPAINCA;调整地址INCCOUNT;记数值加1SJMPLOOP;继续查LAST:INCCOUNT;记数值加1SJMP$;等待ALPT:DB‘A’,‘B’,…,‘Y’,‘Z’求y=n!(n=0,1,…9),已知n的值在片内RAM的NUM单元中,求得的y值用BCD码表示且存放到片内GETY开始的单元中。例4.6分析:可将函数值列成表,采用查表法求y=n!当n=9时,y=362880H,需要三个单元存放结果。设每个n对应的y值都用3个单元来存放,则y值在表格中的首地址的计算公式为:y地址=函数表首址+n*3参考程序ORG1000HNUMEQU50HGETYEQU60HMOVR1,#03HMOVR0,#GETYMOVA,NUM;取n值MOVB,#03HMULAB;n*3MOVR2,A;暂存MOVDPTR,#TAB;表的首地址送DPTR中LOOP:MOVCA,@A+DPTR;查表MOV@R0,A;存结果INCR0;调整存放结果地址指针INCDPTR;修正表地址指针MOVA,R2;恢复n*3DJNZR1,LOOP;未查完,继续SJMP$TAB:DB01H,00H,00H,01H,00H,00HDB02H,00H,00H,06H,00H,00HDB24H,00H,00H,20H,01H,00HDB20H,07H,00H,40H,50H,00HDB20H,03H,04H,80H,28H,36H编写子程序时应注意:a.子程序的第一条指令之前必须有标号;b.子程序中需要保护现场;c.主程序和子程序之间的参数传递。4.2.5子程序设计1)数码转换子程序已知R0的低半个字节为一个四位的二进制数,要求将其转换为ASCAII码后送回R0中。例4.7二进制码转换为ASCII码二进制数与ASCII码的对应关系:四位二进制ASCII0000(0)30H0001(1)31H1001(9)39H1010(A)41H1111(F)46H…………ORG1000BTOASC:PUSHAPUSHPSWMOVA,R0ANLA,#0FH;取四位二进制数MOVR0,A;暂存CLRCSUBBA,#0AH;与10比较MOVA,R0;恢复四位二进制数JCLOOP;小于10转LOOPADDA,#07H;大于10先加07HLOOP:ADDA,#30H;加30HMOVR0,A;存结果POPPSWPOPARET已知片内BLOCKB单元开始处有一数据块,块长在R2中,在此数据块的每个单元中存放有一个未组合的BCD码,要求将每个BCD码转换为ASCAII码,并存放在片内BLOCKA开始的连续单元中。例4.8BCD码转换为ASCAII码主程序:ORG1000HMOVR2,#LMOVR0,#BLOCKB;设BCD码的地址指针MOVR1,#BLOCKA;设ASCAII的地址指针LOOP:ACALLTRANS;调用TRANS子程序DJNZR2,LOOP;未转换完,继续SJMP$将一位BCD码转换为ASCAII码的子程序:TRANS:MOVA,@R0;取BCD码ADDA,#30HMOV@R1,A;送ASCAII码INCR0INCR1RET;返回分析:将一位BCD码转换为相应的ASCAII码只需加上30H即可。设片内40H单元中有一组合的BCD码,要求将其转换为二进制数存回原处。分析:1)组合的BCD码为两位十进制数,需将十位数和个位数分开处理。2)十位数转换时只要累加0AH就能将其转换为相应的二进制数,然后再将个位数直接加到转换后的十位数上就可完成整个数的转换。例4.9BCD码转换为二进制数ORG1000HBCDTOB:MOVA,40HANLA,#F0H;取十位数SWAPA;将十位数换到低4位JZDONE;十位数为0,返回MOVR3,#00H;累加和寄存器清0LOOP:DECAACALLADD0AH;调用加10子程序JNZLOOP;未加完继续MOVA,40HANLA,#0FH;取个位数ADDA,R3;与十位相加MOV40H,A;转换完,存结果DONE:RET;(书上此处有错)ADD0AH:PUSHA;保护现场MOVA,R3;累加的和送A中ADDA,#0AHMOVR3,A;加10后送回原处POPA;恢复现场RET;返回加10子程序:已知8位二进制数在R0中,要求将其转换为BCD码并存放在片内30H、31H单元中。分析:8位二进制数转换成3位BCD码数,用压缩BCD码存放,百位数放31H单元,十位、个位数放30H单元。例4.10二进制数转换为BCD码程序流程图为:开始A/100→商为BCD码的百位数存于31H单元余数→A中A/10→商为BCD码的十位数余数为BCD码的个位数BCD码十位数、个位数组成压缩BCD码→30H单元结束R0→AORG1000HBTOBCD:MOVA,R0;取二进制数MOVB,#100DIVAB;除以100MOVR1,#31HMOV@R1,A;百位的BCD码数存31H单元DECR1XCHA,B;二进制数除以100的余数送累加器中MOVB,#10DIVAB;除以10SWAPAADDA,B;十位与个位的BCD码进行组合MOV@R1,A;组合的BCD码送30H中RET已知片内RAMADR1开始

1 / 60
下载文档,编辑使用

©2015-2020 m.777doc.com 三七文档.

备案号:鲁ICP备2024069028号-1 客服联系 QQ:2149211541

×
保存成功