| /* Support for ignoring signals. | 
 |  | 
 |    Copyright (C) 2021-2023 Free Software Foundation, Inc. | 
 |  | 
 |    This file is part of GDB. | 
 |  | 
 |    This program is free software; you can redistribute it and/or modify | 
 |    it under the terms of the GNU General Public License as published by | 
 |    the Free Software Foundation; either version 3 of the License, or | 
 |    (at your option) any later version. | 
 |  | 
 |    This program is distributed in the hope that it will be useful, | 
 |    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
 |    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
 |    GNU General Public License for more details. | 
 |  | 
 |    You should have received a copy of the GNU General Public License | 
 |    along with this program.  If not, see <http://www.gnu.org/licenses/>.  */ | 
 |  | 
 | #ifndef SCOPED_IGNORE_SIGNAL_H | 
 | #define SCOPED_IGNORE_SIGNAL_H | 
 |  | 
 | #include <signal.h> | 
 |  | 
 | /* RAII class used to ignore a signal in a scope.  If sigprocmask is | 
 |    supported, then the signal is only ignored by the calling thread. | 
 |    Otherwise, the signal disposition is set to SIG_IGN, which affects | 
 |    the whole process.  If ConsumePending is true, the destructor | 
 |    consumes a pending Sig.  SIGPIPE for example is queued on the | 
 |    thread even if blocked at the time the pipe is written to.  SIGTTOU | 
 |    OTOH is not raised at all if the thread writing to the terminal has | 
 |    it blocked.  Because SIGTTOU is sent to the whole process instead | 
 |    of to a specific thread, consuming a pending SIGTTOU in the | 
 |    destructor could consume a signal raised due to actions done by | 
 |    some other thread.  */ | 
 |  | 
 | template <int Sig, bool ConsumePending> | 
 | class scoped_ignore_signal | 
 | { | 
 | public: | 
 |   scoped_ignore_signal () | 
 |   { | 
 | #ifdef HAVE_SIGPROCMASK | 
 |     sigset_t set, old_state; | 
 |  | 
 |     sigemptyset (&set); | 
 |     sigaddset (&set, Sig); | 
 |     sigprocmask (SIG_BLOCK, &set, &old_state); | 
 |     m_was_blocked = sigismember (&old_state, Sig); | 
 | #else | 
 |     m_osig = signal (Sig, SIG_IGN); | 
 | #endif | 
 |   } | 
 |  | 
 |   ~scoped_ignore_signal () | 
 |   { | 
 | #ifdef HAVE_SIGPROCMASK | 
 |     if (!m_was_blocked) | 
 |       { | 
 | 	sigset_t set; | 
 |  | 
 | 	sigemptyset (&set); | 
 | 	sigaddset (&set, Sig); | 
 |  | 
 | 	/* If we got a pending Sig signal, consume it before | 
 | 	   unblocking.  */ | 
 | 	if (ConsumePending) | 
 | 	  { | 
 | #ifdef HAVE_SIGTIMEDWAIT | 
 | 	    const timespec zero_timeout = {}; | 
 |  | 
 | 	    sigtimedwait (&set, nullptr, &zero_timeout); | 
 | #else | 
 | 	    sigset_t pending; | 
 |  | 
 | 	    sigpending (&pending); | 
 | 	    if (sigismember (&pending, Sig)) | 
 | 	      { | 
 | 		int sig_found; | 
 |  | 
 | 		sigwait (&set, &sig_found); | 
 | 		gdb_assert (sig_found == Sig); | 
 | 	      } | 
 | #endif | 
 | 	  } | 
 |  | 
 | 	sigprocmask (SIG_UNBLOCK, &set, nullptr); | 
 |       } | 
 | #else | 
 |     signal (Sig, m_osig); | 
 | #endif | 
 |   } | 
 |  | 
 |   DISABLE_COPY_AND_ASSIGN (scoped_ignore_signal); | 
 |  | 
 | private: | 
 | #ifdef HAVE_SIGPROCMASK | 
 |   bool m_was_blocked; | 
 | #else | 
 |   sighandler_t m_osig; | 
 | #endif | 
 | }; | 
 |  | 
 | struct scoped_ignore_signal_nop | 
 | { | 
 |   /* Note, these can't both be "= default", because otherwise the | 
 |      compiler warns that variables of this type are not used.  */ | 
 |   scoped_ignore_signal_nop () | 
 |   {} | 
 |   ~scoped_ignore_signal_nop () | 
 |   {} | 
 |   DISABLE_COPY_AND_ASSIGN (scoped_ignore_signal_nop); | 
 | }; | 
 |  | 
 | #ifdef SIGPIPE | 
 | using scoped_ignore_sigpipe = scoped_ignore_signal<SIGPIPE, true>; | 
 | #else | 
 | using scoped_ignore_sigpipe = scoped_ignore_signal_nop; | 
 | #endif | 
 |  | 
 | #endif /* SCOPED_IGNORE_SIGNAL_H */ |