windbg通过qemu的gdbserver调试虚机windows的方法

2023-12-24 23:20

本文主要是介绍windbg通过qemu的gdbserver调试虚机windows的方法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1. 概述

1.1. 原理

通过qemu-system程序启动的虚机,命令行里一般带有monitor。连接上这个monitor,可以启动gdbserver。通过此gdbserver,可以查看虚机的cpu寄存器,虚拟地址内存等。同时,qemu-system提供了一个tcpserver。另一台机的windbg可以连接上这个tcpserver,进而调试虚机。

1.2. 推荐场景

虚机windows未开启调试,但是虚机偶发的卡死发生时。

1.3. 优劣

优势:

A. 不需被调试机bcdedit事先配置

B. 若开启嵌套虚拟化,还能观察L1虚机的hvix64的代码运行

缺陷和限制:

仅限于qemu-system启动的虚机,tci产品不支持。

windbg里定位pe模块比较复杂。

2. qemu主机配置

2.1. 以新平台的idv为例

在qemu-system的命令行中可以看到-qmp unix:/tmp/vm.monitor,server,nowait,由此,在linux shell里执行qmp-shell -H /tmp/vm.monitor即可进入qmp的交互终端。

输入gdbserver tcp::35100得到Waiting for gdb connection on device 'tcp::35100'即表明gdbserver开启成功。gdbserver默认端口1234,会被ranios的防火墙拦截。这里推荐端口选35100,它在rainos防火墙的白名单中。

退出qmp交互终端用Ctrl+d,不可用Ctrl+c,它会让qemu-system进程终止。

2.2 vdi

先找出vdi虚机的id和所在运行服务器。ssh连上服务器,先找出虚机对应的domain:

virsh list|grep 120fb070-6b04-4766-b534-38e3131a6c11

得到315   120fb070-6b04-4766-b534-38e3131a6c11   running

遂用libvirt打开它的gdbserver

virsh qemu-monitor-command 315 --hmp gdbserver tcp::35900

3. 调试机配置

参考使用 EXDI 设置 QEMU 内核模式调试

ExdiGdbSrv.dll(64位),exdiConfigData.xml,systemregisters.xml我已集成到\\rcc.ruijie.net\Upload\lida\Microsoft.WinDbg_1.2202.7001.0.zip

设置环境变量:

EXDI_GDBSRV_XML_CONFIG_FILE为"C:\Program Files (x86)\Windows Kits\10\Debuggers\x64\exdiConfigData.xml"

EXDI_SYSTEM_REGISTERS_MAP_XML_FILE为"C:\Program Files (x86)\Windows Kits\10\Debuggers\x64\systemregisters.xml"

解压此zip,在里面有这个dll和两个xml。以管理员权限执行如下:

注册dll:

regsvr32 ExdiGdbSrv.dll

编辑exdiconfigData.xml,找到<  = "QEMU">里的<Value HostNameAndPort=",这里的值改为qemu主机的ip和gdbserver的port

cmd进入windbg路径,执行.\dbgx.shell.exe -v -kx exdi:CLSID={29f9906e-9dbe-4d4b-b0fb-6acf7fb6d014},Kd=Guess,DataBreaks=Exdi,随后能看到新的cmd窗口和windbg preview的窗口出现。在windbg中看到Connected to eXDI Device 0 x64 target at (即可认为连上了被调试机

4. windbg内定位pe模块

4.1 定位nt(ntoskrnl.exe)模块

在windbg里输入Ctrl+break或已经发现windbg暂停了target机,输入k,lm,.reload等命令发现没有任何模块被识别。具体原因不明。但是我们可以自行帮windbg定位符号。暂停时,通过r rip查看rip寄存器,若为fffff802`3c7f5e8f前5位是fffff如此字样则表示暂停在内核态,若为00007ffa`f2ebc742前四位是0如此字样则表明暂停在用户态,此时可g再暂停或~1s切换一个cpu线程,直到rip走到内核态。

以rip=fffff802`3c7f5e8f为例,猜测这段地址处于nt模块中,而nt模块的载入地址一般是0x100000的整数倍。用windbg的!dh来解析这个pe头,挨个输入700000,600000,500000……直到解析成功

0: kd> !dh fffff8023c700000
No file header
0: kd> !dh fffff8023c600000
No file header
0: kd> !dh fffff8023c500000
No file header
0: kd> !dh fffff8023c400000

File Type: EXECUTABLE IMAGE
FILE HEADER VALUES
    8664 machine (X64)
      21 number of sections
C3F2329E time date stamp Mon Mar  5 06:56:30 2074

如上,表明fffff8023c400000是个pe头位置。翻阅pe信息,查看

OPTIONAL HEADER VALUES
     20B magic #
   14.20 linker version
  8BB400 size of code
  1B7400 size of initialized data
  48E000 size of uninitialized data
  98F010 address of entry point
    1000 base of code
         ----- new -----
fffff8023c400000 image base
    1000 section alignment
     200 file alignment
       1 subsystem (Native)
   10.00 operating system version
   10.00 image version
   10.00 subsystem version
 1046000 size of image

可知这个pe文件占据空间0x1046000字节。

查看

Debug Directories(3)
        Type       Size     Address  Pointer
        cv           25       406e0    3fee0        Format: RSDS, guid, 1, ntkrnlmp.pdb

可知这个pe文件是nt。

如此,可用.reload指定这段空间为nt:.reload /f nt=fffff802`3c400000,1046000

之后,可以使用!process 0 0查看所有进程。

4.2 lm后又失去所有模块的指定了。注意不要随便lm。原因未知

4.3 定位用户态dll的位置

例如给新进程的诞生处下断点:bp nt!PspInsertProcess;g。待断点触发时,看到栈回溯是:

Breakpoint 0 hit
nt!PspInsertProcess:
0010:fffff8023cae52a4 488bc4          mov     rax,rsp
0: kd> k
 # Child-SP          RetAddr               Call Site
00 ffff85052613aeb8 fffff8023cac9028     nt!PspInsertProcess
01 ffff85052613aec0 fffff8023c80d8f5     nt!NtCreateUserProcess+0xd88
02 ffff85052613bb90 00007ffaf2ece9b4     nt!KiSystemServiceCopyEnd+0x25
03 00000062d50fcd08 00007ffaf0529053     0x00007ffaf2ece9b4
04 00000062d50fcd10 0000000000000000     0x00007ffa`f0529053

看RetAddr,猜测地址7ffaf2ece9b4位于用户态,而且一般是ntdll.dll或kernelbase.dll。输入!dh 7ffa`f0520000,果然它是kernelbase.dll:

0: kd> !dh 7ffa`f0520000

File Type: DLL
FILE HEADER VALUES
    8664 machine (X64)
       7 number of sections
E8E9AC9B time date stamp Thu Oct 29 12:16:27 2093

……

  2D2000 size of image

……

Debug Directories(4)
        Type       Size     Address  Pointer
        cv           27      26a170   269370        Format: RSDS, guid, 1, kernelbase.pdb

如此可指定kernelbase的地址:.reload /f /user kernelbase=7ffa`f0520000,2D2000

输入k查看栈回溯:

0: kd> k
 # Child-SP          RetAddr               Call Site
00 ffff85052613aeb8 fffff8023cac9028     nt!PspInsertProcess
01 ffff85052613aec0 fffff8023c80d8f5     nt!NtCreateUserProcess+0xd88
02 ffff85052613bb90 00007ffaf2ece9b4     nt!KiSystemServiceCopyEnd+0x25
03 00000062d50fcd08 00007ffaf0529053     0x00007ffaf2ece9b4
04 00000062d50fcd10 00007ffaf05265e3     kernelbase!CreateProcessInternalW+0xfe3
05 00000062d50fe2e0 00007ffaf26ee2e0     kernelbase!CreateProcessAsUserW+0x63
06 00000062d50fe350 0000000000000001     0x00007ffaf26ee2e0
07 00000062d50fe358 00007ffaee361e09     0x1
08 00000062d50fe360 00007ffaee488dd0     0x00007ffaee361e09
09 00000062d50fe368 00007ffaee3a3a48     0x00007ffaee488dd0
0a 00000062d50fe370 0000000000000000     0x00007ffa`ee3a3a48

猜测7ffaf2ece9b4位于ntdll,从7ffaf2ec0000逐个递减0x1000用!dh翻译,直到找出了pe头:

0: kd> !dh 7ffa`f2e30000

File Type: DLL
FILE HEADER VALUES
    8664 machine (X64)

……

  1F8000 size of image

……

Debug Directories(4)
        Type       Size     Address  Pointer
        cv           22      140160   13d360        Can't read debug data cb=0

这里Debug Directories无法显示pdb文件名,原因未知。强行把它指定为ntdll:

.reload /f /user ntdll=7ffa`f2e30000,1f8000

查看栈回溯:

 # Child-SP          RetAddr               Call Site
00 ffff85052613aeb8 fffff8023cac9028     nt!PspInsertProcess
01 ffff85052613aec0 fffff8023c80d8f5     nt!NtCreateUserProcess+0xd88
02 ffff85052613bb90 00007ffaf2ece9b4     nt!KiSystemServiceCopyEnd+0x25
03 00000062d50fcd08 00007ffaf0529053     ntdll!NtCreateUserProcess+0x14
04 00000062d50fcd10 00007ffaf05265e3     kernelbase!CreateProcessInternalW+0xfe3
05 00000062d50fe2e0 00007ffaf26ee2e0     kernelbase!CreateProcessAsUserW+0x63
06 00000062d50fe350 0000000000000001     0x00007ffaf26ee2e0
07 00000062d50fe358 00007ffaee361e09     0x1
08 00000062d50fe360 00007ffaee488dd0     0x00007ffaee361e09
09 00000062d50fe368 00007ffaee3a3a48     0x00007ffaee488dd0
0a 00000062d50fe370 0000000000000000     0x00007ffa`ee3a3a48

猜测7ffa`f26ee2e0地址位于user32.dll里。果然7ffa`f26d0000是个pe头。但是也是不显示pdb名。那强行指定吧:.reload /user /f user32=7ffa`f26d0000,BF000

5. 其它windbg命令

!running.crash.dump都不行。原因有待继续研究。

这篇关于windbg通过qemu的gdbserver调试虚机windows的方法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Python中提取文件名扩展名的多种方法实现

《Python中提取文件名扩展名的多种方法实现》在Python编程中,经常会遇到需要从文件名中提取扩展名的场景,Python提供了多种方法来实现这一功能,不同方法适用于不同的场景和需求,包括os.pa... 目录技术背景实现步骤方法一:使用os.path.splitext方法二:使用pathlib模块方法三

Python打印对象所有属性和值的方法小结

《Python打印对象所有属性和值的方法小结》在Python开发过程中,调试代码时经常需要查看对象的当前状态,也就是对象的所有属性和对应的值,然而,Python并没有像PHP的print_r那样直接提... 目录python中打印对象所有属性和值的方法实现步骤1. 使用vars()和pprint()2. 使

CSS实现元素撑满剩余空间的五种方法

《CSS实现元素撑满剩余空间的五种方法》在日常开发中,我们经常需要让某个元素占据容器的剩余空间,本文将介绍5种不同的方法来实现这个需求,并分析各种方法的优缺点,感兴趣的朋友一起看看吧... css实现元素撑满剩余空间的5种方法 在日常开发中,我们经常需要让某个元素占据容器的剩余空间。这是一个常见的布局需求

在Windows上使用qemu安装ubuntu24.04服务器的详细指南

《在Windows上使用qemu安装ubuntu24.04服务器的详细指南》本文介绍了在Windows上使用QEMU安装Ubuntu24.04的全流程:安装QEMU、准备ISO镜像、创建虚拟磁盘、配置... 目录1. 安装QEMU环境2. 准备Ubuntu 24.04镜像3. 启动QEMU安装Ubuntu4

Windows下C++使用SQLitede的操作过程

《Windows下C++使用SQLitede的操作过程》本文介绍了Windows下C++使用SQLite的安装配置、CppSQLite库封装优势、核心功能(如数据库连接、事务管理)、跨平台支持及性能优... 目录Windows下C++使用SQLite1、安装2、代码示例CppSQLite:C++轻松操作SQ

Python常用命令提示符使用方法详解

《Python常用命令提示符使用方法详解》在学习python的过程中,我们需要用到命令提示符(CMD)进行环境的配置,:本文主要介绍Python常用命令提示符使用方法的相关资料,文中通过代码介绍的... 目录一、python环境基础命令【Windows】1、检查Python是否安装2、 查看Python的安

基于Python实现一个Windows Tree命令工具

《基于Python实现一个WindowsTree命令工具》今天想要在Windows平台的CMD命令终端窗口中使用像Linux下的tree命令,打印一下目录结构层级树,然而还真有tree命令,但是发现... 目录引言实现代码使用说明可用选项示例用法功能特点添加到环境变量方法一:创建批处理文件并添加到PATH1

Maven 配置中的 <mirror>绕过 HTTP 阻断机制的方法

《Maven配置中的<mirror>绕过HTTP阻断机制的方法》:本文主要介绍Maven配置中的<mirror>绕过HTTP阻断机制的方法,本文给大家分享问题原因及解决方案,感兴趣的朋友一... 目录一、问题场景:升级 Maven 后构建失败二、解决方案:通过 <mirror> 配置覆盖默认行为1. 配置示

SpringBoot排查和解决JSON解析错误(400 Bad Request)的方法

《SpringBoot排查和解决JSON解析错误(400BadRequest)的方法》在开发SpringBootRESTfulAPI时,客户端与服务端的数据交互通常使用JSON格式,然而,JSON... 目录问题背景1. 问题描述2. 错误分析解决方案1. 手动重新输入jsON2. 使用工具清理JSON3.

使用jenv工具管理多个JDK版本的方法步骤

《使用jenv工具管理多个JDK版本的方法步骤》jenv是一个开源的Java环境管理工具,旨在帮助开发者在同一台机器上轻松管理和切换多个Java版本,:本文主要介绍使用jenv工具管理多个JD... 目录一、jenv到底是干啥的?二、jenv的核心功能(一)管理多个Java版本(二)支持插件扩展(三)环境隔