一、前 言
Windows提供強大的功能以及友好的圖形用戶界面(GUI),使得它不僅廣泛的用作管理事務型工作的支持平台,也被工業領域的工程人員所關注。
但Windows3.1並非基於優先級來調度任務,無法立即響應外部事件中斷,也就不能滿足工業應用環境中實時事件處理和實時控制應用的要求。因此,如何在Windows環境中處理外部實時事件一直是技術人員尤其是實時領域工程人員所關注的問題。目前已有的方法大都采用內掛實時多任務內核的方式,如windows的實時控制軟件包FLX等,而iRMX實時操作系統則把Windows3.1當作它的一個任務來運行。對於大型的工程項目,開發人員可采用購買實時軟件然後集成方式。對中小項目,從投資上考慮就不很經濟。如何尋找一種簡明的方法來處理外部實時事件依然顯得很必要。
本文首先闡述windows的消息機制及中斷機制,然後結合DPMI接口,給出一種保護模式下中斷程序的設計方法,以處理外部實時事件。經實際運行結果表明,該方法具有簡潔、實用、可靠的特點,並同樣可運行於Win95。
二、Windows的消息機制
Windows是一消息驅動式系統,。 Windows消息提供了應用程序與應用程序之間、應用程序與Windows系統之間進行通訊的於段。應用程序要實現的功能由消息來觸發,並靠對消息的響應和處理來完成。
Windows系統中有兩種消息隊列,一種是系統消息隊列,另一種是應用程序消息隊列。計算機的所有輸入設備由 Windows監控,當一個事件發生時,windows先將輸入的消息放入系統消息隊列中,然後再將輸入的消息拷貝到相應的應用程序隊列中”應用程序中的消息循環從它的消息隊列中檢索每一個消息並發送給相應的窗口函數中。一個事件的發生,到達處理它的窗口函數必須經歷上述過程。值得注意的是消息的非搶先性,即不論事件的急與緩,總是按到達的先後排隊(一些系統消息除外),這就使得一些外部實時事件可能得不到及時的處理。
三、windows的保護模式及中斷機制
1.Windows的保護模式
保護模式指的是線性地址由一個選擇符間接生成的,該選擇符指向描述表中的某一項;而實模式中則通過一個段/偏移量對來直接尋址。80386(486) CPU提儀的保護模式能力包括一個64K的虛擬地址空間和一個4G的段尺寸。Windows3.1實現時有所差別,它支持標准模式和增強模式。標准模式針對286機器,不周本文探討范圍。增強模式是對386以上CPU而言,windows正是使用保護模式來打破lM的屏障並且執行簡單的內存保護。它使用選擇器、描述器和描述器表控制訪問指定內存的位置和段。描述器表包括全局描述器表局部描述器表、中斷描述器表。保護模式與實模式有許多不同。其中顯著的差異是訪問內存的機制不同。
2.中斷機制
(1)實模式中斷
為了便於理解,我們先回顧實模式中斷。
在實模式下,中斷向量表IVT起到相當重要的作用。無論來自外部硬件的中斷或是內部的軟中斷INTn,在CPU中都產生同樣的響應。
①CPU將當前的指令指針寄存器(IP)、代碼段寄存器(CS)、標志寄存器壓入堆棧。
②然後CPU使用 n值作為指向中斷向量表IVT的索引,在IVT中找出服務例程的遠地址。
②CPU將此遠地垃裝入CS:IP寄存器中,並開始執行服務例程。
④中斷例程總以IRET指令結束。此指令使存在堆棧中的三個值彈出並填入CS、IP和標志寄存器,CPU繼續執行原來的指令。
(2)保護模式中斷
保護模式中斷過程與實模式中斷過程類似,但它不再使用中斷向量表IVT,而使用中斷描述符表(IDT)。值得一提的是,Windows運行時IVT還存在,應用程序並不使用它,Windows仍然使用,但含義已不同‘
(1)IVT結構:IVT在RAM的 0000:0000之上,占據開始的1024字節。它仍然由 BIOS啟動例程設置,由DOS填充到RAM中。
②IDT中斷描述符表:保護模式下,Windows操作系統為實現中斷機制而建立的一個特殊表,即中斷描述符表IDT。該表被用來保存中斷服務例程的線性地址,它們是真正的24位或32位地址,沒有段:偏移值結構。中斷描述器表最多可含有256個例
程說明,詳細說明請見[3]。
②當中斷或異常發生時,處理過程與實模式類丁當前的CS; IP值和標志寄存器值被存儲。保存的內容還包括CPU其他內部寄存器的值,以及目前正在被執行的任務的有關信息(若必須發生任務切換的話)。CPU設法獲取中斷向量後,以它為索引值查找IDT中的服務例程遠地址,接著將控制轉移到該處的服務例程。這是與實模式轉移到IVT的不同所在。保護模式使用IDTR寄存器分配和定位內存中的IDT中斷描述符表。
IDT在內存中是可移動的,與IVT固定在內存中剛好相反。 IDT中斷描述符表在 Windows中起決定性的作用。理解了windows保護模式的中斷機制。有助於我們理解中斷服務程序的設計,它的關鍵就在於如何將服務例程的地址放入IDT中斷描述符表中。當中斷發生時,如何將斷點地址及CPU各寄存器值保護起來,中斷結束時,如何將保護的值恢復。 windows系統本身並不提供實現上述功能的API,而DOS保護模式接口DPMI正具備了上述的功能。
下面我們首先介紹DPMI接口,然後基於它實現Windows下中斷服務程序的設計。
四、DOS保護模式接口
DMPI Windows除了標准服務外,還支持一組特殊的DOS服務,稱為DOS保護模式接口 DPMI,由一些INT2FH和INT31H服務組成。它使應用程序能夠訪問 PC系列計算機的擴充內存,同時維護系統的保護功能。 DPMI通過軟件中斷31h來定義了一個新的接口,使得保護模式的應用程序能夠用它作分配內存,修改描述符以及調用實模式軟件等工作。
Windows為應用程序提供 DPMI服務。即Windows是DPMI的宿主(host),應用程序是DPMI的客戶(client),可通過INT31H調用得到DPMI服務。INT 31H本身提供多功能。其中它的中斷管理服務允許保護模式用於攔截實模式中斷,並且掛住處理器異常。有些服務能夠和 DPMI宿主合作,以維護應用程序的虛擬中斷標志。
可以用INT31H來掛住保護模式中斷向量,以中斷方式處理外部實時事件。利用 INT 21H,功能0205H:設置保護模式中斷向量,將特定中斷的保護模式處理程序的地址置入中斷向量裡。調用方式:
AX=0205H,BL=中斷號,CX:(E)DX=中斷處理程序選擇符:偏移值。返回:執行成功CF=清零,執行失敗CF,置位。
掛住/解掛中斷向量的時機很重要。主窗口第一次被創建時會傳送它WM—CREATE消息,這時是掛住中斷向量的最好時機。退出時需解掛向量,否則Windows可能崩潰。上窗口接收到WM_DESTROY之後進行解掛工作,是最適合的。解掛向量可先用INT35H,0204H功能將老的中斷向量保存,退出時用INT35H,0205H恢復。
五、編程實現
有了DPMI的支持,我們就可以很方便地處理數據采集、串行通信等工業過程中的實時事件。下面以Windows3.1平台下中斷方式實現的串行通信為例,說明中斷程序的編制和實現。為便於參考,給出了詳細的代碼。開發平台BC3.1/BC4.5,其本身支持0.9版的DPMI,無需運行其它支持DPMI的軟件。編程語言C,可與C++混合編譯。
初始化COM1,9600波特率,每字符8bits,1個停止位,中斷接收,查詢發送。
//windows asy COmmunica60n
//by Li Xiumi98
//last modified on June25,1996
#include<windows.h>
#include<dos.h>
void interrupt far DataReceive() ;
void interrupt far( * old_vector)();
unsigned char dataCom_r[1024],datacom_s[1024]:
int inflag=0 ;
unsigned int s8259;
int InitCom1()
{
s8259=inportb(0x21);
outportb(0x21,s8259&0xe8);
outportb(0x3fb,0x83);
outportb(0x3f8,0x0c);
outportb(0x3f9,0x00);
outportb(0x3fb,0x03);
outportb(0x3fc,0x08);
outportb(0x3f9,0x01);
return 1;
}
void interrupt far DataReceive()
{
static int i=0 ;
char rechar =0 ;
rechar=inportb(0x3f8);
if(inflag==0)
{
if(rechar!='s'&&i==0)
{
i=0;
goto l1;
}
datacom_r[i++]=rechar;
if(rechar=='e')
{
inflag=1;
i=0;
}
}
l1:outportb(0x20,0x20);
}
void InitCom(void)
{
asm{
cli;
mov ax,204h
mov bl,0ch
int 31h
sti
}
old_vector=MK_FP(_CX,_DX);
asm{
cli
mov ax,205h
mov bl,0ch
mov cx,seg datareceive
mov dx,offset datareceive
int 31h
sti
}
InitCom();
}
void restore_Comm(void)
{
outportb(0x21,s8259);
asm{
cli
mov ax,205h
mov bl,0ch
mov cx,seg old_vector
mov dx,offset old_vector
int 31h
sti
}
}
在窗口第一次被創建時會傳送它WM_CREATE消息,這時調用initCom()即可。在主窗口關閉時,即主窗口中收到 WM_DESTROY消息時,調用Restore Comm()恢復原來的狀態。
這樣在對串口初始化,設置中斷服務例程後,通信事件發生時,會立即跳入中斷子程序中執行,越過系統的消息隊列,達到實時處理通信事件的目的。而數據處理模塊可通過全局標志f1,8訪問全局的數據通信緩沖區獲取實時數據。
這種實現方式與基於消息機制的Windows通信API實現相比具有實時性強的的特點,因為它超過了Windows 系統的兩極消息機制,上述程序已在實際系統中得到應用。在windows3.1支持下同時運行三個Windows任務,服務器SERVER(內有實時串行通信,多個網絡數據子服務,),客戶CLIENT,FOXPRO數據庫系統。整個系統運行良好。切換到WIN95平台下,系統也運行良好 。