[转载]DWARF dSYM
A
When debug symbols are embedded in the binary itself, the file can
then grow significantly larger (sometimes by several megabytes). To
avoid this extra size, modern compilers and early mainframe
debugging systems output the symbolic information into a separate
file; for Microsoft compilers, this file is called
a
我简单翻译下上面两段话(英语比较烂,翻译得不好,请谅解):
调试符号(debug symbol)是代表在给定的可执行的模块中由编程语言产生的一段特定的机器码的信息。有时这些符号信息被编译进模块的二进制文件中,或者以单独的文件的形式进行分发,或者在编译/链接的时候直接丢弃。这些信息使得人们可以一种符号调试程序来获取关于二进制文件的额外信息,比如变量的名称和来自于源代码的routines(不知道怎么翻译比较好)。这些信息在我们研究和修复程序的崩溃问题或者别的错误的时候非常有用。
-----------------------为什么会非常有用呢?----------------------------
因为借助符号调试程序可以将类似
Thread 0 Crashed:
0 libobjc.A.dylib 0×300c87ec 0×300bb000 + 55276
1 MobileLines 0×00006434 0×1000 + 21556
2 MobileLines 0×000064c2 0×1000 + 21698
3 UIKit 0×30a740ac 0×30a54000 + 131244
的log信息转换成
Thread 0 Crashed:
0 libobjc.A.dylib 0×300c87ec objc_msgSend + 20
1 MobileLines 0×00006434 -[BoardView setSelectedPiece:] (BoardView.m:321)
2 MobileLines 0×000064c2 -[BoardView touchesBegan:withEvent:] (BoardView.m:349)
3 UIKit 0×30a740ac -[UIWindow sendEvent:] + 264
-------------------这样应该能非常明了地指出问题所在吧?-------------------
当调试符号被嵌入进二进制文件本身的时候,文件的尺寸会显著地增长(不难理解吧?文件中放入了额外的信息,自然就大了)。为了避免这个额外的文件尺寸(为什么要避免,应该不难理解吧),现在的编译器(新的,和老旧的相对)和早期的大型机调试系统将符号信息输出到单独的文件中(为什么要输出到单独的文件中?调试符号信息对于普通的用户是没有用的,它只对开发人员追踪问题有用,嵌入到二进制文件中只会增大文件尺寸,延长用户下载的时间,而且也会暴露你程序的内部信息,如果你想保护你的代码不被别人猜到,你应该考虑不要在二进制文件中包含调试信息)。比如微软的编译器,这个文件叫做PDB文件,有的公司通过CD/DVD的方式来传输,有的公司提供在线的服务器单独进行下载,通过PDB文件,你可以进行程序故障诊断(发现程序出错的地方)。
通过上面简单的介绍,我想大家应该对Debug Symbol应该已经有了个大概的认识了。
二、DWARF
接下来我们看看DWARF是什么。
官方网站对DWARF的简单介绍:
DWARF
is a debugging file format used by many compilers and debuggers to
support source level debugging. It addresses the requirements of a
number of procedural languages, such as C, C++, and Fortran, and is
designed to be extensible to other languages. DWARF is architecture
independent and applicable to any processor or operating system. It
is widely used on Unix, Linux and other operating systems, as well
as in stand-alone environments.
DWARF是一种被众多编译器和调试器使用的用于支持源代码级别调试的调试文件格式。它满足了许多程序语言的需求,比如C,C++和Fortran,而且被设计成可拓展到其它语言。DWARF是平台独立的且适用于任何处理器任何操作系统。
为了更好地理解DWARF,可以阅读下这篇文章:http://lapcatsoftware.com/blog/2008/03/09/stabs-is-deprecated/
从2008-02-27开始, STABS(一种调试数据格式)被苹果Apple弃用。XCode项目build setting中DEBUG_INFORMATION_FORMAT默认的STABS被取代。
通过STABS, 你可以创建带有调试符号(debugging
symbols)的程序的release版本,拷贝一份可执行程序MyApp.app/Contents/MacOS/MyApp保存起来,
而把此可执行文件除去调试符号用于分发(strip the executable for shipping),
然后用未除去调试符号的可执行程序来进行崩溃分析。和STABS不同,
DWARF不将调试符号包含到可执行文件自身里面,而是仅在可执行文件里包含对中间对象文件的引用( stabs
,
这些对象文件则是完全无用的.) 如果你把调试符号从你的程序中除去(strip debugging symbols from your
app),你也不能进行单步调试,因为可执行文件中已经不存在对中间对象文件的引用了。
三、dSYM
为了避免进行stripping操作后调试符号的丢失,你可以使用dwarf-with-dsym选项
.
DWARF with dSYM 选项在标准的DWARF之外执行一个额外的步骤:创建一个单独的MyApp.app.dSYM文件,这个文件包含你的程序的所有调试符号(这个文件其实是一个包,可以通过右键->显示包内容进行查看)。事实上,DWARF
with
dSYM选项允许你对你进行单步调试而不管可执行程序是否被剥离了调试信息(stripped)。这是可能的,这是因为gdb将会在你的程序的目录下查找.dSYM
文件。它不需要知道对象文件(object
files)的名字或者路径。如果你不除去调试符号 (strip debugging symbols),
你可以使用.o或者.dSYM
文件来调试。