• 欢迎访问搞代码网站,推荐使用最新版火狐浏览器和Chrome浏览器访问本网站!
  • 如果您觉得本站非常有看点,那么赶紧使用Ctrl+D 收藏搞代码吧

用Delphi编写VxD设备驱动程序_Delphi

Delphi 搞代码 3年前 (2018-08-09) 151次浏览 已收录 0个评论

前言  
 用 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 至少应该响应以下两个系统信息和编写以下的一个函数:  

http://www.gaodaima.com/?p=65941用Delphi编写VxD设备驱动程序_Delphi

  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  
[email protected]  
…  
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~1/98ddk/bin/win98/ml -coff -dbld_coff -dis_32 -w2 -c -cx -zm -dmasm6 vxdmain.asm  
call dcc3.bat -j vxdprocs.pas  
d:/visual~1/98ddk/bin/link /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  

顺便说一下,delphi中有个编译选项可以控制程序加载的入口

一般是0x00400000,你可以改.

欢迎大家阅读《用Delphi编写VxD设备驱动程序_Delphi》,跪求各位点评,若觉得好的话请收藏本文,by 搞代码


喜欢 (0)
[搞代码]
分享 (0)
发表我的评论
取消评论

表情 贴图 加粗 删除线 居中 斜体 签到

Hi,您需要填写昵称和邮箱!

  • 昵称 (必填)
  • 邮箱 (必填)
  • 网址