本人微信公众号"aeolian"~

汇编语言-14端口

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

在PC机系统中,和CPU通过总线相连的芯片除各种存储器外,还有以下3种芯片:
1. 各种接口卡(比如,网卡、显卡)上的接口芯片,它们控制接口卡进行工作;
2. 主板上的接口芯片,CPU通过它们对部分外设进行访问;
3. 其他芯片,用来存储相关的系统信息,或进行相关的输入输出处理。

在这些芯片中,都有一组可以由CPU 读写的寄存器。这些寄存器,它们在物理上可能处于不同的芯片中,但是它们在以下两点上相同:

1. 都和CPU的总线相连,当然这种连接是通过它们所在的芯片进行的;
2. CPU对它们进行读或写的时候都通过控制线向它们所在的芯片发出端口读写命令。

从CPU的角度,将这些寄存器都当作端口,对它们进行统一编址,从而建立了一个统一的端口地址空间。每一个端口在地址空间中都有一个地址。

CPU可以直接读写以下3个地方的数据:

1. CPU内部的寄存器;
2. 内存单元;
3. 端口。

## 端口读写
在访问端口的时候,CPU通过端口地址来定位端口。因为端口所在的芯片和CPU通过总线相连,所以,端口地址和内存地址一样,通过地址总线来传送。在PC系统中,CPU最多可以定位64KB个不同的端口。则端口地址的范围为0~65535。

对端口的读写不能用mov、push、pop等内存读写指令。端口的读写指令只有两条:
**in**和**out**,分别用于从端口读取数据和往端口写入数据。

CPU执行内存访问指令和端口访问指令时候,总线上的信息:
* 访问内存;
![](https://img2020.cnblogs.com/blog/1208477/202007/1208477-20200703202348711-1913317698.png)
* 访问端口;
![](https://img2020.cnblogs.com/blog/1208477/202007/1208477-20200703202455395-132776620.png)

## CMOS RAM芯片
PC机中,有一个CMOS RAM芯片,一般简称为CMOS。此芯片的特征如下:

1. 包含一个实时钟和一个有128个存储单元的RAM存储器(早期的计算机为64个字节)。
2. 该芯片靠电池供电。所以,关机后其内部的实时钟仍可正常工作,RAM中的信息不丢失。
3. 128个字节的RAM中,内部实时钟占用O~Odh单元来保存时间信息,其余大部分单元用于保存系统配置信息,供系统启动时BIOS程序读取。BIOS也提供了相关的程序,使我们可以在开机的时候配置CMOS RAM中的系统信息。
4. 该芯片内部有两个端口,端口地址为70h和71h。CPU通过这两个端口来读写CMOS RAM。
5. 70h为地址端口,存放要访问的CMOS RAM 单元的地址; 71h为数据端口,存放从选定的CMOS RAM单元中读取的数据,或要写入到其中的数据。可见,CPU对CMOS RAM的读写分两步进行,比如,读CMOS RAM的2号单元:1. 将2送入端口70h;2. 从端口71h读出2号单元的内容。

## shl和shr指令

shl和shr是逻辑移位指令。

### shl是逻辑左移指令

功能:
(1) 将一个寄存器或内存单元中的数据向左移位;
(2) 将最后移出的一位写入CF中;
(3) 最低位用0补充。
![](https://img2020.cnblogs.com/blog/1208477/202007/1208477-20200703210912017-2072517992.png)

再次执行一条shl al,1
![](https://img2020.cnblogs.com/blog/1208477/202007/1208477-20200703211052430-446388771.png)

#### 一次左移多位
![](https://img2020.cnblogs.com/blog/1208477/202007/1208477-20200703211150358-730998117.png)

### shr是逻辑右移指令
![](https://img2020.cnblogs.com/blog/1208477/202007/1208477-20200703211319504-565699489.png)
继续执行一条shr al,l,则执行后:(al)=00100000b,CF=0

如果移动位数大于1时,必须将移动位数放在cl中
![](https://img2020.cnblogs.com/blog/1208477/202007/1208477-20200703211520780-1792398213.png)

## CMOS RAM中存储的时间信息
在CMOS RAM中,存放着当前的时间:年、月、日、时、分、秒。这6个信息的长度都为1个字节,存放单元为
秒:0 分:2 时:4 日:7 月:8 年:9

![](https://img2020.cnblogs.com/blog/1208477/202007/1208477-20200703211802024-273551586.png)

一个字节可表示两个BCD码。则CMOS RAM存储时间信息的单元中,存储了用两个BCD码表示的两位十进制数,高4位的BCD码表示十位,低4位的BCD码表示个位。比如,00010100b表示14。

## 屏幕中间显示当前的月份

1.从CMOS RAM的8号单元读出当前月份的BCD码。

要读取CMOS RAM的信息,首先要向地址端口70h写入要访问的单元的地址:
“`assembly
mov al,8
out 70h,al
“`
然后从数据端口71h中取得指定单元中的数据:
“`assembly
in al,7lh
“`

2.将用BCD码表示的月份以十进制的形式显示到屏幕上。
BCD码值=十进制数码值,则BCD码值+30h=十进制数对应的ASCII码。
从CMOS RAM的8号单元读出的一个字节中,包含了用两个BCD码表示的两位十进制数,高4位的BCD码表示十位,低4 位的BCD码表示个位。比如,00010100b表示14。

(1)将从CMOS RAM的8号单元中读出的一个字节,分为两个表示BCD码值的数据。
“`assembly
mov ah,al ;al中为从CMOS RAM的8号单元中读出的数据
mov cl,4
shr ah,cl ;ah中为月份的十位数码值
and al,00001111b ;al中为月份的个位数码值
“`
(2)显示(ah)+30h和(al)+30h对应的ASCII码字符。
“`assembly
assume cs:code
code segment
start: mov al,8
out 70h,al
in al,7lh

mov ah,al
mov cl,4
shr ah,cl
and al, 00001111b

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
“`

点赞

Leave a Reply

Your email address will not be published. Required fields are marked *