`
aigo
  • 浏览: 2644836 次
  • 性别: Icon_minigender_1
  • 来自: 宜昌
社区版块
存档分类
最新评论

VC程序Release程序崩溃后问题根源查找

 
阅读更多

文章1:http://hi.baidu.com/0x9000/item/7b355010f4be8c413a176e81

如何去做崩溃后的定位是一个有效的方法。

1. 可以先利用连接器的配置,产生 map 文件:

在 VC Project Setting 对话框中打开 Generate mapfile,然后在Project Options对话框中键入“/mapinfo:lines /mapinfo:exports”,这样在 ./release 目录下就有 .pdb 文件了。
然后在配合 这个方法 就可以获取源码行数了。

2. 可以产生一下 mfc42.dll 的 pdb 文件:

1). cd C:\Program Files\Microsoft Visual Studio\VC98\MFC\SRC

2). ..\..\bin\vcvars32.bat
设置一下 vc 的编译环境

3). nmake /f mfcdll.mak libname=mfc42 debug=0 编译的中间文件在 $DLL.W。

4). 将产生的 mfc42.dll 放到待调试程序的相同路径下,开始调试。

后者在调试 mfc 程序的时候非常有效,通过调用堆栈即可获悉一些很重要的信息。

文章2:http://www.cppblog.com/woaidongmao/archive/2009/10/19/98929.html

我把这个试验的源代码列出来:

const int x =10000;

int main(int argc, char* argv[])

{

       int *y=0;

       y=(int*)&x;

       *y=10;

       return 0;

}

 

我们用Microsoft Visual C++ 6.0(SP5)编译出一个Debug版本的EXE。双击运行它。在Windows 2000 Server下,你将会得到这样一个对话框:

标题:Pointer.exe ? 应用程序错误

正文:”0x00401279”指令引用的”0x0043101c”内存。该内存不能为”written”

要终止程序,请单击确定

要调试程序,请单击取消

   知道了这些信息后,如何找到错误发生策源地呢?

   请记住这个地址“0x00401279”,它是崩溃发生地。

如何找到崩溃的源头:

   有两种情况:

μ        一是我们拥有源代码,可以现场调试;

μ        二是现场绝对不可以安装VC,无法调试,但是我们有它的MAP文件。

   第一种情况,有源代码,这被叫做事后调试”:

首先我们用VC IDE装载这个工程,按F11执行它,切换至反汇编窗口(Disassembly)

   按下Ctrl+G热键。

   你就会得到一个“Go To”的窗口。默认选择是“Address”。在“Enter address expression”编辑框中输入崩溃发生地0x00401279。然后点击“Go To”按钮。你就来到了这个地方:

00401279   mov         dword ptr [eax],0Ah

好了,我们看到了发生崩溃时执行的是这行反汇编代码,但是为什么会崩溃呢?

我们在这里设置一个断点,按F5来到这里。

Watch窗口中键入“@EAX”察看EAX寄存器,得到的数值是“0xcccccccc”。显然这是因为向一个空指针指向的地址复制一个数据,从而造成了崩溃。

好了,针对这个问题,你已经调试成功了。

还有一个问题,对于Release版本的EXE,也可以这么调试吗?

当然可以。同样是这个例子,运行它的Release版本,得到的崩溃地址是0x0040108a

我们在VC中装载这个工程的Release版本,按F11运行它。

来到它的反汇编代码的0x004018a处,我们看到:

0040108A   mov         dword ptr ds:[40B0D0h],0Ah

 

第二种情况,有映射文件Pointer.map:

值得注意的是,如果你只在VC Project Setting对话框中打开Generate mapfile,还是不够的。因为你一定还要输出程序代码地址和源代码行号!!这非常的重要!

要得到这些信息,请在Project Options对话框中键入“/mapinfo:lines /mapinfo:exports”。请你一定要养成这种习惯!因为这不是默认设置。

我们得到的map文件大致如下,我删节了大多数输出:

Pointer

(应用程序名)

Timestamp is 3d4407a7 (Sun Jul 28 23:03:03 2002)

(时间戳)

Preferred load address is 00400000

(最佳装载基地址。非常重要的一个数据。不过一般都是这个数。)

 

Address         Publics by Value              Rva+Base     Lib:Object

 

0001:00000250       _main                      00401250 f   Pointer.obj

(_main的虚地址)

 

Line numbers for .\Debug\Pointer.obj(E:\ Pointer\Pointer.cpp) segment .text

 

12 0001:00000250    14 0001:00000268    15 0001:0000026f    16 0001:00000276

18 0001:0000027f    20 0001:00000291    23 0001:000002a4    24 0001:000002a6

(这就是我们的Pointer.cpp所对应的程序代码行号和相对虚拟地址的对应表)

我们可以从中看到,最佳装载基地址是0x00400000_main的虚地址是0x00401250,而0001:00000250又是什么意思呢?

0x00000250就是_main的相对虚拟地址(RVA)

0x00010000就是PE头文件的大小,一般都是这个数。

所以虚地址就是这么算出来的:

0x00401250 = 0x00400000     + 0x00010000     +  0x00000250

虚地址      = 最佳装载基地址 + PE头文件的大小 + 相对虚拟地址(RVA)

通过_mainRVA的计算,我们也就知道了怎么计算崩溃地址0x00401279RVA,0x00000279,对吧?

然后,在这个MAP映射文件的“Line numbers for .\Debug\Pointer.obj(E:\ Pointer\Pointer.cpp) segment .text”这个行号段中查找这个地址。如你所看到的,只有16行对应的0000027618行对应的0000027F,没有00000279呀?

没有17行的对应关系,说明17行是空行。

那么00000279就一定是16行的了!这样你不用看那个程序员的代码,就可以通知他:崩溃发生在你的Pointer.cpp的第16行了!很酷吧!

 

分享到:
评论

相关推荐

    vc 程序崩溃的捕捉 vc 程序崩溃的捕捉

    VC 程序崩溃的捕捉是一个常见的问题,在编程过程中,程序崩溃是一个非常让人感到苦恼的问题。为了解决这个问题,我们需要了解程序崩溃的捕捉方法。 首先,在 Debug 信息中,我们可以选择 Program Database for Edit...

    vc实现程序崩溃时弹出个自定义的对话框.仿QQ崩溃了会出一个对话框提示程序错误bug上报

    在开发Windows应用程序时,我们经常会遇到程序因各种原因崩溃的情况。为了提高用户体验并收集有用的调试信息,开发者可以自定义在程序崩溃时显示的错误报告对话框,而不是让系统默认的错误对话框出现。Visual C++ ...

    VC程序异常崩溃源代码位置查找助手

    使用方法: 在程序初始化时LoadLibrary( "exceptlog.dll" ); 程序打包时把exceptlog.dll和系统的...当程序异常时不会弹错误对话框,会在当前目录下生成"crack"字样的日志文件,打开后就可以追查到异常具体位置了!

    VC6_VC2005通过MAP和COD文件找出程序崩溃位置

    本文将详细介绍如何在VC6和VC2005中设置相应的编译选项,以及如何利用这些文件来确定程序崩溃的具体位置。 #### 一、编译器设置 **1. VC6 设置** 在VC6中,首先需要通过菜单进行以下设置: - **选择菜单**:...

    vc Release版本调试

    综上所述,“vc Release版本调试”的核心在于通过合理的编译器和链接器配置来保证即便是在优化后的Release版本中也能够有效地进行调试工作。同时结合恰当的调试技巧和注意事项,可以显著提升开发者处理生产环境问题...

    VC程序1 经典VC程序

    VC程序VC程序VC程序VC程序VC程序VC程序VC程序

    VC6.0将程序打包成一个可执行文件(release)

    VC6.0 将程序打包成一个可执行文件(release) VC6.0 将程序打包成一个可执行文件(release) 是一种常见的操作,在这种操作中,我们需要将编译好的程序打包成一个独立的可执行文件,以便在不同的环境中运行。在这篇...

    VC 记录程序崩溃堆栈

    当程序出现未捕获的异常或错误导致崩溃时,通过记录崩溃堆栈,开发者可以追踪到问题发生的具体位置,从而更快地定位和修复错误。本文将详细介绍如何在VC++中实现这一功能。 首先,我们需要理解什么是崩溃堆栈。崩溃...

    VC6.0行号及崩溃处理插件

    在软件开发过程中,程序崩溃是常见的问题,而有效的崩溃处理能够帮助我们迅速找出问题原因。这款插件集成了崩溃日志记录和异常捕获功能,当程序发生异常时,插件会捕获异常信息,包括堆栈跟踪、内存状态等,以便...

    VC下关于debug和release的不同的讨论

    在使用Microsoft Visual C++ (VC) 开发软件时,我们经常会遇到Debug和Release两种配置下程序行为不一致的情况。这是由于两者在编译、链接以及优化策略上的显著差异导致的。下面将详细讨论这些差异以及如何解决由此...

    VC-Debug-Release]出错的问题解决办法

    在调试 VC 程序时,经常出现 Release 版本或者 Debug 版本运行崩溃的问题,本文给出了多种解决方法,以帮助开发者解决这些问题。 一、内存分配问题 在 Debug 版中,变量会自动初始化,但是 Release 版中不会。因此...

    程序崩溃后,记录下崩溃的堆栈地址.rar vc6工程

    "程序崩溃后,记录下崩溃的堆栈地址"是一个关键的调试技术,它能帮助开发者定位程序错误的源头。 堆栈地址是程序执行过程中内存分配的一部分,它存储了函数调用的顺序。当程序崩溃时,堆栈信息通常包含导致崩溃的...

    windows 应用程序崩溃时的内存转储及dump文件的分析

    在 Windows 系统中,当应用程序崩溃时,系统会自动将内存转储到 dump 文件中,以便开发人员可以通过 dump 文件来定位问题。为了实现这个功能,我们需要在 Windows 注册表中设置调试器的相关配置。 在 HKEY_LOCAL_...

    VC 重复文件查找程序 附源码

    【标题】:“VC 重复文件查找程序 附源码” 这个标题揭示了我们正在讨论一个用Visual C++(简称VC)编写的程序,其主要功能是查找并定位系统中的重复文件。这种工具对于优化系统存储空间,清理无用的冗余文件非常...

    vc2008查找失效

    在探讨“vc2008查找失效”的问题时,我们首先需要理解Visual C++ 2008(简称VC2008)是一款由微软公司开发的集成开发环境(IDE),它基于Microsoft Visual Studio 2008平台,主要用于C++语言的软件开发。在VC2008中...

    VC程序的反编译工具depends2.2_x86

    Depends2.2_x86是一款专为VC程序设计的反编译工具,它能帮助开发者分析和调试程序对各种动态链接库(DLL)的依赖性,从而解决运行时错误和兼容性问题。下面我们将详细探讨Depends2.2_x86的功能、使用方法以及其在...

    VC程序调试步骤~~~

    VC程序调试是一种在开发程序的过程中查找程序中的错误的方法,它需要利用调试工具来帮助开发者进行程序的调试。在VC中,调试工具的使用是非常重要的,本文将详细介绍VC中的调试工具的使用。 一、调试环境的建立 在...

    VC下发布的Release版程序的异常捕捉

    本文将详细介绍如何在Visual C++(以下简称VC)环境下对发布的Release版程序进行异常捕捉,主要通过生成并利用MAP文件来定位问题。 #### 二、MAP文件的作用及生成方法 **1. MAP文件简介** MAP文件是一种文本文件...

    VC Debug 和 Release的区别

    在 Release 版本中,优化会省略 EBP 栈基址指针,這样通过一个全局指针访问栈就会造成返回地址错误是程序崩溃。 最后,让我们来说说如何避免这些错误。在编译时,我们可以添加一些选项来避免错误,例如 /Oy- 选项...

    VC程序崩溃_barstat.cpp_Line260解决办法

    在提供的压缩包文件中,"VC程序崩溃_barstat.cpp_Line260解决办法"可能包含了错误现场的详细信息,如调试日志、源代码修改记录、或者修复后的版本。通过深入研究这些资料,开发者可以更有效地定位和解决问题,提高...

Global site tag (gtag.js) - Google Analytics