C ++在信号处理程序之后继续执行

时间:2020-06-02 11:34:46

标签: c++ linux multithreading signals signal-handling

我试图用C ++管理信号,然后正常继续执行程序。我使用信号代替异常,因为当未连​​接传感器时程序会自行阻塞(我无法更改)。

引发信号的代码部分使用线程作为计时器。由于线程不能引发异常,因此我决定发出一个信号(在这种情况下为SIGTERM)。

当我尝试以与异常但信号处理程序相同的方式继续执行程序时,问题开始了。我利用setjmp库来解决问题:

jmp_buf gBuffer;
void signalHandler(int signal) {
    longjmp(gBuffer, 1);
}

int main() {
    std::signal(SIGTERM, signalHandler);

    if (setjmp(gBuffer) == 0) {
        /* ---------------------
           Part of code that raise the signal
        --------------------- */
    }
}

它工作正常,但是当程序完成时,出现内存错误:

*** stack smashing detected ***: <unknown> terminated
Aborted (core dumped)

在调试模式下执行程序时的回溯是:

Thread 2 "EXECUTABLE_FILE" received signal SIGTERM, Terminated.
[Switching to Thread 0x7ffff01f2700 (LWP 31718)]
raise (sig=<optimized out>) at ../sysdeps/unix/sysv/linux/raise.c:51
51  ../sysdeps/unix/sysv/linux/raise.c: No such file or directory.
(gdb) bt
#0  raise (sig=<optimized out>) at ../sysdeps/unix/sysv/linux/raise.c:51
#1  0x00007ffff78531ab in "THREAD THAT RAISE THE SIGNAL" (futureObj=...)
    at "LINE THAT RAISE THE SIGNAL"
#2  0x00007ffff7856ffd in std::__invoke_impl<void, void (*)(std::future<void>), std::future<void> >
    (__f=<optimized out>) at /usr/include/c++/7/bits/invoke.h:60
#3  std::__invoke<void (*)(std::future<void>), std::future<void> > (__fn=<optimized out>)
    at /usr/include/c++/7/bits/invoke.h:95
#4  std::thread::_Invoker<std::tuple<void (*)(std::future<void>), std::future<void> > >::_M_invoke<0ul, 1ul> (this=<optimized out>) at /usr/include/c++/7/thread:234
#5  std::thread::_Invoker<std::tuple<void (*)(std::future<void>), std::future<void> > >::operator()
    (this=<optimized out>) at /usr/include/c++/7/thread:243
#6  std::thread::_State_impl<std::thread::_Invoker<std::tuple<void (*)(std::future<void>), std::future<void> > > >::_M_run (this=<optimized out>) at /usr/include/c++/7/thread:186
#7  0x00007ffff6e41d80 in ?? () from /usr/lib/x86_64-linux-gnu/libstdc++.so.6
#8  0x00007ffff714c6db in start_thread (arg=0x7ffff01f2700) at pthread_create.c:463
#9  0x00007ffff688988f in clone () at ../sysdeps/unix/sysv/linux/x86_64/clone.S:95

我真的很感谢有人能帮助我。我还可以更改引发信号的代码部分,引发异常,但是我不知道如何在线程中执行该操作。

1 个答案:

答案 0 :(得分:1)

您无法按照您的设想直接进行操作-对不起。是的,您的信号处理程序被调用了,但是通常由于limitations列出的信号处理程序,您不能(也不应该尝试)从信号处理程序中继续执行主程序 (例如,可能会阻止发出新信号等)。

因此,您的信号处理程序应设置某种标志,例如。原子布尔或条件变量或您喜欢的任何方法。然后从信号处理程序返回。然后在“违规代码”附近循环,可以检查该代码,然后可以返回/展开/终止计时器。

相关问题