LCOV - differential code coverage report
Current view: top level - src/backend/utils/activity - pgstat_replslot.c (source / functions) Coverage Total Hit UNC UBC GNC CBC
Current: bed3ffbf9d952be6c7d739d068cdce44c046dfb7 vs 574581b50ac9c63dd9e4abebb731a3b67e5b50f6 Lines: 98.8 % 86 85 1 13 72
Current Date: 2026-05-05 10:23:31 +0900 Functions: 100.0 % 11 11 2 9
Baseline: lcov-20260505-025707-baseline Branches: 62.5 % 32 20 3 9 3 17
Baseline Date: 2026-05-05 10:27:06 +0900 Line coverage date bins:
Legend: Lines:     hit not hit
Branches: + taken - not taken # not executed
(30,360] days: 100.0 % 13 13 13
(360..) days: 98.6 % 73 72 1 72
Function coverage date bins:
(30,360] days: 100.0 % 1 1 1
(360..) days: 100.0 % 10 10 1 9
Branch coverage date bins:
(30,360] days: 50.0 % 6 3 3 3
(360..) days: 65.4 % 26 17 9 17

 Age         Owner                    Branch data    TLA  Line data    Source code
                                  1                 :                : /* -------------------------------------------------------------------------
                                  2                 :                :  *
                                  3                 :                :  * pgstat_replslot.c
                                  4                 :                :  *    Implementation of replication slot statistics.
                                  5                 :                :  *
                                  6                 :                :  * This file contains the implementation of replication slot statistics. It is kept
                                  7                 :                :  * separate from pgstat.c to enforce the line between the statistics access /
                                  8                 :                :  * storage implementation and the details about individual types of
                                  9                 :                :  * statistics.
                                 10                 :                :  *
                                 11                 :                :  * Replication slot stats work a bit different than other variable-numbered
                                 12                 :                :  * stats. Slots do not have oids (so they can be created on physical
                                 13                 :                :  * replicas). Use the slot index as object id while running. However, the slot
                                 14                 :                :  * index can change when restarting. That is addressed by using the name when
                                 15                 :                :  * (de-)serializing. After a restart it is possible for slots to have been
                                 16                 :                :  * dropped while shut down, which is addressed by not restoring stats for
                                 17                 :                :  * slots that cannot be found by name when starting up.
                                 18                 :                :  *
                                 19                 :                :  * Copyright (c) 2001-2026, PostgreSQL Global Development Group
                                 20                 :                :  *
                                 21                 :                :  * IDENTIFICATION
                                 22                 :                :  *    src/backend/utils/activity/pgstat_replslot.c
                                 23                 :                :  * -------------------------------------------------------------------------
                                 24                 :                :  */
                                 25                 :                : 
                                 26                 :                : #include "postgres.h"
                                 27                 :                : 
                                 28                 :                : #include "replication/slot.h"
                                 29                 :                : #include "utils/pgstat_internal.h"
                                 30                 :                : 
                                 31                 :                : 
                                 32                 :                : static int  get_replslot_index(const char *name, bool need_lock);
                                 33                 :                : 
                                 34                 :                : 
                                 35                 :                : /*
                                 36                 :                :  * Reset counters for a single replication slot.
                                 37                 :                :  *
                                 38                 :                :  * Permission checking for this function is managed through the normal
                                 39                 :                :  * GRANT system.
                                 40                 :                :  */
                                 41                 :                : void
 1490 andres@anarazel.de         42                 :CBC           4 : pgstat_reset_replslot(const char *name)
                                 43                 :                : {
                                 44                 :                :     ReplicationSlot *slot;
                                 45                 :                : 
 1285 peter@eisentraut.org       46         [ -  + ]:              4 :     Assert(name != NULL);
                                 47                 :                : 
  785 michael@paquier.xyz        48                 :              4 :     LWLockAcquire(ReplicationSlotControlLock, LW_SHARED);
                                 49                 :                : 
                                 50                 :                :     /* Check if the slot exists with the given name. */
                                 51                 :              4 :     slot = SearchNamedReplicationSlot(name, false);
 1490 andres@anarazel.de         52         [ +  + ]:              4 :     if (!slot)
                                 53         [ +  - ]:              1 :         ereport(ERROR,
                                 54                 :                :                 (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
                                 55                 :                :                  errmsg("replication slot \"%s\" does not exist",
                                 56                 :                :                         name)));
                                 57                 :                : 
                                 58                 :                :     /*
                                 59                 :                :      * Reset stats if it is a logical slot. Nothing to do for physical slots
                                 60                 :                :      * as we collect stats only for logical slots.
                                 61                 :                :      */
  785 michael@paquier.xyz        62         [ +  - ]:              3 :     if (SlotIsLogical(slot))
                                 63                 :              3 :         pgstat_reset(PGSTAT_KIND_REPLSLOT, InvalidOid,
                                 64                 :              3 :                      ReplicationSlotIndex(slot));
                                 65                 :                : 
                                 66                 :              3 :     LWLockRelease(ReplicationSlotControlLock);
 1506 andres@anarazel.de         67                 :              3 : }
                                 68                 :                : 
                                 69                 :                : /*
                                 70                 :                :  * Report replication slot statistics.
                                 71                 :                :  *
                                 72                 :                :  * We can rely on the stats for the slot to exist and to belong to this
                                 73                 :                :  * slot. We can only get here if pgstat_create_replslot() or
                                 74                 :                :  * pgstat_acquire_replslot() have already been called.
                                 75                 :                :  */
                                 76                 :                : void
 1490                            77                 :           6002 : pgstat_report_replslot(ReplicationSlot *slot, const PgStat_StatReplSlotEntry *repSlotStat)
                                 78                 :                : {
                                 79                 :                :     PgStat_EntryRef *entry_ref;
                                 80                 :                :     PgStatShared_ReplSlot *shstatent;
                                 81                 :                :     PgStat_StatReplSlotEntry *statent;
                                 82                 :                : 
                                 83                 :           6002 :     entry_ref = pgstat_get_entry_ref_locked(PGSTAT_KIND_REPLSLOT, InvalidOid,
                                 84                 :           6002 :                                             ReplicationSlotIndex(slot), false);
                                 85                 :           6002 :     shstatent = (PgStatShared_ReplSlot *) entry_ref->shared_stats;
                                 86                 :           6002 :     statent = &shstatent->stats;
                                 87                 :                : 
                                 88                 :                :     /* Update the replication slot statistics */
                                 89                 :                : #define REPLSLOT_ACC(fld) statent->fld += repSlotStat->fld
                                 90                 :           6002 :     REPLSLOT_ACC(spill_txns);
                                 91                 :           6002 :     REPLSLOT_ACC(spill_count);
                                 92                 :           6002 :     REPLSLOT_ACC(spill_bytes);
                                 93                 :           6002 :     REPLSLOT_ACC(stream_txns);
                                 94                 :           6002 :     REPLSLOT_ACC(stream_count);
                                 95                 :           6002 :     REPLSLOT_ACC(stream_bytes);
  209 msawada@postgresql.o       96                 :GNC        6002 :     REPLSLOT_ACC(mem_exceeded_count);
 1490 andres@anarazel.de         97                 :CBC        6002 :     REPLSLOT_ACC(total_txns);
                                 98                 :           6002 :     REPLSLOT_ACC(total_bytes);
                                 99                 :                : #undef REPLSLOT_ACC
                                100                 :                : 
                                101                 :           6002 :     pgstat_unlock_entry(entry_ref);
 1506                           102                 :           6002 : }
                                103                 :                : 
                                104                 :                : /*
                                105                 :                :  * Report replication slot sync skip statistics.
                                106                 :                :  *
                                107                 :                :  * Similar to pgstat_report_replslot(), we can rely on the stats for the
                                108                 :                :  * slot to exist and to belong to this slot.
                                109                 :                :  */
                                110                 :                : void
  161 akapila@postgresql.o      111                 :GNC           7 : pgstat_report_replslotsync(ReplicationSlot *slot)
                                112                 :                : {
                                113                 :                :     PgStat_EntryRef *entry_ref;
                                114                 :                :     PgStatShared_ReplSlot *shstatent;
                                115                 :                :     PgStat_StatReplSlotEntry *statent;
                                116                 :                : 
                                117                 :                :     /* Slot sync stats are valid only for synced logical slots on standby. */
  158                           118         [ -  + ]:              7 :     Assert(slot->data.synced);
  161                           119         [ -  + ]:              7 :     Assert(RecoveryInProgress());
                                120                 :                : 
                                121                 :              7 :     entry_ref = pgstat_get_entry_ref_locked(PGSTAT_KIND_REPLSLOT, InvalidOid,
                                122                 :              7 :                                             ReplicationSlotIndex(slot), false);
                                123         [ -  + ]:              7 :     Assert(entry_ref != NULL);
                                124                 :                : 
                                125                 :              7 :     shstatent = (PgStatShared_ReplSlot *) entry_ref->shared_stats;
                                126                 :              7 :     statent = &shstatent->stats;
                                127                 :                : 
                                128                 :              7 :     statent->slotsync_skip_count += 1;
  151                           129                 :              7 :     statent->slotsync_last_skip = GetCurrentTimestamp();
                                130                 :                : 
  161                           131                 :              7 :     pgstat_unlock_entry(entry_ref);
                                132                 :              7 : }
                                133                 :                : 
                                134                 :                : /*
                                135                 :                :  * Report replication slot creation.
                                136                 :                :  *
                                137                 :                :  * NB: This gets called with ReplicationSlotAllocationLock already held, be
                                138                 :                :  * careful about calling back into slot.c.
                                139                 :                :  */
                                140                 :                : void
 1490 andres@anarazel.de        141                 :CBC         511 : pgstat_create_replslot(ReplicationSlot *slot)
                                142                 :                : {
                                143                 :                :     PgStat_EntryRef *entry_ref;
                                144                 :                :     PgStatShared_ReplSlot *shstatent;
                                145                 :                : 
  783 michael@paquier.xyz       146         [ -  + ]:            511 :     Assert(LWLockHeldByMeInMode(ReplicationSlotAllocationLock, LW_EXCLUSIVE));
                                147                 :                : 
 1490 andres@anarazel.de        148                 :            511 :     entry_ref = pgstat_get_entry_ref_locked(PGSTAT_KIND_REPLSLOT, InvalidOid,
                                149                 :            511 :                                             ReplicationSlotIndex(slot), false);
                                150                 :            511 :     shstatent = (PgStatShared_ReplSlot *) entry_ref->shared_stats;
                                151                 :                : 
                                152                 :                :     /*
                                153                 :                :      * NB: need to accept that there might be stats from an older slot, e.g.
                                154                 :                :      * if we previously crashed after dropping a slot.
                                155                 :                :      */
                                156                 :            511 :     memset(&shstatent->stats, 0, sizeof(shstatent->stats));
                                157                 :                : 
                                158                 :            511 :     pgstat_unlock_entry(entry_ref);
                                159                 :            511 : }
                                160                 :                : 
                                161                 :                : /*
                                162                 :                :  * Report replication slot has been acquired.
                                163                 :                :  *
                                164                 :                :  * This guarantees that a stats entry exists during later
                                165                 :                :  * pgstat_report_replslot() or pgstat_report_replslotsync() calls.
                                166                 :                :  *
                                167                 :                :  * If we previously crashed, no stats data exists. But if we did not crash,
                                168                 :                :  * the stats do belong to this slot:
                                169                 :                :  * - the stats cannot belong to a dropped slot, pgstat_drop_replslot() would
                                170                 :                :  *   have been called
                                171                 :                :  * - if the slot was removed while shut down,
                                172                 :                :  *   pgstat_replslot_from_serialized_name_cb() returning false would have
                                173                 :                :  *   caused the stats to be dropped
                                174                 :                :  */
                                175                 :                : void
                                176                 :           1145 : pgstat_acquire_replslot(ReplicationSlot *slot)
                                177                 :                : {
 1305                           178                 :           1145 :     pgstat_get_entry_ref(PGSTAT_KIND_REPLSLOT, InvalidOid,
                                179                 :           1145 :                          ReplicationSlotIndex(slot), true, NULL);
 1506                           180                 :           1145 : }
                                181                 :                : 
                                182                 :                : /*
                                183                 :                :  * Report replication slot drop.
                                184                 :                :  */
                                185                 :                : void
 1490                           186                 :            441 : pgstat_drop_replslot(ReplicationSlot *slot)
                                187                 :                : {
  783 michael@paquier.xyz       188         [ -  + ]:            441 :     Assert(LWLockHeldByMeInMode(ReplicationSlotAllocationLock, LW_EXCLUSIVE));
                                189                 :                : 
  698                           190         [ +  + ]:            441 :     if (!pgstat_drop_entry(PGSTAT_KIND_REPLSLOT, InvalidOid,
                                191                 :            441 :                            ReplicationSlotIndex(slot)))
                                192                 :             48 :         pgstat_request_entry_refs_gc();
 1490 andres@anarazel.de        193                 :            441 : }
                                194                 :                : 
                                195                 :                : /*
                                196                 :                :  * Support function for the SQL-callable pgstat* functions. Returns
                                197                 :                :  * a pointer to the replication slot statistics struct.
                                198                 :                :  */
                                199                 :                : PgStat_StatReplSlotEntry *
                                200                 :             52 : pgstat_fetch_replslot(NameData slotname)
                                201                 :                : {
                                202                 :                :     int         idx;
  785 michael@paquier.xyz       203                 :             52 :     PgStat_StatReplSlotEntry *slotentry = NULL;
                                204                 :                : 
                                205                 :             52 :     LWLockAcquire(ReplicationSlotControlLock, LW_SHARED);
                                206                 :                : 
                                207                 :             52 :     idx = get_replslot_index(NameStr(slotname), false);
                                208                 :                : 
                                209         [ +  + ]:             52 :     if (idx != -1)
                                210                 :             50 :         slotentry = (PgStat_StatReplSlotEntry *) pgstat_fetch_entry(PGSTAT_KIND_REPLSLOT,
                                211                 :                :                                                                     InvalidOid, idx,
                                212                 :                :                                                                     NULL);
                                213                 :                : 
                                214                 :             52 :     LWLockRelease(ReplicationSlotControlLock);
                                215                 :                : 
                                216                 :             52 :     return slotentry;
                                217                 :                : }
                                218                 :                : 
                                219                 :                : void
 1305 andres@anarazel.de        220                 :            110 : pgstat_replslot_to_serialized_name_cb(const PgStat_HashKey *key, const PgStatShared_Common *header, NameData *name)
                                221                 :                : {
                                222                 :                :     /*
                                223                 :                :      * This is only called late during shutdown. The set of existing slots
                                224                 :                :      * isn't allowed to change at this point, we can assume that a slot exists
                                225                 :                :      * at the offset.
                                226                 :                :      */
  594 michael@paquier.xyz       227         [ -  + ]:            110 :     if (!ReplicationSlotName(key->objid, name))
  402 peter@eisentraut.org      228         [ #  # ]:UBC           0 :         elog(ERROR, "could not find name for replication slot index %" PRIu64,
                                229                 :                :              key->objid);
 1490 andres@anarazel.de        230                 :CBC         110 : }
                                231                 :                : 
                                232                 :                : bool
                                233                 :             78 : pgstat_replslot_from_serialized_name_cb(const NameData *name, PgStat_HashKey *key)
                                234                 :                : {
  785 michael@paquier.xyz       235                 :             78 :     int         idx = get_replslot_index(NameStr(*name), true);
                                236                 :                : 
                                237                 :                :     /* slot might have been deleted */
 1490 andres@anarazel.de        238         [ +  + ]:             78 :     if (idx == -1)
                                239                 :              1 :         return false;
                                240                 :                : 
                                241                 :             77 :     key->kind = PGSTAT_KIND_REPLSLOT;
                                242                 :             77 :     key->dboid = InvalidOid;
  594 michael@paquier.xyz       243                 :             77 :     key->objid = idx;
                                244                 :                : 
 1490 andres@anarazel.de        245                 :             77 :     return true;
                                246                 :                : }
                                247                 :                : 
                                248                 :                : void
                                249                 :              8 : pgstat_replslot_reset_timestamp_cb(PgStatShared_Common *header, TimestampTz ts)
                                250                 :                : {
                                251                 :              8 :     ((PgStatShared_ReplSlot *) header)->stats.stat_reset_timestamp = ts;
                                252                 :              8 : }
                                253                 :                : 
                                254                 :                : static int
  785 michael@paquier.xyz       255                 :            130 : get_replslot_index(const char *name, bool need_lock)
                                256                 :                : {
                                257                 :                :     ReplicationSlot *slot;
                                258                 :                : 
 1285 peter@eisentraut.org      259         [ -  + ]:            130 :     Assert(name != NULL);
                                260                 :                : 
  785 michael@paquier.xyz       261                 :            130 :     slot = SearchNamedReplicationSlot(name, need_lock);
                                262                 :                : 
 1490 andres@anarazel.de        263         [ +  + ]:            130 :     if (!slot)
                                264                 :              3 :         return -1;
                                265                 :                : 
                                266                 :            127 :     return ReplicationSlotIndex(slot);
                                267                 :                : }
        

Generated by: LCOV version 2.5.0-beta