青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

loop_in_codes

低調做技術__歡迎移步我的獨立博客 codemaro.com 微博 kevinlynx

使用dbghelp獲取調用堆棧--release下的調試方法學

Author : Kevin Lynx

當軟件作為release模式被發布給用戶時,當程序崩潰時我們很難去查找原因。常見的手法是輸出LOG文件,根據LOG文件分析
程序崩潰時的運行情況。我們可以通過SEH來捕獲程序錯誤,然后輸出一些有用的信息作為我們分析錯誤的資料。一般我們需要
輸出的信息包括:系統信息、CPU寄存器信息、堆棧信息、調用堆棧等。而調用堆棧則是最有用的部分,它可以直接幫我們定位
到程序崩潰時所處的位置(在何處崩潰)。(codeproject上關于這個專題的常見開場白 = =#)

要獲取call stack(所謂的調用堆棧),就需要查看(unwind)stack的內容。We could conceivably attempt to unwind the
stack ourselves using inline assembly. But stack frames can be organized in different ways, depending on compiler
optimizations and calling conventions, so it could become complicated to do it that way.(摘自vld文檔)要獲取棧的
內容,我們可以自己使用內聯匯編獲取,但是考慮到兼容性,內聯匯編并不是一個好的解決方案。我們可以使用微軟的dbghelp
中的StackWalk64來獲取棧的內容。

StackWalk64聲明如下:
BOOL StackWalk64(
  DWORD MachineType,
  HANDLE hProcess,
  HANDLE hThread,
  LPSTACKFRAME64 StackFrame,
  PVOID ContextRecord,
  PREAD_PROCESS_MEMORY_ROUTINE64 ReadMemoryRoutine,
  PFUNCTION_TABLE_ACCESS_ROUTINE64 FunctionTableAccessRoutine,
  PGET_MODULE_BASE_ROUTINE64 GetModuleBaseRoutine,
  PTRANSLATE_ADDRESS_ROUTINE64 TranslateAddress
);

具體每個參數的含義可以參見MSDN。這里說下ContextRecord參數,該參數指定了CPU各個寄存器的內容。StackFrame指定了stack
frame的內容。stack frame是什么,我也不知道。(= =) StackWalk64函數需要用戶指定當前frame的地址,以及當前程序的指令
地址。這兩個信息都被填充進ContextRecord,然后傳進StackWalk64函數。

那么如何獲取當前的stack frame地址和當前程序指令地址呢?如前所說,你可以使用內聯匯編。(對于程序指令地址,因為要獲取
EIP寄存器的內容,而該寄存器不能被軟件訪問)也可以使用GetThreadContext一次性獲取當前線程當前運行情況下的CPU各個寄存器
內容。補充下,當前frame地址被放在EBP寄存器里,當前程序指令地址放在EIP寄存器里。但是,如同MSDN對GetThreadContext函數
的說明一樣,該函數可能獲取到錯誤的寄存器內容(You cannot get a valid context for a running thread)。

另一種獲取Context(包含EBP and EIP)的方法就是使用SEH(結構化異常處理),在__except中使用GetExceptionInformation獲取。

GetExceptionInformation 傳回一個LPEXCEPTION_POINTERS指針,該指針指向一個EXCEPTION_POINTERS結構,該結構里包含一個
Context的指針,即達到目標,可以使用StackWalk函數。

補充一下,你可以直接使用StackWalk函數,StackWalk被define為StackWalk64(windows平臺相關)。

unwind棧后,可以進一步獲取一個stack frame的內容,例如函數名。這里涉及到SymFromAddr函數,該函數可以根據一個地址返回
符號名(函數名)。還有一個有意思的函數:SymGetLineFromAddr,可以獲取函數對應的源代碼的文件名和行號。

當然,這一切都依賴于VC產生的程序數據庫文件(pdb),以及提供以上API函數的dbghelp.dll。

參考一段簡單的代碼:

///
///
///

#include <windows.h>
#include 
<stdio.h>
#include 
<dbghelp.h>

#pragma comment( lib, 
"dbghelp.lib" )

void dump_callstack( CONTEXT *context )
{
 STACKFRAME sf;
 memset( 
&sf, 0sizeof( STACKFRAME ) );

 sf.AddrPC.Offset 
= context->Eip;
 sf.AddrPC.Mode 
= AddrModeFlat;
 sf.AddrStack.Offset 
= context->Esp;
 sf.AddrStack.Mode 
= AddrModeFlat;
 sf.AddrFrame.Offset 
= context->Ebp;
 sf.AddrFrame.Mode 
= AddrModeFlat;

 DWORD machineType 
= IMAGE_FILE_MACHINE_I386;

 HANDLE hProcess 
= GetCurrentProcess();
 HANDLE hThread 
= GetCurrentThread();

 
for( ; ; )
 
{
  
if!StackWalk(machineType, hProcess, hThread, &sf, context, 0, SymFunctionTableAccess, SymGetModuleBase, 0 ) )
  
{
   
break;
  }


  
if( sf.AddrFrame.Offset == 0 )
  
{
   
break;
  }

  BYTE symbolBuffer[ 
sizeof( SYMBOL_INFO ) + 1024 ];
  PSYMBOL_INFO pSymbol 
= ( PSYMBOL_INFO ) symbolBuffer;
 
  pSymbol
->SizeOfStruct = sizeof( symbolBuffer );
  pSymbol
->MaxNameLen = 1024;

  DWORD64 symDisplacement 
= 0;
  
if( SymFromAddr( hProcess, sf.AddrPC.Offset, 0, pSymbol ) )
  
{
   printf( 
"Function : %s\n", pSymbol->Name );
  }

  
else
  
{
   printf( 
"SymFromAdd failed!\n" );
  }


  IMAGEHLP_LINE lineInfo 
= sizeof(IMAGEHLP_LINE) };
  DWORD dwLineDisplacement;

  
if( SymGetLineFromAddr( hProcess, sf.AddrPC.Offset, &dwLineDisplacement, &lineInfo ) )
  
{
   printf( 
"[Source File : %s]\n", lineInfo.FileName ); 
   printf( 
"[Source Line : %u]\n", lineInfo.LineNumber ); 
  }

  
else
  
{
   printf( 
"SymGetLineFromAddr failed!\n" );
  }

 }

}


DWORD excep_filter( LPEXCEPTION_POINTERS lpEP )
{
 
/// init dbghelp.dll
 if( SymInitialize( GetCurrentProcess(), NULL, TRUE ) )
 
{
  printf( 
"Init dbghelp ok.\n" );
 }


 dump_callstack( lpEP
->ContextRecord );

 
if( SymCleanup( GetCurrentProcess() ) )
 
{
  printf( 
"Cleanup dbghelp ok.\n" );
 }


 
return EXCEPTION_EXECUTE_HANDLER;
}


void func1( int i )
{
 
int *= 0;
 
*= i;
}


void func2( int i )
{
 func1( i 
- 1 );
}


void func3( int i )
{
 func2( i 
- 1 );
}


void test( int i )
{
 func3( i 
- 1 );
}


int main()
{
 __try
 
{
  test( 
10 );
 }

 __except( excep_filter( GetExceptionInformation() ) )
 
{
  printf( 
"Some exception occures.\n" );
 }


 
return 0;
}



以上代碼在release模式下需要關掉優化,否則調用堆棧顯示不正確(某些函數被去掉了?),同時需要pdb文件。

參考資料:
http://www.codeproject.com/KB/threads/StackWalker.aspx
http://www.cnblogs.com/protalfox/articles/84723.html
http://www.codeproject.com/KB/debug/XCrashReportPt1.aspx
http://www.codeproject.com/KB/applications/visualleakdetector.aspx

ps,本文技術淺嘗輒止,部分內容是否完全準確(正確)我個人都持保留態度,僅供參考。:D

posted on 2008-03-28 16:37 Kevin Lynx 閱讀(13718) 評論(10)  編輯 收藏 引用 所屬分類: 通用編程

評論

# re: 使用dbghelp獲取調用堆棧--release下的調試方法學 2008-03-29 12:45 Sil

不如用dr.watson  回復  更多評論   

# re: 使用dbghelp獲取調用堆棧--release下的調試方法學 2008-03-29 15:00 yafare

一般的做法都是在程序里面設置一下最終異常處理,就是 SetUnhandledExceptionFilter,沒有設置 SEH 的異常最終就會轉移到這里處理了。

這個函數的參數是個需要自己實現的回調函數,Windows會把 EXCEPTION_POINTERS 作為參數傳給這個回調,然后就可以用你文章里面的代碼打印相應的信息了。
  回復  更多評論   

# re: 使用dbghelp獲取調用堆棧--release下的調試方法學 2008-03-29 22:09 Kevin Lynx

@yafare
也行,不知道這個屬于不屬于矢量異常  回復  更多評論   

# re: 使用dbghelp獲取調用堆棧--release下的調試方法學 2008-03-29 22:40 yafare

@Kevin Lynx

沒用過VEH,剛查了下msdn,發現VEH是這樣定義的:

[quote]

Vectored exception handlers are not frame-based handlers. Therefore, you can add a handler and ensure that it gets called regardless of where you are in a call frame. The handlers are called in the order that they were added, after the debugger gets a first chance notification, but before frame-based dispatching occurs.


To add a handler, use the AddVectoredExceptionHandler function. The handler is called for all future exceptions for the process. To remove a handler, use the RemoveVectoredExceptionHandler function.

[/quote]

看他的意思就是VEH 優先級高于SEH的,但是必須得顯式的用AddVectoredExceptionHandler 來添加。

VEH 的handler,跟 SetUnhandledExceptionFilter 參數是一樣的,貌似除了被調用的順序不同,別的也都差不多。
  回復  更多評論   

# re: 使用dbghelp獲取調用堆棧--release下的調試方法學 2008-03-30 21:59 Bugs

有沒有辦法處理Linux平臺的異常處理?研究一下:)  回復  更多評論   

# re: 使用dbghelp獲取調用堆棧--release下的調試方法學 2008-04-18 09:45 lbq1221119

Stack Frame 是托管代碼里面,給EE提供執行體信息的東西.  回復  更多評論   

# re: 使用dbghelp獲取調用堆棧--release下的調試方法學 2008-04-18 09:56 Kevin Lynx

@lbq1221119
native application里也應該有stack frame這個概念吧?  回復  更多評論   

# re: 使用dbghelp獲取調用堆棧--release下的調試方法學 2008-04-18 10:34 lbq1221119

@Kevin Lynx
額 這個我倒是不是特別清楚..我一直研究.Net和托管代碼的...
對于托管stack的Frame調試的時候看的比較多 呵呵

現在開始研究非托管的

__asm
{
mov [_ebp], ebp
}
也可以這樣使用內聯匯編來獲取當前Frame 的Pointer呢  回復  更多評論   

# re: 使用dbghelp獲取調用堆棧--release下的調試方法學 2010-10-20 12:14 Tori

您好啊 我想請問一下 為什么我按照您這里寫的 SymGetLineFromAddr讀不出東西來???
謝謝。  回復  更多評論   

# re: 使用dbghelp獲取調用堆棧--release下的調試方法學 2013-08-10 08:41 王小亮

恩。很不錯的。  回復  更多評論   

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            欧美精品一区二区三区在线看午夜 | 亚洲欧美日韩综合| 久久亚洲精品视频| 国产精品丝袜xxxxxxx| 国产精品视频导航| 在线视频一区二区| 亚洲男人av电影| 久久久久久9| 国产精品99久久久久久久女警| 老司机精品视频一区二区三区| 国内精品久久久久久久97牛牛| 欧美一区二区大片| 葵司免费一区二区三区四区五区| 最近中文字幕日韩精品 | 久久免费视频在线| 国产亚洲一区二区三区| 久久精品91久久香蕉加勒比 | 欧美风情在线| 亚洲精品一区二区在线观看| 欧美大片专区| 欧美成人日韩| 亚洲视频播放| 亚洲一区不卡| 国产日韩一区| 欧美wwwwww| 你懂的国产精品永久在线| 亚洲国产毛片完整版| 91久久夜色精品国产网站| 久久精品视频在线播放| 亚洲国产精品女人久久久| 欧美国产亚洲视频| 欧美色偷偷大香| 久久精品国产亚洲aⅴ| 欧美在线视频全部完| 亚洲黄网站黄| 在线性视频日韩欧美| 国产主播精品在线| 亚洲激情另类| 国产欧美日韩一区二区三区在线观看| 欧美一区二区日韩| 久热精品视频在线观看一区| 欧美精品国产| 亚洲欧美一区二区三区在线| 亚洲欧美在线x视频| 亚洲国产日韩欧美一区二区三区| 亚洲精品久久久蜜桃| 国产精品久久久一区二区三区| 久久精品123| 欧美激情2020午夜免费观看| 亚洲欧美美女| 欧美成人按摩| 亚洲尤物影院| 另类酷文…触手系列精品集v1小说| 一本久道久久久| 欧美影院成年免费版| 久久国产精品久久久久久电车| 免费日韩视频| 亚洲日本成人在线观看| 国产精品美女www爽爽爽| 久久精品一区蜜桃臀影院| 美女在线一区二区| 久久国产88| 欧美二区在线观看| 久久手机免费观看| 欧美性猛交一区二区三区精品| 久久婷婷国产综合精品青草| 欧美日韩在线视频一区| 玖玖玖国产精品| 国产精品丝袜91| aⅴ色国产欧美| 亚洲精品免费一区二区三区| 欧美在线网址| 欧美在线观看视频一区二区三区| 欧美成人国产| 免费在线播放第一区高清av| 国产欧美精品在线观看| 宅男噜噜噜66国产日韩在线观看| 91久久亚洲| 免费亚洲电影在线观看| 免费在线看一区| 狠久久av成人天堂| 久久gogo国模裸体人体| 亚洲欧美综合另类中字| 欧美日韩国产综合网| 亚洲国产三级网| 99国产精品| 亚洲国产精品ⅴa在线观看| 国产亚洲精品高潮| 亚洲自拍电影| 性做久久久久久久免费看| 久久久久久伊人| 久热精品视频在线观看| 在线日韩中文| 久久综合久久综合九色| 欧美xxxx在线观看| 国产一区白浆| 亚洲欧美国产视频| 欧美在线观看天堂一区二区三区 | 欧美激情一区二区三区蜜桃视频| 欧美中文字幕不卡| 国产日韩欧美在线视频观看| 羞羞视频在线观看欧美| 欧美一区二区三区啪啪| 国产视频在线观看一区二区三区 | 国产精品入口日韩视频大尺度| 一本久道综合久久精品| 亚洲专区一区| 国产精品久久一区二区三区| 一区二区三区四区在线| 欧美一区免费视频| 伊人久久久大香线蕉综合直播 | 欧美成人免费全部观看天天性色| 欧美成人精品一区二区三区| 亚洲人成小说网站色在线| 欧美激情综合五月色丁香小说| 欧美中文字幕在线观看| 性欧美超级视频| 久久久999精品| 国内激情久久| 久久精品国产欧美亚洲人人爽| 牛人盗摄一区二区三区视频| 亚洲高清资源综合久久精品| 欧美精品国产精品| 欧美高清不卡| 亚洲在线1234| 欧美日韩国产首页在线观看| 一本色道久久综合狠狠躁篇怎么玩| 亚洲欧美综合| 亚洲丶国产丶欧美一区二区三区| 欧美精品成人一区二区在线观看| 亚洲精品中文字| 久久精品国产在热久久| 亚洲国产天堂网精品网站| 欧美日韩国产91| 久久九九热免费视频| 99视频精品全国免费| 麻豆国产精品777777在线| 亚洲私人影院| 亚洲成人资源网| 国产精品久久久久久久久久ktv| 久久久国产精品亚洲一区| 亚洲精品免费一区二区三区| 久久av一区二区三区| 在线播放日韩| 国产精品美女久久久免费| 免播放器亚洲一区| 亚洲欧美综合v| 日韩午夜在线播放| 女仆av观看一区| 久久精品国产亚洲5555| 亚洲一区二区视频在线| 亚洲国产老妈| 伊人精品成人久久综合软件| 国产精品自拍一区| 欧美少妇一区二区| 欧美激情综合在线| 免费欧美日韩| 久久亚洲私人国产精品va| 亚洲欧美大片| 亚洲一区二区三区四区中文 | 久久国产加勒比精品无码| 99综合电影在线视频| 樱桃成人精品视频在线播放| 国产精品成人午夜| 欧美日韩亚洲三区| 欧美日本在线一区| 欧美1区2区| 免费成人性网站| 久久人人爽人人| 久久久精彩视频| 久久国产精品网站| 亚洲一区二区四区| 欧美激情视频网站| 久久亚洲精品网站| 久久九九99视频| 欧美一区二区成人| 午夜精品久久久久久久蜜桃app| 一区二区三区久久网| 99在线热播精品免费| 9久re热视频在线精品| 夜夜精品视频一区二区| 一区二区三区视频在线看| 99精品国产在热久久| 亚洲免费高清| 亚洲天堂成人| 亚洲欧美在线网| 久久狠狠亚洲综合| 欧美一区二区| 亚洲一区二区三区中文字幕 | 亚洲国产婷婷| 亚洲高清免费| 亚洲精品视频在线看| 一区二区三区四区五区精品| 亚洲最快最全在线视频| 亚洲午夜在线观看| 久久超碰97人人做人人爱| 久久精品国产96久久久香蕉| 久热这里只精品99re8久| 欧美激情第二页| 日韩一级在线观看|