驱动开发:内核R3与R0内存映射拷贝

打印 上一主题 下一主题

主题 561|帖子 561|积分 1683

在上一篇博文《驱动开发:内核通过PEB得到进程参数》中我们通过使用KeStackAttachProcess附加进程的方式得到了该进程的PEB结构信息,本篇文章同样需要使用进程附加功能,但这次我们将实现一个更加有趣的功能,在某些情况下应用层与内核层需要共享一片内存区域通过这片区域可打通内核与应用层的隔离,此类功能的实现依附于MDL内存映射机制实现。
应用层(R3)数据映射到内核层(R0)
先来实现将R3内存数据拷贝到R0中,功能实现所调用的API如下:

  • IoAllocateMdl 该函数用于创建MDL(类似初始化)
  • MmProbeAndLockPages 用于锁定创建的地址其中UserMode代表用户层,IoReadAccess以读取的方式锁定
  • MmGetSystemAddressForMdlSafe 用于从MDL中得到映射内存地址
  • RtlCopyMemory 用于内存拷贝,将DstAddr应用层中的数据拷贝到pMappedSrc中
  • MmUnlockPages 拷贝结束后解锁pSrcMdl
  • IoFreeMdl 释放MDL
内存拷贝SafeCopyMemory_R3_to_R0函数封装代码如下:
  1. #include <ntifs.h>
  2. #include <windef.h>
  3. // 分配内存
  4. void* RtlAllocateMemory(BOOLEAN InZeroMemory, SIZE_T InSize)
  5. {
  6.         void* Result = ExAllocatePoolWithTag(NonPagedPool, InSize, 'lysh');
  7.         if (InZeroMemory && (Result != NULL))
  8.                 RtlZeroMemory(Result, InSize);
  9.         return Result;
  10. }
  11. // 释放内存
  12. void RtlFreeMemory(void* InPointer)
  13. {
  14.         ExFreePool(InPointer);
  15. }
  16. /*
  17. 将应用层中的内存复制到内核变量中
  18. SrcAddr  r3地址要复制
  19. DstAddr  R0申请的地址
  20. Size     拷贝长度
  21. */
  22. NTSTATUS SafeCopyMemory_R3_to_R0(ULONG_PTR SrcAddr, ULONG_PTR DstAddr, ULONG Size)
  23. {
  24.         NTSTATUS status = STATUS_UNSUCCESSFUL;
  25.         ULONG nRemainSize = PAGE_SIZE - (SrcAddr & 0xFFF);
  26.         ULONG nCopyedSize = 0;
  27.         if (!SrcAddr || !DstAddr || !Size)
  28.         {
  29.                 return status;
  30.         }
  31.         while (nCopyedSize < Size)
  32.         {
  33.                 PMDL pSrcMdl = NULL;
  34.                 PVOID pMappedSrc = NULL;
  35.                 if (Size - nCopyedSize < nRemainSize)
  36.                 {
  37.                         nRemainSize = Size - nCopyedSize;
  38.                 }
  39.                 // 创建MDL
  40.                 pSrcMdl = IoAllocateMdl((PVOID)(SrcAddr & 0xFFFFFFFFFFFFF000), PAGE_SIZE, FALSE, FALSE, NULL);
  41.                 if (pSrcMdl)
  42.                 {
  43.                         __try
  44.                         {
  45.                                 // 锁定内存页面(UserMode代表应用层)
  46.                                 MmProbeAndLockPages(pSrcMdl, UserMode, IoReadAccess);
  47.                                 // 从MDL中得到映射内存地址
  48.                                 pMappedSrc = MmGetSystemAddressForMdlSafe(pSrcMdl, NormalPagePriority);
  49.                         }
  50.                         __except (EXCEPTION_EXECUTE_HANDLER)
  51.                         {
  52.                         }
  53.                 }
  54.                 if (pMappedSrc)
  55.                 {
  56.                         __try
  57.                         {
  58.                                 // 将MDL中的映射拷贝到pMappedSrc内存中
  59.                                 RtlCopyMemory((PVOID)DstAddr, (PVOID)((ULONG_PTR)pMappedSrc + (SrcAddr & 0xFFF)), nRemainSize);
  60.                         }
  61.                         __except (1)
  62.                         {
  63.                                 // 拷贝内存异常
  64.                         }
  65.                         // 释放锁
  66.                         MmUnlockPages(pSrcMdl);
  67.                 }
  68.                 if (pSrcMdl)
  69.                 {
  70.                         // 释放MDL
  71.                         IoFreeMdl(pSrcMdl);
  72.                 }
  73.                 if (nCopyedSize)
  74.                 {
  75.                         nRemainSize = PAGE_SIZE;
  76.                 }
  77.                 nCopyedSize += nRemainSize;
  78.                 SrcAddr += nRemainSize;
  79.                 DstAddr += nRemainSize;
  80.         }
  81.         status = STATUS_SUCCESS;
  82.         return status;
  83. }
复制代码
调用该函数实现拷贝,如下代码中首先PsLookupProcessByProcessId得到进程EProcess结构,并KeStackAttachProcess附加进程,声明pTempBuffer指针用于存储RtlAllocateMemory开辟的内存空间,nSize则代表读取应用层进程数据长度,ModuleBase则是读入进程基址,调用SafeCopyMemory_R3_to_R0即可将应用层数据拷贝到内核空间,并最终BYTE* data转为BYTE字节的方式输出。
  1. VOID UnDriver(PDRIVER_OBJECT driver)
  2. {
  3.         DbgPrint(("Uninstall Driver Is OK \n"));
  4. }
  5. // lyshark.com
  6. NTSTATUS DriverEntry(IN PDRIVER_OBJECT Driver, PUNICODE_STRING RegistryPath)
  7. {
  8.         DbgPrint("hello lyshark.com \n");
  9.         NTSTATUS status = STATUS_UNSUCCESSFUL;
  10.         PEPROCESS eproc = NULL;
  11.         KAPC_STATE kpc = { 0 };
  12.         __try
  13.         {
  14.                 // HANDLE 进程PID
  15.                 status = PsLookupProcessByProcessId((HANDLE)4556, &eproc);
  16.                 if (NT_SUCCESS(status))
  17.                 {
  18.                         // 附加进程
  19.                         KeStackAttachProcess(eproc, &kpc);
  20.                         // -------------------------------------------------------------------
  21.                         // 开始映射
  22.                         // -------------------------------------------------------------------
  23.                         // 将用户空间内存映射到内核空间
  24.                         PVOID pTempBuffer = NULL;
  25.                         ULONG nSize = 0x1024;
  26.                         ULONG_PTR ModuleBase = 0x0000000140001000;
  27.                         // 分配内存
  28.                         pTempBuffer = RtlAllocateMemory(TRUE, nSize);
  29.                         if (pTempBuffer)
  30.                         {
  31.                                 // 拷贝数据到R0
  32.                                 status = SafeCopyMemory_R3_to_R0(ModuleBase, (ULONG_PTR)pTempBuffer, nSize);
  33.                                 if (NT_SUCCESS(status))
  34.                                 {
  35.                                         DbgPrint("[*] 拷贝应用层数据到内核里 \n");
  36.                                 }
  37.                                 // 转成BYTE方便读取
  38.                                 BYTE* data = pTempBuffer;
  39.                                 for (size_t i = 0; i < 10; i++)
  40.                                 {
  41.                                         DbgPrint("%02X \n", data[i]);
  42.                                 }
  43.                         }
  44.                         // 释放空间
  45.                         RtlFreeMemory(pTempBuffer);
  46.                         // 脱离进程
  47.                         KeUnstackDetachProcess(&kpc);
  48.                 }
  49.         }
  50.         __except (EXCEPTION_EXECUTE_HANDLER)
  51.         {
  52.                 Driver->DriverUnload = UnDriver;
  53.                 return STATUS_SUCCESS;
  54.         }
  55.         Driver->DriverUnload = UnDriver;
  56.         return STATUS_SUCCESS;
  57. }
复制代码
代码运行后即可将进程中0x0000000140001000处的数据读入内核空间并输出:


内核层(R0)数据映射到应用层(R3)
与上方功能实现相反SafeCopyMemory_R0_to_R3函数则用于将一个内核层中的缓冲区写出到应用层中,写出过程:

  • IoAllocateMdl 分别调用MDL分配,源地址SrcAddr目标地址DstAddr均创建
  • MmBuildMdlForNonPagedPool 该 MDL 指定非分页虚拟内存缓冲区,并对其进行更新以描述基础物理页
  • MmGetSystemAddressForMdlSafe 调用两次得到源地址,分别获取pSrcMdl,pDstMdl两个MDL的
  • MmProbeAndLockPages 以写入方式锁定用户层中pDstMdl的地址
内存拷贝SafeCopyMemory_R0_to_R3函数封装代码如下:
  1. // 分配内存
  2. void* RtlAllocateMemory(BOOLEAN InZeroMemory, SIZE_T InSize)
  3. {
  4.         void* Result = ExAllocatePoolWithTag(NonPagedPool, InSize, 'lysh');
  5.         if (InZeroMemory && (Result != NULL))
  6.                 RtlZeroMemory(Result, InSize);
  7.         return Result;
  8. }
  9. // 释放内存
  10. void RtlFreeMemory(void* InPointer)
  11. {
  12.         ExFreePool(InPointer);
  13. }
  14. /*
  15. 将内存中的数据复制到R3中
  16. SrcAddr  R0要复制的地址
  17. DstAddr  返回R3的地址
  18. Size     拷贝长度
  19. */
  20. NTSTATUS SafeCopyMemory_R0_to_R3(PVOID SrcAddr, PVOID DstAddr, ULONG Size)
  21. {
  22.         PMDL  pSrcMdl = NULL, pDstMdl = NULL;
  23.         PUCHAR pSrcAddress = NULL, pDstAddress = NULL;
  24.         NTSTATUS st = STATUS_UNSUCCESSFUL;
  25.         // 分配MDL 源地址
  26.         pSrcMdl = IoAllocateMdl(SrcAddr, Size, FALSE, FALSE, NULL);
  27.         if (!pSrcMdl)
  28.         {
  29.                 return st;
  30.         }
  31.         // 该 MDL 指定非分页虚拟内存缓冲区,并对其进行更新以描述基础物理页。
  32.         MmBuildMdlForNonPagedPool(pSrcMdl);
  33.         // 获取源地址MDL地址
  34.         pSrcAddress = MmGetSystemAddressForMdlSafe(pSrcMdl, NormalPagePriority);
  35.         if (!pSrcAddress)
  36.         {
  37.                 IoFreeMdl(pSrcMdl);
  38.                 return st;
  39.         }
  40.         // 分配MDL 目标地址
  41.         pDstMdl = IoAllocateMdl(DstAddr, Size, FALSE, FALSE, NULL);
  42.         if (!pDstMdl)
  43.         {
  44.                 IoFreeMdl(pSrcMdl);
  45.                 return st;
  46.         }
  47.         __try
  48.         {
  49.                 // 以写入的方式锁定目标MDL
  50.                 MmProbeAndLockPages(pDstMdl, UserMode, IoWriteAccess);
  51.                 // 获取目标地址MDL地址
  52.                 pDstAddress = MmGetSystemAddressForMdlSafe(pDstMdl, NormalPagePriority);
  53.         }
  54.         __except (EXCEPTION_EXECUTE_HANDLER)
  55.         {
  56.         }
  57.         if (pDstAddress)
  58.         {
  59.                 __try
  60.                 {
  61.                         // 将源地址拷贝到目标地址
  62.                         RtlCopyMemory(pDstAddress, pSrcAddress, Size);
  63.                 }
  64.                 __except (1)
  65.                 {
  66.                         // 拷贝内存异常
  67.                 }
  68.                 MmUnlockPages(pDstMdl);
  69.                 st = STATUS_SUCCESS;
  70.         }
  71.         IoFreeMdl(pDstMdl);
  72.         IoFreeMdl(pSrcMdl);
  73.         return st;
  74. }
复制代码
调用该函数实现拷贝,此处除去附加进程以外,在拷贝之前调用了ZwAllocateVirtualMemory将内存属性设置为PAGE_EXECUTE_READWRITE可读可写可执行状态,然后在向该内存中写出pTempBuffer变量中的内容,此变量中的数据是0x90填充的区域。
  1. VOID UnDriver(PDRIVER_OBJECT driver)
  2. {
  3.         DbgPrint(("Uninstall Driver Is OK \n"));
  4. }
  5. // lyshark.com
  6. NTSTATUS DriverEntry(IN PDRIVER_OBJECT Driver, PUNICODE_STRING RegistryPath)
  7. {
  8.         DbgPrint("hello lyshark.com \n");
  9.         NTSTATUS status = STATUS_UNSUCCESSFUL;
  10.         PEPROCESS eproc = NULL;
  11.         KAPC_STATE kpc = { 0 };
  12.         __try
  13.         {
  14.                 // HANDLE 进程PID
  15.                 status = PsLookupProcessByProcessId((HANDLE)4556, &eproc);
  16.                 if (NT_SUCCESS(status))
  17.                 {
  18.                         // 附加进程
  19.                         KeStackAttachProcess(eproc, &kpc);
  20.                         // -------------------------------------------------------------------
  21.                         // 开始映射
  22.                         // -------------------------------------------------------------------
  23.                         // 将用户空间内存映射到内核空间
  24.                         PVOID pTempBuffer = NULL;
  25.                         ULONG nSize = 0x1024;
  26.                         PVOID ModuleBase = 0x0000000140001000;
  27.                         // 分配内存
  28.                         pTempBuffer = RtlAllocateMemory(TRUE, nSize);
  29.                         if (pTempBuffer)
  30.                         {
  31.                                 memset(pTempBuffer, 0x90, nSize);
  32.                                 // 设置内存属性 PAGE_EXECUTE_READWRITE
  33.                                 ZwAllocateVirtualMemory(NtCurrentProcess(), &ModuleBase, 0, &nSize, MEM_RESERVE, PAGE_EXECUTE_READWRITE);
  34.                                 ZwAllocateVirtualMemory(NtCurrentProcess(), &ModuleBase, 0, &nSize, MEM_COMMIT, PAGE_EXECUTE_READWRITE);
  35.                                 // 将数据拷贝到R3中
  36.                                 status = SafeCopyMemory_R0_to_R3(pTempBuffer, &ModuleBase, nSize);
  37.                                 if (NT_SUCCESS(status))
  38.                                 {
  39.                                         DbgPrint("[*] 拷贝内核数据到应用层 \n");
  40.                                 }
  41.                         }
  42.                         // 释放空间
  43.                         RtlFreeMemory(pTempBuffer);
  44.                         // 脱离进程
  45.                         KeUnstackDetachProcess(&kpc);
  46.                 }
  47.         }
  48.         __except (EXCEPTION_EXECUTE_HANDLER)
  49.         {
  50.                 Driver->DriverUnload = UnDriver;
  51.                 return STATUS_SUCCESS;
  52.         }
  53.         Driver->DriverUnload = UnDriver;
  54.         return STATUS_SUCCESS;
  55. }
复制代码
拷贝成功后,应用层进程内将会被填充为Nop指令。


免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

去皮卡多

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表