四、
在驱动中使用链表
4.1 内存的分配与释放
传统的C语言中,分配内存常常使用的函数是malloc,但在驱动开发过程中这个函数不再有效。驱动中分配内存,最常用的是调用ExAllocatePoolWithTag或ExAllocatePool。

// 定义一个内存分配标记 

 #defineMEM_TAG ‘MyTt’ 

 // 目标字符串,接下来它需要分配空间。 

 UNICODE_STRINGdst={0}; 

 // 分配空间给目标字符串。根据源字符串的长度。 

 Windows 下设备驱动程序的开发方法 2120080411 计算机应用 赖锡盛 

 11 

 dst.Buffer=(PWCHAR)ExAllocatePoolWithTag(NonpagedPool,src->Length,MEM_TAG); 

 if(dst.Buffer==NULL) 

 { 

 // 错误处理 

 status=STATUS_INSUFFICIENT_RESOUCRES; 

 …… 

 } 

 dst.Length=dst.MaximumLength=src->Length;


ExAllocatePoolWithTag的第一个参数NonpagedPool表明分配的内存是非分页内存,这样它们可以永远存在于物理内存,而不会被分页交换到硬盘上去;第二个参数是长度;第三个参数是一个所谓的“内存分配标记”。
内存分配标记用于检测内存泄漏。想象一下,我们根据占用越来越多的内存的分配标记,就能大概知道泄漏的来源。一般每个驱动程序定义一个自己的内存标记。也可以在每个模块中定义单独的内存标记。内存标记是随意的32位数字。即使冲突也不会有什么问题。
此外也可以分配可分页内存,使用PagedPool标识第一个参数即可。
ExAllocatePoolWithTag分配的内存可以使用ExFreePool来释放,否则的话这块内存就会产生泄漏。虽然用户进程关闭后自动释放进程内分配的空间,但驱动不太一样,即使它已经被卸载,空间也不会自动释放,除非重启计算机。
ExFreePool只需要提供需要释放的指针即可。举例如下:
ExFreePool(dst.Buffer);
dst.Buffer=NULL;
dst.Length=dst.MaximumLength=0;
注意,ExFreePool不能用来释放一个栈空间的指针,否则系统立刻崩溃。诸如下面的代码将会招致立刻蓝屏的灾难:
UNICODE_STRINGsrc=RTL_CONST_STRING(L”Mysourcestring!”);
ExFreePool(src.Buffer);
请务必保持ExAllocatePoolWithTag或ExAllocatePool和ExFreePool的成对关系。