基于GWES的WinCE Display驅(qū)動開發(fā)介紹
引言
本文引用地址:http://www.ex-cimer.com/article/148946.htm在WinCE中,Display驅(qū)動由GWES模塊來管理。WinCE提供了兩種架構(gòu)的Display驅(qū)動模型,可以滿足不同的硬件需求。一種是基于WinCE DDI的Display驅(qū)動模型,另一種是基于DirectDraw的Display驅(qū)動模型。下面將對兩種架構(gòu)作簡單介紹。
1 Display驅(qū)動模型
WinCE下的Display驅(qū)動直接由GWES模塊管理,它會直接被GWES模塊管理和調(diào)用。Display驅(qū)動實際上也是分層的,其中包括GPE庫,該庫處理一些默認(rèn)的繪圖,相當(dāng)于驅(qū)動的MDD層。用戶只需要開發(fā)和硬件相關(guān)的PDD層驅(qū)動就可以了。在WinCE中,整個架構(gòu)如圖1:
圖1
如圖,ApplicatiON為一個應(yīng)用程序,該程序會調(diào)用圖形設(shè)備接口函數(shù)(GDI),而GDI函數(shù)是由Coredll.dll模塊導(dǎo)出的。Coredll.dll會將函數(shù)調(diào)用的參數(shù)打包,然后觸發(fā)對另一個進(jìn)程的本地過程調(diào)用(LPC),所有的繪圖和開窗口的工作被傳給內(nèi)核中GWES模塊。GWES模塊被稱為圖形,窗口和事件子系統(tǒng),專門處理圖形輸出和用戶輸入等事件及相關(guān)的所有交互。GWES模塊會調(diào)用Display驅(qū)動完成對顯示硬件的操作。Display驅(qū)動由GPE和DDL.dll組成,GPE完成基本的默認(rèn)繪圖工作,而DDI.dll實際上從GPE類上繼承而來的,并實現(xiàn)了相關(guān)的顯示硬件的操作。
2 DirectDraw Display驅(qū)動模型
DirectDraw提供了獨立于硬件的直接訪問顯示設(shè)備的能力。它可以通過直接訪問硬件抽象層(HAL)中的一些函數(shù)來達(dá)到直接操作顯示設(shè)備的目的,在這個過程中,不再需要圖形設(shè)備接口(GDI)的轉(zhuǎn)換。這種直接的方法可以使圖像更加連貫,也提高了顯示的性能。為了實現(xiàn)這樣的功能,需要在顯示驅(qū)動上擴(kuò)展能夠直接訪問相關(guān)硬件的函數(shù)。這些函數(shù)會被DirectDraw模塊調(diào)用,并形成DirectDraw的硬件抽象層(DDHAL)。DirectDraw顯示驅(qū)動架構(gòu)如圖2:
如圖,DirectDraw的真正實現(xiàn)代碼都駐在gwes.dll模塊中,應(yīng)用程序只是連接了一個小的客戶端,被稱為DDRAW.dll代理,該代理主要負(fù)責(zé)用戶進(jìn)程與系統(tǒng)之間的遠(yuǎn)程DirectDraw COM接口連接。這樣,用戶請求會被傳送到內(nèi)核的GWES模塊中。針對DirectDraw,WinCE提供了一個名為DirectDraw的GPE庫(DDGPE),它是從GPE類上面繼承而來的。實際上,DirectDraw顯示驅(qū)動是由DDGPE和DDHAL組成,而DDGPE中已經(jīng)包含了DDHAL的功能。用戶需要從DDGPE類繼承并實現(xiàn)相關(guān)函數(shù)即可。GWES.dll模塊中包含GDI和DDRAW兩個組件,這兩個組件會調(diào)用驅(qū)動中的DDGPE的相關(guān)接口完成對硬件的操作。
在上述兩種架構(gòu)中,用戶可以根據(jù)自己的硬件情況選擇相應(yīng)的架構(gòu)。第一種架構(gòu)是基于GPE類繼承來實現(xiàn)的,第二種架構(gòu)是基于DDGPE類繼承來實現(xiàn)的,而第二種架構(gòu)的DDGPE類又是從第一種架構(gòu)的GPE類繼承而來。關(guān)于兩種類的具體定義,可參見” WINCE600PUBLICCOMMONOAKINC”路徑下的gpe.h和ddgpe.h文件。
本Blog將基于Display驅(qū)動模型來介紹,DirectDraw Display驅(qū)動模型不在這里介紹。
WinCE下的Display驅(qū)動是基于GPE類來實現(xiàn)的,其中GPE中已經(jīng)實現(xiàn)了基本的繪制工作,相當(dāng)于MDD層。用戶需要繼承該類,并實現(xiàn)里面的其他一些函數(shù),所以用戶實現(xiàn)的相當(dāng)于PDD層。
GPE類是一個抽象類,其中包含很多純虛函數(shù),只能用于繼承。用戶在繼承了GPE類以后,要對GPE類中的純虛函數(shù)做相應(yīng)的實現(xiàn)。開發(fā)Display驅(qū)動的大致步驟如下:
(1) 繼承GPE類并定義一個該類的實例。
(2) 實現(xiàn)GetGPE()函數(shù),把該類的實例返回給上層的DDI接口。
(3) 實現(xiàn)DrvEnabLEDriver()和DisplayInit()函數(shù)并導(dǎo)出這兩個接口。
(4) 實現(xiàn)GPE類中的函數(shù)。
下面將具體介紹實現(xiàn)的步驟:
2.1 繼承GPE類
首先,基于GPE類進(jìn)行繼承,如果想在Display驅(qū)動支持Rotation可以從GPERotate類上面繼承。實際上,在”gpe.h”中有如下定義:
typedef GPE GPERotate;
可以看出GPERotate類就是GPE類。在這里,用戶從GPE類上面繼承就可以了,舉個例子如下:
class NewGPE: public GPE
{
private:
GPEMode m_ModeInfo;
DWORD m_colorDepth;
DWORD m_VirtualFrameBuffer;
DWORD m_FrameBufferSize;
BOOL m_CursorDisabLED;
BOOL m_CursorVisible;
…
public:
NewGPE(void);
virtual INT NumModes(void);
virtual SCODE SetMode(INT modeId, HPALETTE *palette);
virtual INT InVBlank(void);
virtual SCODE SetPalette(conST PALETTEENTRY *source, USHORT firstEntry, USHORT numEntries);
virtual SCODE GetModeInfo(GPEMode *pMode, INT modeNumber);
virtual SCODE SetPointerShape(GPESurf *mask, GPESurf *colorSurface, INT xHot, INT yHot, INT cX, INT cY);
virtual SCODE MovePointer(INT xPosition, INT yPosition);
virtual void WaitForNotBusy(void);
virtual INT IsBusy(void);
virtual void GetPhysicalVideoMemory(unsigned long *physicalMemoryBase, unsigned long *videoMemorySize);
virtual SCODE AllocSurface(GPESurf **surface, INT width, INT height, EGPEFormat format, INT surfaceFlags);
virtual SCODE Line(GPELineParms *lineParameters, EGPEPhase phase);
virtual SCODE BltPrepare(GPEBltParms *blitParameters);
virtual SCODE BltComplete(GPEBltParms *blitParameters);
virtual ULONG GetGraphicsCaps();
virtual ULONG DrvEscape(
SURFOBJ *pso,
ULONG iEsc,
ULONG cjIn,
PVOID pvIn,
ULONG cjOut,
PVOID pvOut);
SCODE WrappedEmulatedLine (GPELineParms *lineParameters);
評論