mirror of
https://github.com/postgres/postgres.git
synced 2026-05-27 12:00:13 -04:00
Rework signal handler infrastructure to pass sender info as argument.
Commit 095c9d4cf06 added errdetail() reporting of the PID and UID of the process that sent a termination signal. However, as noted by Andres Freund, the implementation had architectural problems: 1. wrapper_handler() in pqsignal.c contained SIGTERM-specific logic (setting ProcDieSenderPid/Uid), violating its role as a generic signal dispatch wrapper. 2. Using globals to pass sender info between wrapper_handler and the real handler is unsafe when signals nest on some platforms. 3. The syncrep.c errdetail used psprintf() to conditionally embed text via %s, breaking translatability. Adopt the approach proposed by Andres Freund: introduce a pg_signal_info struct that is passed as an argument to all signal handlers via the SIGNAL_ARGS macro. wrapper_handler populates it from siginfo_t when SA_SIGINFO is available, or with zeros otherwise. This keeps wrapper_handler fully generic and avoids any globals for passing signal metadata. Since pqsigfunc now has a different signature from the system's signal handler type, SIG_IGN and SIG_DFL can no longer be passed directly to pqsignal(). Introduce PG_SIG_IGN and PG_SIG_DFL macros that cast to the new pqsigfunc type, and update all call sites. The legacy pqsignal() in libpq retains its original signature via a local typedef. Only die() reads pg_siginfo today, copying the sender PID/UID into ProcDieSenderPid/Uid for later use by ProcessInterrupts(). Only the first SIGTERM's sender info is recorded. Also fix the syncrep.c translatability issue by using separate ereport calls with complete, independently translatable errdetail strings. Also make the psql TAP test require the DETAIL line on platforms with SA_SIGINFO, rather than making it unconditionally optional. On Windows, pg_signal_info uses uint32_t for pid and uid fields since pid_t/uid_t are not available early enough in the include chain. The Windows signal dispatch in pgwin32_dispatch_queued_signals() passes a zeroed pg_signal_info to handlers. Author: Andres Freund <andres@anarazel.de> Author: Jakub Wartak <jakub.wartak@enterprisedb.com> Reviewed-by: Andrew Dunstan <andrew@dunslane.net> Reviewed-by: Chao Li <li.evan.chao@gmail.com> Discussion: https://postgr.es/m/cwyyryh2veejuxbj5ifzyaejw7jhhqc5mrdeq56xckknsdecn2@6hzfcxde2nm5 Discussion: https://postgr.es/m/jygesyr7mwg7ovdbxpmjvvbi3hccptpkcreqb645h7f56puwbz@hmkkwi3melfe
This commit is contained in:
parent
972c14fb91
commit
3e2a1496ba
32 changed files with 218 additions and 149 deletions
|
|
@ -463,10 +463,10 @@ bootstrap_signals(void)
|
||||||
* mode; "curl up and die" is a sufficient response for all these cases.
|
* mode; "curl up and die" is a sufficient response for all these cases.
|
||||||
* Let's set that handling explicitly, as documentation if nothing else.
|
* Let's set that handling explicitly, as documentation if nothing else.
|
||||||
*/
|
*/
|
||||||
pqsignal(SIGHUP, SIG_DFL);
|
pqsignal(SIGHUP, PG_SIG_DFL);
|
||||||
pqsignal(SIGINT, SIG_DFL);
|
pqsignal(SIGINT, PG_SIG_DFL);
|
||||||
pqsignal(SIGTERM, SIG_DFL);
|
pqsignal(SIGTERM, PG_SIG_DFL);
|
||||||
pqsignal(SIGQUIT, SIG_DFL);
|
pqsignal(SIGQUIT, PG_SIG_DFL);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ----------------------------------------------------------------
|
/* ----------------------------------------------------------------
|
||||||
|
|
|
||||||
|
|
@ -88,7 +88,7 @@ pgwin32_signal_initialize(void)
|
||||||
pg_signal_array[i].sa_handler = SIG_DFL;
|
pg_signal_array[i].sa_handler = SIG_DFL;
|
||||||
pg_signal_array[i].sa_mask = 0;
|
pg_signal_array[i].sa_mask = 0;
|
||||||
pg_signal_array[i].sa_flags = 0;
|
pg_signal_array[i].sa_flags = 0;
|
||||||
pg_signal_defaults[i] = SIG_IGN;
|
pg_signal_defaults[i] = PG_SIG_IGN;
|
||||||
}
|
}
|
||||||
pg_signal_mask = 0;
|
pg_signal_mask = 0;
|
||||||
pg_signal_queue = 0;
|
pg_signal_queue = 0;
|
||||||
|
|
@ -134,15 +134,19 @@ pgwin32_dispatch_queued_signals(void)
|
||||||
{
|
{
|
||||||
/* Execute this signal */
|
/* Execute this signal */
|
||||||
struct sigaction *act = &pg_signal_array[i];
|
struct sigaction *act = &pg_signal_array[i];
|
||||||
pqsigfunc sig = act->sa_handler;
|
pqsigfunc sig = (pqsigfunc) (pg_funcptr_t) act->sa_handler;
|
||||||
|
|
||||||
if (sig == SIG_DFL)
|
if (sig == PG_SIG_DFL)
|
||||||
sig = pg_signal_defaults[i];
|
sig = pg_signal_defaults[i];
|
||||||
pg_signal_queue &= ~sigmask(i);
|
pg_signal_queue &= ~sigmask(i);
|
||||||
if (sig != SIG_ERR && sig != SIG_IGN && sig != SIG_DFL)
|
if (sig != (pqsigfunc) (pg_funcptr_t) SIG_ERR && sig != PG_SIG_IGN && sig != PG_SIG_DFL)
|
||||||
{
|
{
|
||||||
sigset_t block_mask;
|
sigset_t block_mask;
|
||||||
sigset_t save_mask;
|
sigset_t save_mask;
|
||||||
|
struct pg_signal_info nodata;
|
||||||
|
|
||||||
|
nodata.pid = 0;
|
||||||
|
nodata.uid = 0;
|
||||||
|
|
||||||
LeaveCriticalSection(&pg_signal_crit_sec);
|
LeaveCriticalSection(&pg_signal_crit_sec);
|
||||||
|
|
||||||
|
|
@ -151,7 +155,7 @@ pgwin32_dispatch_queued_signals(void)
|
||||||
block_mask |= sigmask(i);
|
block_mask |= sigmask(i);
|
||||||
|
|
||||||
sigprocmask(SIG_BLOCK, &block_mask, &save_mask);
|
sigprocmask(SIG_BLOCK, &block_mask, &save_mask);
|
||||||
sig(i);
|
sig(i, &nodata);
|
||||||
sigprocmask(SIG_SETMASK, &save_mask, NULL);
|
sigprocmask(SIG_SETMASK, &save_mask, NULL);
|
||||||
|
|
||||||
EnterCriticalSection(&pg_signal_crit_sec);
|
EnterCriticalSection(&pg_signal_crit_sec);
|
||||||
|
|
|
||||||
|
|
@ -445,11 +445,11 @@ AutoVacLauncherMain(const void *startup_data, size_t startup_data_len)
|
||||||
|
|
||||||
InitializeTimeouts(); /* establishes SIGALRM handler */
|
InitializeTimeouts(); /* establishes SIGALRM handler */
|
||||||
|
|
||||||
pqsignal(SIGPIPE, SIG_IGN);
|
pqsignal(SIGPIPE, PG_SIG_IGN);
|
||||||
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
||||||
pqsignal(SIGUSR2, avl_sigusr2_handler);
|
pqsignal(SIGUSR2, avl_sigusr2_handler);
|
||||||
pqsignal(SIGFPE, FloatExceptionHandler);
|
pqsignal(SIGFPE, FloatExceptionHandler);
|
||||||
pqsignal(SIGCHLD, SIG_DFL);
|
pqsignal(SIGCHLD, PG_SIG_DFL);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Create a per-backend PGPROC struct in shared memory. We must do this
|
* Create a per-backend PGPROC struct in shared memory. We must do this
|
||||||
|
|
@ -1456,11 +1456,11 @@ AutoVacWorkerMain(const void *startup_data, size_t startup_data_len)
|
||||||
|
|
||||||
InitializeTimeouts(); /* establishes SIGALRM handler */
|
InitializeTimeouts(); /* establishes SIGALRM handler */
|
||||||
|
|
||||||
pqsignal(SIGPIPE, SIG_IGN);
|
pqsignal(SIGPIPE, PG_SIG_IGN);
|
||||||
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
||||||
pqsignal(SIGUSR2, SIG_IGN);
|
pqsignal(SIGUSR2, PG_SIG_IGN);
|
||||||
pqsignal(SIGFPE, FloatExceptionHandler);
|
pqsignal(SIGFPE, FloatExceptionHandler);
|
||||||
pqsignal(SIGCHLD, SIG_DFL);
|
pqsignal(SIGCHLD, PG_SIG_DFL);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Create a per-backend PGPROC struct in shared memory. We must do this
|
* Create a per-backend PGPROC struct in shared memory. We must do this
|
||||||
|
|
|
||||||
|
|
@ -785,19 +785,19 @@ BackgroundWorkerMain(const void *startup_data, size_t startup_data_len)
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
pqsignal(SIGINT, SIG_IGN);
|
pqsignal(SIGINT, PG_SIG_IGN);
|
||||||
pqsignal(SIGUSR1, SIG_IGN);
|
pqsignal(SIGUSR1, PG_SIG_IGN);
|
||||||
pqsignal(SIGFPE, SIG_IGN);
|
pqsignal(SIGFPE, PG_SIG_IGN);
|
||||||
}
|
}
|
||||||
pqsignal(SIGTERM, die);
|
pqsignal(SIGTERM, die);
|
||||||
/* SIGQUIT handler was already set up by InitPostmasterChild */
|
/* SIGQUIT handler was already set up by InitPostmasterChild */
|
||||||
pqsignal(SIGHUP, SIG_IGN);
|
pqsignal(SIGHUP, PG_SIG_IGN);
|
||||||
|
|
||||||
InitializeTimeouts(); /* establishes SIGALRM handler */
|
InitializeTimeouts(); /* establishes SIGALRM handler */
|
||||||
|
|
||||||
pqsignal(SIGPIPE, SIG_IGN);
|
pqsignal(SIGPIPE, PG_SIG_IGN);
|
||||||
pqsignal(SIGUSR2, SIG_IGN);
|
pqsignal(SIGUSR2, PG_SIG_IGN);
|
||||||
pqsignal(SIGCHLD, SIG_DFL);
|
pqsignal(SIGCHLD, PG_SIG_DFL);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If an exception is encountered, processing resumes here.
|
* If an exception is encountered, processing resumes here.
|
||||||
|
|
|
||||||
|
|
@ -101,18 +101,18 @@ BackgroundWriterMain(const void *startup_data, size_t startup_data_len)
|
||||||
* Properly accept or ignore signals that might be sent to us.
|
* Properly accept or ignore signals that might be sent to us.
|
||||||
*/
|
*/
|
||||||
pqsignal(SIGHUP, SignalHandlerForConfigReload);
|
pqsignal(SIGHUP, SignalHandlerForConfigReload);
|
||||||
pqsignal(SIGINT, SIG_IGN);
|
pqsignal(SIGINT, PG_SIG_IGN);
|
||||||
pqsignal(SIGTERM, SignalHandlerForShutdownRequest);
|
pqsignal(SIGTERM, SignalHandlerForShutdownRequest);
|
||||||
/* SIGQUIT handler was already set up by InitPostmasterChild */
|
/* SIGQUIT handler was already set up by InitPostmasterChild */
|
||||||
pqsignal(SIGALRM, SIG_IGN);
|
pqsignal(SIGALRM, PG_SIG_IGN);
|
||||||
pqsignal(SIGPIPE, SIG_IGN);
|
pqsignal(SIGPIPE, PG_SIG_IGN);
|
||||||
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
||||||
pqsignal(SIGUSR2, SIG_IGN);
|
pqsignal(SIGUSR2, PG_SIG_IGN);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Reset some signals that are accepted by postmaster but not here
|
* Reset some signals that are accepted by postmaster but not here
|
||||||
*/
|
*/
|
||||||
pqsignal(SIGCHLD, SIG_DFL);
|
pqsignal(SIGCHLD, PG_SIG_DFL);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We just started, assume there has been either a shutdown or
|
* We just started, assume there has been either a shutdown or
|
||||||
|
|
|
||||||
|
|
@ -223,17 +223,17 @@ CheckpointerMain(const void *startup_data, size_t startup_data_len)
|
||||||
*/
|
*/
|
||||||
pqsignal(SIGHUP, SignalHandlerForConfigReload);
|
pqsignal(SIGHUP, SignalHandlerForConfigReload);
|
||||||
pqsignal(SIGINT, ReqShutdownXLOG);
|
pqsignal(SIGINT, ReqShutdownXLOG);
|
||||||
pqsignal(SIGTERM, SIG_IGN); /* ignore SIGTERM */
|
pqsignal(SIGTERM, PG_SIG_IGN); /* ignore SIGTERM */
|
||||||
/* SIGQUIT handler was already set up by InitPostmasterChild */
|
/* SIGQUIT handler was already set up by InitPostmasterChild */
|
||||||
pqsignal(SIGALRM, SIG_IGN);
|
pqsignal(SIGALRM, PG_SIG_IGN);
|
||||||
pqsignal(SIGPIPE, SIG_IGN);
|
pqsignal(SIGPIPE, PG_SIG_IGN);
|
||||||
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
||||||
pqsignal(SIGUSR2, SignalHandlerForShutdownRequest);
|
pqsignal(SIGUSR2, SignalHandlerForShutdownRequest);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Reset some signals that are accepted by postmaster but not here
|
* Reset some signals that are accepted by postmaster but not here
|
||||||
*/
|
*/
|
||||||
pqsignal(SIGCHLD, SIG_DFL);
|
pqsignal(SIGCHLD, PG_SIG_DFL);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Initialize so that first time-driven event happens at the correct time.
|
* Initialize so that first time-driven event happens at the correct time.
|
||||||
|
|
|
||||||
|
|
@ -1020,7 +1020,7 @@ DataChecksumsWorkerLauncherMain(Datum arg)
|
||||||
pqsignal(SIGTERM, die);
|
pqsignal(SIGTERM, die);
|
||||||
pqsignal(SIGINT, launcher_cancel_handler);
|
pqsignal(SIGINT, launcher_cancel_handler);
|
||||||
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
||||||
pqsignal(SIGUSR2, SIG_IGN);
|
pqsignal(SIGUSR2, PG_SIG_IGN);
|
||||||
|
|
||||||
BackgroundWorkerUnblockSignals();
|
BackgroundWorkerUnblockSignals();
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -229,16 +229,16 @@ PgArchiverMain(const void *startup_data, size_t startup_data_len)
|
||||||
* except for SIGHUP, SIGTERM, SIGUSR1, SIGUSR2, and SIGQUIT.
|
* except for SIGHUP, SIGTERM, SIGUSR1, SIGUSR2, and SIGQUIT.
|
||||||
*/
|
*/
|
||||||
pqsignal(SIGHUP, SignalHandlerForConfigReload);
|
pqsignal(SIGHUP, SignalHandlerForConfigReload);
|
||||||
pqsignal(SIGINT, SIG_IGN);
|
pqsignal(SIGINT, PG_SIG_IGN);
|
||||||
pqsignal(SIGTERM, SignalHandlerForShutdownRequest);
|
pqsignal(SIGTERM, SignalHandlerForShutdownRequest);
|
||||||
/* SIGQUIT handler was already set up by InitPostmasterChild */
|
/* SIGQUIT handler was already set up by InitPostmasterChild */
|
||||||
pqsignal(SIGALRM, SIG_IGN);
|
pqsignal(SIGALRM, PG_SIG_IGN);
|
||||||
pqsignal(SIGPIPE, SIG_IGN);
|
pqsignal(SIGPIPE, PG_SIG_IGN);
|
||||||
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
||||||
pqsignal(SIGUSR2, pgarch_waken_stop);
|
pqsignal(SIGUSR2, pgarch_waken_stop);
|
||||||
|
|
||||||
/* Reset some signals that are accepted by postmaster but not here */
|
/* Reset some signals that are accepted by postmaster but not here */
|
||||||
pqsignal(SIGCHLD, SIG_DFL);
|
pqsignal(SIGCHLD, PG_SIG_DFL);
|
||||||
|
|
||||||
/* Unblock signals (they were blocked when the postmaster forked us) */
|
/* Unblock signals (they were blocked when the postmaster forked us) */
|
||||||
sigprocmask(SIG_SETMASK, &UnBlockSig, NULL);
|
sigprocmask(SIG_SETMASK, &UnBlockSig, NULL);
|
||||||
|
|
|
||||||
|
|
@ -555,8 +555,8 @@ PostmasterMain(int argc, char *argv[])
|
||||||
pqsignal(SIGINT, handle_pm_shutdown_request_signal);
|
pqsignal(SIGINT, handle_pm_shutdown_request_signal);
|
||||||
pqsignal(SIGQUIT, handle_pm_shutdown_request_signal);
|
pqsignal(SIGQUIT, handle_pm_shutdown_request_signal);
|
||||||
pqsignal(SIGTERM, handle_pm_shutdown_request_signal);
|
pqsignal(SIGTERM, handle_pm_shutdown_request_signal);
|
||||||
pqsignal(SIGALRM, SIG_IGN); /* ignored */
|
pqsignal(SIGALRM, PG_SIG_IGN); /* ignored */
|
||||||
pqsignal(SIGPIPE, SIG_IGN); /* ignored */
|
pqsignal(SIGPIPE, PG_SIG_IGN); /* ignored */
|
||||||
pqsignal(SIGUSR1, handle_pm_pmsignal_signal);
|
pqsignal(SIGUSR1, handle_pm_pmsignal_signal);
|
||||||
pqsignal(SIGUSR2, dummy_handler); /* unused, reserve for children */
|
pqsignal(SIGUSR2, dummy_handler); /* unused, reserve for children */
|
||||||
pqsignal(SIGCHLD, handle_pm_child_exit_signal);
|
pqsignal(SIGCHLD, handle_pm_child_exit_signal);
|
||||||
|
|
@ -573,15 +573,15 @@ PostmasterMain(int argc, char *argv[])
|
||||||
* child processes should just allow the inherited settings to stand.
|
* child processes should just allow the inherited settings to stand.
|
||||||
*/
|
*/
|
||||||
#ifdef SIGTTIN
|
#ifdef SIGTTIN
|
||||||
pqsignal(SIGTTIN, SIG_IGN); /* ignored */
|
pqsignal(SIGTTIN, PG_SIG_IGN); /* ignored */
|
||||||
#endif
|
#endif
|
||||||
#ifdef SIGTTOU
|
#ifdef SIGTTOU
|
||||||
pqsignal(SIGTTOU, SIG_IGN); /* ignored */
|
pqsignal(SIGTTOU, PG_SIG_IGN); /* ignored */
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* ignore SIGXFSZ, so that ulimit violations work like disk full */
|
/* ignore SIGXFSZ, so that ulimit violations work like disk full */
|
||||||
#ifdef SIGXFSZ
|
#ifdef SIGXFSZ
|
||||||
pqsignal(SIGXFSZ, SIG_IGN); /* ignored */
|
pqsignal(SIGXFSZ, PG_SIG_IGN); /* ignored */
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* Begin accepting signals. */
|
/* Begin accepting signals. */
|
||||||
|
|
@ -3967,7 +3967,7 @@ process_pm_pmsignal(void)
|
||||||
* Dummy signal handler
|
* Dummy signal handler
|
||||||
*
|
*
|
||||||
* We use this for signals that we don't actually use in the postmaster,
|
* We use this for signals that we don't actually use in the postmaster,
|
||||||
* but we do use in backends. If we were to SIG_IGN such signals in the
|
* but we do use in backends. If we were to PG_SIG_IGN such signals in the
|
||||||
* postmaster, then a newly started backend might drop a signal that arrives
|
* postmaster, then a newly started backend might drop a signal that arrives
|
||||||
* before it's able to reconfigure its signal processing. (See notes in
|
* before it's able to reconfigure its signal processing. (See notes in
|
||||||
* tcop/postgres.c.)
|
* tcop/postgres.c.)
|
||||||
|
|
|
||||||
|
|
@ -226,18 +226,18 @@ StartupProcessMain(const void *startup_data, size_t startup_data_len)
|
||||||
* Properly accept or ignore signals the postmaster might send us.
|
* Properly accept or ignore signals the postmaster might send us.
|
||||||
*/
|
*/
|
||||||
pqsignal(SIGHUP, StartupProcSigHupHandler); /* reload config file */
|
pqsignal(SIGHUP, StartupProcSigHupHandler); /* reload config file */
|
||||||
pqsignal(SIGINT, SIG_IGN); /* ignore query cancel */
|
pqsignal(SIGINT, PG_SIG_IGN); /* ignore query cancel */
|
||||||
pqsignal(SIGTERM, StartupProcShutdownHandler); /* request shutdown */
|
pqsignal(SIGTERM, StartupProcShutdownHandler); /* request shutdown */
|
||||||
/* SIGQUIT handler was already set up by InitPostmasterChild */
|
/* SIGQUIT handler was already set up by InitPostmasterChild */
|
||||||
InitializeTimeouts(); /* establishes SIGALRM handler */
|
InitializeTimeouts(); /* establishes SIGALRM handler */
|
||||||
pqsignal(SIGPIPE, SIG_IGN);
|
pqsignal(SIGPIPE, PG_SIG_IGN);
|
||||||
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
||||||
pqsignal(SIGUSR2, StartupProcTriggerHandler);
|
pqsignal(SIGUSR2, StartupProcTriggerHandler);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Reset some signals that are accepted by postmaster but not here
|
* Reset some signals that are accepted by postmaster but not here
|
||||||
*/
|
*/
|
||||||
pqsignal(SIGCHLD, SIG_DFL);
|
pqsignal(SIGCHLD, PG_SIG_DFL);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Register timeouts needed for standby mode
|
* Register timeouts needed for standby mode
|
||||||
|
|
|
||||||
|
|
@ -276,18 +276,18 @@ SysLoggerMain(const void *startup_data, size_t startup_data_len)
|
||||||
|
|
||||||
pqsignal(SIGHUP, SignalHandlerForConfigReload); /* set flag to read config
|
pqsignal(SIGHUP, SignalHandlerForConfigReload); /* set flag to read config
|
||||||
* file */
|
* file */
|
||||||
pqsignal(SIGINT, SIG_IGN);
|
pqsignal(SIGINT, PG_SIG_IGN);
|
||||||
pqsignal(SIGTERM, SIG_IGN);
|
pqsignal(SIGTERM, PG_SIG_IGN);
|
||||||
pqsignal(SIGQUIT, SIG_IGN);
|
pqsignal(SIGQUIT, PG_SIG_IGN);
|
||||||
pqsignal(SIGALRM, SIG_IGN);
|
pqsignal(SIGALRM, PG_SIG_IGN);
|
||||||
pqsignal(SIGPIPE, SIG_IGN);
|
pqsignal(SIGPIPE, PG_SIG_IGN);
|
||||||
pqsignal(SIGUSR1, sigUsr1Handler); /* request log rotation */
|
pqsignal(SIGUSR1, sigUsr1Handler); /* request log rotation */
|
||||||
pqsignal(SIGUSR2, SIG_IGN);
|
pqsignal(SIGUSR2, PG_SIG_IGN);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Reset some signals that are accepted by postmaster but not here
|
* Reset some signals that are accepted by postmaster but not here
|
||||||
*/
|
*/
|
||||||
pqsignal(SIGCHLD, SIG_DFL);
|
pqsignal(SIGCHLD, PG_SIG_DFL);
|
||||||
|
|
||||||
sigprocmask(SIG_SETMASK, &UnBlockSig, NULL);
|
sigprocmask(SIG_SETMASK, &UnBlockSig, NULL);
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -244,13 +244,13 @@ WalSummarizerMain(const void *startup_data, size_t startup_data_len)
|
||||||
* Properly accept or ignore signals the postmaster might send us
|
* Properly accept or ignore signals the postmaster might send us
|
||||||
*/
|
*/
|
||||||
pqsignal(SIGHUP, SignalHandlerForConfigReload);
|
pqsignal(SIGHUP, SignalHandlerForConfigReload);
|
||||||
pqsignal(SIGINT, SIG_IGN); /* no query to cancel */
|
pqsignal(SIGINT, PG_SIG_IGN); /* no query to cancel */
|
||||||
pqsignal(SIGTERM, SignalHandlerForShutdownRequest);
|
pqsignal(SIGTERM, SignalHandlerForShutdownRequest);
|
||||||
/* SIGQUIT handler was already set up by InitPostmasterChild */
|
/* SIGQUIT handler was already set up by InitPostmasterChild */
|
||||||
pqsignal(SIGALRM, SIG_IGN);
|
pqsignal(SIGALRM, PG_SIG_IGN);
|
||||||
pqsignal(SIGPIPE, SIG_IGN);
|
pqsignal(SIGPIPE, PG_SIG_IGN);
|
||||||
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
||||||
pqsignal(SIGUSR2, SIG_IGN); /* not used */
|
pqsignal(SIGUSR2, PG_SIG_IGN); /* not used */
|
||||||
|
|
||||||
/* Advertise ourselves. */
|
/* Advertise ourselves. */
|
||||||
on_shmem_exit(WalSummarizerShutdown, (Datum) 0);
|
on_shmem_exit(WalSummarizerShutdown, (Datum) 0);
|
||||||
|
|
@ -267,7 +267,7 @@ WalSummarizerMain(const void *startup_data, size_t startup_data_len)
|
||||||
/*
|
/*
|
||||||
* Reset some signals that are accepted by postmaster but not here
|
* Reset some signals that are accepted by postmaster but not here
|
||||||
*/
|
*/
|
||||||
pqsignal(SIGCHLD, SIG_DFL);
|
pqsignal(SIGCHLD, PG_SIG_DFL);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If an exception is encountered, processing resumes here.
|
* If an exception is encountered, processing resumes here.
|
||||||
|
|
|
||||||
|
|
@ -101,18 +101,18 @@ WalWriterMain(const void *startup_data, size_t startup_data_len)
|
||||||
* Properly accept or ignore signals the postmaster might send us
|
* Properly accept or ignore signals the postmaster might send us
|
||||||
*/
|
*/
|
||||||
pqsignal(SIGHUP, SignalHandlerForConfigReload);
|
pqsignal(SIGHUP, SignalHandlerForConfigReload);
|
||||||
pqsignal(SIGINT, SIG_IGN); /* no query to cancel */
|
pqsignal(SIGINT, PG_SIG_IGN); /* no query to cancel */
|
||||||
pqsignal(SIGTERM, SignalHandlerForShutdownRequest);
|
pqsignal(SIGTERM, SignalHandlerForShutdownRequest);
|
||||||
/* SIGQUIT handler was already set up by InitPostmasterChild */
|
/* SIGQUIT handler was already set up by InitPostmasterChild */
|
||||||
pqsignal(SIGALRM, SIG_IGN);
|
pqsignal(SIGALRM, PG_SIG_IGN);
|
||||||
pqsignal(SIGPIPE, SIG_IGN);
|
pqsignal(SIGPIPE, PG_SIG_IGN);
|
||||||
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
||||||
pqsignal(SIGUSR2, SIG_IGN); /* not used */
|
pqsignal(SIGUSR2, PG_SIG_IGN); /* not used */
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Reset some signals that are accepted by postmaster but not here
|
* Reset some signals that are accepted by postmaster but not here
|
||||||
*/
|
*/
|
||||||
pqsignal(SIGCHLD, SIG_DFL);
|
pqsignal(SIGCHLD, PG_SIG_DFL);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Create a memory context that we will do all our work in. We do this so
|
* Create a memory context that we will do all our work in. We do this so
|
||||||
|
|
|
||||||
|
|
@ -1620,9 +1620,9 @@ ReplSlotSyncWorkerMain(const void *startup_data, size_t startup_data_len)
|
||||||
pqsignal(SIGTERM, die);
|
pqsignal(SIGTERM, die);
|
||||||
pqsignal(SIGFPE, FloatExceptionHandler);
|
pqsignal(SIGFPE, FloatExceptionHandler);
|
||||||
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
||||||
pqsignal(SIGUSR2, SIG_IGN);
|
pqsignal(SIGUSR2, PG_SIG_IGN);
|
||||||
pqsignal(SIGPIPE, SIG_IGN);
|
pqsignal(SIGPIPE, PG_SIG_IGN);
|
||||||
pqsignal(SIGCHLD, SIG_DFL);
|
pqsignal(SIGCHLD, PG_SIG_DFL);
|
||||||
|
|
||||||
check_and_set_sync_info(MyProcPid);
|
check_and_set_sync_info(MyProcPid);
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -300,22 +300,18 @@ SyncRepWaitForLSN(XLogRecPtr lsn, bool commit)
|
||||||
*/
|
*/
|
||||||
if (ProcDiePending)
|
if (ProcDiePending)
|
||||||
{
|
{
|
||||||
/*
|
if (ProcDieSenderPid != 0)
|
||||||
* ProcDieSenderPid/Uid are read directly from the globals here
|
ereport(WARNING,
|
||||||
* rather than copied to locals first; a second SIGTERM could
|
(errcode(ERRCODE_ADMIN_SHUTDOWN),
|
||||||
* change them between reads, but that is harmless because the
|
errmsg("canceling the wait for synchronous replication and terminating connection due to administrator command"),
|
||||||
* process is about to die anyway. The signal sender detail is
|
errdetail("The transaction has already committed locally, but might not have been replicated to the standby. Signal sent by PID %d, UID %d.",
|
||||||
* inlined rather than using a separate errdetail() call because
|
(int) ProcDieSenderPid,
|
||||||
* it must be appended to the existing detail message.
|
(int) ProcDieSenderUid)));
|
||||||
*/
|
else
|
||||||
ereport(WARNING,
|
ereport(WARNING,
|
||||||
(errcode(ERRCODE_ADMIN_SHUTDOWN),
|
(errcode(ERRCODE_ADMIN_SHUTDOWN),
|
||||||
errmsg("canceling the wait for synchronous replication and terminating connection due to administrator command"),
|
errmsg("canceling the wait for synchronous replication and terminating connection due to administrator command"),
|
||||||
errdetail("The transaction has already committed locally, but might not have been replicated to the standby.%s",
|
errdetail("The transaction has already committed locally, but might not have been replicated to the standby.")));
|
||||||
ProcDieSenderPid == 0 ? "" :
|
|
||||||
psprintf("\nSignal sent by PID %d, UID %d.",
|
|
||||||
(int) ProcDieSenderPid,
|
|
||||||
(int) ProcDieSenderUid))));
|
|
||||||
whereToSendOutput = DestNone;
|
whereToSendOutput = DestNone;
|
||||||
SyncRepCancelWait();
|
SyncRepCancelWait();
|
||||||
break;
|
break;
|
||||||
|
|
|
||||||
|
|
@ -248,16 +248,16 @@ WalReceiverMain(const void *startup_data, size_t startup_data_len)
|
||||||
/* Properly accept or ignore signals the postmaster might send us */
|
/* Properly accept or ignore signals the postmaster might send us */
|
||||||
pqsignal(SIGHUP, SignalHandlerForConfigReload); /* set flag to read config
|
pqsignal(SIGHUP, SignalHandlerForConfigReload); /* set flag to read config
|
||||||
* file */
|
* file */
|
||||||
pqsignal(SIGINT, SIG_IGN);
|
pqsignal(SIGINT, PG_SIG_IGN);
|
||||||
pqsignal(SIGTERM, die); /* request shutdown */
|
pqsignal(SIGTERM, die); /* request shutdown */
|
||||||
/* SIGQUIT handler was already set up by InitPostmasterChild */
|
/* SIGQUIT handler was already set up by InitPostmasterChild */
|
||||||
pqsignal(SIGALRM, SIG_IGN);
|
pqsignal(SIGALRM, PG_SIG_IGN);
|
||||||
pqsignal(SIGPIPE, SIG_IGN);
|
pqsignal(SIGPIPE, PG_SIG_IGN);
|
||||||
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
||||||
pqsignal(SIGUSR2, SIG_IGN);
|
pqsignal(SIGUSR2, PG_SIG_IGN);
|
||||||
|
|
||||||
/* Reset some signals that are accepted by postmaster but not here */
|
/* Reset some signals that are accepted by postmaster but not here */
|
||||||
pqsignal(SIGCHLD, SIG_DFL);
|
pqsignal(SIGCHLD, PG_SIG_DFL);
|
||||||
|
|
||||||
/* Load the libpq-specific functions */
|
/* Load the libpq-specific functions */
|
||||||
load_file("libpqwalreceiver", false);
|
load_file("libpqwalreceiver", false);
|
||||||
|
|
|
||||||
|
|
@ -3897,13 +3897,13 @@ WalSndSignals(void)
|
||||||
pqsignal(SIGTERM, die); /* request shutdown */
|
pqsignal(SIGTERM, die); /* request shutdown */
|
||||||
/* SIGQUIT handler was already set up by InitPostmasterChild */
|
/* SIGQUIT handler was already set up by InitPostmasterChild */
|
||||||
InitializeTimeouts(); /* establishes SIGALRM handler */
|
InitializeTimeouts(); /* establishes SIGALRM handler */
|
||||||
pqsignal(SIGPIPE, SIG_IGN);
|
pqsignal(SIGPIPE, PG_SIG_IGN);
|
||||||
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
||||||
pqsignal(SIGUSR2, WalSndLastCycleHandler); /* request a last cycle and
|
pqsignal(SIGUSR2, WalSndLastCycleHandler); /* request a last cycle and
|
||||||
* shutdown */
|
* shutdown */
|
||||||
|
|
||||||
/* Reset some signals that are accepted by postmaster but not here */
|
/* Reset some signals that are accepted by postmaster but not here */
|
||||||
pqsignal(SIGCHLD, SIG_DFL);
|
pqsignal(SIGCHLD, PG_SIG_DFL);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Register shared-memory space needed by walsender */
|
/* Register shared-memory space needed by walsender */
|
||||||
|
|
|
||||||
|
|
@ -684,10 +684,10 @@ IoWorkerMain(const void *startup_data, size_t startup_data_len)
|
||||||
* Ignore SIGTERM, will get explicit shutdown via SIGUSR2 later in the
|
* Ignore SIGTERM, will get explicit shutdown via SIGUSR2 later in the
|
||||||
* shutdown sequence, similar to checkpointer.
|
* shutdown sequence, similar to checkpointer.
|
||||||
*/
|
*/
|
||||||
pqsignal(SIGTERM, SIG_IGN);
|
pqsignal(SIGTERM, PG_SIG_IGN);
|
||||||
/* SIGQUIT handler was already set up by InitPostmasterChild */
|
/* SIGQUIT handler was already set up by InitPostmasterChild */
|
||||||
pqsignal(SIGALRM, SIG_IGN);
|
pqsignal(SIGALRM, PG_SIG_IGN);
|
||||||
pqsignal(SIGPIPE, SIG_IGN);
|
pqsignal(SIGPIPE, PG_SIG_IGN);
|
||||||
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
||||||
pqsignal(SIGUSR2, SignalHandlerForShutdownRequest);
|
pqsignal(SIGUSR2, SignalHandlerForShutdownRequest);
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -2748,11 +2748,11 @@ OpenPipeStream(const char *command, const char *mode)
|
||||||
|
|
||||||
TryAgain:
|
TryAgain:
|
||||||
fflush(NULL);
|
fflush(NULL);
|
||||||
pqsignal(SIGPIPE, SIG_DFL);
|
pqsignal(SIGPIPE, PG_SIG_DFL);
|
||||||
errno = 0;
|
errno = 0;
|
||||||
file = popen(command, mode);
|
file = popen(command, mode);
|
||||||
save_errno = errno;
|
save_errno = errno;
|
||||||
pqsignal(SIGPIPE, SIG_IGN);
|
pqsignal(SIGPIPE, PG_SIG_IGN);
|
||||||
errno = save_errno;
|
errno = save_errno;
|
||||||
if (file != NULL)
|
if (file != NULL)
|
||||||
{
|
{
|
||||||
|
|
|
||||||
|
|
@ -348,7 +348,7 @@ InitializeWaitEventSupport(void)
|
||||||
|
|
||||||
#ifdef WAIT_USE_KQUEUE
|
#ifdef WAIT_USE_KQUEUE
|
||||||
/* Ignore SIGURG, because we'll receive it via kqueue. */
|
/* Ignore SIGURG, because we'll receive it via kqueue. */
|
||||||
pqsignal(SIGURG, SIG_IGN);
|
pqsignal(SIGURG, PG_SIG_IGN);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -3027,6 +3027,17 @@ die(SIGNAL_ARGS)
|
||||||
{
|
{
|
||||||
InterruptPending = true;
|
InterruptPending = true;
|
||||||
ProcDiePending = true;
|
ProcDiePending = true;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Record who sent the signal. Will be 0 on platforms without
|
||||||
|
* SA_SIGINFO, which is fine -- ProcessInterrupts() checks for that.
|
||||||
|
* Only set on the first SIGTERM so we report the original sender.
|
||||||
|
*/
|
||||||
|
if (ProcDieSenderPid == 0)
|
||||||
|
{
|
||||||
|
ProcDieSenderPid = pg_siginfo->pid;
|
||||||
|
ProcDieSenderUid = pg_siginfo->uid;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* for the cumulative stats system */
|
/* for the cumulative stats system */
|
||||||
|
|
@ -4316,17 +4327,17 @@ PostgresMain(const char *dbname, const char *username)
|
||||||
* returns to outer loop. This seems safer than forcing exit in the
|
* returns to outer loop. This seems safer than forcing exit in the
|
||||||
* midst of output during who-knows-what operation...
|
* midst of output during who-knows-what operation...
|
||||||
*/
|
*/
|
||||||
pqsignal(SIGPIPE, SIG_IGN);
|
pqsignal(SIGPIPE, PG_SIG_IGN);
|
||||||
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
pqsignal(SIGUSR1, procsignal_sigusr1_handler);
|
||||||
pqsignal(SIGUSR2, SIG_IGN);
|
pqsignal(SIGUSR2, PG_SIG_IGN);
|
||||||
pqsignal(SIGFPE, FloatExceptionHandler);
|
pqsignal(SIGFPE, FloatExceptionHandler);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Reset some signals that are accepted by postmaster but not by
|
* Reset some signals that are accepted by postmaster but not by
|
||||||
* backend
|
* backend
|
||||||
*/
|
*/
|
||||||
pqsignal(SIGCHLD, SIG_DFL); /* system() requires this on some
|
pqsignal(SIGCHLD, PG_SIG_DFL); /* system() requires this on some
|
||||||
* platforms */
|
* platforms */
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Early initialization */
|
/* Early initialization */
|
||||||
|
|
|
||||||
|
|
@ -2903,10 +2903,10 @@ setup_signals(void)
|
||||||
pqsignal(SIGQUIT, trapsig);
|
pqsignal(SIGQUIT, trapsig);
|
||||||
|
|
||||||
/* Ignore SIGPIPE when writing to backend, so we can clean up */
|
/* Ignore SIGPIPE when writing to backend, so we can clean up */
|
||||||
pqsignal(SIGPIPE, SIG_IGN);
|
pqsignal(SIGPIPE, PG_SIG_IGN);
|
||||||
|
|
||||||
/* Prevent SIGSYS so we can probe for kernel calls that might not work */
|
/* Prevent SIGSYS so we can probe for kernel calls that might not work */
|
||||||
pqsignal(SIGSYS, SIG_IGN);
|
pqsignal(SIGSYS, PG_SIG_IGN);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -868,7 +868,7 @@ trap_sigint_during_startup(SIGNAL_ARGS)
|
||||||
* Clear the signal handler, and send the signal again, to terminate the
|
* Clear the signal handler, and send the signal again, to terminate the
|
||||||
* process as normal.
|
* process as normal.
|
||||||
*/
|
*/
|
||||||
pqsignal(postgres_signal_arg, SIG_DFL);
|
pqsignal(postgres_signal_arg, PG_SIG_DFL);
|
||||||
raise(postgres_signal_arg);
|
raise(postgres_signal_arg);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -568,9 +568,9 @@ sigTermHandler(SIGNAL_ARGS)
|
||||||
* signal handler. That could muck up our attempt to send PQcancel, so
|
* signal handler. That could muck up our attempt to send PQcancel, so
|
||||||
* disable the signals that set_cancel_handler enabled.
|
* disable the signals that set_cancel_handler enabled.
|
||||||
*/
|
*/
|
||||||
pqsignal(SIGINT, SIG_IGN);
|
pqsignal(SIGINT, PG_SIG_IGN);
|
||||||
pqsignal(SIGTERM, SIG_IGN);
|
pqsignal(SIGTERM, PG_SIG_IGN);
|
||||||
pqsignal(SIGQUIT, SIG_IGN);
|
pqsignal(SIGQUIT, PG_SIG_IGN);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If we're in the leader, forward signal to all workers. (It seems best
|
* If we're in the leader, forward signal to all workers. (It seems best
|
||||||
|
|
@ -1049,7 +1049,7 @@ ParallelBackupStart(ArchiveHandle *AH)
|
||||||
* the workers to inherit this setting, though.
|
* the workers to inherit this setting, though.
|
||||||
*/
|
*/
|
||||||
#ifndef WIN32
|
#ifndef WIN32
|
||||||
pqsignal(SIGPIPE, SIG_IGN);
|
pqsignal(SIGPIPE, PG_SIG_IGN);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
|
||||||
|
|
@ -142,8 +142,11 @@ my ($ret, $out, $err) = $node->psql('postgres',
|
||||||
is($ret, 2, 'server crash: psql exit code');
|
is($ret, 2, 'server crash: psql exit code');
|
||||||
like($out, qr/before/, 'server crash: output before crash');
|
like($out, qr/before/, 'server crash: output before crash');
|
||||||
unlike($out, qr/AFTER/, 'server crash: no output after crash');
|
unlike($out, qr/AFTER/, 'server crash: no output after crash');
|
||||||
|
my $detail_re = check_pg_config("#define HAVE_SA_SIGINFO 1")
|
||||||
|
? qr/DETAIL: Signal sent by PID \d+, UID \d+\.\n/
|
||||||
|
: qr//;
|
||||||
like( $err, qr/psql:<stdin>:2: FATAL: terminating connection due to administrator command
|
like( $err, qr/psql:<stdin>:2: FATAL: terminating connection due to administrator command
|
||||||
(?:DETAIL: Signal sent by PID \d+, UID \d+\.\n)?psql:<stdin>:2: server closed the connection unexpectedly
|
${detail_re}psql:<stdin>:2: server closed the connection unexpectedly
|
||||||
This probably means the server terminated abnormally
|
This probably means the server terminated abnormally
|
||||||
before or while processing the request.
|
before or while processing the request.
|
||||||
psql:<stdin>:2: error: connection to server was lost/,
|
psql:<stdin>:2: error: connection to server was lost/,
|
||||||
|
|
|
||||||
|
|
@ -3024,7 +3024,7 @@ void
|
||||||
disable_sigpipe_trap(void)
|
disable_sigpipe_trap(void)
|
||||||
{
|
{
|
||||||
#ifndef WIN32
|
#ifndef WIN32
|
||||||
pqsignal(SIGPIPE, SIG_IGN);
|
pqsignal(SIGPIPE, PG_SIG_IGN);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -3047,7 +3047,7 @@ void
|
||||||
restore_sigpipe_trap(void)
|
restore_sigpipe_trap(void)
|
||||||
{
|
{
|
||||||
#ifndef WIN32
|
#ifndef WIN32
|
||||||
pqsignal(SIGPIPE, always_ignore_sigpipe ? SIG_IGN : SIG_DFL);
|
pqsignal(SIGPIPE, always_ignore_sigpipe ? PG_SIG_IGN : PG_SIG_DFL);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1441,17 +1441,28 @@ extern int fdatasync(int fd);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The following is used as the arg list for signal handlers. Any ports
|
* Platform independent struct representing additional information about the
|
||||||
* that take something other than an int argument should override this in
|
* received signal. If the system does not support the extended information,
|
||||||
* their pg_config_os.h file. Note that variable names are required
|
* or a field does not apply to the signal, the value is instead reset to the
|
||||||
* because it is used in both the prototypes as well as the definitions.
|
* documented default value.
|
||||||
* Note also the long name. We expect that this won't collide with
|
|
||||||
* other names causing compiler warnings.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef SIGNAL_ARGS
|
typedef struct pg_signal_info
|
||||||
#define SIGNAL_ARGS int postgres_signal_arg
|
{
|
||||||
#endif
|
uint32_t pid; /* pid of sending process or 0 if unknown */
|
||||||
|
uint32_t uid; /* uid of sending process; only meaningful
|
||||||
|
* when pid is not 0 */
|
||||||
|
} pg_signal_info;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The following is used as the arg list for signal handlers. These days we
|
||||||
|
* use the same argument to all signal handlers and hide the difference
|
||||||
|
* between platforms in wrapper functions.
|
||||||
|
*
|
||||||
|
* SIGNAL_ARGS just exists separately from the pqsignal() definition for
|
||||||
|
* historical reasons.
|
||||||
|
*/
|
||||||
|
#define SIGNAL_ARGS int postgres_signal_arg, const pg_signal_info *pg_siginfo
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* When there is no sigsetjmp, its functionality is provided by plain
|
* When there is no sigsetjmp, its functionality is provided by plain
|
||||||
|
|
|
||||||
|
|
@ -546,6 +546,9 @@ extern int pg_mkdir_p(char *path, int omode);
|
||||||
#else
|
#else
|
||||||
#define pqsignal pqsignal_be
|
#define pqsignal pqsignal_be
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#define PG_SIG_DFL (pqsigfunc) (pg_funcptr_t) SIG_DFL
|
||||||
|
#define PG_SIG_IGN (pqsigfunc) (pg_funcptr_t) SIG_IGN
|
||||||
typedef void (*pqsigfunc) (SIGNAL_ARGS);
|
typedef void (*pqsigfunc) (SIGNAL_ARGS);
|
||||||
extern void pqsignal(int signo, pqsigfunc func);
|
extern void pqsignal(int signo, pqsigfunc func);
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -36,10 +36,12 @@
|
||||||
* is to ensure that no in-tree code accidentally calls this version.)
|
* is to ensure that no in-tree code accidentally calls this version.)
|
||||||
*/
|
*/
|
||||||
#undef pqsignal
|
#undef pqsignal
|
||||||
extern pqsigfunc pqsignal(int signo, pqsigfunc func);
|
|
||||||
|
|
||||||
pqsigfunc
|
typedef void (*pqsigfunc_legacy) (int postgres_signal_arg);
|
||||||
pqsignal(int signo, pqsigfunc func)
|
extern pqsigfunc_legacy pqsignal(int signo, pqsigfunc_legacy func);
|
||||||
|
|
||||||
|
pqsigfunc_legacy
|
||||||
|
pqsignal(int signo, pqsigfunc_legacy func)
|
||||||
{
|
{
|
||||||
#ifndef WIN32
|
#ifndef WIN32
|
||||||
struct sigaction act,
|
struct sigaction act,
|
||||||
|
|
|
||||||
|
|
@ -63,6 +63,14 @@
|
||||||
#define PG_NSIG (64) /* XXX: wild guess */
|
#define PG_NSIG (64) /* XXX: wild guess */
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#if !(defined(WIN32) && defined(FRONTEND))
|
||||||
|
#define USE_SIGACTION
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if defined(USE_SIGACTION) && defined(HAVE_SA_SIGINFO)
|
||||||
|
#define USE_SIGINFO
|
||||||
|
#endif
|
||||||
|
|
||||||
/* Check a couple of common signals to make sure PG_NSIG is accurate. */
|
/* Check a couple of common signals to make sure PG_NSIG is accurate. */
|
||||||
StaticAssertDecl(SIGUSR2 < PG_NSIG, "SIGUSR2 >= PG_NSIG");
|
StaticAssertDecl(SIGUSR2 < PG_NSIG, "SIGUSR2 >= PG_NSIG");
|
||||||
StaticAssertDecl(SIGHUP < PG_NSIG, "SIGHUP >= PG_NSIG");
|
StaticAssertDecl(SIGHUP < PG_NSIG, "SIGHUP >= PG_NSIG");
|
||||||
|
|
@ -82,19 +90,16 @@ static volatile pqsigfunc pqsignal_handlers[PG_NSIG];
|
||||||
*
|
*
|
||||||
* This wrapper also handles restoring the value of errno.
|
* This wrapper also handles restoring the value of errno.
|
||||||
*/
|
*/
|
||||||
#if !defined(FRONTEND) && defined(HAVE_SA_SIGINFO)
|
#if defined(USE_SIGACTION) && defined(USE_SIGINFO)
|
||||||
static void
|
static void
|
||||||
wrapper_handler(int signo, siginfo_t * info, void *context)
|
wrapper_handler(int postgres_signal_arg, siginfo_t * info, void *context)
|
||||||
#else
|
#else /* no USE_SIGINFO */
|
||||||
static void
|
static void
|
||||||
wrapper_handler(SIGNAL_ARGS)
|
wrapper_handler(int postgres_signal_arg)
|
||||||
#endif
|
#endif
|
||||||
{
|
{
|
||||||
int save_errno = errno;
|
int save_errno = errno;
|
||||||
#if !defined(FRONTEND) && defined(HAVE_SA_SIGINFO)
|
pg_signal_info pg_info;
|
||||||
/* SA_SIGINFO signature uses signo, not SIGNAL_ARGS macro */
|
|
||||||
int postgres_signal_arg = signo;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
Assert(postgres_signal_arg > 0);
|
Assert(postgres_signal_arg > 0);
|
||||||
Assert(postgres_signal_arg < PG_NSIG);
|
Assert(postgres_signal_arg < PG_NSIG);
|
||||||
|
|
@ -110,21 +115,32 @@ wrapper_handler(SIGNAL_ARGS)
|
||||||
|
|
||||||
if (unlikely(MyProcPid != (int) getpid()))
|
if (unlikely(MyProcPid != (int) getpid()))
|
||||||
{
|
{
|
||||||
pqsignal(postgres_signal_arg, SIG_DFL);
|
pqsignal(postgres_signal_arg, PG_SIG_DFL);
|
||||||
raise(postgres_signal_arg);
|
raise(postgres_signal_arg);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
#ifdef HAVE_SA_SIGINFO
|
#ifdef HAVE_SA_SIGINFO
|
||||||
if (signo == SIGTERM && info)
|
|
||||||
{
|
/*
|
||||||
ProcDieSenderPid = info->si_pid;
|
* If supported by the system, forward interesting information from the
|
||||||
ProcDieSenderUid = info->si_uid;
|
* system's extended signal information to our platform independent
|
||||||
}
|
* format.
|
||||||
#endif
|
*/
|
||||||
|
pg_info.pid = info->si_pid;
|
||||||
|
pg_info.uid = info->si_uid;
|
||||||
|
#else
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Otherwise forward values indicating that we do not have the
|
||||||
|
* information.
|
||||||
|
*/
|
||||||
|
pg_info.pid = 0;
|
||||||
|
pg_info.uid = 0;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
(*pqsignal_handlers[postgres_signal_arg]) (postgres_signal_arg);
|
(*pqsignal_handlers[postgres_signal_arg]) (postgres_signal_arg, &pg_info);
|
||||||
|
|
||||||
errno = save_errno;
|
errno = save_errno;
|
||||||
}
|
}
|
||||||
|
|
@ -139,33 +155,44 @@ wrapper_handler(SIGNAL_ARGS)
|
||||||
void
|
void
|
||||||
pqsignal(int signo, pqsigfunc func)
|
pqsignal(int signo, pqsigfunc func)
|
||||||
{
|
{
|
||||||
#if !(defined(WIN32) && defined(FRONTEND))
|
#ifdef USE_SIGACTION
|
||||||
struct sigaction act;
|
struct sigaction act;
|
||||||
|
#else
|
||||||
|
void (*wrapper_func_ptr) (int);
|
||||||
#endif
|
#endif
|
||||||
bool use_wrapper = false;
|
bool is_ign = func == PG_SIG_IGN;
|
||||||
|
bool is_dfl = func == PG_SIG_DFL;
|
||||||
|
|
||||||
Assert(signo > 0);
|
Assert(signo > 0);
|
||||||
Assert(signo < PG_NSIG);
|
Assert(signo < PG_NSIG);
|
||||||
|
|
||||||
if (func != SIG_IGN && func != SIG_DFL)
|
/* set up indirection handler */
|
||||||
|
if (!(is_ign || is_dfl))
|
||||||
{
|
{
|
||||||
pqsignal_handlers[signo] = func; /* assumed atomic */
|
pqsignal_handlers[signo] = func; /* assumed atomic */
|
||||||
use_wrapper = true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#if !(defined(WIN32) && defined(FRONTEND))
|
/*
|
||||||
|
* Configure system to either ignore/reset the signal handler, or to
|
||||||
|
* forward it to wrapper_handler.
|
||||||
|
*/
|
||||||
|
#ifdef USE_SIGACTION
|
||||||
sigemptyset(&act.sa_mask);
|
sigemptyset(&act.sa_mask);
|
||||||
act.sa_flags = SA_RESTART;
|
act.sa_flags = SA_RESTART;
|
||||||
#if !defined(FRONTEND) && defined(HAVE_SA_SIGINFO)
|
|
||||||
if (use_wrapper)
|
if (is_ign)
|
||||||
|
act.sa_handler = SIG_IGN;
|
||||||
|
else if (is_dfl)
|
||||||
|
act.sa_handler = SIG_DFL;
|
||||||
|
#ifdef USE_SIGINFO
|
||||||
|
else
|
||||||
{
|
{
|
||||||
act.sa_sigaction = wrapper_handler;
|
act.sa_sigaction = wrapper_handler;
|
||||||
act.sa_flags |= SA_SIGINFO;
|
act.sa_flags |= SA_SIGINFO;
|
||||||
}
|
}
|
||||||
else
|
|
||||||
act.sa_handler = func;
|
|
||||||
#else
|
#else
|
||||||
act.sa_handler = use_wrapper ? wrapper_handler : func;
|
else
|
||||||
|
act.sa_handler = wrapper_handler;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef SA_NOCLDSTOP
|
#ifdef SA_NOCLDSTOP
|
||||||
|
|
@ -174,9 +201,20 @@ pqsignal(int signo, pqsigfunc func)
|
||||||
#endif
|
#endif
|
||||||
if (sigaction(signo, &act, NULL) < 0)
|
if (sigaction(signo, &act, NULL) < 0)
|
||||||
Assert(false); /* probably indicates coding error */
|
Assert(false); /* probably indicates coding error */
|
||||||
#else
|
#else /* no USE_SIGACTION */
|
||||||
/* Forward to Windows native signal system. */
|
|
||||||
if (signal(signo, use_wrapper ? wrapper_handler : func) == SIG_ERR)
|
/*
|
||||||
|
* Forward to Windows native signal system, we need to send this though
|
||||||
|
* wrapper handler as it it needs to take single argument only.
|
||||||
|
*/
|
||||||
|
if (is_ign)
|
||||||
|
wrapper_func_ptr = SIG_IGN;
|
||||||
|
else if (is_dfl)
|
||||||
|
wrapper_func_ptr = SIG_DFL;
|
||||||
|
else
|
||||||
|
wrapper_func_ptr = wrapper_handler;
|
||||||
|
|
||||||
|
if (signal(signo, wrapper_func_ptr) == SIG_ERR)
|
||||||
Assert(false); /* probably indicates coding error */
|
Assert(false); /* probably indicates coding error */
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -492,7 +492,7 @@ signal_remove_temp(SIGNAL_ARGS)
|
||||||
{
|
{
|
||||||
remove_temp();
|
remove_temp();
|
||||||
|
|
||||||
pqsignal(postgres_signal_arg, SIG_DFL);
|
pqsignal(postgres_signal_arg, PG_SIG_DFL);
|
||||||
raise(postgres_signal_arg);
|
raise(postgres_signal_arg);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -4039,6 +4039,7 @@ pg_sha224_ctx
|
||||||
pg_sha256_ctx
|
pg_sha256_ctx
|
||||||
pg_sha384_ctx
|
pg_sha384_ctx
|
||||||
pg_sha512_ctx
|
pg_sha512_ctx
|
||||||
|
pg_signal_info
|
||||||
pg_snapshot
|
pg_snapshot
|
||||||
pg_special_case
|
pg_special_case
|
||||||
pg_stack_base_t
|
pg_stack_base_t
|
||||||
|
|
|
||||||
Loading…
Reference in a new issue