疯狂的暑假学习之 汇编入门学习笔记 (十二)—— int指令、端口
参考: 《汇编语言》 王爽 第13、14章
int n指令,相当于引发一个n号中断。
执行过程相当于:
(1)取中断类型吗n。
(2)标志寄存器入栈;设置IF=0,TF=0。
(3)CS,IP入栈
(4)(IP)=(n*4),(CS)=(n*4+2)
例子1:编写、安装中断7ch,实现求一个word型数据的平方,用ax存放这个数据。
assume cs:code code segment start: mov ax,cs mov ds,ax mov si,offset sqr mov ax,0 mov es,ax mov di,200h mov cx,offset sqrend - offset sqr cld rep movsb mov ax,0 mov es,ax mov word ptr es:[7ch*4],200h mov word ptr es:[7ch*4+2],0 mov ax,4c00h int 21h sqr: mul ax iret sqrend: nop code ends end start
assume cs:code code segment start: mov ax,3 int 7ch mov ax,4c00h int 21h code ends end start
例子2:编写、安装7ch中断,实现将data段中的字符串转化为大写。
assume cs:code code segment start: mov ax,cs mov ds,ax mov si,offset change mov ax,0 mov es,ax mov di,200h mov cx,offset changeend - offset change cld rep movsb mov ax,0 mov es,ax mov word ptr es:[7ch*4],200h mov word ptr es:[7ch*4+2],0 mov ax,4c00h int 21h change: push cx push si s: mov ch,0 mov cl,ds:[si] jcxz ok and cl,11011111b mov ds:[si],cl inc si jmp short s ok: pop si pop cx iret changeend: nop code ends end start
assume cs:code data segment db 'conversation',0 data ends code segment start: mov ax,data mov ds,ax mov si,0 int 7ch mov ax,4c00h int 21h code ends end start
例子3:使中断7ch实现loop指令的功能。
assume cs:code code segment start: mov ax,cs mov ds,ax mov si,offset lp mov ax,0 mov es,ax mov di,200h mov cx,offset lpend - offset lp cld rep movsb mov ax,0 mov es,ax mov word ptr es:[7ch*4],200h mov word ptr es:[7ch*4+2],0 mov ax,4c00h int 21h lp: push bp mov bp,sp dec cx jcxz lpret add [bp+2],bx lpret: pop bp iret lpend: nop code ends end start
assume cs:code code segment start: mov ax,0b800h mov es,ax mov di,160*12 mov bx,offset s - offset se mov cx,80 s: mov byte ptr es:[di],'!' add di,2 int 7ch se: nop mov ax,4c00h int 21h code ends end start
BIOS中主要包含:
(1)硬件系统的检测和初始化程序
(2)外部中断和内部中断的中断例程
(3)用于对硬件设备进行I/O操作的中断例程
(4)其他和硬件系统相关的中断例程
BIOS和DOS提供的中断例程安装过程
(1)开机,初始化CS=0FFFFH,IP=0,执行FFFF:0程序。FFFF:0程序有一条跳转指令,执行完后,CPU会去执行BIOS中的硬件系统检测和初始化程序。
(2)初始化程序将建立BIOS所支持的中断向量,即将BIOS中提供的中断例程的入口地址登记在中断向量表中。注意:BIOS所提供的中断例程,只需要将入口地址登记在中断向量表中即可,因为它们是固化在ROM中的。
(3)硬件系统检测和初始化完成后,调用int 19h进行操作系统的引导。从此将计算机交给操作系统控制
(4)DOS启动后,除了完成其他工作外,还将他所提供的中断例程装入内存中,并建立相应的中断向量。
一般来说,一个中断例程包含了多个子程序,用ah来传递内部子程序的编号
它是BIOS提供的中断例程,其中包含了多个和屏幕输出有关的子程序
ah=2 设置光标位置,此时bh存放 第几页;dh存放行号;dl存放列号
ah=9 在光标位置显示字符,此时al存放字符;bl存放颜色属性;bh存放第几页;cx存放这个字符重复显示几
例子:
assume cs:code code segment start: mov ah,2 mov bh,0 mov dh,5 mov dl,12 int 10h mov ah,9 mov al,'a' mov bl,11001010b mov bh,0 mov cx,3 int 10h mov ax,4c00h int 21h code ends end start
它是DOS提供的中断例程,包含了DOS提供给程序员在编程时调用的子程序。
前面我们一直用下面的代码返回
mov ax,4c00h int 21h
其实这是用了21号中断例程的4ch号子程序,al是用来存放程序返回值的,上面的代码即0本质就是:
mov ah,4ch mov al,0 int 21h
21h号中断例程的9号子程序功能是在光标位置显示字符串,用ds:dx指定字符串的首地址,字符串要以‘$’结尾例子:
assume cs:code data segment db 'Welcome to masm!','$' data ends code segment start: mov ah,2 mov bh,0 mov dh,5 mov dl,12 int 10h mov ax,data mov ds,ax mov dx,0 ;ds:dx指向字符串的首地址 data:0 mov ah,9 int 21h mov ax,4c00h int 21h code ends end start
各种芯片(网卡,显卡等)都有一组可以由CPU读写的寄存器。CPU可以通过这些寄存器读取数据,或者操控芯片。CPU把这些寄存器当做端口,对这些寄存器进行统一编址。每个端口宅地址空间中都有一个地址。
访问端口用in和out
如果:
in al,60h ; 从60h端口读取数据放到al中
注意:只能用ax或者al来存放从端口中读入的数据或要发送到端口的数据。
对0~255以内的端口进行读写时:
in al,20h
out 20h,al
对256~65535的端口进行读写时,端口号放在dx中:
mov dx,3f8h
in al.dx
ou dx,al
shl:左移指令,最低位补0,将移出的追后以为写入CF中
如果是移1位就:
shl al,1
如果移动大于1位,必须将移动的位数放在cl中
mov cl,3
shl al,cl
shr是右移,其他与shl一样
PC中有一个靠电池供电CMOS RAM芯片,它包含一个实时种和一个有128个存储单元的RAM存储器。
它有两个端口:
70h:保存要读取内存单元的地址
71h:读取内存单元
CMOS RAM中存储的时间信息
0:秒;2:分;
4:时;7:日;
8:月;9:年;
例子:在屏幕显示月份。
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 ;'0' 的ASCII值 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
原文地址:http://blog.csdn.net/billvsme/article/details/38014881