寫程式出現segmentation fault是常有的事。如果這個bug是固定會出現的,那還算幸運,因為你已經知道發生segmentation fault的地方在哪哩,只要好好看source code應該就可以解決了。但是,如果segmentation fault是三不五時才出現,而且沒有一定的規則會使這個錯誤出現,那就很麻煩了。不過,我們其實可以利用signal handler的方式,攔截segmentation fault發生時的錯誤處理方式,就可以輕鬆的找到錯誤發生的那行程式碼了。
原理是這樣的,原本程式如果發生segmentation fault,作業系統在印出錯誤訊息之後就會將該程式結束。可是,這種處理方式對我們並沒有多大的幫助,因為該程式的process都已經從記憶體中一除了。所以,我們要做的就是,改變發生segmentation fault時處理process的方式。改變的方法就是利用signal()這個系統function。其使用方式很簡單,就是指定要攔截的signal以及處理該signal的handler就可以了。
在下面的程式碼中,我們寫了一個超級簡單的signal handler來代替原來系統預設的segmentation fault handler。就是寫一個無窮迴圈將程式停住,這樣我們就可以用gdb來看看到底是哪一行程式碼觸發segmentation fault了。
#include <signal.h>
#include <stdio.h>
void sighandler(int sig)
{
while(1)
;
}
void error_func()
{
char *msg = NULL;
printf("%s\n", msg); // 這裡會發生segmentation fault
}
int main()
{
signal(SIGSEGV, sighandler);
error_func();
return 0;
}