汇编语言伪指令.docx

上传人:b****5 文档编号:3786115 上传时间:2022-11-25 格式:DOCX 页数:18 大小:28.06KB
下载 相关 举报
汇编语言伪指令.docx_第1页
第1页 / 共18页
汇编语言伪指令.docx_第2页
第2页 / 共18页
汇编语言伪指令.docx_第3页
第3页 / 共18页
汇编语言伪指令.docx_第4页
第4页 / 共18页
汇编语言伪指令.docx_第5页
第5页 / 共18页
点击查看更多>>
下载资源
资源描述

汇编语言伪指令.docx

《汇编语言伪指令.docx》由会员分享,可在线阅读,更多相关《汇编语言伪指令.docx(18页珍藏版)》请在冰豆网上搜索。

汇编语言伪指令.docx

汇编语言伪指令

汇编语言伪指令

在汇编语言程序里,有一些特殊的助记符,这些助记符与指令系统的助记符不同,它们没有对应的机器码。

这些助记符在源程序中的作用是完成汇编程序的各种准备工作,包括定义变量、分配数据存储空间、控制汇编过程、定义程序入口等。

它们仅仅在汇编的过程中起作用,一旦汇编过程结束,它们的使命也就完成了。

这些助记符称为伪指令,它们所完成的操作称为伪操作。

不同汇编器的伪指令可能存在少量的区别,并非所有的伪指令在任何编译器上都能被识别。

一、符号定义伪指令

符号定义(SymbolDefinition)伪指令用于定义ARM汇编程序中的变量,对变量赋值和定义寄存器别名等,如表1所列。

表1符号定义伪指令

助记符

指令功能描述

GBLA

定义一个全局的数字变量,并初始化为0

GBLL

定义一个全局的逻辑变量,并初始化为F(假)

GBLS

定义一个全局的字符串变量,并初始化为空

LCLA

定义一个局部的数字变量,并初始化为0

LCLL

定义一个局部的逻辑变量,并初始化为F(假)

LCLS

定义一个局部的字符串变量,并初始化为空

SETA

用于给一个数字变量赋值

SETL

用于给一个逻辑变量赋值

SETS

用于给一个字符串变量赋值

RN

用于给一个特殊的寄存器命名

CN

用于给一个协处理器的寄存器命名

CP

用于给一个协处理器命名

DN

用于给一个双精度的VFP寄存器命名

SN

用于给一个单精度的VFP寄存器命名

FN

用于给一个FPA浮点寄存器命名

RLIST

定义一个通用寄存器列表,根据寄存器编号由低到高访问

实例:

GBLLP_ON;定义全局逻辑变量P_ON

P_ONSETL{TRUE};给全局逻辑变量P_ON赋值为真

LCLANUM;定义局部数字变量NUM

NUMSETA100;给全局数字变量NUM赋值为100

RegListRLIST{R0-R5,R8,R10};定义一个寄存器列表RegList,可用

;LDM/STM指令访问该列表

二、数据定义伪指令

数据定义(DataDenfinition)伪指令一般用于为特定的数据分配存储单元,同时完成对已分配存储单元的初始化工作。

数据定义伪指令如表2所示。

表2数据定义伪指令

助记符

指令功能描述

SPACE

用于分配一片连续的存储单元,并初始化为0

MAP

用于定义一个结构化内存表的首地址

FIELD

用于定义一个结构化内存表的数据域

LTORG

用于声明一个数据缓冲池的开始

DCI

分配一段字节的存储单元,用指定表达式初始化,并指定存放的是代码

DATA

在代码段中使用数据

DCB

用于分配一段连续的字节存储单元,并用指定的表达式初始化

DCW(DCWU)

用于分配一段连续的半字存储单元,并用指定的表达式初始化

DCD(DCDU)

用于分配一段连续的字存储单元,并用指定的表达式初始化

DCFS(DCFSU)

用于为单精度浮点数分配连续字节存储单元,一个数分配1个字单元,并用指定的表达式初始化

DCFD(DCFDU)

用于为双精度浮点数分配连续字节存储单元,一个数分配2个字单元,并用指定的表达式初始化

DCQ(DCQU)

用于分配一段连续的双字存储单元,并用指定的64位整数初始化

DCDO

用于分配一段连续的字存储单元,并将内容初始化为相对于静态基址寄存器的偏移量

从使用方法上来讲,数据定义伪指令可以分为以下3类。

1.SPACE

伪指令SPACE用于分配一片连续的存储区,并初始化为0。

其中表达式中的数字表示分配的字节数。

SPACE也可以用%代替。

实例:

DataSpaceSPACE100;分配连续100字节的存储单元并初始化为0

2.MAP和FIELD

伪指令MAP和伪指令FIELD经常结合在一起使用。

MAP用于定义一个结构化的内存表的首地址,可以用“^”替代。

FIELD用于定义一个结构化的内存表中的数据域,可以用“#”代替。

FIELD与MAP配合使用能够定义结构化的内存表。

首先用伪指令MAP定义表的首地址,再用伪指令FIELD定义表中的各个数据域,并为每个数据域指定一个标号以供其他指令引用。

注意MAP和FIELD仅用于定义数据结构,并不实际分配存储单元。

实例:

MAP0x1000;定义结构化内存表首地址为0x1000

F1FIELD32;长度为32字节数据域F1,起始位置0x1000

F2FIELD64;长度为64字节数据域F2,起始位置0x1020

F3FIELD128;长度为128字节数据域F3,起始位置0x1060

3.存储单元分配

伪指令DCB用于分配一片连续的以字节为最小单位的存储单元,并可以使用指定的表达式对其进行初始化。

换句话说,初始化时数据的取值范围为0~255的数字或字符,每个数字或字符存储时占1字节。

DCB也可以用“=”代替。

伪指令DCW用于分配一片连续的以半字为最小单位的存储单元,并可以使用指定的表达式对其进行初始化。

DCWU与DCW的区别在于DCW分配存储单元时严格要求半字对齐,而DCWU则不严格要求半字对齐。

DCD、DCFS、DCFD、DCQ与DCW类似并要求字或双字对齐,而DCDU、DCFSU、DCFDU、DCQU与DCWU类似,不严格要求字或双字对齐。

实例:

helloDCB“HelloWorld\n\0”;用字符串初始化标号为hello的连续字节存储单元

DCD((B0<<13)+(B1<<6));使用表达式初始化连续字存储单元

srcDCD1,2,3,4,5;用数字常量初始化标号为src的连续字存储单元

三、汇编控制伪指令

汇编控制(AssemblyControl)伪指令用于控制汇编程序的执行流程,可以让汇编器按照一定的条件选择性地编译汇编指令或多次编译一段指令,从而根据程序员设定的参数生成不同的映像文件,减少程序员的工作量。

常用的汇编控制伪指令包括以下几条,如表3所列。

表3汇编控制伪指令

助记符

指令功能描述

IF、ELSE和ENDIF

根据条件成立与否决定是否编译某个指令序列

WHILE和WEND

根据条件成立与否决定是否循环编译某个指令序列

MACRO和MEND

将一段代码用一个标号替代,可在程序中多次调用该段代码

MEXIT

从宏定义中跳转出去

汇编控制指令可以分为以下3类。

1.条件判断

条件判断指令包括IF、ELSE和ENDIF三条伪指令,其用法和高级语言中的if语句类似,指令书写格式如下。

IFlogicalexpression

指令序列1

ELSE

指令序列2

ENDIF

IF,ELSE和ENDIF伪指令能根据条件的成立与否决定是否编译某个指令序列。

若IF后面的逻辑表达式为真,则编译指令序列1,否则执行指令序列2。

其中,ELSE及指令序列2可以没有,此时当IF后面的逻辑表达式为真,则编译指令序列1,否则继续执行后面的指令。

IF、ELSE和ENDIF伪指令可以嵌套使用。

实例:

GBLLTest;声明一个全局的逻辑变量,变量名为Test

……

IFTest=TRUE

指令序列1

ELSE

指令序列2

ENDIF

通过判断Test的值是否为真,汇编器决定编译哪一段指令序列。

程序员可以通过设置Test的值,灵活地生成不同的映像文件。

2.循环控制

循环控制指令包括WHILE和WEND指令,指令书写格式如下所示。

WHILElogicalexpression

指令序列

WEND

WHILE和WEND伪指令能根据条件的成立与否决定是否循环编译某个指令序列。

若WHILE后面的逻辑表达式为真,则编译指令序列。

该指令序列编译完毕后,再判断逻辑表达式的值,若为真则继续再次编译,一直到逻辑表达式的值为假。

WHILE和WEND伪指令可以嵌套使用。

实例:

GBLACounter;声明一个全局的数学变量,变量名为Counter

CounterSETA3;由变量Counter控制循环次数

……

WHILECounter<10

指令序列

WEND

通过循环控制指令,可以将一段相同的代码多次编译,类似于把代码复制多次。

通过对循环次数的控制可以灵活地控制代码复制次数,极大地减少了程序员的工作量。

3.宏指令

MACRO和MEND伪指令可以将一段代码定义为一个整体,这两条指令称为宏指令。

程序员可以在程序中通过宏指令多次调用完成宏定义的代码。

指令书写格式如下所示。

MACRO

{$label}Microname{$parameter1,$parameter2,……}

;code

MEND

其中,在宏指令展开时,{$label}中的label会替换为用户定义的符号。

宏指令可以使用一个或多个参数,当宏指令展开时,这些参数被相应的值替换。

包含在MACRO和MEND之间的指令序列称为宏定义体,在宏定义体的第一行应声明宏的原型(包含宏名和所需的参数),然后就可以在汇编程序中通过宏名来调用该指令序列。

在源程序被编译时,汇编器将宏调用展开,用宏定义中的指令序列代替程序中的宏调用,并将实际参数的值传递给宏定义中的形式参数。

MACRO和MEND伪指令可以嵌套使用。

宏定义实例:

MACRO;宏定义开始

$lablexmac$p1,$p2;宏的名称为xmac,有两个参数$p1和$p2

;宏的标号$lable可用于构造宏定义体内的其他标号名称

指令序列

$lable.loop1指令;$lable.loop1为宏定义体的内部标号

指令序列

BGE$lable.loop1

$lable.loop2指令;$lable.loop2为宏定义体的内部标号

BL$p1;参数$p1为一个子程序的名称

BGT$lable.loop2

指令序列

ADR$p2

指令序列

MEND;宏定义结束

在上述代码中,宏名为xmac,标号为$lable,有两个参数$p1和$p2。

标号和参数在实际调用中可根据需要被替换成不同的符号。

而宏名是唯一确定的,在调用宏时以宏名为依据。

调用上述宏可以采用下面的方式:

abcxmacsubr1,de;通过宏的名称xmac调用宏,其中宏的标号为abc,

;第一个参数为subr1,第二个参数为de

可以看出宏调用是通过宏名完成的,调用宏时标号$lable被替换成了abc,两个参数$p1和$p2分别被替换成了subr1和de。

汇编器处理宏时会将宏展开还原成为一段代码。

程序被汇编后,宏展开的结果为

指令序列

abc.loop1指令;用标号$lable的实际值abc代替$lable而构成标号

;abc.loop1

指令序列

BGEabc.loop1

abc.loop2指令

BLsubr1;第一个参数的实际值为subr1

BGTabc.loop2

指令序列

ADRde;第二个参数的实际值为de

指令序列

对比宏定义可以看出,原来宏定义中凡是出现了$lable的地方均被替换成了abc,出现$p1的地方被替换成了subr1,而出现$p2的地方被替换成了de。

对于程序员而言,采用宏定义的方法可以用一条语句替代一大段指令序列,灵活使用可以极大地提高编程效率。

宏指令的使用方式和功能与子程序有些相似,但其本质是完全不同的。

采用宏指令时,用一条语句表示了一大段指令序列,汇编器在完成编译时会将宏指令还原成指令序列。

换句话说,如果在程序中调用了N次宏指令,最终将汇编器还原成N段指令序列。

这N段代码都会被编译成映像文件,最终ARM会将这N段代码都执行一遍,从本质上来讲程序最终生成的映像文件大小不会发生变化。

子程序调用则是通过改变程序计数器(PC)的值,从而改变程序的执行顺序来完成的。

换句话说,如果在程序中调用N次子程序,子程序本身并不会被复制N次,在编译时也只会生成一段映像文件。

但在ARM执行到调用子程序的指令时,会通过改变PC的值将同一段子程序执行N次。

与宏定义相比,子程序可以提供模块化的程序设计、节省存储空间并提高运行速度。

但在使用子程序结构时需要保护现场,从而增加了系统开销。

因此,在代码较短且需要传递的参数较多时,可以使用宏指令代替子程序,通过占用更多的存储空间而减少系统开销,从而优化系统运行速度。

MEXIT用于从宏定义中跳转出去。

MEXIT使用实例如下:

MACRO;宏定义开始

$abcmicroabc$param1,$param2

指令序列

WHILEcondition1

指令序列

IFcondition2

指令序列

MEXIT;从宏中跳转出去

ELSE

指令序列

ENDIF

WEND

指令序列

MEND

四、其他常用伪指令

还有一些其他常用的伪指令,在汇编程序中也经常用到,如表4所示。

表4杂项伪指令

助记符

指令功能描述

ALIGN

通过添加填充字节的方式,使当前位置满足一定的对齐方式

AREA

段定义,定义一个代码段或数据段

CODE16

指示汇编器后面的指令为16位的Thumb指令

CODE32

指示汇编器后面的指令为32位的ARM指令

END

指示汇编器源文件已经结束

ENTRY

指定程序的入口点

EQU

为数字常量,基于寄存器的值和程序中的标号定义一个名称

EXPORT

声明一个符号可以被其他文件引用

GLOBAL

同上

IMPORT

指示编译器当前符号在其他文件中定义,本文件中可能引用该符号

(续表)

助记符

指令功能描述

EXTERN

同上

GET/INCLUDE

引用文件,包含一个源文件并在当前位置进行汇编

INCBIN

引用文件,包含一个源文件但并不进行汇编处理

KEEP

指示编译器保留符号表中的局部符号

NOFP

禁止源程序中包含浮点运算指令

REQUIRE

指定段间的依赖关系

REQUORE8

指示当前文件请求堆栈为8字节对齐

PRESERVE8

指示当前文件保存堆栈为8字节对齐

ROUT

定义局部标号的有效范围

下面简要介绍杂项指令中较为常用的几条指令。

1.AREA

AREA伪指令用于定义一个代码段或数据段。

指令书写格式:

AREAsectionname{,attr}{,attr},……

其中sectionname表示段名,段名若以数字开头,则该段名需用“|”括起来,如|1_test|。

属性字段{attr},表示该代码段(或数据段)的相关属性,多个属性以逗号分隔。

常用属性如下所示。

lCODE属性。

用于定义代码段。

lDATA属性。

用于定义数据段。

lREADONLY属性。

指定本段为只读属性,代码段默认为READONLY。

lREADWRITE属性。

指定本段为可读写属性,数据段的默认属性为READWRITE。

lALIGN属性。

书写格式为ALIGNExpr。

默认情况下,ELF(可执行连接文件)的代码段和数据段是按字对齐的,目的:

建立库房管理制度,保证物料在储存期安全储存。

2适用范围:

适用于本公司库存的原料、辅料、包装材料、半成品和成品的储存管理。

3职责:

3.1供应链部负责库房的管理工作。

3.2库管员负责按本制度实施储存管理。

4控制要求:

4.1物料管理

4.1.1物料应按不同的状态分区存放。

应放在洁净的托盘上,间距合适通风良好,严禁直接堆放在地上,堆放以不超出托板为宜。

4.1.2存放原则

4.1.2.1不合格品应设专区存放;其他有特殊存放要求的,应存放于与其存放条件相适宜的库房(冷库、化学品库)。

4.1.2.3成品应按品种、规格分区存放;同一品种、同一规格,应按批号存放,以利先进先出。

零箱应放在明显的位置,以利清点。

4.1.2.4内包材与外包材、液体物料与固体物料应分区隔离存放。

容易相互发生化学反应或者灭火方法不同的物品,必须分间、分库储存,并在醒目处标明储存物品的名称、性质和灭火方法。

4.1.2.5含有挥发性成分的物料,应密闭保存。

4.1.3库存物料有明显的状态标记(待验:

黄色;合格:

绿色;不合格:

红色),以区别不同的状态。

4.1.4凡采用单独地区储藏来保证隔离状况时,这些地区应当有明显标志,且应当仅限于被批准者方可进入。

4.1.5物料进入库房时,需要按照物料堆放的要求进行码垛,垛与垛间距不小于0.5米,垛与墙间距不小于0.5米,垛与梁、柱间距不小于0.3米,主要通道宽度不少于2米。

4.1.6库房内设备、管道、设施与货垛应保持至少0.3米距离,电器设施、设备周围,架空线路下方不准堆放物料。

4.1.7仓储区内物料的码放、搬运应严格按操作规定进行,轻拿轻放,以防止损坏。

4.1.8库存物料的收发、保管应实行二人复核,所有物料、帐、卡、记录、单据、状态标志等应由表达式的取值范围为0~31。

lCOMMON属性。

该属性定义一个通用的段,不包含任何用户代码和数据。

各源文件中同名的COMMON段共享同一段存储单元。

一个汇编语言程序至少要包含一个段,当程序太长时,也可以将程序分为多个代码段和数据段。

例如,

AREAInit,CODE,READONLY;该伪指令定义了一个代码段,段名为Init,属性为只读

AREAData1,DATA,READWRITE;该伪指令定义了一个数据段,段名为Data1,属性为可读写

2.ALIGN

指令书写格式:

ALIGN{expr{,offset}}

ALIGN伪指令可通过添加填充字节的方式,使当前位置满足一定的对齐方式。

其中,expr(表达式)的值用于指定对齐方式,可能的取值为2的幂,如1,2,4,8和16等。

若未指定expr,则将当前位置对齐到下一个字的位置。

offset(偏移量)也是一个数字表达式,若使用该字段,则当前位置的对齐方式为2的expr次幂+offset。

例如,

AREAInit,CODE,READONLY,ALIGN=3;指定后面的指令为8字节对齐

指令序列

END

3.CODE16和CODE32

CODE16伪指令通知编译器,其后的指令序列为16位的Thumb指令。

CODE32伪指令通知编译器,其后的指令序列为32位的ARM指令。

若在汇编源程序中同时包含ARM指令和Thumb指令,则可用CODE16伪指令通知编译器其后的指令序列为16位的Thumb指令,用CODE32伪指令通知编译器其后的指令序列为32位的ARM指令。

因此,在使用ARM指令和Thumb指令混合编程的代码里,可用这两条伪指令进行切换,但要注意的是,它们只通知编译器其后指令的类型,并不能对处理器进行状态的切换。

例如,

AREAInit,CODE,READONLY

CODE32;通知编译器其后的指令为32位的ARM指令

LDRR0,=NEXT+1;将跳转地址放入寄存器R0

BXR0;程序跳转到新的位置执行,并将处理器切换到Thumb工作状态

CODE16;通知编译器其后的指令为16位的Thumb指令

NEXTLDRR3,=0x3FF

END;程序结束

4.ENTRY

ENTRY伪指令用于指定汇编程序的入口点。

在一个完整的汇编程序中至少要有一个ENTRY(也可以有多个,当有多个ENTRY时,程序的真正入口点由连接器指定),但在一个源文件里最多只能有一个ENTRY(也可以没有)。

例如,

AREAInit,CODE,READONLY

ENTRY;指定应用程序的入口点

5.EQU

指令书写格式:

nameEQUexpr{,type}

EQU伪指令用于为程序中的常量和标号等定义一个等效的字符名称,类似于C语言中的#define。

其中EQU可用“*”代替。

name(名称)为EQU伪指令定义的字符名称,当expr(表达式)为32位常量时,可以指定其type(数据类型),可为三种类型之一:

CODE16,CODE32和DATA。

例如,

TestEQU50;定义标号Test的值为50

AddrEQU0x55,CODE32;定义Addr的值为0x55,且该处为32位的ARM指令

6.引用指令

EXPORT、GLOBAL、IMPORT和EXTERN伪指令属于引用指令。

其主要作用是声明一个符号可以被其他文件引用,或在本文件中将引用其他文件中的符号。

指令书写格式如下:

EXPORT(GLOBAL/IMPORT/EXTERN)Symbol{[WEAK]}

EXPORT伪指令用于在程序中声明一个全局的标号,该标号可在其他文件中引用。

EXPORT可用GLOBAL代替。

IMPORT伪指令用于通知编译器要使用的标号在其他源文件中定义,但要在当前源文件中引用,而且无论当前源文件是否引用该标号,该标号均会被加入到当前源文件的符号表中。

EXTERN伪指令用于通知编译器要使用的标号在其他源文件中定义,但要在当前源文件中引用。

如果当前源文件实际并未引用该标号,则该标号不会被加入到当前源文件的符号表中。

标号在程序中区分大小写,[WEAK]选项表示当所有源文件都没有定义这样一个标号时,编译器不会给出错误信息,在多数情况下将该标号置为0。

若该标号被B或BL指令引用,则将B或BL指令置为NOP操作。

例如,

AREAInit,CODE,READONLY

EXTERNMain;通知编译器当前文件要引用标号Main,但Main在其他源文件中定义

END

7.包含指令

GET、INCLUDE和INCBIN等伪指令属于包含伪指令,这类伪指令通常表示当前的源文件编译时需要使用到源文件或目标文件。

GET伪指令用于将一个源文件包含到当前的源文件中,并将被包含的源文件在当前位置进行汇编处理。

可以使用INCLUDE代替GET。

汇编程序中常用的方法是在某源文件中定义一些宏指令,用EQU定义常量的符号名称,用MAP和FIELD定义结构化的数据类型,然后用GET伪指令将这个源文件包含到其他源文件中。

使用方法与C语言中的include相似。

GET伪指令只能用于包含源文件,包含目标文件需要使用INCBIN伪指令。

INCBIN伪指令用于将一个目标文件或数据文件包含到当前的源文件中,被包含的文件不做任何变动而存放在当前文件中,编译器从其后开始继续处理。

包含指令使用实例如下:

AREAInit,CODE,READONLY

GETa1.s;通知编译器当前源文件包含源文件a1.s

GETC:

\b1.s;通知编译器当前源文件包含源文件C:

\a2.s

INCBINa2.dat

展开阅读全文
相关资源
猜你喜欢
相关搜索
资源标签

当前位置:首页 > 小学教育 > 数学

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1