【Linux】进程信号中的 core dump 标记位

简介: 【Linux】进程信号中的 core dump 标记位

一、什么是core dump

我们知道所有的程序最终运行起来,都会变成进程,进程在运行时可能会异常终止或崩溃,而Linux操作系统会将程序当时的内存状态记录下来,保存在一个文件中,这种行为就叫做Core Dump(中文有的翻译成核心转储)。

保存的这个文件通常是:该进程的同目录下以core.PID的方式命名的文件。

二、core dump的使用

1、开启core dump

在Linux下core dump选项一般是被关闭的,我们可以通过ulimit -a查看当前Linux下系统资源的限制。

可以看到,core file size的大小是0,这说明系统不允许我们生成core file文件 ,我们可以使用命令设置生成的core file文件的大小的最大限制。

ulimit -c 10240

可以看到使用此命令以后我们生成的core file文件的大小的最大限制就变为了10240 blocks了。

2、生成core file文件

在Linux下有很多信号我们可以使用kill -l查看:

kill -l

但是并不是所有的信号引起的退出都会产生core file文件,只有有core标志的信号引起的退出才会产生core file文件,我们可以通过 man 7 signal 查看信号的详细信息

man 7 signal

11 号信号SIGSEGV是一个段错误的信号,当你的进程有内存越界等问题时,通常会收到该信号,可以看到该信号是有core标志的。

2号信号SIGINT其实就是我们常用的Ctrl + C 键产生的信号,可以看到该进程是没有core标志的

下面我们用代码来验证:

#include <iostream>
#include <unistd.h>
using namespace std;
int main()
{
    while (true)
    {
        cout << "我是一个正在运行的进程..." << endl;
        sleep(1);
    }
    return 0;
}

对于这个死循环代码我们使用Ctrl + C 来进行终止,观察是否有core file文件的产生。

可以看到并没有core file文件的生成。

我们再来看下面的代码:

#include <iostream>
#include <unistd.h>
using namespace std;
int main()
{
    int* p = nullptr;
    *p = 10;
    cout << "野指针问题" << endl;
    return 0;
}

很明显这里会收到SIGSEGV信号,而SIGSEGV是有core标志的,因此此进程运行完毕以后应该生成core file文件。

运行结果:

可以看到确实生成了core file文件,我们打开该文件:

发现是乱码,这时因为core file里面是数据都是内存中的二进制数据,我们不使用特殊编码是看不懂里面的含义的。

3、验证进程退出码里面的core dump标志位

在以前我们学习进程等待时一定学习过:对于一个存储了进程的退出码的变量,其内部结构是这样的: 次第8位表示退出码,最低7位表示终止信号,终止信号的前一位就是core dump标志位。

按照这样的结构,如果我们创建一个子进程,让子进程直接遇到野指针收到SIGSEGV信号直接退出,然后我们在父进程里面检查core dump的标志位是否被置为1

#include <iostream>
#include <cstdlib>
#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>
using namespace std;
int main()
{
    pid_t id = fork();
    if (id == 0)
    {
        // 子进程
        int* p = nullptr;
        *p = 10;
        cout << "野指针问题" << endl;
        exit(0);
    }
    int status = 0;
    wait(&status);
    cout << "子进程的退出信号:" << (status & 0x7F) << endl;
    cout << "子进程的core dump标志位:" << ((status >> 7) & 0x1) << endl;
    return 0;
}

运行结果:

当然这个结果是在core dump被开启的条件下,那么我们将core dump关闭运行的结果还会一致吗?

我们继续实验:

运行同样的代码,结果是:

可以看到 core dump标志位被改为了0

结论

  1. 同样的程序,在core dump是否开启时会有不同的效果。
  2. 如果core dump开启,遇到有core 标志的信号,会进行核心转储,并且退出码里面的core dump 标志位会被置为1
  3. 如果core dump关闭,遇到有core 标志的信号,也不会进行核心转储,并且退出码里面的core dump 标志位始终置为0

三、 core dump的应用

因为core file 文件内部有进程退出时的内存中的相关信息 ,所以我们可以用这些信息在gdb里面进行调试我们的代码,注意调试的程序要以debug模式发布。

例如下面的代码生成的core file 文件:

#include <iostream>
#include <unistd.h>
using namespace std;
int main()
{
    int* p = nullptr;
    *p = 10;
    cout << "野指针问题" << endl;
    return 0;
}

我们在gdb里面进行调试,然后直接使用下面的命令,gdb就能直接帮我们找到问题的根源了,这种调试手段一般被称为事后调试

core-file core文件

相关文章
|
9月前
|
安全 Linux
【Linux】阻塞信号|信号原理
本教程从信号的基本概念入手,逐步讲解了阻塞信号的实现方法及其应用场景。通过对这些技术的掌握,您可以更好地控制进程在处理信号时的行为,确保应用程序在复杂的多任务环境中正常运行。
324 84
|
8月前
|
并行计算 Linux
Linux内核中的线程和进程实现详解
了解进程和线程如何工作,可以帮助我们更好地编写程序,充分利用多核CPU,实现并行计算,提高系统的响应速度和计算效能。记住,适当平衡进程和线程的使用,既要拥有独立空间的'兄弟',也需要在'家庭'中分享和并行的成员。对于这个世界,现在,你应该有一个全新的认识。
305 67
|
7月前
|
Web App开发 Linux 程序员
获取和理解Linux进程以及其PID的基础知识。
总的来说,理解Linux进程及其PID需要我们明白,进程就如同汽车,负责执行任务,而PID则是独特的车牌号,为我们提供了管理的便利。知道这个,我们就可以更好地理解和操作Linux系统,甚至通过对进程的有效管理,让系统运行得更加顺畅。
223 16
|
7月前
|
Unix Linux
对于Linux的进程概念以及进程状态的理解和解析
现在,我们已经了解了Linux进程的基础知识和进程状态的理解了。这就像我们理解了城市中行人的行走和行为模式!希望这个形象的例子能帮助我们更好地理解这个重要的概念,并在实际应用中发挥作用。
143 20
|
6月前
|
监控 Shell Linux
Linux进程控制(详细讲解)
进程等待是系统通过调用特定的接口(如waitwaitpid)来实现的。来进行对子进程状态检测与回收的功能。
131 0
|
6月前
|
存储 负载均衡 算法
Linux2.6内核进程调度队列
本篇文章是Linux进程系列中的最后一篇文章,本来是想放在上一篇文章的结尾的,但是想了想还是单独写一篇文章吧,虽然说这部分内容是比较难的,所有一般来说是简单的提及带过的,但是为了让大家对进程有更深的理解与认识,还是看了一些别人的文章,然后学习了学习,然后对此做了总结,尽可能详细的介绍明白。最后推荐一篇文章Linux的进程优先级 NI 和 PR - 简书。
203 0
|
6月前
|
存储 Linux Shell
Linux进程概念-详细版(二)
在Linux进程概念-详细版(一)中我们解释了什么是进程,以及进程的各种状态,已经对进程有了一定的认识,那么这篇文章将会继续补全上篇文章剩余没有说到的,进程优先级,环境变量,程序地址空间,进程地址空间,以及调度队列。
133 0
|
6月前
|
Linux 调度 C语言
Linux进程概念-详细版(一)
子进程与父进程代码共享,其子进程直接用父进程的代码,其自己本身无代码,所以子进程无法改动代码,平时所说的修改是修改的数据。为什么要创建子进程:为了让其父子进程执行不同的代码块。子进程的数据相对于父进程是会进行写时拷贝(COW)。
184 0
|
9月前
|
存储 Linux 调度
【Linux】进程概念和进程状态
本文详细介绍了Linux系统中进程的核心概念与管理机制。从进程的定义出发,阐述了其作为操作系统资源管理的基本单位的重要性,并深入解析了task_struct结构体的内容及其在进程管理中的作用。同时,文章讲解了进程的基本操作(如获取PID、查看进程信息等)、父进程与子进程的关系(重点分析fork函数)、以及进程的三种主要状态(运行、阻塞、挂起)。此外,还探讨了Linux特有的进程状态表示和孤儿进程的处理方式。通过学习这些内容,读者可以更好地理解Linux进程的运行原理并优化系统性能。
350 4
|
9月前
|
Linux 数据库 Perl
【YashanDB 知识库】如何避免 yasdb 进程被 Linux OOM Killer 杀掉
本文来自YashanDB官网,探讨Linux系统中OOM Killer对数据库服务器的影响及解决方法。当内存接近耗尽时,OOM Killer会杀死占用最多内存的进程,这可能导致数据库主进程被误杀。为避免此问题,可采取两种方法:一是在OS层面关闭OOM Killer,通过修改`/etc/sysctl.conf`文件并重启生效;二是豁免数据库进程,由数据库实例用户借助`sudo`权限调整`oom_score_adj`值。这些措施有助于保护数据库进程免受系统内存管理机制的影响。