编写驱动程序
驱动分类简述
驱动大体分为三种,分别是:NT式驱动、WDM式驱动、WDF式驱动(KWDF内核驱动,UWDF用户驱动)。
NT式驱动
NT虚拟驱动,老式驱动,从WIN95开始使用NT式驱动。 若所开发的驱动不与硬件打交道,建议使用NT式驱动或WDM式驱动。如果NT式驱动出现了绑定设备的情况,该驱动将无法卸载。只能通过重启系统进行卸载。对于服务器来说重启很伤。比如说你要插个鼠标 就要重启
WDM式驱动
相对于NT式驱动来讲,WDM式驱动支持卸载(热拔插)。无需重启即可卸载。并且WDM式驱动对于NT式驱动进行了一些封装和优化。本质区别不大。
WDF式驱动
WDF式驱动相较前两种,其最大的意义是简化开发。不像NT与WDM驱动那么底层化。WDF式驱动将WDM式驱动进行了封装,做成了一套架构,使得开发驱动变得更简单。同时带来的弊端就是无法掌控底层。
由于开发简便,不容易蓝屏,所以公司开发驱动一般选用WDF式驱动。
想要学习WDF式驱动,需要了解COM相关知识。
只有系统中存在WDFLDR.sys驱动,我们编写的WDF驱动才可以跑起来。并且项目中需要一个inf文件,NT/WDM式驱动则不需要这个inf文件。
创建驱动项目
创建项目
打开VS2017,新建项目选择Visual C++ -> Windwos Drivers -> Legacy -> Empty WDM Driver
右键SourceFiles目录,新建项。创建一个扩展名为C的C++文件。(不要用cpp扩展名)。文件名随意起,不是非要和项目名一样。
在.c文件中先引入头文件 ntifs.h
。
删除INF文件
驱动入口函数(DriverEntry)
1 2 3 4 5
| #include "ntifs.h" NTSTATUS DriverEntry(PDRIVER_OBJECT DriverObject,PUNICODE_STRING RegistryPath){ //代码 return STATUS_UNSUCCESSFUL; }
|
DriverEntry是我们写代码时的入口函数。其编译生成的sys文件真正的入口点并不是DriverEntry。在IDA中可以看到驱动真正的入口点函数是GsDriverEntry。其内部调用了我们的DriverEntry函数。这个函数的返回值是一个NTSTATUS类型,这个返回值的宏定义在ntstatus.h这个头文件中
2 参数PDRIVER_OBJECT
它代表的是windows中的一个指向驱动对象的指针,前面的P就是Pointer的意思,这个驱动对象对应的就是我们要操作的.sys驱动
这个驱动对象是Windows系统中对某个驱动的唯一标识,里面包括了这个驱动的各种信息,各个功能函数的入口地址等重要信息,这些信息非常的庞大和复杂。
驱动对象一般包括一个及以上的设备对象,总之驱动就是要在一系列设备上进行信息交互实现功能。
3 参数PUNICODE_STRING
这是一个UNICODE类型的字符串,它代表了驱动在注册表中的参数所存放的位置,由于每个驱动都是以一个类似服务的形式存在,在系统注册表存放,注册表可以通过cmd输入regedit进入。
HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Services
Windows内核在启动时加载了一个最小文件系统,分析磁盘并将注册表树下的所有内容读到内存中,这样保证这一部分的注册表内容在Windows内核刚加载之后就是可以读写状态。
指定入口函数
如果不想让编译器生成GsDriverEntry而是直接将入口函数设置为DriverEntry,可以按照下图设置。
编写代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| #include "ntddk.h" void UnloadDriver(PDRIVER_OBJECT driver); NTSTATUS DriverEntry(PDRIVER_OBJECT DriverObject, PUNICODE_STRING RegistryPath) { DbgBreakPoint(); DbgPrint("驱动加载了。\r\n"); DriverObject->DriverUnload = UnloadDriver; return STATUS_SUCCESS; }
void UnloadDriver(PDRIVER_OBJECT driver) { DbgPrint("驱动停止了。\r\n"); }
|
打印字符串对象
如果想要打印字符串对象中的字符串,可以使用如下格式:
1 2 3 4
| NTSTATUS DriverEntry(PDRIVER_OBJECT DriverObject, PUNICODE_STRING pReg) { DbgPrint("-------%wZ--------",pReg); return STATUS_SUCCESS; }
|
生成驱动
点击生成解决方案即可。若报一些格式错误,就删除一些特殊符号之类的东西。
加载驱动(部署-启动-停止-卸载)
使用InstDrv.exe加载驱动。使用DbgView.exe查看输出(必须选中监视核心,否则无法监视驱动层输出)。
调试驱动
通过调用函数DbgBreakPoint为驱动增加一个断点。这个函数相当于int 3指令。
在虚拟机用驱动加载工具运行驱动我们生成的驱动
驱动对象PDRIVER_OBJECT初识
在成功断在我们的代码中后,查看驱动对象结构。
Type:驱动对象类型。
Size:驱动对象大小
DeviceObject:设备对象,我们这里没添加设备,因此是null
DriverStart:驱动文件基址,也就是PE格式中的ImageBase。通过db命令可以看到4D 5A。
-
DriverSize:驱动模块大小,也就是PE格式中的SizeOfImage。
DriverExtension:驱动扩展对象。使用dt命令查看该对象
DriverObject:指向当前驱动对象首地址。
ServiceKeyName:驱动服务注册表文件夹名。
DriverName:驱动名,也就是驱动的文件名前面加个\Driver\。这个名字是个字符串结构体。
查看该字符串结构:
HardwareDatabase:驱动服务注册表路径。前往注册表查看该路径,可以发现一个名为“hellodriver”的文件夹,这就是我们的驱动。
- ErrorControl:当驱动加载失败时会设置这个值。
- ImagePath:驱动文件路径。??\是设备路径,我们平时访问各种文件夹其实都带这个??\,只是windows底层帮我们补充了。
- Start:驱动加载类型。手动启动为3,开机自启为2,BIOS自启为1。
- Type:服务类型。1为驱动。
- DriverInit :驱动入口点,也就是PE文件的AddressOfEntryPoint。
- DriverUnload:驱动卸载函数地址。
驱动加载方法
加载驱动大体分为两种:服务加载和直接加载。实际应用中可以将两种方法都利用上。
服务加载
- 调用OpenSCManager打开服务控制。
- 调用CreateService创建服务。实际上就是创建注册表相关键值。在执行完该API后,驱动已经被注册为服务了。这时我们通过CMD执行net start XXXX也可以加载我们的驱动。
- 调用OpenService打开现有服务。
- 调用StartService启动服务
这种方式实际上加载该驱动的进程,并不是调用API的进程。而是通过API向系统通知我要加载一个驱动。系统进程接收到通知后加入到系统中的一个队列。并由系统进程在某时某刻加载该驱动。
也就是这种方式是通知系统进程来进行加载。
直接加载
调用ZwLoadDriver或NtLoadDriver加载一个已被正确注册的驱动。
这种方法需要我们自己手动去注册表内注册该驱动的相关信息。这样该驱动才可以被加载。
直接加载的方式在调用API后就会直接加载该驱动,所以该驱动的加载者就是调用该API的进程。相比于服务加载会留下痕迹。
第一个练习
编写两个驱动A和B,在A中定义全局变量值为100,打印A的地址pA。在B中打印pA的数据,观察是否与A中定义的相同。
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| #include "ntddk.h"
void UnloadDriver(PDRIVER_OBJECT driver); UINT32 i = 100; NTSTATUS DriverEntry(PDRIVER_OBJECT DriverObject, PUNICODE_STRING RegistryPath) { DbgPrint("驱动加载了。\r\n"); DbgPrint("i addr = %08x\r\n", &i); DriverObject->DriverUnload = UnloadDriver; return STATUS_SUCCESS; }
void UnloadDriver(PDRIVER_OBJECT driver) { DbgPrint("驱动停止了。\r\n"); }
|
老规矩 生成了之后 拖入虚拟机里面加载 然后运行 程序就会断到vs设置断点的位置 我们此时给输出i哪一行 按下f9 看看 windbg窗口 db一下地址显示 64 也就是十进制100
此时虚拟机是卡着的很正常,
然后根据作业要求 打印这个地址 看看是不是64
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| //B #include "ntddk.h"
void UnloadDriver(PDRIVER_OBJECT driver); UINT32 i=9b4fa000; NTSTATUS DriverEntry(PDRIVER_OBJECT DriverObject, PUNICODE_STRING RegistryPath) { DbgPrint("驱动加载了。\r\n"); //驱动的打印函数,相当于3环的printf DbgPrint("%08x\r\n", &i); DriverObject->DriverUnload = UnloadDriver; //为驱动指定卸载函数 return STATUS_SUCCESS; } //驱动卸载函数 void UnloadDriver(PDRIVER_OBJECT driver) { DbgPrint("驱动停止了。\r\n"); }
|
数值不见了
驱动常用类型及API
在驱动中写代码与3环不同,一些数据类型及常用API也最好使用驱动开发专用的版本。这算是一种代码规范。
基本数据类型
在驱动中,原数据类型int char等均被封装、重定义。在驱动开发中应使用如下数据类型:
1 2 3 4 5 6 7 8 9 10 11
| UINT8,PUINT8 -> unsigned char UINT16,PUINT16 -> unsigned short UINT32,PUINT32 -> unsigned int UINT64,PUINT64 -> unsigned __int64 INT8,PINT8 -> char INT16,PINT16 -> short INT32,PINT32 -> int INT64,PINT64 -> __int64 LONG32,PLONG32 -> int ULONG32,PULONG32 -> unsigned int DWORD32,PDWRD32 -> int
|
错误码返回值
绝大多数内核函数都会有一个返回值,类型为NTSTATUS
。该类型本质就是一个LONG。
如GetLastError这种取错误码的函数,取到的值其实就是NTSTATUS转化后的错误码。
常用的NTSTATUS宏如下,负数(大于0X80000000)的返回值为错误,大于等于0为成功
1 2 3 4 5 6
| STATUS_SEVERITY_SUCCESS 0x0 STATUS_SEVERITY_INFORMATIONAL 0x1 STATUS_SEVERITY_WARNING 0x2 STATUS_SEVERITY_ERROR 0x3 STATUS_UNSUCCESSFUL 0xC0000001 STATUS_ACCESS_VIOLATION 0xC0000005
|
同时有一个宏用于判断返回值是成功还是失败:
1
| NT_SUCCESS(NTSATUS类型参数) ``/``/``#define NT_SUCCESS(Status) (((NTSTATUS)(Status)) >= 0)
|
字符串相关
在内核开发中,字符串不要定义为char* x = “xx”,WDK为我们准备了一些字符串相关的API。
定义字符串
1 2 3
| cUNICODE_STRING uStr = {0}; STRING aStr = {0}; ANSI_STRING aStr = {0};
|
初始化字符串
1 2 3
| RtlInitUnicodeString(&uStr,L"unicode string"); RtlInitString(&aStr,"ascii string"); RtlInitAnsiString(&aStr,"ascii string");
|
字符串转化
1 2
| RtlAnsiStringToUnicodeString(&uStr,&aStr,true); RtlUnicodeStringToAnsiString();
|
释放字符串
1 2
| RtlFreeUnicodeString(); RtlFreeAnsiString();
|
字符串格式化
1 2 3 4 5
| #include <ntstrsafe.h> char aStr[0x1000]= {0}; RtlStringCbPrintfA(aStr, 0x1000, "%d---%s", 123, "test"); wchar uStr[0x1000] = {0}; RtlStringCbPrintfW(uStr, 0x1000, L"%d---%s", 123, L"test");
|
字符串比较
1 2
| RtlCompareUnicodeString(&uStr1,&uStr2,TRUE); RtlCompareString
|
内存相关
申请内存
1 2 3 4
| ExAllocatePool(type,size);
ExAllocatePoolWithTag(type,size,tag);
|
拷贝、设置、比较内存
1 2 3 4 5
| RtlFillMemory(pointer,length,value); RtlEqualMemory(pointer,Source,Length) RtlMoveMemory(pointer,Source,Length) RtlCopyMemory(pointer,Source,Length) RtlZeroMemory(pointer,Length)
|
释放内存
延迟
1 2 3 4 5 6 7
| LARGE_INTEGER li = { 0 }; li.QuadPart = -10000 * 5000; KeDelayExecutionThread(KernelMode,FALSE,&li);
|
创建线程
1 2 3 4 5 6 7 8 9 10 11
| VOID myThreadFun(_In_ PVOID StartContext) { } HANDLE tHandle = NULL; NTSTATUS tRet = PsCreateSystemThread(&tHandle,THREAD_ALL_ACCESS,NULL,NULL,NULL, myThreadFun,NULL);
if(NT_SUCCESS(tRet)){ ZwClose(tHandle); }
|
内核链表API
windows开发人员很喜欢使用链表,你可以在很多内核结构中看到LIST_ENTRY
成员。这就是链表节点结构。也是WDK中提供的一个官方链表结构。LIST_ENTRY是一个双向链表。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
| typedef struct _Monster { UINT32 ID; LIST_ENTRY node; UINT32 hp; UINT32 level; UNICODE_STRING name; }Monster,*PMonster; Monster m1 = { 0 }; InitializeListHead(&m1.node); IsListEmpty(&m1.node); Monster m2 = { 0 }; InsertHeadList(&m1.node, &m2.node); Monster m3 = { 0 }; InsertTailList(&m1.node, &m3.node); RemoveHeadList(&m2.node); RemoveTailList(&m2.node); RemoveEntryList(&m3.node);
PMonster pm = (PMonster)((UCHAR)m2.node.Flink - ((UCHAR)(&m2.node) - (UCHAR)&m2));
|
内核二叉树API
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59
| typedef struct _Monster { UINT32 id; UINT32 hp; UINT32 level; UNICODE_STRING name; }Monster,*PMonster;
RTL_GENERIC_COMPARE_RESULTS NTAPI myCmpFunc(_In_ struct _RTL_GENERIC_TABLE *Table,_In_ PVOID FirstStruct,_In_ PVOID SecondStruct) { PMonster m1 = (PMonster)FirstStruct; PMonster m2 = (PMonster)SecondStruct; if (m1->id == m2->id) { return GenericEqual; } return m1->id > m2->id ? GenericGreaterThan : GenericLessThan; }
VOID NTAPI myAllocFunc( _In_ struct _RTL_GENERIC_TABLE *Table, _In_ CLONG ByteSize ) { ExAllocatePool(NonPagedPool,ByteSize); }
VOID NTAPI myFreeFunc( _In_ struct _RTL_GENERIC_TABLE *Table, _In_ __drv_freesMem(Mem) _Post_invalid_ PVOID Buffer ) { ExFreePool(Buffer); } Monster m1 = {0,100,10,L"monster 1"}; Monster m2 = {1,100,10,L"monster 2"}; Monster m3 = {2,100,10,L"monster 3"}; RTL_GENERIC_TABLE table = {0};
RtlInitializeGenericTable(&table, myCmpFunc, myAllocFunc, myFreeFunc,NULL); BOOLEAN isNewEle = FALSE;
RtlInsertElementGenericTable(&table, (PVOID)&m1,sizeof(m1),&isNewEle); RtlInsertElementGenericTable(&table, (PVOID)&m2,sizeof(m1),&isNewEle); RtlInsertElementGenericTable(&table, (PVOID)&m3,sizeof(m1),&isNewEle);
Monster lookupM = { 0,0,0,0 }; PMonster lookupResult = (PMonster)RtlLookupElementGenericTable(&table,&lookupM);
RtlDeleteElementGenericTable(&table, &lookupM);
ULONG nodeNum = RtlNumberGenericTableElements(&table); ULONG nodeNum = RtlNumberGenericTableElementsAvl(&table);
PVOID key = NULL; PMonster pm = (PMonster)RtlEnumerateGenericTableWithoutSplaying(&table, &key); while (pm!=NULL) { DbgPrint(pm->name.Buffer); pm = (PMonster)RtlEnumerateGenericTableWithoutSplaying(&table, &key); }
|
驱动对象-DriverSection
驱动对象中有一个成员名为DriverSection,其数据类型为未公开类型_KLDR_DATA_TABLE_ENTRY
的结构指针。该结构信息可以在WRK源码中搜索到。成员如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
| typedef struct _KLDR_DATA_TABLE_ENTRY { LIST_ENTRY InLoadOrderLinks; ULONG __Undefined1; ULONG __Undefined2; ULONG __Undefined3; ULONG NonPagedDebugInfo; ULONG DllBase; ULONG EntryPoint; ULONG SizeOfImage; UNICODE_STRING FullDllName; UNICODE_STRING BaseDllName; ULONG Flags; USHORT LoadCount; USHORT __Undefined5; ULONG __Undefined6; ULONG CheckSum; ULONG TimeDateStamp; } KLDR_DATA_TABLE_ENTRY, *PKLDR_DATA_TABLE_ENTRY;
|
其中第一个成员InLoadOrderLinks是一个链表节点结构。由此可知,_KLDR_DATA_TABLE_ENTRY是一个双向链表。
驱动模块遍历-手动
编写如下代码,加载该驱动:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
| typedef struct _KLDR_DATA_TABLE_ENTRY { LIST_ENTRY InLoadOrderLinks; ULONG __Undefined1; ULONG __Undefined2; ULONG __Undefined3; ULONG NonPagedDebugInfo; ULONG DllBase; ULONG EntryPoint; ULONG SizeOfImage; UNICODE_STRING FullDllName; UNICODE_STRING BaseDllName; ULONG Flags; USHORT LoadCount; USHORT __Undefined5; ULONG __Undefined6; ULONG CheckSum; ULONG TimeDateStamp; } KLDR_DATA_TABLE_ENTRY, *PKLDR_DATA_TABLE_ENTRY; NTSTATUS DriverEntry(PDRIVER_OBJECT DriverObject, PUNICODE_STRING RegistryPath) { DbgBreakPoint(); KLDR_DATA_TABLE_ENTRY * ldr = DriverObject->DriverSection; DriverObject->DriverUnload = UnloadDriver; return STATUS_SUCCESS; }
|
还是昨天那个代码 跑起来 下断
输入命令dt ldr
查看自身节点结构:
输入命令dt _KLDR_DATA_TABLE_ENTRY 0x83f62850
查看下一个节点的结构。
可以发现很多属性都是0。这里我们需要知道一个常识,windows很多链表都喜欢将头部节点成员置位null,从第二个节点开始才是真正的有效数据。
继续执行命令dt _KLDR_DATA_TABLE_ENTRY 0x86341c98
查看下一个节点。
可以发现找到了ntoskrnl模块。经过多次重复寻找,可以发现这个链表是一个双向循环链表
驱动模块遍历-代码
将手动遍历的方法写入代码中:
1 2 3 4 5 6 7 8 9 10 11
| NTSTATUS DriverEntry(PDRIVER_OBJECT DriverObject, PUNICODE_STRING RegistryPath) { PKLDR_DATA_TABLE_ENTRY selfNode = DriverObject->DriverSection; PKLDR_DATA_TABLE_ENTRY preNode = selfNode; UINT32 index = 1; do { DbgPrint("[db] %d driver name = %wZ \r\n", index++,&preNode->BaseDllName); preNode = preNode->InLoadOrderLinks.Flink; } while (preNode != selfNode); DriverObject->DriverUnload = UnloadDriver; return STATUS_SUCCESS; }
|
加载驱动,观察输出内容,与PCHUNTER做对比,可以发现已经将全部驱动遍历出来了:(多一个因为吧空节点字符串也打印出来了,理应过滤掉)
驱动模块隐藏-断链
完成对驱动模块的遍历后,我们要开始搞事情了。在实际攻防对抗中,无论是外挂开发者或是内核木马开发者,为了让自己的驱动悄悄的运行起来,都会对自身的驱动模块做一些隐藏操作。使其自身无法被检测到。而断链就是一个古老但又有效的一个方法。无论你将来从事攻或防,了解一些老技术都是必不可少的。
断链1-HTTP.sys
断链练习不要乱找一个驱动就开始断链, 否则可能对系统造成影响,这里我们拿HTTP.sys做断链练习,理论上如果断链成功,PCHUNTER中应该看不到HTTP.sys驱动。代码如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
| NTSTATUS DriverEntry(PDRIVER_OBJECT DriverObject, PUNICODE_STRING RegistryPath) { PKLDR_DATA_TABLE_ENTRY selfNode = DriverObject->DriverSection; PKLDR_DATA_TABLE_ENTRY preNode = selfNode; UNICODE_STRING httpName = { 0 }; RtlInitUnicodeString(&httpName,L"HTTP.sys"); do { if (preNode->BaseDllName.Length != 0 && RtlCompareUnicodeString(&preNode->BaseDllName,&httpName,TRUE) == 0) { DbgPrint("%wZ\r\n", &preNode->BaseDllName); RemoveEntryList(preNode); break; } preNode = preNode->InLoadOrderLinks.Flink; } while (preNode != selfNode); DriverObject->DriverUnload = UnloadDriver; return STATUS_SUCCESS; }
|
加载驱动后,dbgview成功打印,说明此时已经成功断链了。去PCHUNTER中观察一下效果:
可以发现HTTP.sys仍然在列表中,但是它变红了。这是因为PCHUNTER的遍历方法更健壮一些,不单单是通过链表遍历。还会涉及特征、文件等。有精力的话可以逆向一下PCHUNTER。
所以为了达到完美隐藏,我们需要改善一下我们的代码,抹掉驱动对象中的一些特征,在此之前,我们需要了解一下如何通过驱动名来获取驱动对象指针。
获取驱动对象指针
微软有一个未公开的导出函数ObReferenceObjectByName
。这个函数可以根据驱动名获取驱动对象指针。在WRK源码中可以搜索到。
1 2 3 4 5 6 7 8 9 10 11
| NTKERNELAPI NTSTATUS ObReferenceObjectByName( __in PUNICODE_STRING ObjectName, __in ULONG Attributes, __in_opt PACCESS_STATE AccessState, __in_opt ACCESS_MASK DesiredAccess, __in POBJECT_TYPE ObjectType, __in KPROCESSOR_MODE AccessMode, __inout_opt PVOID ParseContext, __out PVOID *Object );
|
其中ObjectType对象类型我们通过F12未找到该类型都有哪些值。这些类型是未公开的。同样在WRK中可以找到。此处我们使用一个名为IoDriverObjectType
的导出变量。
1
| extern POBJECT_TYPE * IoDriverObjectType;
|
断链2-HTTP.sys增强
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27
| extern POBJECT_TYPE * IoDriverObjectType; NTSTATUS DriverEntry(PDRIVER_OBJECT DriverObject, PUNICODE_STRING RegistryPath) { PKLDR_DATA_TABLE_ENTRY selfNode = DriverObject->DriverSection; PKLDR_DATA_TABLE_ENTRY preNode = selfNode; UNICODE_STRING httpName = { 0 }; RtlInitUnicodeString(&httpName,L"HTTP.sys"); UNICODE_STRING httpObjName = { 0 }; RtlInitUnicodeString(&httpObjName, L"\\Driver\\HTTP"); do { if (preNode->BaseDllName.Length != 0 && RtlCompareUnicodeString(&preNode->BaseDllName,&httpName,TRUE) == 0) { DbgPrint("%wZ\r\n", &preNode->BaseDllName); PDRIVER_OBJECT pHttpObj = NULL; ObReferenceObjectByName(&httpObjName,FILE_ALL_ACCESS,NULL,NULL, *IoDriverObjectType, KernelMode,NULL, &pHttpObj); pHttpObj->Flags = 0; pHttpObj->DriverSection = 0; pHttpObj->DriverInit = 0; RemoveEntryList(preNode); break; } preNode = preNode->InLoadOrderLinks.Flink; } while (preNode != selfNode); DriverObject->DriverUnload = UnloadDriver; return STATUS_SUCCESS; }
|
加载驱动,观察效果(记得重启,刚刚链表已经断掉HTTP了。):
可以看到HTTP已经彻底从PCHUNTER中消失了,也没有了红色HTTP的记录。我们的断链操作成功了。
断链3-自身驱动
对自身驱动模块进行断链,省去了遍历和取驱动对象的步骤,理论上是更简单的,我们尝试一下。
1 2 3 4 5 6 7 8 9 10 11
| NTSTATUS DriverEntry(PDRIVER_OBJECT DriverObject, PUNICODE_STRING RegistryPath) { PKLDR_DATA_TABLE_ENTRY selfNode = DriverObject->DriverSection; DriverObject->Flags = 0; DriverObject->DriverSection = 0; DriverObject->DriverInit = 0; RemoveEntryList(selfNode); DriverObject->DriverUnload = UnloadDriver; return STATUS_SUCCESS; }
|
加载驱动,观察效果:
发现驱动无法被加载。这是因为系统调用完DriverEntry后仍需要做一些处理。而我们吧自己的驱动隐藏掉了,导致系统找不到我们的驱动没办法做后续处理。从而返回驱动加载失败。
所以我们需要在驱动成功加载后再进行断链操作,这里使用新线程+延迟执行的方法来规避。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
| VOID hideSelf(PVOID pDriverObj) { LARGE_INTEGER li = { 0 }; li.QuadPart = -10000 * 10000; KeDelayExecutionThread(KernelMode,FALSE,&li); PDRIVER_OBJECT DriverObject = (PDRIVER_OBJECT)pDriverObj; PKLDR_DATA_TABLE_ENTRY selfNode = DriverObject->DriverSection; DriverObject->Flags = 0; DriverObject->DriverSection = 0; DriverObject->DriverInit = 0; RemoveEntryList(selfNode); } NTSTATUS DriverEntry(PDRIVER_OBJECT DriverObject, PUNICODE_STRING RegistryPath) { HANDLE tHandle = NULL; NTSTATUS ret = PsCreateSystemThread(&tHandle,THREAD_ALL_ACCESS,NULL,NULL,NULL, hideSelf, DriverObject); if (NT_SUCCESS(ret)) { ZwClose(ret); } DriverObject->DriverUnload = UnloadDriver; return STATUS_SUCCESS; }
|
再次尝试加载驱动,分别观察刚加载驱动时PCHUNTER的列表和延迟10秒后PCHUNTER的列表。
可以看到我们的驱动已经成功被加载了,并在10秒后做了断链隐藏处理。但目前仍有一个BUG,那就是卸载驱动时会提示“请求的控件对此服务无效。”。若想修复这个BUG,需要在卸载之前还原我们的驱动,将节点重新接入链表中并恢复被清空的属性。
驱动通信-常规
驱动在实际使用中不可能从入口点一路执行到结束。 将驱动按功能分成模块, 需要时调用才是实际的应用。通过用户层与内核层的通信,可以让用户程序在需要时调用驱动的特定功能。无论是攻击方或是防守方,驱动通信都是一个关键的战场。
这里的常规通信使用设备交互的方式,其类似与WIN32的消息和回调函数的组合。在内核中,消息被封装为一个结构体IRP(I/O Request Packae)。设备对象可以接收IRP数据从而实现通信。
0环代码-创建设备
用户应用想要向驱动发起通信,其本质是向驱动所绑定的设备发起通信,再由设备向下分发。所以我们需要首先创建一个设备:
1 2 3 4 5 6 7 8 9 10 11
| UNICODE_STRING deviceName = { 0 }; RtlInitUnicodeString(&deviceName,L"\\Device\\MyDevice"); DEVICE_OBJECT devObj = {0}; NTSTATUS retStatus = IoCreateDevice(pDriverObj,NULL,&deviceName,FILE_DEVICE_UNKNOWN,FILE_DEVICE_SECURE_OPEN,FALSE,&devObj);
|
0环代码-设置数据交互方式
1 2 3 4 5 6
| pDeviceObj->Flags |= DO_BUFFERED_IO;
pDeviceObj->Flags &= DO_DEVICE_INITIALIZING;
|
0环代码-创建符号链接
3环想要打开我们的设备无法直接使用\\Device\\XXX
这种设备名,我们需要指定一个符号链接(别名)用于3环的访问。
1 2 3 4 5
| UNICODE_STRING symName = { 0 }; RtlInitUnicodeString(&symName, L"\\??\\MyDeviceSymbol"); retStatus = IoCreateSymbolicLink(&symName,&deviceName);
|
IRP消息
在用户层,我们每次调用CreateFile、OpenFIle、DeleteFile、CloseHandle等API时,都会向0环发送一个消息,这个消息成为IRP数据包。这些API称为设备操作API
。如:当调用CreateFile时,会向内核层发送一个名为IRP_MJ_CREATE
的打开设备的IRP消息。其他常用IRP类型如下:
1 2 3 4 5
| CreateFile -》 IRP_MJ_CREATE ReadFile -》 IRP_MJ_READ WriteFile -》 IRP_MJ_WRITE CloseHandle -》 IRP_MJ_CLOSE DeviceControl -》 IRP_MJ_DEVICE_CONTROL
|
派遣函数
在用户层我们使用WIN32开发GUI时,通过回调函数
来处理窗口消息
。 而在内核层,我们通过派遣函数
来处理IRP消息
。只是换了个名字而已,本质一样。
在驱动对象中,有个属性名为MajorFunction
,这是个数组,每个元素都是一个函数指针,对应了各种类型IRP的派遣函数。
派遣函数格式如下:
1 2 3 4 5 6 7 8 9
| NTSTATUS MyDispatchFunction(PDEVICE_OBJECT pDevObj,PIRP pIrp){ ... pIrp->IoStatus.Status = STATUS_SUCCESS; pIrp->IoStatus.Information = 0; IoCompleteRequest(pIrp,IO_NO_INCREMENT); return STATUS_SUCCESS; }
|
0环代码-处理IRP消息1
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
| NTSTATUS NullFunc(DEVICE_OBJECT *DeviceObject, IRP *Irp) { Irp->IoStatus.Status = STATUS_SUCCESS; Irp->IoStatus.Information = 0; IoCompleteRequest(Irp, IO_NO_INCREMENT); return STATUS_SUCCESS; } NTSTATUS DeviceControlFunc(DEVICE_OBJECT *DeviceObject, IRP *Irp) { Irp->IoStatus.Status = STATUS_SUCCESS; Irp->IoStatus.Information = 0; IoCompleteRequest(Irp, IO_NO_INCREMENT); return STATUS_SUCCESS; }
NTSTATUS DriverEntry(PDRIVER_OBJECT pDriverObj,PUNICODE_STRING pReg) { pDriverObj->MajorFunction[IRP_MJ_DEVICE_CONTROL] = DeviceControlFunc; pDriverObj->MajorFunction[IRP_MJ_CREATE] = NullFunc; pDriverObj->MajorFunction[IRP_MJ_CLOSE] = NullFunc; }
|
3环代码-发送IRP消息
驱动现在已经可以接收IRP消息了,那么我们在3环中就可以发送一个IRP消息了。使用DeviceIoControl
函数来向设备发送一个IRP_MJ_DEVICE_CONTROL
类型的IRP消息。(也可以用CreateFile进行通信,此处不做演示。)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
| #include <iostream> #include <Windows.h> #include <winioctl.h>
#define code1 CTL_CODE(FILE_DEVICE_UNKNOWN,0x800,METHOD_BUFFERED,FILE_ANY_ACCESS) #define code2 CTL_CODE(FILE_DEVICE_UNKNOWN,0x900,METHOD_BUFFERED,FILE_ANY_ACCESS) int main() { CHAR* devName = (CHAR*)"\\\\.\\MyDeviceSymbol"; HANDLE devHandle = CreateFileA(devName,GENERIC_READ|GENERIC_WRITE, FILE_SHARE_READ| FILE_SHARE_WRITE,NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL,NULL); DWORD str = 100; DWORD back = 0; DWORD backLen = 0; DeviceIoControl(devHandle, code1, &str,0x4,&back,0x4,&backLen,NULL); printf("back = %d\r\n", back); str = 200; DeviceIoControl(devHandle, code2, &str, 0x4, &back, 0x4, &backLen, NULL); printf("back = %d\r\n", back); getchar(); CloseHandle(devHandle); return 0; }
|
0环代码-处理IRP消息-扩展
3环的代码已经写完了,我们需要对IRP中的控制码再做一个详细的分支处理:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33
| #define code1 CTL_CODE(FILE_DEVICE_UNKNOWN,0x800,METHOD_BUFFERED,FILE_ANY_ACCESS) #define code2 CTL_CODE(FILE_DEVICE_UNKNOWN,0x900,METHOD_BUFFERED,FILE_ANY_ACCESS)
NTSTATUS DeviceControlFunc(DEVICE_OBJECT *DeviceObject, IRP *Irp) { PIO_STACK_LOCATION ioStack = IoGetCurrentIrpStackLocation(Irp); ULONG code = ioStack->Parameters.DeviceIoControl.IoControlCode; PVOID buffer = Irp->AssociatedIrp.SystemBuffer; switch (code) { case code1: DbgPrint("param = %d\r\n",*(PUINT32)buffer); *(PUINT32)buffer = 800; Irp->IoStatus.Information = 4; break; case code2: DbgPrint("param = %d\r\n", *(PUINT32)buffer); *(PUINT32)buffer = 900; Irp->IoStatus.Information = 4; break; default: break; } Irp->IoStatus.Status = STATUS_SUCCESS; IoCompleteRequest(Irp, IO_NO_INCREMENT); return STATUS_SUCCESS; }
|
0环代码-卸载设备和符号链接
在驱动卸载时要删除设备和符号链接,否则会一直存在内核空间中,并且无法创建同名设备。
先创建设备,后创建符号链接。所以删除时先删除符号链接,再卸载设备。
1 2 3 4
| VOID UnloadDriver( DRIVER_OBJECT *DriverObject ) { IoDeleteSymbolicLink(&symName); IoDeleteDevice(DriverObject->DeviceObject); }
|
执行效果
如果代码没问题,执行后的效果如下:
转载请注明来源,欢迎对文章中的引用来源进行考证,欢迎指出任何有错误或不够清晰的表达。可以在下面评论区评论,也可以邮件至 jaytp@qq.com