linux 信号机制
软中断信号 Signal,简称信号,用来通知进程发生了异步事件,进程之间可以互相通过系统调用 kill 等函数来发送软中断信号。内核也可以因为内部事件而给进程发送信号,通知进程发生了某个事件。
进程对信号的处理
进程对信号的处理有三种方式:
忽略信号: 大部分信号可被忽略,除SIGSTOP和SIGKILL信号外(这是超级用户杀掉或停掉任意进程的手段)。
捕获信号: 注册信号处理函数,它对产生的特定信号做处理。
信号默认动作: unix内核定义的默认动作,有5种情况:
a) 流产abort:终止进程并产生core文件。
b) 终止stop:终止进程但不生成core文件。
c) 忽略:忽略信号。
d) 挂起suspend:挂起进程。
e) 继续continue:若进程是挂起的,则resume进程,否则忽略此信号。
捕获进程退出信号
当程序由于某种原因要退出时,系统也会给进程发送信号。
此时,如果没有对信号进行捕获,进程就会执行默认退出动作。
当软件在客户环境运行时,为了方便 debug, 可以在程序异常退出时,打印程序当前的堆栈信息
下面五种都是进程退出信号。
// 捕获信号,对于这些信号都执行 coreDumpHandle
signal(SIGQUIT, coreDumpHandle);
signal(SIGILL, coreDumpHandle);
signal(SIGABRT, coreDumpHandle);
signal(SIGSEGV, coreDumpHandle);
signal(SIGTRAP, coreDumpHandle);
可以通过在终端执行 man 7 signal
查看各种信号的说明
c++ 打印堆栈信息
#include <signal.h>
#include <execinfo.h>
#include <dlfcn.h>
#include <cxxabi.h>
#include <QFile>
#include <QTextStream>
void coreDumpHandle(int signum) {
string path = "/xxx/xxx/coreDump.log";
void* callStack[128];
int frames = backstrace(callStack, sizeof(callStack) / sizeof(callStack[0]));
char** strs = backtrack_symbols(callStack, frames);
if (strs == nullptr)
exit(0);
QFile out(path.c_str());
if (!out.open(QIODevice::WrietOnly | QIODevece::Text | QIODevice::Truncate)) {
exit(0);
}
QTextStream textStream(&out);
textStream << "Get signal: " << strsignal(signum) << "\n";
textStream << "Call stack:\n";
unsigned count = 0;
for (int i = 0; i < frames; ++i) {
Dl_info info;
if (dladdr(callStack[i], &info) && info.dli_sname) {
int status;
char* demangled = abi::__cxa_demangle(info.dli_sname, nullptr, nullptr, &status);
if (status == 0) {
QString str = QString("[%1] %2 : %3 0x%4\n").arg(count++)
.arg(info.dli_fname)
.arg(demangled)
.arg(reinterpret_cast<quintptr>(((void*)callStack[i])), 0, 16);
textStream << str;
} else {
QString str = QString("[%1] %2 : %3 0x%4\n").arg(count++)
.arg(info.dli_fname)
.arg(info.dli_fname)
.arg(reinterpret_cast<quintptr>(((void*)callStack[i])), 0, 16);
textStream << str;
}
free(demangled);
}
else {
// printf("[%d] %s : ???\n", i, info.dli_fname);
}
}
out.close();
free(strs);
exit(0);
}