LCOV - code coverage report
Current view: top level - src/backend/utils/activity - pgstat_wal.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 56 57 98.2 %
Date: 2025-01-18 04:15:08 Functions: 9 9 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /* -------------------------------------------------------------------------
       2             :  *
       3             :  * pgstat_wal.c
       4             :  *    Implementation of WAL statistics.
       5             :  *
       6             :  * This file contains the implementation of WAL 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             :  * Copyright (c) 2001-2025, PostgreSQL Global Development Group
      12             :  *
      13             :  * IDENTIFICATION
      14             :  *    src/backend/utils/activity/pgstat_wal.c
      15             :  * -------------------------------------------------------------------------
      16             :  */
      17             : 
      18             : #include "postgres.h"
      19             : 
      20             : #include "executor/instrument.h"
      21             : #include "utils/pgstat_internal.h"
      22             : 
      23             : 
      24             : PgStat_PendingWalStats PendingWalStats = {0};
      25             : 
      26             : /*
      27             :  * WAL usage counters saved from pgWalUsage at the previous call to
      28             :  * pgstat_report_wal(). This is used to calculate how much WAL usage
      29             :  * happens between pgstat_report_wal() calls, by subtracting
      30             :  * the previous counters from the current ones.
      31             :  */
      32             : static WalUsage prevWalUsage;
      33             : 
      34             : 
      35             : /*
      36             :  * Calculate how much WAL usage counters have increased and update
      37             :  * shared WAL and IO statistics.
      38             :  *
      39             :  * Must be called by processes that generate WAL, that do not call
      40             :  * pgstat_report_stat(), like walwriter.
      41             :  *
      42             :  * "force" set to true ensures that the statistics are flushed; note that
      43             :  * this needs to acquire the pgstat shmem LWLock, waiting on it.  When
      44             :  * set to false, the statistics may not be flushed if the lock could not
      45             :  * be acquired.
      46             :  */
      47             : void
      48       42900 : pgstat_report_wal(bool force)
      49             : {
      50             :     bool        nowait;
      51             : 
      52             :     /* like in pgstat.c, don't wait for lock acquisition when !force */
      53       42900 :     nowait = !force;
      54             : 
      55             :     /* flush wal stats */
      56       42900 :     pgstat_flush_wal(nowait);
      57             : 
      58             :     /* flush IO stats */
      59       42900 :     pgstat_flush_io(nowait);
      60       42900 : }
      61             : 
      62             : /*
      63             :  * Support function for the SQL-callable pgstat* functions. Returns
      64             :  * a pointer to the WAL statistics struct.
      65             :  */
      66             : PgStat_WalStats *
      67          70 : pgstat_fetch_stat_wal(void)
      68             : {
      69          70 :     pgstat_snapshot_fixed(PGSTAT_KIND_WAL);
      70             : 
      71          70 :     return &pgStatLocal.snapshot.wal;
      72             : }
      73             : 
      74             : /*
      75             :  * Simple wrapper of pgstat_wal_flush_cb()
      76             :  */
      77             : void
      78       42900 : pgstat_flush_wal(bool nowait)
      79             : {
      80       42900 :     (void) pgstat_wal_flush_cb(nowait);
      81       42900 : }
      82             : 
      83             : /*
      84             :  * Calculate how much WAL usage counters have increased by subtracting the
      85             :  * previous counters from the current ones.
      86             :  *
      87             :  * If nowait is true, this function returns true if the lock could not be
      88             :  * acquired. Otherwise return false.
      89             :  */
      90             : bool
      91      102308 : pgstat_wal_flush_cb(bool nowait)
      92             : {
      93      102308 :     PgStatShared_Wal *stats_shmem = &pgStatLocal.shmem->wal;
      94      102308 :     WalUsage    wal_usage_diff = {0};
      95             : 
      96             :     Assert(IsUnderPostmaster || !IsPostmasterEnvironment);
      97             :     Assert(pgStatLocal.shmem != NULL &&
      98             :            !pgStatLocal.shmem->is_shutdown);
      99             : 
     100             :     /*
     101             :      * This function can be called even if nothing at all has happened. Avoid
     102             :      * taking lock for nothing in that case.
     103             :      */
     104      102308 :     if (!pgstat_wal_have_pending_cb())
     105       77222 :         return false;
     106             : 
     107             :     /*
     108             :      * We don't update the WAL usage portion of the local WalStats elsewhere.
     109             :      * Calculate how much WAL usage counters were increased by subtracting the
     110             :      * previous counters from the current ones.
     111             :      */
     112       25086 :     WalUsageAccumDiff(&wal_usage_diff, &pgWalUsage, &prevWalUsage);
     113             : 
     114       25086 :     if (!nowait)
     115       15898 :         LWLockAcquire(&stats_shmem->lock, LW_EXCLUSIVE);
     116        9188 :     else if (!LWLockConditionalAcquire(&stats_shmem->lock, LW_EXCLUSIVE))
     117           0 :         return true;
     118             : 
     119             : #define WALSTAT_ACC(fld, var_to_add) \
     120             :     (stats_shmem->stats.fld += var_to_add.fld)
     121             : #define WALSTAT_ACC_INSTR_TIME(fld) \
     122             :     (stats_shmem->stats.fld += INSTR_TIME_GET_MICROSEC(PendingWalStats.fld))
     123       25086 :     WALSTAT_ACC(wal_records, wal_usage_diff);
     124       25086 :     WALSTAT_ACC(wal_fpi, wal_usage_diff);
     125       25086 :     WALSTAT_ACC(wal_bytes, wal_usage_diff);
     126       25086 :     WALSTAT_ACC(wal_buffers_full, PendingWalStats);
     127       25086 :     WALSTAT_ACC(wal_write, PendingWalStats);
     128       25086 :     WALSTAT_ACC(wal_sync, PendingWalStats);
     129       25086 :     WALSTAT_ACC_INSTR_TIME(wal_write_time);
     130       25086 :     WALSTAT_ACC_INSTR_TIME(wal_sync_time);
     131             : #undef WALSTAT_ACC_INSTR_TIME
     132             : #undef WALSTAT_ACC
     133             : 
     134       25086 :     LWLockRelease(&stats_shmem->lock);
     135             : 
     136             :     /*
     137             :      * Save the current counters for the subsequent calculation of WAL usage.
     138             :      */
     139       25086 :     prevWalUsage = pgWalUsage;
     140             : 
     141             :     /*
     142             :      * Clear out the statistics buffer, so it can be re-used.
     143             :      */
     144      150516 :     MemSet(&PendingWalStats, 0, sizeof(PendingWalStats));
     145             : 
     146       25086 :     return false;
     147             : }
     148             : 
     149             : void
     150       34726 : pgstat_wal_init_backend_cb(void)
     151             : {
     152             :     /*
     153             :      * Initialize prevWalUsage with pgWalUsage so that pgstat_wal_flush_cb()
     154             :      * can calculate how much pgWalUsage counters are increased by subtracting
     155             :      * prevWalUsage from pgWalUsage.
     156             :      */
     157       34726 :     prevWalUsage = pgWalUsage;
     158       34726 : }
     159             : 
     160             : /*
     161             :  * To determine whether any WAL activity has occurred since last time, not
     162             :  * only the number of generated WAL records but also the numbers of WAL
     163             :  * writes and syncs need to be checked. Because even transaction that
     164             :  * generates no WAL records can write or sync WAL data when flushing the
     165             :  * data pages.
     166             :  */
     167             : bool
     168      112456 : pgstat_wal_have_pending_cb(void)
     169             : {
     170      206864 :     return pgWalUsage.wal_records != prevWalUsage.wal_records ||
     171      199816 :         PendingWalStats.wal_write != 0 ||
     172       87360 :         PendingWalStats.wal_sync != 0;
     173             : }
     174             : 
     175             : void
     176        1918 : pgstat_wal_init_shmem_cb(void *stats)
     177             : {
     178        1918 :     PgStatShared_Wal *stats_shmem = (PgStatShared_Wal *) stats;
     179             : 
     180        1918 :     LWLockInitialize(&stats_shmem->lock, LWTRANCHE_PGSTATS_DATA);
     181        1918 : }
     182             : 
     183             : void
     184         466 : pgstat_wal_reset_all_cb(TimestampTz ts)
     185             : {
     186         466 :     PgStatShared_Wal *stats_shmem = &pgStatLocal.shmem->wal;
     187             : 
     188         466 :     LWLockAcquire(&stats_shmem->lock, LW_EXCLUSIVE);
     189         466 :     memset(&stats_shmem->stats, 0, sizeof(stats_shmem->stats));
     190         466 :     stats_shmem->stats.stat_reset_timestamp = ts;
     191         466 :     LWLockRelease(&stats_shmem->lock);
     192         466 : }
     193             : 
     194             : void
     195        1210 : pgstat_wal_snapshot_cb(void)
     196             : {
     197        1210 :     PgStatShared_Wal *stats_shmem = &pgStatLocal.shmem->wal;
     198             : 
     199        1210 :     LWLockAcquire(&stats_shmem->lock, LW_SHARED);
     200        1210 :     memcpy(&pgStatLocal.snapshot.wal, &stats_shmem->stats,
     201             :            sizeof(pgStatLocal.snapshot.wal));
     202        1210 :     LWLockRelease(&stats_shmem->lock);
     203        1210 : }

Generated by: LCOV version 1.14