VS2005/VC2008中SetUnhandledExceptionFilter函數無效

 

很多軟件通過設置自己的異常捕獲函數,捕獲未處理的異常,生成報告或者日誌(例如生成mini-dump 文件),達到Release 版本下追蹤Bug 的目的。但是,到了VS2005 (即VC8 ),MicrosoftCRTC 運行時庫)的一些與安全相關的代碼做了些改動,典型的,例如增加了對緩衝溢出的檢查。新CRT 版本在出現錯誤時強制把異常拋給默認的調試器(如果沒有配置的話,默認是Dr.Watson ),而不再通知應用程序設置的異常捕獲函數,這種行爲主要在以下三種情況出現。

1         調用abort 函數,並且設置了_CALL_REPORTFAULT 選項(這個選項在Release 版本是默認設置的)。

2         啓用了運行時安全檢查選項,並且在軟件運行時檢查出安全性錯誤,例如出現緩存溢出。(安全檢查選項  /GS  默認也是打開的)

3         遇到_invalid_parameter 錯誤,而應用程序又沒有主動調用

_set_invalid_parameter_handler 設置錯誤捕獲函數。

所以結論是,使用VS2005VC8 )編譯的程序,許多錯誤都不能在SetUnhandledExceptionFilter 捕獲到。這是CRT 相對於前面版本的一個比較大的改變,但是很遺憾,Microsoft 卻沒有在相應的文檔明確指出。

解決方法

         之所以應用程序捕獲不到那些異常,原因是因爲新版本的CRT 實現在異常處理中強制刪除所有應用程序先前設置的捕獲函數,如下所示:

  /* Make sure any filter already in place is deleted. */

  SetUnhandledExceptionFilter(NULL);

  UnhandledExceptionFilter(&ExceptionPointers);

解決方法是攔截CRT 調用SetUnhandledExceptionFilter 函數,使之無效。在X86 平臺下,可以使用以下代碼。

#ifndef _M_IX86

        #error "The following code only works for x86!"

#endif

 

void DisableSetUnhandledExceptionFilter()

{

     void *addr = (void*)GetProcAddress(LoadLibrary(_T("kernel32.dll")),

                                                          "SetUnhandledExceptionFilter");

     if (addr)

     {

               unsigned char code[16];

               int size = 0;

               code[size++] = 0x33;

               code[size++] = 0xC0;

               code[size++] = 0xC2;

               code[size++] = 0x04;

               code[size++] = 0x00;

 

                  DWORD dwOldFlag, dwTempFlag;

                VirtualProtect(addr, size, PAGE_READWRITE, &dwOldFlag);

               WriteProcessMemory(GetCurrentProcess(), addr, code, size, NULL);

               VirtualProtect(addr, size, dwOldFlag, &dwTempFlag);

        }

}

在設置自己的異常處理函數後,調用DisableSetUnhandledExceptionFilter 禁止CRT 設置即可。

其它討論

         上面通過設置api hook ,解決了在VS2005 上的異常捕獲問題,這種雖然不是那麼“乾淨”的解決方案,確是目前唯一簡單有效的方式。

         雖然也可以通過_set_abort_behavior(0, _WRITE_ABORT_MSG | _CALL_REPORTFAULT), signal(SIGABRT, ...),  _set_invalid_parameter_handler(...)  解決(1 )(3 ),但是對於(2 ),設置api hook 是唯一的方式。

發佈了80 篇原創文章 · 獲贊 8 · 訪問量 34萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章