4.5 Windows驱动开发:内核中实现进程数据转储

多数ark反内核工具中都存在驱动级别的内存转存功能,该功能可以将应用层中运行进程的内存镜像转存到特定目录下,内存转存功能在应对加壳程序的分析尤为重要,当进程在内存中解码后,我们可以很容易的将内存镜像导出,从而更好的对样本进行分析,当然某些加密壳可能无效但绝大多数情况下是可以被转存的。

4.5 Windows驱动开发:内核中实现进程数据转储

在上一篇文章

《内核R3与R0内存映射拷贝》

介绍了一种方式

SafeCopyMemory_R3_to_R0

可以将应用层进程的内存空间映射到内核中,要实现内存转储功能我们还是需要使用这个映射函数,只是需要在此函数上增加一些功能而已。

在实现转存之前,需要得到两个东西,进程内

模块基地址

以及

模块长度

这两个参数是必不可少的,至于内核中如何得到指定进程的模块数据,在很早之前的文章

《内核中枚举进线程与模块》

中有详细的参考方法,这里就在此基础之上实现一个简单的进程模块遍历功能。

如下代码中使用的就是

枚举

进程

PEB

结构得到更多参数的具体实现,如果不懂得可以研读

《内核通过PEB得到进程参数》

这篇文章此处不再赘述。

代码语言:javascript代码运行次数:0运行复制

#include #include // 声明结构体typedef struct _KAPC_STATE{    LIST_ENTRY ApcListHead[2];    PKPROCESS Process;    UCHAR KernelApcInProgress;    UCHAR KernelApcPending;    UCHAR UserApcPending;} KAPC_STATE, *PKAPC_STATE;typedef struct _LDR_DATA_TABLE_ENTRY{    LIST_ENTRY64    InLoadOrderLinks;    LIST_ENTRY64    InMemoryOrderLinks;    LIST_ENTRY64    InInitializationOrderLinks;    PVOID           DllBase;    PVOID           EntryPoint;    ULONG           SizeOfImage;    UNICODE_STRING  FullDllName;    UNICODE_STRING  BaseDllName;    ULONG           Flags;    USHORT          LoadCount;    USHORT          TlsIndex;    PVOID           SectionPointer;    ULONG           CheckSum;    PVOID           LoadedImports;    PVOID           EntryPointActivationContext;    PVOID           PatchInformation;    LIST_ENTRY64    ForwarderLinks;    LIST_ENTRY64    ServiceTagLinks;    LIST_ENTRY64    StaticLinks;    PVOID           ContextInformation;    ULONG64         OriginalBase;    LARGE_INTEGER   LoadTime;} LDR_DATA_TABLE_ENTRY, *PLDR_DATA_TABLE_ENTRY;// 偏移地址ULONG64 LdrInPebOffset = 0x018;     //peb.ldrULONG64 ModListInPebOffset = 0x010; //peb.ldr.InLoadOrderModuleList// 声明APINTKERNELAPI UCHAR* PsGetProcessImageFileName(IN PEPROCESS Process);NTKERNELAPI PPEB PsGetProcessPeb(PEPROCESS Process);NTKERNELAPI HANDLE PsGetProcessInheritedFromUniqueProcessId(IN PEPROCESS Process);// 根据进程ID返回进程EPROCESS,失败返回NULLPEPROCESS LookupProcess(HANDLE Pid){    PEPROCESS eprocess = NULL;    if (NT_SUCCESS(PsLookupProcessByProcessId(Pid, &eprocess)))        return eprocess;    else        return NULL;}// 枚举指定进程的模块// By: LyShark.comVOID EnumModule(PEPROCESS Process){    SIZE_T Peb = 0;    SIZE_T Ldr = 0;    PLIST_ENTRY ModListHead = 0;    PLIST_ENTRY Module = 0;    ANSI_STRING AnsiString;    KAPC_STATE ks;        // EPROCESS地址无效则退出    if (!MmIsAddressValid(Process))        return;        // 获取PEB地址    Peb = (SIZE_T)PsGetProcessPeb(Process);        // PEB地址无效则退出    if (!Peb)        return;        // 依附进程    KeStackAttachProcess(Process, &ks);    __try    {        // 获得LDR地址        Ldr = Peb + (SIZE_T)LdrInPebOffset;        // 测试是否可读,不可读则抛出异常退出        ProbeForRead((CONST PVOID)Ldr, 8, 8);        // 获得链表头        ModListHead = (PLIST_ENTRY)(*(PULONG64)Ldr + ModListInPebOffset);        // 再次测试可读性        ProbeForRead((CONST PVOID)ModListHead, 8, 8);        // 获得第一个模块的信息        Module = ModListHead->Flink;                while (ModListHead != Module)        {            //打印信息:基址、大小、DLL路径            DbgPrint("模块基址 = %p | 大小 = %ld | 模块名 = %wZ | 完整路径= %wZ ",                (PVOID)(((PLDR_DATA_TABLE_ENTRY)Module)->DllBase),                (ULONG)(((PLDR_DATA_TABLE_ENTRY)Module)->SizeOfImage),                &(((PLDR_DATA_TABLE_ENTRY)Module)->BaseDllName),                &(((PLDR_DATA_TABLE_ENTRY)Module)->FullDllName)                );            Module = Module->Flink;                        // 测试下一个模块信息的可读性            ProbeForRead((CONST PVOID)Module, 80, 8);        }    }    __except (EXCEPTION_EXECUTE_HANDLER){ ; }        // 取消依附进程    KeUnstackDetachProcess(&ks);}VOID DriverUnload(IN PDRIVER_OBJECT DriverObject){}NTSTATUS DriverEntry(IN PDRIVER_OBJECT DriverObject, IN PUNICODE_STRING RegistryPath){    DbgPrint("hello lyshark.com ");    ULONG i = 0;    PEPROCESS eproc = NULL;    for (i = 4; iDriverUnload = DriverUnload;    return STATUS_SUCCESS;}

如上我们指定获取应用层

lyshark.exe

进程的模块信息,并可得到以下输出效果:

4.5 Windows驱动开发:内核中实现进程数据转储

上篇文章中的代码就不再啰嗦了,这里只给出内存转存的核心代码

ProcessDumps

的实现流程:

ProcessDumps 代码的功能是将一个进程的内存空间转储(Dump)到磁盘上的一个文件中,该函数接收三个参数,并返回内存转存的状态;

Trae国内版 Trae国内版

国内首款AI原生IDE,专为中国开发者打造

Trae国内版 815 查看详情 Trae国内版 参数 pEprocess:要转储的进程的PEPROCESS结构体指针。参数 nBase:要转储的内存空间的基地址。参数 nSize:要转储的内存空间的大小。函数返回值:转储操作的状态,如果成功则返回 STATUS_SUCCESS,否则返回一个表示错误原因的 NTSTATUS 值。

该函数的实现也非常简单,通过

SafeCopyMemory_R3_to_R0

函数将应用层中的进程内存映射到内核层中的

pBuffer

堆中,当映射完成后再通过

ZwWriteFile

函数将这段内存写出到磁盘中完成转存,函数

ProcessDumps

的具体流程如下:

1.检查参数 pEprocess 和 nSize 是否为 NULL 或为 0,如果是,则直接返回 STATUS_UNSUCCESSFUL,表示操作失败。2.分配一个大小为 nSize 的缓冲区,用于存储要转储的内存空间。3.如果要转储的进程不是当前进程,则将当前线程切换到要转储的进程的上下文中,以便能够访问要转储的进程的内存空间。4.调用函数 SafeCopyMemory_R3_to_R0,将要转储的内存空间中的数据复制到缓冲区中。5.如果线程被切换到了要转储的进程的上下文中,则将线程切换回当前进程的上下文中。6.调用ZwCreateFile创建一个表示输出文件的句柄。7.通过ZwWriteFile将缓冲区中的数据写入到输出文件中。8.最后ZwClose关闭输出文件句柄并释放缓冲区内存。

很简单只是利用了

SafeCopyMemory_R3_to_R0

将进程内存读取到缓冲区内,并将缓冲区写出到C盘目录下,默认将转存数据保存为

lyshark_dumps.exe

代码语言:javascript代码运行次数:0运行复制

NTSTATUS ProcessDumps(PEPROCESS pEprocess, ULONG_PTR nBase, ULONG nSize){    BOOLEAN bAttach = FALSE;    KAPC_STATE ks = { 0 };    PVOID pBuffer = NULL;    NTSTATUS status = STATUS_UNSUCCESSFUL;    if (nSize == 0 || pEprocess == NULL)    {        return status;    }    pBuffer = ExAllocatePoolWithTag(PagedPool, nSize, 'lysh');    if (!pBuffer)    {        return status;    }    memset(pBuffer, 0, nSize);    if (pEprocess != IoGetCurrentProcess())    {        KeStackAttachProcess(pEprocess, &ks);        bAttach = TRUE;    }    status = SafeCopyMemory_R3_to_R0(nBase, (ULONG_PTR)pBuffer, nSize);    if (bAttach)    {        KeUnstackDetachProcess(&ks);        bAttach = FALSE;    }    OBJECT_ATTRIBUTES object;    IO_STATUS_BLOCK io;    HANDLE hFile;    UNICODE_STRING log;    // 导出文件名称    RtlInitUnicodeString(&log, L"??C:lyshark_dumps.exe");    InitializeObjectAttributes(&object, &log, OBJ_CASE_INSENSITIVE, NULL, NULL);    status = ZwCreateFile(&hFile,        GENERIC_WRITE,        &object,        &io,        NULL,        FILE_ATTRIBUTE_NORMAL,        FILE_SHARE_WRITE,        FILE_OPEN_IF,        FILE_SYNCHRONOUS_IO_NONALERT,        NULL,        0);    if (!NT_SUCCESS(status))    {        DbgPrint("打开文件错误 ");        return STATUS_SUCCESS;    }    ZwWriteFile(hFile, NULL, NULL, NULL, &io, pBuffer, nSize, NULL, NULL);    DbgPrint("写出字节数: %d ", io.Information);    DbgPrint("[*] LyShark.exe 已转存");    ZwClose(hFile);    if (pBuffer)    {        ExFreePoolWithTag(pBuffer, 'lysh');        pBuffer = NULL;    }    return status;}VOID UnDriver(PDRIVER_OBJECT driver){    DbgPrint(("Uninstall Driver Is OK "));}NTSTATUS DriverEntry(IN PDRIVER_OBJECT Driver, PUNICODE_STRING RegistryPath){    DbgPrint("hello lyshark.com ");    NTSTATUS ntStatus;    PEPROCESS pCurProcess = NULL;    __try    {        ntStatus = PsLookupProcessByProcessId((HANDLE)272, &pCurProcess);        if (NT_SUCCESS(ntStatus))        {            // 设置基地址以及长度            ntStatus = ProcessDumps(pCurProcess, 0x140000000, 1024);            ObDereferenceObject(pCurProcess);        }    }    __except (1)    {        ntStatus = GetExceptionCode();    }    Driver->DriverUnload = UnDriver;    return STATUS_SUCCESS;}

转存后效果如下图所示:

4.5 Windows驱动开发:内核中实现进程数据转储

至于导出的进程无法运行只是没有修复而已(后期会讲),可以打开看看是没错的。

4.5 Windows驱动开发:内核中实现进程数据转储

以上就是4.5 Windows驱动开发:内核中实现进程数据转储的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/411398.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
济南大学原长洲教授课题组AEM:微小高熵掺杂Na3V2(PO4)2O2F在钠离子电池中高效储钠的研究进展
上一篇 2025年11月6日 23:27:09
防止AO3访问失败 收藏这个AO3备用官方域名
下一篇 2025年11月6日 23:27:14

相关推荐

  • composer require-dev和require有什么不同_Composer Require与Require-Dev区别解析

    require用于声明项目运行必需的依赖,如框架、数据库组件和第三方SDK,这些包会随项目部署到生产环境;2. require-dev用于声明仅在开发和测试阶段需要的工具,如PHPUnit、PHPStan、Faker等,不会默认部署到生产环境;3. 安装时composer install根据环境决定…

    2026年5月10日
    1000
  • 修复Django电商项目中AJAX过滤产品列表图片不显示问题

    在Django电商项目中,当使用AJAX动态加载过滤后的产品列表时,常遇到图片无法正常显示的问题。这通常是由于前端模板中图片加载方式(如data-setbg属性结合JavaScript库)与AJAX动态内容更新机制不兼容所致。解决方案是直接在AJAX返回的HTML中使用标准的标签来渲染图片,确保浏览…

    2026年5月10日
    000
  • 利用海象运算符简化条件赋值:Python教程与最佳实践

    本文旨在探讨Python中海象运算符(:=)在条件赋值场景下的应用。通过对比传统if/else语句与海象运算符,以及条件表达式,分析海象运算符在简化代码、提高可读性方面的优势与局限性。并通过具体示例,展示如何在列表推导式等场景下合理使用海象运算符,同时强调其潜在的复杂性及替代方案,帮助开发者更好地掌…

    2026年5月10日
    100
  • Debian syslog性能优化技巧有哪些

    提升Debian系统syslog (通常基于rsyslog)性能,关键在于精简配置和高效处理日志。以下策略能有效优化日志管理,提升系统整体性能: 精简配置,高效加载: 在rsyslog配置文件中,仅加载必要的输入、输出和解析模块。 使用全局指令设置日志级别和格式,避免不必要的处理。 自定义模板: 创…

    2026年5月10日
    000
  • 怎么在PHP代码中实现图片上传功能_PHP图片上传功能实现与安全处理教程

    首先创建含enctype的HTML表单,再用PHP接收文件,检查目录、移动临时文件,验证类型与大小,生成唯一文件名,并调整php.ini限制以确保上传成功。 如果您尝试在PHP项目中添加图片上传功能,但服务器无法正确接收或保存文件,则可能是由于表单配置、文件处理逻辑或安全限制的问题。以下是实现该功能…

    2026年5月10日
    100
  • c++中的SFINAE技术是什么_c++模板编程中的SFINAE原理与应用

    SFINAE 是“替换失败不是错误”的原则,指模板实例化时若参数替换导致错误,只要存在其他合法候选,编译器不报错而是继续重载决议。它用于条件启用模板、类型检测等场景,如通过 decltype 或 enable_if 控制函数重载,实现类型特征判断。尽管 C++20 引入 Concepts 简化了部分…

    2026年5月10日
    000
  • 修复点击时按钮抖动:CSS垂直对齐实践

    本文探讨了在Web开发中,交互式按钮(如播放/暂停按钮)在点击时发生意外垂直位移的问题。通过分析CSS样式变化对元素布局的影响,我们发现这是由于按钮不同状态下的边框样式和内边距改变,以及默认的垂直对齐行为共同作用所致。核心解决方案是利用CSS的vertical-align属性,将其设置为middle…

    2026年5月10日
    100
  • Golang goroutine与channel调试技巧

    使用go run -race检测数据竞争,结合runtime.NumGoroutine监控协程数量,通过pprof分析阻塞调用栈,利用select超时避免永久阻塞,有效排查goroutine泄漏、死锁和数据竞争问题。 Go语言的goroutine和channel是并发编程的核心,但它们也带来了调试上…

    2026年5月10日
    000
  • 使用 Jupyter Notebook 进行探索性数据分析

    Jupyter Notebook通过单元格实现代码与Markdown结合,支持数据导入(pandas)、清洗(fillna)、探索(matplotlib/seaborn可视化)、统计分析(describe/corr)和特征工程,便于记录与分享分析过程。 Jupyter Notebook 是进行探索性…

    2026年5月10日
    000
  • 如何在HTML中插入表单元素_HTML表单控件与输入类型使用指南

    HTML表单通过标签构建,包含action和method属性定义数据提交目标与方式,常用input类型如text、password、email等适配不同输入需求,配合label、required、placeholder提升可用性,结合textarea、select、button等控件实现完整交互,是…

    2026年5月10日
    100
  • 前端缓存策略与JavaScript存储管理

    根据数据特性选择合适的存储方式并制定清晰的读写与清理逻辑,能显著提升前端性能;合理运用Cookie、localStorage、sessionStorage、IndexedDB及Cache API,结合缓存策略与定期清理机制,可在保证用户体验的同时避免安全与性能隐患。 前端缓存和JavaScript存…

    2026年5月10日
    200
  • 网站标题关键词更新后,搜索引擎为何仍显示旧标题?

    网站标题更新后,搜索引擎为何显示旧标题? 网站SEO优化中,站长常修改网站标题关键词,期望搜索结果显示自定义标题。然而,即使更新标签、meta keywords、meta description和结构化数据中的name属性后,搜索结果仍显示旧标题,这令人费解。本文将对此进行解释。 问题:站长修改了网…

    2026年5月10日
    100
  • HTML5网页如何实现手势操作 HTML5网页移动端交互的处理技巧

    首先利用原生touch事件实现滑动判断,再通过preventDefault解决滚动冲突,接着引入Hammer.js处理复杂手势,最后通过优化点击区域、避免事件冲突和增加视觉反馈提升体验。 在移动端浏览器中,HTML5网页可以通过触摸事件实现手势操作,提升用户体验。虽然原生JavaScript提供了基…

    2026年5月10日
    000
  • Python命令怎样使用profile分析脚本性能 Python命令性能分析的基础教程

    使用Python的cProfile模块分析脚本性能最直接的方式是通过命令行执行python -m cProfile your_script.py,它会输出每个函数的调用次数、总耗时、累积耗时等关键指标,帮助定位性能瓶颈;为进一步分析,可将结果保存为文件python -m cProfile -o ou…

    2026年5月10日
    000
  • 如何插入查询结果数据_SQL插入Select查询结果方法

    如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法

    使用INSERT INTO…SELECT语句可高效插入数据,通过NOT EXISTS、LEFT JOIN、MERGE语句或唯一约束避免重复;表结构不一致时可通过别名、类型转换、默认值或计算字段处理;结合存储过程可提升可维护性,支持参数化与动态SQL。 将查询结果数据插入到另一个表中,可以…

    2026年5月10日 用户投稿
    000
  • python中zip函数详解 python多序列压缩zip函数应用场景

    zip函数的应用场景包括:1) 同时遍历多个序列,2) 合并多个列表的数据,3) 数据分析和科学计算中的元素运算,4) 处理csv文件,5) 性能优化。zip函数是一个强大的工具,能够简化代码并提高处理多个序列时的效率。 在Python中,zip函数是一个非常有用的工具,它能够将多个可迭代对象打包成…

    2026年5月10日
    000
  • JavaScript 闭包:理解闭包原理与内存泄漏问题

    闭包是函数访问其外部作用域变量的能力,即使外部函数已执行完毕。如 inner 函数引用 outer 中的 count,形成闭包,使变量持久存在。闭包本身无害,但可能因延长变量生命周期导致内存泄漏,例如事件监听器引用大对象时。若未及时清理 DOM 事件或定时器,闭包会阻止垃圾回收,造成内存占用过高。解…

    2026年5月10日
    100
  • JavaScript 动态菜单点击高亮效果实现教程

    本教程详细介绍了如何使用 JavaScript 实现动态菜单的点击高亮功能。通过事件委托和状态管理,当用户点击菜单项时,被点击项会高亮显示(绿色),同时其他菜单项恢复默认样式(白色)。这种方法避免了不必要的DOM操作,提高了性能和代码可维护性,确保了无论点击方向如何,功能都能稳定运行。 动态菜单高亮…

    2026年5月10日
    200
  • c++如何实现UDP通信_c++基于UDP的网络通信示例

    UDP通信基于套接字实现,适用于实时性要求高的场景。1. 流程包括创建套接字、绑定地址(接收方)、发送(sendto)与接收(recvfrom)数据、关闭套接字;2. 服务端监听指定端口,接收客户端消息并回传;3. 客户端发送消息至服务端并接收响应;4. 跨平台需处理Winsock初始化与库链接,编…

    2026年5月10日
    100
  • 谷歌浏览器如何截图 谷歌浏览器页面截图技巧

    谷歌浏览器如何截图 谷歌浏览器页面截图技巧谷歌浏览器如何截图 谷歌浏览器页面截图技巧谷歌浏览器如何截图 谷歌浏览器页面截图技巧谷歌浏览器如何截图 谷歌浏览器页面截图技巧

    使用谷歌浏览器的开发者工具截图步骤:1. 按ctrl+shift+i(windows/linux)或cmd+option+i(mac)打开开发者工具。2. 点击右上角三个点,选择”更多工具”,再选择”截图”。3. 选择截取整个页面。推荐的谷歌浏览器扩展…

    2026年5月10日 用户投稿
    100

发表回复

登录后才能评论
关注微信