PostgreSQL Source Code git master
pgstat.c File Reference
#include "postgres.h"
#include <unistd.h>
#include "access/xact.h"
#include "access/xlog.h"
#include "lib/dshash.h"
#include "pgstat.h"
#include "storage/fd.h"
#include "storage/ipc.h"
#include "storage/lwlock.h"
#include "utils/guc_hooks.h"
#include "utils/memutils.h"
#include "utils/pgstat_internal.h"
#include "utils/timestamp.h"
#include "lib/simplehash.h"
Include dependency graph for pgstat.c:

Go to the source code of this file.

Data Structures

struct  PgStat_SnapshotEntry
 

Macros

#define PGSTAT_MIN_INTERVAL   1000
 
#define PGSTAT_MAX_INTERVAL   60000
 
#define PGSTAT_IDLE_INTERVAL   10000
 
#define PGSTAT_SNAPSHOT_HASH_SIZE   512
 
#define PGSTAT_FILE_ENTRY_END   'E' /* end of file */
 
#define PGSTAT_FILE_ENTRY_FIXED   'F' /* fixed-numbered stats entry */
 
#define PGSTAT_FILE_ENTRY_NAME   'N' /* stats entry identified by name */
 
#define PGSTAT_FILE_ENTRY_HASH
 
#define SH_PREFIX   pgstat_snapshot
 
#define SH_ELEMENT_TYPE   PgStat_SnapshotEntry
 
#define SH_KEY_TYPE   PgStat_HashKey
 
#define SH_KEY   key
 
#define SH_HASH_KEY(tb, key)    pgstat_hash_hash_key(&key, sizeof(PgStat_HashKey), NULL)
 
#define SH_EQUAL(tb, a, b)    pgstat_cmp_hash_key(&a, &b, sizeof(PgStat_HashKey), NULL) == 0
 
#define SH_SCOPE   static inline
 
#define SH_DEFINE
 
#define SH_DECLARE
 
#define write_chunk_s(fpout, ptr)   write_chunk(fpout, ptr, sizeof(*ptr))
 
#define read_chunk_s(fpin, ptr)   read_chunk(fpin, ptr, sizeof(*ptr))
 

Typedefs

typedef struct PgStat_SnapshotEntry PgStat_SnapshotEntry
 

Functions

static void pgstat_write_statsfile (XLogRecPtr redo)
 
static void pgstat_read_statsfile (XLogRecPtr redo)
 
static void pgstat_init_snapshot_fixed (void)
 
static void pgstat_reset_after_failure (void)
 
static bool pgstat_flush_pending_entries (bool nowait)
 
static void pgstat_prep_snapshot (void)
 
static void pgstat_build_snapshot (void)
 
static void pgstat_build_snapshot_fixed (PgStat_Kind kind)
 
static bool pgstat_is_kind_valid (PgStat_Kind kind)
 
void pgstat_restore_stats (XLogRecPtr redo)
 
void pgstat_discard_stats (void)
 
void pgstat_before_server_shutdown (int code, Datum arg)
 
static void pgstat_shutdown_hook (int code, Datum arg)
 
void pgstat_initialize (void)
 
long pgstat_report_stat (bool force)
 
void pgstat_force_next_flush (void)
 
static bool match_db_entries (PgStatShared_HashEntry *entry, Datum match_data)
 
void pgstat_reset_counters (void)
 
void pgstat_reset (PgStat_Kind kind, Oid dboid, uint64 objid)
 
void pgstat_reset_of_kind (PgStat_Kind kind)
 
void pgstat_clear_snapshot (void)
 
void * pgstat_fetch_entry (PgStat_Kind kind, Oid dboid, uint64 objid)
 
TimestampTz pgstat_get_stat_snapshot_timestamp (bool *have_snapshot)
 
bool pgstat_have_entry (PgStat_Kind kind, Oid dboid, uint64 objid)
 
void pgstat_snapshot_fixed (PgStat_Kind kind)
 
PgStat_EntryRefpgstat_prep_pending_entry (PgStat_Kind kind, Oid dboid, uint64 objid, bool *created_entry)
 
PgStat_EntryRefpgstat_fetch_pending_entry (PgStat_Kind kind, Oid dboid, uint64 objid)
 
void pgstat_delete_pending_entry (PgStat_EntryRef *entry_ref)
 
PgStat_Kind pgstat_get_kind_from_str (char *kind_str)
 
const PgStat_KindInfopgstat_get_kind_info (PgStat_Kind kind)
 
void pgstat_register_kind (PgStat_Kind kind, const PgStat_KindInfo *kind_info)
 
static void write_chunk (FILE *fpout, void *ptr, size_t len)
 
static bool read_chunk (FILE *fpin, void *ptr, size_t len)
 
void assign_stats_fetch_consistency (int newval, void *extra)
 

Variables

bool pgstat_track_counts = false
 
int pgstat_fetch_consistency = PGSTAT_FETCH_CONSISTENCY_CACHE
 
PgStat_LocalState pgStatLocal
 
static MemoryContext pgStatPendingContext = NULL
 
static dlist_head pgStatPending = DLIST_STATIC_INIT(pgStatPending)
 
static bool pgStatForceNextFlush = false
 
static bool force_stats_snapshot_clear = false
 
static const PgStat_KindInfo pgstat_kind_builtin_infos [PGSTAT_KIND_BUILTIN_SIZE]
 
static const PgStat_KindInfo ** pgstat_kind_custom_infos = NULL
 

Macro Definition Documentation

◆ PGSTAT_FILE_ENTRY_END

#define PGSTAT_FILE_ENTRY_END   'E' /* end of file */

Definition at line 144 of file pgstat.c.

◆ PGSTAT_FILE_ENTRY_FIXED

#define PGSTAT_FILE_ENTRY_FIXED   'F' /* fixed-numbered stats entry */

Definition at line 145 of file pgstat.c.

◆ PGSTAT_FILE_ENTRY_HASH

#define PGSTAT_FILE_ENTRY_HASH
Value:
'S' /* stats entry identified by
* PgStat_HashKey */

Definition at line 147 of file pgstat.c.

◆ PGSTAT_FILE_ENTRY_NAME

#define PGSTAT_FILE_ENTRY_NAME   'N' /* stats entry identified by name */

Definition at line 146 of file pgstat.c.

◆ PGSTAT_IDLE_INTERVAL

#define PGSTAT_IDLE_INTERVAL   10000

Definition at line 131 of file pgstat.c.

◆ PGSTAT_MAX_INTERVAL

#define PGSTAT_MAX_INTERVAL   60000

Definition at line 129 of file pgstat.c.

◆ PGSTAT_MIN_INTERVAL

#define PGSTAT_MIN_INTERVAL   1000

Definition at line 127 of file pgstat.c.

◆ PGSTAT_SNAPSHOT_HASH_SIZE

#define PGSTAT_SNAPSHOT_HASH_SIZE   512

Definition at line 138 of file pgstat.c.

◆ read_chunk_s

#define read_chunk_s (   fpin,
  ptr 
)    read_chunk(fpin, ptr, sizeof(*ptr))

Definition at line 1766 of file pgstat.c.

◆ SH_DECLARE

#define SH_DECLARE

Definition at line 174 of file pgstat.c.

◆ SH_DEFINE

#define SH_DEFINE

Definition at line 173 of file pgstat.c.

◆ SH_ELEMENT_TYPE

#define SH_ELEMENT_TYPE   PgStat_SnapshotEntry

Definition at line 165 of file pgstat.c.

◆ SH_EQUAL

#define SH_EQUAL (   tb,
  a,
  b 
)     pgstat_cmp_hash_key(&a, &b, sizeof(PgStat_HashKey), NULL) == 0

Definition at line 170 of file pgstat.c.

◆ SH_HASH_KEY

#define SH_HASH_KEY (   tb,
  key 
)     pgstat_hash_hash_key(&key, sizeof(PgStat_HashKey), NULL)

Definition at line 168 of file pgstat.c.

◆ SH_KEY

#define SH_KEY   key

Definition at line 167 of file pgstat.c.

◆ SH_KEY_TYPE

#define SH_KEY_TYPE   PgStat_HashKey

Definition at line 166 of file pgstat.c.

◆ SH_PREFIX

#define SH_PREFIX   pgstat_snapshot

Definition at line 164 of file pgstat.c.

◆ SH_SCOPE

#define SH_SCOPE   static inline

Definition at line 172 of file pgstat.c.

◆ write_chunk_s

#define write_chunk_s (   fpout,
  ptr 
)    write_chunk(fpout, ptr, sizeof(*ptr))

Definition at line 1580 of file pgstat.c.

Typedef Documentation

◆ PgStat_SnapshotEntry

Function Documentation

◆ assign_stats_fetch_consistency()

void assign_stats_fetch_consistency ( int  newval,
void *  extra 
)

Definition at line 2087 of file pgstat.c.

2089{
2090 /*
2091 * Changing this value in a transaction may cause snapshot state
2092 * inconsistencies, so force a clear of the current snapshot on the next
2093 * snapshot build attempt.
2094 */
#define newval
int pgstat_fetch_consistency
Definition: pgstat.c:205
static bool force_stats_snapshot_clear
Definition: pgstat.c:251

References force_stats_snapshot_clear, newval, and pgstat_fetch_consistency.

◆ match_db_entries()

static bool match_db_entries ( PgStatShared_HashEntry entry,
Datum  match_data 
)
static

Definition at line 839 of file pgstat.c.

841{
842 return entry->key.dboid == DatumGetObjectId(MyDatabaseId);
Oid MyDatabaseId
Definition: globals.c:93
static Oid DatumGetObjectId(Datum X)
Definition: postgres.h:247

References DatumGetObjectId(), PgStat_HashKey::dboid, PgStatShared_HashEntry::key, and MyDatabaseId.

Referenced by pgstat_reset_counters().

◆ pgstat_before_server_shutdown()

void pgstat_before_server_shutdown ( int  code,
Datum  arg 
)

Definition at line 560 of file pgstat.c.

562{
563 Assert(pgStatLocal.shmem != NULL);
565
566 /*
567 * Stats should only be reported after pgstat_initialize() and before
568 * pgstat_shutdown(). This is a convenient point to catch most violations
569 * of this rule.
570 */
571 Assert(pgstat_is_initialized && !pgstat_is_shutdown);
572
573 /* flush out our own pending changes before writing out */
574 pgstat_report_stat(true);
575
576 /*
577 * Only write out file during normal shutdown. Don't even signal that
578 * we've shutdown during irregular shutdowns, because the shutdown
579 * sequence isn't coordinated to ensure this backend shuts down last.
580 */
581 if (code == 0)
582 {
585 }
#define Assert(condition)
Definition: c.h:815
long pgstat_report_stat(bool force)
Definition: pgstat.c:692
static void pgstat_write_statsfile(XLogRecPtr redo)
Definition: pgstat.c:1587
PgStat_LocalState pgStatLocal
Definition: pgstat.c:213
PgStat_ShmemControl * shmem
XLogRecPtr GetRedoRecPtr(void)
Definition: xlog.c:6458

References Assert, GetRedoRecPtr(), PgStat_ShmemControl::is_shutdown, pgstat_report_stat(), pgstat_write_statsfile(), pgStatLocal, and PgStat_LocalState::shmem.

Referenced by CheckpointerMain(), and InitPostgres().

◆ pgstat_build_snapshot()

static void pgstat_build_snapshot ( void  )
static

Definition at line 1139 of file pgstat.c.

1141{
1142 dshash_seq_status hstat;
1144
1145 /* should only be called when we need a snapshot */
1147
1148 /* snapshot already built */
1150 return;
1151
1153
1154 Assert(pgStatLocal.snapshot.stats->members == 0);
1155
1157
1158 /*
1159 * Snapshot all variable stats.
1160 */
1161 dshash_seq_init(&hstat, pgStatLocal.shared_hash, false);
1162 while ((p = dshash_seq_next(&hstat)) != NULL)
1163 {
1164 PgStat_Kind kind = p->key.kind;
1165 const PgStat_KindInfo *kind_info = pgstat_get_kind_info(kind);
1166 bool found;
1167 PgStat_SnapshotEntry *entry;
1168 PgStatShared_Common *stats_data;
1169
1170 /*
1171 * Check if the stats object should be included in the snapshot.
1172 * Unless the stats kind can be accessed from all databases (e.g.,
1173 * database stats themselves), we only include stats for the current
1174 * database or objects not associated with a database (e.g. shared
1175 * relations).
1176 */
1177 if (p->key.dboid != MyDatabaseId &&
1178 p->key.dboid != InvalidOid &&
1179 !kind_info->accessed_across_databases)
1180 continue;
1181
1182 if (p->dropped)
1183 continue;
1184
1186
1187 stats_data = dsa_get_address(pgStatLocal.dsa, p->body);
1188 Assert(stats_data);
1189
1190 entry = pgstat_snapshot_insert(pgStatLocal.snapshot.stats, p->key, &found);
1191 Assert(!found);
1192
1194 kind_info->shared_size);
1195
1196 /*
1197 * Acquire the LWLock directly instead of using
1198 * pg_stat_lock_entry_shared() which requires a reference.
1199 */
1200 LWLockAcquire(&stats_data->lock, LW_SHARED);
1201 memcpy(entry->data,
1202 pgstat_get_entry_data(kind, stats_data),
1203 kind_info->shared_size);
1204 LWLockRelease(&stats_data->lock);
1205 }
1206 dshash_seq_term(&hstat);
1207
1208 /*
1209 * Build snapshot of all fixed-numbered stats.
1210 */
1211 for (PgStat_Kind kind = PGSTAT_KIND_MIN; kind <= PGSTAT_KIND_MAX; kind++)
1212 {
1213 const PgStat_KindInfo *kind_info = pgstat_get_kind_info(kind);
1214
1215 if (!kind_info)
1216 continue;
1217 if (!kind_info->fixed_amount)
1218 {
1219 Assert(kind_info->snapshot_cb == NULL);
1220 continue;
1221 }
1222
1224 }
1225
static uint32 pg_atomic_read_u32(volatile pg_atomic_uint32 *ptr)
Definition: atomics.h:239
TimestampTz GetCurrentTimestamp(void)
Definition: timestamp.c:1644
void * dsa_get_address(dsa_area *area, dsa_pointer dp)
Definition: dsa.c:942
void dshash_seq_init(dshash_seq_status *status, dshash_table *hash_table, bool exclusive)
Definition: dshash.c:638
void dshash_seq_term(dshash_seq_status *status)
Definition: dshash.c:747
void * dshash_seq_next(dshash_seq_status *status)
Definition: dshash.c:657
bool LWLockAcquire(LWLock *lock, LWLockMode mode)
Definition: lwlock.c:1168
void LWLockRelease(LWLock *lock)
Definition: lwlock.c:1781
@ LW_SHARED
Definition: lwlock.h:115
void * MemoryContextAlloc(MemoryContext context, Size size)
Definition: mcxt.c:1181
static void pgstat_prep_snapshot(void)
Definition: pgstat.c:1118
static void pgstat_build_snapshot_fixed(PgStat_Kind kind)
Definition: pgstat.c:1229
const PgStat_KindInfo * pgstat_get_kind_info(PgStat_Kind kind)
Definition: pgstat.c:1454
@ PGSTAT_FETCH_CONSISTENCY_SNAPSHOT
Definition: pgstat.h:49
static void * pgstat_get_entry_data(PgStat_Kind kind, PgStatShared_Common *entry)
#define PgStat_Kind
Definition: pgstat_kind.h:17
#define PGSTAT_KIND_MAX
Definition: pgstat_kind.h:21
#define PGSTAT_KIND_MIN
Definition: pgstat_kind.h:20
#define InvalidOid
Definition: postgres_ext.h:37
pg_atomic_uint32 refcount
PgStat_Kind kind
bool accessed_across_databases
void(* snapshot_cb)(void)
PgStat_Snapshot snapshot
dshash_table * shared_hash
TimestampTz snapshot_timestamp
MemoryContext context
PgStat_FetchConsistency mode
struct pgstat_snapshot_hash * stats

References PgStat_KindInfo::accessed_across_databases, Assert, PgStatShared_HashEntry::body, PgStat_Snapshot::context, PgStat_SnapshotEntry::data, PgStat_HashKey::dboid, PgStatShared_HashEntry::dropped, PgStat_LocalState::dsa, dsa_get_address(), dshash_seq_init(), dshash_seq_next(), dshash_seq_term(), PgStat_KindInfo::fixed_amount, GetCurrentTimestamp(), InvalidOid, PgStatShared_HashEntry::key, PgStat_HashKey::kind, PgStatShared_Common::lock, LW_SHARED, LWLockAcquire(), LWLockRelease(), MemoryContextAlloc(), PgStat_Snapshot::mode, MyDatabaseId, pg_atomic_read_u32(), pgstat_build_snapshot_fixed(), pgstat_fetch_consistency, PGSTAT_FETCH_CONSISTENCY_SNAPSHOT, pgstat_get_entry_data(), pgstat_get_kind_info(), PgStat_Kind, PGSTAT_KIND_MAX, PGSTAT_KIND_MIN, pgstat_prep_snapshot(), pgStatLocal, PgStatShared_HashEntry::refcount, PgStat_LocalState::shared_hash, PgStat_KindInfo::shared_size, PgStat_LocalState::snapshot, PgStat_KindInfo::snapshot_cb, PgStat_Snapshot::snapshot_timestamp, and PgStat_Snapshot::stats.

Referenced by pgstat_fetch_entry(), and pgstat_snapshot_fixed().

◆ pgstat_build_snapshot_fixed()

static void pgstat_build_snapshot_fixed ( PgStat_Kind  kind)
static

Definition at line 1229 of file pgstat.c.

1231{
1232 const PgStat_KindInfo *kind_info = pgstat_get_kind_info(kind);
1233 int idx;
1234 bool *valid;
1235
1236 /* Position in fixed_valid or custom_valid */
1237 if (pgstat_is_kind_builtin(kind))
1238 {
1239 idx = kind;
1241 }
1242 else
1243 {
1244 idx = kind - PGSTAT_KIND_CUSTOM_MIN;
1246 }
1247
1248 Assert(kind_info->fixed_amount);
1249 Assert(kind_info->snapshot_cb != NULL);
1250
1252 {
1253 /* rebuild every time */
1254 valid[idx] = false;
1255 }
1256 else if (valid[idx])
1257 {
1258 /* in snapshot mode we shouldn't get called again */
1260 return;
1261 }
1262
1263 Assert(!valid[idx]);
1264
1265 kind_info->snapshot_cb();
1266
1267 Assert(!valid[idx]);
1268 valid[idx] = true;
Datum idx(PG_FUNCTION_ARGS)
Definition: _int_op.c:259
@ PGSTAT_FETCH_CONSISTENCY_NONE
Definition: pgstat.h:47
@ PGSTAT_FETCH_CONSISTENCY_CACHE
Definition: pgstat.h:48
static bool pgstat_is_kind_builtin(PgStat_Kind kind)
Definition: pgstat_kind.h:61
#define PGSTAT_KIND_CUSTOM_MIN
Definition: pgstat_kind.h:49
bool custom_valid[PGSTAT_KIND_CUSTOM_SIZE]
bool fixed_valid[PGSTAT_KIND_BUILTIN_SIZE]

References Assert, PgStat_Snapshot::custom_valid, PgStat_KindInfo::fixed_amount, PgStat_Snapshot::fixed_valid, idx(), pgstat_fetch_consistency, PGSTAT_FETCH_CONSISTENCY_CACHE, PGSTAT_FETCH_CONSISTENCY_NONE, pgstat_get_kind_info(), pgstat_is_kind_builtin(), PGSTAT_KIND_CUSTOM_MIN, pgStatLocal, PgStat_LocalState::snapshot, and PgStat_KindInfo::snapshot_cb.

Referenced by pgstat_build_snapshot(), pgstat_snapshot_fixed(), and pgstat_write_statsfile().

◆ pgstat_clear_snapshot()

void pgstat_clear_snapshot ( void  )

Definition at line 918 of file pgstat.c.

920{
922
929
930 /* Release memory, if any was allocated */
932 {
934
935 /* Reset variables */
937 }
938
939 /*
940 * Historically the backend_status.c facilities lived in this file, and
941 * were reset with the same function. For now keep it that way, and
942 * forward the reset request.
943 */
945
946 /* Reset this flag, as it may be possible that a cleanup was forced. */
void pgstat_clear_backend_activity_snapshot(void)
void MemoryContextDelete(MemoryContext context)
Definition: mcxt.c:454
#define pgstat_assert_is_up()

References PgStat_Snapshot::context, PgStat_Snapshot::custom_valid, PgStat_Snapshot::fixed_valid, force_stats_snapshot_clear, MemoryContextDelete(), PgStat_Snapshot::mode, pgstat_assert_is_up, pgstat_clear_backend_activity_snapshot(), PGSTAT_FETCH_CONSISTENCY_NONE, pgStatLocal, PgStat_LocalState::snapshot, and PgStat_Snapshot::stats.

Referenced by AtEOXact_PgStat(), pg_stat_clear_snapshot(), pgstat_get_stat_snapshot_timestamp(), pgstat_prep_snapshot(), pgstat_snapshot_fixed(), and PostPrepare_PgStat().

◆ pgstat_delete_pending_entry()

void pgstat_delete_pending_entry ( PgStat_EntryRef entry_ref)

Definition at line 1335 of file pgstat.c.

1337{
1338 PgStat_Kind kind = entry_ref->shared_entry->key.kind;
1339 const PgStat_KindInfo *kind_info = pgstat_get_kind_info(kind);
1340 void *pending_data = entry_ref->pending;
1341
1342 Assert(pending_data != NULL);
1343 /* !fixed_amount stats should be handled explicitly */
1344 Assert(!pgstat_get_kind_info(kind)->fixed_amount);
1345
1346 if (kind_info->delete_pending_cb)
1347 kind_info->delete_pending_cb(entry_ref);
1348
1349 pfree(pending_data);
1350 entry_ref->pending = NULL;
1351
1352 dlist_delete(&entry_ref->pending_node);
static void dlist_delete(dlist_node *node)
Definition: ilist.h:405
void pfree(void *pointer)
Definition: mcxt.c:1521
PgStatShared_HashEntry * shared_entry
dlist_node pending_node
void(* delete_pending_cb)(PgStat_EntryRef *sr)

References Assert, PgStat_KindInfo::delete_pending_cb, dlist_delete(), PgStatShared_HashEntry::key, PgStat_HashKey::kind, PgStat_EntryRef::pending, PgStat_EntryRef::pending_node, pfree(), pgstat_get_kind_info(), PgStat_Kind, and PgStat_EntryRef::shared_entry.

Referenced by pgstat_flush_pending_entries(), and pgstat_release_entry_ref().

◆ pgstat_discard_stats()

void pgstat_discard_stats ( void  )

Definition at line 517 of file pgstat.c.

519{
520 int ret;
521
522 /* NB: this needs to be done even in single user mode */
523
524 ret = unlink(PGSTAT_STAT_PERMANENT_FILENAME);
525 if (ret != 0)
526 {
527 if (errno == ENOENT)
528 elog(DEBUG2,
529 "didn't need to unlink permanent stats file \"%s\" - didn't exist",
531 else
532 ereport(LOG,
534 errmsg("could not unlink permanent statistics file \"%s\": %m",
536 }
537 else
538 {
541 errmsg_internal("unlinked permanent statistics file \"%s\"",
543 }
544
545 /*
546 * Reset stats contents. This will set reset timestamps of fixed-numbered
547 * stats to the current time (no variable stats exist).
548 */
int errmsg_internal(const char *fmt,...)
Definition: elog.c:1157
int errcode_for_file_access(void)
Definition: elog.c:876
int errmsg(const char *fmt,...)
Definition: elog.c:1070
#define LOG
Definition: elog.h:31
#define DEBUG2
Definition: elog.h:29
#define elog(elevel,...)
Definition: elog.h:225
#define ereport(elevel,...)
Definition: elog.h:149
static void pgstat_reset_after_failure(void)
Definition: pgstat.c:2064
#define PGSTAT_STAT_PERMANENT_FILENAME
Definition: pgstat.h:31

References DEBUG2, elog, ereport, errcode_for_file_access(), errmsg(), errmsg_internal(), LOG, pgstat_reset_after_failure(), and PGSTAT_STAT_PERMANENT_FILENAME.

Referenced by StartupXLOG().

◆ pgstat_fetch_entry()

void * pgstat_fetch_entry ( PgStat_Kind  kind,
Oid  dboid,
uint64  objid 
)

Definition at line 950 of file pgstat.c.

952{
954 PgStat_EntryRef *entry_ref;
955 void *stats_data;
956 const PgStat_KindInfo *kind_info = pgstat_get_kind_info(kind);
957
958 /* should be called from backends */
960 Assert(!kind_info->fixed_amount);
961
963
964 /* clear padding */
965 memset(&key, 0, sizeof(struct PgStat_HashKey));
966
967 key.kind = kind;
968 key.dboid = dboid;
969 key.objid = objid;
970
971 /* if we need to build a full snapshot, do so */
974
975 /* if caching is desired, look up in cache */
977 {
978 PgStat_SnapshotEntry *entry = NULL;
979
980 entry = pgstat_snapshot_lookup(pgStatLocal.snapshot.stats, key);
981
982 if (entry)
983 return entry->data;
984
985 /*
986 * If we built a full snapshot and the key is not in
987 * pgStatLocal.snapshot.stats, there are no matching stats.
988 */
990 return NULL;
991 }
992
994
995 entry_ref = pgstat_get_entry_ref(kind, dboid, objid, false, NULL);
996
997 if (entry_ref == NULL || entry_ref->shared_entry->dropped)
998 {
999 /* create empty entry when using PGSTAT_FETCH_CONSISTENCY_CACHE */
1001 {
1002 PgStat_SnapshotEntry *entry = NULL;
1003 bool found;
1004
1005 entry = pgstat_snapshot_insert(pgStatLocal.snapshot.stats, key, &found);
1006 Assert(!found);
1007 entry->data = NULL;
1008 }
1009 return NULL;
1010 }
1011
1012 /*
1013 * Allocate in caller's context for PGSTAT_FETCH_CONSISTENCY_NONE,
1014 * otherwise we could quickly end up with a fair bit of memory used due to
1015 * repeated accesses.
1016 */
1018 stats_data = palloc(kind_info->shared_data_len);
1019 else
1021 kind_info->shared_data_len);
1022
1023 pgstat_lock_entry_shared(entry_ref, false);
1024 memcpy(stats_data,
1025 pgstat_get_entry_data(kind, entry_ref->shared_stats),
1026 kind_info->shared_data_len);
1027 pgstat_unlock_entry(entry_ref);
1028
1030 {
1031 PgStat_SnapshotEntry *entry = NULL;
1032 bool found;
1033
1034 entry = pgstat_snapshot_insert(pgStatLocal.snapshot.stats, key, &found);
1035 entry->data = stats_data;
1036 }
1037
1038 return stats_data;
bool IsUnderPostmaster
Definition: globals.c:119
bool IsPostmasterEnvironment
Definition: globals.c:118
void * palloc(Size size)
Definition: mcxt.c:1317
static void pgstat_build_snapshot(void)
Definition: pgstat.c:1139
PgStat_EntryRef * pgstat_get_entry_ref(PgStat_Kind kind, Oid dboid, uint64 objid, bool create, bool *created_entry)
Definition: pgstat_shmem.c:444
bool pgstat_lock_entry_shared(PgStat_EntryRef *entry_ref, bool nowait)
Definition: pgstat_shmem.c:663
void pgstat_unlock_entry(PgStat_EntryRef *entry_ref)
Definition: pgstat_shmem.c:675
PgStatShared_Common * shared_stats

References Assert, PgStat_Snapshot::context, PgStat_SnapshotEntry::data, PgStatShared_HashEntry::dropped, PgStat_KindInfo::fixed_amount, IsPostmasterEnvironment, IsUnderPostmaster, sort-test::key, MemoryContextAlloc(), PgStat_Snapshot::mode, palloc(), pgstat_build_snapshot(), pgstat_fetch_consistency, PGSTAT_FETCH_CONSISTENCY_CACHE, PGSTAT_FETCH_CONSISTENCY_NONE, PGSTAT_FETCH_CONSISTENCY_SNAPSHOT, pgstat_get_entry_data(), pgstat_get_entry_ref(), pgstat_get_kind_info(), pgstat_lock_entry_shared(), pgstat_prep_snapshot(), pgstat_unlock_entry(), pgStatLocal, PgStat_KindInfo::shared_data_len, PgStat_EntryRef::shared_entry, PgStat_EntryRef::shared_stats, PgStat_LocalState::snapshot, and PgStat_Snapshot::stats.

Referenced by pgstat_fetch_replslot(), pgstat_fetch_stat_backend(), pgstat_fetch_stat_dbentry(), pgstat_fetch_stat_funcentry(), pgstat_fetch_stat_injentry(), pgstat_fetch_stat_subscription(), and pgstat_fetch_stat_tabentry_ext().

◆ pgstat_fetch_pending_entry()

PgStat_EntryRef * pgstat_fetch_pending_entry ( PgStat_Kind  kind,
Oid  dboid,
uint64  objid 
)

Definition at line 1322 of file pgstat.c.

1324{
1325 PgStat_EntryRef *entry_ref;
1326
1327 entry_ref = pgstat_get_entry_ref(kind, dboid, objid, false, NULL);
1328
1329 if (entry_ref == NULL || entry_ref->pending == NULL)
1330 return NULL;
1331
1332 return entry_ref;

References PgStat_EntryRef::pending, and pgstat_get_entry_ref().

Referenced by find_funcstat_entry(), and find_tabstat_entry().

◆ pgstat_flush_pending_entries()

static bool pgstat_flush_pending_entries ( bool  nowait)
static

Definition at line 1358 of file pgstat.c.

1360{
1361 bool have_pending = false;
1362 dlist_node *cur = NULL;
1363
1364 /*
1365 * Need to be a bit careful iterating over the list of pending entries.
1366 * Processing a pending entry may queue further pending entries to the end
1367 * of the list that we want to process, so a simple iteration won't do.
1368 * Further complicating matters is that we want to delete the current
1369 * entry in each iteration from the list if we flushed successfully.
1370 *
1371 * So we just keep track of the next pointer in each loop iteration.
1372 */
1375
1376 while (cur)
1377 {
1378 PgStat_EntryRef *entry_ref =
1379 dlist_container(PgStat_EntryRef, pending_node, cur);
1380 PgStat_HashKey key = entry_ref->shared_entry->key;
1381 PgStat_Kind kind = key.kind;
1382 const PgStat_KindInfo *kind_info = pgstat_get_kind_info(kind);
1383 bool did_flush;
1385
1386 Assert(!kind_info->fixed_amount);
1387 Assert(kind_info->flush_pending_cb != NULL);
1388
1389 /* flush the stats, if possible */
1390 did_flush = kind_info->flush_pending_cb(entry_ref, nowait);
1391
1392 Assert(did_flush || nowait);
1393
1394 /* determine next entry, before deleting the pending entry */
1397 else
1398 next = NULL;
1399
1400 /* if successfully flushed, remove entry */
1401 if (did_flush)
1402 pgstat_delete_pending_entry(entry_ref);
1403 else
1404 have_pending = true;
1405
1406 cur = next;
1407 }
1408
1409 Assert(dlist_is_empty(&pgStatPending) == !have_pending);
1410
1411 return have_pending;
static int32 next
Definition: blutils.c:221
struct cursor * cur
Definition: ecpg.c:29
static bool dlist_has_next(const dlist_head *head, const dlist_node *node)
Definition: ilist.h:503
static dlist_node * dlist_next_node(dlist_head *head, dlist_node *node)
Definition: ilist.h:537
static dlist_node * dlist_head_node(dlist_head *head)
Definition: ilist.h:565
static bool dlist_is_empty(const dlist_head *head)
Definition: ilist.h:336
#define dlist_container(type, membername, ptr)
Definition: ilist.h:593
void pgstat_delete_pending_entry(PgStat_EntryRef *entry_ref)
Definition: pgstat.c:1335
static dlist_head pgStatPending
Definition: pgstat.c:238
bool(* flush_pending_cb)(PgStat_EntryRef *sr, bool nowait)

References Assert, cur, dlist_container, dlist_has_next(), dlist_head_node(), dlist_is_empty(), dlist_next_node(), PgStat_KindInfo::fixed_amount, PgStat_KindInfo::flush_pending_cb, PgStatShared_HashEntry::key, sort-test::key, next, pgstat_delete_pending_entry(), pgstat_get_kind_info(), PgStat_Kind, pgStatPending, and PgStat_EntryRef::shared_entry.

Referenced by pgstat_report_stat().

◆ pgstat_force_next_flush()

void pgstat_force_next_flush ( void  )

Definition at line 830 of file pgstat.c.

832{
static bool pgStatForceNextFlush
Definition: pgstat.c:245

References pgStatForceNextFlush.

Referenced by pg_stat_force_next_flush().

◆ pgstat_get_kind_from_str()

PgStat_Kind pgstat_get_kind_from_str ( char *  kind_str)

Definition at line 1420 of file pgstat.c.

1422{
1423 for (PgStat_Kind kind = PGSTAT_KIND_BUILTIN_MIN; kind <= PGSTAT_KIND_BUILTIN_MAX; kind++)
1424 {
1425 if (pg_strcasecmp(kind_str, pgstat_kind_builtin_infos[kind].name) == 0)
1426 return kind;
1427 }
1428
1429 /* Check the custom set of cumulative stats */
1431 {
1432 for (PgStat_Kind kind = PGSTAT_KIND_CUSTOM_MIN; kind <= PGSTAT_KIND_CUSTOM_MAX; kind++)
1433 {
1435
1438 return kind;
1439 }
1440 }
1441
1442 ereport(ERROR,
1443 (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
1444 errmsg("invalid statistics kind: \"%s\"", kind_str)));
1445 return PGSTAT_KIND_INVALID; /* avoid compiler warnings */
uint32_t uint32
Definition: c.h:488
int errcode(int sqlerrcode)
Definition: elog.c:853
#define ERROR
Definition: elog.h:39
static const PgStat_KindInfo ** pgstat_kind_custom_infos
Definition: pgstat.c:492
static const PgStat_KindInfo pgstat_kind_builtin_infos[PGSTAT_KIND_BUILTIN_SIZE]
Definition: pgstat.c:277
#define PGSTAT_KIND_CUSTOM_MAX
Definition: pgstat_kind.h:50
#define PGSTAT_KIND_BUILTIN_MAX
Definition: pgstat_kind.h:43
#define PGSTAT_KIND_INVALID
Definition: pgstat_kind.h:24
#define PGSTAT_KIND_BUILTIN_MIN
Definition: pgstat_kind.h:42
int pg_strcasecmp(const char *s1, const char *s2)
Definition: pgstrcasecmp.c:36
const char * name

References ereport, errcode(), errmsg(), ERROR, idx(), name, pg_strcasecmp(), PgStat_Kind, pgstat_kind_builtin_infos, PGSTAT_KIND_BUILTIN_MAX, PGSTAT_KIND_BUILTIN_MIN, pgstat_kind_custom_infos, PGSTAT_KIND_CUSTOM_MAX, PGSTAT_KIND_CUSTOM_MIN, and PGSTAT_KIND_INVALID.

Referenced by pg_stat_have_stats().

◆ pgstat_get_kind_info()

◆ pgstat_get_stat_snapshot_timestamp()

TimestampTz pgstat_get_stat_snapshot_timestamp ( bool *  have_snapshot)

Definition at line 1046 of file pgstat.c.

1048{
1051
1053 {
1054 *have_snapshot = true;
1056 }
1057
1058 *have_snapshot = false;
1059
1060 return 0;
void pgstat_clear_snapshot(void)
Definition: pgstat.c:918

References force_stats_snapshot_clear, PgStat_Snapshot::mode, pgstat_clear_snapshot(), PGSTAT_FETCH_CONSISTENCY_SNAPSHOT, pgStatLocal, PgStat_LocalState::snapshot, and PgStat_Snapshot::snapshot_timestamp.

Referenced by pg_stat_get_snapshot_timestamp().

◆ pgstat_have_entry()

bool pgstat_have_entry ( PgStat_Kind  kind,
Oid  dboid,
uint64  objid 
)

Definition at line 1063 of file pgstat.c.

1065{
1066 /* fixed-numbered stats always exist */
1067 if (pgstat_get_kind_info(kind)->fixed_amount)
1068 return true;
1069
1070 return pgstat_get_entry_ref(kind, dboid, objid, false, NULL) != NULL;

References pgstat_get_entry_ref(), and pgstat_get_kind_info().

Referenced by pg_stat_have_stats().

◆ pgstat_init_snapshot_fixed()

static void pgstat_init_snapshot_fixed ( void  )
static

Definition at line 1099 of file pgstat.c.

1101{
1102 /*
1103 * Initialize fixed-numbered statistics data in snapshots, only for custom
1104 * stats kinds.
1105 */
1106 for (PgStat_Kind kind = PGSTAT_KIND_CUSTOM_MIN; kind <= PGSTAT_KIND_CUSTOM_MAX; kind++)
1107 {
1108 const PgStat_KindInfo *kind_info = pgstat_get_kind_info(kind);
1109
1110 if (!kind_info || !kind_info->fixed_amount)
1111 continue;
1112
1115 }
MemoryContext TopMemoryContext
Definition: mcxt.c:149
void * custom_data[PGSTAT_KIND_CUSTOM_SIZE]

References PgStat_Snapshot::custom_data, PgStat_KindInfo::fixed_amount, MemoryContextAlloc(), pgstat_get_kind_info(), PgStat_Kind, PGSTAT_KIND_CUSTOM_MAX, PGSTAT_KIND_CUSTOM_MIN, pgStatLocal, PgStat_KindInfo::shared_data_len, PgStat_LocalState::snapshot, and TopMemoryContext.

Referenced by pgstat_initialize().

◆ pgstat_initialize()

void pgstat_initialize ( void  )

Definition at line 639 of file pgstat.c.

641{
642 Assert(!pgstat_is_initialized);
643
645
647
648 /* Backend initialization callbacks */
649 for (PgStat_Kind kind = PGSTAT_KIND_MIN; kind <= PGSTAT_KIND_MAX; kind++)
650 {
651 const PgStat_KindInfo *kind_info = pgstat_get_kind_info(kind);
652
653 if (kind_info == NULL || kind_info->init_backend_cb == NULL)
654 continue;
655
656 kind_info->init_backend_cb();
657 }
658
659 /* Set up a process-exit hook to clean up */
661
662#ifdef USE_ASSERT_CHECKING
663 pgstat_is_initialized = true;
664#endif
void before_shmem_exit(pg_on_exit_callback function, Datum arg)
Definition: ipc.c:337
static void pgstat_init_snapshot_fixed(void)
Definition: pgstat.c:1099
static void pgstat_shutdown_hook(int code, Datum arg)
Definition: pgstat.c:601
void pgstat_attach_shmem(void)
Definition: pgstat_shmem.c:244
void(* init_backend_cb)(void)

References Assert, before_shmem_exit(), PgStat_KindInfo::init_backend_cb, pgstat_attach_shmem(), pgstat_get_kind_info(), pgstat_init_snapshot_fixed(), PgStat_Kind, PGSTAT_KIND_MAX, PGSTAT_KIND_MIN, and pgstat_shutdown_hook().

Referenced by BaseInit().

◆ pgstat_is_kind_valid()

static bool pgstat_is_kind_valid ( PgStat_Kind  kind)
inlinestatic

◆ pgstat_prep_pending_entry()

PgStat_EntryRef * pgstat_prep_pending_entry ( PgStat_Kind  kind,
Oid  dboid,
uint64  objid,
bool *  created_entry 
)

Definition at line 1284 of file pgstat.c.

1286{
1287 PgStat_EntryRef *entry_ref;
1288
1289 /* need to be able to flush out */
1290 Assert(pgstat_get_kind_info(kind)->flush_pending_cb != NULL);
1291
1293 {
1296 "PgStat Pending",
1298 }
1299
1300 entry_ref = pgstat_get_entry_ref(kind, dboid, objid,
1301 true, created_entry);
1302
1303 if (entry_ref->pending == NULL)
1304 {
1305 size_t entrysize = pgstat_get_kind_info(kind)->pending_size;
1306
1307 Assert(entrysize != (size_t) -1);
1308
1309 entry_ref->pending = MemoryContextAllocZero(pgStatPendingContext, entrysize);
1311 }
1312
1313 return entry_ref;
#define unlikely(x)
Definition: c.h:333
static void dlist_push_tail(dlist_head *head, dlist_node *node)
Definition: ilist.h:364
void * MemoryContextAllocZero(MemoryContext context, Size size)
Definition: mcxt.c:1215
#define AllocSetContextCreate
Definition: memutils.h:129
#define ALLOCSET_SMALL_SIZES
Definition: memutils.h:170
static MemoryContext pgStatPendingContext
Definition: pgstat.c:230

References ALLOCSET_SMALL_SIZES, AllocSetContextCreate, Assert, dlist_push_tail(), MemoryContextAllocZero(), PgStat_EntryRef::pending, PgStat_EntryRef::pending_node, PgStat_KindInfo::pending_size, pgstat_get_entry_ref(), pgstat_get_kind_info(), pgStatPending, pgStatPendingContext, TopMemoryContext, and unlikely.

Referenced by pgstat_create_inj(), pgstat_init_function_usage(), pgstat_prep_database_pending(), pgstat_prep_relation_pending(), pgstat_report_inj(), pgstat_report_subscription_conflict(), and pgstat_report_subscription_error().

◆ pgstat_prep_snapshot()

◆ pgstat_read_statsfile()

static void pgstat_read_statsfile ( XLogRecPtr  redo)
static

Definition at line 1775 of file pgstat.c.

1777{
1778 FILE *fpin;
1779 int32 format_id;
1780 bool found;
1781 const char *statfile = PGSTAT_STAT_PERMANENT_FILENAME;
1783 XLogRecPtr file_redo;
1784
1785 /* shouldn't be called from postmaster */
1787
1788 elog(DEBUG2, "reading stats file \"%s\" with redo %X/%X", statfile,
1789 LSN_FORMAT_ARGS(redo));
1790
1791 /*
1792 * Try to open the stats file. If it doesn't exist, the backends simply
1793 * returns zero for anything and statistics simply starts from scratch
1794 * with empty counters.
1795 *
1796 * ENOENT is a possibility if stats collection was previously disabled or
1797 * has not yet written the stats file for the first time. Any other
1798 * failure condition is suspicious.
1799 */
1800 if ((fpin = AllocateFile(statfile, PG_BINARY_R)) == NULL)
1801 {
1802 if (errno != ENOENT)
1803 ereport(LOG,
1805 errmsg("could not open statistics file \"%s\": %m",
1806 statfile)));
1808 return;
1809 }
1810
1811 /*
1812 * Verify it's of the expected format.
1813 */
1814 if (!read_chunk_s(fpin, &format_id))
1815 {
1816 elog(WARNING, "could not read format ID");
1817 goto error;
1818 }
1819
1820 if (format_id != PGSTAT_FILE_FORMAT_ID)
1821 {
1822 elog(WARNING, "found incorrect format ID %d (expected %d)",
1823 format_id, PGSTAT_FILE_FORMAT_ID);
1824 goto error;
1825 }
1826
1827 /*
1828 * Read the redo LSN stored in the file.
1829 */
1830 if (!read_chunk_s(fpin, &file_redo))
1831 {
1832 elog(WARNING, "could not read redo LSN");
1833 goto error;
1834 }
1835
1836 if (file_redo != redo)
1837 {
1838 elog(WARNING, "found incorrect redo LSN %X/%X (expected %X/%X)",
1839 LSN_FORMAT_ARGS(file_redo), LSN_FORMAT_ARGS(redo));
1840 goto error;
1841 }
1842
1843 /*
1844 * We found an existing statistics file. Read it and put all the stats
1845 * data into place.
1846 */
1847 for (;;)
1848 {
1849 int t = fgetc(fpin);
1850
1851 switch (t)
1852 {
1854 {
1855 PgStat_Kind kind;
1856 const PgStat_KindInfo *info;
1857 char *ptr;
1858
1859 /* entry for fixed-numbered stats */
1860 if (!read_chunk_s(fpin, &kind))
1861 {
1862 elog(WARNING, "could not read stats kind for entry of type %c", t);
1863 goto error;
1864 }
1865
1866 if (!pgstat_is_kind_valid(kind))
1867 {
1868 elog(WARNING, "invalid stats kind %u for entry of type %c",
1869 kind, t);
1870 goto error;
1871 }
1872
1873 info = pgstat_get_kind_info(kind);
1874 if (!info)
1875 {
1876 elog(WARNING, "could not find information of kind %u for entry of type %c",
1877 kind, t);
1878 goto error;
1879 }
1880
1881 if (!info->fixed_amount)
1882 {
1883 elog(WARNING, "invalid fixed_amount in stats kind %u for entry of type %c",
1884 kind, t);
1885 goto error;
1886 }
1887
1888 /* Load back stats into shared memory */
1889 if (pgstat_is_kind_builtin(kind))
1890 ptr = ((char *) shmem) + info->shared_ctl_off +
1891 info->shared_data_off;
1892 else
1893 {
1894 int idx = kind - PGSTAT_KIND_CUSTOM_MIN;
1895
1896 ptr = ((char *) shmem->custom_data[idx]) +
1897 info->shared_data_off;
1898 }
1899
1900 if (!read_chunk(fpin, ptr, info->shared_data_len))
1901 {
1902 elog(WARNING, "could not read data of stats kind %u for entry of type %c with size %u",
1903 kind, t, info->shared_data_len);
1904 goto error;
1905 }
1906
1907 break;
1908 }
1911 {
1914 PgStatShared_Common *header;
1915
1917
1918 if (t == PGSTAT_FILE_ENTRY_HASH)
1919 {
1920 /* normal stats entry, identified by PgStat_HashKey */
1921 if (!read_chunk_s(fpin, &key))
1922 {
1923 elog(WARNING, "could not read key for entry of type %c", t);
1924 goto error;
1925 }
1926
1927 if (!pgstat_is_kind_valid(key.kind))
1928 {
1929 elog(WARNING, "invalid stats kind for entry %u/%u/%llu of type %c",
1930 key.kind, key.dboid,
1931 (unsigned long long) key.objid, t);
1932 goto error;
1933 }
1934 }
1935 else
1936 {
1937 /* stats entry identified by name on disk (e.g. slots) */
1938 const PgStat_KindInfo *kind_info = NULL;
1939 PgStat_Kind kind;
1940 NameData name;
1941
1942 if (!read_chunk_s(fpin, &kind))
1943 {
1944 elog(WARNING, "could not read stats kind for entry of type %c", t);
1945 goto error;
1946 }
1947 if (!read_chunk_s(fpin, &name))
1948 {
1949 elog(WARNING, "could not read name of stats kind %u for entry of type %c",
1950 kind, t);
1951 goto error;
1952 }
1953 if (!pgstat_is_kind_valid(kind))
1954 {
1955 elog(WARNING, "invalid stats kind %u for entry of type %c",
1956 kind, t);
1957 goto error;
1958 }
1959
1960 kind_info = pgstat_get_kind_info(kind);
1961 if (!kind_info)
1962 {
1963 elog(WARNING, "could not find information of kind %u for entry of type %c",
1964 kind, t);
1965 goto error;
1966 }
1967
1968 if (!kind_info->from_serialized_name)
1969 {
1970 elog(WARNING, "invalid from_serialized_name in stats kind %u for entry of type %c",
1971 kind, t);
1972 goto error;
1973 }
1974
1975 if (!kind_info->from_serialized_name(&name, &key))
1976 {
1977 /* skip over data for entry we don't care about */
1978 if (fseek(fpin, pgstat_get_entry_len(kind), SEEK_CUR) != 0)
1979 {
1980 elog(WARNING, "could not seek \"%s\" of stats kind %u for entry of type %c",
1981 NameStr(name), kind, t);
1982 goto error;
1983 }
1984
1985 continue;
1986 }
1987
1988 Assert(key.kind == kind);
1989 }
1990
1991 /*
1992 * This intentionally doesn't use pgstat_get_entry_ref() -
1993 * putting all stats into checkpointer's
1994 * pgStatEntryRefHash would be wasted effort and memory.
1995 */
1997
1998 /* don't allow duplicate entries */
1999 if (found)
2000 {
2002 elog(WARNING, "found duplicate stats entry %u/%u/%llu of type %c",
2003 key.kind, key.dboid,
2004 (unsigned long long) key.objid, t);
2005 goto error;
2006 }
2007
2008 header = pgstat_init_entry(key.kind, p);
2010
2011 if (!read_chunk(fpin,
2012 pgstat_get_entry_data(key.kind, header),
2014 {
2015 elog(WARNING, "could not read data for entry %u/%u/%llu of type %c",
2016 key.kind, key.dboid,
2017 (unsigned long long) key.objid, t);
2018 goto error;
2019 }
2020
2021 break;
2022 }
2024
2025 /*
2026 * check that PGSTAT_FILE_ENTRY_END actually signals end of
2027 * file
2028 */
2029 if (fgetc(fpin) != EOF)
2030 {
2031 elog(WARNING, "could not read end-of-file");
2032 goto error;
2033 }
2034
2035 goto done;
2036
2037 default:
2038 elog(WARNING, "could not read entry of type %c", t);
2039 goto error;
2040 }
2041 }
2042
2043done:
2044 FreeFile(fpin);
2045
2046 elog(DEBUG2, "removing permanent stats file \"%s\"", statfile);
2047 unlink(statfile);
2048
2049 return;
2050
2051error:
2052 ereport(LOG,
2053 (errmsg("corrupted statistics file \"%s\"", statfile)));
2054
2056
2057 goto done;
#define NameStr(name)
Definition: c.h:703
#define PG_BINARY_R
Definition: c.h:1232
int32_t int32
Definition: c.h:484
void dshash_release_lock(dshash_table *hash_table, void *entry)
Definition: dshash.c:558
void * dshash_find_or_insert(dshash_table *hash_table, const void *key, bool *found)
Definition: dshash.c:433
#define WARNING
Definition: elog.h:36
int FreeFile(FILE *file)
Definition: fd.c:2803
FILE * AllocateFile(const char *name, const char *mode)
Definition: fd.c:2605
#define CHECK_FOR_INTERRUPTS()
Definition: miscadmin.h:122
#define PGSTAT_FILE_ENTRY_END
Definition: pgstat.c:144
static bool pgstat_is_kind_valid(PgStat_Kind kind)
Definition: pgstat.c:1448
#define PGSTAT_FILE_ENTRY_HASH
Definition: pgstat.c:147
#define PGSTAT_FILE_ENTRY_FIXED
Definition: pgstat.c:145
static bool read_chunk(FILE *fpin, void *ptr, size_t len)
Definition: pgstat.c:1761
#define read_chunk_s(fpin, ptr)
Definition: pgstat.c:1766
#define PGSTAT_FILE_ENTRY_NAME
Definition: pgstat.c:146
#define PGSTAT_FILE_FORMAT_ID
Definition: pgstat.h:215
static size_t pgstat_get_entry_len(PgStat_Kind kind)
PgStatShared_Common * pgstat_init_entry(PgStat_Kind kind, PgStatShared_HashEntry *shhashent)
Definition: pgstat_shmem.c:293
static void error(void)
Definition: sql-dyntest.c:147
bool(* from_serialized_name)(const NameData *name, PgStat_HashKey *key)
void * custom_data[PGSTAT_KIND_CUSTOM_SIZE]
Definition: c.h:698
#define LSN_FORMAT_ARGS(lsn)
Definition: xlogdefs.h:43
uint64 XLogRecPtr
Definition: xlogdefs.h:21

References AllocateFile(), Assert, CHECK_FOR_INTERRUPTS, PgStat_ShmemControl::custom_data, DEBUG2, dshash_find_or_insert(), dshash_release_lock(), elog, ereport, errcode_for_file_access(), errmsg(), error(), PgStat_KindInfo::fixed_amount, FreeFile(), PgStat_KindInfo::from_serialized_name, idx(), IsPostmasterEnvironment, IsUnderPostmaster, sort-test::key, LOG, LSN_FORMAT_ARGS, name, NameStr, PG_BINARY_R, PGSTAT_FILE_ENTRY_END, PGSTAT_FILE_ENTRY_FIXED, PGSTAT_FILE_ENTRY_HASH, PGSTAT_FILE_ENTRY_NAME, PGSTAT_FILE_FORMAT_ID, pgstat_get_entry_data(), pgstat_get_entry_len(), pgstat_get_kind_info(), pgstat_init_entry(), pgstat_is_kind_builtin(), pgstat_is_kind_valid(), PgStat_Kind, PGSTAT_KIND_CUSTOM_MIN, pgstat_reset_after_failure(), PGSTAT_STAT_PERMANENT_FILENAME, pgStatLocal, read_chunk(), read_chunk_s, PgStat_KindInfo::shared_ctl_off, PgStat_KindInfo::shared_data_len, PgStat_KindInfo::shared_data_off, PgStat_LocalState::shared_hash, PgStat_LocalState::shmem, and WARNING.

Referenced by pgstat_restore_stats().

◆ pgstat_register_kind()

void pgstat_register_kind ( PgStat_Kind  kind,
const PgStat_KindInfo kind_info 
)

Definition at line 1482 of file pgstat.c.

1484{
1486
1487 if (kind_info->name == NULL || strlen(kind_info->name) == 0)
1488 ereport(ERROR,
1489 (errmsg("custom cumulative statistics name is invalid"),
1490 errhint("Provide a non-empty name for the custom cumulative statistics.")));
1491
1492 if (!pgstat_is_kind_custom(kind))
1493 ereport(ERROR, (errmsg("custom cumulative statistics ID %u is out of range", kind),
1494 errhint("Provide a custom cumulative statistics ID between %u and %u.",
1496
1498 ereport(ERROR,
1499 (errmsg("failed to register custom cumulative statistics \"%s\" with ID %u", kind_info->name, kind),
1500 errdetail("Custom cumulative statistics must be registered while initializing modules in \"shared_preload_libraries\".")));
1501
1502 /*
1503 * Check some data for fixed-numbered stats.
1504 */
1505 if (kind_info->fixed_amount)
1506 {
1507 if (kind_info->shared_size == 0)
1508 ereport(ERROR,
1509 (errmsg("custom cumulative statistics property is invalid"),
1510 errhint("Custom cumulative statistics require a shared memory size for fixed-numbered objects.")));
1511 }
1512
1513 /*
1514 * If pgstat_kind_custom_infos is not available yet, allocate it.
1515 */
1516 if (pgstat_kind_custom_infos == NULL)
1517 {
1521 }
1522
1523 if (pgstat_kind_custom_infos[idx] != NULL &&
1525 ereport(ERROR,
1526 (errmsg("failed to register custom cumulative statistics \"%s\" with ID %u", kind_info->name, kind),
1527 errdetail("Custom cumulative statistics \"%s\" already registered with the same ID.",
1529
1530 /* check for existing custom stats with the same name */
1531 for (PgStat_Kind existing_kind = PGSTAT_KIND_CUSTOM_MIN; existing_kind <= PGSTAT_KIND_CUSTOM_MAX; existing_kind++)
1532 {
1533 uint32 existing_idx = existing_kind - PGSTAT_KIND_CUSTOM_MIN;
1534
1535 if (pgstat_kind_custom_infos[existing_idx] == NULL)
1536 continue;
1537 if (!pg_strcasecmp(pgstat_kind_custom_infos[existing_idx]->name, kind_info->name))
1538 ereport(ERROR,
1539 (errmsg("failed to register custom cumulative statistics \"%s\" with ID %u", kind_info->name, kind),
1540 errdetail("Existing cumulative statistics with ID %u has the same name.", existing_kind)));
1541 }
1542
1543 /* Register it */
1544 pgstat_kind_custom_infos[idx] = kind_info;
1545 ereport(LOG,
1546 (errmsg("registered custom cumulative statistics \"%s\" with ID %u",
1547 kind_info->name, kind)));
int errdetail(const char *fmt,...)
Definition: elog.c:1203
int errhint(const char *fmt,...)
Definition: elog.c:1317
bool process_shared_preload_libraries_in_progress
Definition: miscinit.c:1834
#define PGSTAT_KIND_CUSTOM_SIZE
Definition: pgstat_kind.h:51
const char *const name

References ereport, errdetail(), errhint(), errmsg(), ERROR, PgStat_KindInfo::fixed_amount, idx(), LOG, MemoryContextAllocZero(), name, PgStat_KindInfo::name, pg_strcasecmp(), pgstat_is_kind_custom(), PgStat_Kind, pgstat_kind_custom_infos, PGSTAT_KIND_CUSTOM_MAX, PGSTAT_KIND_CUSTOM_MIN, PGSTAT_KIND_CUSTOM_SIZE, process_shared_preload_libraries_in_progress, PgStat_KindInfo::shared_size, and TopMemoryContext.

Referenced by pgstat_register_inj(), and pgstat_register_inj_fixed().

◆ pgstat_report_stat()

long pgstat_report_stat ( bool  force)

Definition at line 692 of file pgstat.c.

694{
695 static TimestampTz pending_since = 0;
696 static TimestampTz last_flush = 0;
697 bool partial_flush;
699 bool nowait;
700
703
704 /* "absorb" the forced flush even if there's nothing to flush */
706 {
707 force = true;
708 pgStatForceNextFlush = false;
709 }
710
711 /* Don't expend a clock check if nothing to do */
713 {
714 bool do_flush = false;
715
716 /* Check for pending stats */
717 for (PgStat_Kind kind = PGSTAT_KIND_MIN; kind <= PGSTAT_KIND_MAX; kind++)
718 {
719 const PgStat_KindInfo *kind_info = pgstat_get_kind_info(kind);
720
721 if (!kind_info)
722 continue;
723 if (!kind_info->have_static_pending_cb)
724 continue;
725
726 if (kind_info->have_static_pending_cb())
727 {
728 do_flush = true;
729 break;
730 }
731 }
732
733 if (!do_flush)
734 {
735 Assert(pending_since == 0);
736 return 0;
737 }
738 }
739
740 /*
741 * There should never be stats to report once stats are shut down. Can't
742 * assert that before the checks above, as there is an unconditional
743 * pgstat_report_stat() call in pgstat_shutdown_hook() - which at least
744 * the process that ran pgstat_before_server_shutdown() will still call.
745 */
747
748 if (force)
749 {
750 /*
751 * Stats reports are forced either when it's been too long since stats
752 * have been reported or in processes that force stats reporting to
753 * happen at specific points (including shutdown). In the former case
754 * the transaction stop time might be quite old, in the latter it
755 * would never get cleared.
756 */
758 }
759 else
760 {
762
763 if (pending_since > 0 &&
765 {
766 /* don't keep pending updates longer than PGSTAT_MAX_INTERVAL */
767 force = true;
768 }
769 else if (last_flush > 0 &&
771 {
772 /* don't flush too frequently */
773 if (pending_since == 0)
774 pending_since = now;
775
777 }
778 }
779
781
782 /* don't wait for lock acquisition when !force */
783 nowait = !force;
784
785 partial_flush = false;
786
787 /* flush of variable-numbered stats tracked in pending entries list */
788 partial_flush |= pgstat_flush_pending_entries(nowait);
789
790 /* flush of other stats kinds */
791 for (PgStat_Kind kind = PGSTAT_KIND_MIN; kind <= PGSTAT_KIND_MAX; kind++)
792 {
793 const PgStat_KindInfo *kind_info = pgstat_get_kind_info(kind);
794
795 if (!kind_info)
796 continue;
797 if (!kind_info->flush_static_cb)
798 continue;
799
800 partial_flush |= kind_info->flush_static_cb(nowait);
801 }
802
803 last_flush = now;
804
805 /*
806 * If some of the pending stats could not be flushed due to lock
807 * contention, let the caller know when to retry.
808 */
809 if (partial_flush)
810 {
811 /* force should have prevented us from getting here */
812 Assert(!force);
813
814 /* remember since when stats have been pending */
815 if (pending_since == 0)
816 pending_since = now;
817
819 }
820
821 pending_since = 0;
822
823 return 0;
bool TimestampDifferenceExceeds(TimestampTz start_time, TimestampTz stop_time, int msec)
Definition: timestamp.c:1780
Datum now(PG_FUNCTION_ARGS)
Definition: timestamp.c:1608
int64 TimestampTz
Definition: timestamp.h:39
#define PGSTAT_MIN_INTERVAL
Definition: pgstat.c:127
static bool pgstat_flush_pending_entries(bool nowait)
Definition: pgstat.c:1358
#define PGSTAT_MAX_INTERVAL
Definition: pgstat.c:129
#define PGSTAT_IDLE_INTERVAL
Definition: pgstat.c:131
void pgstat_update_dbstats(TimestampTz ts)
bool(* flush_static_cb)(bool nowait)
bool(* have_static_pending_cb)(void)
bool IsTransactionOrTransactionBlock(void)
Definition: xact.c:4981
TimestampTz GetCurrentTransactionStopTimestamp(void)
Definition: xact.c:890

References Assert, dlist_is_empty(), PgStat_KindInfo::flush_static_cb, GetCurrentTimestamp(), GetCurrentTransactionStopTimestamp(), PgStat_KindInfo::have_static_pending_cb, PgStat_ShmemControl::is_shutdown, IsTransactionOrTransactionBlock(), now(), pgstat_assert_is_up, pgstat_flush_pending_entries(), pgstat_get_kind_info(), PGSTAT_IDLE_INTERVAL, PgStat_Kind, PGSTAT_KIND_MAX, PGSTAT_KIND_MIN, PGSTAT_MAX_INTERVAL, PGSTAT_MIN_INTERVAL, pgstat_update_dbstats(), pgStatForceNextFlush, pgStatLocal, pgStatPending, PgStat_LocalState::shmem, and TimestampDifferenceExceeds().

Referenced by AllTablesyncsReady(), apply_handle_commit_internal(), apply_handle_commit_prepared(), apply_handle_prepare(), apply_handle_rollback_prepared(), apply_handle_stream_prepare(), LogicalRepApplyLoop(), LogicalRepSyncTableStart(), pg_attribute_noreturn(), pgstat_before_server_shutdown(), pgstat_shutdown_hook(), PostgresMain(), process_syncing_tables_for_apply(), process_syncing_tables_for_sync(), ProcessInterrupts(), standby_redo(), and worker_spi_main().

◆ pgstat_reset()

void pgstat_reset ( PgStat_Kind  kind,
Oid  dboid,
uint64  objid 
)

Definition at line 870 of file pgstat.c.

872{
873 const PgStat_KindInfo *kind_info = pgstat_get_kind_info(kind);
875
876 /* not needed atm, and doesn't make sense with the current signature */
877 Assert(!pgstat_get_kind_info(kind)->fixed_amount);
878
879 /* reset the "single counter" */
880 pgstat_reset_entry(kind, dboid, objid, ts);
881
882 if (!kind_info->accessed_across_databases)
void pgstat_reset_database_timestamp(Oid dboid, TimestampTz ts)
void pgstat_reset_entry(PgStat_Kind kind, Oid dboid, uint64 objid, TimestampTz ts)

References PgStat_KindInfo::accessed_across_databases, Assert, GetCurrentTimestamp(), pgstat_get_kind_info(), pgstat_reset_database_timestamp(), and pgstat_reset_entry().

Referenced by pg_stat_reset_backend_stats(), pg_stat_reset_single_function_counters(), pg_stat_reset_single_table_counters(), pg_stat_reset_subscription_stats(), pgstat_create_transactional(), and pgstat_reset_replslot().

◆ pgstat_reset_after_failure()

static void pgstat_reset_after_failure ( void  )
static

Definition at line 2064 of file pgstat.c.

2066{
2068
2069 /* reset fixed-numbered stats */
2070 for (PgStat_Kind kind = PGSTAT_KIND_MIN; kind <= PGSTAT_KIND_MAX; kind++)
2071 {
2072 const PgStat_KindInfo *kind_info = pgstat_get_kind_info(kind);
2073
2074 if (!kind_info || !kind_info->fixed_amount)
2075 continue;
2076
2077 kind_info->reset_all_cb(ts);
2078 }
2079
2080 /* and drop variable-numbered ones */
void pgstat_drop_all_entries(void)
void(* reset_all_cb)(TimestampTz ts)

References PgStat_KindInfo::fixed_amount, GetCurrentTimestamp(), pgstat_drop_all_entries(), pgstat_get_kind_info(), PgStat_Kind, PGSTAT_KIND_MAX, PGSTAT_KIND_MIN, and PgStat_KindInfo::reset_all_cb.

Referenced by pgstat_discard_stats(), and pgstat_read_statsfile().

◆ pgstat_reset_counters()

void pgstat_reset_counters ( void  )

Definition at line 851 of file pgstat.c.

853{
855
858 ts);
static bool match_db_entries(PgStatShared_HashEntry *entry, Datum match_data)
Definition: pgstat.c:839
void pgstat_reset_matching_entries(bool(*do_reset)(PgStatShared_HashEntry *, Datum), Datum match_data, TimestampTz ts)
static Datum ObjectIdGetDatum(Oid X)
Definition: postgres.h:257

References GetCurrentTimestamp(), match_db_entries(), MyDatabaseId, ObjectIdGetDatum(), and pgstat_reset_matching_entries().

Referenced by pg_stat_reset().

◆ pgstat_reset_of_kind()

void pgstat_reset_of_kind ( PgStat_Kind  kind)

Definition at line 892 of file pgstat.c.

894{
895 const PgStat_KindInfo *kind_info = pgstat_get_kind_info(kind);
897
898 if (kind_info->fixed_amount)
899 kind_info->reset_all_cb(ts);
900 else
void pgstat_reset_entries_of_kind(PgStat_Kind kind, TimestampTz ts)

References PgStat_KindInfo::fixed_amount, GetCurrentTimestamp(), pgstat_get_kind_info(), pgstat_reset_entries_of_kind(), and PgStat_KindInfo::reset_all_cb.

Referenced by pg_stat_reset_replication_slot(), pg_stat_reset_shared(), pg_stat_reset_slru(), and pg_stat_reset_subscription_stats().

◆ pgstat_restore_stats()

void pgstat_restore_stats ( XLogRecPtr  redo)

Definition at line 505 of file pgstat.c.

507{
static void pgstat_read_statsfile(XLogRecPtr redo)
Definition: pgstat.c:1775

References pgstat_read_statsfile().

Referenced by StartupXLOG().

◆ pgstat_shutdown_hook()

static void pgstat_shutdown_hook ( int  code,
Datum  arg 
)
static

Definition at line 601 of file pgstat.c.

603{
604 Assert(!pgstat_is_shutdown);
606
607 /*
608 * If we got as far as discovering our own database ID, we can flush out
609 * what we did so far. Otherwise, we'd be reporting an invalid database
610 * ID, so forget it. (This means that accesses to pg_database during
611 * failed backend starts might never get counted.)
612 */
615
616 pgstat_report_stat(true);
617
618 /* there shouldn't be any pending changes left */
621
622 /* drop the backend stats entry */
625
627
628#ifdef USE_ASSERT_CHECKING
629 pgstat_is_shutdown = true;
630#endif
#define OidIsValid(objectId)
Definition: c.h:732
ProcNumber MyProcNumber
Definition: globals.c:89
static void dlist_init(dlist_head *head)
Definition: ilist.h:314
void pgstat_report_disconnect(Oid dboid)
#define PGSTAT_KIND_BACKEND
Definition: pgstat_kind.h:32
void pgstat_request_entry_refs_gc(void)
Definition: pgstat_shmem.c:700
bool pgstat_drop_entry(PgStat_Kind kind, Oid dboid, uint64 objid)
Definition: pgstat_shmem.c:953
void pgstat_detach_shmem(void)
Definition: pgstat_shmem.c:264

References Assert, dlist_init(), dlist_is_empty(), InvalidOid, IsPostmasterEnvironment, IsUnderPostmaster, MyDatabaseId, MyProcNumber, OidIsValid, pgstat_detach_shmem(), pgstat_drop_entry(), PGSTAT_KIND_BACKEND, pgstat_report_disconnect(), pgstat_report_stat(), pgstat_request_entry_refs_gc(), and pgStatPending.

Referenced by pgstat_initialize().

◆ pgstat_snapshot_fixed()

◆ pgstat_write_statsfile()

static void pgstat_write_statsfile ( XLogRecPtr  redo)
static

Definition at line 1587 of file pgstat.c.

1589{
1590 FILE *fpout;
1591 int32 format_id;
1592 const char *tmpfile = PGSTAT_STAT_PERMANENT_TMPFILE;
1593 const char *statfile = PGSTAT_STAT_PERMANENT_FILENAME;
1594 dshash_seq_status hstat;
1596
1598
1599 /* should be called only by the checkpointer or single user mode */
1601
1602 /* we're shutting down, so it's ok to just override this */
1604
1605 elog(DEBUG2, "writing stats file \"%s\" with redo %X/%X", statfile,
1606 LSN_FORMAT_ARGS(redo));
1607
1608 /*
1609 * Open the statistics temp file to write out the current values.
1610 */
1611 fpout = AllocateFile(tmpfile, PG_BINARY_W);
1612 if (fpout == NULL)
1613 {
1614 ereport(LOG,
1616 errmsg("could not open temporary statistics file \"%s\": %m",
1617 tmpfile)));
1618 return;
1619 }
1620
1621 /*
1622 * Write the file header --- currently just a format ID.
1623 */
1624 format_id = PGSTAT_FILE_FORMAT_ID;
1625 write_chunk_s(fpout, &format_id);
1626
1627 /* Write the redo LSN, used to cross check the file read */
1628 write_chunk_s(fpout, &redo);
1629
1630 /* Write various stats structs for fixed number of objects */
1631 for (PgStat_Kind kind = PGSTAT_KIND_MIN; kind <= PGSTAT_KIND_MAX; kind++)
1632 {
1633 char *ptr;
1634 const PgStat_KindInfo *info = pgstat_get_kind_info(kind);
1635
1636 if (!info || !info->fixed_amount)
1637 continue;
1638
1639 if (pgstat_is_kind_builtin(kind))
1640 Assert(info->snapshot_ctl_off != 0);
1641
1642 /* skip if no need to write to file */
1643 if (!info->write_to_file)
1644 continue;
1645
1647 if (pgstat_is_kind_builtin(kind))
1648 ptr = ((char *) &pgStatLocal.snapshot) + info->snapshot_ctl_off;
1649 else
1651
1652 fputc(PGSTAT_FILE_ENTRY_FIXED, fpout);
1653 write_chunk_s(fpout, &kind);
1654 write_chunk(fpout, ptr, info->shared_data_len);
1655 }
1656
1657 /*
1658 * Walk through the stats entries
1659 */
1660 dshash_seq_init(&hstat, pgStatLocal.shared_hash, false);
1661 while ((ps = dshash_seq_next(&hstat)) != NULL)
1662 {
1663 PgStatShared_Common *shstats;
1664 const PgStat_KindInfo *kind_info = NULL;
1665
1667
1668 /*
1669 * We should not see any "dropped" entries when writing the stats
1670 * file, as all backends and auxiliary processes should have cleaned
1671 * up their references before they terminated.
1672 *
1673 * However, since we are already shutting down, it is not worth
1674 * crashing the server over any potential cleanup issues, so we simply
1675 * skip such entries if encountered.
1676 */
1677 Assert(!ps->dropped);
1678 if (ps->dropped)
1679 continue;
1680
1681 /*
1682 * This discards data related to custom stats kinds that are unknown
1683 * to this process.
1684 */
1685 if (!pgstat_is_kind_valid(ps->key.kind))
1686 {
1687 elog(WARNING, "found unknown stats entry %u/%u/%llu",
1688 ps->key.kind, ps->key.dboid,
1689 (unsigned long long) ps->key.objid);
1690 continue;
1691 }
1692
1694
1695 kind_info = pgstat_get_kind_info(ps->key.kind);
1696
1697 /* if not dropped the valid-entry refcount should exist */
1698 Assert(pg_atomic_read_u32(&ps->refcount) > 0);
1699
1700 /* skip if no need to write to file */
1701 if (!kind_info->write_to_file)
1702 continue;
1703
1704 if (!kind_info->to_serialized_name)
1705 {
1706 /* normal stats entry, identified by PgStat_HashKey */
1707 fputc(PGSTAT_FILE_ENTRY_HASH, fpout);
1708 write_chunk_s(fpout, &ps->key);
1709 }
1710 else
1711 {
1712 /* stats entry identified by name on disk (e.g. slots) */
1713 NameData name;
1714
1715 kind_info->to_serialized_name(&ps->key, shstats, &name);
1716
1717 fputc(PGSTAT_FILE_ENTRY_NAME, fpout);
1718 write_chunk_s(fpout, &ps->key.kind);
1719 write_chunk_s(fpout, &name);
1720 }
1721
1722 /* Write except the header part of the entry */
1723 write_chunk(fpout,
1724 pgstat_get_entry_data(ps->key.kind, shstats),
1725 pgstat_get_entry_len(ps->key.kind));
1726 }
1727 dshash_seq_term(&hstat);
1728
1729 /*
1730 * No more output to be done. Close the temp file and replace the old
1731 * pgstat.stat with it. The ferror() check replaces testing for error
1732 * after each individual fputc or fwrite (in write_chunk()) above.
1733 */
1734 fputc(PGSTAT_FILE_ENTRY_END, fpout);
1735
1736 if (ferror(fpout))
1737 {
1738 ereport(LOG,
1740 errmsg("could not write temporary statistics file \"%s\": %m",
1741 tmpfile)));
1742 FreeFile(fpout);
1743 unlink(tmpfile);
1744 }
1745 else if (FreeFile(fpout) < 0)
1746 {
1747 ereport(LOG,
1749 errmsg("could not close temporary statistics file \"%s\": %m",
1750 tmpfile)));
1751 unlink(tmpfile);
1752 }
1753 else if (durable_rename(tmpfile, statfile, LOG) < 0)
1754 {
1755 /* durable_rename already emitted log message */
1756 unlink(tmpfile);
1757 }
#define PG_BINARY_W
Definition: c.h:1233
int durable_rename(const char *oldfile, const char *newfile, int elevel)
Definition: fd.c:781
struct parser_state ps
@ B_CHECKPOINTER
Definition: miscadmin.h:362
BackendType MyBackendType
Definition: miscinit.c:64
static void write_chunk(FILE *fpout, void *ptr, size_t len)
Definition: pgstat.c:1570
#define write_chunk_s(fpout, ptr)
Definition: pgstat.c:1580
#define PGSTAT_STAT_PERMANENT_TMPFILE
Definition: pgstat.h:32
void(* to_serialized_name)(const PgStat_HashKey *key, const PgStatShared_Common *header, NameData *name)

References AllocateFile(), Assert, B_CHECKPOINTER, CHECK_FOR_INTERRUPTS, PgStat_Snapshot::custom_data, DEBUG2, PgStat_LocalState::dsa, dsa_get_address(), dshash_seq_init(), dshash_seq_next(), dshash_seq_term(), durable_rename(), elog, ereport, errcode_for_file_access(), errmsg(), PgStat_KindInfo::fixed_amount, FreeFile(), IsUnderPostmaster, LOG, LSN_FORMAT_ARGS, MyBackendType, name, pg_atomic_read_u32(), PG_BINARY_W, pgstat_assert_is_up, pgstat_build_snapshot_fixed(), pgstat_fetch_consistency, PGSTAT_FETCH_CONSISTENCY_NONE, PGSTAT_FILE_ENTRY_END, PGSTAT_FILE_ENTRY_FIXED, PGSTAT_FILE_ENTRY_HASH, PGSTAT_FILE_ENTRY_NAME, PGSTAT_FILE_FORMAT_ID, pgstat_get_entry_data(), pgstat_get_entry_len(), pgstat_get_kind_info(), pgstat_is_kind_builtin(), pgstat_is_kind_valid(), PgStat_Kind, PGSTAT_KIND_CUSTOM_MIN, PGSTAT_KIND_MAX, PGSTAT_KIND_MIN, PGSTAT_STAT_PERMANENT_FILENAME, PGSTAT_STAT_PERMANENT_TMPFILE, pgStatLocal, ps, PgStat_KindInfo::shared_data_len, PgStat_LocalState::shared_hash, PgStat_LocalState::snapshot, PgStat_KindInfo::snapshot_ctl_off, PgStat_KindInfo::to_serialized_name, WARNING, write_chunk(), write_chunk_s, and PgStat_KindInfo::write_to_file.

Referenced by pgstat_before_server_shutdown().

◆ read_chunk()

static bool read_chunk ( FILE *  fpin,
void *  ptr,
size_t  len 
)
static

Definition at line 1761 of file pgstat.c.

1763{
1764 return fread(ptr, 1, len, fpin) == len;
const void size_t len

References len.

Referenced by pgstat_read_statsfile().

◆ write_chunk()

static void write_chunk ( FILE *  fpout,
void *  ptr,
size_t  len 
)
static

Definition at line 1570 of file pgstat.c.

1572{
1573 int rc;
1574
1575 rc = fwrite(ptr, len, 1, fpout);
1576
1577 /* we'll check for errors with ferror once at the end */
1578 (void) rc;

References len.

Referenced by pgstat_write_statsfile().

Variable Documentation

◆ force_stats_snapshot_clear

◆ pgstat_fetch_consistency

◆ pgstat_kind_builtin_infos

const PgStat_KindInfo pgstat_kind_builtin_infos[PGSTAT_KIND_BUILTIN_SIZE]
static

Definition at line 277 of file pgstat.c.

Referenced by pgstat_get_kind_from_str(), and pgstat_get_kind_info().

◆ pgstat_kind_custom_infos

const PgStat_KindInfo** pgstat_kind_custom_infos = NULL
static

Definition at line 492 of file pgstat.c.

Referenced by pgstat_get_kind_from_str(), pgstat_get_kind_info(), and pgstat_register_kind().

◆ pgstat_track_counts

◆ pgStatForceNextFlush

bool pgStatForceNextFlush = false
static

Definition at line 245 of file pgstat.c.

Referenced by pgstat_force_next_flush(), and pgstat_report_stat().

◆ pgStatLocal

PgStat_LocalState pgStatLocal

Definition at line 213 of file pgstat.c.

Referenced by pgstat_acquire_entry_ref(), pgstat_archiver_reset_all_cb(), pgstat_archiver_snapshot_cb(), pgstat_attach_shmem(), pgstat_before_server_shutdown(), pgstat_bgwriter_reset_all_cb(), pgstat_bgwriter_snapshot_cb(), pgstat_build_snapshot(), pgstat_build_snapshot_fixed(), pgstat_checkpointer_reset_all_cb(), pgstat_checkpointer_snapshot_cb(), pgstat_clear_snapshot(), pgstat_detach_shmem(), pgstat_drop_database_and_contents(), pgstat_drop_entry(), pgstat_drop_entry_internal(), pgstat_drop_matching_entries(), pgstat_fetch_entry(), pgstat_fetch_slru(), pgstat_fetch_stat_archiver(), pgstat_fetch_stat_bgwriter(), pgstat_fetch_stat_checkpointer(), pgstat_fetch_stat_io(), pgstat_fetch_stat_wal(), pgstat_free_entry(), pgstat_gc_entry_refs(), pgstat_get_custom_shmem_data(), pgstat_get_custom_snapshot_data(), pgstat_get_entry_ref(), pgstat_get_stat_snapshot_timestamp(), pgstat_init_entry(), pgstat_init_snapshot_fixed(), pgstat_io_flush_cb(), pgstat_io_reset_all_cb(), pgstat_io_snapshot_cb(), pgstat_need_entry_refs_gc(), pgstat_prep_snapshot(), pgstat_read_statsfile(), pgstat_reinit_entry(), pgstat_release_entry_ref(), pgstat_report_archiver(), pgstat_report_bgwriter(), pgstat_report_checkpointer(), pgstat_report_stat(), pgstat_request_entry_refs_gc(), pgstat_reset_matching_entries(), pgstat_reset_slru_counter_internal(), pgstat_setup_shared_refs(), pgstat_slru_flush_cb(), pgstat_slru_snapshot_cb(), pgstat_snapshot_fixed(), pgstat_wal_flush_cb(), pgstat_wal_reset_all_cb(), pgstat_wal_snapshot_cb(), pgstat_write_statsfile(), and StatsShmemInit().

◆ pgStatPending

dlist_head pgStatPending = DLIST_STATIC_INIT(pgStatPending)
static

◆ pgStatPendingContext

MemoryContext pgStatPendingContext = NULL
static

Definition at line 230 of file pgstat.c.

Referenced by pgstat_prep_pending_entry().