mirror of
https://git.postgresql.org/git/postgresql.git
synced 2025-01-06 15:24:56 +08:00
Temporarily revert stats collector latch changes so we can ship beta1.
This patch reverts commit 49340037ee
and some
follow-on tweaking in pgstat.c. While the basic scheme of latch-ifying the
stats collector seems sound enough, it's failing on most Windows buildfarm
members for unknown reasons, and there's no time left to debug that before
9.2beta1. Better to ship a beta version without this improvement. I hope
to re-revert this once beta1 is out, though.
This commit is contained in:
parent
5428ff4af8
commit
cb2f2873d6
@ -28,6 +28,12 @@
|
|||||||
#include <arpa/inet.h>
|
#include <arpa/inet.h>
|
||||||
#include <signal.h>
|
#include <signal.h>
|
||||||
#include <time.h>
|
#include <time.h>
|
||||||
|
#ifdef HAVE_POLL_H
|
||||||
|
#include <poll.h>
|
||||||
|
#endif
|
||||||
|
#ifdef HAVE_SYS_POLL_H
|
||||||
|
#include <sys/poll.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
#include "pgstat.h"
|
#include "pgstat.h"
|
||||||
|
|
||||||
@ -49,8 +55,8 @@
|
|||||||
#include "storage/backendid.h"
|
#include "storage/backendid.h"
|
||||||
#include "storage/fd.h"
|
#include "storage/fd.h"
|
||||||
#include "storage/ipc.h"
|
#include "storage/ipc.h"
|
||||||
#include "storage/latch.h"
|
|
||||||
#include "storage/pg_shmem.h"
|
#include "storage/pg_shmem.h"
|
||||||
|
#include "storage/pmsignal.h"
|
||||||
#include "storage/procsignal.h"
|
#include "storage/procsignal.h"
|
||||||
#include "utils/ascii.h"
|
#include "utils/ascii.h"
|
||||||
#include "utils/guc.h"
|
#include "utils/guc.h"
|
||||||
@ -88,6 +94,9 @@
|
|||||||
* failed statistics collector; in
|
* failed statistics collector; in
|
||||||
* seconds. */
|
* seconds. */
|
||||||
|
|
||||||
|
#define PGSTAT_SELECT_TIMEOUT 2 /* How often to check for postmaster
|
||||||
|
* death; in seconds. */
|
||||||
|
|
||||||
#define PGSTAT_POLL_LOOP_COUNT (PGSTAT_MAX_WAIT_TIME / PGSTAT_RETRY_DELAY)
|
#define PGSTAT_POLL_LOOP_COUNT (PGSTAT_MAX_WAIT_TIME / PGSTAT_RETRY_DELAY)
|
||||||
#define PGSTAT_INQ_LOOP_COUNT (PGSTAT_INQ_INTERVAL / PGSTAT_RETRY_DELAY)
|
#define PGSTAT_INQ_LOOP_COUNT (PGSTAT_INQ_INTERVAL / PGSTAT_RETRY_DELAY)
|
||||||
|
|
||||||
@ -130,8 +139,6 @@ PgStat_MsgBgWriter BgWriterStats;
|
|||||||
*/
|
*/
|
||||||
NON_EXEC_STATIC pgsocket pgStatSock = PGINVALID_SOCKET;
|
NON_EXEC_STATIC pgsocket pgStatSock = PGINVALID_SOCKET;
|
||||||
|
|
||||||
static Latch pgStatLatch;
|
|
||||||
|
|
||||||
static struct sockaddr_storage pgStatAddr;
|
static struct sockaddr_storage pgStatAddr;
|
||||||
|
|
||||||
static time_t last_pgstat_start_time;
|
static time_t last_pgstat_start_time;
|
||||||
@ -3002,7 +3009,15 @@ PgstatCollectorMain(int argc, char *argv[])
|
|||||||
{
|
{
|
||||||
int len;
|
int len;
|
||||||
PgStat_Msg msg;
|
PgStat_Msg msg;
|
||||||
int wr;
|
|
||||||
|
#ifndef WIN32
|
||||||
|
#ifdef HAVE_POLL
|
||||||
|
struct pollfd input_fd;
|
||||||
|
#else
|
||||||
|
struct timeval sel_timeout;
|
||||||
|
fd_set rfds;
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
IsUnderPostmaster = true; /* we are a postmaster subprocess now */
|
IsUnderPostmaster = true; /* we are a postmaster subprocess now */
|
||||||
|
|
||||||
@ -3021,13 +3036,9 @@ PgstatCollectorMain(int argc, char *argv[])
|
|||||||
elog(FATAL, "setsid() failed: %m");
|
elog(FATAL, "setsid() failed: %m");
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* Initialize private latch for use by signal handlers */
|
|
||||||
InitLatch(&pgStatLatch);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Ignore all signals usually bound to some action in the postmaster,
|
* Ignore all signals usually bound to some action in the postmaster,
|
||||||
* except SIGHUP and SIGQUIT. Note we don't need a SIGUSR1 handler to
|
* except SIGQUIT.
|
||||||
* support latch operations, because pgStatLatch is local not shared.
|
|
||||||
*/
|
*/
|
||||||
pqsignal(SIGHUP, pgstat_sighup_handler);
|
pqsignal(SIGHUP, pgstat_sighup_handler);
|
||||||
pqsignal(SIGINT, SIG_IGN);
|
pqsignal(SIGINT, SIG_IGN);
|
||||||
@ -3062,24 +3073,26 @@ PgstatCollectorMain(int argc, char *argv[])
|
|||||||
pgStatRunningInCollector = true;
|
pgStatRunningInCollector = true;
|
||||||
pgStatDBHash = pgstat_read_statsfile(InvalidOid, true);
|
pgStatDBHash = pgstat_read_statsfile(InvalidOid, true);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Setup the descriptor set for select(2). Since only one bit in the set
|
||||||
|
* ever changes, we need not repeat FD_ZERO each time.
|
||||||
|
*/
|
||||||
|
#if !defined(HAVE_POLL) && !defined(WIN32)
|
||||||
|
FD_ZERO(&rfds);
|
||||||
|
#endif
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Loop to process messages until we get SIGQUIT or detect ungraceful
|
* Loop to process messages until we get SIGQUIT or detect ungraceful
|
||||||
* death of our parent postmaster.
|
* death of our parent postmaster.
|
||||||
*
|
*
|
||||||
* For performance reasons, we don't want to do ResetLatch/WaitLatch after
|
* For performance reasons, we don't want to do a PostmasterIsAlive() test
|
||||||
* every message; instead, do that only after a recv() fails to obtain a
|
* after every message; instead, do it only when select()/poll() is
|
||||||
* message. (This effectively means that if backends are sending us stuff
|
* interrupted by timeout. In essence, we'll stay alive as long as
|
||||||
* like mad, we won't notice postmaster death until things slack off a
|
* backends keep sending us stuff often, even if the postmaster is gone.
|
||||||
* bit; which seems fine.) To do that, we have an inner loop that
|
|
||||||
* iterates as long as recv() succeeds. We do recognize got_SIGHUP inside
|
|
||||||
* the inner loop, which means that such interrupts will get serviced but
|
|
||||||
* the latch won't get cleared until next time there is a break in the
|
|
||||||
* action.
|
|
||||||
*/
|
*/
|
||||||
for (;;)
|
for (;;)
|
||||||
{
|
{
|
||||||
/* Clear any already-pending wakeups */
|
int got_data;
|
||||||
ResetLatch(&pgStatLatch);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Quit if we get SIGQUIT from the postmaster.
|
* Quit if we get SIGQUIT from the postmaster.
|
||||||
@ -3088,37 +3101,87 @@ PgstatCollectorMain(int argc, char *argv[])
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Inner loop iterates as long as we keep getting messages, or until
|
* Reload configuration if we got SIGHUP from the postmaster.
|
||||||
* need_exit becomes set.
|
|
||||||
*/
|
*/
|
||||||
while (!need_exit)
|
if (got_SIGHUP)
|
||||||
{
|
{
|
||||||
/*
|
ProcessConfigFile(PGC_SIGHUP);
|
||||||
* Reload configuration if we got SIGHUP from the postmaster.
|
got_SIGHUP = false;
|
||||||
*/
|
}
|
||||||
if (got_SIGHUP)
|
|
||||||
{
|
|
||||||
got_SIGHUP = false;
|
|
||||||
ProcessConfigFile(PGC_SIGHUP);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Write the stats file if a new request has arrived that is not
|
* Write the stats file if a new request has arrived that is not
|
||||||
* satisfied by existing file.
|
* satisfied by existing file.
|
||||||
*/
|
*/
|
||||||
if (last_statwrite < last_statrequest)
|
if (last_statwrite < last_statrequest)
|
||||||
pgstat_write_statsfile(false);
|
pgstat_write_statsfile(false);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Try to receive and process a message. This will not block,
|
* Wait for a message to arrive; but not for more than
|
||||||
* since the socket is set to non-blocking mode.
|
* PGSTAT_SELECT_TIMEOUT seconds. (This determines how quickly we will
|
||||||
*/
|
* shut down after an ungraceful postmaster termination; so it needn't
|
||||||
|
* be very fast. However, on some systems SIGQUIT won't interrupt the
|
||||||
|
* poll/select call, so this also limits speed of response to SIGQUIT,
|
||||||
|
* which is more important.)
|
||||||
|
*
|
||||||
|
* We use poll(2) if available, otherwise select(2). Win32 has its own
|
||||||
|
* implementation.
|
||||||
|
*/
|
||||||
|
#ifndef WIN32
|
||||||
|
#ifdef HAVE_POLL
|
||||||
|
input_fd.fd = pgStatSock;
|
||||||
|
input_fd.events = POLLIN | POLLERR;
|
||||||
|
input_fd.revents = 0;
|
||||||
|
|
||||||
|
if (poll(&input_fd, 1, PGSTAT_SELECT_TIMEOUT * 1000) < 0)
|
||||||
|
{
|
||||||
|
if (errno == EINTR)
|
||||||
|
continue;
|
||||||
|
ereport(ERROR,
|
||||||
|
(errcode_for_socket_access(),
|
||||||
|
errmsg("poll() failed in statistics collector: %m")));
|
||||||
|
}
|
||||||
|
|
||||||
|
got_data = (input_fd.revents != 0);
|
||||||
|
#else /* !HAVE_POLL */
|
||||||
|
|
||||||
|
FD_SET(pgStatSock, &rfds);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* timeout struct is modified by select() on some operating systems,
|
||||||
|
* so re-fill it each time.
|
||||||
|
*/
|
||||||
|
sel_timeout.tv_sec = PGSTAT_SELECT_TIMEOUT;
|
||||||
|
sel_timeout.tv_usec = 0;
|
||||||
|
|
||||||
|
if (select(pgStatSock + 1, &rfds, NULL, NULL, &sel_timeout) < 0)
|
||||||
|
{
|
||||||
|
if (errno == EINTR)
|
||||||
|
continue;
|
||||||
|
ereport(ERROR,
|
||||||
|
(errcode_for_socket_access(),
|
||||||
|
errmsg("select() failed in statistics collector: %m")));
|
||||||
|
}
|
||||||
|
|
||||||
|
got_data = FD_ISSET(pgStatSock, &rfds);
|
||||||
|
#endif /* HAVE_POLL */
|
||||||
|
#else /* WIN32 */
|
||||||
|
got_data = pgwin32_waitforsinglesocket(pgStatSock, FD_READ,
|
||||||
|
PGSTAT_SELECT_TIMEOUT * 1000);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If there is a message on the socket, read it and check for
|
||||||
|
* validity.
|
||||||
|
*/
|
||||||
|
if (got_data)
|
||||||
|
{
|
||||||
len = recv(pgStatSock, (char *) &msg,
|
len = recv(pgStatSock, (char *) &msg,
|
||||||
sizeof(PgStat_Msg), 0);
|
sizeof(PgStat_Msg), 0);
|
||||||
if (len < 0)
|
if (len < 0)
|
||||||
{
|
{
|
||||||
if (errno == EAGAIN || errno == EWOULDBLOCK || errno == EINTR)
|
if (errno == EINTR)
|
||||||
break; /* out of inner loop */
|
continue;
|
||||||
ereport(ERROR,
|
ereport(ERROR,
|
||||||
(errcode_for_socket_access(),
|
(errcode_for_socket_access(),
|
||||||
errmsg("could not read statistics message: %m")));
|
errmsg("could not read statistics message: %m")));
|
||||||
@ -3216,21 +3279,17 @@ PgstatCollectorMain(int argc, char *argv[])
|
|||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
} /* end of inner message-processing loop */
|
}
|
||||||
|
else
|
||||||
/* Sleep until there's something to do */
|
{
|
||||||
wr = WaitLatchOrSocket(&pgStatLatch,
|
/*
|
||||||
WL_LATCH_SET | WL_POSTMASTER_DEATH | WL_SOCKET_READABLE,
|
* We can only get here if the select/poll timeout elapsed. Check
|
||||||
pgStatSock,
|
* for postmaster death.
|
||||||
-1L);
|
*/
|
||||||
|
if (!PostmasterIsAlive())
|
||||||
/*
|
break;
|
||||||
* Emergency bailout if postmaster has died. This is to avoid the
|
}
|
||||||
* necessity for manual cleanup of all postmaster children.
|
} /* end of message-processing loop */
|
||||||
*/
|
|
||||||
if (wr & WL_POSTMASTER_DEATH)
|
|
||||||
break;
|
|
||||||
} /* end of outer loop */
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Save the final stats to reuse at next startup.
|
* Save the final stats to reuse at next startup.
|
||||||
@ -3245,24 +3304,14 @@ PgstatCollectorMain(int argc, char *argv[])
|
|||||||
static void
|
static void
|
||||||
pgstat_exit(SIGNAL_ARGS)
|
pgstat_exit(SIGNAL_ARGS)
|
||||||
{
|
{
|
||||||
int save_errno = errno;
|
|
||||||
|
|
||||||
need_exit = true;
|
need_exit = true;
|
||||||
SetLatch(&pgStatLatch);
|
|
||||||
|
|
||||||
errno = save_errno;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* SIGHUP handler for collector process */
|
/* SIGHUP handler for collector process */
|
||||||
static void
|
static void
|
||||||
pgstat_sighup_handler(SIGNAL_ARGS)
|
pgstat_sighup_handler(SIGNAL_ARGS)
|
||||||
{
|
{
|
||||||
int save_errno = errno;
|
|
||||||
|
|
||||||
got_SIGHUP = true;
|
got_SIGHUP = true;
|
||||||
SetLatch(&pgStatLatch);
|
|
||||||
|
|
||||||
errno = save_errno;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user