第3章MCS-51系列单片机指令系统及汇编语言程序设计MCS-51单片机的寻址方式MCS-51单片机分类指令MCS-51单片机汇编语言程序设计基础3.1MCS-51单片机的寻址方式MCS-51单片机共有7种寻址方式:立即寻址直接寻址寄存器寻址寄存器间接寻址变址寻址相对寻址位寻址1.立即寻址立即寻址就是把操作数直接在指令中给出,即操作数包含在指令中。立即操作数有8位和16位两种形式,前面加“#”来表示。例如:MOVA,#0FFH;将立即数FFH传送到累加器A中MOVDPTR,#2000H;将16位立即数2000H传送到数据指针DPTR中MOV40H,#60H;将立即数60H传送到40H内部RAM单元中2.直接寻址指令中直接给出操作数地址的寻址方式。例如:MOVA,30H;内部RAM中的30H单元中的数据内容传送到累加器A中3.寄存器寻址以通用寄存器的内容为操作数的寻址方式。通用寄存器包括:A,B,DPTR,R0~R7。其中,R0~R7必须在工作寄存器组之中。例如:INCR0;(R0)+1→R0需要注意的是,A和B既是通用寄存器,又是具有直接地址的特殊功能寄存器。4.寄存器间接寻址以寄存器中的内容为地址,该地址中的内容为操作数的寻址方式。能够用于寄存器间接寻址的寄存器有:R0,R1,DPTR,SP。其中,R0,R1必须在工作寄存器组之中,SP仅用于堆栈操作。例如:MOVXA,@R1;外部数据RAM中地址为P2R1的单元内容→AMOVX@DPTR,A;A→外部数据RAM中DPTR为地址的单元寄存器间接寻址的存储器空间包括:内部数据RAM和外部数据RAM。内部数据RAM共用128字节,用一个字节的R0或R1可寻址整个空间。外部数据RAM最大可达64K字节,仅用R0或R1无法寻址整个空间。在MCS-51系列单片机指令中,对外部数据RAM作间接寻址有两种方法:第一种由P2口提供高8位外部数据RAM地址,由R0或R1提供低8位地址,由此共同寻址64K空间;第二种是用16位的DPTR作寄存器间接寻址。5.变址寻址由程序计数器PC或DPTR加上偏移量寄存器A中内容之和形成操作数地址的寻址方式。变址寻址只能对程序存储器中的数据作寻址操作。由于程序存储器是只读存储器,因此变址寻址操作只有读操作而无写操作。例如:MOVCA,@A+DPTR;A+DPTR为地址的存储器单元内容→AMOVCA,@A+PC;A+PC为地址的存储器单元内容→A6.相对寻址以当前程序计数器PC的内容为基址,加上指令给出的一字节补码数形成新的PC值的寻址方式。PC中的当前值称为基地址,一字节补码数称为偏移量,新的PC值称为转移目的地址。例如:JC80H;C=1跳转表示进位位C为0,则程序计数器PC中的内容不改变,即不转移。若进位位C为1,则PC中的当前值为基地址加上偏移量80H后所得到的结果作为该转移指令的目的地址。7.位寻址对位地址中的内容作位操作的寻址方式。单片机中只有内部RAM和特殊功能寄存器的部分单元有位地址(两者统一编址,地址空间为00H~FFH),因此,位寻址只能对有位地址的这两个空间作寻址操作。例如SETB20H;1→20H位MOV32H,C;进位位C→32H位3.2分类指令MCS-51单片机共有111条指令,按功能分类,MCS-51指令系统可分为5大类:数据传送类指令(共29条)算术操作类指令(共24条)逻辑操作类指令(共24条)控制转移类指令(共17条)布尔变量操作类指令(共17条)1.数据传送类指令(共29条)以累加器A为目的操作数类指令(4条)MOVA,dirMOVA,#dataMOVA,RnMOVA,@Ri例:R1=20H,(20H)=55H,指令MOVA,@R1执行后,A=55H。以寄存器Rn为目的操作数的指令(3条)MOVRn,dirMOVRn,#dataMOVRn,A例:(40H)=30H,指令MOVR7,40H执行后,R7=30H。以直接地址为目的操作数的指令(5条)MOVdir,dirMOVdir,#dataMOVdir,AMOVdir,RnMOVdir,@Ri例:R0=50H,(50H)=10H,指令MOV35H,@R0执行后,(35H)=10H。以间接地址为目的操作数的指令(3条)MOV@Ri,dirMOV@Ri,#dataMOV@Ri,A查表指令(2条)MOVCA,@A+DPTRMOVCA,@A+PC例:A=20H,DPTR=2000H,指令MOVCA,@A+DPTR执行后,程序存储器2020H单元中的内容送入A。累加器A与片外数据存储器RAM传送指令(4条)MOVX@DPTR,AMOVXA,@DPTRMOVXA,,@RiMOVX@Ri,A例:DPTR=2000H,外部RAM中(2000H)=18H,指令MOVXA,@DPTR执行后,A=18H。例:P2=10H,R1=50H,A=64H,指令MOVX@R1,A执行后,外部RAM中(1050H)=64H。堆栈操作类指令(2条)PUSHdirPOPdir例:SP=07H,(35H)=55H,指令PUSH35H执行后,55H送入08H地址单元,SP=08H。交换指令(5条)XCHA,RnXCHA,@RiXCHA,dirXCHDA,@RiSWAPA例:A=80H,R0=32H,(32H)=FFH,指令XCHDA,@R0执行后,A=8FH,(32H)=F0H。16位数据传送指令(1条)MOVDPTR,#data162.算术操作类指令(共24条)算术运算指令共有24条,算术运算主要是执行加、减、乘、除、增量、减量和十进制调整等指令。加法指令(4条)ADDA,#dataADDA,dirADDA,RnADDA,@Ri带进位加法指令(4条)ADDCA,dirADDCA,#dataADDCA,RnADDCA,@Ri带借位减法指令(4条)SUBBA,dirSUBBA,#dataSUBBA,RnSUBBA,@Ri例:A=38H,R1=20H,(20H)=23H,C=1,指令SUBBA,@R1执行后,A=14H乘法指令(1条)MULA,B例:A=50H,B=A0H,指令MULA,B执行后,A=00H,B=32H除法指令(1条)DIVA,B例:A=28H,B=12H,指令DIVA,B执行后,A=02H,B=04H加1指令(5条)INCAINCdirINC@RiINCRnINCDPTR减1指令(4条)DECADECdirDEC@RiDECRn十进制调整指令(1条)DAA;把A中按二进制相加后的结果调整成按BCD数相加的结果3.逻辑操作类指令(共24条)逻辑操作类指令共有24条,主要包括“与”、“或”、“异或”、求反、清0、左右移位等逻辑操作。“与”操作指令(6条)ANLA,dirANLA,#dataANLA,RnANLA,@RiANLdir,AANLdir,#data“或”操作指令(6条)ORLA,dirORLA,#dataORLA,RnORLA,@RiORLdir,AORLdir,#data例:(21H)=3AH,A=14H,指令ORL21H,A执行后,(21H)=3EH。“异或”操作指令(6条)XRLA,dirXRLA,#dataXRLA,RnXRLA,@RiXRLdir,AXRLdir,#data例:R0=23H,(23H)=32H,A=53H,指令XRLA,@R0执行后,A=61H求反指令(1条)CPLA;累加器A中的内容按位取反清零指令(1条)CLRA;0→A,累加器A中的内容清零循环移位指令(4条)RLA;累加器A中的内容左移一位RRA;累加器A中的内容右移一位RLCA;累加器A中的内容连同进位位C左移一位RRCA;累加器A中的内容连同进位位C右移一位例:A=16H,指令RRA执行后,A=0BH。4.控制转移类指令(共17条)控制转移类指令共有17条,用于控制程序的流向,所控制的范围即为程序存储器区间。子程序调用指令(4条)绝对调用指令ACALLaddr11绝对调用指令实现在2K地址范围内的子程序调用。本指令实现的操作将不改变原PC的高5位,仅把11位地址addr11送入PC的低11位,以此确定子程序的入口地址。由于整个64K程序存储器空间被分成32个基本2K地址范围(见表3.4),编程时,必须保证紧接ACALL指令后面的那一条指令的第一字节与被调用子程序的入口地址在同一2K范围内,否则将不能使用ACALL指令实现这种调用。表3.4程序存储器空间中的32个基本2K地址范围0000H~07FFH5800H~5FFFHB000H~B7FFH0800H~0FFFH6000H~67FFHB800H~BFFFH1000H~17FFH6800H~6FFFHC000H~C7FFH1800H~1FFFH7000H~77FFHC800H~CFFFH2000H~27FFH7800H~7FFFHD000H~D7FFH2800H~2FFFH8000H~87FFHD800H~DFFFH3000H~37FFH8800H~8FFFHE000H~E7FFH3800H~3FFFH9000H~97FFHE800H~EFFFH4000H~47FFH9800H~9FFFHF000H~F7FFH4800H~4FFFHA000H~A7FFHF800H~FFFFH5000H~57FFHA800H~AFFFH长调用指令LCALLaddr16长调用指令为64K程序存储器空间中的全范围子程序调用指令,子程序的入口地址可在64K地址空间中的任一处。子程序返回指令RET从被调用子程序返回,与LCALL与ACALL指令配合使用。中断返回指令RETI从中断服务程序中返回,除具有RET功能外,还具有恢复中断逻辑的功能,需注意的是,RETI指令不能用RET代替。转移类指令(12条)转移类指令分为无条件转移指令和条件转移指令两组。无条件转移指令包括绝对转移、长转移、短转移和间接转移;条件转移指令包括结果为零、结果为非零、减“1”后结果为非零以及两数不相等的转移条件等,它们全部采用相对转移的方式。无条件转移指令(4条)【1】长转移指令LJMPaddr16长转移指令为64K程序存储器空间的全范围转移指令,转移地址可为16位地址值中的任一值。【2】绝对转移指令AJMPaddr11绝对转移指令为2K地址范围内的转移指令,对转移目的地址的要求与ACALL指令中对子程序入口地址的要求相同。【3】短转移指令SJMPrel;PC+2+rel→PC短转移指令为一页地址范围内的相对转移指令。因为rel为1字节补码偏移量,且SJMPrel指令为2字节指令,所以转移范围为-126D~+129D【4】间接转移指令JMP@A+DPTR条件转移指令(8条)【1】累加器为零/非零转移指令JZrel若A=0,则PC+2+rel→PC,程序转移到偏移量所指向的地址若A≠0,程序顺序执行JNZrel若A≠0,则PC+2+rel→PC,程序转移到偏移量所指向的地址若A=0,程序顺序执行【2】两数不等转移指令CJNEA,dir,rel若A≠(dir),则PC+3+rel→PC,程序转移;若A=(dir),则程序顺序执行CJNEA,#data,rel若A≠#data,则PC+3+rel→PC,程序转移;若A=#data,则程序顺序执行CJNERn,#data,rel若Rn≠#data,则PC+3+rel→PC,程序转移;若Rn=#data,则程序顺序执行CJNE@Ri,#data,rel若((Ri))≠#data,则PC+3+rel→PC,程序转移;若((Ri))=#data,则程序顺序执行【3】减“1”非零转移指令DJNZRn,relRn-1→Rn,若Rn≠0,则PC+2+rel→PC,程序转移;若Rn=0,则程序顺序执行DJNZdir,rel(dir)-1→dir,若(dir)≠0,则PC+3+rel→PC,程序转移;若(dir)=0,则程序顺序执行需要注意的是,DJNZRn,rel是2字节指令,而DJ