用Delphi编写VxD设备驱动程序

王朝delphi·作者佚名  2006-01-08
窄屏简体版  字體: |||超大  

用Delphi编写VxD设备驱动程序

介绍

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

...

publicdelphiio_ddb

public@@handlefinally

public@initialization

...

control_0proc

cmpeax, sys_dynamic_device_init

jnzshort chksysdynexit

callsysdynamicdeviceinit

cmpeax, 1

retn

;-------------

chksysdynexit:

cmpeax, sys_dynamic_device_exit

jnzshort chkdevioctl

callsysdynamicdeviceexit

cmpeax, 1

retn

;-------------

chkdevioctl:

cmpeax, w32_deviceiocontrol

jnzshort loc_ret

pushesi

pushedx

pushebx

pushecx

callw32deviceiocontrol

cmpeax, 1

retn

;-------------

loc_ret:

clc

retn

control_0endp

@@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

movebx, handle// virtual machine handle

moveax, flags// message box flags

movecx, message// address of message text

movedi, caption// address of caption text

movesi, callback// address of callback

movedx, referencedata// reference data for callback

int20h// 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,你可以改。

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
 
 
© 2005- 王朝網路 版權所有 導航