diff --git a/src/backend/postmaster/pgstat.c b/src/backend/postmaster/pgstat.c index 788dedb9ee97c5981bcfeae20e9cc050f9924330..c40b90b5993ad745795793e0eaf8acfe060adcc8 100644 --- a/src/backend/postmaster/pgstat.c +++ b/src/backend/postmaster/pgstat.c @@ -13,7 +13,7 @@ * * Copyright (c) 2001-2006, PostgreSQL Global Development Group * - * $PostgreSQL: pgsql/src/backend/postmaster/pgstat.c,v 1.123 2006/04/20 10:51:32 momjian Exp $ + * $PostgreSQL: pgsql/src/backend/postmaster/pgstat.c,v 1.124 2006/04/27 00:06:58 momjian Exp $ * ---------- */ #include "postgres.h" @@ -28,6 +28,7 @@ #include <arpa/inet.h> #include <signal.h> #include <time.h> +#include <sys/stat.h> #include "pgstat.h" @@ -66,12 +67,15 @@ * Timer definitions. * ---------- */ -#define PGSTAT_STAT_INTERVAL 500 /* How often to write the status file; - * in milliseconds. */ -#define PGSTAT_RESTART_INTERVAL 60 /* How often to attempt to restart a - * failed statistics collector; in - * seconds. */ +/* How often to write the status file, in milliseconds. */ +#define PGSTAT_STAT_INTERVAL (5*60*1000) + +/* + * How often to attempt to restart a failed statistics collector; in ms. + * Must be at least PGSTAT_STAT_INTERVAL. + */ +#define PGSTAT_RESTART_INTERVAL (5*60*1000) /* ---------- * Amount of space reserved in pgstat_recvbuffer(). @@ -172,11 +176,12 @@ static void pgstat_drop_database(Oid databaseid); static void pgstat_write_statsfile(void); static void pgstat_read_statsfile(HTAB **dbhash, Oid onlydb, PgStat_StatBeEntry **betab, - int *numbackends); + int *numbackends, bool rewrite); static void backend_read_statsfile(void); static void pgstat_setheader(PgStat_MsgHdr *hdr, StatMsgType mtype); static void pgstat_send(void *msg, int len); +static void pgstat_send_rewrite(void); static void pgstat_recv_bestart(PgStat_MsgBestart *msg, int len); static void pgstat_recv_beterm(PgStat_MsgBeterm *msg, int len); @@ -1449,6 +1454,24 @@ pgstat_send(void *msg, int len) #endif } +/* + * pgstat_send_rewrite() - + * + * Send a command to the collector to rewrite the stats file. + * ---------- + */ +static void +pgstat_send_rewrite(void) +{ + PgStat_MsgRewrite msg; + + if (pgStatSock < 0) + return; + + pgstat_setheader(&msg.m_hdr, PGSTAT_MTYPE_REWRITE); + pgstat_send(&msg, sizeof(msg)); +} + /* ---------- * PgstatBufferMain() - @@ -1549,7 +1572,7 @@ PgstatCollectorMain(int argc, char *argv[]) fd_set rfds; int readPipe; int len = 0; - struct itimerval timeout; + struct itimerval timeout, canceltimeout; bool need_timer = false; MyProcPid = getpid(); /* reset MyProcPid */ @@ -1604,12 +1627,15 @@ PgstatCollectorMain(int argc, char *argv[]) timeout.it_value.tv_sec = PGSTAT_STAT_INTERVAL / 1000; timeout.it_value.tv_usec = PGSTAT_STAT_INTERVAL % 1000; + /* Values set to zero will cancel the active timer */ + MemSet(&canceltimeout, 0, sizeof(struct itimerval)); + /* * Read in an existing statistics stats file or initialize the stats to * zero. */ pgStatRunningInCollector = true; - pgstat_read_statsfile(&pgStatDBHash, InvalidOid, NULL, NULL); + pgstat_read_statsfile(&pgStatDBHash, InvalidOid, NULL, NULL, false); /* * Create the known backends table @@ -1764,6 +1790,12 @@ PgstatCollectorMain(int argc, char *argv[]) pgstat_recv_analyze((PgStat_MsgAnalyze *) &msg, nread); break; + case PGSTAT_MTYPE_REWRITE: + need_statwrite = true; + /* Disable the timer - it will be restarted on next data update */ + setitimer(ITIMER_REAL, &canceltimeout, NULL); + break; + default: break; } @@ -2344,7 +2376,7 @@ comparePids(const void *v1, const void *v2) */ static void pgstat_read_statsfile(HTAB **dbhash, Oid onlydb, - PgStat_StatBeEntry **betab, int *numbackends) + PgStat_StatBeEntry **betab, int *numbackends, bool rewrite) { PgStat_StatDBEntry *dbentry; PgStat_StatDBEntry dbbuf; @@ -2363,6 +2395,71 @@ pgstat_read_statsfile(HTAB **dbhash, Oid onlydb, MemoryContext use_mcxt; int mcxt_flags; + + if (rewrite) + { + /* + * To force a rewrite of the stats file from the collector, send + * a REWRITE message to the stats collector. Then wait for the file + * to change. On Unix, we wait for the inode to change (as the file + * is renamed into place from a different file). Win32 has no concept + * of inodes, so we wait for the date on the file to change instead. + * We can do this on win32 because we have high-res timing on the + * file dates, but we can't on unix, because it has 1sec resolution + * on the fields in struct stat. + */ + int i; +#ifndef WIN32 + struct stat st1, st2; + + if (stat(PGSTAT_STAT_FILENAME, &st1)) + { + /* Assume no file there yet */ + st1.st_ino = 0; + } + st2.st_ino = 0; +#else + WIN32_FILE_ATTRIBUTE_DATA fd1, fd2; + + if (!GetFileAttributesEx(PGSTAT_STAT_FILENAME, GetFileExInfoStandard, &fd1)) + { + fd1.ftLastWriteTime.dwLowDateTime = 0; + fd1.ftLastWriteTime.dwHighDateTime = 0; + } + fd2.ftLastWriteTime.dwLowDateTime = 0; + fd2.ftLastWriteTime.dwHighDateTime = 0; +#endif + + + /* Send rewrite message */ + pgstat_send_rewrite(); + + /* Now wait for the file to change */ + for (i=0; i < 50; i++) + { +#ifndef WIN32 + if (!stat(PGSTAT_STAT_FILENAME, &st2)) + { + if (st2.st_ino != st1.st_ino) + break; + } +#else + if (GetFileAttributesEx(PGSTAT_STAT_FILENAME, GetFileExInfoStandard, &fd2)) + { + if (fd1.ftLastWriteTime.dwLowDateTime != fd2.ftLastWriteTime.dwLowDateTime || + fd1.ftLastWriteTime.dwHighDateTime != fd2.ftLastWriteTime.dwHighDateTime) + break; + } +#endif + + pg_usleep(50000); + } + if (i >= 50) + ereport(WARNING, + (errmsg("pgstat update timeout"))); + /* Fallthrough and read the old file anyway - old data better than no data */ + } + /* * If running in the collector or the autovacuum process, we use the * DynaHashCxt memory context. If running in a backend, we use the @@ -2681,7 +2778,7 @@ backend_read_statsfile(void) return; Assert(!pgStatRunningInCollector); pgstat_read_statsfile(&pgStatDBHash, InvalidOid, - &pgStatBeTable, &pgStatNumBackends); + &pgStatBeTable, &pgStatNumBackends, true); } else { @@ -2691,7 +2788,7 @@ backend_read_statsfile(void) { Assert(!pgStatRunningInCollector); pgstat_read_statsfile(&pgStatDBHash, MyDatabaseId, - &pgStatBeTable, &pgStatNumBackends); + &pgStatBeTable, &pgStatNumBackends, true); pgStatDBHashXact = topXid; } } diff --git a/src/include/pgstat.h b/src/include/pgstat.h index 00631b3e12b7b9ef0bde89e8d6b01505f2dfbc1e..4497e4d1f58fd80a18fd7d423bdbc57ff0dd1253 100644 --- a/src/include/pgstat.h +++ b/src/include/pgstat.h @@ -5,7 +5,7 @@ * * Copyright (c) 2001-2006, PostgreSQL Global Development Group * - * $PostgreSQL: pgsql/src/include/pgstat.h,v 1.43 2006/04/06 20:38:00 tgl Exp $ + * $PostgreSQL: pgsql/src/include/pgstat.h,v 1.44 2006/04/27 00:06:59 momjian Exp $ * ---------- */ #ifndef PGSTAT_H @@ -32,7 +32,8 @@ typedef enum StatMsgType PGSTAT_MTYPE_RESETCOUNTER, PGSTAT_MTYPE_AUTOVAC_START, PGSTAT_MTYPE_VACUUM, - PGSTAT_MTYPE_ANALYZE + PGSTAT_MTYPE_ANALYZE, + PGSTAT_MTYPE_REWRITE } StatMsgType; /* ---------- @@ -107,6 +108,15 @@ typedef struct PgStat_MsgDummy char m_dummy[512]; } PgStat_MsgDummy; +/* ---------- + * PgStat_MsgRewrite Sent by backends to cause a rewrite of the stats file + * ---------- + */ +typedef struct Pgstat_MsgRewrite +{ + PgStat_MsgHdr m_hdr; +} PgStat_MsgRewrite; + /* ---------- * PgStat_MsgBestart Sent by the backend on startup * ----------