驱动开发:内核LoadLibrary实现DLL注入

2023-11-20 17:40

本文主要是介绍驱动开发:内核LoadLibrary实现DLL注入,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

远程线程注入是最常用的一种注入技术,在应用层注入是通过CreateRemoteThread这个函数实现的,该函数通过创建线程并调用 LoadLibrary 动态载入指定的DLL来实现注入,而在内核层同样存在一个类似的内核函数RtlCreateUserThread,但需要注意的是此函数未被公开,RtlCreateUserThread其实是对NtCreateThreadEx的包装,但最终会调用ZwCreateThread来实现注入,RtlCreateUserThreadCreateRemoteThread的底层实现。

基于LoadLibrary实现的注入原理可以具体分为如下几步;

  • 1.调用AllocMemory,在对端应用层开辟空间,函数封装来源于《内核远程堆分配与销毁》章节;
  • 2.调用MDLWriteMemory,将DLL路径字符串写出到对端内存,函数封装来源于《内核MDL读写进程内存》章节;
  • 3.调用GetUserModuleAddress,获取到kernel32.dll模块基址,函数封装来源于《内核远程线程实现DLL注入》章节;
  • 4.调用GetModuleExportAddress,获取到LoadLibraryW函数的内存地址,函数封装来源于《内核远程线程实现DLL注入》章节;
  • 5.最后调用本章封装函数MyCreateRemoteThread,将应用层DLL动态转载到进程内,实现DLL注入;

总结起来就是首先在目标进程申请一块空间,空间里面写入要注入的DLL的路径字符串或者是一段ShellCode,找到该内存中LoadLibrary的基址并传入到RtlCreateUserThread中,此时进程自动加载我们指定路径下的DLL文件。

注入依赖于RtlCreateUserThread这个未到处内核函数,该内核函数中最需要关心的参数是ProcessHandle用于接收进程句柄,StartAddress接收一个函数地址,StartParameter用于对函数传递参数,具体的函数原型如下所示;

typedef DWORD(WINAPI* pRtlCreateUserThread)(IN HANDLE                    ProcessHandle,          // 进程句柄IN PSECURITY_DESCRIPTOR      SecurityDescriptor,IN BOOL                      CreateSuspended,IN ULONG                     StackZeroBits,IN OUT PULONG                StackReserved,IN OUT PULONG                StackCommit,IN LPVOID                    StartAddress,          // 执行函数地址 LoadLibraryWIN LPVOID                    StartParameter,        // 参数传递OUT HANDLE                   ThreadHandle,          // 线程句柄OUT LPVOID                   ClientID);

由于我们加载DLL使用的是LoadLibraryW函数,此函数在运行时只需要一个参数,我们可以将DLL的路径传递进去,并调用LoadLibraryW以此来将特定模块拉起,该函数的定义规范如下所示;

HMODULE LoadLibraryW([in] LPCWSTR lpLibFileName
);

根据上一篇文章中针对注入头文件lyshark.h的封装,本章将继续使用这个头文件中的函数,首先我们实现这样一个功能,将一段准备好的UCHAR字符串动态的写出到应用层进程内存,并以宽字节模式写出在对端内存中,这段代码可以写为如下样子;

// 署名权
// right to sign one's name on a piece of work
// PowerBy: LyShark
// Email: me@lyshark.com#include "lyshark.h"// 驱动卸载例程
VOID UnDriver(PDRIVER_OBJECT driver)
{DbgPrint("Uninstall Driver \n");
}// 驱动入口地址
NTSTATUS DriverEntry(IN PDRIVER_OBJECT Driver, PUNICODE_STRING RegistryPath)
{DbgPrint("Hello LyShark \n");DWORD process_id = 7112;DWORD create_size = 1024;DWORD64 ref_address = 0;// 分配内存堆 《内核远程堆分配与销毁》 核心代码NTSTATUS Status = AllocMemory(process_id, create_size, &ref_address);DbgPrint("对端进程: %d \n", process_id);DbgPrint("分配长度: %d \n", create_size);DbgPrint("[*] 分配内核堆基址: %p \n", ref_address);UCHAR DllPath[256] = "C:\\hook.dll";UCHAR Item[256] = { 0 };// 将字节转为双字for (int x = 0, y = 0; x < strlen(DllPath) * 2; x += 2, y++){Item[x] = DllPath[y];}// 写出内存 《内核MDL读写进程内存》 核心代码ReadMemoryStruct ptr;ptr.pid = process_id;ptr.address = ref_address;ptr.size = strlen(DllPath) * 2;// 需要写入的数据ptr.data = ExAllocatePool(PagedPool, ptr.size);// 循环设置for (int i = 0; i < ptr.size; i++){ptr.data[i] = Item[i];}// 写内存MDLWriteMemory(&ptr);Driver->DriverUnload = UnDriver;return STATUS_SUCCESS;
}

运行如上方所示的代码,将会在目标进程7112中开辟一段内存空间,并写出C:\hook.dll字符串,运行效果图如下所示;

此处你可以通过x64dbg附加到应用层进程内,并观察内存0000000002200000会看到如下字符串已被写出,双字类型则是每一个字符空一格,效果图如下所示;

继续实现所需要的子功能,实现动态获取Kernel32.dll模块里面LiadLibraryW这个导出函数的内存地址,这段代码相信你可以很容易的写出来,根据上节课的知识点我们可以二次封装一个GetProcessAddress来实现对特定模块基址的获取功能,如下是完整代码案例;

// 署名权
// right to sign one's name on a piece of work
// PowerBy: LyShark
// Email: me@lyshark.com#include "lyshark.h"// 实现取模块基址
PVOID GetProcessAddress(HANDLE ProcessID, PWCHAR DllName, PCCHAR FunctionName)
{PEPROCESS EProcess = NULL;NTSTATUS Status = STATUS_SUCCESS;KAPC_STATE ApcState;PVOID RefAddress = 0;// 根据PID得到进程EProcess结构Status = PsLookupProcessByProcessId(ProcessID, &EProcess);if (Status != STATUS_SUCCESS){return Status;}// 判断目标进程是32位还是64位BOOLEAN IsWow64 = (PsGetProcessWow64Process(EProcess) != NULL) ? TRUE : FALSE;// 验证地址是否可读if (!MmIsAddressValid(EProcess)){return NULL;}// 将当前线程连接到目标进程的地址空间(附加进程)KeStackAttachProcess((PRKPROCESS)EProcess, &ApcState);__try{UNICODE_STRING DllUnicodeString = { 0 };PVOID BaseAddress = NULL;// 得到进程内模块基地址RtlInitUnicodeString(&DllUnicodeString, DllName);BaseAddress = GetUserModuleAddress(EProcess, &DllUnicodeString, IsWow64);if (!BaseAddress){return NULL;}DbgPrint("[*] 模块基址: %p \n", BaseAddress);// 得到该函数地址RefAddress = GetModuleExportAddress(BaseAddress, FunctionName, EProcess);DbgPrint("[*] 函数地址: %p \n", RefAddress);}__except (EXCEPTION_EXECUTE_HANDLER){return NULL;}// 取消附加KeUnstackDetachProcess(&ApcState);return RefAddress;
}VOID Unload(PDRIVER_OBJECT pDriverObj)
{DbgPrint("[-] 驱动卸载 \n");
}NTSTATUS DriverEntry(PDRIVER_OBJECT DriverObject, PUNICODE_STRING RegPath)
{DbgPrint("Hello LyShark.com \n");// 取模块基址PVOID pLoadLibraryW = GetProcessAddress(5200, L"kernel32.dll", "LoadLibraryW");DbgPrint("[*] 所在内存地址 = %p \n", pLoadLibraryW);DriverObject->DriverUnload = Unload;return STATUS_SUCCESS;
}

编译并运行如上驱动代码,将自动获取PID=5200进程中Kernel32.dll模块内的LoadLibraryW的内存地址,输出效果图如下所示;

实现注入的最后一步就是调用自定义函数MyCreateRemoteThread该函数实现原理是调用RtlCreateUserThread开线程执行,这段代码的最终实现如下所示;

// 署名权
// right to sign one's name on a piece of work
// PowerBy: LyShark
// Email: me@lyshark.com#include "lyshark.h"// 定义函数指针
typedef PVOID(NTAPI* PfnRtlCreateUserThread)
(IN HANDLE ProcessHandle,IN PSECURITY_DESCRIPTOR SecurityDescriptor,IN BOOLEAN CreateSuspended,IN ULONG StackZeroBits,IN OUT size_t StackReserved,IN OUT size_t StackCommit,IN PVOID StartAddress,IN PVOID StartParameter,OUT PHANDLE ThreadHandle,OUT PCLIENT_ID ClientID
);// 实现取模块基址
PVOID GetProcessAddress(HANDLE ProcessID, PWCHAR DllName, PCCHAR FunctionName)
{PEPROCESS EProcess = NULL;NTSTATUS Status = STATUS_SUCCESS;KAPC_STATE ApcState;PVOID RefAddress = 0;// 根据PID得到进程EProcess结构Status = PsLookupProcessByProcessId(ProcessID, &EProcess);if (Status != STATUS_SUCCESS){return Status;}// 判断目标进程是32位还是64位BOOLEAN IsWow64 = (PsGetProcessWow64Process(EProcess) != NULL) ? TRUE : FALSE;// 验证地址是否可读if (!MmIsAddressValid(EProcess)){return NULL;}// 将当前线程连接到目标进程的地址空间(附加进程)KeStackAttachProcess((PRKPROCESS)EProcess, &ApcState);__try{UNICODE_STRING DllUnicodeString = { 0 };PVOID BaseAddress = NULL;// 得到进程内模块基地址RtlInitUnicodeString(&DllUnicodeString, DllName);BaseAddress = GetUserModuleAddress(EProcess, &DllUnicodeString, IsWow64);if (!BaseAddress){return NULL;}DbgPrint("[*] 模块基址: %p \n", BaseAddress);// 得到该函数地址RefAddress = GetModuleExportAddress(BaseAddress, FunctionName, EProcess);DbgPrint("[*] 函数地址: %p \n", RefAddress);}__except (EXCEPTION_EXECUTE_HANDLER){return NULL;}// 取消附加KeUnstackDetachProcess(&ApcState);return RefAddress;
}// 远程线程注入函数
BOOLEAN MyCreateRemoteThread(ULONG pid, PVOID pRing3Address, PVOID PParam)
{NTSTATUS status = STATUS_UNSUCCESSFUL;PEPROCESS pEProcess = NULL;KAPC_STATE ApcState = { 0 };PfnRtlCreateUserThread RtlCreateUserThread = NULL;HANDLE hThread = 0;__try{// 获取RtlCreateUserThread函数的内存地址UNICODE_STRING ustrRtlCreateUserThread;RtlInitUnicodeString(&ustrRtlCreateUserThread, L"RtlCreateUserThread");RtlCreateUserThread = (PfnRtlCreateUserThread)MmGetSystemRoutineAddress(&ustrRtlCreateUserThread);if (RtlCreateUserThread == NULL){return FALSE;}// 根据进程PID获取进程EProcess结构status = PsLookupProcessByProcessId((HANDLE)pid, &pEProcess);if (!NT_SUCCESS(status)){return FALSE;}// 附加到目标进程内KeStackAttachProcess(pEProcess, &ApcState);// 验证进程是否可读写if (!MmIsAddressValid(pRing3Address)){return FALSE;}// 启动注入线程status = RtlCreateUserThread(ZwCurrentProcess(),NULL,FALSE,0,0,0,pRing3Address,PParam,&hThread,NULL);if (!NT_SUCCESS(status)){return FALSE;}return TRUE;}__finally{// 释放对象if (pEProcess != NULL){ObDereferenceObject(pEProcess);pEProcess = NULL;}// 取消附加进程KeUnstackDetachProcess(&ApcState);}return FALSE;
}VOID Unload(PDRIVER_OBJECT pDriverObj)
{DbgPrint("[-] 驱动卸载 \n");
}NTSTATUS DriverEntry(PDRIVER_OBJECT DriverObject, PUNICODE_STRING RegPath)
{DbgPrint("Hello LyShark.com \n");ULONG process_id = 5200;DWORD create_size = 1024;DWORD64 ref_address = 0;// -------------------------------------------------------// 取模块基址// -------------------------------------------------------PVOID pLoadLibraryW = GetProcessAddress(process_id, L"kernel32.dll", "LoadLibraryW");DbgPrint("[*] 所在内存地址 = %p \n", pLoadLibraryW);// -------------------------------------------------------// 应用层开堆// -------------------------------------------------------NTSTATUS Status = AllocMemory(process_id, create_size, &ref_address);DbgPrint("对端进程: %d \n", process_id);DbgPrint("分配长度: %d \n", create_size);DbgPrint("分配的内核堆基址: %p \n", ref_address);// 设置注入路径,转换为多字节UCHAR DllPath[256] = "C:\\lyshark_hook.dll";UCHAR Item[256] = { 0 };for (int x = 0, y = 0; x < strlen(DllPath) * 2; x += 2, y++){Item[x] = DllPath[y];}// -------------------------------------------------------// 写出数据到内存// -------------------------------------------------------ReadMemoryStruct ptr;ptr.pid = process_id;ptr.address = ref_address;ptr.size = strlen(DllPath) * 2;// 需要写入的数据ptr.data = ExAllocatePool(PagedPool, ptr.size);// 循环设置for (int i = 0; i < ptr.size; i++){ptr.data[i] = Item[i];}// 写内存MDLWriteMemory(&ptr);// -------------------------------------------------------// 执行开线程函数// -------------------------------------------------------// 执行线程注入// 参数1:PID// 参数2:LoadLibraryW内存地址// 参数3:当前DLL路径BOOLEAN flag = MyCreateRemoteThread(process_id, pLoadLibraryW, ref_address);if (flag == TRUE){DbgPrint("[*] 已完成进程 %d 注入文件 %s \n", process_id, DllPath);}DriverObject->DriverUnload = Unload;return STATUS_SUCCESS;
}

编译这段驱动程序,并将其放入虚拟机中,在C盘下面放置好一个名为lyshark_hook.dll文件,运行驱动程序将自动插入DLL到Win32Project进程内,输出效果图如下所示;

回到应用层进程,则可看到如下图所示的注入成功提示信息;

这篇关于驱动开发:内核LoadLibrary实现DLL注入的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/396322

相关文章

使用Python实现矢量路径的压缩、解压与可视化

《使用Python实现矢量路径的压缩、解压与可视化》在图形设计和Web开发中,矢量路径数据的高效存储与传输至关重要,本文将通过一个Python示例,展示如何将复杂的矢量路径命令序列压缩为JSON格式,... 目录引言核心功能概述1. 路径命令解析2. 路径数据压缩3. 路径数据解压4. 可视化代码实现详解1

PyQt6/PySide6中QTableView类的实现

《PyQt6/PySide6中QTableView类的实现》本文主要介绍了PyQt6/PySide6中QTableView类的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学... 目录1. 基本概念2. 创建 QTableView 实例3. QTableView 的常用属性和方法

PyQt6/PySide6中QTreeView类的实现

《PyQt6/PySide6中QTreeView类的实现》QTreeView是PyQt6或PySide6库中用于显示分层数据的控件,本文主要介绍了PyQt6/PySide6中QTreeView类的实现... 目录1. 基本概念2. 创建 QTreeView 实例3. QTreeView 的常用属性和方法属性

Android使用ImageView.ScaleType实现图片的缩放与裁剪功能

《Android使用ImageView.ScaleType实现图片的缩放与裁剪功能》ImageView是最常用的控件之一,它用于展示各种类型的图片,为了能够根据需求调整图片的显示效果,Android提... 目录什么是 ImageView.ScaleType?FIT_XYFIT_STARTFIT_CENTE

pandas中位数填充空值的实现示例

《pandas中位数填充空值的实现示例》中位数填充是一种简单而有效的方法,用于填充数据集中缺失的值,本文就来介绍一下pandas中位数填充空值的实现,具有一定的参考价值,感兴趣的可以了解一下... 目录什么是中位数填充?为什么选择中位数填充?示例数据结果分析完整代码总结在数据分析和机器学习过程中,处理缺失数

Golang HashMap实现原理解析

《GolangHashMap实现原理解析》HashMap是一种基于哈希表实现的键值对存储结构,它通过哈希函数将键映射到数组的索引位置,支持高效的插入、查找和删除操作,:本文主要介绍GolangH... 目录HashMap是一种基于哈希表实现的键值对存储结构,它通过哈希函数将键映射到数组的索引位置,支持

Pandas使用AdaBoost进行分类的实现

《Pandas使用AdaBoost进行分类的实现》Pandas和AdaBoost分类算法,可以高效地进行数据预处理和分类任务,本文主要介绍了Pandas使用AdaBoost进行分类的实现,具有一定的参... 目录什么是 AdaBoost?使用 AdaBoost 的步骤安装必要的库步骤一:数据准备步骤二:模型

使用Pandas进行均值填充的实现

《使用Pandas进行均值填充的实现》缺失数据(NaN值)是一个常见的问题,我们可以通过多种方法来处理缺失数据,其中一种常用的方法是均值填充,本文主要介绍了使用Pandas进行均值填充的实现,感兴趣的... 目录什么是均值填充?为什么选择均值填充?均值填充的步骤实际代码示例总结在数据分析和处理过程中,缺失数

Java对象转换的实现方式汇总

《Java对象转换的实现方式汇总》:本文主要介绍Java对象转换的多种实现方式,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录Java对象转换的多种实现方式1. 手动映射(Manual Mapping)2. Builder模式3. 工具类辅助映

Go语言开发实现查询IP信息的MCP服务器

《Go语言开发实现查询IP信息的MCP服务器》随着MCP的快速普及和广泛应用,MCP服务器也层出不穷,本文将详细介绍如何在Go语言中使用go-mcp库来开发一个查询IP信息的MCP... 目录前言mcp-ip-geo 服务器目录结构说明查询 IP 信息功能实现工具实现工具管理查询单个 IP 信息工具的实现服