程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> 更多編程語言 >> Delphi >> 用 Delphi 編寫 VxD 設備驅動程序(轉)

用 Delphi 編寫 VxD 設備驅動程序(轉)

編輯:Delphi

用 Delphi 編寫 VxD 設備驅動程序(轉)

關鍵詞:Delphi控件雜項

作者:Emil Biserov([email protected])(Russion)
英語翻譯:Vitaly Zayko([email protected])
中文翻譯改編:Riceball([email protected])

前言
  用 Delphi 3.0 編寫 VxD 設備驅動程序,在Delphi 3 下編譯通過,Delphi 2 下沒有測試,Delphi 4 建立的 Object 文件 M$ Linker 5.12.8181 不能識別,這裡使用的匯編器是M$的Macro Assembler ver. 6.11d ,聯結器是M$ Incremental Linker ver. 5.12.8181 ,它們來自 Windows 98DDK(http://www.microsoft.com/ddk/ddk98.htm)。

介紹
  Windows 存在有兩種類型的 VxD 設備驅動程序:
    1、靜態(Static) VxD ,裝入操作系統並永久的存在於內存中;
    2、動態(Dynamic) VxD,當需要時才調入內存,用完後關閉VxD即可釋放內存。
  Inprise Delphi 有能力建立任何一種類型的 VxD 設備驅動程序,下面我們將介紹如何建立動態 VxD。
  當 Win32 應用程序打開一個 VxD “虛擬”設備時,VWIN32 使用 LoadDevice 將 VxD 裝入內存,並建立消息W32_DEVICEIOCONTROL ,發向 VxD。
  也就是說,VxD 至少應該響應以下兩個系統信息和編寫以下的一個函數:
    SYS_DYNAMIC_DEVICE_INIT
    SYS_DYNAMIC_DEVICE_EXIT
    W32_DEVICEIOCONTROL 函數.
  消息 SYS_DYNAMIC_DEVICE_INIT 在嘗試裝入 VxD 時發送到 VxD ,消息 SYS_DYNAMIC_DEVICE_EXIT 在嘗試動態交換時發送到 VxD ,消息的處理者在成功處理後,應該在寄存器 AX 中返回 VXD_SUCCESS 標志。

  W32_DEVICEIOCONTROL 的 dwService 參數有以下的值:
    DIOC_OPEN 當 VxD 通過 CreateFile() 函數嘗試打開操作時發送(在 SYS_DYNAMIC_DEVICE_INIT 消息後),如果成功返回 NO_ERROR (0); 
    DIOC_CLOSEHANDLE 當 VxD 通過 CloseHandle() 函數嘗試關閉操作時發送(在 SYS_DYNAMIC_DEVICE_EXIT 前)
    所有其它的值 > 0 意味著不同的函數調用(由 dwIoControlCode 給出),當 VxD 被 DeviceIoControl 函數調用時。

啟動模塊(vxdmain.asm)
...
extrn SysDynamicDeviceInit :PROC
extrn SysDynamicDeviceExit :PROC
extrn W32DeviceIoControl  :PROC
...
                        PUBLIC  DELPHIIO_DDB
            Public  @@HandleFinally
            Public  @initialization
...
Control_0    proc
    cmp    eax, SYS_DYNAMIC_DEVICE_INIT
    jnz    short chkSysDynExit
    call    SysDynamicDeviceInit
    cmp    eax, 1
    retn   
;-------------

chkSysDynExit:
    cmp    eax, SYS_DYNAMIC_DEVICE_EXIT
    jnz    short chkDevIOCtl
    call    SysDynamicDeviceExit
    cmp    eax, 1
    retn   
;-------------
chkDevIOCtl:
    cmp    eax, W32_DEVICEIOCONTROL
    jnz    short loc_ret
    push    esi
    push    edx
    push    ebx
    push    ecx
    call    W32DeviceIoControl
    cmp    eax, 1
    retn   
;-------------
loc_ret:
    clc   
    retn   

Control_0    endp

@@HandleFinally:
@initialization:
            ret

_LTEXT  ends
                        END

  Delphi 會為單元的 initialization/finalization 建立代碼調用外部過程 HandleFinaly 和 initialization ,即使 initialization/finalization 在單元中不存在。因此我們在匯編的啟動文件中建立空的外部過程入口。

主 Delphi 程序單元(vxdProcs.pas) 
...
procedure ShellMessage(Handle, Flags : integer; const Message, Caption : PChar; 
  Callback, ReferenceData : pointer); stdcall; assembler;
asm
  mov    ebx, Handle        // virtual machine handle
  mov    eax, Flags        // message box flags
  mov    ecx, Message        // address of message text
  mov    edi, Caption        // address of caption text
  mov    esi, Callback        // address of callback
  mov    edx, ReferenceData        // reference data for callback

  int    20H            // VxDCall
  dd    170004h            // Shell_Message
end;

function SysDynamicDeviceInit : INTEGER;
begin
  ShellMessage(0, $10, Copyright, SysDynInit: Hello from Delphi VxD !!!, nil, nil);
  Result := VXD_SUCCESS;
end;

function SysDynamicDeviceExit : INTEGER;
begin
  ShellMessage(0, $10, Copyright, SysDynDevExit: Bye from Delphi VxD !!!, nil, nil);
  Result := VXD_SUCCESS;
end;

function W32DeviceIoControl(dwService : INTEGER;
                            dwDDB : INTEGER;
                            hDevice : INTEGER;
                            lpDIOCParms : pointer) : INTEGER;
begin
  ShellMessage(0, $10, Copyright, W32DevIOCtl, nil, nil);

  if (dwService = DIOC_OPEN) then
  begin
      Result := NO_ERROR;
  end
  else if (dwService = DIOC_CLOSEHANDLE) then
  begin
      Result := VXD_SUCCESS;
  end
  else if (dwService > MAX_PASVXD_W32_API) then
  begin
      Result := ERROR_NOT_SUPPORTED;
  end
    else
  begin
      Result := VXD_SUCCESS; 
  end;
end;
...


[譯者:好了,簡單的 VxD 設備驅動程序編寫完畢了。你可以將它當作一個寫 VxD 設備驅動程序的模板。]

附一:Make.bat
D:VISUAL~198DDKBINWin98ml -coff -DBLD_COFF -DIS_32 -W2 -c -Cx -Zm -DMASM6 vxdmain.asm
call dcc3.bat -J vxdprocs.pas
D:VISUAL~198DDKBINlink /DEF:vxddef.def /VXD vxdmain.obj vxdprocs /OUT:delphiio.vxd

附二:
現在讓我們來編寫對該 VxD 的測試程序,兩個按鈕:一個打開 VxD;一個關閉 VxD。

const
VxDName = \.DELPHIIO.VXD;

...

function TVxDTestForm.OpenVxDDriver: boolean;
begin
HVxDHandle := CreateFile(VxDName,0,0,nil,0,FILE_FLAG_DELETE_ON_CLOSE,0);
Result := HVxDHandle <> INVALID_HANDLE_VALUE;
end;

procedure TVxDTestForm.CloseVxDDriver;
begin
if HVxDHandle <> INVALID_HANDLE_VALUE then begin
  CloseHandle(HVxDHandle);
  HVxDHandle := INVALID_HANDLE_VALUE;
end;
end

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved