您好,欢迎访问三七文档
当前位置:首页 > 电子/通信 > 综合/其它 > 单片机原理与接口技术第4章
2020/6/121第4章80C51的汇编语言程序设计程序编制的方法和技巧4.1源程序的编辑和汇编4.2基本程序结构4.3子程序及其调用4.4简单I/O设备的并口直接驱动示例4.52020/6/122单片机应用系统由硬件系统和应用程序构成汇编语言高级语言应用程序设计方法汇编语言,生成的目标程序占内存空间少、运行速度快,具有效率高、实时性强。高级语言,对系统的功能描述与实现简单,程序阅读、修改和移植方便,适合于编写复杂的程序。2020/6/1234.1程序编制的方法和技巧4.1.1程序编制的步骤明确任务:功能要求、技术指标运行环境调研任务分析将实际问题转化为计算机处理的程序算法算法比较与优化(内存需求与运行速度)算法设计2020/6/124流程描述流程图符号开始或结束符号工作任务符号判断分支符号程序连接符号程序流向符号程序流向符号“超级循环”框架初始化开始循环扫描、处理2020/6/125强化模块观念使程序占用空间减少、结构清晰循环初值和结束条件,避免“死机”现象子程序的现场保护(注意栈平衡、寄存器内容)程序模块(主程序模块、各种子程序模块)模块化优点:分块设计、便于阅读、调试方便4.1.2程序编制的方法和技巧采用循环和子程序对中断子程序还有注意保护PSW的内容2020/6/1264.1.3汇编语言的语句格式非数字字符开头,后跟字母、数字、“-”、“?”等不能用已定义的保留字(指令助记符、伪指令等)后跟英文冒号“:”Keil的汇编器A51可以识别的语句形式为:标号(即符号地址)[标号:]指令助记符[操作数1,][操作数2,][操作数3,][;注释]指令助记符是指令功能的英文缩写。2020/6/127数据:二进制(B)十进制(D或省略D)十六进制(H),注意A~F开头时要加“0”ASCII码,如‘A’,‘1245’符号:符号名、标号或“$”(PC的当前值)表达式:由运算符和数据构成(见表4.1)操作数注释英文分号“;”开头2020/6/128优先级运算符功能表达式及其结果示例高↓↓↓↓↓↓↓低·()括号4*(5+6)即44NOT、HIGH、LOW取反、取高字节、取低字节NOT55H即AAH;HIGH1234H即12H+、-正号、负号+5、-6*、/、MOD乘、除(取商)、取余数17/5即3;17MOD5即2+、-加、减5+4即9;5-4=即1SHL、SHR左移、右移2SHL2即8;8SHR2即2AND、OR、XOR与、或、异或45HAND0FH即05H、、=、、=、=比较运算符MOVA,X8;若X8为真,则为MOVA,01H若X8为假,则为MOVA,00H表4.12020/6/1294.2源程序的编辑和汇编目标程序的产生过程如下图:汇编器A51.EXE编译器C51.EXE目标文件(浮动地址).OBJC源文件.C汇编源文件.ASM连接器BL51.EXE调试目标文件(绝对地址)无扩展名转换器OH51.EXE可烧写目标文件.HEX.LST.LST.M51库文件.LIB映像文件仿真调试写入芯片2020/6/12104.2.1源程序的编辑和汇编源程序的编辑ORG0000HLJMPMAINORG0040HMAIN:MOVR7,#16MOVR0,#60HMOVA,#55HLOOP:MOV@R0,AINCR0DJNZR7,LOOPSJMP$END依据汇编语言规则用好伪指令符号不用中文SJMP$用于调试以.ASM存盘2020/6/1211源程序的汇编汇编源程序转为目标程序的过程叫汇编汇编通常在Windows下的集成开发环境完成用A51.EXE汇编生成.OBJ、.LIB及.LST目标程序的连接.OBJ、.LIB经BL51.EXE生成无扩展名的绝对地址目标文件绝对地址目标文件可以用于仿真器调试调试无误的目标文件用OH51.EXE转换为.HEX文件.HEX文件经编程器写入单片机存储器2020/6/12124.2.2伪指令伪指令,也叫汇编命令。仅对汇编过程进行指示伪指令无对应的单片机可执行代码起始地址设定伪指令ORGORG表达式表达式通常为十六进制地址,例:ORG8000HSTART:MOVA,#30H……ORG可多次使用,但地址值的顺序要由小到大结束汇编伪指令ENDEND该伪指令位于源程序的最后一行。2020/6/1213定义字节数据表伪指令DB定义字数据表伪指令DW1000H[标号:]DB字节数据表如:ORG1000HDB-2,-4,-6,8,10,18FEHFCHFAH08H0AH12H1001H[标号:]DW字数据表ORG1400HDATA1:DW324AH,3CH……1400H32H4AH00H3CH1401H1402H1403H大端模式2020/6/1214定义常值为符号名伪指令EQU符号名EQU常值表达式LENEQU10SUMEQU21HBLOCKEQU22HCLRAMOVR7,#LENMOVR0,#BLOCKLOOP:ADDA,@R0INCR0DJNZR7,LOOPMOVSUM,A符号名为:地址常数段名字符串寄存器名位名比较:标号只能是地址2020/6/1215定义位地址为符号名伪指令BIT符号名BIT位地址表达式如:STBITP1.0;将P1.0的位地址赋给符号名STCFBIT0D7H;将位地址为D7H的位定义为符号名用BIT定义的“符号名”一经定义便不能重新定义和改变其它一些伪指令参见教材表4.22020/6/12164.3基本程序结构一般不影响标志寄存器PSW的状态。传送类指令有两大类一般传送(MOV)特殊传送,如:MOVCMOVXPUSH、POPXCH、XCHDSWAP2020/6/12174.3.1顺序程序(无分支、无循环)4.3基本程序结构【例4-1】片内RAM的21H单元存放一个十进制数据十位的ASCII码,22H单元存放该数据个位的ASCII码。编写程序将该数据转换成压缩BCD码存放在20H单元。开始结束取十位的ASCII码保留低半字节移至高半字节,存回取个位的ASCII码保留低半字节合并到结果单元2020/6/1218ORG0040HSTART:MOVA,21H;取十位ASCII码ANLA,#0FH;保留低半字节SWAPA;移至高半字节MOV20H,A;存于20H单元MOVA,22H;取个位ASCII码ANLA,#0FH;保留低半字节ORL20H,A;合并到结果单元SJMP$END2020/6/12194.3.2分支程序(单分支、双分支、多分支)【例4-2】设变量x以补码的形式存放在片内RAM的30H单元,变量y与x的关系是:当x大于0时,y=x;当x=0时,y=20H;当x小于0时,y=x+5。编制程序,根据x的大小求y并送回原单元。开始结束取x至累加器X=0?X+05H送Y20H送YNX0?NYY2020/6/1220ORG0040HSTART:MOVA,30H;取x至累加器JZNEXT;x=0,转NEXTANLA,#80H;否,保留符号位JZDONE;x0,转结束MOVA,#05H;x0处理ADDA,30HMOV30H,A;X+05H送YSJMPDONENEXT:MOV30H,#20H;x=0,20H送YDONE:SJMPDONEEND2020/6/1221【例4-3】根据R7的内容x(转移序号)转向相应的处理程序。设R7内容为0~4,对应的处理程序入口地址分别为PP0~PP4。取分支号,乘2X=?分支程序PP1地址高、低字节存DPTR分支程序PP0分支程序PP2分支程序PP3返回分支程序PP4X=0X=1X=2X=3X=42020/6/1222START:MOVR7,#3;以转移序号3为例ACALLJPNUMAJMPSTARTJPNUM:MOVDPTR,#TAB;置分支入口地址表首址MOVA,R7ADDA,R7;乘2,调整偏移量MOVR3,AMOVCA,@A+DPTR;取地址高字节,暂存于R3XCHA,R3INCAMOVCA,@A+DPTR;取地址低字节MOVDPL,A;处理程序入口地址低8位送DPLMOVDPH,R3;处理程序入口地址高8位送DPHCLRAJMP@A+DPTR2020/6/1223TAB:DWPP0DWPP1DWPP2DWPP3DWPP4PP0:MOV30H,#0;转移序号为0时,置功能号“0”于30H单元RETPP1:MOV30H,#1;转移序号为1时,置功能号“1”于30H单元RETPP2:MOV30H,#2;转移序号为2时,置功能号“2”于30H单元RETPP3:MOV30H,#3;转移序号为3时,置功能号“3”于30H单元RETPP4:MOV30H,#4;转移序号为4时,置功能号“4”于30H单元RET2020/6/12244.3.3循环程序(2种:先执行,后判断;先判断,后执行)【例4-4】将内部RAM的30H至3FH单元初始化为00H。置初值开始结束循环处理循环修改结束处理循环结束?YNMAIN:MOVR0,#30H;置初值MOVA,#00H;MOVR7,#16;LOOP:MOV@R0,A;循环处理INCR0;DJNZR7,LOOP;循环修改,判结束SJMP$;结束处理2020/6/1225【例4-5】将内部RAM起始地址为60H的数据串传送到外部RAM中起始地址为1000H的存储区域,直到发现‘$’字符停止传送。置初值开始结束循环处理循环修改结束处理循环结束?YNMAIN:MOVR0,#60H;置初值MOVDPTR,#1000HLOOP0:MOVA,@R0;取数据CJNEA,#24H,LOOP1;循环结束?SJMPDONE;是LOOP1:MOVX@DPTR,A;循环处理INCR0;循环修改INCDPTRSJMPLOOP0;继续循环DONE:SJMPDONE;结束处理2020/6/12264.4子程序及其调用完成通用功能、反复使用的程序设计成子程序。使应用程序结构清晰紧凑,便于阅读和调试执行要由其它程序来调用,执行完后要返回到调用程序结构上仍然采用一般程序的3种结构调用时注意:一是现场的保护和恢复;二是主程序与子程序间的参数传递。2020/6/12274.4.1现场保护与恢复在主程序中实现(结构灵活)PUSHPSW;保护现场(含当前工作寄存器组号)PUSHACC;PUSHB;MOVPSW,#10H;切换当前工作寄存器组LCALLaddr16;子程序调用,POPB;恢复现场POPACC;POPPSW;含当前工作寄存器组切换2020/6/1228在子程序中实现(程序规范、清晰)SUB1:PUSHPSW;保护现场(含当前工作寄存器组号)PUSHACC;PUSHB;MOVPSW,#10H;切换当前工作寄存器组……POPB;恢复现场POPACC;POPPSW;内含当前工作寄存器组切换RET2020/6/12294.4.2参数传递利用累加器或寄存器(简单、快速,但参数个数不多)【例4-6】实现两个8位的十六进制无符号数求和的子程序。SADD:MOVA,R3;取加数(在R3中)CLRCADDA,R4;被加数(在R4中)加AJCPP1MOVR3,#00H;结果小于255时,高字节R3内容为00HSJMPPP2PP1:MOVR3,#01H;结果大于255时,高字节R3内容为01HPP2:MOVR4,A;结果的低字节在R4中RET入口:(R3)=加数;(R4)=被加数。出口:(R3)=和的高字节;(R4)=和的低字节。2020/6/1230利用存储器(个数多,用R0或R1及DPTR为参数表指针)【例4-7】将内部RAM中两个4字节无符号整数相加,和的高字节由R0指向。数据采用大端模式存储。入口:(R0)=加数低字节地址;(R1)=被加数低字节地址。出口:(R0)=和的高字节起始地址。NADD:MOVR7,#4;字节数4送计数器CLRC;NADD1:MOVA,@R0;利用指针,取加数低字节ADDCA,@R1;利用指针,被加数低字节加AMOV@R0,A;DECR0DECR1DJNZR7,NADD1INCR0;调整指针,指向出口RET2020/6/123
本文标题:单片机原理与接口技术第4章
链接地址:https://www.777doc.com/doc-5833596 .html