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

loop_in_codes

低調(diào)做技術(shù)__歡迎移步我的獨(dú)立博客 codemaro.com 微博 kevinlynx

使用dbghelp獲取調(diào)用堆棧--release下的調(diào)試方法學(xué)

Author : Kevin Lynx

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

要獲取call stack(所謂的調(diào)用堆棧),就需要查看(unwind)stack的內(nèi)容。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文檔)要獲取棧的
內(nèi)容,我們可以自己使用內(nèi)聯(lián)匯編獲取,但是考慮到兼容性,內(nèi)聯(lián)匯編并不是一個(gè)好的解決方案。我們可以使用微軟的dbghelp
中的StackWalk64來獲取棧的內(nèi)容。

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
);

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

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

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

GetExceptionInformation 傳回一個(gè)LPEXCEPTION_POINTERS指針,該指針指向一個(gè)EXCEPTION_POINTERS結(jié)構(gòu),該結(jié)構(gòu)里包含一個(gè)
Context的指針,即達(dá)到目標(biāo),可以使用StackWalk函數(shù)。

補(bǔ)充一下,你可以直接使用StackWalk函數(shù),StackWalk被define為StackWalk64(windows平臺相關(guān))。

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

當(dāng)然,這一切都依賴于VC產(chǎn)生的程序數(shù)據(jù)庫文件(pdb),以及提供以上API函數(shù)的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模式下需要關(guān)掉優(yōu)化,否則調(diào)用堆棧顯示不正確(某些函數(shù)被去掉了?),同時(shí)需要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,本文技術(shù)淺嘗輒止,部分內(nèi)容是否完全準(zhǔn)確(正確)我個(gè)人都持保留態(tài)度,僅供參考。:D

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

評論

# re: 使用dbghelp獲取調(diào)用堆棧--release下的調(diào)試方法學(xué) 2008-03-29 12:45 Sil

不如用dr.watson  回復(fù)  更多評論   

# re: 使用dbghelp獲取調(diào)用堆棧--release下的調(diào)試方法學(xué) 2008-03-29 15:00 yafare

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

這個(gè)函數(shù)的參數(shù)是個(gè)需要自己實(shí)現(xiàn)的回調(diào)函數(shù),Windows會把 EXCEPTION_POINTERS 作為參數(shù)傳給這個(gè)回調(diào),然后就可以用你文章里面的代碼打印相應(yīng)的信息了。
  回復(fù)  更多評論   

# re: 使用dbghelp獲取調(diào)用堆棧--release下的調(diào)試方法學(xué) 2008-03-29 22:09 Kevin Lynx

@yafare
也行,不知道這個(gè)屬于不屬于矢量異常  回復(fù)  更多評論   

# re: 使用dbghelp獲取調(diào)用堆棧--release下的調(diào)試方法學(xué) 2008-03-29 22:40 yafare

@Kevin Lynx

沒用過VEH,剛查了下msdn,發(fā)現(xiàn)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 優(yōu)先級高于SEH的,但是必須得顯式的用AddVectoredExceptionHandler 來添加。

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

# re: 使用dbghelp獲取調(diào)用堆棧--release下的調(diào)試方法學(xué) 2008-03-30 21:59 Bugs

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

# re: 使用dbghelp獲取調(diào)用堆棧--release下的調(diào)試方法學(xué) 2008-04-18 09:45 lbq1221119

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

# re: 使用dbghelp獲取調(diào)用堆棧--release下的調(diào)試方法學(xué) 2008-04-18 09:56 Kevin Lynx

@lbq1221119
native application里也應(yīng)該有stack frame這個(gè)概念吧?  回復(fù)  更多評論   

# re: 使用dbghelp獲取調(diào)用堆棧--release下的調(diào)試方法學(xué) 2008-04-18 10:34 lbq1221119

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

現(xiàn)在開始研究非托管的

__asm
{
mov [_ebp], ebp
}
也可以這樣使用內(nèi)聯(lián)匯編來獲取當(dāng)前Frame 的Pointer呢  回復(fù)  更多評論   

# re: 使用dbghelp獲取調(diào)用堆棧--release下的調(diào)試方法學(xué) 2010-10-20 12:14 Tori

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

# re: 使用dbghelp獲取調(diào)用堆棧--release下的調(diào)試方法學(xué) 2013-08-10 08:41 王小亮

恩。很不錯(cuò)的。  回復(fù)  更多評論   

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            久久视频在线免费观看| 老司机午夜精品| 久久久久久欧美| 久久精品国产第一区二区三区最新章节 | 欧美**人妖| 欧美日韩色一区| 国产在线播精品第三| 久久亚洲精品一区二区| 欧美理论在线| 欧美激情在线| 亚洲欧洲精品天堂一级| 国产精品欧美经典| 欧美日韩一区二区三区在线| 99热在线精品观看| 亚洲国产精品久久精品怡红院| 国产精品理论片在线观看| 欧美激情网友自拍| 午夜伦欧美伦电影理论片| 久久大香伊蕉在人线观看热2| 久久综合狠狠| 狠狠色噜噜狠狠狠狠色吗综合| 99热在这里有精品免费| 久久国内精品自在自线400部| 欧美在线观看一区| 亚洲欧美日本日韩| 欧美日本精品一区二区三区| 国产乱肥老妇国产一区二| 亚洲欧美精品在线观看| 欧美激情一区二区三区蜜桃视频| 亚洲一区免费在线观看| 亚洲欧美日韩在线一区| 一区二区视频欧美| 亚洲一区制服诱惑| 亚洲电影在线看| 欧美成人精品在线观看| 国产在线精品一区二区中文| 亚洲免费在线观看| 99精品国产99久久久久久福利| 国产一区二区三区日韩| 午夜欧美大片免费观看| 日韩一级精品视频在线观看| 亚洲黑丝在线| 亚洲福利小视频| 欧美激情免费观看| 欧美精品一区二区三| 亚洲午夜小视频| 久久av一区| 亚洲一区激情| 女生裸体视频一区二区三区| 国产精品萝li| 亚洲国产综合在线看不卡| 久久综合网hezyo| 国产亚洲精品久久久久动| 欧美一区成人| 久久国产免费| 久久久人成影片一区二区三区观看| 欧美午夜三级| 亚洲一本视频| 亚洲素人一区二区| 另类av导航| 欧美一区二区精品久久911| 性18欧美另类| 国产欧美精品一区| 亚洲淫性视频| 亚洲男人第一av网站| 激情一区二区| 亚洲高清二区| 欧美成人蜜桃| 久热re这里精品视频在线6| 久久精品人人做人人爽电影蜜月| 亚洲麻豆一区| 亚洲性夜色噜噜噜7777| 国产精品久久久一区二区| 日韩午夜三级在线| 亚洲欧美国产日韩天堂区| 国产九九精品| 蜜臀av在线播放一区二区三区| 国产精品美女久久久| 欧美一级理论性理论a| 亚洲欧美在线另类| 99精品国产福利在线观看免费| 久久久久久高潮国产精品视| 亚洲国产精品成人久久综合一区| 欧美色精品在线视频| 最新高清无码专区| 亚洲欧洲精品一区二区精品久久久| 欧美午夜不卡在线观看免费| 国产日韩精品一区二区浪潮av| 国产在线视频欧美| 亚洲精品一区在线| 欧美一级二区| 亚洲国产一成人久久精品| 亚洲一区中文| 暖暖成人免费视频| 国产精品日日摸夜夜添夜夜av| 国产一区二区三区四区三区四| 亚洲精品一区二区在线| 欧美成人一区二区三区| 欧美电影免费观看| 99国产精品一区| 亚洲欧美日韩国产另类专区| 欧美色区777第一页| 国产日韩欧美麻豆| 国产精品久久久久毛片大屁完整版| 美女黄色成人网| 久久er精品视频| 亚洲免费网站| 一本久久青青| 亚洲校园激情| 欧美三区在线| 亚洲视频在线观看网站| 欧美成人蜜桃| 亚洲精品日产精品乱码不卡| 99热在这里有精品免费| 欧美日韩免费在线观看| 亚洲在线观看视频| 蜜桃av一区二区三区| 亚洲人精品午夜| 欧美午夜www高清视频| 亚洲欧美国产毛片在线| 免费不卡视频| 一本色道久久综合狠狠躁的推荐| 国产精品成人在线| 欧美一区三区三区高中清蜜桃| 蜜臀91精品一区二区三区| 日韩视频在线免费| 国产欧美日韩亚洲精品| 久久久久在线观看| 一本久道综合久久精品| 麻豆av一区二区三区| 99re国产精品| 国产一区二区三区久久| 欧美国产激情二区三区| 亚洲视频axxx| 欧美肥婆bbw| 亚洲欧美精品在线观看| 欧美亚洲第一页| 午夜精品久久久久久99热软件 | 亚洲欧美日韩成人| 国产亚洲成av人片在线观看桃| 香蕉国产精品偷在线观看不卡| 久久久www成人免费无遮挡大片| 亚洲欧洲一区二区在线观看| 香蕉国产精品偷在线观看不卡| 亚洲激情视频网站| 欧美一区二区三区在线| 亚洲男人第一av网站| 欧美精品久久久久久久免费观看 | 在线播放一区| 午夜精品久久久久久| 亚洲欧美日本精品| 国内在线观看一区二区三区| 午夜激情综合网| 午夜精品久久久久久99热软件| 欧美亚洲三级| 91久久亚洲| 国产在线拍揄自揄视频不卡99| 国产精品久久久对白| 欧美日韩不卡合集视频| 久久先锋影音av| 欧美一区二区精品| 国产精品99久久不卡二区| 亚洲国产精品成人va在线观看| 国产精品毛片a∨一区二区三区| 欧美—级a级欧美特级ar全黄| 久久久精品国产免大香伊| 午夜激情久久久| 亚洲免费在线精品一区| 亚洲视频一二三| 中文欧美字幕免费| 一区二区三区蜜桃网| 99成人精品| 99在线精品观看| 国语精品一区| 久久综合久色欧美综合狠狠| 亚洲欧美视频一区| 欧美一区激情视频在线观看| 一区二区三区产品免费精品久久75 | 欧美一区二区三区免费视| 国产亚洲精品资源在线26u| 欧美电影打屁股sp| 先锋影音久久| 亚洲精品免费观看| 欧美专区在线播放| 亚洲伦伦在线| 伊人久久婷婷色综合98网| 欧美经典一区二区三区| 日韩午夜免费| 国产精品视频久久| 亚洲素人一区二区| 亚洲日本免费电影| 欧美大片专区| 亚洲国内自拍| aa亚洲婷婷| 日韩图片一区| 在线一区二区三区做爰视频网站 | 最新国产成人av网站网址麻豆 | 欧美在线free| 亚洲视频综合| 亚洲综合大片69999|