用 Delphi 编写 VxD 设备驱动程序(转)

<script type="text/javascript"> </script> <script type="text/javascript" src="http://pagead2.googlesyndication.com/pagead/show_ads.js"> </script>
<script type="text/javascript"> </script><script type="text/javascript" src="http://pagead2.googlesyndication.com/pagead/show_ads.js"> </script>
Delphi 编写 VxD 设备驱动程序(转)

关键词: Delphi控件杂项

作者:Emil Biserov(dinfo@mail.primorye.ru)(Russion)
英语翻译:Vitaly Zayko(zayko@vitsoftware.com)
中文翻译改编:Riceball(teditor@mailroom.com)

前言
  用 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_PAS VxD_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 T VxDTestForm.Open VxDDriver: boolean;
begin
H VxDHandle := CreateFile( VxDName,0,0,nil,0,FILE_FLAG_DELETE_ON_CLOSE,0);
Result := H VxDHandle <> INVALID_HANDLE_VALUE;
end;

procedure T VxDTestForm.Close VxDDriver;
begin
if H VxDHandle <> INVALID_HANDLE_VALUE then begin
  CloseHandle(H VxDHandle);
  H VxDHandle := INVALID_HANDLE_VALUE;
end;
end

<script type="text/javascript"> </script> <script type="text/javascript" src="http://pagead2.googlesyndication.com/pagead/show_ads.js"> </script>
<script type="text/javascript"> </script><script type="text/javascript" src="http://pagead2.googlesyndication.com/pagead/show_ads.js"> </script>
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值