《汇编语言》第14章 端口
我们前面讲过,各种存储器都和CPU的地址线、数据线、控制线相连。CPU在操控它们的时候,把它们都当作内存来对待,把它们总地看做一个由若干存储单元组成的逻辑存储器,这个逻辑存储器我们称其为内存地址空间(可参见1.15节)。
在PC机系统中,和CPU通过总线相连的芯片除各种存储器外,还有以下3种芯片。
(1)各种接口卡(比如,网卡、显卡)上的接口芯片,它们控制接口卡进行工作;
(2)主板上的接口芯片,CPU通过它们对部分外设进行访问;
(3)其他芯片,用来存储相关的系统信息,或进行相关的输入输出处理。在这些芯片中,都有一组可以由 CPU 读写的寄存器。这些寄存器,它们在物理上可能处于不同的芯片中,但是它们在以下两点上相同。
(1)都和CPU的总线相连,当然这种连接是通过它们所在的芯片进行的;
(2)CPU 对它们进行读或写的时候都通过控制线向它们所在的芯片发出端口读写命令。
可见,从 CPU 的角度,将这些寄存器都当作端口,对它们进行统一编址,从而建立了一个统一的端口地址空间。每一个端口在地址空间中都有一个地址。
CPU可以直接读写以下3个地方的数据。
(1)CPU内部的寄存器;
(2)内存单元;
(3)端口。
这一章,我们讨论端口的读写。
14.1 端口的读写
在访问端口的时候,CPU通过端口地址来定位端口。因为端口所在的芯片和CPU通过总线相连,所以,端口地址和内存地址一样,通过地址总线来传送。在PC系统中,CPU最多可以定位64KB个不同的端口。则端口地址的范围为0~65535。
对端口的读写不能用 mov、push、pop 等内存读写指令。端口的读写指令只有两条:in和 out,分别用于从端口读取数据和往端口写入数据。
我们看一下CPU执行内存访问指令和端口访问指令时候,总线上的信息:
(1)访问内存:
mov ax,ds:[8];假设执行前(ds)=0
执行时与总线相关的操作如下所示。
① CPU通过地址线将地址信息8发出;
② CPU 通过控制线发出内存读命令,选中存储器芯片,并通知它,将要从中读取数据:
③ 存储器将8号单元中的数据通过数据线送入CPU。
(2)访问端口:
in al,60h;从60h号端口读入一个字节
执行时与总线相关的操作如下。
① CPU通过地址线将地址信息60h发出;
② CPU 通过控制线发出端口读命令,选中端口所在的芯片,并通知它,将要从中读取数据;
③ 端口所在的芯片将60h端口中的数据通过数据线送入CPU。
注意,在in和out指令中,只能使用ax或al来存放从端口中读入的数据或要发送到端口中的数据。访问8位端口时用al,访问16位端口时用ax。
对0~255以内的端口进行读写时:
in al,20h;从20h端口读入一个字节
out 20h,al;往20h端口写入一个字节
对256~65535的端口进行读写时,端口号放在dx中:
mov dx,3f8h;将端口号3f8h送入dx
in al,dx;从3f8h端口读入一个字节
out dx,al;向3f8h端口写入一个字节
14.2 CMOS RAM芯片
下面的内容中,我们通过对CMOS RAM的读写来体会一下对端口的访问。PC机中,有一个CMOS RAM芯片,一般简称为CMOS。此芯片的特征如下。
(1)包含一个实时钟和一个有128个存储单元的RAM存储器(早期的计算机为64个字节)。
(2)该芯片靠电池供电。所以,关机后其内部的实时钟仍可正常工作,RAM 中的信息不丢失。
(3) 128个字节的RAM中,内部实时钟占用0~0dh单元来保存时间信息,其余大部分单元用于保存系统配置信息,供系统启动时 BIOS 程序读取。BIOS 也提供了相关的程序,使我们可以在开机的时候配置CMOS RAM中的系统信息。
(4)该芯片内部有两个端口,端口地址为 70h 和 71h。CPU 通过这两个端口来读写CMOS RAM。
(5) 70h为地址端口,存放要访问的 CMOS RAM 单元的地址:71h为数据端口,存放从选定的 CMOS RAM 单元中读取的数据,或要写入到其中的数据。可见,CPU 对CMOS RAM的读写分两步进行,比如,读 CMOS RAM的2号单元:
① 将2送入端口70h;
② 从端口71h读出2号单元的内容。
检测点14.1
(1)编程,读取CMOS RAM的2号单元的内容。
;jc14_1_1.asm 检测点14.1
;(1)编程,读取CMOS RAM的2号单元的内容。
assume cs:codesg
codesg segment
start: mov al,2h ;赋值alout 70h,al ;将al送入端口70hin al,71h ;从端口71h处读出单元内容mov ax,4c00h int 21h codesg ends
end start
运行调试
(2)编程,向CMOS RAM的2号单元写入0。
;jc14_1_2.asm 检测点14.1
;(2)编程,向CMOS RAM的2号单元写入0。assume cs:codesg
codesg segment
start: mov al,2h out 70h,al ;70h是地址端口,设置址址单元mov al,0out 71h,al ;71h是数据端口,通过71h端口向2号单元写入0;in al,71h ;从71h端口读取数据mov ax,4c00hint 21hcodesg ends
end start
运行调试:
14.3 shl和shr指令
shl和shr是逻辑移位指令,后面的课程中我们要用到移位指令,这里进行一下讲解。shl是逻辑左移指令,它的功能为:
(1)将一个寄存器或内存单元中的数据向左移位;
(2)将最后移出的一位写入CF中;
(3)最低位用0补充。
指令:
mov al,01001000b
shl al,1;将al中的数据左移一位
执行后(al)=10010000b,CF=0。
我们来看一下shl al,1的操作过程。
(1)左移
原数据:01001000
左移后:01001000
(2)将最后移出的一位写入CF中
原数据:01001000
左移后:1001000 CF=0
(3)最低位用0补充
原数据:01001000
左移后:1001000
如果接着上面,继续执行一条 shl al,1,则执行后:(al)=00100000b,CF=1。shl 指令的操作过程如下。
(1)左移
原数据:10010000
左移后:10010000
(2)将最后移出的一位写入CF中
原数据:10010000
左移后:0010000 CF=1
(3)最低位用0补充
原数据:1001000
左移后:0010000
如果移动位数大于1时,必须将移动位数放在cl中。
比如,指令:
mov al,01010001b
mov cl,3
shl al,cl
执行后(al)=10001000b,因为最后移出的一位是0,所以CF=0。
可以看出,将X逻辑左移一位,相当于执行X=X*2。
比如:
mov al,00000001b;执行后(al)=00000001b=1
shl al,1;执行后(al)=00000010b=2
shl al,1;执行后(al)=00000100b=4
shl al,1;执行后(al)=00001000b=8
mov cl,3
shl al,cl;执行后(al)=01000000b=64
shr是逻辑右移指令,它和shl所进行的操作刚好相反。
(1)将一个寄存器或内存单元中的数据向右移位;
(2)将最后移出的一位写入CF中;
(3)最高位用0补充。
指令:
mov al,10000001b
shr al,1;将al中的数据右移一位
执行后(al)=0100000b,CF=1。
如果接着上面,继续执行一条 shral,1,则执行后:(al)=00100000b,CF=0。
如果移动位数大于1时,必须将移动位数放在cl中。
比如,指令:
mov al,01010001b
mov cl,3
shr al,cl
执行后(al)=00001010b,因为最后移出的一位是0,所以CF=0。
可以看出将X逻辑右移一位,相当于执行X=X/2。
检测点14.2
编程,用加法和移位指令计算(ax)=(ax)*10。
提示,(ax)*10=(ax)*2+(ax)*8。
;jc14_2.asm assume cs:codesg
codesg segment
start: mov ax,10shl ax,1 ;10左移1位,表示乘2mov bx,ax ;把左移结果放在bx中mov cl,3mov ax,10shl ax,cl ;10左移3位,表示乘以8add ax,bx ;ax=(ax)*2+(ax)*8。 = 100mov ax,4c00hint 21hcodesg ends
end start
运行调试:
先计算ax*2,再计算ax*8,最后两者相加
14.4 CMOS RAM中存储的时间信息
在CMOS RAM中,存放着当前的时间:年、月、日、时、分、秒。这6个信息的长度都为1个字节,存放单元为:
秒:0 分:2 时:4 日:7 月:8 年:9
这些数据以BCD码的方式存放。
BCD码是以4位二进制数表示十进制数码的编码方法,如下所示。
十进制数码: 0 1 2 3 4 5 6 7 8 9
对应的BCD码:0000000010010011010101010101001111110001001
比如,数值26,用BCD码表示为:00100110。
可见,一个字节可表示两个BCD码。则CMOS RAM存储时间信息的单元中,存储了用两个BCD码表示的两位十进制数,高4位的BCD码表示十位,低4位的BCD码表示个位。比如,00010100b表示14。
编程,在屏幕中间显示当前的月份。
分析,这个程序主要做以下两部分工作。
(1)从CMOS RAM的8号单元读出当前月份的BCD码。
要读取CMOS RAM的信息,首先要向地址端口70h写入要访问的单元的地址:
mov al,8
out 70h,al
然后从数据端口71h中取得指定单元中的数据:
in al,71h
(2)将用BCD码表示的月份以十进制的形式显示到屏幕上。
我们可以看出,BCD码值=十进制数码值,则BCD码值+30h=十进制数对应的ASCII码。
从CMOS RAM的8号单元读出的一个字节中,包含了用两个BCD码表示的两位十进制数,高4位的BCD码表示十位,低4位的BCD码表示个位。比如,00010100b表示14。
我们需要进行以下两步工作。
① 将从CMOS RAM的8号单元中读出的一个字节,分为两个表示BCD码值的数据。
mov ah,al;al中为从CMOS RAM的8号单元中读出的数据
mov cl,4
shr ah,cl;ah中为月份的十位数码值
and al,0000111b;al中为月份的个位数码值
② 显示(ah)+30h和(al)+30h对应的ASCII码字符。
完整的程序如下。
;p14_1.asm 14.4 CMOS RAM中存储的时间信息
;编程,在屏幕中间显示当前的月份。assume cs:code
code segment
start: mov al,8out 70h,al in al,71h mov ah,al mov cl,4shr ah,cl and al,00011111b add ah,30h add al,30h mov bx,0b800h mov es,bx mov byte ptr es:[160*12+40*2],ah ;显示月份的十位数码mov byte ptr es:[160*12+40*2 +2],al ;接着显示月份的个位数码mov ax,4c00h int 21h code ends
end start
运行效果: