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

loop_in_codes

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

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

Author : Kevin Lynx

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

要獲取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)匯編并不是一個好的解決方案。我們可以使用微軟的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
);

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

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

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

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

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

unwind棧后,可以進一步獲取一個stack frame的內(nèi)容,例如函數(shù)名。這里涉及到SymFromAddr函數(shù),該函數(shù)可以根據(jù)一個地址返回
符號名(函數(shù)名)。還有一個有意思的函數(shù):SymGetLineFromAddr,可以獲取函數(shù)對應(yī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ù)被去掉了?),同時需要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)容是否完全準確(正確)我個人都持保留態(tài)度,僅供參考。:D

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

評論

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

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

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

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

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

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

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

# re: 使用dbghelp獲取調(diào)用堆棧--release下的調(diào)試方法學 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)試方法學 2008-03-30 21:59 Bugs

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

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

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

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

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

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

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

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

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

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

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

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

恩。很不錯的。  回復(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>
            91久久精品国产91久久性色| 亚洲欧美一区二区激情| 欧美黑人国产人伦爽爽爽| 欧美在线视频不卡| 激情欧美日韩一区| 免费欧美日韩| 国产精品美女久久久久久2018| 欧美日韩精品一区二区| 亚洲图片激情小说| 亚洲天堂成人| 激情伊人五月天久久综合| 亚洲美洲欧洲综合国产一区| 亚洲精品色图| 国产无一区二区| 亚洲精品中文字幕女同| 国产真实乱子伦精品视频| 亚洲激情在线视频| 欧美另类在线播放| 欧美电影免费| 久久免费国产精品| 欧美日韩一级黄| 欧美福利网址| 黄网动漫久久久| 亚洲视屏一区| 国产精品99久久久久久有的能看| 欧美在线视频在线播放完整版免费观看| 亚洲美女免费精品视频在线观看| 一区二区三区成人| 亚洲视频免费在线观看| 欧美18av| 亚洲国产第一页| 91久久午夜| 免播放器亚洲| 亚洲免费电影在线| 一区二区三区国产精华| 久久免费视频这里只有精品| 亚洲午夜影视影院在线观看| 欧美gay视频激情| 欧美激情第10页| 在线一区二区三区做爰视频网站| 欧美精品啪啪| 亚洲自拍偷拍麻豆| 亚洲欧美日韩精品一区二区 | 亚洲欧美区自拍先锋| 欧美日韩国产成人在线91| 亚洲第一在线视频| 99精品福利视频| 国产精品一级二级三级| 欧美一区二区福利在线| 国产精品久久激情| 久久午夜精品| 亚洲伊人观看| 91久久精品www人人做人人爽| 欧美专区18| 最新中文字幕一区二区三区| 国产精品免费福利| 欧美另类综合| 美女精品国产| 久久精品二区三区| 午夜免费久久久久| 亚洲国产精品成人一区二区| 99视频精品全部免费在线| 欧美日韩一区二区三区在线观看免 | 欧美一级理论性理论a| 亚洲国产福利在线| 激情综合网激情| 国产亚洲女人久久久久毛片| 欧美国产视频一区二区| 久久一区欧美| 欧美在线视屏| 久久久青草婷婷精品综合日韩| 国产一区亚洲| 在线精品国产成人综合| 一区二区三区无毛| 在线播放日韩| 国产视频不卡| 国产一区清纯| 曰韩精品一区二区| 极品少妇一区二区| 亚洲电影天堂av| 在线亚洲一区观看| 午夜一区不卡| 欧美高清视频| 日韩视频在线观看一区二区| 亚洲人成在线观看一区二区| 一区二区精品在线| 亚洲女同在线| 亚洲乱码国产乱码精品精天堂| 国内精品久久久久久久果冻传媒| 伊人色综合久久天天| 久久精品一区二区国产| 亚洲影院色在线观看免费| 久久精品国产第一区二区三区| 久久黄色级2电影| 亚洲人成网站777色婷婷| 亚洲欧美在线看| 欧美精品手机在线| 在线日韩电影| 久久av一区二区三区漫画| 国产精品久99| 中国女人久久久| 免费不卡欧美自拍视频| 国产日韩精品入口| 久久精品主播| 国产精品色网| 99国内精品久久| 91久久夜色精品国产九色| 香蕉久久夜色精品国产使用方法 | 尤物yw午夜国产精品视频| 国产精品99久久久久久白浆小说| 欧美电影免费观看高清| 欧美一区二区免费| 国产欧美日韩三区| 欧美一区二区视频在线| 中文高清一区| 欧美另类videos死尸| 亚洲午夜精品17c| 亚洲区在线播放| 欧美精品久久99久久在免费线| 伊人蜜桃色噜噜激情综合| 久久国产手机看片| 久久精品午夜| 亚洲精品在线一区二区| 亚洲国产精品ⅴa在线观看| 免费在线欧美视频| 妖精成人www高清在线观看| 激情丁香综合| 美女视频一区免费观看| 久久久综合网站| 91久久国产综合久久| 亚洲人成网站在线播| 国产精品xvideos88| 久久精品91久久香蕉加勒比| 久久久久久久久久久成人| 91久久精品一区二区三区| 最新亚洲一区| 欧美午夜片欧美片在线观看| 午夜精品影院在线观看| 美玉足脚交一区二区三区图片| 国产精品99久久久久久久女警| 欧美一区二区三区视频免费播放| 亚洲欧洲久久| 亚洲已满18点击进入久久| 在线观看中文字幕不卡| 欧美影片第一页| 午夜精彩视频在线观看不卡| 久久婷婷av| 久久资源在线| 国产日韩专区| 亚洲网站在线看| 亚洲精品免费在线播放| 欧美专区日韩专区| 久久久国产精品一区| 国产精品日韩一区二区| 亚洲少妇一区| 91久久精品日日躁夜夜躁欧美 | 亚洲午夜av电影| 鲁大师成人一区二区三区| 久久九九99| 国产欧美不卡| 亚洲欧美日本国产有色| 亚洲午夜精品福利| 久久久午夜电影| 亚洲电影成人| 欧美性一二三区| 亚洲视频欧美视频| 欧美一区二区三区婷婷月色| 国产精品毛片a∨一区二区三区| 亚洲一区二区在线播放| 久久精品一区二区国产| 国产专区精品视频| 久久精品欧美| 亚洲国产精彩中文乱码av在线播放| 久久精彩视频| 亚洲国产日韩一级| 午夜国产一区| 亚洲国产一区二区a毛片| 免费黄网站欧美| 亚洲视频日本| 欧美成人中文字幕| 久久精品国产亚洲高清剧情介绍| 亚洲国产日韩精品| 国产精品一区二区三区久久| 久久久免费av| 亚洲欧美国产日韩天堂区| 亚洲国产日韩欧美在线99 | 亚洲成人原创| 欧美激情视频在线播放| 欧美诱惑福利视频| 一区二区日韩精品| 91久久国产精品91久久性色| 久久成人久久爱| 亚洲一区二区三区在线播放| 在线看成人片| 精品成人免费| 亚洲国产高清一区| 亚洲福利视频一区| 亚洲精品视频免费| 亚洲第一免费播放区| 国产在线精品成人一区二区三区|