簡體   English   中英

為什么我的linux信號處理程序只運行一次

[英]Why my linux signal handler run only once

#include <iostream>
#include <signal.h>
#include <fenv.h>
#include <string.h>

void signal_handler(int sig, siginfo_t *siginfo, void* context) 
{ 
  std::cout << " signal_handler " << fetestexcept(FE_ALL_EXCEPT) << std::endl;
  throw "exception"; 
}

void divide() {
  float a = 1000., b = 0.,  c, f = 1e-300;
  c = a / b;

  std::cout << c << " and f = " << f << std::endl;  
}

void init_sig_hanlder() {
  feenableexcept(FE_ALL_EXCEPT);

  struct sigaction sa, initial_sa;

  sa.sa_sigaction   = &signal_handler ;
  sigemptyset( &sa.sa_mask ) ;
  sa.sa_flags   = SA_SIGINFO;   // man sigaction(3) // allows for void(*)(int,siginfo_t*,void*) handler

  sigaction(SIGFPE, &sa, &initial_sa);

}

int main(int argc, char** argv) {
  init_sig_hanlder();

  while(true)
    {
      try {
    sleep(1);
    divide();
      }
      catch(const char * a) {
    std::cout << "Exception in catch: " << a << std::endl;
      }    
      catch(...) {
    std::cout << "Exception in ..." << std::endl;
      }    
    }

  return 0;
}

在Linux / g ++ 4.2上產生以下結果:

signal_handler 0
catch中的異常:異常
inf和f = 0
inf和f = 0
inf和f = 0
inf和f = 0

因此,第一次執行信號處理程序,但下一個fp異常不會再次觸發處理程序。 我哪里錯了?

我記得信號處理程序必須被聲明為'extern“C”',因為庫/內核將為您的函數使用C調用約定,而不是C ++調用約定。 但是外部“C”的函數不能拋出異常,所以至少在形式上你的代碼不是“正確的”

在引擎蓋下,我猜想Linux中的信號傳遞代碼沒有機會重置或清除信號掩碼,因為你從未將控制權返回給運行時和內核。

我不認為在信號處理程序中拋出異常是一種很好的做法。 操作系統期望信號處理程序return ,因為在調用它的處理程序時信號被阻塞。 通過拋出異常可以防止系統解除阻塞信號。

您應該將sigsetjmp / siglongjmp用於此類代碼,而不是異常。

在這里使用異常是錯誤的,jmp更改應該是排除奇怪平台特定行為的第一步(特別是因為C ++標准允許使用信號傳遞機制實現異常)。

但是,我很好奇SIGFPE的失敗是否與fe * except()狀態有關。

問題:在初始除以零之前和之后該狀態會發生什么變化? 也許有人期望當你再次嘗試時,需要feclearexcept()來獲得另一個SIGFPE。

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM