资源描述
单击此处编辑母版标题样式,单击此处编辑母版文本样式,第二级,第三级,第四级,第五级,*,*,4.1 汇编语言语句种类及其格式,第四章 汇编语言程序格式,不同的汇编程序有不同的汇编语言编程规定。目前支持Intel8086/8088系列微机,常用的汇编程序有ASM、MASM、TASM、OPTASM等 。,本章主要介绍汇编语言程序设计中的一些基本书写格式与语法规则。,1,4.1 汇编语言语句种类及其格式第四章 汇编语言程序格式,每一条指令语句在汇编时都要产生一个可供CPU执行的机器目标代码,它又叫可执行语句。,一、指令语句,指令语句的一般格式为:,标号,指令助记符,操作数,注释,;,:,,,一条指令语句最多可以包含4个字段,汇编语言的语句可以分为指令语句、伪指令语句和宏指令语句(宏调用语句)。,2,每一条指令语句在汇编时都要产生一个可供CPU执行,标号是可选字段,它后面必须有“:”。标号是一条指令的符号地址,代表了该指令的第一个字节存放地址。,1.标号字段,标号一般放在一个程序段或子程序的入口处,控制程序的执行转到该程序位置。,例 ADDR1:MOV AL,100,在转移指令或子程序调用指令中,可直接引用这个标号。,3,标号是可选字段,它后面必须有“:”。标号是一条指令的符号地址,该字段是一条指令的必选项,它表示这条语句要求CPU完成什么具体操作,如MOV、ADD、SHL等。,2.指令助记符字段,一条指令可以有一个操作数、两个操作数或者无操作数。,3.操作数字段,有些指令还可以在指令助记符的前面加上前缀,实现一定的附加操作。如串操作指令前所加的重复前缀REP(见第7章介绍)等。,如ADD、MOV指令需要两个操作数,INC、NOT指令只需一个操作数,而CLC指令不需要操作数。,4,该字段是一条指令的必选项,它表示这条语句要求CPU完成什么具,4.注释字段,例:LABEL1:ADD AX,BX,;功能为AXMOV AX ,0FF0FHMOV AL , NOT 0F0H =MOV AL , 0FHMOV BL , 55H AND 0F0H =MOV BL,50HMOV BH , 55H OR 0F0H =MOV BH, 0F5HMOV CL , 55H XOR 0F0H =MOV CL, 0A5H,三、关系运算符,关系运算符包括:EQ(等于)、NE(不等于)、LT(小于)、 LE(小于等于)、GT(大于)、 GE(大于等于),39,二、逻辑运算符 逻辑运算符有NOT、AND、O,关系运算符用于比较两个表达式的大小。关系运算符比较的两个表达式必须同为常数或同一逻辑段中的变量。,如果是常量的比较,则按无符号数进行比较;如果是变量的比较,则比较它们的偏移量的大小。,关系运算的结果只能是“真”(全1)或“假”(全0),MOV AX,0FH EQ 1111B =MOV AX ,0FFFFHMOV BX , 0FH NE 1111B =MOV BX, 0,例1:,VAR DW NUM LT 0ABH,该语句在汇编时,根据符号常量NUM的大小来决定VAR存储单元的值,当NUM0ABH,时,则变量VAR的内容为0FFFFH,否则VAR的内容为0。,例2,40,关系运算符用于比较两个表达式的大小。关系运算,四、数值返回运算符,该类运算符有5个,它们将变量或标号的某些特征值或存储单元地址的一部分提取出来。,取变量或标号所在段的段基值。,1.SEG运算符,DATA SEGMENT K1 DW 1,2 K2 DW 3,4 MOV AX , SEG K1MOV BX , SEG K2,例如:,设DATA逻辑段的段基值为1FFEH,则两条传送指令将被汇编为:,MOV AX ,1FFEHMOV BX ,1FFEH,作用,41,四、数值返回运算符 该类运算符有5个,它们将变量,该运算符的作用是取变量或标号在段内的偏移量。,2.OFFSET运算符,例如:,DATA SEGMENTVAR1 DB 20H DUP(0)VAR2 DW 5A49HADDR DW VAR2 ;将VAR2的偏移量20H存入ADDR中 . MOV BX,VAR2;(BX)=5A49H MOV SI , 0FFSET VAR2 ;(SI)=20H MOV DI, ADDR ;DI,的内容与SI相同 MOV BP,OFFSET ADDR ;(BP)=22H,42,该运算符的作用是取变量或标号在段内的偏移量。2.OFFSET,该运算符的作用为取变量或标号的类型属性,并用数字形式来表示。对变量来说就是取它的字节长度。,3.TYPE运算符,变量,BYTE 1WORD 2DWORD 4,标号,NEAR -1FAR -2,例如:,V1 DB ABCDEV2 DW 1234H, 5678HV3 DD V2 ;,存放V2的段基值和偏移量 MOV AL , TYPE V1 MOV CL , TYPE V2 MOV CH , TYPE V3,上述程序段汇编后的等效指令序列如下:,43,该运算符的作用为取变量或标号的类型属性,并用数字形式来表示。,MOV AL,01H,MOV CL,02H,MOV CH,04H,该运算符只能加在变量的前面。如果变量是用重复数据操作符DUP说明的,则返回外层DUP给定的值。如果没有用DUP说明,则返回值总是1。,4.LENGTH运算符,K1 DB 10H DUP(0)K2 DB 10H,20H,30H,40HK3 DW 20H DUP(0,1,2 DUP(0)K4 DB ABCDEFGH.,44,MOV AL,01H 该运算符只能加在变量的,MOV AL, LENGTH K1; (AL)=10HMOV BL, LENGTH K2 ; (BL)=1MOV CX, LENGTH K3 ; (CX)=20HMOV DX, LENGTH K4 ; (DX)=1,该运算符只能作用于变量,SIZE取值等于LENGTH和TYPE两个运算符返回值的乘积。,5.SIZE运算符,例如,对于上面例子,加上以下指令:,MOV AL ,SIZE K1 ;(AL)=10HMOV BL,SIZE K2 ; (BL)=1MOV CL,SIZE K3 ; (CL)=20H*2=40HMOV DL,SIZE K4 ; (DL)=1,45,MOV AL, LENGTH K1; (AL)=1,五、属性修改运算符,这一类运算符用来对变量、标号或存储器操作数的类型属性进行修改或指定。,1.PTR运算符,类型 PTR 地址表达式,使用格式:,作用,: 将地址表达式所指定的标号、变量或用其它形式表示的存储器地址的类型属性修改为 “类型”所指的值。,类型可以是BYTE、WORD、DWORD、NEAR和FAR。这种修改是临时性的,只在含有该运算符的语句内有效。,46,五、属性修改运算符 这一类运算符用来对变量、标,例如:,DA_BYTE DB 20H DUP(0)DA_WORD DW 30H DUP(0) .MOV AX , WORD PTR DA_BYTE10ADD BYTE PTR DA_WORD20, BLINC BYTE PTR BX SUB WORD PTR SI, 100JMP FAR PTR SUB1,;,指明SUB1不是本段中的地址,2.HIGH/LOW运算符,HIGH 表达式,LOW 表达式,使用格式:,这两个运算符用来将一个数据分离出高字节和低字节。,47,例如:DA_BYTE DB 20H DUP(0),如果表达式为一个常量,则将其分离成高8位和低8位;如果表达式是一个地址(段基值或偏移量)时,则分离出它的高字节和低字节。,例如:,DATA SEGMENTCONST EQU 0ABCDHDA1 DB 10H DUP (0)DA2 DW 20H DUP (0)DATA ENDS .MOV AH ,HIGH CONSTMOV AL, LOW CONSTMOV BH, HIGH (OFFSET DA1)MOV BL, LOW (OFFSET DA2)MOV CH, HIGH (SEG DA1)MOV CL, LOW (SEG DA2),设DATA段的段基值是0926H,则上述指令序列汇编后的等效指令为:,MOV AH , 0ABHMOV AL , 0CDHMOV BH , 00HMOV BL , 10HMOV CH , 09HMOV CL , 26H,48,如果表达式为一个常量,则将其分离成高8位和低8位;如果表达式,注意,:HIGH/LOW运算符不能用来分离一个变量、寄存器或存储器单元的高字节与低字节。,下面语句使用是错误的用法。,DA1 DW 1234HMOV AH,HIGH DA1MOV BH, LOW AXMOV CH,HIGH SI,49,注意:HIGH/LOW运算符不能用来分离一个变量、寄存器或存,3、THIS运算符,THIS运算符一般与等值运算符EQU连用,用来定义一个变量或标号的类型属性。所定义的变量或标号的段基值和偏移量与紧跟其后的变量或标号相同。,例如:,DATA_BYTE EQU THIS BYTEDATA_WORD DW 10 DUP (0), MOV AX, DATA_WORD MOV BL, DATA_BYTE .,又如:,LFAR EQU THIS FARLNEAR:MOV AX,B,标号LFAR与LNEAR具有相同的逻辑地址值,但类型不同。LNEAR只能被本段中的指令调用,而LFAR可以被其它段的指令调用。,50,3、THIS运算符 THIS运算符一般与等值运,六、运算符的优先级,在一个表达式中如果存在多个运算符时,在计算时就有先后顺序问题。不同的运算符具有不同的运算优先级别。,优先级别 运算符,(最高)1 LENGTH,SIZE ,圆括号 2 PTR,OFFSET,SEG,TYPE,THIS 3 HIGH,LOW 4 *,/,MOD,SHR,SHL 5 +,- 6 EQ,NE,LT,LE,GT,GE 7 NOT 8 AND(最低)9 OR,XOR,51,六、运算符的优先级在一个表达式中如果存在多个运算符时,在计算,汇编程序在计算表达式时,按以下规则进行运算。,先执行优先级别高的运算,再算较低级别运算;,相同优先级别的操作,按照在表达式中的顺序,从左到右进行;,可以用圆括号改变运算的顺序,。,例如:,K1= 10 OR 5 AND 1 ;结果为K1=11K2=(10 OR 5) AND 1 ;结果为K2=1,52,汇编程序在计算表达式时,按以下规则进行运算。先执行优先级别高,4.5 程序的段结构,8086/8088在管理内存时,需要按照逻辑段进行划分,不同的逻辑段可以用来存放不同目的的数据。在程序中使用四个段寄存器CS,DS,ES和SS来访问它们。,在源程序设计时,使用伪指令来定义和使用这些逻辑段。,53,4.5 程序的段结构 8086/8088在管,段名 SEGMENT 定位类型 组合类型 类别名,.,. . 本段语句序列, .,段名 ENDS,一般格式:,一、段定义伪指令,伪指令SEGMENT和ENDS用于定义一个逻辑段。使用时必须配对,分别表示定义的开始与结束。,54,段名 SEGMENT 定位类型 组合类型 ,段名是由用户自己任意选定的,符合标识符定义规则的一个名称。,段定义伪指令语句各部分的作用如下:,1、段名,最好选用与该逻辑段用途相关的名称。如第一个数据段为DATA1,第二个数据为DATA2等。,一个段的开始与结尾用的段名必须一致。,55,段名是由用户自己任意选定的,符合标识符定义,2、定位类型,定位类型用于决定段的起始边界,即第一个可存放数据的位置(不是段基址)。它可以有4种取值。,由于一个页面为256个字节,并且页面编号从0开始,因此,PAGE定位类型的段起始地址的最后8位二进制数一定为0,即以00H结尾的地址。,(1),PAGE,: 表示该段从一个页面的边界开始,(2),PARA,:表示该段从一个小节的边界开始,如果用户未选定位类型,则缺省为PARA。,56,2、定位类型 定位类型用于决定段的起始边界,即第,(3),WORD,:表示该段从一个偶数字节地址开始,即段起始单元地址的最后一位二进制数一定是0。,(4),BYTE,:表示该段起始单元地址可以是任一地址值。,注意,:定位类型为PAGE和PARA时,段起始地址与段基址相同。定位类型为WORD和BYTE时,段起始地址与段基址可能不同。,57,(3)WORD:表示该段从一个偶数字节地址开始,即段起始单元,(1),NONE:,若未指定组合类型,表示本段与其它段无连接关系。在装入内存时,本段有自己的物理段,因此有自己的段基址。,3、组合类型,组合类型说明符用来指定段与段之间的连接关系和定位。它有六种取值选择。,(2),PUBLIC,:在满足定位类型的前提下,将与该段同名的段邻接在一起,形成一个新的逻辑段,共用一个段基址。段内的所有偏移量调整为相对于新逻辑段的段基址。,(3),COMMON,: 产生一个覆盖段。在多个模块连接时,把该段与其它也用COMMON说明的同名段置成相同的段基址,这样就达到了共享同一存储区。共享存储区的长度由同名段中最大的段确定。,58,(1) NONE:若未指定组合类型,表示本段与其它段无连接关,(4),STACK,:把所有同名段连接成一个连续段,且系统自动对SS段寄存器初始化为该连续段的段基址。并初始化堆栈指针SP。,用户程序中应至少有一个段用STACK说明,否则需要用户程序自己初始化SS和SP。,(5),AT表达式,:表示本段可定位在表达式所指示的小节边界上。表达式的值也就是段基值。,(6),MEMORY,:表示本段在存储器中应定位在所有其它段这之后的最高地址上。如果有多个用MEMORY说明的段,则只处理第一个用MEMORY说明的段。其余的被视为COMMON.,59,(4)STACK:把所有同名段连接成一个连续段,且系统自动对,类别名为某一个段或几个相同类型段设定的类型名称。系统在进行连接处理时,把类别名相同的段存放在相邻的存储区,但段的划分与使用仍按原来的设定。,4.类别名,类别名必须用单引号引起来。所用字符串可任意选定,但它不能使用程序中的标号、变量名或其它定义的符号。,在定义一个段时,段名是必须有的项,而定位类型、组合类型和类别名三个参数是可选项。各个参数之间用空格分隔。各参数之间的顺序不能改变。,下面是一个分段结构的源程序框架。,60,类别名为某一个段或几个相同类型段设定的类型名,STACK1,SEGMENT PARA STACK STACK0,.,STACK1,ENDS,DATA1,SEGMENT PARA DATA,.,DATA1,ENDS,STACK2,SEGMENT PARA STACK0,.,STACK2,ENDS,CODE,SEGMENT PARA MEMORY,ASSUME CS:CODE,DS:DATA1,SS:STACK1,MAIN,: .,.,CODE,ENDS,END MAIN,DATA2,SEGMENT BYTE DATA,.,DATA2,ENDS,61,STACK1 SEGMENT PARA STACK,上述源程序经LINK程序进行连接处理后,程序被装入内存的情况如右图所示。,如果在段定义中选用了PARA定位类型说明,则在一个段的结尾与另一个段的开始之间可能存在一些空白,图中以兰色框表示。CODE段的组合类型为MEMORY,因此被装入在其它段之后。,在进行程序设计时,如果程序不大,一般只需要定义三个段就可以了。,0000H,STACK1,STACK2,DATA1,DATA2,CODE,间隔,62,上述源程序经LINK程序进行连接处理后,程序,二、段寻址伪指令,段寻址伪指令ASSUME的作用是告诉汇编程序,在处理源程序时,定义的段与哪个寄存器关联。,ASSUME 段寄存器名:段名,段寄存器名:段名,.,一般格式 :,其中段寄存器名为CS,DS,ES和SS四个之一,段名是用SEGMENT/ENDS伪指令定义的段名。,ASSUME并不设置各个段寄存器的具体内容,段寄存器的值是在程序运行时设定的。,63,二、段寻址伪指令 段寻址伪指令ASSUME的,DATA1,SEGMENT,VAR1,DB 12H,DATA1,ENDS,DATA2,SEGMENT,VAR2,DB 34H,DATA2,ENDS,CODE,SEGMENT,VAR3,DB 56H,ASSUME CS:CODE,DS:DATA1,ES:DATA2,START,: .,.,INC VAR1,INC VAR2,INC VAR3,.,CODE,ENDS,END START,例如:,该指令汇编时,VAR1使用的是DS,该指令被汇编时,VAR2使用的是ES,即指令编码中有段前缀,64,DATA1 SEGMENT例如:该指令汇编时,VAR1使用,在一个代码段中可以有几条ASSUME伪指令,对于前面的设置,可以用ASSUME改变原来的设置。,一条ASSUME语句不一定设置全部段寄存器,可以选择其中一个或几个段寄存器。,可以使用关键字NOTHING将前面的设置删除。,ASSUME ES:NOTHING ;删除前面对ES与某个定义段的关联,ASSUME NOTHING ;删除全部4个段寄存器的设置,例如:,65,在一个代码段中可以有几条ASSUME伪指令,对于前面的设置,,三、段寄存器的装入,段寄存器的初值(段基值)装入需要用程序的方法来实现。四个段寄存器的装入方法略有不同。,在程序中,使用数据传送语句来实现对DS和ES的装入。,1、DS和ES的装入,66,三、段寄存器的装入 段寄存器的初值(段基值)装,DATA1 SEGMENT,DBYTE1 DB 12H,DATA1 ENDS,DATA2 SEGMENT,DBYTE2 DB 14H DUP(?),DATA2 ENDS,CODE SEGMENT,ASSUME CS:CODE,DS:DATA1,START: MOV AX,DATA1,MOV DS,AX,MOV AX,DATA2,MOV ES,AX,MOV AL,DBYTE1,MOV DBYTE22,AL,.,CODE ENDS,例如:,该指令在汇编时出错,因为在ASSUME指令中未指定ES与DATA2的联系。,67,DATA1 SEGMENT 例如:该指令在汇编时出错,,为了改正上述程序中的错误,可以在变量DBYTE2前加一个段前缀说明即可。即:,MOV ES:DBYTE22, AL,SS的装入有两种方法,2、SS的装入,(1)在段定义伪指令的组合类型项中,使用STACK参数,并在段寻址伪指令ASSUME语句中把该段与SS段寄存器关联。,68,为了改正上述程序中的错误,可以在变量DBYTE,STACK1 SEGMENT PARA STACK,DB 40H DUP(?),STACK1 ENDS,.,CODE SEGMENT,ASSUME CS:CODE,SS:STACK1,.,例如:,SS将被自动装入STACK1段的段基值,堆栈指针SP也将指向堆栈底部+2的存储单元。上例中(SP)=40H。,(2)如果在段定义伪指令的组合类型中,未使用STACK参数,或者是在程序中要调换到另一个堆栈,这时,可以使用类似于DS和ES的装入方法。,69,STACK1 SEGMENT PARA STACK例如,DATA_STACK SEGMENT,DB 40H DUP(?),TOP LABEL WORDDATA_STACK ENDS,.,CODE SEGMENT,.,MOV AX,DATA_STACK,MOV SS,AX,MOV SP,OFFSET TOP,.,例如:,TOP变量的偏移量为40H,70,DATA_STACK SEGMENT例如:TOP变量的偏,CS保存的是当前目标代码的段基值,而IP提供下一条将要执行的目标代码的偏移量。CPU在执行指令之前根据CS和IP的内容来从内存中提取指令。因此,必须在程序执行之前装入CS和IP的值。由于这时程序还没有运行,就无法用几条可执行语句来装入CS和IP。,3、CS的装入,END 起始地址,装入CS和IP一般有下面两种情况。,(1)按照结束伪指令指定的地址装入CS和IP,任何一个源程序都必须以END伪指令来结束。,起始地址可以是一个标号或表达式,它与程序中第一条指令语句前所加的标号必须一致。,其格式为:,71,CS保存的是当前目标代码的段基值,而IP提供,END伪指令的作用是标识源程序结束和指定程序运行时的起始地址。当程序被装入内存时,根据起始地址的段基值和偏移量分别自动装入CS和IP中。,.,CODE SEGMENT,ASSUME CS:CODE,.,START: .,.,CODE ENDS,END START,例如:,72,END伪指令的作用是标识源程序结束和指定程序,(2)在程序运行期间,当执行某些指令时,CPU自动修改CS和IP,使它们指向新的代码段。,执行段间过程调用CALL和段间返回指令RET;,执行段间无条件转移指令JMP;,响应中断及中断返回指令;,执行硬件复位操作。,例如:,73,(2)在程序运行期间,当执行某些指令时,CPU自动修改CS和,在程序设计过程中,常常将具有一定功能的程序段设计成一个子程序。在MASM宏汇编程序中,用过程(PROCEDURE)来构造子程序。,4.6 过程定义伪指令(PROC/ENDP),过程名 PROC NEAR/FAR,.,RET,.,过程名 ENDP,过程定义伪指令格式如下:,74,在程序设计过程中,常常将具有一定功能的程序段设计成一个子程,其中,过程名是该子程序的名称,它被用作过程调用指令CALL的目的操作数。它类同一个标号的作用。具有段、偏移量和距离三个属性。而距离属性使用NEAR和FAR来指定,若没有指定,则隐含为NEAR。,NEAR过程只能被本段指令调用,而FAR过程可以供其它段的指令调用。,每一个过程中必须包含有返回指令RET,其作用是控制CPU从子程序中返回到调用该过程的主程序。,75,其中,过程名是该子程序的名称,它被用作过程调用指令CALL的,4.7 当前位置计数器$与定位伪指令ORG(Origin),汇编程序在汇编源程序时,每遇到一个逻辑段,就要为其设置一个位置计数器,它随时记录着定义的每一个数据或每一条指令在程序中的位置。,在源程序中,使用符号$来表示位置计数器的当前值。因此,$被称为当前计数器。它位于不同的位置具有不同的值。,位置计数器$在使用上完全类似变量的使用.,76,4.7 当前位置计数器$与定位伪指令ORG(Origin,作用:将数值表达式的值赋给当前位置计数器。ORG语句为其后的数据或指令设置起始偏移量。,表达式的值必须为正值。表达式中也可以包含有当前位置计数器的现行值$。,定位伪指令ORG,用来改变位置计数器的值。,ORG 数值表达式,格式:,77,作用:将数值表达式的值赋给当前位置计数器。ORG语句为其后的,DATA1 SEGMENT,ORG 30H,DB1 DB 12H,34H ;DB1在DATA1段内的偏移量为30H,ORG $+20H;保留20H个字节单元,其后再存放ABCD.,STRING DB ABCDEFGHI,COUT EQU $-STRING;计算STRING的长度,DB2 DW $; 取$的偏移量,类似变量的用法,DB3 DB $ ;,此语句错误!,DATA1 ENDS,CODE SEGMENT,ASSUME CS:CODE.,ORG 10H,START: MOV AX,DATA,MOV DS,AX,.,CODE ENDS,END START,78,DATA1 SEGMENT78,4.8 标题伪指令TITLE,TITLE 标题名,语句格式:,作用:给所在程序指定一个标题。以便在列表文件的每一页的第一行都显示这个标题。其中标题是用户任意选用的字符串,字符个数不能超过60。,79,4.8 标题伪指令TITLETITLE 标题名语句格式:,一、使用程序段前缀PSP(Program Segment Prefix)实现返回,4.10 从程序返回操作系统的方法,为了使程序运行结束后,能够正确地返回到操作系统,需要在用户程序中加上一些必要的语句。一般有以下两种方法。,DOS系统将一个.EXE(可执行文件)装入内存时,在该文件的前面生成一个程序段前缀PSP,其长度为100H字节。同时让DS和ES都指向PSP的开始,而CS指向该程序的代码段,即第一条可执行指令。,80,一、使用程序段前缀PSP(Program Segment P,为了使程序执行完后,正确返回DOS,需要做以下三个工作:,1. 将用户程序编制成一个过程,类型为FAR;,2. 将PSP的起始逻辑地址压栈,即将INT 20H指令的地址压栈;,如图所示。PSP中一开始就是一条中断指令 INT 20H,执行该指令将终止用户程序,返回DOS系统。,81,为了使程序执行完后,正确返回DOS,需要做以下三个工作: 1,3. 在用户程序结束时,使用RET指令。执行该指令将使保存在堆栈中的INT 20H 的逻辑地址弹出到CS和IP中。,DATA SEGMENT,DATA ENDS,STACK1 SEGMENT STACK,STACK1 ENDS,CODE SEGM
展开阅读全文