|
阅读:6572回复:3
一個簡單的WDK範例
直接存取I/O埠與記憶體(Direct access I/O and Memory)
網路上有一個工具程式名為'WINIO”,透過它我們可以輕易的存取記憶體與I/O埠,但是它使用了一些已經過時或未公開的函數,因為不知微軟何時會將這些函數移除,所以使用這些函數會有一定的危險性,雖然到目前32位元的Windows 7版本中還可使用。 在此,我以WDK(Windows Driver Kit)的架構與相關函數,編寫一個用於存取I/O與記憶體的驅動程式,希望此程式能對您有所幫助。要重新編譯驅動程式或範例,請至微軟下載WDK 驅動程式與範例已在32位元的WinXP/Win7與64位元Win7上測試成功。因為在64位元的Win7作業系統上所有的驅動程式需經過微軟認證,否則無法執行,所以如果您要在64位元的系統上測試,請在開機時按下F8選擇停用驅動程式的強制簽章。 程式架構 描述:fig1 图片:arch.png 程式簡介 NTSTATUS
DriverEntry(IN OUT PDRIVER_OBJECT DriverObject,
IN PUNICODE_STRING RegistryPath)
{
NTSTATUS status;
WDF_DRIVER_CONFIG config;
WDFDRIVER hDriver;
PWDFDEVICE_INIT pInit = NULL;
//
#if DBG
DbgBreakPoint();
#endif
//初始化Driver Config結構,不設定任何回呼(CALLBACK)函數
WDF_DRIVER_CONFIG_INIT(&config, WDF_NO_EVENT_CALLBACK);
config.DriverInitFlags |= WdfDriverInitNonPnpDriver;
config.EvtDriverUnload = EvtDriverUnload;
//
//建立WDFDRIVER物件
//因為我們不需要清除(CLEANUP)或刪除回呼函數,所以不需要設定物件的屬性
status = WdfDriverCreate(DriverObject,
RegistryPath,
WDF_NO_OBJECT_ATTRIBUTES,
&config,
&hDriver);
if (!NT_SUCCESS(status))
{
KdPrint(("iomem: WdfDriverCreate failed with status 0x%x\n", status));
return status;
};
//
pInit = WdfControlDeviceInitAllocate(hDriver,
&SDDL_DEVOBJ_SYS_ALL_ADM_RWX_WORLD_RW_RES_R);
if (pInit == NULL)
{
status = STATUS_INSUFFICIENT_RESOURCES;
return status;
};
//
status = DeviceAdd(pInit);
return status;
}; 要在Windows NT中存取I/O埠,我們可以透過READ_PORT_UCHAR(USHORT/ULONG)或WRITE_PORT_UCHAR(USHORT/ULONG)等API來存取。這些API的使用方式與C語言提供的Runtime Library中的_outp、_inp等類似,所以不在詳加說明。 case 1: //UCHAR
if (pvInputBuffer->Command == 0) //判斷是寫出還是讀入
WRITE_PORT_UCHAR((PUCHAR)pvInputBuffer->PortBase, (UCHAR)pvInputBuffer->Data);
else
*(PUCHAR)pvOutputBuffer = READ_PORT_UCHAR((PUCHAR)pvInputBuffer->PortBase);
break;
case 2: // USHORT
if (pvInputBuffer->Command == 0)
WRITE_PORT_USHORT((PUSHORT)pvInputBuffer->PortBase, (USHORT)pvInputBuffer->Data);
else
*(PUSHORT)pvOutputBuffer = READ_PORT_USHORT((PUSHORT)pvInputBuffer->PortBase);
break;
case 4: //ULONG
if (pvInputBuffer->Command == 0)
WRITE_PORT_ULONG((PULONG)pvInputBuffer->PortBase, (ULONG)pvInputBuffer->Data);
else
*(PULONG)pvOutputBuffer = READ_PORT_ULONG((PULONG)pvInputBuffer->PortBase);
break; 要存取記憶體在程式的處理上比較繁瑣,無法像存取I/O埠一樣呼叫一個API就解決。 1. 要存取記憶體首先我們要先取得使用者所希望存取的位址與長度,這可以透過MmMapIoSpace這個API將指定的位址與長度對映(mapping)成非頁面(non-paged)的系統空間。 2. 如果步驟1執行成功,之後使用IoAllocateMdl與MmBuildMdlForNonPagedPool配置足夠的緩衝區以對映實體記憶體。 3. 使用MmMapLockedPagesSpecifyCache將對映的虛擬記憶體傳給呼叫者,讓處於RING 3的應用程式可以直接存取指定的記憶體。 //Step 1.
memReq.MapIo = MmMapIoSpace(Address, memReq.Length, MmNonCached);
if (memReq.MapIo == NULL)
{
status = STATUS_INSUFFICIENT_RESOURCES;
__leave;
};
// Step 2.
memReq.Mdl = IoAllocateMdl(memReq.MapIo, memReq.Length, FALSE, FALSE, NULL);
if (memReq.Mdl == NULL)
{
status = STATUS_INSUFFICIENT_RESOURCES;
__leave;
};
MmBuildMdlForNonPagedPool(memReq.Mdl);
// Step 3.
memReq.Buffer = (PUCHAR)MmMapLockedPagesSpecifyCache(memReq.Mdl,
UserMode, //此處要設定成UserMode
MmNonCached,
NULL,
FALSE,
NormalPagePriority);
if (memReq.Buffer == NULL)
{
status = STATUS_INSUFFICIENT_RESOURCES;
__leave;
}; IOMEM.DLL 這個使用者模式的動態函式庫提供outp、_outpw、_outpd、_inp、_inpw與_inpd等六個函數來存取I/O埠。MapMemory、UnmapMemory、ReadPhyicalMemory與WritePhyicalMemory四個函數來存取記憶體。 功能:設定存取記憶體的位址與長度 PUCHAR MapMemory(__in ULONG Address,__in ULONG Length); 參數說明: Address : 要存取的記憶體的起始位址。 Length : 所要存取的長度。 函數執行成功後傳回相映後可以使用的記憶體起始位址。 功能:解除先前所指定的記憶體 BOOLEAN UnmapMemory(VOID); 範例程式 這裡提供3個範例程式,分別是IO-MEM的存取範例,透過Winbond 83627xx Super I/O來存取系統的相關資訊。例如CPU溫度、冷卻風扇轉速與電壓等資訊。最後一個範例是列舉系統內PCI的資訊。 描述:fig2 图片:iomem.png 源代碼可至 http://bbs.codeheaven.com.tw/htm_data/2/1004/1.html 下載 |
|
|
|
沙发#
发布于:2010-04-27 20:34
牛!佩服。
|
|
|
|
板凳#
发布于:2010-05-03 23:33
谢谢,欢迎有时间到我的网站发帖,交流. http://bbs.codeheaven.com.tw
|
|
|
地板#
发布于:2011-04-01 14:24
附件咋下载不了了呢?
|
|
