PDB文件:每一个开发人员都必须知道的

PDB Files: What Every Developer Must Know
http://www.wintellect.com/CS/blogs/jrobbins/archive/2009/05/11/pdb-files-what-every-developer-must-know.aspx
 
PDB文件:每一个开发人员都必须知道的
 

一 什么是PDB文件web

大部分的开发人员应该都知道PDB文件是用来帮助软件的调试的。可是他到底是如何工做的呢,咱们可能并不熟悉。本文描述了PDB文件的存储和内容。同时还描 述了debugger如何找到binay相应的PDB文件,以及debugger如何找到与binay对应的源代码文件。本文适用于全部的Native和 Managed的开发人员。 

在开始前,咱们先定义2个术语:private build, 用来表示在开发人员本身机器上生成的build;public build,表示在公用的build机器上生成的build。private build相对来讲比较简单,由于PDB和binay在相同的地方,一般地咱们遇到的问题都是关于public build。  
 
全部的的开发人员须要知道的最重要的事情是”PDB文件跟源代码一样的重要“, 没有PDB文件,你甚至不能debugging。对于public build,须要symbol server存储全部的PDB,而后当用户报告错误的时候,debugger才能够自动地找到binay相应的PDB文件, visual studio 和 windbg都知道如何访问symbol server。在将PDB和binay存储到symbol server前,还须要对PDB运行进行source indexing, source indexing的做用是将PDB和source关联起来。  
 
接下来的部分假设有已经设置好了symbol server和source server indexing。TFS2010中能够很简单地完成对一个新的build的source indexing 和 symbol server copying。
 
app

二 PDB文件的内容函数

正式开始PDB的内容,PDB不是公开的文件格式,可是Microsoft提供了API来帮助从PDB中获取数据。
 
Native C++ PDB包含了以下的信息:
 * public,private 和static函数地址;
 * 全局变量的名字和地址;
 * 参数和局部变量的名字和在堆栈的偏移量;
 * class,structure 和数据的类型定义;
 * Frame Pointer Omission 数据,用来在x86上的native堆栈的遍历;
 * 源代码文件的名字和行数;
 
.NET PDB只包含了2部分信息:
 * 源代码文件名字和行数;
 * 和局部变量的名字;
 * 全部的其余的数据都已经包含在了.NET Metadata中了;  
 
工具

三 PDB如何工做post

当你加载一个模块到进程的地址空间的时候,debugger用2中信息来找到相应的PDB文件。第一个毫无疑问就是文件的名字,若是加载 zzz.dll,debugger则查找zzz.pdb文件。在文件名字相同的状况下debugger还经过嵌入到PDB和binay的GUID来确保 PDB和binay的真正的匹配。 因此即便没有任何的代码修改,昨天的binay和今天的PDB是不能匹配的。能够使用dempbin.exe来查看binary的GUID。 
 
在VisualStudio中的modules窗口的symbol file列能够查看PDB的load顺序。第一个搜索的路径是binary所在的路径,若是不在binary所在的路径,则查找binary中hardcode记录的build目录,例如obj\debug\*.pdb, 若是以上两个路径都没有找到PDB,则根据symbol server的设置,在本地的symbol server的cache中查找,若是在本地的symbol server的cache中没有对应的PDB,则最后才到远程的symbol server中查找。经过上面的查找顺序咱们能够看出为何public build和private build的PDB查找不会冲突。 
 
对于private build有时咱们须要在别人的机器上debug的状况,须要将相应的PDB与binary一块儿拷贝,对于加入GAC的.NET的binary,须要将PDB文件拷贝到C:\Windows\assembly\GAC_MSIL\Example\1.0.0.0__682bc775ff82796a相似的binary所在的目录。另外一个变通的方法是定义环境变量DEVPATH,从而代替使用命令GACUTIL将binary放入GAC中。在定义DEVPATH后,只须要将binary和PDB放到DEVPATH的路径,在DEVPATH下的binary至关于在GAC下。使用DEVPATH,首先须要建立目录且对当前build用户有写权限,而后建立环境变量DEVPATH且值为刚才建立的目录,而后在web.config,app.config或machine.config中开启development模式,启动对DEVPATH的使用 
<configuration> 
   <runtime> 
      <developmentMode developerInstallation="true"/> 
   </runtime> 
</configuration> 

在你打开了development模式后,若是DEVPATH没有定义或路径不存在的话会致使程序启动时异常"Invalid value for registry"。并且若是在machine.config中开启DEVPATH的使用会影响其余的全部的程序,因此要慎重使用machine.config。 
 
最后开发人员须要知道的是源代码信息是如何存储在PDB文件中的。对于public builds,在运行source indexing tool后,版本控制工具将代码存储到你设置的代码cache中。对于private builds,只是存储了PDB文件的全路径,例如在c:\foo下的源文件mycode.cpp,在pdb文件中存储的路径为c:\foo\mycode.cpp。对于private builds能够使用虚拟盘来增长PDB对绝对路径的依赖,例如能够使用subst.exe将源代码路径挂载为V:,在别人的机器上debug的时候也挂载V:。ui

 

完! 
debug


做者: iTech
出处: http://itech.cnblogs.com/ 
相关文章
相关标签/搜索