外中斷
以前我們討論的都是CPU對(duì)指令的執(zhí)行。
我們知道,CPU在計(jì)算機(jī)系統(tǒng)中,除了能夠執(zhí)行指令,進(jìn)行運(yùn)算以外,還應(yīng)該能夠?qū)ν獠吭O(shè)備進(jìn)行控制,接收它們的輸入,向它們進(jìn)行輸出。也就是說(shuō),CPU除了有運(yùn)算能力外,還要有I/O(Input/Output,輸入/輸出)能力。
比如,我們按下鍵盤(pán)上的確個(gè)鍵,CPU最終要能夠處理這個(gè)鍵。在使用文本編輯器時(shí),按下a鍵后,我們可以看到屏幕上出現(xiàn)“a”,是CPU將從鍵盤(pán)上輸入的鍵所對(duì)應(yīng)的字符送到顯示上的。
要及時(shí)處理外設(shè)的輸入,顯然需要解決兩個(gè)問(wèn)題:
一是外設(shè)的輸入隨時(shí)可能發(fā)生,CPU如何得知?
二是CPU從何處得到外設(shè)的輸入?
接口芯片和端口
在前面我們知道,在PC系統(tǒng)的接口卡和主板上,裝有各種接口芯片。
這些外設(shè)接口芯片的內(nèi)部有若干寄存器,CPU將這些寄存器當(dāng)作端口來(lái)訪問(wèn)。
外設(shè)的輸入不直接送入內(nèi)存和CPU,而是送入相關(guān)的接口芯片的端口中;
CPU向外設(shè)的輸出也不是直接送入外設(shè),而是先送入端口中,再由相關(guān)的芯片送到外設(shè)。
CPU還可以向外設(shè)輸出控制命令,而這些控制命令也是先送到相關(guān)芯片的端口中,然后再由相關(guān)的芯片根據(jù)命令對(duì)外設(shè)實(shí)施控制。
可見(jiàn),CPU通過(guò)端口和外部設(shè)備進(jìn)行聯(lián)系。
外中斷信息
現(xiàn)在,我們知道了外設(shè)的輸入被存放在端口中,可是外設(shè)的輸入隨時(shí)都有可能到達(dá),CPU如何及時(shí)地知道,并進(jìn)行處理呢?更一般地講,就是外設(shè)隨時(shí)都可能發(fā)生需要CPU及時(shí)處理的事件,CPU如何及時(shí)得知并進(jìn)行處理?
CPU提供中斷機(jī)制來(lái)滿足這種需要。前面講過(guò),當(dāng)CPU的內(nèi)部有需要處理的事情發(fā)生的時(shí)候,將產(chǎn)生中斷信息,引發(fā)中斷過(guò)程。這種中斷信息來(lái)自CPU的內(nèi)部。
還有一種中斷信息,來(lái)自于CPU外部,當(dāng)CPU外部有需要處理的事情發(fā)生的時(shí)候,比如說(shuō),外設(shè)的輸入到達(dá),相關(guān)芯片將向CPU發(fā)出相應(yīng)的中斷信息。CPU在執(zhí)行完當(dāng)前指令后,可以檢測(cè)到發(fā)送過(guò)來(lái)的中斷信息,引發(fā)中斷過(guò)程,處理外設(shè)的輸入。
在PC系統(tǒng)中,外中斷源一共有兩類:
1、可屏蔽中斷
可屏蔽中斷是CPU可以不響應(yīng)的外中斷,CPU是否響應(yīng)可屏蔽中斷,要看標(biāo)志寄存器的IF位的設(shè)置。當(dāng)CPU檢測(cè)到可屏蔽中斷信息時(shí),如果IF=1,則CPU在執(zhí)行完當(dāng)前指令后響應(yīng)中斷,引發(fā)中斷過(guò)程;如果IF=0,則不響應(yīng)可屏蔽中斷。
回憶一下內(nèi)中斷所引發(fā)的中斷過(guò)程:
1)取中斷類型碼n;
2)標(biāo)志寄存器入棧,IF=0,TF=0;
3)CS、IP入棧;
4)(IP)=(n*4),(CS)=(n*4+2)
由此轉(zhuǎn)去執(zhí)行中斷處理程序。
可屏蔽中斷所引發(fā)的中斷過(guò)程,除在第1步的實(shí)現(xiàn)上有所不同外,基本上和內(nèi)中斷的中斷過(guò)程相同。因?yàn)榭善帘沃袛嘈畔?lái)自于CPU外部,中斷類型碼是通過(guò)數(shù)據(jù)總路線送入CPU的;而內(nèi)中斷的中斷類型碼是在CPU內(nèi)部產(chǎn)生的。
現(xiàn)在,我們可以解釋中斷過(guò)程中將IF設(shè)置為0的原因了。將IF置0的原因就是,在進(jìn)入中斷處理程序后,禁止其他的可屏蔽中斷。
當(dāng)然,如果在中斷處理程序中需要處理可屏蔽中斷,可以用指令將IF置1。8086CPU提供的設(shè)置IF的指令如下:
sti,用于設(shè)置IF=1;
cli,用于設(shè)置IF=0。
2、不可屏蔽中斷
不可屏蔽中斷是CPU必須響應(yīng)的外中斷。當(dāng)CPU檢測(cè)到不可屏蔽中斷信息時(shí),則在執(zhí)行完當(dāng)前指令后,立即響應(yīng),引發(fā)中斷過(guò)程。
對(duì)于8086CPU,不可屏蔽中斷的中斷類型碼固定為2,所以中斷過(guò)程中,不需要取中斷類型碼。則不可屏蔽中斷的中斷過(guò)程為:
(1)標(biāo)志寄存器入棧,IF=0,TF=0;
(2)CS、IP入棧;
(3)(IP)=(8),(CS)=(0AH)。
幾乎所有由外設(shè)引發(fā)的外中斷,都是可屏蔽中斷。當(dāng)外設(shè)有需要處理的事件(比如說(shuō)鍵盤(pán)輸入)發(fā)生時(shí),相關(guān)芯片向CPU發(fā)出可屏蔽中斷信息。不可屏蔽中斷是在系統(tǒng)中有必須處理的情況發(fā)生時(shí)用來(lái)通知CPU的中斷信息。
PC機(jī)鍵盤(pán)的處理過(guò)程
下面來(lái)看一下鍵盤(pán)輸入的處理過(guò)程,并以此來(lái)體會(huì)一下PC機(jī)處理外設(shè)輸入的基本方法。
1、鍵盤(pán)輸入
鍵盤(pán)上的每一個(gè)鍵相當(dāng)于一個(gè)開(kāi)關(guān),鍵盤(pán)中有一個(gè)芯片對(duì)鍵盤(pán)上的每一個(gè)鍵的開(kāi)關(guān)狀態(tài)進(jìn)行掃描。
按下一個(gè)鍵時(shí),開(kāi)關(guān)接通,該芯片就產(chǎn)生一個(gè)掃描碼,掃描碼說(shuō)明了按下的鍵在鍵盤(pán)上的位置。掃描碼被送入主板上的相關(guān)接口芯片的寄存器中,該寄存器的端口地址為60H。
松開(kāi)按下的鍵時(shí),也產(chǎn)生一個(gè)掃描碼,掃描碼說(shuō)明了松開(kāi)的鍵在鍵盤(pán)上的位置。松開(kāi)按鍵時(shí)產(chǎn)生的掃描碼也被送入60H端口中。
一般將按下一個(gè)鍵時(shí)產(chǎn)生的掃描碼稱為通碼,松開(kāi)一個(gè)鍵產(chǎn)生的掃描碼稱為斷碼。
掃描碼長(zhǎng)度為一個(gè)字節(jié),通碼的第7位為0,斷碼的第7位為1,即:
斷碼=通碼+80H
比如:g鍵的通碼為22H,斷碼為a2H。
2、引發(fā)9號(hào)中斷
鍵盤(pán)的輸入到達(dá)60H端口時(shí),相關(guān)的芯片就會(huì)向CPU發(fā)出中斷類型碼為9的可屏蔽中斷信息。CPU檢測(cè)到該中斷信息后,如果IF=1,則響應(yīng)中斷,引發(fā)中斷過(guò)程,轉(zhuǎn)去執(zhí)行int 9中斷例程。
3、執(zhí)行int 9中斷例程
BIOS提供了int 9中斷例程,用來(lái)進(jìn)行基本的鍵盤(pán)輸入處理,主要的工作如下:
(1)讀出60H端口中的掃描碼;
(2)如果是字符鍵的掃描碼,將該掃描碼和它所對(duì)應(yīng)的字符碼(即ASCII碼)送入內(nèi)存中的BIOS鍵盤(pán)緩沖區(qū);如果是控制鍵(比如Ctrl)和切換鍵(比如CapsLock)的掃描碼,則將其轉(zhuǎn)變?yōu)?span>狀態(tài)字節(jié)(用二進(jìn)制位記錄控制鍵和切換鍵狀態(tài)的字節(jié))寫(xiě)入內(nèi)存中存儲(chǔ)狀態(tài)字節(jié)的單元。
(3)對(duì)鍵盤(pán)系統(tǒng)進(jìn)行相關(guān)的控制,比如說(shuō),向相關(guān)芯片發(fā)出應(yīng)答信息。
BIOS鍵盤(pán)緩沖區(qū)是系統(tǒng)啟動(dòng)后,BIOS用于存放int 9中斷例程所接收的鍵盤(pán)輸入的內(nèi)存區(qū)。該內(nèi)存區(qū)可以存儲(chǔ)15個(gè)鍵盤(pán)輸入,因?yàn)?/span>int 9中斷例程除了接收掃描碼外,還要產(chǎn)生和掃描碼對(duì)應(yīng)的字符碼,所以在BIOS鍵盤(pán)緩沖區(qū)中,一個(gè)鍵盤(pán)輸入用一個(gè)字單元存放,高位字節(jié)存放掃描碼,低位字節(jié)存放字符碼。
0040:17單元存儲(chǔ)鍵盤(pán)狀態(tài)字節(jié),該字節(jié)記錄了控制鍵和切換鍵的狀態(tài)。鍵盤(pán)狀態(tài)字節(jié)各位記錄的信息如下:
0:右shift狀態(tài),置1表示按下右shift鍵;
1:左shift狀態(tài),置1表示按下左shift鍵;
2:Ctrl狀態(tài),置1表示按下Ctrl鍵;
3:Alt狀態(tài),置1表示按下Alt鍵;
4:ScrolLock狀態(tài),置1表示Scroll指示燈亮;
5:NumLock狀態(tài),置1表示小鍵盤(pán)輸入的是數(shù)字;
6:CapsLock狀態(tài),置1表示輸入大寫(xiě)字母;
7:Insert狀態(tài):置1表示處于刪除狀態(tài)。
編寫(xiě)int 9中斷例程
鍵盤(pán)輸入的處理過(guò)程:
1)鍵盤(pán)產(chǎn)生掃描碼;
2)掃描碼送入60h端口;
3)引發(fā)9號(hào)中斷;
4)CPU執(zhí)行int 9中斷例程處理鍵盤(pán)輸入。
上面的過(guò)程中,第(1)、(2)、(3)步都是由硬件系統(tǒng)完成的。我們能夠改變的只有int 9的中斷處理程序。
編程:在屏蔽中間依次顯示“a”~“z”,并可以讓人看清。在顯示的過(guò)程中,按下Esc鍵后,改變顯示的顏色。
assume cs:code
code segment
start: mov ax,0b800h
mov es,ax
mov ah,’a’
s: mov es:[160*12+40*2],ah
inc ah
cmp ah,’z’
jna s
mov ax,4c00h
int 21h
code ends
end start
上面的程序的執(zhí)行過(guò)程中,我們無(wú)法看清屏幕上的顯示,因?yàn)橐粋€(gè)字母剛顯示到屏幕上,因?yàn)?/span>CPU執(zhí)行指令太快了。
我們應(yīng)該在每顯示一個(gè)字母后,延時(shí)一段時(shí)間,讓人看清后,再顯示下一個(gè)字母。
那么如何延時(shí)呢?我們讓CPU執(zhí)行一段時(shí)間的空循環(huán),因?yàn)楝F(xiàn)在的CPU的速度都非常快,所以循環(huán)的次數(shù)一定要大,我們用兩個(gè)16位寄存器來(lái)存放32位的循環(huán)次數(shù)。如下:
mov dx,10h
mov ax,0
s: sub ax,1
sbb dx,0
cmp ax,0
jne s
cmp dx,0
jne s
上面的程序,循環(huán)100000h次。我們可以將循環(huán)延時(shí)的程序段寫(xiě)為一個(gè)子程序。
assume cs:code
stack segment
db 128 dup (0)
stack ends
code segment
start: mov ax,stack
mov ss,ax
mov sp,128
mov ax,0b800h
mov ex,ax
mov ah,’a’
s: mov es:[160*12+40*2],ah
call delay
inc ah
cmp ah,’z’
jna s
mov ax,4c00h
int 21h
delay: push ax
push dx
mov dx,1000h ;循環(huán)10000000次,讀者可以根據(jù)自己機(jī)器的速度調(diào)整循環(huán)次數(shù)
mov ax,0
s1: sub ax,1
sbb dx,0
jne sl
cmp dx,0
jne s1
pop dx
pop ax
ret
code ends
end start
上面的程序,顯示“a”~“z”,并可以讓人看清,這個(gè)任務(wù)已經(jīng)實(shí)現(xiàn)。
那么如何實(shí)現(xiàn),按下Esc鍵后,改變顯示的顏色呢?
鍵盤(pán)輸入到達(dá)60h端口后,就會(huì)引發(fā)9號(hào)中斷,CPU則轉(zhuǎn)去執(zhí)行int 9中斷例程。我們可以編寫(xiě)int 9中斷例程,功能如下:
(1)從60h端口讀出鍵盤(pán)的輸入;
(2)調(diào)用BIOS的int 9中斷全程,處理其他硬件細(xì)節(jié);
(3)判斷是否為Esc鍵的掃描碼,如果是,改變顯示的顏色后返回;如果不是則直接返回。
分析:
1、從端口60h讀出鍵盤(pán)的輸入
in al,60h
2、調(diào)用BIOS的int 9中斷例程
注意,我們寫(xiě)的中斷處理程序要成為新的int 9中斷例程,主程序必須要將中斷向量表中的int 9中斷例程的入口地址改為我們寫(xiě)的中斷處理程序的入口地址。則在新的中斷處理程序中調(diào)用原來(lái)的int 9中斷例程時(shí),中斷向量表中的int 9中斷例程的入口地址卻不是原來(lái)的int 9中斷例程的地址,所以我們不能使用int 指令直接調(diào)用。
要能在我們寫(xiě)的新的中斷例程中調(diào)用原來(lái)的中斷例程,就必須在將中斷向量表中的中斷例程的入口地址改為新地址之前,將原來(lái)的入口地址保存起來(lái)。這樣,在需要調(diào)用的時(shí)候我們才能找到原來(lái)的中斷例程的入口。
有了入口地址后,我們?nèi)绾芜M(jìn)行調(diào)用呢?
當(dāng)然不能使用指令int 9來(lái)調(diào)用,我們可以用別的指令來(lái)對(duì)int 指令進(jìn)行一些模擬,從而實(shí)現(xiàn)對(duì)中斷例程的調(diào)用。
int指令在執(zhí)行的時(shí)候,CPU進(jìn)行下面的工作:
1)取中斷類型碼n;
2)標(biāo)志寄存器入棧;
3)IF=0,TF=0;
4)CS、IP入棧;
5)(IP)=(n*4),(CS)=(n*4+2)。
取中斷類型碼是為了定位中斷例程的入口地址,在我們的問(wèn)題中,中斷例程的入口地址已經(jīng)知道。所以,我們用別的指令模擬int指令時(shí)候,不需要做第(1)步。在假設(shè)要調(diào)用的中斷例程的入口地址在ds:0和ds:2單元中的前提下,我們將int過(guò)程用下面幾步模擬:
1)標(biāo)志寄存器入棧;
2)IF=0,TF=0;
3)CS、IP入棧;
4)(IP)=((ds)*16+0),(CS)=((ds)*16+2)。
所以int 過(guò)程的模擬過(guò)程變?yōu)椋?/span>
1)標(biāo)志寄存器入棧;
2)IF=0,TF=0;
3)call dword ptr ds:[0]
對(duì)于(1),可用pushf實(shí)現(xiàn)。
對(duì)于(2),可用下面的指令實(shí)現(xiàn)。
pushf
pop ax
and ah,11111100b ;IF和TF為標(biāo)志寄存的第9位和第8位
push ax
popf
則,模擬int指令的調(diào)用功能,調(diào)用入口地址在ds:0、ds:2中的中斷例程的程序?yàn)椋?/span>
pushf ;標(biāo)志寄存器入棧
pushf
pop ax
and ah,11111100b
push ax
popf ;IF=0, TF=0
call dword ptr ds:[0] ;CS、IP入棧;(IP)=((ds)*16+0), (CS)=((ds)*16+2)
3、如果是Esc鍵掃描碼,改變顯示的顏色后返回
如果改變顯示的顏色?
顯示的位置是屏幕的中間,即第12行40列,顯存中的偏移地址為:160*12+40*2。所以字符的ASCII碼要送入b800:160*12+40*2處。而b800:160*12+40*2+1處是字符的屬性,我們只要改變此處的數(shù)據(jù)就可以改變?cè)?/span>b800:160*12+40*2處顯示的字符的顏色了。
該程序的最后一個(gè)問(wèn)題是,要在程序返回前,將中斷向量表中的int 9中斷例程的入口地址恢復(fù)為原來(lái)的地址。否則程序返回后,別的程序?qū)o(wú)法使用鍵盤(pán)。
完整的程序如下:
assume cs:code
stack segment
db 128 dup (0)
stack ends
data segment
dw 0,0
data ends
code segment
start:mov ax,stack
mov ss,ax
mov sp,128
mov ax,data
mov ds,ax
mov ax,0
mov es,ax
push es:[9*4]
pop ds:[0]
push es:[9*4+2]
pop ds:[2] ;將原來(lái)的int 9中斷例程的入口地址保存在ds:0、ds:2單元中
mov word ptr es:[9*4],offset int9
mov es:[9*4+2],cs ;在中斷向量表中設(shè)置新的int 9中斷例程的入口地址
mov ax,0b800h
mov es,ax
mov ah,'a'
s: mov es:[160*12+40*2],ah
call delay
inc ah
cmp ah,'z'
jna s
mov ax,0
mov es,ax
push ds:[0]
pop es:[9*4]
push ds:[2]
pop es:[9*4+2] ;將中斷向量表中int 9中斷例程的入口恢復(fù)為原來(lái)的地址
mov ax,4c00h
int 21h
delay:push ax
push dx
mov dx,1000h
mov ax,0
s1:sub ax,1
sbb dx,0
cmp ax,0
jne s1
cmp dx,0
jne s1
pop dx
pop ax
ret
;-------以下為新的int 9中斷例程------------------
int9: push ax
push bx
push es
in al,60h
pushf
pushf
pop bx
and bh,11111100b
push bx
popf
call dword ptr ds:[0] ;對(duì)int指令進(jìn)行模擬,調(diào)用原來(lái)的int 9中斷例程
cmp al,1
jne int9ret
mov ax,0b800h
mov es,ax
inc byte ptr es:[160*12+40*2+1] ;將屬性值加1,改變顏色
int9ret:pop es
pop bx
pop ax
iret
code ends
end start
檢測(cè)點(diǎn)15.1
(1)模擬int指令調(diào)用原int9中斷例程的程序段是可以精簡(jiǎn)的,因?yàn)樵谶M(jìn)入中斷例程后,IF和TF都已經(jīng)置0,沒(méi)有必要再進(jìn)行設(shè)置了。
對(duì)于程序段:
pushf
pushf
pop ax
and ah,11111100b
push ax
popf
call dword ptr ds:[0]
可以精簡(jiǎn)為:
pushf
call dword ptr ds:[0]
再條指令。
(2)仔細(xì)分析上面程序中的主程序,會(huì)發(fā)現(xiàn)一個(gè)潛在的問(wèn)題?在主程序中,如果在執(zhí)行設(shè)置int9中斷例程的段地址和偏移地址的指令之間發(fā)生了鍵盤(pán)中斷,則CPU將轉(zhuǎn)去一個(gè)錯(cuò)誤的地址執(zhí)行,將發(fā)生錯(cuò)誤。
排除潛在問(wèn)題方法:
在pop ds:[2]指令后加入一條cli指令,并在mov es:[9*4+2],cs指令后加入一條sti指令即可。意思是在這段期間,讓IF=0,禁止或屏蔽外中斷處事程序的執(zhí)行。
安裝新的int9中斷處理例程
安裝一個(gè)新的int 9中斷例程,使得原int 9中斷例程的功能得到擴(kuò)展。
任務(wù):安裝一個(gè)新的int 9中斷例程,功能:在DOS下,按F1鍵后改變當(dāng)前屏幕的顯示顏色,其他的鍵照常處理。
分析:
(1)改變屏幕的顯示顏色
改變從B8000開(kāi)始的4000個(gè)字節(jié)中的所有奇地址單元中的內(nèi)容,當(dāng)前屏幕的顯示顏色即發(fā)生改變。程序如下:
mov ax,0b800h
mov es,ax
mov bx,1
mov cx,2000
s: nc byte ptr es:[bx]
add bx,2
loop s
(2)其他鍵照常處理
可以調(diào)用原int 9中斷處理程序,來(lái)處理其他的鍵盤(pán)輸入。
(3)原int 9中斷例程入口地址的保存
因?yàn)樵诰帉?xiě)的新int 9中斷例程中要調(diào)用原int 9中斷例程,所以,要保存原int 9中斷例程的入口地址。保存在哪里?顯然不能保存在安裝程序中,因?yàn)榘惭b程序返回后地址將丟失。我們將地址保存在0:200單元處。
(4)新int 9中斷例程的安裝
我們可將新的int 9中斷例程安裝在0:204處。
完整的程序如下:
assume cs:code
stack segment
db 128 dup (0)
stack ends
code segment
start:
mov ax,stack
mov ss,ax
mov sp,128
push cs
pop ds
mov ax,0
mov es,ax
mov si,offset int9 ;設(shè)置ds:si指向源地址
mov di,204h ;設(shè)置es:di指向目的地址
mov cx,offset int9end - offset int9 ;設(shè)置cx為傳輸長(zhǎng)度
cld ;設(shè)置傳輸方向?yàn)檎?/span>
rep movsb
push es:[9*4]
pop es:[200h]
push es:[9*4+2]
pop es:[202h] ;保存原有的int 9中斷例程入口地址
cli ;設(shè)置IF為0,禁止CPU執(zhí)行可屏蔽外中斷。
mov word ptr es:[9*4],204h
mov word ptr es:[9*4+2],0 ;設(shè)置int 9中斷向量,指向新的中斷例程入口地址
sti ;恢復(fù)設(shè)置IF為1
mov ax,4c00h
int 21h
int9:push ax
push bx
push cx
push es
in al,60h
pushf
call dword ptr cs:[200h] ;當(dāng)此中斷例程執(zhí)行時(shí)(CS)=0
;cmp al,1 ;F1的掃描碼為3bh
;jne int9ret
mov ax,0b800h
mov es,ax
mov bx,1
mov cx,2000
s:mov byte ptr es:[bx],2
;inc byte ptr es:[bx]
and bx,2
loop s
int9ret:pop es
pop cx
pop bx
pop ax
iret
int9end:nop
code ends
end start
通過(guò)對(duì)鍵盤(pán)輸入的處理,了解了CPU對(duì)外設(shè)輸入的通常 通常方法,即:
1)外設(shè)的輸入送入端口;
2)向CPU發(fā)出外中斷(可屏蔽中斷)信息;
3)CPU檢測(cè)到可屏蔽中斷信息,如果IF=1,CPU在執(zhí)行完當(dāng)前指令后響應(yīng)中斷,執(zhí)行相應(yīng)的中斷例程;
4)可在中斷例程中實(shí)現(xiàn)對(duì)外設(shè)輸入的處理。
端口和中斷機(jī)制,是CPU進(jìn)行I/O的基礎(chǔ)。
指令系統(tǒng)總結(jié)
8086CPU提供以下幾大類指令:
1、數(shù)據(jù)傳送指令
mov, push, pop, pushf, popf, xchg等。
實(shí)現(xiàn)寄存器和內(nèi)存、寄存器和寄存器之間的單個(gè)數(shù)據(jù)傳送。
2、自述運(yùn)算指令
add, sub, adc, sbb, inc, dec, cmp, imul, idiv, aaa 等。
實(shí)現(xiàn)寄存器和內(nèi)存中的數(shù)據(jù)的算術(shù)運(yùn)算。它們的執(zhí)行結(jié)果影響標(biāo)志寄存器的:sf, zf, of, cf, pf, af位。
3、邏輯指令
and, or, not, xor, test, shl, shr, sal, sar, rol, ror, rcl, rcr等。
除了not指令外,它們的執(zhí)行結(jié)果都影響標(biāo)志寄存器的相關(guān)標(biāo)志位。
4、轉(zhuǎn)移指令
可以修改IP,或同時(shí)修改CS和IP的指令統(tǒng)稱為轉(zhuǎn)移指令。分為以下幾類:
1)無(wú)條件轉(zhuǎn)移指令,比如:jmp;
2)條件轉(zhuǎn)移指令,比如:jcxz, je, jb, ja, jnb, jna等;
3)循環(huán)指令,比如:loop;
4)過(guò)程,比如:call, ret, reft;
5)中斷,比如:int, iret。
5、處理機(jī)控制指令
這些指令對(duì)標(biāo)志寄存器或其他處理機(jī)狀態(tài)進(jìn)行設(shè)置,比如:cld, std, cli, sti, nop, clc, cmc, stc, hlt, wait, csc, lock等都是處理機(jī)控制指令。
6、串處理指令
這些指令對(duì)內(nèi)存中的批量數(shù)據(jù)進(jìn)行處理,比如:movsb, movsw, cmps, scas, lods, stos等。
基要使用這些指令方便地進(jìn)行批量數(shù)據(jù)的處理,則需要和rep, repe, repne等前綴指令配合使用。