汇编语言——第14章 端口

目录

引言

14.1 端口的读写

14.2 CMOS RAM芯片

14.3 shl和shr指令

14.4 CMOS RAM中存储的时间信息


引言

各种存储器都和CPU的地址线、数据线、控制线相连。CPU在操控它们的时候,把它们都当作内存来对待,把它们总的看作一个由若干个存储单元组成的逻辑存储器,这个逻辑存储器我们称其为内存地址空间。

在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芯片

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.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

         左移后:                 10010000

如果接着上面,继续执行一条shl al,1,则执行后,(al)=00100000b,CF=1。shl al,1操作过程如下:

(1)左移

         原数据:                 10010000

         左移后:               10010000

(2)将最后移出的一位写入CF中

         原数据:                 10010000

         左移后:                 0010000           CF=1

(3)最低位用0补充

         原数据:                 10010000

         左移后:                 00100000

如果移动位数大于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)=01000000b,CF=1。

如果接着上面,继续执行一条shr al,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.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码:0000 0001 0010 0011 0100 0101 0110 0111 1000 1001

数值26,用BCD码表示为:0010 0110。

可见,一个字节可表示两个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码值的数据。

mov ah,al                    ;al中为从CMOS RAM的8号单元中读出的数据

mov cl,4

shr ah,cl                      ;ah中为月份的十位数码值

and al,00001111b        ;al中为月份的个位数码值

②显示(ah)+30h和(al)+30h对应的ASCII码字符。

完整的程序如下:

assume cs:code

code segment

start:   mov al,8

           out 70h,al 

           in al,71h

           mov ah,al

           mov cl,4

           shr ah,cl

           and al,00001111b

           add ah,30h

           add al,3h

           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

       

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

关了个尔

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值