|
6 s) _) N9 U* E! D5 @
! ~* i7 M U3 p, f, C1 X摘 要: 本文介绍了DLL技术在Windows编程中的基本运用方法及应用,给出了直
( m0 g+ J$ X$ o接内存 9 |& e3 V# S/ Y: k
访问及端口I/O的两个实用DLL的全部源代码。
( A' P. ^) \4 }/ D6 f6 m; ~. t% a关键词: DLL Windows编程 内存访问 I/O
) X5 f* k/ q2 O. S! [# |
, Q# }; x5 w/ \) M4 i. V一 、引 言 1 c- Y3 J& Y0 K: l; q" B
由于Windows为微机提供了前所未有的标准用户界面、图形处理能力和简单灵便 " j( c& U0 I8 t1 b
的操作,绝大多数程序编制人员都已转向或正在转向Windows编程。在许多用户 1 O- s1 b1 i4 _3 E
设计的实际应用系统的编程任务中,常常要实现软件对硬件资源和内存资源的访
W7 c" D; s1 z$ R9 q问,例如端口I/O、DMA、中断、直接内存访问等等 。若是编制DOS程序,这是轻
0 ~# N# l' O: v9 X而易举的事情,但要是编制Windows程序,尤其是WindowsNT环境下的程序,就会 9 |9 m& x* }( [( `- h! I
显得较困难。 : e" `2 x0 w& t+ I1 V. |% r+ I
因为Windows具有"与设备无关"的特性,不提倡与机器底层的东西打交道,如果 % |( o/ S& h* d, p8 D1 K
直接用Windows的 API函数或I/O读写指令进行访问和操作,程序运行时往往就会
- _8 ?) U/ I; m" U, @产生保护模式错误甚至死机,更严重的情况会导致系统崩溃。那么在Windows下
4 S! T2 e8 c- V; Y+ F怎样方便地解决上述问题呢?用DLL(Dynamic Link Libraries)技术就是良好途
$ o- O1 B4 W) m7 K# Z- A, Z3 @7 k5 p径之一。 ( }# `; z* a+ z
DLL是Windows最重要的组成要素,Windows中的许多新功能、新特性都是通过
2 i# l; X0 a( S4 q. PDLL来实现的,因此掌握它、应用它是非常重要的。其实Windows本身就是由许多 * r6 \% h6 t7 i x$ B
的DLL组成的,它最基本的三大组成模块Kernel、GDI和User 都是DLL,它所有的
4 H% a; v, G2 x {3 p% w6 l库模块也都设计成DLL。凡是以.DLL、.DRV、.FON、.SYS和许多以.EXE为扩展名 7 I! [) s$ e# z
的系统文件都是DLL,要是打开Windows\System目录,就可以看到许多的DLL模块 . n: K! o" i9 d/ Z, f% V
。尽管DLL在Ring3优先级下运行,仍是实现硬件接口的简便途径。DLL可以有自
2 {: F! u1 P& p+ E' C+ n己的数据段,但没有自己的堆栈,使用与调用它的应用程序相同的堆栈模式,减 v* [ | ?+ [5 y; C, u. ?
少了编程设计上的不便;同时,一个DLL在内存中只有一个实例,使之能高效经
0 T0 B5 g* H1 w1 i* U济地使用内存;DLL实现的代码封装性,使得程序简洁明晰;此外还有一个最大
3 Z8 b G& _+ t6 ?. s的特点,即DLL的编制与具体的编程语言及编译器无关,只要遵守DLL的开发规范
& T7 ~& x: L6 k和编程策略,并安排正确的调用接口,不管用何种编程语言编制的DLL都具有通 7 M2 a M/ C8 h' S3 a
用性。例如在BC31中编制的DLL程序,可用于BC、VC、VB、Delphi等多种语言环 : t$ O% _8 \' M
境中。笔者在BC31环境下编译了Windows下直接内存访问和端口I/O两个DLL,用
' E) T8 f+ @8 e' |! ]* V: V3 I! y1 ?7 O在多个自制系统的应用软件中,
8 p* a3 }2 |& [ E+ |运行良好。 2 t: `; m' s f O/ f$ U5 o
' D3 \( x* c' H$ c @* _) \: f( U二、DLL的建立和调用 3 K% W& p P! E" X
DLL的建立及调用方法在许多资料上有详细的介绍,为了节省篇幅,在这里仅作 9 k% C4 x/ D; M3 _5 r
一些 ; `, b9 G& Q' Q
主要的概括。
% Z( T+ i* x, U% |3 Y1.DLL的建立
4 z; H' m8 {3 E* E: e. s关于DLL的建立,有如下几个方面的要素是不可缺少和必须掌握的:
6 T* f; {0 }9 f: H5 f' \?. 入口函数LibMain( ) 8 U' h/ g* A. s
就象C程序中的WinMain( )一样,Windows每次加载DLL时都要执行LibMain( )函
" t/ L* {. [5 q/ X3 r数,主要用来进行一些初始化工作。通常的形式是:
: `: _* k9 X, X5 k% }5 w- h7 D* q2 a 3 N+ F1 @' w" z( ~; L
; ~6 \3 G1 ~. d9 l
, |" [) |( P; aint FAR PASCAL LibMain(HINSTANCE hInstance,WORD wDataSeg,WORD 4 M9 V2 Y2 H* S" x4 ?: v" ]
wHeapSize,LPSTR lpszCmdLine)
5 U4 M! M9 o$ D! a9 v2 C% R7 m{
" z M% U* k# @* {# {0 L: I0 vif(wHeapSize!=0) //使局部堆、数据段可移动
' Q2 Y' }0 V. G3 {/ wUnlockData(0); //解锁数据段 3 f2 z& Y" q0 T! F' I, A
/*此处可进行一些用户必要的初始化工作*/
# f: Y) K0 H( Q! o8 R% Breturn 1; //初始化成功 3 v" Z% b- F5 Z3 ]" r s( p4 B
} . I* Q, k+ E6 P
?出口函数WEP( ) 6 z1 V& E% N8 b+ S5 J
Windows从内存中卸载DLL时,调用相应的出口函数WEP( ),主要做一些清理工作
' H) Q; F8 k' O9 V6 S, g. N,如释放占用的内存资源;丢弃某些字串、位图等资源;关闭打开的文件等等。 $ w I+ A, }; }/ D5 M9 [6 M p
) e* a8 l$ N/ D) O8 u* R2 N
?自定义的输出函数
+ M$ [+ R. U0 ~- H, Q$ M% S为了让位于不同内存段的应用程序进行远程调用,自定义的输出函数必须定义为 + S. D7 a$ n) @6 R- m( I
远程函数(使用FAR关键字),以防使用近程指针而得到意外的结果;同时,加上
7 @# O* `9 h3 {3 ~6 Q) W, j+ OPASCAL关键字可加快程序的运行速度,使代码简单高效,提高程序的运行速度。 ' y c; r( G; c& m; w" X0 n P
n7 ~5 |; I1 h2 i. N8 q?输出函数的引出方法 0 F& _# R% c% m5 K/ G, O' W
? 在DLL的模块定义文件中(.DEF)由EXPORTS语句对输出函数逐一列出。例如:
0 }. }3 a" t. s$ n% VEXPORTS WEP @1 residentname //residentname可提高DLL效率和处理速度
- e: B* }( S# [4 APortIn @2 % h! D2 e+ d" U
PortOut @3 //通常对所有输出函数附加系列号 , s6 S* T1 ], ?: x
? 在每个输出函数定义的说明中使用_export关键字来对其引出。
' q+ v: e: Z3 t4 N+ d1 L w以上两种方法任选其中的一种即可,不可重复。后面的两个实例分别使用了上述 - y5 [7 b$ ^, m0 i1 W4 K
两种不同的引出方式,请留意。
0 J: J' X, G2 q; }1 i7 g " \9 m7 y9 N2 e# Q$ o" K
2.DLL的调用 ' t: i) p. e3 J' A! g3 R
加载DLL时,Windows寻找相应DLL的次序如下: / V9 F8 I$ |# @0 O
?.当前工作盘。
2 i# H [5 p! V6 \& k) Z; ]: A3 E- s?Windows目录;GetWindowsDirectory( )函数可提供该目录的路径名。
( ?6 y9 ?4 F" d! W5 K7 \3 ??Windows系统目录,即System子目录;调用GetSystemDiretory( )函数可获得这
g( Y0 P0 b" t1 |4 W7 u6 ]个目录的路径名。 9 O" M1 l2 I/ K# Y4 W' f
?DOS的PATH命令中罗列的所有目录。
}4 N; b" y1 W+ e?网络中映象的目录列表中的全部目录。 $ {/ @9 Q5 Y% {6 A9 L! |
# A# x9 M S4 Y- I7 W8 nDLL模块中输出函数的调用方法:
- k! H9 a# Y: R! a不论使用何种语言对编译好的DLL进行调用时,基本上都有两种调用方式,即静
- B# b ~5 ^3 s& w3 [( X. E% z态调用方式和动态调用方式。静态调用方式由编译系统完成对DLL的加载和应用 7 R$ Y/ f* g% L- g+ `* w* M
程序结束时DLL卸载的编码(如还有其它程序使用该DLL,则Windows对DLL的应用
3 _8 L" S2 u* Y记录减1,直到所有相关程序都结束对该DLL的使用时才释放它),简单实用,但
0 \! q8 Z$ o+ V# q8 o0 \8 [不够灵活,只能满足一般要求。动态调用方式是由编程者用API函数加载和卸载
( Z+ H+ j% C7 E1 t- uDLL来达到调用DLL的目的,使用上较复杂,但能更加有效地使用内存,是编制大 ) ^) R8 O- A/ M, ^
型应用程序时的重要方式。具体来说,可用如下的方法调用:
2 f. I" l2 I) R9 A, N) j2 X0 F- U?.在应用程序模块定义文件中,用IMPORTS语句列出所要调用DLL的函数名。如: ( k' e5 g8 S- D$ |
/ o4 I1 y$ ?5 A( g2 |8 ] h# v
IMPORTS MEMORYDLL.MemoryRead
2 h5 \$ y9 Q0 `. WMEMORYDLL.MemoryWrite 3 [/ M' D$ A; D3 G( P' F+ N6 t
?让应用程序运行时与DLL模块动态链接
: i7 ?0 p8 s% ~! }
# c% ~0 q3 w m* E( s
2 J; T% b& y0 P) e$ R5 |# U
+ ]" G9 u6 S. R' }: \! h先用LoadLibrary加载DLL,再用GetProcAddress函数检取其输出函数的地址,获
i4 [7 w; Q: I; b# J得其指针来调用。如: " M" S% l4 Y' D" P
HANDLE hLibrary; 4 \% _; l$ u0 \4 T; x$ }+ q& j u
FARPROC lpFunc; 7 z- Z# T$ J' r
int PortValue; ' C9 Z! O$ Z/ H
M ( o& D$ }8 y/ r
hLibrary=LoadLibrary("PORTDLL.DLL"); //加载DLL 9 k7 j, F, w) g# n9 E2 W
if(hLibrary>31) //加载成功
. ]" ]% O' K9 G* [8 h{
# ]3 V' u; {" d1 k9 GlpFunc=GetProcAddress(hLibrary,"PortIn"); //检取PortIn函数地址
; E, ]- M2 E* m4 G( u4 Oif(lpFunc!=(FARPROC)NULL) //检取成功则调用 7 O' h, H; b& G9 v
PortValue=(*lpFunc)(port); //读port端口的值 ' _, `3 h* `! J5 d: U
FreeLibrary(hLibrary); //释放占用的内存 , @6 B, d6 m2 \
}
0 X: W. Z* k* J6 |, O$ r2 v0 Y( SM - x. U$ S+ h# J8 ]0 }+ k# O
* E4 V% T4 g G/ E& Z1 j ^+ O三、DLL应用实例源程序
1 \" k: c) X- p. R( \. ^+ a8 b1.直接内存访问的DLL源代码 % |# T# ^! x+ j9 R+ n- G |1 s
//.DEF文件 $ r4 C. c6 Y# ^
LIBRARY MEMORYDLL
+ P3 M$ M2 u0 {8 n1 }7 SDESCRIPTION 'DLL FOR MEMORY_READ_WRITE '
7 p) `% `3 {- N3 _# G! TEXETYPE WINDOWS
9 ]* ~( u) S$ v$ q+ x( ACODE PRELOAD MOVEABLE DISCARDABLE - H7 \1 B) y2 [9 F9 [. k4 |
DATA PRELOAD MOVEABLE SINGLE 4 E3 f3 h. l) @
HEAPSIZE 1024 //DLL无自己的堆栈,故没有STACKSIZE语句 . @5 l: I+ }: \* v8 z+ \
EXPORTS WEP @1 residentname
) v# j9 y5 D4 Q5 c# Q" N. oReadMemory @2
8 h* Y. I L) L0 R. a: d! q) U& wWriteMemory @3 ) F& q- f" W3 I( f0 P3 g }0 J) }
' S" d8 v7 X) U1 F' ?
//.CPP文件
$ B# N, C: t; f4 V1 ^#include <windows.h> & [4 ~8 C; S4 _* p0 ]% q3 {
|1 T1 u# ]4 d p2 ?3 T
int FAR PASCAL LibMain(HINSTANCE hInstance,WORD wDataSeg,WORD D z* t8 I4 J" }
wHeapSize,LPSTR lpszCmdLine)
0 x/ G9 C' L3 l [5 x& V{ # A, B4 K9 Y6 y% |5 P U
if(wHeapSize!=0) ' I# |2 z t# Y5 J- i- h/ s
UnlockData(0);
( `" v7 K! J' ^% ]# ]: `# freturn 1; ( e% o3 q) X+ e ~: o* Q; t
} ( g/ B5 u6 t4 {8 g
" R/ t2 S% v; [9 L. xint FAR PASCAL MemoryRead(unsigned int DosSeg,unsigned int # V9 O' n9 S0 \- Y- S, ] M5 F
DosOffset) / }8 g; m5 d. T4 z" z
{
' o* o; j1 [( C: E. r% N4 t# @3 FWORD wDataSelector,wSelector;
( }+ L) |% G+ K. P0 Z- ychar far *pData;
3 f% ^ Y9 P% G, ochar value; + j3 J- s" p$ J
wDataSelector=HIWORD((DWORD)(WORD FAR *)&wDataSelector); - r' t1 r2 d1 e$ A/ |- _
wSelector=AllocSelector(wDataSelector); //分配选择器
! F p% h+ A$ \! CSetSelectorLimit(wSelector,0x2000); //置存取界限
* }9 ^$ ]& B6 I1 y$ J" N; hSetSelectorBase(wSelector,(((DWORD)DosSeg)<<4)+(DWORD)DosOffset); //置
" s# Z6 N" A! ]. l: M& {& I2 L基地址
0 Y c, D# x# A& y/ B6 I# ypData=(char far *)((DWORD)wSelector<<16);
6 L( I ]: D; X! j9 Qvalue=*pData; " _8 K' [) z! z2 e8 R0 r# p' t4 B
FreeSelector(wSelector); //释放选择器
! |" d. v5 p5 c) H! ~return (value); 8 W% m" ~; v" s; J) O- Y7 y
}
7 a% ~( p* ?8 d0 o: b
8 C9 n$ w1 l U
( ?+ I1 M; V% C* w: a: o/ ?
& {9 _7 g9 r. a, ^8 o/ J) Yvoid FAR PASCAL MemoryWrite(unsigned int DosSeg,unsigned int 8 c( x) c9 r/ I. I, a
DosOffset,char Data)
* n }' B7 k7 x2 q% `. G{
' z' n7 ?, d b. l; eWORD wDataSelector,wSelector; 7 V, ]- F+ B# f4 L) P2 F" W
char far *pData;
7 _; Z0 `: q I. k0 `, Q6 b! HwDataSelector=HIWORD((DWORD)(WORD FAR *)&wDataSelector);
5 a. T, {# a" V owSelector=AllocSelector(wDataSelector); ' N+ P& f% t- Y
SetSelectorLimit(wSelector,0x2000);
) e- n" k' T7 HSetSelectorBase(wSelector,(((DWORD)DosSeg)<<4)+(DWORD)DosOffset);
, L# U: k/ L. r6 p3 f0 r) TpData=(char far *)((DWORD)wSelector<<16); ) ~3 y/ I2 E- C$ D) D: D5 y* f
*pData=Data;
" f2 [4 L# I# z% |5 mFreeSelector(wSelector); & a9 y% Y x, x) { o7 c- F
}
* n, y% Y4 f8 z& t 9 j* c+ S0 l# P: c
int FAR PASCAL WEP(int nParam) 2 Z/ N9 O; ]" a: F. |' u
{
" \, w' l. N! Z4 s5 _" ^return 1; & ~; e, d& w: B$ V6 b
}
/ W& S B$ |% J7 C! v + {- l; P; }1 X* M4 _* W3 Y
2.端口读写I/O的DLL源代码
6 k" n8 A+ q: V) V//.DEF文件
, o# H- e: ] v& ^) LLIBRARY PORTDLL
: E% g+ t6 D2 IDESCRIPTION 'DLL FOR PORT_IN_OUT '
. [ M2 g1 Z: F S0 H6 G% XEXETYPE WINDOWS 0 w5 H: V5 e. O3 o k9 p# ]7 [
CODE PRELOAD MOVEABLE DISCARDABLE 9 Z" k1 e9 w; i7 j* Z# Y6 _
DATA PRELOAD MOVEABLE SINGLE
' g/ i8 L5 E/ @' b/ n& eHEAPSIZE 1024
, s# N/ b: f8 l/ O/ [1 R( Y & V ?# \; o' _ U& I5 l
//.CPP文件
* Q8 N: x$ q4 Z/ h" o#include <windows.h>
9 d, R* U( p" w _#include <dos.h> 1 n; u4 F: V, l) @% n& D/ K
: s" k4 ]6 M# Y' x- [int FAR PASCAL LibMain(HINSTANCE hInstance,WORD wDataSeg,WORD
8 ?. T7 _* O+ X2 m9 AwHeapSize,LPSTR lpszCmdLine)
+ L$ M1 S$ O4 u) d- y9 v{
! H' e) s6 K2 g% r0 Y. s; `7 pif(wHeapSize!=0) , s: K" G! M6 g! r1 ]1 G& ~# V; c& i
UnlockData(0);
2 `. l, i' [; p7 t; {3 \return 1; 8 I1 q6 L M6 T1 o: l, o M) X
} 9 o+ ]) e& L5 ]' F
$ P) @6 T! f: z u: p7 b6 hint FAR PASCAL _export PortOut(int port,unsigned char value)
( ~0 M# h2 C6 s1 g{ 8 t+ Z& X# v* @5 b1 C' N6 P
outp(port,value); 1 @; M# i8 [1 u6 v0 ?
return 1;
2 s& ?9 W9 V, o: Y m} : N- w1 U: h C
: J: a# m, e' `- r8 q6 Z; u
int FAR PASCAL _export PortIn(int port) $ l u' f# m) m+ R, g" J7 u
{ 1 }7 ^+ x( U; o
int result;
) W1 ^5 `5 u/ B* @. dresult=inp(port);
9 w# e" m: ^& Oreturn (result);
& o" J z/ I& n; A' I' w2 W}
}$ I- [% [4 p( S; U
# {0 Q9 ^! d. eint FAR PASCAL _export WEP(int nParam) # B4 ]. X4 d% ? F
{
8 d" w4 p, T) i4 \/ [% A& n E$ o( ^return 1;
# G/ x# R6 X! j7 ^} , _" k/ N1 k( N X0 n. L. A" J s
{
U* o/ e5 t; g" ?- l' p+ k9 O% R, jreturn 1; 7 P- H0 k* ]' c4 R6 [/ L: |2 j+ E
} " G$ I8 d- u0 I7 Y
, a+ ]6 }. A ~# A. c3 D ) @, S$ F. w% f# O5 a. r
分别将上面两个实例的.DEF文件和.CPP文件各自组成一个.PRJ文件,并进行编译
- Q+ Y8 X- A2 K2 ?6 M8 m链接成.EXE或.DLL文件就可以在应用程序中对其进行调用。
5 v+ V& q }. s# L/ {
* J T3 r+ l0 i* c- G0 {四、结 束 语
4 v0 L* q6 O7 r6 _6 w在上面,我们利用DLL技术方便地实现了Windows环境下对内存的直接访问和端口
- }. z& T1 @( n9 R8 Y/ tI/O的访问,仿效这两个例子,还可以编制出更多的适合自己应用系统所需的 # U7 C9 e3 Z; `; ~; d- A
DLL,如用于数据采集卡的端口操作及扩展内存区访问、视频区缓冲区及BIOS数
. Y4 D2 L: f! K0 R2 ~3 E1 [# B据区操作等许多实际应用的编程任务中。必要时只需直接更新DLL,而用不着对
; e) j/ e3 a% S" ]1 ]应用程序本身作任何改动就可以对应用程序的功能和用户接口作较大的改善,实
: @5 k0 E( v1 \+ s0 J9 x1 W( i现版本升级。因此,掌握好DLL技术对Windows程序开发者很有裨益。 |