Coverage Report

Created: 2025-06-15 06:31

/src/postgres/src/backend/utils/activity/pgstat_checkpointer.c
Line
Count
Source (jump to first uncovered line)
1
/* -------------------------------------------------------------------------
2
 *
3
 * pgstat_checkpointer.c
4
 *    Implementation of checkpoint statistics.
5
 *
6
 * This file contains the implementation of checkpoint 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_checkpointer.c
15
 * -------------------------------------------------------------------------
16
 */
17
18
#include "postgres.h"
19
20
#include "utils/memutils.h"
21
#include "utils/pgstat_internal.h"
22
23
24
PgStat_CheckpointerStats PendingCheckpointerStats = {0};
25
26
27
/*
28
 * Report checkpointer and IO statistics
29
 */
30
void
31
pgstat_report_checkpointer(void)
32
0
{
33
0
  PgStatShared_Checkpointer *stats_shmem = &pgStatLocal.shmem->checkpointer;
34
35
0
  Assert(!pgStatLocal.shmem->is_shutdown);
36
0
  pgstat_assert_is_up();
37
38
  /*
39
   * This function can be called even if nothing at all has happened. In
40
   * this case, avoid unnecessarily modifying the stats entry.
41
   */
42
0
  if (pg_memory_is_all_zeros(&PendingCheckpointerStats,
43
0
                 sizeof(struct PgStat_CheckpointerStats)))
44
0
    return;
45
46
0
  pgstat_begin_changecount_write(&stats_shmem->changecount);
47
48
0
#define CHECKPOINTER_ACC(fld) stats_shmem->stats.fld += PendingCheckpointerStats.fld
49
0
  CHECKPOINTER_ACC(num_timed);
50
0
  CHECKPOINTER_ACC(num_requested);
51
0
  CHECKPOINTER_ACC(num_performed);
52
0
  CHECKPOINTER_ACC(restartpoints_timed);
53
0
  CHECKPOINTER_ACC(restartpoints_requested);
54
0
  CHECKPOINTER_ACC(restartpoints_performed);
55
0
  CHECKPOINTER_ACC(write_time);
56
0
  CHECKPOINTER_ACC(sync_time);
57
0
  CHECKPOINTER_ACC(buffers_written);
58
0
  CHECKPOINTER_ACC(slru_written);
59
0
#undef CHECKPOINTER_ACC
60
61
0
  pgstat_end_changecount_write(&stats_shmem->changecount);
62
63
  /*
64
   * Clear out the statistics buffer, so it can be re-used.
65
   */
66
0
  MemSet(&PendingCheckpointerStats, 0, sizeof(PendingCheckpointerStats));
67
68
  /*
69
   * Report IO statistics
70
   */
71
0
  pgstat_flush_io(false);
72
0
}
73
74
/*
75
 * pgstat_fetch_stat_checkpointer() -
76
 *
77
 * Support function for the SQL-callable pgstat* functions. Returns
78
 * a pointer to the checkpointer statistics struct.
79
 */
80
PgStat_CheckpointerStats *
81
pgstat_fetch_stat_checkpointer(void)
82
0
{
83
0
  pgstat_snapshot_fixed(PGSTAT_KIND_CHECKPOINTER);
84
85
0
  return &pgStatLocal.snapshot.checkpointer;
86
0
}
87
88
void
89
pgstat_checkpointer_init_shmem_cb(void *stats)
90
0
{
91
0
  PgStatShared_Checkpointer *stats_shmem = (PgStatShared_Checkpointer *) stats;
92
93
0
  LWLockInitialize(&stats_shmem->lock, LWTRANCHE_PGSTATS_DATA);
94
0
}
95
96
void
97
pgstat_checkpointer_reset_all_cb(TimestampTz ts)
98
0
{
99
0
  PgStatShared_Checkpointer *stats_shmem = &pgStatLocal.shmem->checkpointer;
100
101
  /* see explanation above PgStatShared_Checkpointer for the reset protocol */
102
0
  LWLockAcquire(&stats_shmem->lock, LW_EXCLUSIVE);
103
0
  pgstat_copy_changecounted_stats(&stats_shmem->reset_offset,
104
0
                  &stats_shmem->stats,
105
0
                  sizeof(stats_shmem->stats),
106
0
                  &stats_shmem->changecount);
107
0
  stats_shmem->stats.stat_reset_timestamp = ts;
108
0
  LWLockRelease(&stats_shmem->lock);
109
0
}
110
111
void
112
pgstat_checkpointer_snapshot_cb(void)
113
0
{
114
0
  PgStatShared_Checkpointer *stats_shmem = &pgStatLocal.shmem->checkpointer;
115
0
  PgStat_CheckpointerStats *reset_offset = &stats_shmem->reset_offset;
116
0
  PgStat_CheckpointerStats reset;
117
118
0
  pgstat_copy_changecounted_stats(&pgStatLocal.snapshot.checkpointer,
119
0
                  &stats_shmem->stats,
120
0
                  sizeof(stats_shmem->stats),
121
0
                  &stats_shmem->changecount);
122
123
0
  LWLockAcquire(&stats_shmem->lock, LW_SHARED);
124
0
  memcpy(&reset, reset_offset, sizeof(stats_shmem->stats));
125
0
  LWLockRelease(&stats_shmem->lock);
126
127
  /* compensate by reset offsets */
128
0
#define CHECKPOINTER_COMP(fld) pgStatLocal.snapshot.checkpointer.fld -= reset.fld;
129
0
  CHECKPOINTER_COMP(num_timed);
130
0
  CHECKPOINTER_COMP(num_requested);
131
0
  CHECKPOINTER_COMP(num_performed);
132
0
  CHECKPOINTER_COMP(restartpoints_timed);
133
0
  CHECKPOINTER_COMP(restartpoints_requested);
134
0
  CHECKPOINTER_COMP(restartpoints_performed);
135
0
  CHECKPOINTER_COMP(write_time);
136
0
  CHECKPOINTER_COMP(sync_time);
137
0
  CHECKPOINTER_COMP(buffers_written);
138
0
  CHECKPOINTER_COMP(slru_written);
139
0
#undef CHECKPOINTER_COMP
140
0
}