当前位置:   article > 正文

Process finished with exit code 139 (interrupted by signal 11: SIGSEGV)

process finished with exit code 139 (interrupted by signal 11: sigsegv)

在PyCharm运行Python代码运行时出现错误导致退出:Process finished with exit code 139 (interrupted by signal 11: SIGSEGV)

139错误是Linux错误代码(参考Linux code error 139),异常导致某个进程崩溃产生的,可以出现在任何语言、程序中。其产生的原因也有很多,可能时内存泄漏、依赖的库版本不匹配、环境不满足等。在程序中只能debug逐步定位出错位置,揣测出错原因。

类似的段错误包括:SIGSEGV, SIGABRT, SIGBUS,以下转载自:程序段错误处理

SIGSEGV
SIGSEGV --- Segment Fault. The possible cases of your encountering this error are:
buffer overflow --- usually caused by a pointer reference out of range.
stack overflow --- please keep in mind that the default stack size is 8192K.
illegal file access --- file operations are forbidden on our judge system.
Example

  1. char *p = NULL;
  2. *p = 1;

SIGBUS
 address alignment可能会导致SIGBUS.
SIGABRT
对同一个指针free() 2次可能会产生SIGABRT
SIGBUS与SIGSEGV区别:
SIGBUS(Bus error)意味着指针所对应的地址是有效地址,但总线不能正常使用该指针。通常是未对齐的数据访问所致。
SIGSEGV(Segment fault)意味着指针所对应的地址是无效地址,没有物理内存对应该地址。
结论
SIGSEGV:  一般是非法内存访问错误;
SIGABRT:  重复释放内存则会导致;
SIGBUS :  address alignment可能会导致。

程序中处理
1. 函数说明

  1. #include <execinfo.h>
  2. int backtrace(void **buffer, int size);

       该函数获取当前线程的调用堆栈,获取的信息将会被存放在buffer中,它是一个指针数组,参数size用来指定buffer中可以保存多少个void元素。函数的返回值是实际返回的void元素个数。buffer中的void*元素实际是从堆栈中获取的返回地址。

char **backtrace_symbols(void *const *buffer, int size);

       该函数将backtrace函数获取的信息转化为一个字符串数组,参数buffer是backtrace获取的堆栈指针,size是backtrace返回值。函数返回值是一个指向字符串数组的指针,它包含char*元素个数为size。每个字符串包含了一个相对于buffer中对应元素的可打印信息,包括函数名、函数偏移地址和实际返回地址。
       backtrace_symbols生成的字符串占用的内存是malloc出来的,但是是一次性malloc出来的,释放是只需要一次性释放返回的二级指针即可。

void backtrace_symbols_fd(void *const *buffer, int size, int fd);

       该函数与backtrace_symbols函数功能相同,只是它不会malloc内存,而是将结果写入文件描述符为fd的文件中,每个函数对应一行。该函数可重入。

2. 首先注册信号

  1. signal(SIGSEGV, segv_signal_handler);
  2. signal(SIGABRT, segv_signal_handler);
  3. signal(SIGBUS, segv_signal_handler);

3. 在信号处理函数中,打印堆栈信息
       当程序出现崩溃等异常时,会接收到内核发送给进程的异常信号,进程接收到异常信号后,可以在处理信号的时候将程序的堆栈信息打印出来,以便于程序调试。

  1. #include <stdio.h>
  2. #include <string.h>
  3. #include <signal.h>
  4. #include <stdlib.h>
  5. #include <execinfo.h>
  6. void segv_signal_handler(int signum)
  7. {
  8. #define BACKTRACE_SIZE 16
  9.     int nptrs, i;
  10.     void *buffer[BACKTRACE_SIZE]={0};
  11.     char **strings;
  12.     
  13.     printf("\n>>>>>>>>> Catch Signal [%d] <<<<<<<<<\n", signum);
  14.     nptrs = backtrace(buffer, BACKTRACE_SIZE);
  15.     strings = backtrace_symbols(buffer, nptrs);
  16.     if (!strings) {
  17.         printf("backtrace_symbols() error");
  18.         exit(-1);
  19.     }
  20.     
  21.     printf("===== print backtrace begin =====\n");
  22.     for (i=0; i<nptrs; i++) {
  23.         printf("[%02d] %s\n", i, strings[i]);
  24.     }
  25.     printf("===== print backtrace end =====\n\n");
  26.     
  27.     free(strings);
  28.     exit(-1);

4. 函数使用注意事项
backtrace() 的实现依赖于栈指针(fp寄存器),在gcc编译过程中任何非零的优化等级(-On参数)或加入了栈指针优化参数-fomit-frame-pointer后多将不能正确得到程序栈信息;
backtrace_symbols() 的实现需要符号名称的支持,在gcc编译过程中需要加入 -rdynamic 参数;
内联函数没有栈帧,它在编译过程中被展开在调用的位置;
尾调用优化(Tail-call Optimization)将复用当前函数栈,而不再生成新的函数栈,这将导致栈信息不能正确被获取。
 

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/Cpp五条/article/detail/88770
推荐阅读
相关标签
  

闽ICP备14008679号