王爽《汇编语言》(14-16章)学习笔记

《汇编语言》王爽–学习笔记

十二、端口

1. 端口基本概念

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

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

  1. 各种接口卡(如网卡、显卡)上的接口芯片,它们控制接口卡进行工作;
  2. 主板上的接口芯片,CPU通过它们对部分外设进行访问;
  3. 其他芯片,用来存储相关系统信息,或进行相应的输入输出处理。

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

  1. 都和CPU的总线相连(通过它们所在的芯片进行);
  2. CPU对他们进行读、写的时候都通过控制线向他们所在的芯片发出端口读写指令。

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

CPU可以直接读取下面三个地方的数据:

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

2. 端口的读写

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

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

示例:

;对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端口写入一个字节

注意:在 in 和 out 指令中,只能使用ax或al来存放从端口中读入的数据或要发送到端口中的数据。

3. CMOS RAM芯片

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

  1. 包含一个实时钟和一个有128个存储单元的RAM存储器

  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号单元的内容。

CMOS RAM中存储的时间信息

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

024789

这些数据以BCD码的方式存放。

BCD码是以4位二进制数表示十进制数码的编码方法,如下表所示:

0123456789
0000000100100011010001010110011110001001

4. shl和shr指令

shl和shr是逻辑移位指令。

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

  1. 将一个寄存器或内存单元中的数据向左移位;
  2. 将最后移出的一位写入CF中;
  3. 最低位用0补充。

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

mov al, 01001000b 
shl al, 1 			;将a1中的数据左移一位
;执行后(al)=10010000b,CF=0。

mov al, 10000001b 
shr al, 1  ;将al中的数据右移一位
;执行后(al)=01000000b,CF=1。

十三、外中断

CPU在计算机系统中,除了能够执行指令,进行运算以外,还应该拥有I/O(输入输出)能力。

1. 接口芯片和端口

PC系统的接口卡和主板上,装有各种接口芯片。这些外设接口芯片的内部有若干寄存器,CPU将这些寄存器当作端口来访问。

外设的输入不直接送入内存和CPU,而是送入相关的接口芯片的端口中;CPU向外设的输出也不是直接送入外设,而是先送入端口中,再由相关的芯片送到外设。CPU还可以向外设输出控制命令,而这些控制命令也是先送到相关芯片的端口中,然后再由相关的芯片根据命令对外设实施控制。

即: CPU通过端口和外部设备进行联系。

2. 外中断信息

当CPU外部有需要处理的事情发生的时候,比如说,外设的输入到达,相关芯片将向CPU发出相应的中断信息。CPU在执行完当前指令后,可以检测到发送过来的中断信息,引发中断过程,处理外设的输入。

在PC系统中,外中断源有两类:

2.1 可屏蔽中断

可屏蔽中断是CPU可以不响应的外中断。CPU是否响应可屏蔽中断,要看标志寄存器的IF位的设置。

当CPU检测到可屏蔽中断信息时,如果IF=1,则CPU在执行完当前指令后响应中断,引发中断过程;如果IF=0,则不响应可屏蔽中断。可屏蔽中断过程类似于内部中断:

  1. 获取中断类型码n(从外部通过总线输入)
  2. 标志寄存器入栈,IF=0,TF=0
  3. CS,IP入栈
  4. (IP)=(n*4),(CS)=(n*4+2)

中断过程中将IF置0的原因:在进入中断处理程序后,禁止其他的可屏蔽中断。如果在中断处理程序中需要处理可屏蔽中断,可以用指令将IF置1。

8086CPU提供的设置IF的指令:

  • sti,设置IF=1;
  • cli,设置IF=0。

2.2 不可屏蔽中断

不可屏蔽中断是CPU必须响应的外中断。当CPU检测到不可屏蔽中断信息时,则在执行完当前指令后,立即响应,引发中断过程。

对于8086CPU,不可屏蔽中断的中断类型码固定为2,所以中断过程中,不需要取中断类型码。则不可屏蔽中断的中断过程为:

  1. 标志寄存器入栈,IF=0,TF=0;
  2. CS、IP入栈;
  3. (IP)=(8),(CS)=(0AH)。

几乎所有由外设引发的外中断,都是可屏蔽中断。如键盘输入,不可屏蔽中断通常是在系统中有必须处理的紧急情况发生时通知CPU的中断信息。

3. PC机键盘的处理过程

1.键盘输入

键盘上每个按键都相当于一个开关,键盘中有一个芯片对键盘上的每一个键的开关状态进行扫描。

按下一个键时,开关接通,该芯片就产生一个扫描码,扫描码说明了按下的键在键盘上的位置。扫描码被送入主板上的相关接口芯片的寄存器中,该寄存器的端口地址为 60h。

松开按下的键时,也产生一个扫描码,扫描码说明了松开的键在键盘上的位置。松开按键时产生的扫描码也被送入 60h 端口中。

一般将按下一个键时产生的扫描码称为通码,松开一个键产生的扫描码称为断码。

扫描码长度为一个字节,通码的第7位为0,断码的第7位为1。
即:断码 = 通码 + 80h
比如,g键的通码为 22h,断码为 a2h。

在这里插入图片描述

在这里插入图片描述

2.引发9号中断

键盘的输入到达60h端口时,相关的芯片就会向CPU发出中断类型码为9的可屏蔽中断信息。CPU检测到该中断信息后,如果IF=1,则响应中断,引发中断过程,转去执行int 9中断例程。

3.执行 int 9 中断例程

BIOS提供了int 9中断例程,用来进行基本的键盘输入处理,主要的工作如下:

  1. 读出60h端口中的扫描码;
  2. 如果是字符键的扫描码,将该扫描码和它所对应的字符码(即ASCII码)送入内存中的BIOS键盘缓冲区;
    如果是控制键(比如Ctrl)和切换键(比如CapsLock)的扫描码,则将其转变为状态字节(用二进制位记录控制键和切换键状态的字节)写入内存中存储状态字节的单元;
  3. 对键盘系统进行相关的控制,比如说,向相关芯片发出应答信息。

BIOS 键盘缓冲区(是系统启动后,BIOS用于存放int 9中断例程所接收的键盘输入的内存区)可以存储15个键盘输入,在该缓冲区中一个键盘输入用一个字单元存放,高位字节存放扫描码,低位字节存放字符码。

0040:17单元存储键盘状态字节,该字节记录了控制键和切换键的状态。键盘状态字节各位记录的信息如下。

状态置1
0右shift状态置1表示按下右shift键
1左shift状态置1表示按下左shift键
2Ctrl状态置1表示按下Ctrl键
3Alt状态置1表示按下Alt键
4ScrollLock状态置1表示Scroll指示灯亮
5NumLock状态置1表示小键盘输入的是数字
6CapsLock状态置1表示输入大写字母
7Insert状态置1表示处于删除态

4. 编写int 9中断例程

示例,见王爽《汇编语言》P276-P284

CPU对外设输入的通常处理方法:

  1. 外设的输入送入端口;
  2. 向CPU发出外中断(可屏蔽中断)信息;
  3. CPU检测到可屏蔽中断信息,如果IF=1,CPU在执行完当前指令后响应中断,执行相应的中断例程;
  4. 可在中断例程中实现对外设输入的处理。

端口和中断机制,是CPU进行I/O的基础。

十四、指令系统总结

对8086CPU的指令系统进行一下总结。读者若要详细了解8086指令系统中的各个指令的用,可以查看有关的指令手册。

8086CPU提供以下几大类指令。

  1. 数据传送指令
    如:mov、push、pop、pushf、popf、xchg 等都是数据传送指令,这些指令实现寄存器和内存、寄存器和寄存器之间的单个数据传送。
  2. 算术运算指令
    如:add、sub、adc、sbb、inc、dec、cmp、imul、idiv、aaa等都是算术运算指令,这些指令实现寄存器和内存中的数据的算数运算。它们的执行结果影响标志寄存器的sf、zf、of、cf、pf、af位。
  3. 逻辑指令
    如:and、or、not、xor、test、shl、shr、sal、sar、rol、ror、rcl、rcr等都是逻辑指令。除了not指外,它们的执行结果都影响标志寄存器的相关标志位。
  4. 转移指令
    可以修改IP,或同时修改CS和IP的指令统称为转移指令。转移指令分为以下几类。
    (1)无条件转移指令,比如,jmp;
    (2)条件转移指令,比如,jcxz、je、jb、ja、jnb、jna等;
    (3)循环指令,比如,loop;
    (4)过程,比如,call、ret、retf;
    (5)中断,比如,int、iret。
  5. 处理机控制指令
    这些指令对标志寄存器或其他处理机状态进行设置,如:cld、std、cli、sti、nop、clc、cmc、stc、hlt、wait、esc、lock等都是处理机控制指令。
  6. 串处理指令
    这些指令对内存中的批量数据进行处理,如:movsb、movsw、cmps、scas、lods、stos等。若要使用这些指令方便地进行批量数据的处理,则需要和rep、repe、repne 等前缀指令配合使用。

十五、直接定值表

讨论如何有效合理地组织数据,以及相关的编程技术。

1. 描述单元长度的标号

在之前,我们一直在代码段中使用标号来标记指令、数据、段的起始地址。这些标号仅仅表示了内存单元的地址。

如:

assume cs:code
code segment
         a : db 1,2,3,4,5,6,7,8  ;在a后面加有":"
         b : dw 0
start :mov si,offset a
         mov bx,offset b
         mov cx,8
    s : mov al,cs:[si]
         mov ah,0
         add cs:[bx],ax
         inc si
         loop s
         mov ax,4c00h
         int 21h
code ends
end start

注意:在后面加有 “ :” 的地址标号,只能在代码段中使用,不能在其他段中使用。

现在我们引出新的概念:数据标号,这种标号 不但表示内存单元的地址,还表示了内存单元的长度(即此标号处的单元,是一个字节单元还是字单元,还是双字单元)。

上面的程序现在可以写成这样:

assume cs:code
code segment
          a db 1,2,3,4,5,6,7,8 	;标号a、b后面没有":"
          b dw 0               
start :  mov si,0
          mov cx,8
    s :   mov al,a[si]
          mov ah,0
          add b,ax
          inc si
          loop s
          mov ax,4c00h
          int 21h
code ends
end start

在code段中a和b后并没有 ”:” 号,这种写法同时描述内存地址和单元长度的标号。

标号a ,描述了地址 code:0 和从这个地址开始后的内存单元都是字节单元;
而标号b,描述了地址 code:8 和从这个地址开始以后的内存单元都是字单元。

2.在其他段中使用数据标号

注意:
如果想在代码段中直接使用数据标号访问数据,需要用伪指令 assume 将标号所在的段和一个段寄存器联系起来。否则,编译器在编译的时候,无法确定标号的段地址在哪一个寄存器中。

当然,并不是说用伪指令 assume 将标号所在的段和某个段相联系,段寄存器中就真的存放该段的地址。在程序中还要使用指令对段寄存器进行设置。

如:在程序中用ds寄存器和data段相连,则编译器对相关指令的编译如下

指令:  mov al,a[si]
编译为:mov al,[si+0]

指令:  add b,ax
编译为:add [8],ax
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

ClimberCoding

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

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

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

打赏作者

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

抵扣说明:

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

余额充值