威尼斯wns.9778官网活动_vnsc威尼斯城官网

热门关键词: 威尼斯wns.9778官网活动,vnsc威尼斯城官网
当前位置:威尼斯wns.9778官网活动 > 计算机教程 > 威尼斯wns.9778官网活动:Linux下的段错误(Segmenta

威尼斯wns.9778官网活动:Linux下的段错误(Segmenta

文章作者:计算机教程 上传时间:2019-05-11

1    使用非法的内存地址(指针),包括使用未经初始化及已经释放的指针、不存在的地址、受系统保护的地址,只读的地址等,这一类也是最常见和最好解决的段错误问题,使用GDB print一下即可知道原因。

 

2    内存读/写越界。包括数组访问越界,或在使用一些写内存的函数时,长度指定不正确或者这些函数本身不能指定长度,典型的函数有strcpy(strncpy),sprintf(snprint)等等。

Linux开发中常见段错误问题原因分析

1 使用非法的内存地址(指针),包括使用未经初始化及已经释放的指针、不存在的地址、受系统保护的地址,只读的地址等,这一类也是最常见和最好解决的段错误问题,使用GDB print一下即可知道原因。

2 内存读/写越界。包括数组访问越界,或在使用一些写内存的函数时,长度指定不正确或者这些函数本身不能指定长度,典型的函数有strcpy(strncpy),sprintf(snprint)等等。

3 对于C 对象,应该通过相应类的接口来去内存进行操作,禁止通过其返回的指针对内存进行写操作,典型的如string类的c_str()接口,如果你强制往其返回的指针进行写操作肯定会段错误的,因为其返回的地址是只读的。

4 函数不要返回其中局部对象的引用或地址,当函数返回时,函数栈弹出,局部对象的地址将失效,改写或读这些地址都会造成未知的后果。

5 避免在栈中定义过大的数组,否则可能导致进程的栈空间不足,此时也会出现段错误,同样的,在创建进程/线程时如果不知道此线程/进程最大需要多少栈空间时最好不要在代码中指定栈大小,应该使用系统默认的,这样问题比较好查,ulimit一下即可知道。这类问题也是为什么我的程序在其他平台跑得好好的,为什么一移植到这个平台就段错误了。

6 操作系统的相关限制,如:进程可以分配的最大内存,进程可以打开的最大文件描述符个数等,在Linux下这些需要通过ulimit、setrlimit、sysctl等来解除相关的限制,这类段错误问题在系统移植中也经常发现,以前我们移植Linux的程序到VxWorks下时经常遇到(VxWorks要改内核配置来解决)。

7 多线程的程序,涉及到多个线程同时操作一块内存时必须进行互斥,否则内存中的内容将不可预料。

8 在多线程环境下使用非线程安全的函数调用,例如 strerror 函数等。

9 在有信号的环境中,使用不可重入函数调用,而这些函数内部会读或写某片内存区,当信号中断时,内存写操作将被打断,而下次进入时将无法避免地出错。

10 跨进程传递某个地址,传递的都是经过映射的虚拟地址,对另外一个进程是不通用的。

11 某些有特殊要求的系统调用,例如epool_wait,正常情况下使用close关闭一个套接字后,epool会不再返回这个socket上的事件,但是如果你使用dup或dup2操作,将导致epool无法进行移除操作,此时再进行读写操作肯定是段错误的。

 

3    对于C 对象,应该通过相应类的接口来去内存进行操作,禁止通过其返回的指针对内存进行写操作,典型的如string类的c_str()接口,如果你强制往其返回的指针进行写操作肯定会段错误的,因为其返回的地址是只读的。

威尼斯wns.9778官网活动,Linux环境下段错误的产生原因及调试方法小结  

  1. 段错误是什么 

一句话来说,段错误是指访问的内存超出了系统给这个程序所设定的内存空间,例如访问了不存在的内存地址、访问了系统保护的内存地址、访问了只读的内存地址等等情况。这里贴一个对于“段错误”的准确定义(参考Answers.com): 

A segmentation fault (often shortened to segfault) is a particular error condition that can occur during the operation of computer software. In short, a segmentation fault occurs when a program attempts to access a memory location that it is not allowed to access, or attempts to access a memory location in a way that is not allowed (e.g., attempts to write to a read-only location, or to overwrite part of the operating system). Systems based on processors like the Motorola 68000 tend to refer to these events as Address or Bus errors. 

Segmentation is one approach to memory management and protection in the operating system. It has been superseded by paging for most purposes, but much of the terminology of segmentation is still used, "segmentation fault" being an example. Some operating systems still have segmentation at some logical level although paging is used as the main memory management policy. 

On Unix-like operating systems, a process that accesses invalid memory receives the SIGSEGV signal. On Microsoft Windows, a process that accesses invalid memory receives the STATUS_ACCESS_VIOLATION exception. 
  1. 段错误产生的原因 
    2.1 访问不存在的内存地址 
#include<stdio.h> 
#include<stdlib.h> 
void main() 
{ 
int *ptr = NULL; 
*ptr = 0; 
} 

 

2.2 访问系统保护的内存地址 

#include<stdio.h> 
#include<stdlib.h> 
void main() 
{ 
int *ptr = (int *)0; 
*ptr = 100; 
} 

  

2.3 访问只读的内存地址 

#include<stdio.h> 
#include<stdlib.h> 
#include<string.h> 
void main() 
{ 
char *ptr = "test"; 
strcpy(ptr, "TEST"); 
}

 

2.4 栈溢出 

#include<stdio.h> 
#include<stdlib.h> 
void main() 
{ 
main(); 
} 

  

等等其他原因。 

  1. 段错误信息的获取 
    程序发生段错误时,提示信息很少,下面有几种查看段错误的发生信息的途径。 

3.1 dmesg 
dmesg可以在应用程序crash掉时,显示内核中保存的相关信息。如下所示,通过dmesg命令可以查看发生段错误的程序名称、引起段错误发生的内存地址、指令指针地址、堆栈指针地址、错误代码、错误原因等。以程序2.3为例: 

panfeng@ubuntu:~/segfault$ dmesg 
[ 2329.479037] segfault3[2700]: segfault at 80484e0 ip 00d2906a sp bfbbec3c error 7 in libc-2.10.1.so[cb4000 13e000] 
3.2 -g 

使用gcc编译程序的源码时,加上-g参数,这样可以使得生成的二进制文件中加入可以用于gdb调试的有用信息。以程序2.3为例: 

panfeng@ubuntu:~/segfault$ gcc -g -o segfault3 segfault3.c 
3.3 nm 

使用nm命令列出二进制文件中的符号表,包括符号地址、符号类型、符号名等,这样可以帮助定位在哪里发生了段错误。以程序2.3为例: 

复制代码代码如下:

panfeng@ubuntu:~/segfault$ nm segfault3 
08049f20 d _DYNAMIC 
08049ff4 d _GLOBAL_OFFSET_TABLE_ 
080484dc R _IO_stdin_used 
w _Jv_RegisterClasses 
08049f10 d __CTOR_END__ 
08049f0c d __CTOR_LIST__ 
08049f18 D __DTOR_END__ 
08049f14 d __DTOR_LIST__ 
080484ec r __FRAME_END__ 
08049f1c d __JCR_END__ 
08049f1c d __JCR_LIST__ 
0804a014 A __bss_start 
0804a00c D __data_start 
08048490 t __do_global_ctors_aux 
08048360 t __do_global_dtors_aux 
0804a010 D __dso_handle 
w __gmon_start__ 
0804848a T __i686.get_pc_thunk.bx 
08049f0c d __init_array_end 
08049f0c d __init_array_start 
08048420 T __libc_csu_fini 
08048430 T __libc_csu_init 
U __libc_start_main@@GLIBC_2.0 
0804a014 A _edata 
0804a01c A _end 
080484bc T _fini 
080484d8 R _fp_hw 
080482bc T _init 
08048330 T _start 
0804a014 b completed.6990 
0804a00c W data_start 
0804a018 b dtor_idx.6992 
080483c0 t frame_dummy 
080483e4 T main 
U memcpy@@GLIBC_2.0 

  

3.4 ldd 
使用ldd命令查看二进制程序的共享链接库依赖,包括库的名称、起始地址,这样可以确定段错误到底是发生在了自己的程序中还是依赖的共享库中。以程序2.3为例: 

panfeng@ubuntu:~/segfault$ ldd ./segfault3 
linux-gate.so.1 => (0x00e08000) 
libc.so.6 => /lib/tls/i686/cmov/libc.so.6 (0x00675000) 
/lib/ld-linux.so.2 (0x00482000) 

  

  1. 段错误的调试方法 
    4.1 使用printf输出信息 
    这个是看似最简单但往往很多情况下十分有效的调试方式,也许可以说是程序员用的最多的调试方式。简单来说,就是在程序的重要代码附近加上像printf这类输出信息,这样可以跟踪并打印出段错误在代码中可能出现的位置。 

为了方便使用这种方法,可以使用条件编译指令#ifdef DEBUG和#endif把printf函数包起来。这样在程序编译时,如果加上-DDEBUG参数就能查看调试信息;否则不加该参数就不会显示调试信息。 

4.2 使用gcc和gdb 
4.2.1 调试步骤 
1、为了能够使用gdb调试程序,在编译阶段加上-g参数,以程序2.3为例: 

panfeng@ubuntu:~/segfault$ gcc -g -o segfault3 segfault3.c 
2、使用gdb命令调试程序: 

panfeng@ubuntu:~/segfault$ gdb ./segfault3 
GNU gdb (GDB) 7.0-ubuntu 
Copyright (C) 2009 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 "i486-linux-gnu". 
For bug reporting instructions, please see: 
<http://www.gnu.org/software/gdb/bugs/>... 
Reading symbols from /home/panfeng/segfault/segfault3...done. 
(gdb) 

  

3、进入gdb后,运行程序: 

(gdb) run 
Starting program: /home/panfeng/segfault/segfault3 

Program received signal SIGSEGV, Segmentation fault. 
0x001a306a in memcpy () from /lib/tls/i686/cmov/libc.so.6 
(gdb) 

 

从输出看出,程序2.3收到SIGSEGV信号,触发段错误,并提示地址0x001a306a、调用memcpy报的错,位于/lib/tls/i686/cmov/libc.so.6库中。 

4、完成调试后,输入quit命令退出gdb: 

(gdb) quit 
A debugging session is active. 

Inferior 1 [process 3207] will be killed. 

Quit anyway? (y or n) y 

 

4.2.2 适用场景 
1、仅当能确定程序一定会发生段错误的情况下使用。 

2、当程序的源码可以获得的情况下,使用-g参数编译程序。 

3、一般用于测试阶段,生产环境下gdb会有副作用:使程序运行减慢,运行不够稳定,等等。 

4、即使在测试阶段,如果程序过于复杂,gdb也不能处理。 

4.3 使用core文件和gdb 
在4.2节中提到段错误会触发SIGSEGV信号,通过man 7 signal,可以看到SIGSEGV默认的handler会打印段错误出错信息,并产生core文件,由此我们可以借助于程序异常退出时生成的core文件中的调试信息,使用gdb工具来调试程序中的段错误。 

4.3.1 调试步骤 
1、在一些Linux版本下,默认是不产生core文件的,首先可以查看一下系统core文件的大小限制: 

panfeng@ubuntu:~/segfault$ ulimit -c 

2、可以看到默认设置情况下,本机Linux环境下发生段错误时不会自动生成core文件,下面设置下core文件的大小限制(单位为KB): 

panfeng@ubuntu:~/segfault$ ulimit -c 1024 
panfeng@ubuntu:~/segfault$ ulimit -c 
1024 
3、运行程序2.3,发生段错误生成core文件: 

panfeng@ubuntu:~/segfault$ ./segfault3 
段错误 (core dumped) 
4、加载core文件,使用gdb工具进行调试: 

复制代码代码如下:

panfeng@ubuntu:~/segfault$ gdb ./segfault3 ./core 
GNU gdb (GDB) 7.0-ubuntu 
Copyright (C) 2009 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 "i486-linux-gnu". 
For bug reporting instructions, please see: 
<http://www.gnu.org/software/gdb/bugs/>... 
Reading symbols from /home/panfeng/segfault/segfault3...done. 

warning: Can't read pathname for load map: 输入/输出错误. 
Reading symbols from /lib/tls/i686/cmov/libc.so.6...(no debugging symbols found)...done. 
Loaded symbols for /lib/tls/i686/cmov/libc.so.6 
Reading symbols from /lib/ld-linux.so.2...(no debugging symbols found)...done. 
Loaded symbols for /lib/ld-linux.so.2 
Core was generated by `./segfault3'. 
Program terminated with signal 11, Segmentation fault. 
#0 0x0018506a in memcpy () from /lib/tls/i686/cmov/libc.6 

  

从输出看出,同4.2.1中一样的段错误信息。 

5、完成调试后,输入quit命令退出gdb: 

(gdb) quit 
4.3.2 适用场景 
1、适合于在实际生成环境下调试程序的段错误(即在不用重新发生段错误的情况下重现段错误)。 

2、当程序很复杂,core文件相当大时,该方法不可用。 

4.4 使用objdump 
4.4.1 调试步骤 
1、使用dmesg命令,找到最近发生的段错误输出信息: 

panfeng@ubuntu:~/segfault$ dmesg 
... ... 
[17257.502808] segfault3[3320]: segfault at 80484e0 ip 0018506a sp bfc1cd6c error 7 in libc-2.10.1.so[110000 13e000] 
其中,对我们接下来的调试过程有用的是发生段错误的地址:80484e0和指令指针地址:0018506a。 

2、使用objdump生成二进制的相关信息,重定向到文件中: 

panfeng@ubuntu:~/segfault$ objdump -d ./segfault3 > segfault3Dump 
其中,生成的segfault3Dump文件中包含了二进制文件的segfault3的汇编代码。 

3、在segfault3Dump文件中查找发生段错误的地址: 

panfeng@ubuntu:~/segfault$ grep -n -A 10 -B 10 "80484e0" ./segfault3Dump 
121- 80483df: ff d0 call * 
		

本文由威尼斯wns.9778官网活动发布于计算机教程,转载请注明出处:威尼斯wns.9778官网活动:Linux下的段错误(Segmenta

关键词:

Copyright © 2018 http://www.itcinternet.net All Rights Reserved |

威尼斯wns888_wns888 coom是一个很好的娱乐平台,因为威尼斯wns888_wns888 coom能够为大家带来真正的真钱享受,点击进入威尼斯wns888_wns888 coom.,每天向广大玩家不定时的提供各种娱乐资讯。