luqingfei@C++

为中华之崛起而崛起!
兼听则明,偏听则暗。

汇编语言--端口

 

端口

 

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

 

PC机系统中,和CPU通过总线相连的芯片除各种存储器外,还有以下3种芯片:

1)各种接口卡(比如:网卡、显卡)上的接口芯片,它们控制接口卡进行工作;

2)主板上的接口芯片,CPU通过它们对部分外设进行访问;

3)其他芯片,用来存储相关的系统信息,或进行相关的输入输出处理。

 

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

1)都和CPU的总线相连,当然这种连接是通过它们所在的芯片进行的;

2CPU对它们进行读或写的时候都通过控制线向它们所在的芯片发出端口读写命令。

 

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

 

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

1CPU内部的寄存器;

2)内存单元;

3)端口中。(芯片中的寄存器)

 

 

 

端口的读写

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

 

对端口的读写不能用movpushpop等内存读写指令。

端口的读写指令只有两条:inout,分别用于从端口读出数据和往端口写入数据。

 

我们来看一下CPU执行内存访问指令和端口访问指令时候,总线上的信息:

1)访问内存:

       mov ax,ds:[8]               ;假设执行前(ds)=0

       执行时与总路线相关的操作:

       1CPU通过地址线将地址信息8发出;

       2CPU通过控制线发出内存读命令,选中存储器芯片,并通知它,将要从中读取数据;

       3)存储器将8号单元中的数据通过数据线送入CPU

 

2)访问端口:

       in al,60h         ;60h号端口读入一个字节

       执行时与总路线相关的操作:

       1CPU通过地址线将地址信息60h发出;

       2CPU通过控制线发出端口读命令,选中端口所在的芯片,并通知它,将要从中读取数据;

       3)端口所在的芯片将60h端口中的数据通过数据线送入CPU

 

inout指令中,只能使用axal来存放从端口中读入的数据或要发送到端口中的数据。访问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端口写入一个字节

 

 

 

CMOS RAM芯片

我们通过对CMOS RAM的读写来体会一下对端口的访问。

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

1)包含一个实时钟和一个有128个存储单元的RAM存储器(早期的计算机为64个字节)。

2)该芯片靠电池供电。所以,关机后其内部的实时钟仍可正常工作,RAM中的信息不丢失。

3128个字节的RAM中,内部实时钟占用0~0dh单元来保存时间信息,其余大部分单元用来保存系统配置信息,供系统启动时BIOS程序读取。BIOS也提供了相关的程序,使我们可以在开机的时候配置CMOS RAM中的系统信息。

4)该芯片内部有两个端口,端口地址为70h71hCPU通过这两个端口来读写CMOS RAM

570h为地址端口,存放要访问的CMOS RAM单元的地址:71h为数据端口,存放从选定的CMOS RAM单元中读取的数据。可见,CPUCMOS RAM的读写分两步进行了,比如:读CMOS RAM2号单元:

       1)将2送入端口70h

       2)从71h读出2号单元的内容。

 

检测点14.1

1)编程:读取CMOS RAM2号单元的内容。

       mov al,2h

       out 70h,al              ;2写入地址端口。

       in 71h,al         ;把数据端口的内容读出放入al寄存器。

 

2)编程:向CMOS RAM2号单元写入0

       mov al,2h

       out 70h,al              ;2写入地址端口

       mov al,0

       out 71h,al              ;0写入数据端口

 

 

 

shlshr指令

shlshr是逻辑移位指令。

 

shl是逻辑左移指令,它的功能为:

1)将一个寄存器或内存单元中的数据向左移位;

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

3)最低位用0补充。

指令:

mov al,01001000b

shl al,1                  ;al中的数据左移一位

执行后(al)=10010000bCF=0

 

如果移动位数大于1时,必须将移动位数放在cl中。

比如,指令:

mov al,01010001b

mov cl,3

shl al,cl

执行后(al)=10001000b,因为最后移出的一位是1,所以CF=1

 

可以看出,将X逻辑左移一位,相当于执行X=X*2

 

 

shr是逻辑右移指令,它和shl所进行的操作刚好相反:

1)将一个寄存器或内存单元中的数据向右移位;

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

3)最高位用0补充。

指令:

mov al,10000001b

shr al,1                  ;al中的数据右移一位

执行后(al)=01000000bCF=1

如果移动位数大于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

 

assume cs:code

code segment

 start:    mov ax,1000h

              mov bx,ax

              shl bx,1          ;(ax)=(ax)*2

 

              mov cl,3

              shl ax,cl         ;(ax)=(ax)*8

              add ax,bx              ;(ax)=(ax)*10

code ends

end start

 

 

 

 

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 2 6

 

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

 

编程:在屏幕中间显示当前的月份。

分析:这个程序主要做两部分工作:

1)从CMOS RAM8号单元读出当前月份的BCD码: 

       mov al,8         ;首先要向地址端口70h

       out 70h,al              ;写入要访问的单元的地址。

       in al,71h         ;然后,从数据端口71h中取得指定单元中的数据。

 

2)将用BCD码表示的月份以十进制的形式显示到屏幕上。

       BCD码值=十进制数码值,则BCD码值+30h=十进制数对应的ASCII码。

       CMOS RAM8号单元读出的一个字节中,包含了两个BCD码表示的两位十进制数,高4位的BCD码表示十位,低4位的BCD码表示个位。比如:00010100表示14

       我们需要进行两步工作:

1)将从CMOS RAM8号单元中读出的一个字节,分为两个表示BCD码值的数据。

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

       mov cl,4

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

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

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

 

完整程序如下:

assume cs:code

code segment

 start:

        mov al,8

        out 70h,al

        in al,71h   ;CMOS RAM8号单元读出当前月份的BCD码。

       

        mov ah,al   ;ah,al都存储着当前月份的BCD码。

        mov cl,4

        shr ah,cl   ;ah中内容逻辑右移4位,则只剩下月份的十位数码。

        and al,00001111b ;al中内容将只剩下月份的个位数码。

       

        add ah,30h

        add al,30h ;得到十进制数码的ASCII码值。

       

        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

 

 

 

 

 

 

 

posted on 2010-08-04 10:37 luqingfei 阅读(1707) 评论(4)  编辑 收藏 引用 所属分类: 汇编语言基础学习

评论

# re: 汇编语言--端口 2010-09-02 01:38 D

检测点14.1

1)编程:读取CMOS RAM的2号单元的内容。

mov al,2h

out 70h,al ;把2写入地址端口。

in 71h,al ;把数据端口的内容读出放入al寄存器。

其中应该是:in al,71h吧!!  回复  更多评论   

# re: 汇编语言--端口 2010-09-02 01:55 D

检测点14.2中,该种方法会影响cf的正确的值吗?  回复  更多评论   

# re: 汇编语言--端口 2010-09-02 08:35 路青飞

@D
你说的对,我写错了。
in 寄存器al/ax, 端口地址 ;从端口读入数据放到寄存器al或ax中
out 端口地址, 寄存器al/ax ;把寄存器al/ax中的数据读入到端口
这个地方是有点记不住,其实大部分的命令有个规则:
数据的传输方向是从右边的操作数流向左边的操作数。
比如:mov ax, 10h
  回复  更多评论   

# re: 汇编语言--端口 2010-09-02 14:55 D

@路青飞
对,没错!  回复  更多评论   


只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   知识库   博问   管理


导航

<2010年8月>
25262728293031
1234567
891011121314
15161718192021
22232425262728
2930311234

统计

留言簿(6)

随笔分类(109)

随笔档案(105)

Blogers

Game

Life

NodeJs

Python

Useful Webs

大牛

搜索

积分与排名

最新评论

阅读排行榜

评论排行榜