pdb文件名称被修改导致pdb文件加载失败的实战排查案例分享

本文主要是介绍pdb文件名称被修改导致pdb文件加载失败的实战排查案例分享,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

1、概述

2、问题说明

3、pdb文件加载失败的可能原因有哪些?

4、使用!sym noisy打开pdb加载详情,发现pdb文件名称确实被修改了

5、Windbg是如何知道要加载pdb文件名称的?


C++软件异常排查从入门到精通系列教程(专栏文章列表,欢迎订阅,持续更新...)icon-default.png?t=N7T8https://blog.csdn.net/chenlycly/article/details/125529931C/C++基础与进阶(专栏文章,持续更新中...)icon-default.png?t=N7T8https://blog.csdn.net/chenlycly/category_11931267.htmlVC++常用功能开发汇总(专栏文章列表,欢迎订阅,持续更新...)icon-default.png?t=N7T8https://blog.csdn.net/chenlycly/article/details/124272585C++软件分析工具从入门到精通案例集锦(专栏文章,持续更新中...)icon-default.png?t=N7T8https://blog.csdn.net/chenlycly/article/details/131405795开源组件及数据库技术(专栏文章,持续更新中...)icon-default.png?t=N7T8https://blog.csdn.net/chenlycly/category_12458859.html网络编程与网络问题分享(专栏文章,持续更新中...)icon-default.png?t=N7T8https://blog.csdn.net/chenlycly/category_2276111.html       最近技群一朋友在使用Windbg分析软件异常时遇到问题,pdb文件与二进制文件的版本(时间戳)明明是一致的,但Windbg就是加载不了pdb文件。于是找到我,让我帮他看看到底是怎么回事。本文大概地讲述一下这个pdb文件加载失败问题的排查过程,以供借鉴或参考。

1、概述

       我们用Windbg排查软件异常时,无论是静态分析dump文件,还是动态调试目标进程,如果要查看详细的函数调用堆栈(比如具体的函数名、代码的行号等),就需要加载堆栈中相关模块的pdb文件,需要使用到pdb文件中的函数、变量等符号信息。

       pdb(Program Database)文件是Windows平台用于存储程序调试信息的文件格式,pdb文件中包含了编译后的可执行文件或动态链接库的符号表、源代码文件路径、局部变量和全局变量的信息等。pdb文件通常与对应的可执行文件(如.exe或.dll)一起使用,以支持程序的调试和分析。

       pdb文件的主要特点和作用:

1)符号表信息:pdb文件包含了编译后的程序的符号表信息,包括函数名、类名、结构体名、变量名等。这些符号信息可以帮助调试器在调试过程中准确地定位和查看变量的值、函数的调用关系等。
2)源代码文件路径:pdb文件记录了源代码文件的路径信息,这使得调试器能够在需要时自动加载对应的源代码文件,以便开发人员在调试过程中查看和修改源代码。
3)局部变量和全局变量信息:pdb文件还包含了局部变量和全局变量的信息,包括变量名、类型和内存地址等。这些信息对于调试器来说很重要,它们使得开发人员可以在调试过程中查看和修改变量的值,有助于分析程序中的问题。
4)回溯栈信息:pdb文件中还包含了堆栈帧和函数调用关系的信息,这对于调试器来说非常重要。通过pdb文件,调试器可以在堆栈回溯时准确地还原函数调用的顺序,帮助开发人员理解程序的执行流程。

       pdb文件是Microsoft开发工具链(如Visual Studio)生成的调试信息文件。在进行程序的调试和分析时,通常需要将pdb文件与对应的可执行文件一起使用。调试器(比如Visual Studio、Windbg等)会根据pdb文件中存放的函数符号及变量符号等信息,来展示变量值、源代码行号、函数调用堆栈等相关信息,以供调试使用。

       我们使用Windbg分析软件异常时,有时不仅要查看详细的函数调用堆栈信息,甚至还需要查看相关变量的值,这些都需要使用到pdb文件中的函数及变量的符号。

2、问题说明

       已经获取了包含异常上下文的dump文件,并拿来了exe主程序及对应时间点的pdb文件,如下所示:(dump文件比较大,是手动导出的全dump文件)

但用Windbg打开dump文件并给Windbg设置pdb路径后,一直加载不了exe主程序的pdb文件,使用.reload强制加载pdb文件也不行。

       用Windbg打开dump文件,并设置pdb的路径,查看函数调用堆栈,如下所示:

但在堆栈中始终看不到主程序Phaseye_12_01.exe模块中的具体函数应该是Phaseye_12_01.exe对应的pdb没有加载起来。pdb文件路径已经设置到Windbg中了,为啥加载不起来呢?于是尝试使用.reload /f Phaseye_12_01.exe命令去强制加载pdb,还是加载不了。

分析问题时,主要通过查看异常发生时的函数调用堆栈,去对照着源代码进行分析的,所以详细的函数调用的堆栈很重要(比如可以看到具体的函数名和代码的行号)。

       使用lm vm Phaseye_12_01*,可以查看到其pdb文件确实没加载起来:

我再三和朋友确认,Phaseye_12_01.pdb和主程序Phaseye_12_01.exe是匹配的,是同个时间点,这就奇怪了,为啥加载不起来呢?难道pdb文件被修改过?朋友说这些模块是其他同事负责维护的,pdb文件名称应该没有修改过的。


       在这里,给大家重点推荐一下我的几个热门畅销专栏,欢迎订阅:(博客主页还有其他专栏,可以去查看)

专栏1:(该精品技术专栏的订阅量已达到430多个,专栏中包含大量项目实战分析案例,有很强的实战参考价值,广受好评!专栏文章持续更新中,预计更新到200篇以上!欢迎订阅!)

C++软件异常排查从入门到精通系列教程(专栏文章列表,欢迎订阅,持续更新...)icon-default.png?t=N7T8https://blog.csdn.net/chenlycly/article/details/125529931

本专栏根据多年C++软件异常排查的项目实践,系统地总结了引发C++软件异常的常见原因以及排查C++软件异常的常用思路与方法,详细讲述了C++软件的调试方法与手段,以图文并茂的方式给出具体的项目问题实战分析实例(很有实战参考价值),带领大家逐步掌握C++软件调试与异常排查的相关技术,适合基础进阶和想做技术提升的相关C++开发人员!

考察一个开发人员的水平,一是看其编码及设计能力,二是要看其软件调试能力!所以软件调试能力(排查软件异常的能力)很重要,必须重视起来!能解决一般人解决不了的问题,既能提升个人能力及价值,也能体现对团队及公司的贡献!

专栏中的文章都是通过项目实战总结出来的,包含大量项目问题实战分析案例,有很强的实战参考价值!专栏文章还在持续更新中,预计文章篇数能更新到200篇以上!

专栏2: 

C/C++基础与进阶(专栏文章,持续更新中...)icon-default.png?t=N7T8https://blog.csdn.net/chenlycly/category_11931267.html

以多年的开发实战经验为基础,总结并讲解一些的C/C++基础与进阶内容,以图文并茂的方式对C++相关知识点进行详细地展开与剖析!专栏涉及了C/C++开发领域多个方面的内容,同时给出C/C++及网络方面的常见笔试面试题,并详细讲述Visual Studio常用调试手段与技巧!

专栏3: 

VC++常用功能开发汇总icon-default.png?t=N7T8https://blog.csdn.net/chenlycly/article/details/124272585

专栏将10多年C++开发实践中常用的功能,以高质量的代码展现出来,并对相关功能的实现细节进行了详细的说明。这些常用的代码,其质量与稳定性是有保证的,可以直接拿过去使用,可以有效地解决C++软件开发过程中遇到的问题。


3、pdb文件加载失败的可能原因有哪些?

        pdb文件加载失败可能是以下某个原因导致的:

1)没有设置正确的pdb文件路径
没有将pdb文件路径设置到Windbg中,或者设置了错误的pdb文件路径。

2)pdb文件的时间戳与二进制文件不一致
pdb文件的时间戳和对应的二进制文件(dll或exe文件)不一致,即pdb文件和二进制文件不是同一个时间点生成的。加载pdb文件时,会严格校验pdb文件的时间戳,必须要和对应的二进制文件完全一致。即便代码没修改过,不同时间点编译的pdb文件,也是不能交叉使用的。

3)pdb文件名称不能修改
在Visual Studio的工程属性中,默认配置了生成pdb文件的,pdb文件的名称默认就是工程名称。在QT Creator中编译的程序,也可以配置生成pdb文件。这些pdb生成时,会自动将pdb文件的名称写到对应的二进制文件中:


后面Windbg去根据写入的pdb名称去加载pdb文件的。一旦pdb文件生成了,其名称是不能人为修改的,人为修改后会导致pdb文件无法加载的。

4、使用!sym noisy打开pdb加载详情,发现pdb文件名称确实被修改了

       执行.reload /f Phaseye-12-01.exe命令去强制加载Phaseye-12-01.exe的pdb文件,提示加载失败:

提示可以使用!sym noisy命令打开pdb加载详情,看看搜索加载pdb的完整过程。

       于是输入!sym noisy命令打开详情开关,再次输入.reload /f Phaseye-12-01.exe命令去强制加载pdb,看到详细加载过程:

突然看到了一个关键线索,Windbg要加载的pdb名称是Phaseye.pdb,而不是当前的Phaseye_12_01.pdb,由此确定pdb文件名称肯定是被修改过了。于是手动将pdb名称由Phaseye_12_01.pdb改成Phaseye.pdb,重新去执行.reload强制加载命令,加载成功了!

        使用lm vm Phaseye_12_01*查看,看到了加载的pdb路径,如下所示:

再次使用kn命令查看函数调用堆栈,看到了Phaseye-12-01.exe模块的具体函数名和代码行号了:

所以下次遇到不确定pdb文件是否被修改时,可以直接使用!sym noisy命令打开加载详情,然后使用.reload命令去强制加载,看看Windbg加载的pdb名称即可。

5、Windbg是如何知道要加载pdb文件名称的?

       Windbg加载pdb文件时,是如何知道呀加载的pdb文件名称的呢?无论是使用Visual Studio,还是QT Creator,亦或是其他的IDE开发工具,都可以配置生成pdb文件,比如Visual Studio编译时默认生成pdb文件的。也可以不使用默认名称,可以在工程属性配置中指定pdb名称。

        IDE在编译代码时,会自动将对应的pdb文件名称写到生成的dll或exe二进制文件中,比如:

而Windbg正是通过读取二进制文件中写入的pdb名称去加载对应名称的pdb的。

        pdb名称可以在编译配置中指定(在工程属性或者编译脚本中指定),但不能在pdb文件生成后再去修改pdb文件的名称。如果修改了,Windbg就找不到pdb文件,就加载不了了。

这篇关于pdb文件名称被修改导致pdb文件加载失败的实战排查案例分享的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

使用 sql-research-assistant进行 SQL 数据库研究的实战指南(代码实现演示)

《使用sql-research-assistant进行SQL数据库研究的实战指南(代码实现演示)》本文介绍了sql-research-assistant工具,该工具基于LangChain框架,集... 目录技术背景介绍核心原理解析代码实现演示安装和配置项目集成LangSmith 配置(可选)启动服务应用场景

pip install jupyterlab失败的原因问题及探索

《pipinstalljupyterlab失败的原因问题及探索》在学习Yolo模型时,尝试安装JupyterLab但遇到错误,错误提示缺少Rust和Cargo编译环境,因为pywinpty包需要它... 目录背景问题解决方案总结背景最近在学习Yolo模型,然后其中要下载jupyter(有点LSVmu像一个

关于Spring @Bean 相同加载顺序不同结果不同的问题记录

《关于Spring@Bean相同加载顺序不同结果不同的问题记录》本文主要探讨了在Spring5.1.3.RELEASE版本下,当有两个全注解类定义相同类型的Bean时,由于加载顺序不同,最终生成的... 目录问题说明测试输出1测试输出2@Bean注解的BeanDefiChina编程nition加入时机总结问题说明

linux进程D状态的解决思路分享

《linux进程D状态的解决思路分享》在Linux系统中,进程在内核模式下等待I/O完成时会进入不间断睡眠状态(D状态),这种状态下,进程无法通过普通方式被杀死,本文通过实验模拟了这种状态,并分析了如... 目录1. 问题描述2. 问题分析3. 实验模拟3.1 使用losetup创建一个卷作为pv的磁盘3.

使用Navicat工具比对两个数据库所有表结构的差异案例详解

《使用Navicat工具比对两个数据库所有表结构的差异案例详解》:本文主要介绍如何使用Navicat工具对比两个数据库test_old和test_new,并生成相应的DDLSQL语句,以便将te... 目录概要案例一、如图两个数据库test_old和test_new进行比较:二、开始比较总结概要公司存在多

MySql死锁怎么排查的方法实现

《MySql死锁怎么排查的方法实现》本文主要介绍了MySql死锁怎么排查的方法实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录前言一、死锁排查方法1. 查看死锁日志方法 1:启用死锁日志输出方法 2:检查 mysql 错误

MySQL8.2.0安装教程分享

《MySQL8.2.0安装教程分享》这篇文章详细介绍了如何在Windows系统上安装MySQL数据库软件,包括下载、安装、配置和设置环境变量的步骤... 目录mysql的安装图文1.python访问网址2javascript.点击3.进入Downloads向下滑动4.选择Community Server5.

修改若依框架Token的过期时间问题

《修改若依框架Token的过期时间问题》本文介绍了如何修改若依框架中Token的过期时间,通过修改`application.yml`文件中的配置来实现,默认单位为分钟,希望此经验对大家有所帮助,也欢迎... 目录修改若依框架Token的过期时间修改Token的过期时间关闭Token的过期时js间总结修改若依

MySQL修改密码的四种实现方式

《MySQL修改密码的四种实现方式》文章主要介绍了如何使用命令行工具修改MySQL密码,包括使用`setpassword`命令和`mysqladmin`命令,此外,还详细描述了忘记密码时的处理方法,包... 目录mysql修改密码四种方式一、set password命令二、使用mysqladmin三、修改u

在Java中使用ModelMapper简化Shapefile属性转JavaBean实战过程

《在Java中使用ModelMapper简化Shapefile属性转JavaBean实战过程》本文介绍了在Java中使用ModelMapper库简化Shapefile属性转JavaBean的过程,对比... 目录前言一、原始的处理办法1、使用Set方法来转换2、使用构造方法转换二、基于ModelMapper