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

S.l.e!ep.¢%

像打了激速一樣,以四倍的速度運轉,開心的工作
簡單、開放、平等的公司文化;尊重個性、自由與個人價值;
posts - 1098, comments - 335, trackbacks - 0, articles - 1
  C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理

Linux GDB 調試程序code Segmentation fault

Posted on 2011-08-03 17:50 S.l.e!ep.¢% 閱讀(926) 評論(0)  編輯 收藏 引用 所屬分類: Unix

Linux GDB 調試程序code Segmentation fault2008-11-05 11:19:24|? 分類: C語言 |? 標簽: |字號大

小 訂閱
產生段錯誤就是訪問了錯誤的內存段,一般是你沒有權限,或者根本就不存在對應的物理內存,尤其常見的是訪問0地址.

一般來說,段錯誤就是指訪問的內存超出了系統所給這個程序的內存空間,通常這個值是由gdtr來保存的,他是一個48位的寄存器,其中的32位是保存由它指向的gdt表,后13位保存相應于gdt的下標,最后3位包括了程序是否在內存中以及程序的在cpu中的運行級別,指向的gdt是由以64位為一個單位的表,在這張表中就保存著程序運行的代碼段以及數據段的起始地址以及與此相應的段限和頁面交換還有程序運行級別還有內存粒度等等的信息。一旦一個程序發生了越界訪問,cpu就會產生相應的異常保護,于是segmentation fault就出現了.

在編程中以下幾類做法容易導致段錯誤,基本是是錯誤地使用指針引起的

1)訪問系統數據區,尤其是往 系統保護的內存地址寫數據
最常見就是給一個指針以0地址
2)內存越界(數組越界,變量類型不一致等) 訪問到不屬于你的內存區域

解決方法

我們在用C/C++語言寫程序的時侯,內存管理的絕大部分工作都是需要我們來做的。實際上,內存管理是一個比較繁瑣的工作,無論你多高明,經驗多豐富,難免會在此處犯些小錯誤,而通常這些錯誤又是那么的淺顯而易于消除。但是手工“除蟲”(debug),往往是效率低下且讓人厭煩的,本文將就"段錯誤"這個內存訪問越界的錯誤談談如何快速定位這些"段錯誤"的語句。
下面將就以下的一個存在段錯誤的程序介紹幾種調試方法:
1 dummy_function (void)
2 {
3 unsigned char *ptr = 0x00;
4 *ptr = 0x00;
5 }
6
7 int main (void)
8 {
9 dummy_function ();
10
11 return 0;
12 }
作為一個熟練的C/C++程序員,以上代碼的bug應該是很清楚的,因為它嘗試操作地址為0的內存區域,而這個內存區域通常是不可訪問的禁區,當然就會出錯了。我們嘗試編譯運行它:
xiaosuo@gentux test $ ./a.out
段錯誤
果然不出所料,它出錯并退出了。
1.利用gdb逐步查找段錯誤:
這種方法也是被大眾所熟知并廣泛采用的方法,首先我們需要一個帶有調試信息的可執行程序,所以我們加上“-g -rdynamic"的參數進行編譯,然后用gdb調試運行這個新編譯的程序,具體步驟如下:
xiaosuo@gentux test $ gcc -g -rdynamic d.c
xiaosuo@gentux test $ gdb ./a.out
GNU gdb 6.5
Copyright (C) 2006 Free Software Foundation, Inc.
GDB is free software, covered by the GNU General Public License, and you are
welcome to change it and/or distribute copies of it under certain conditions.
Type "show copying" to see the conditions.
There is absolutely no warranty for GDB. Type "show warranty" for details.
This GDB was configured as "i686-pc-linux-gnu"...Using host libthread_db library "/lib/libthread_db.so.1".

(gdb) r
Starting program: /home/xiaosuo/test/a.out

Program received signal SIGSEGV, Segmentation fault.
0x08048524 in dummy_function () at d.c:4
4 *ptr = 0x00;
(gdb)?
哦?!好像不用一步步調試我們就找到了出錯位置d.c文件的第4行,其實就是如此的簡單。
從這里我們還發現進程是由于收到了SIGSEGV信號而結束的。通過進一步的查閱文檔(man 7 signal),我們知道SIGSEGV默認handler的動作是打印”段錯誤"的出錯信息,并產生Core文件,由此我們又產生了方法二。
2.分析Core文件:
Core文件是什么呢?
The default action of certain signals is to cause a process to terminate and produce a core dump file, a disk file containing an image of the process's memory at the time of termination. A list of the signals which cause a process to dump core can be found in signal(7).
以上資料摘自man page(man 5 core)。不過奇怪了,我的系統上并沒有找到core文件。后來,憶起為了漸少系統上的拉圾文件的數量(本人有些潔癖,這也是我喜歡Gentoo的原因之一),禁止了core文件的生成,查看了以下果真如此,將系統的core文件的大小限制在512K大小,再試:
xiaosuo@gentux test $ ulimit -c
0
xiaosuo@gentux test $ ulimit -c 1000
xiaosuo@gentux test $ ulimit -c
1000
xiaosuo@gentux test $ ./a.out
段錯誤 (core dumped)
xiaosuo@gentux test $ ls
a.out core d.c f.c g.c pango.c test_iconv.c test_regex.c
core文件終于產生了,用gdb調試一下看看吧:
xiaosuo@gentux test $ gdb ./a.out core
GNU gdb 6.5
Copyright (C) 2006 Free Software Foundation, Inc.
GDB is free software, covered by the GNU General Public License, and you are
welcome to change it and/or distribute copies of it under certain conditions.
Type "show copying" to see the conditions.
There is absolutely no warranty for GDB. Type "show warranty" for details.
This GDB was configured as "i686-pc-linux-gnu"...Using host libthread_db library "/lib/libthread_db.so.1".


warning: Can't read pathname for load map: 輸入/輸出錯誤.
Reading symbols from /lib/libc.so.6...done.
Loaded symbols for /lib/libc.so.6
Reading symbols from /lib/ld-linux.so.2...done.
Loaded symbols for /lib/ld-linux.so.2
Core was generated by `./a.out'.
Program terminated with signal 11, Segmentation fault.
#0 0x08048524 in dummy_function () at d.c:4
4 *ptr = 0x00;
哇,好歷害,還是一步就定位到了錯誤所在地,佩服一下Linux/Unix系統的此類設計。
接著考慮下去,以前用windows系統下的ie的時侯,有時打開某些網頁,會出現“運行時錯誤”,這個時侯如果恰好你的機器上又裝有windows的編譯器的話,他會彈出來一個對話框,問你是否進行調試,如果你選擇是,編譯器將被打開,并進入調試狀態,開始調試。
Linux下如何做到這些呢?我的大腦飛速地旋轉著,有了,讓它在SIGSEGV的handler中調用gdb,于是第三個方法又誕生了:
3.段錯誤時啟動調試:
#include <stdio.h>
#include <stdlib.h>
#include <signal.h>
#include <string.h>

void dump(int signo)
{
char buf[1024];
char cmd[1024];
FILE *fh;

snprintf(buf, sizeof(buf), "/proc/%d/cmdline", getpid());
if(!(fh = fopen(buf, "r")))
exit(0);
if(!fgets(buf, sizeof(buf), fh))
exit(0);
fclose(fh);
if(buf[strlen(buf) - 1] == 'n')
buf[strlen(buf) - 1] = '';
snprintf(cmd, sizeof(cmd), "gdb %s %d", buf, getpid());
system(cmd);

exit(0);
}

void
dummy_function (void)
{
unsigned char *ptr = 0x00;
*ptr = 0x00;
}

int
main (void)
{
signal(SIGSEGV, &dump);
dummy_function ();

return 0;
}
編譯運行效果如下:
xiaosuo@gentux test $ gcc -g -rdynamic f.c
xiaosuo@gentux test $ ./a.out
GNU gdb 6.5
Copyright (C) 2006 Free Software Foundation, Inc.
GDB is free software, covered by the GNU General Public License, and you are
welcome to change it and/or distribute copies of it under certain conditions.
Type "show copying" to see the conditions.
There is absolutely no warranty for GDB. Type "show warranty" for details.
This GDB was configured as "i686-pc-linux-gnu"...Using host libthread_db library "/lib/libthread_db.so.1".

Attaching to program: /home/xiaosuo/test/a.out, process 9563
Reading symbols from /lib/libc.so.6...done.
Loaded symbols for /lib/libc.so.6
Reading symbols from /lib/ld-linux.so.2...done.
Loaded symbols for /lib/ld-linux.so.2
0xffffe410 in __kernel_vsyscall ()
(gdb) bt
#0 0xffffe410 in __kernel_vsyscall ()
#1 0xb7ee4b53 in waitpid () from /lib/libc.so.6
#2 0xb7e925c9 in strtold_l () from /lib/libc.so.6
#3 0x08048830 in dump (signo=11) at f.c:22
#4 <signal handler called>
#5 0x0804884c in dummy_function () at f.c:31
#6 0x08048886 in main () at f.c:38
怎么樣?是不是依舊很酷?
以上方法都是在系統上有gdb的前提下進行的,如果沒有呢?其實glibc為我們提供了此類能夠dump棧內容的函數簇,詳見/usr/include/execinfo.h(這些函數都沒有提供man page,難怪我們找不到),另外你也可以通過gnu的手冊進行學習。
4.利用backtrace和objdump進行分析:
重寫的代碼如下:
#include <execinfo.h>
#include <stdio.h>
#include <stdlib.h>
#include <signal.h>

/* A dummy function to make the backtrace more interesting. */
void
dummy_function (void)
{
unsigned char *ptr = 0x00;
*ptr = 0x00;
}

void dump(int signo)
{
void *array[10];
size_t size;
char **strings;
size_t i;

size = backtrace (array, 10);
strings = backtrace_symbols (array, size);

printf ("Obtained %zd stack frames.n", size);

for (i = 0; i < size; i++)
printf ("%sn", strings[i]);

free (strings);

exit(0);
}

int
main (void)
{
signal(SIGSEGV, &dump);
dummy_function ();

return 0;
}
編譯運行結果如下:
xiaosuo@gentux test $ gcc -g -rdynamic g.c
xiaosuo@gentux test $ ./a.out
Obtained 5 stack frames.
./a.out(dump+0x19) [0x80486c2]
[0xffffe420]
./a.out(main+0x35) [0x804876f]
/lib/libc.so.6(__libc_start_main+0xe6) [0xb7e02866]
./a.out [0x8048601]
這次你可能有些失望,似乎沒能給出足夠的信息來標示錯誤,不急,先看看能分析出來什么吧,用objdump反匯編程序,找到地址0x804876f對應的代碼位置:
xiaosuo@gentux test $ objdump -d a.out

8048765: e8 02 fe ff ff call 804856c <signal@plt>
804876a: e8 25 ff ff ff call 8048694 <dummy_function>
804876f: b8 00 00 00 00 mov $0x0,%eax
8048774: c9 leave
我們還是找到了在哪個函數(dummy_function)中出錯的,信息已然不是很完整,不過有總比沒有好的啊!

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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精品欧美一区| av不卡在线观看| 欧美激情一区二区三区四区| 美国成人毛片| 樱花yy私人影院亚洲| 欧美一区二区视频在线观看| 午夜久久资源| 国产精品chinese| 一区二区三区四区五区精品视频| 日韩午夜激情电影| 欧美福利影院| 亚洲精品一区中文| 99精品国产99久久久久久福利| 久久婷婷丁香| 欧美激情亚洲综合一区| 亚洲国产精品成人综合色在线婷婷| 欧美一区二区三区在线免费观看| 欧美影院在线| 国产一区二区精品| 久久精品道一区二区三区| 欧美中文字幕第一页| 国产一区二区精品| 久久久久**毛片大全| 久久中文久久字幕| 最新日韩在线视频| 欧美女激情福利| 亚洲婷婷在线| 久久xxxx| 亚洲国产精选| 欧美日韩亚洲一区二区| 亚洲一区国产精品| 久久久亚洲精品一区二区三区| 国内精品一区二区三区| 免费看av成人| 日韩一级精品视频在线观看| 亚洲欧美清纯在线制服| 黑人操亚洲美女惩罚| 久久久久久亚洲精品中文字幕| 亚洲国产精品黑人久久久| 在线亚洲电影| 国产日韩一区欧美| 欧美暴力喷水在线| 亚洲一级二级| 欧美激情影院| 性久久久久久久久久久久| 一区三区视频| 欧美午夜不卡| 久久综合久久久久88| 夜夜嗨av一区二区三区四区| 久久精品观看| 9色精品在线| 精品69视频一区二区三区 | 亚洲国产aⅴ天堂久久| 一区二区三区成人 | 国产精品婷婷| 久久这里有精品15一区二区三区 | 欧美激情视频一区二区三区免费 | 欧美成人首页| 亚洲免费视频成人| 亚洲高清在线| 国产日韩欧美精品在线| 欧美精品大片| 久久精品夜夜夜夜久久| 99国产一区二区三精品乱码| 久久尤物视频| 性欧美video另类hd性玩具| 亚洲三级色网| 在线观看成人一级片| 国产精品亚洲欧美| 欧美极品影院| 久久综合九色99| 欧美一区二区三区另类| 亚洲乱码国产乱码精品精98午夜| 久久久久网址| 欧美在线视频一区二区| 一区二区三区日韩精品视频| 亚洲国产欧美日韩精品| 国产一区二区三区久久悠悠色av | 极品裸体白嫩激情啪啪国产精品| 欧美日韩视频免费播放| 免费久久99精品国产自在现线| 香蕉久久夜色精品国产| 中文精品视频| 亚洲视频axxx| 一本大道久久精品懂色aⅴ| 亚洲第一偷拍| 欧美阿v一级看视频| 麻豆av一区二区三区久久| 久久国产精品久久久久久| 午夜视频在线观看一区| 亚洲一区二区伦理| 亚洲手机在线| 亚洲在线不卡| 亚洲欧美综合v| 欧美影院精品一区| 久久av最新网址| 久久国产精品久久久久久久久久| 午夜久久久久久久久久一区二区| 亚洲男人av电影| 欧美一区二区视频网站| 久久av二区| 久久久噜噜噜久久狠狠50岁| 久久久久久久波多野高潮日日| 久久国产视频网| 裸体丰满少妇做受久久99精品| 久久亚洲美女| 亚洲高清免费| 日韩一区二区免费高清| 99精品热视频| 亚洲欧美经典视频| 久久久高清一区二区三区| 久久久免费av| 欧美肥婆bbw| 国产精品成人一区二区三区吃奶| 欧美三级不卡| 国产精品永久| 一区免费观看| 9国产精品视频| 欧美一区午夜视频在线观看| 久久久久在线| 亚洲欧洲日夜超级视频| 亚洲小视频在线| 久久久久综合一区二区三区| 欧美大片免费看| 国产精品久在线观看| 国产一区二区三区丝袜 | 日韩视频―中文字幕| 正在播放欧美一区| 新67194成人永久网站| 久久综合久久综合九色| 亚洲激情图片小说视频| 亚洲一区中文| 免费观看成人鲁鲁鲁鲁鲁视频| 欧美日韩亚洲综合一区| 韩国av一区| 一区二区冒白浆视频| 久久久精品动漫| 91久久国产综合久久| 亚洲欧美一区二区激情| 欧美1区免费| 国产视频久久网| 一区二区三区蜜桃网| 久久久人成影片一区二区三区观看| 亚洲第一综合天堂另类专| 午夜精品久久久久久久男人的天堂 | 久久久精品国产99久久精品芒果| 亚洲国产成人av| 久久xxxx| 国产精品扒开腿做爽爽爽视频| 在线观看不卡av| 欧美一区二区三区精品电影| 亚洲精品1区2区| 久久视频在线视频| 国产精品无码永久免费888| 亚洲日本无吗高清不卡| 久久久99久久精品女同性| 在线亚洲伦理| 欧美精品日韩精品| 今天的高清视频免费播放成人| 亚洲综合视频网| 亚洲欧洲精品一区二区三区不卡 | 欧美呦呦网站| 9色porny自拍视频一区二区| 可以看av的网站久久看| 国产日韩欧美在线视频观看| 亚洲永久在线| 亚洲精品人人| 欧美成人精品影院| 亚洲电影免费观看高清完整版在线| 午夜精品久久久久久久99樱桃| 亚洲欧洲一区二区天堂久久| 美日韩精品视频| 在线欧美福利| 久久亚洲国产成人| 欧美自拍丝袜亚洲| 国产视频在线一区二区| 欧美一级久久| 亚洲欧美日本国产有色| 国产精品男人爽免费视频1| 亚洲一级二级| 亚洲视频免费| 国产精品视频成人| 西西人体一区二区| 亚洲免费一在线| 国产欧美va欧美va香蕉在| 欧美一级视频一区二区| 亚洲制服av| 国产主播一区二区三区四区| 久久精品国产99国产精品| 久久成人18免费网站| 激情文学综合丁香| 欧美va天堂在线|