ielts5316
驱动小牛
驱动小牛
  • 注册日期2005-05-18
  • 最后登录2018-08-27
  • 粉丝0
  • 关注0
  • 积分478分
  • 威望940点
  • 贡献值0点
  • 好评度72点
  • 原创分0分
  • 专家分0分
  • 社区居民
阅读:6126回复:3

一個簡單的WDK範例

楼主#
更多 发布于:2010-04-25 14:45
直接存取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

fig1
 

程式簡介
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埠。MapMemoryUnmapMemoryReadPhyicalMemoryWritePhyicalMemory四個函數來存取記憶體。

功能:設定存取記憶體的位址與長度
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

fig2
 
源代碼可至 http://bbs.codeheaven.com.tw/htm_data/2/1004/1.html 下載



附件名称/大小 下载次数 最后更新
SRC.zip (91KB)  23 2010-04-25 14:45
demo-x86.zip (1758KB)  20 2010-04-25 14:45
Pegram
论坛版主
论坛版主
  • 注册日期2005-12-03
  • 最后登录2013-08-23
  • 粉丝13
  • 关注5
  • 积分1333分
  • 威望4717点
  • 贡献值1点
  • 好评度78点
  • 原创分0分
  • 专家分2分
沙发#
发布于:2010-04-27 20:34
牛!佩服。
《寒江独钓》与《竹林蹊径》的合作作者。精通USB开发,设计了CY001 USB驱动套件(http://bbs.driverdevelop.com/read.php?tid-119314.html)。
ielts5316
驱动小牛
驱动小牛
  • 注册日期2005-05-18
  • 最后登录2018-08-27
  • 粉丝0
  • 关注0
  • 积分478分
  • 威望940点
  • 贡献值0点
  • 好评度72点
  • 原创分0分
  • 专家分0分
  • 社区居民
板凳#
发布于:2010-05-03 23:33
谢谢,欢迎有时间到我的网站发帖,交流. http://bbs.codeheaven.com.tw
xiaojj2005
驱动牛犊
驱动牛犊
  • 注册日期2006-04-21
  • 最后登录2011-05-05
  • 粉丝0
  • 关注0
  • 积分22分
  • 威望23点
  • 贡献值0点
  • 好评度2点
  • 原创分0分
  • 专家分0分
地板#
发布于:2011-04-01 14:24
附件咋下载不了了呢?
游客

返回顶部