文章目录
参考书籍:《汇编语言(第2版)》——王爽,参考视频:《零基础入门学习汇编语言》——小甲鱼
第四章:第一个程序
章节过渡
终于可以编写第一个完整的程序了,以前都是在Debug中写一些指令,在Debug中执行。现在开始我们将开始编写完整的汇编语言程序,用编译和连接程序将它们编译连接成可执行文件(如*.exe文件),在操作系统中运行。
4.1 一个源程序从写出到执行的过程
第一步:编写汇编源程序
使用文本编辑器,用汇编语言编写汇编源程序。
第二步:对源程序进行编译连接。
使用汇编语言编译程序对源程序文件中的源程序进行编译,产生目标文件;再用连接程序对目标文件进行连接,生成可在操作系统中直接运行的可执行文件。
可执行文件包含两部分内容。
- 程序(从源程序中的汇编指令翻译过来的机器码)和数据(源程序中定义的数据)。
- 相关的描述信息。(比如,程序有多大、要占用多少内存空间等)
这一步工作的结果:产生一个可在操作系统中运行的可执行文件。
第三步:执行可执行文件中的程序。
在操作系统中,执行可执行文件中的程序。
操作系统依照可执行文件中的描述信息,将可执行文件中的机器码和数据加载入内存,并进行相关的初始化(比如设置CS:IP指向第一条要执行的指令),然后由CPU执行程序。
4.2 源程序
下面是一段简单的汇编语言源程序。
assume cs:codesg
codesg segment
mov ax,0123H
mov bx,0456H
add ax,bx
add ax,as
mov ax,4c00H
int 21H
codesg ends
end
1.伪指令
在汇编语言源程序中,包含两种指令,一种是汇编指令,一种是伪指令。
汇编指令是有相应的机器码的指令,可以被编译为机器指令,最终为CPU所执行。
而伪指令没有对应的机器指令,最终不被CPU所执行。伪指令是由编译器来执行的指令,引导编译器进行相关的编译工作。
(1)segment和ends
segment和ends的功能是定义一个段,segment说明一个段开始,ends说明一个段结束。
一个段必须有一个名称来标识,使用格式为:
段名 segment
:
段名 ends
一个汇编程序是由多个段组成的,这些段被用来存放代码、数据或当作栈空间来使用。
一个有意义的汇编程序中至少要有一个段,这个段用来存放代码。
(2)end
end是一个汇编程序的结束标记,编译器遇到伪指令end就会停止对源程序的编译。
(3)assume
assume即“假设”,它假设某一段寄存器和程序中的某一个用segment……ends定义的段相关联。
比如上面的源程序,我们用assume cs:codesg将用作代码段的段codesg和CPU中的段寄存器cs联系起来。
2.源程序中的“程序”
用汇编语言写的源程序,包括伪指令和汇编指令。
这里所说的程序就是指源程序中最终由计算机执行、处理的指令或数据。
程序最先以汇编指令的形式存在源程序中,经编译、连接后转变为机器码,存储在可执行文件中。
3.标号
汇编源程序中,除了汇编指令和伪指令外,还有一些标号,比如“codesg”。一个标号指代了一个地址。
比如codesg在segment的前面,作为一个段的名称,这个段的名称最终将被编译、连接程序处理为一个段的段地址。
4.程序的结构
(1)定义一个段,名称为abc。
abc segment
:
abc ends
(2)在这个段中写入汇编指令,来实现我们的任务。
abc segment
mov ax,2
add ax,ax
add ax,ax
abc ends
(3)指出程序在何处结束。
abc segment
mov ax,2
add ax,ax
add ax,ax
abc ends
end
(4)abc被当作代码段,所以,应该将abc和cs联系起来。
assume cs:abc
abc segment
mov ax,2
add ax,ax
add ax,ax
abc ends
end
5.程序返回
一个程序P2在可执行文件中,则必须有一个正在运行的程序P1,将P2从可执行文件中加载入内存后,将CPU的控制权交给P2,P2才能得以运行。
P2运行后,P1暂停运行。而当P2运行完毕后,应该将CPU的控制权交还给使它得以运行的程序P1,然后P1继续运行。
一个程序结束后,将CPU的控制权交还给使它得以运行的程序,我们称这个过程为:程序返回。
例如,在cmd中运行某程序。
4.3 编辑
4.4 编译
下载masm6.15,解压后找到masm.exe,ml.exe,link.exe三个程序,将它们和源程序放在同一个文件夹下,然后在该目录下运行cmd,执行“masm 文件名.asm”命令对源程序进行编译。
4.5 连接
link 文件名.obj,但是我们这个程序没有其它源文件需要连接,所以中间过程直接enter跳过即可,最后生成一个1.exe可执行文件。
4.6 以简化的方式进行编译和连接
在命令后面加上分号即可,可以忽略过程中产生的中间文件。
4.7 1.exe的执行
emmmmmm,这个16位程序在64位Windows上运行不了。