Minifilter 是一种文件过滤驱动,该驱动简称为微过滤驱动,相对于传统的sfilter文件过滤驱动来说,微过滤驱动编写时更简单,其不需要考虑底层RIP如何派发且无需要考虑兼容性问题,微过滤驱动使用过滤管理器FilterManager提供接口,由于提供了管理结构以及一系列管理API函数,所以枚举过滤驱动将变得十分容易 。
通常文件驱动过滤是ARK重要功能之一,如下是一款闭源ARK工具的输出效果图 。

文章插图
【驱动开发:内核枚举Minifilter微过滤驱动】由于
MiniFilter提供了FltEnumerateFilters函数 , 所以只需要调用这些函数即可获取到所有的过滤器地址,我们看下微软公开的信息 。NTSTATUS FLTAPI FltEnumerateFilters([out] PFLT_FILTER *FilterList,[in]ULONGFilterListSize,[out] PULONGNumberFiltersReturned);此函数需要注意 , 如果用户将FilterList设置为NULL则默认是输出当前系统中存在的过滤器数量,而如果传入的是一个内存地址,则将会枚举系统中所有的过滤器信息 。使用
FltEnumerateFilters这个API,它会返回过滤器对象FLT_FILTER的地址,然后根据过滤器对象的地址 , 加上一个偏移,获得记录过滤器PreCall、PostCall、IRP等信息的PFLT_OPERATION_REGISTRATION结构体指针 。上文之所以说要加上偏移 , 是因为
FLT_FILTER的定义在每个系统都不同,比如WIN10 X64中的定义以下样子,这里我们需要记下+0x1a8 Operations因为他指向的就是_FLT_OPERATION_REGISTRATION结构的偏移地址 。lyshark.com: kd> dt fltmgr!_FLT_FILTER+0x000 Base: _FLT_OBJECT+0x030 Frame: Ptr64 _FLTP_FRAME+0x038 Name: _UNICODE_STRING+0x048 DefaultAltitude: _UNICODE_STRING+0x058 Flags: _FLT_FILTER_FLAGS+0x060 DriverObject: Ptr64 _DRIVER_OBJECT+0x068 InstanceList: _FLT_RESOURCE_LIST_HEAD+0x0e8 VerifierExtension : Ptr64 _FLT_VERIFIER_EXTENSION+0x0f0 VerifiedFiltersLink : _LIST_ENTRY+0x100 FilterUnload: Ptr64long+0x108 InstanceSetup: Ptr64long+0x110 InstanceQueryTeardown : Ptr64long+0x118 InstanceTeardownStart : Ptr64void+0x120 InstanceTeardownComplete : Ptr64void+0x128 SupportedContextsListHead : Ptr64 _ALLOCATE_CONTEXT_HEADER+0x130 SupportedContexts : [7] Ptr64 _ALLOCATE_CONTEXT_HEADER+0x168 PreVolumeMount: Ptr64_FLT_PREOP_CALLBACK_STATUS+0x170 PostVolumeMount: Ptr64_FLT_POSTOP_CALLBACK_STATUS+0x178 GenerateFileName : Ptr64long+0x180 NormalizeNameComponent : Ptr64long+0x188 NormalizeNameComponentEx : Ptr64long+0x190 NormalizeContextCleanup : Ptr64void+0x198 KtmNotification: Ptr64long+0x1a0 SectionNotification : Ptr64long+0x1a8 Operations: Ptr64 _FLT_OPERATION_REGISTRATION+0x1b0 OldDriverUnload: Ptr64void+0x1b8 ActiveOpens: _FLT_MUTEX_LIST_HEAD+0x208 ConnectionList: _FLT_MUTEX_LIST_HEAD+0x258 PortList: _FLT_MUTEX_LIST_HEAD+0x2a8 PortLock: _EX_PUSH_LOCK解析FLT_OPERATION_REGISTRATION结构体,可以看到这就是我们需要枚举的过滤器,只要拿到它输出即可:lyshark.com: kd> dt fltmgr!_FLT_OPERATION_REGISTRATION+0x000 MajorFunction: UChar+0x004 Flags: Uint4B+0x008 PreOperation: Ptr64_FLT_PREOP_CALLBACK_STATUS+0x010 PostOperation: Ptr64_FLT_POSTOP_CALLBACK_STATUS+0x018 Reserved1: Ptr64 Void枚举过滤器代码如下所示 。// 配置属性 -> 连接器 -> 输入-> 附加依赖 -> fltMgr.lib// 配置属性 -> C/C++ -> 常规 -> 设置 警告等级2级 (警告视为错误关闭)#include <fltKernel.h>#include <dontuse.h>#include <suppress.h>// 设置默认回调NTSTATUS DriverDefaultHandle(PDEVICE_OBJECT pDevObj, PIRP pIrp){	NTSTATUS status = STATUS_SUCCESS;	pIrp->IoStatus.Status = status;	pIrp->IoStatus.Information = 0;	IoCompleteRequest(pIrp, IO_NO_INCREMENT);	return status;}VOID DriverUnload(PDRIVER_OBJECT pDriverObject){}NTSTATUS DriverEntry(PDRIVER_OBJECT pDriverObject, PUNICODE_STRING pRegPath){	DbgPrint("hello lyshark.com \n");	NTSTATUS status = STATUS_SUCCESS;	pDriverObject->DriverUnload = DriverUnload;	for (ULONG i = 0; i < IRP_MJ_MAXIMUM_FUNCTION; i++)	{pDriverObject->MajorFunction[i] = DriverDefaultHandle;	}	ULONG ulFilterListSize = 0;	PFLT_FILTER *ppFilterList = NULL;	ULONG i = 0;	LONG lOperationsOffset = 0;	PFLT_OPERATION_REGISTRATION pFltOperationRegistration = NULL;	// 获取 Minifilter 过滤器Filter 的数量	FltEnumerateFilters(NULL, 0, &ulFilterListSize);	// 申请内存	ppFilterList = (PFLT_FILTER *)ExAllocatePool(NonPagedPool, ulFilterListSize *sizeof(PFLT_FILTER));	if (NULL == ppFilterList)	{return FALSE;	}	// 获取 Minifilter 中所有过滤器Filter 的信息	status = FltEnumerateFilters(ppFilterList, ulFilterListSize, &ulFilterListSize);	if (!NT_SUCCESS(status))	{return FALSE;	}	DbgPrint("过滤器数量: %d \n", ulFilterListSize);	// 获取 PFLT_FILTER 中 Operations 偏移	lOperationsOffset = 0x1A8;	// 开始遍历 Minifilter	__try	{for (i = 0; i < ulFilterListSize; i++){// 获取 PFLT_FILTER 中 Operations 成员地址pFltOperationRegistration = (PFLT_OPERATION_REGISTRATION)(*(PVOID *)((PUCHAR)ppFilterList[i] + lOperationsOffset));__try{// 同一过滤器下的回调信息while (IRP_MJ_OPERATION_END != pFltOperationRegistration->MajorFunction){if (IRP_MJ_MAXIMUM_FUNCTION > pFltOperationRegistration->MajorFunction){// 显示DbgPrint("Filter: %p | IRP: %d | PreFunc: 0x%p | PostFunc=0x%p \n", ppFilterList[i], pFltOperationRegistration->MajorFunction,pFltOperationRegistration->PreOperation, pFltOperationRegistration->PostOperation);}// 获取下一个消息回调信息pFltOperationRegistration = (PFLT_OPERATION_REGISTRATION)((PUCHAR)pFltOperationRegistration + sizeof(FLT_OPERATION_REGISTRATION));}}__except (EXCEPTION_EXECUTE_HANDLER){}}	}	__except (EXCEPTION_EXECUTE_HANDLER)	{	}	// 释放内存	ExFreePool(ppFilterList);	ppFilterList = NULL;	return status;}
		  	
    
     
    
    
    
推荐阅读
           - 
              
              
            
 
- 
              
              
            
 
- 
              
              
            
 
- 
              
              
            
 
- 
              
              
            
 
- 
              
              
            
 
- 
              
              
            
 
- 
              
              
            
 
- 
              
              
            
 
- 
              
              
            
 
- 
              
              
            
 
- 
              
              
            
 
- 
              
              
            
 
- 
              
              
            
 
- 
              
              
            
 
- 
              
              
            
 
- 
              
              
            
 
- 
              
              
            
 
- 
              
              
            
 
- 
              
              
            
 
          
- Vue3 JS 与 SCSS 变量相互使用 
 
- 24 Node.js躬行记——低代码 
 
- 驱动开发:内核枚举PspCidTable句柄表 
 
- 驱动开发:内核枚举DpcTimer定时器 
 
- envoy开发调试环境搭建 
 
- 如何更新电脑的显卡驱动(笔记本显卡驱动要不要更新) 
 
- 你Win7系统如何将显卡驱动更新到最新版本 
 
- 云原生时代的DevOps平台设计之道 
 
- Taurus.MVC 微服务框架 入门开发教程:项目部署:7、微服务节点的监控与告警。 
 
- IOS12未受信任的企业级开发者怎么解决 
 
            
              
            
          