mirror of
https://github.com/postgres/postgres.git
synced 2025-08-27 07:42:10 +03:00
In the name of ABI stability (that is, to avoid a library major version bump for libpq), libpq still exports a version of pqsignal() that we no longer want to use ourselves. However, since that has the same link name as the function exported by src/port/pqsignal.c, there is a link ordering dependency determining which version will actually get used by code that uses libpq as well as libpgport.a. It now emerges that the wrong version has been used by pgbench and psql since commit06843df4a
rearranged their link commands. This can result in odd failures in pgbench with the -T switch, since its SIGALRM handler will now not be marked SA_RESTART. psql may have some edge-case problems in \watch, too. Since we don't want to depend on link ordering effects anymore, let's fix this in the same spirit asb6c7cfac8
: use macros to change the actual link names of the competing functions. We cannot change legacy-pqsignal.c's exported name of course, so the victim has to be src/port/pqsignal.c. In master, rename its exported name to be pqsignal_fe in frontend or pqsignal_be in backend. (We could perhaps have gotten away with using the same symbol in both cases, but since the FE and BE versions now work a little differently, it seems advisable to use different names.) In back branches, rename to pqsignal_fe in frontend but keep it as pqsignal in backend. The frontend change could affect third-party code that is calling pqsignal from libpgport.a or libpgport_shlib.a, but only if the code is compiled against port.h from a different minor release than libpgport. Since we don't support using libpgport as a shared library, it seems unlikely that there will be such a problem. I left the backend symbol unchanged to avoid an ABI break for extensions. This means that the link ordering hazard still exists for any extension that links against libpq. However, none of our own extensions use both pqsignal() and libpq, and we're not making things any worse for third-party extensions that do. Report from Andy Fan, diagnosis by Fujii Masao, patch by me. Back-patch to all supported branches, as06843df4a
was. Discussion: https://postgr.es/m/87msfz5qv2.fsf@163.com
66 lines
1.9 KiB
C
66 lines
1.9 KiB
C
/*-------------------------------------------------------------------------
|
|
*
|
|
* pqsignal.c
|
|
* reliable BSD-style signal(2) routine stolen from RWW who stole it
|
|
* from Stevens...
|
|
*
|
|
* Portions Copyright (c) 1996-2022, PostgreSQL Global Development Group
|
|
* Portions Copyright (c) 1994, Regents of the University of California
|
|
*
|
|
*
|
|
* IDENTIFICATION
|
|
* src/port/pqsignal.c
|
|
*
|
|
* We now assume that all Unix-oid systems have POSIX sigaction(2)
|
|
* with support for restartable signals (SA_RESTART). We used to also
|
|
* support BSD-style signal(2), but there really shouldn't be anything
|
|
* out there anymore that doesn't have the POSIX API.
|
|
*
|
|
* Windows, of course, is resolutely in a class by itself. In the backend,
|
|
* we don't use this file at all; src/backend/port/win32/signal.c provides
|
|
* pqsignal() for the backend environment. Frontend programs can use
|
|
* this version of pqsignal() if they wish, but beware that this does
|
|
* not provide restartable signals on Windows.
|
|
*
|
|
* ------------------------------------------------------------------------
|
|
*/
|
|
|
|
#include "c.h"
|
|
|
|
#include <signal.h>
|
|
|
|
#if !defined(WIN32) || defined(FRONTEND)
|
|
|
|
/*
|
|
* Set up a signal handler, with SA_RESTART, for signal "signo"
|
|
*
|
|
* Returns the previous handler.
|
|
*
|
|
* Note: the actual name of this function is either pqsignal_fe when
|
|
* compiled with -DFRONTEND, or pqsignal when compiled without that.
|
|
* This is to avoid a name collision with libpq's legacy-pqsignal.c.
|
|
*/
|
|
pqsigfunc
|
|
pqsignal(int signo, pqsigfunc func)
|
|
{
|
|
#ifndef WIN32
|
|
struct sigaction act,
|
|
oact;
|
|
|
|
act.sa_handler = func;
|
|
sigemptyset(&act.sa_mask);
|
|
act.sa_flags = SA_RESTART;
|
|
#ifdef SA_NOCLDSTOP
|
|
if (signo == SIGCHLD)
|
|
act.sa_flags |= SA_NOCLDSTOP;
|
|
#endif
|
|
if (sigaction(signo, &act, &oact) < 0)
|
|
return SIG_ERR;
|
|
return oact.sa_handler;
|
|
#else /* WIN32 */
|
|
return signal(signo, func);
|
|
#endif
|
|
}
|
|
|
|
#endif /* !defined(WIN32) || defined(FRONTEND) */
|