通用寄存器?
四个:AX BX CX DX
存放数据,也就是数据寄存器;
一个数据寄存器大小为 2 byte
,16
位;
0000 0000 0000 0000
~1111 1111 1111 1111
;
0000
~FFFFH
;
0
~65535
;
可以看到,
AX
中已经变成了0005
首先-a
修改 指令
然后-r
进行编译,发现代码 已经变成MOV AX,0005
这是因为在使用mov
指令时会保证数据与寄存器之间的位数的一致性
最后-t
执行
0
也要编译为16
位的0
,AL
是八位的,所以编译成AL,05
一个数据寄存器可以分位高低两个寄存器,各为8
位
如:
AX = AH + AL
BX = BH + BL
这么做一是为了兼容以前的CPU
,二为了方便操作字节
数据线?
数据线 16
根 数据线的根数(宽度)决定了CPU
一次可以处理的数据长度
8086CPU
一次性可以处理两种大小的数据:
字节型: byte
字型: 2 byte
地址寄存器?
段地址寄存器:ds es ss cs
偏移地址寄存器:sp bp si di ip bx
8086CPU
有20
地址线,决定了CPU
的寻址能力,但是数据线只有16
根,不够用,这样可以增强寻址能力
地址加法器
段地址 × 16 +偏移地址 = 物理地址
段地址 × 16 = 基础地址
基础地址 + 偏移地址 = 物理地址
例:F230:00C8
段地址 偏移地址 物理地址
F230H × 10H C8H F23C8H
CPU
如何区分指令与数据?
8080CPU
中在任意时刻,CPU
将CS:IP
所指向的内容 全部当作指令来执行
在内存中指令和数据是没有任何区别的,都是二进制执行,只有在CPU
工作时才会进行区分
指令执行过程:
-
CPU
从CS:IP
所指向的内存单元读取指令,存放到指令寄存器中 -
IP = IP + 所读指令长度
,从而指向下一条指令 - 执行指令寄存器中的内容,回到步骤1
jmp
转移指令,可以修改CS IP
这两个寄存器
CS
和IP
不可以通过MOV
来修改
字节型和字型数据?
高地址内存单元存放 字型数据的高位字节
低地址内存单元存放 字型数据的地位字节
例:
地址 数据
0 20H
1 4EH
2 12H
3 00H
地址0存放的字节型数据是 20H
地址0存放的字型数据是 4E20H
地址2存放的字节型数据是 12H
地址2存放的字型数据是 0012H
地址1存放的字型数据是 124EH
大地址 | 小地址 |
---|---|
高八位 | 低八位 |
数据大小不一致时无法使用MOV
做加法时,超过寄存器的位数会怎样?
7818H + 8H = 7820H
可以看到未溢出时,AX
已变成7820H
93H + 85H = 118H
,AL
只有八位会溢出:
使用AX
来进行计算:
汇编运行时,寄存器是相互独立的,并不会影响其他寄存器
add
指令也需要数据大小匹配
ds
一般访问数据时使用
栈?
入栈 push
将数据放到栈顶标记的内存中 修改栈顶标记
出栈 pop
将栈顶标记的内存中的数据拿出来 修改栈顶标记
入栈出栈操作的长度是字型数据(16
位)
栈顶标记是内存地址,用段地址和偏移地址来表示
在8086CPU
中 在任意时刻 将段地址寄存器ss
和偏移地址寄存器sp
所组合出来的内存地址当作栈顶标记
push ax
修改sp
寄存器中的数值 sp = sp - 2
将ax
中字型数据放到ss:sp
所组合出来的内存地址当中去
pop bx
把ss:sp
所组合出来的内存地址当中的数据放到bx
中去 修改栈顶标记sp = sp + 2
栈的最大的空间:65536
个字节,32768
个字,64kB
栈的作用:
- 临时保存数据
- 交换数据
编译和链接
编译:生成 OBJ
连接:OBJ -> EXE
start
指令是告诉编译软件汇编程序的入口在哪里,然后系统通过修改CS:IP
来执行程序
exe
不止有整个程序,还包括了描述信息,系统根据这些信息进行相关的设置
asm
文件
程序返回:
mov ax,4c00h
int 21h