高级汇编语言技术
合集下载
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
宏定义 macro_name MACRO [哑元表] ; 形参/虚参 [ LOCAL 标号表 ] …… …… END母(注意与名称的差异) * [哑元表] 形参表,哑元之间用逗号隔开 * 如果宏定义体有一个或多个标号,则必须用LOCAL伪操作列出所有的标号。???
[格式] RCPT 表达式 …. RNDM
7.2.1 重复伪操作
IPR伪操作 [格式] IRP 形式参数,<参数表> … (指令体) ENDM
[说明]其中,IRP和ENDM伪指令必须成对出现,其间是要重复的指令体。重复的次数由参数表中参数的个数所决定,而且每重复一次,依次用参数表中的参数来代替形式参数。参数表必须用尖括号括起来,参数间用逗号分隔。
宏库的建立: 由定义即可
7.1.7 PURGE伪操作
宏指令名可与指令助记符和伪指令名同名,但这时宏指令的优先级最高,而同名的指令和伪操作便失效。
格式: PURGE 宏指令名[,宏指令名,…] PURGE可取消多个宏定义,而各宏指令名之间应以逗号分 隔。 功能:取消宏定义,使该宏定义为空。
7.1 宏汇编 为用户提供更加容易、更加灵活、更加向高级语言靠拢的汇编工具。 宏:源程序中一段有独立功能的程序代码。 宏指令:用户自定义的指令。在编程时,将多次使用的功能 用一条宏指令来代替。
汇编语言程序语句
指令 伪指令(伪操作) 宏指令
子程序: 优点:模块化,节省内存,可被多次调用,编程效率高。 缺点:额外开销(保存返回地址,计算转向地址,传递参 数等)大,增加了执行时间。适用于子功能代码较 长、调用比较频繁的情况。 宏调用: 优点:参数传送简单,执行效率高。 缺点:不节省空间,适用于子功能代码较短、传参较多的 情况。
MASM6.0+
;; 注释 ; 宏展开时,;;后面的注释不予展开。
源程序(ASM文件)经汇编程序汇编,主要输出是OBJ文件和LST文件(列表文件)。LST文件同时列出源程序和机器语言程序清单,并给出符号表。
7.1.6 宏库的建立与调用
宏库:只包含若干宏定义的文件,扩展名为MAC或LIB 宏库的调用: 在程序的开始用INCLUDE语句说明,例如 INCLUDE C:\MACRO.MAC INCLUDE D:\AAA\EEE.LIB 注:INCLUDE语句可以不放在程序的最前面,但必须放在所有的宏 调用之前
宏展开 源程序被汇编时,汇编程序把宏调用展开,也就是把宏定义体复制到调用宏指令的位置上,同时用实元取代哑元,由LOCAL定义的标号也由??0000~??FFFF替代。
实元(在使用上要比子程序灵活) 常数、寄存器、存贮单元 用寻址方式能找到的地址、表达式 指令的操作码、操作码的一部分
宏调用与子程序调用的区别: 子程序:在程序执行期间被调用,只占用自身大小的一段空间。
宏汇编操作符 & ;; % 符号1 & 符号2 ; 宏展开时,合并前后两个符号形成一个符号。
leap macro cond, lab j&cond lab endm
宏定义:
宏调用:
leap z, there …… leap nz, here
[格式]LOCAL 局部标号表 [说明] 其中,局部标号表中的各标号之间用逗号分隔。汇编程序在汇编时对局部中标号表中的各标号建立唯一的符号(??000一??FFFF)以取代展开后的各局部标号。该伪指令必须是宏定义体中的第一个语句,而且在MACR O和LOCAL之间还不允许有注释和分号标志。
7.1.4 在宏定义体内使用宏
(2)子程序中的这组指令的机器码在存储器中只需要存放一次;而宏调用则不同,每调用一次就把宏定义体展开一次,因而它占有的存储空间与调用次数有关,调用次数越多则占有的存储空间就越大。
(3)子程序调用时需要用堆栈,调用子程序和返回主程序都要开销必要的时间;而宏调用其宏定义体展开是插入在调用程序中,因此避免了调用时跳到宏和由宏返回所开销的时间。
7.2 重复汇编
在编写汇编语言程序中,有时需要进行连续地重复完成相同的或几乎完全相同的操作,这时可以使用重复汇编,使书写的程序变得相当简洁。
重复汇编有两种格式
重复伪操作 不定重复伪操作
[说明] 其中,REPT和ENDM为伪指令,它们必须成对出现,中间是要重复的指令体,重复的次数由表达式的值来决定。 注意:重复伪操作并不一定要用在宏定义体内。
第七章 高级汇编语言技术
宏汇编 重复汇编 条件汇编
方法2、当这组指令较短或不适合把它写成一个子程序 时,可以把它义成一个宏,每次需要执行时就去调用这个宏。
问 如果在一个程序中多次使用同一组指令时,为了避免每次都要重写,该如何解决?
方法1、把这组指令写成一个子程序,每次需要执行时 只要调用这个子程序就可以了。
限制条件:必须先定义后调用
在宏定义体内不仅可以使用宏, 也可以可以包含宏定义(也就是说,宏定义允许嵌套)
7.1.5 列表伪操作
.XALL 默认值, 只列出产生目标码的宏展开 .LALL 列出包括注释在内的所有宏展开 .SALL 不列出任何展开信息
功能: 控制汇编清单中宏展开的列出情况
.LISTMACROALL .LISTMACRO .NOLISTMACRO
(宏定义体)
一段有独立功能的程序代码
宏调用 macro_name [实元表] ; 实参 * 实元表中的实元与哑元表中的哑元在位置上一一对应; * 若实元数>哑元数,则多余的实元无效; * 若实元数<哑元数,则多余的哑元作“空(NUL)”处理; * 对宏指令的调用必须先定义后调用。 * 应当注意,宏展开后,即用实参取代形式参数之后,所得到的语句应该是有效的,否则汇编时会出错。
宏展开: …… + cmp var, 0 + jge ??0000 + neg var + ??0000: …… + cmp bx, 0 + jge ??0001 + neg bx + ??0001: ……
主程序
…… call sub1 …… call sub1 …… call sub1 ……
子程序 sub1
…… …… …… ret
宏调用:在汇编期间被展开,每调用一次就把宏定义体 展开一次。
宏定义
宏调用
宏展开
Q macro x,y endm
宏展开: + push ax + push bx + push cx + push dx + push si + push di
例 宏指令名与指令助记符或伪操作名相同,宏指令 的优先级最高。 宏定义: add MACRO opr1, opr2, result …… …… ENDM 宏调用: …… add xx, yy, zz purge add ; 取消宏定义 ……
宏展开: + push dx + push ax + mov ax, cx + imul var + mov xyz[bx], ax + pop ax + pop dx
例7.2 保存寄存器 宏定义: savereg MACRO push ax push bx push cx push dx push si push di ENDM 宏调用: savereg
…… Q a,b …… Q c,d …… Q e,f ……
…… …… …… ……
(1)子程序调用是在程序执行期间进行的,主程序通过CALL指令转入于程序的入口地址去执行子程序,子程序执行完后RET指令返回到主程序的调用处(断点处);而宏调用是通过宏指令名和实在参数(如果有要传递的参数的话)进行调用的,而且是在汇编期间展开宏定义体的。
子程序与宏调用工作方式的差异
如何选择用子程序还是用宏调用呢?
可以无变元 可以是操作码 可以是操作码的一部分,但在宏定义体中必须用 &来分隔。 变元是ASCII串 %操作符
7.1.2 宏定义中的参数
7.1.3 LOCAL伪操作
在宏定义中允许使用标号。如果在程序中多次使用该宏定义时。展开后会出现标号的多重定义,这是不允许的。但是可以用伪指令LOCAL指定局部标号。
例 求绝对值(使用LOCAL伪操作) 宏定义: absol MACRO oper LOCAL next cmp oper, 0 jge next neg oper next: ENDM 宏调用: …… absol var …… absol bx ……
伪指令PURGE可取消宏定义,恢复指令的原来功效。
例7.1 两个字相乘 宏定义: multiply MACRO opr1,opr2,result push dx push ax mov ax, opr1 imul opr2 mov result, ax pop ax pop dx ENDM 宏调用: multiply cx, var, xyz[bx]
7.2.2 不定重复伪操作 IPR伪操作
IPRC伪操作 [格式] IRPC 形式参数,字符串(或<字符串>) … (指令体) ENDM
宏展开:
+ jz there …… + jnz here
例7.4
;; 注释 ; 宏展开时,;;后面的注释不予展开。 例:Q MACRO m ; display a message 每次展开保留此注释 ;; m is a string 每次展开不保留此注释 …… ENDM % 表达式 ; 汇编程序将%后面的表达式转换为数字,并在展开期间用这个数取代哑元。