debug-段错误
大多數(shù)的linux發(fā)型版本關(guān)閉了內(nèi)核轉(zhuǎn)存儲(chǔ)功能,使用:
ulimit -c查看自己的linux內(nèi)核是否關(guān)閉了核心轉(zhuǎn)存儲(chǔ)功能;
andrew@andrew-Thurley:/proc/net$ ulimit -c 0輸出為0 說(shuō)明內(nèi)核關(guān)閉了核心轉(zhuǎn)存儲(chǔ)功能,-c選項(xiàng)表示內(nèi)核轉(zhuǎn)存儲(chǔ)文件的大小限制,使用如下命令可以放開內(nèi)核轉(zhuǎn)存儲(chǔ)功能:
ulimit -c unlimited這個(gè)命令是不限制內(nèi)核轉(zhuǎn)存儲(chǔ)文件的大小,但是內(nèi)存有限的情況下可能會(huì)想限制一下內(nèi)核轉(zhuǎn)存儲(chǔ)文件的大小,可以使用以下命令設(shè)置內(nèi)核轉(zhuǎn)存儲(chǔ)文件的大小,如下命令限制內(nèi)核轉(zhuǎn)存儲(chǔ)文件的大小為1G;
ulimit -c 1073741824查看系統(tǒng)的核心轉(zhuǎn)存儲(chǔ)是否被限制為0 ,若是被限制為0使用
將核心轉(zhuǎn)存儲(chǔ)設(shè)置為無(wú)限制
編譯程序加上-g 選項(xiàng)
調(diào)試
要想使用GDB調(diào)試內(nèi)核轉(zhuǎn)存儲(chǔ)文件,應(yīng)當(dāng)以以下方式啟動(dòng)GDB
gdb -c core文件 ./a.out
在使用大型文件系統(tǒng)時(shí),會(huì)希望將內(nèi)核轉(zhuǎn)存儲(chǔ)放在固定的位置。默認(rèn)情況下會(huì)在當(dāng)前目錄下生成,但是可能很難弄清文件在哪在哪生成。這種情況下可以配置/etc/systl.conf文件,來(lái)決定文件的生成目錄和文件的命名;
例如在/etc/systl.conf 文件中添加如下內(nèi)容:
kernel.core_pattern = /var/core/%t-%e-%p-%c.core
kernel.core_uses_pid = 0
然后執(zhí)行
sysctl -p
上述生成核心轉(zhuǎn)存儲(chǔ)文件的命名:
時(shí)刻-進(jìn)程名-PID-內(nèi)核轉(zhuǎn)存儲(chǔ)最大大小.core
kernel.core_pattern中可以設(shè)置的格式符:
kernel.core_uses_pid = 0 設(shè)置為0是因?yàn)槲覀兏淖兞宋募械腜ID的位置,如果設(shè)置該值為1,文件名末尾就會(huì)添加.PID
segment.c文件示例
andrew@andrew-Thurley:/work/linux-sys/DEBUG/segmentation$ gdb -c core ./segment GNU gdb (Ubuntu 7.11.1-0ubuntu1~16.5) 7.11.1 Copyright (C) 2016 Free Software Foundation, Inc. License GPLv3+: GNU GPL version 3 or later <http://gnu.org/licenses/gpl.html> This is free software: you are free to change and redistribute it. There is NO WARRANTY, to the extent permitted by law. Type "show copying" and "show warranty" for details. This GDB was configured as "x86_64-linux-gnu". Type "show configuration" for configuration details. For bug reporting instructions, please see: <http://www.gnu.org/software/gdb/bugs/>. Find the GDB manual and other documentation resources online at: <http://www.gnu.org/software/gdb/documentation/>. For help, type "help". Type "apropos word" to search for commands related to "word"... Reading symbols from ./segment...done. [New LWP 7103] Core was generated by `./segment'. Program terminated with signal SIGSEGV, Segmentation fault. #0 0x00000000004004ed in main (argc=1, argv=0x7ffcab228948) at segment.c:16 16 (*a) = 1; (gdb) bt full #0 0x00000000004004ed in main (argc=1, argv=0x7ffcab228948) at segment.c:16a = 0x0 (gdb) bt #0 0x00000000004004ed in main (argc=1, argv=0x7ffcab228948) at segment.c:16 (gdb) frame 0 #0 0x00000000004004ed in main (argc=1, argv=0x7ffcab228948) at segment.c:16 16 (*a) = 1;(gdb) disassemble 0x00000000004004ed Dump of assembler code for function main:0x00000000004004d6 <+0>: push %rbp0x00000000004004d7 <+1>: mov %rsp,%rbp0x00000000004004da <+4>: mov %edi,-0x14(%rbp)0x00000000004004dd <+7>: mov %rsi,-0x20(%rbp)0x00000000004004e1 <+11>: movq $0x0,-0x8(%rbp)0x00000000004004e9 <+19>: mov -0x8(%rbp),%rax => 0x00000000004004ed <+23>: movl $0x1,(%rax)0x00000000004004f3 <+29>: mov $0x0,%eax0x00000000004004f8 <+34>: pop %rbp0x00000000004004f9 <+35>: retq End of assembler dump. (gdb) list 11 12 int *a = NULL; 13 14 15 16 (*a) = 1; 17 18 19 20 return 0;GDB list命令可以查看函數(shù)周圍的源代碼
可以看出代碼在中為指向NULL的指針賦值,產(chǎn)生段錯(cuò)誤;
4.使用內(nèi)核轉(zhuǎn)儲(chǔ)掩碼來(lái)排除要轉(zhuǎn)儲(chǔ)的內(nèi)存塊
你可能會(huì)因?yàn)椴幌MO(shè)置ulimit 的時(shí)候太僵硬導(dǎo)致空間不夠沒有得到完整的轉(zhuǎn)儲(chǔ),所以設(shè)置ulimit 為"unlimited" (不限制)。但是如果執(zhí)行一個(gè)占用內(nèi)存很恐怖的程序,當(dāng)這個(gè)程序內(nèi)核轉(zhuǎn)儲(chǔ)的時(shí)候也就會(huì)生成體積很恐怖的轉(zhuǎn)儲(chǔ)文件。
為了避免這種情況,可以指定內(nèi)核轉(zhuǎn)儲(chǔ)掩碼來(lái)排除要轉(zhuǎn)儲(chǔ)的內(nèi)存段。
掩碼請(qǐng)查看/usr/src/linux/Documentation/sysctl/kernel.txt 中的3.4 小節(jié),沒有內(nèi)核源碼可以到這里的網(wǎng)絡(luò)版,這里摘錄出來(lái)如下:
The following 7 memory types are supported:
- (bit 0) anonymous private memory(匿名私有內(nèi)存段)
- (bit 1) anonymous shared memory(匿名共享內(nèi)存段)
- (bit 2) file-backed private memory(file-backed 私有內(nèi)存段)
- (bit 3) file-backed shared memory(file-bakced 共享內(nèi)存段)
- (bit 4) ELF header pages in file-backed private memory areas (it is
effective only if the bit 2 is cleared)(ELF 文件映射,只有在bit 2 復(fù)位的時(shí)候才起作用) - (bit 5) hugetlb private memory(大頁(yè)面私有內(nèi)存)
- (bit 6) hugetlb shared memory(大頁(yè)面共享內(nèi)存)
設(shè)置方法很簡(jiǎn)單:找到程序的PID,然后修改/proc/PID/coredump_filter 的值。
如果你要設(shè)置某些還沒有運(yùn)行的進(jìn)程的內(nèi)核轉(zhuǎn)儲(chǔ)掩碼,請(qǐng)修改/proc/self/coredump_filter 的值。
PS:
a. 默認(rèn)的coredump_filter 的值一般是0x23,至于代表什么,請(qǐng)換成二進(jìn)制00100011,從右向左看,bit 0、bit 1、bit 5 被置位,也就是說(shuō)會(huì)轉(zhuǎn)儲(chǔ)所有的匿名內(nèi)存段和大頁(yè)面私有內(nèi)存段。
b. 共享內(nèi)存段都是一樣的,可以不必轉(zhuǎn)儲(chǔ)。
全局設(shè)置沒有什么好說(shuō)的,把你的配置寫入/etc/profile.d/ 目錄下任意一個(gè)新建的文件當(dāng)中,別忘了設(shè)置屬組和所有者為root:root,權(quán)限為751。
根據(jù)上面說(shuō)的,寫入的要有一條ulimit 指令,可能還有一條sysctl 指令,最后可能還有一條cat 指令。
更多的設(shè)置請(qǐng)查看上面提到的kernel.txt 和proc.txt,例如你想把SUID 程序也轉(zhuǎn)儲(chǔ),你需要
sysctl -w 'fs.suid_dumpable=1'總結(jié)
- 上一篇: 作者:李大中(1976-),男,中国联合
- 下一篇: linux中GDB详细使用手册