PostgreSQL Source Code  git master
injection_stats.c
Go to the documentation of this file.
1 /*--------------------------------------------------------------------------
2  *
3  * injection_stats.c
4  * Code for statistics of injection points.
5  *
6  * Portions Copyright (c) 1996-2024, PostgreSQL Global Development Group
7  * Portions Copyright (c) 1994, Regents of the University of California
8  *
9  * IDENTIFICATION
10  * src/test/modules/injection_points/injection_stats.c
11  *
12  * -------------------------------------------------------------------------
13  */
14 
15 #include "postgres.h"
16 
17 #include "fmgr.h"
18 
19 #include "common/hashfn.h"
20 #include "injection_stats.h"
21 #include "pgstat.h"
22 #include "utils/builtins.h"
23 #include "utils/pgstat_internal.h"
24 
25 /* Structures for statistics of injection points */
26 typedef struct PgStat_StatInjEntry
27 {
28  PgStat_Counter numcalls; /* number of times point has been run */
30 
32 {
36 
37 static bool injection_stats_flush_cb(PgStat_EntryRef *entry_ref, bool nowait);
38 
40  .name = "injection_points",
41  .fixed_amount = false, /* Bounded by the number of points */
42 
43  /* Injection points are system-wide */
44  .accessed_across_databases = true,
45 
46  .shared_size = sizeof(PgStatShared_InjectionPoint),
47  .shared_data_off = offsetof(PgStatShared_InjectionPoint, stats),
48  .shared_data_len = sizeof(((PgStatShared_InjectionPoint *) 0)->stats),
49  .pending_size = sizeof(PgStat_StatInjEntry),
50  .flush_pending_cb = injection_stats_flush_cb,
51 };
52 
53 /*
54  * Compute stats entry idx from point name with an 8-byte hash.
55  */
56 #define PGSTAT_INJ_IDX(name) hash_bytes_extended((const unsigned char *) name, strlen(name), 0)
57 
58 /*
59  * Kind ID reserved for statistics of injection points.
60  */
61 #define PGSTAT_KIND_INJECTION 129
62 
63 /* Track if stats are loaded */
64 static bool inj_stats_loaded = false;
65 
66 /*
67  * Callback for stats handling
68  */
69 static bool
70 injection_stats_flush_cb(PgStat_EntryRef *entry_ref, bool nowait)
71 {
72  PgStat_StatInjEntry *localent;
73  PgStatShared_InjectionPoint *shfuncent;
74 
75  localent = (PgStat_StatInjEntry *) entry_ref->pending;
76  shfuncent = (PgStatShared_InjectionPoint *) entry_ref->shared_stats;
77 
78  if (!pgstat_lock_entry(entry_ref, nowait))
79  return false;
80 
81  shfuncent->stats.numcalls += localent->numcalls;
82  return true;
83 }
84 
85 /*
86  * Support function for the SQL-callable pgstat* functions. Returns
87  * a pointer to the injection point statistics struct.
88  */
89 static PgStat_StatInjEntry *
91 {
92  PgStat_StatInjEntry *entry = NULL;
93 
95  return NULL;
96 
97  /* Compile the lookup key as a hash of the point name */
99  InvalidOid,
101  return entry;
102 }
103 
104 /*
105  * Workhorse to do the registration work, called in _PG_init().
106  */
107 void
109 {
111 
112  /* mark stats as loaded */
113  inj_stats_loaded = true;
114 }
115 
116 /*
117  * Report injection point creation.
118  */
119 void
121 {
122  PgStat_EntryRef *entry_ref;
123  PgStatShared_InjectionPoint *shstatent;
124 
125  /* leave if disabled */
127  return;
128 
130  PGSTAT_INJ_IDX(name), false);
131  shstatent = (PgStatShared_InjectionPoint *) entry_ref->shared_stats;
132 
133  /* initialize shared memory data */
134  memset(&shstatent->stats, 0, sizeof(shstatent->stats));
135  pgstat_unlock_entry(entry_ref);
136 }
137 
138 /*
139  * Report injection point drop.
140  */
141 void
142 pgstat_drop_inj(const char *name)
143 {
144  /* leave if disabled */
146  return;
147 
151 }
152 
153 /*
154  * Report statistics for injection point.
155  *
156  * This is simple because the set of stats to report currently is simple:
157  * track the number of times a point has been run.
158  */
159 void
161 {
162  PgStat_EntryRef *entry_ref;
163  PgStatShared_InjectionPoint *shstatent;
164  PgStat_StatInjEntry *statent;
165 
166  /* leave if disabled */
168  return;
169 
171  PGSTAT_INJ_IDX(name), false);
172 
173  shstatent = (PgStatShared_InjectionPoint *) entry_ref->shared_stats;
174  statent = &shstatent->stats;
175 
176  /* Update the injection point statistics */
177  statent->numcalls++;
178 
179  pgstat_unlock_entry(entry_ref);
180 }
181 
182 /*
183  * SQL function returning the number of times an injection point
184  * has been called.
185  */
187 Datum
189 {
192 
193  if (entry == NULL)
194  PG_RETURN_NULL();
195 
196  PG_RETURN_INT64(entry->numcalls);
197 }
#define PG_GETARG_TEXT_PP(n)
Definition: fmgr.h:309
#define PG_RETURN_INT64(x)
Definition: fmgr.h:368
#define PG_RETURN_NULL()
Definition: fmgr.h:345
#define PG_FUNCTION_ARGS
Definition: fmgr.h:193
bool inj_stats_enabled
void pgstat_report_inj(const char *name)
void pgstat_register_inj(void)
static bool inj_stats_loaded
#define PGSTAT_INJ_IDX(name)
void pgstat_create_inj(const char *name)
struct PgStat_StatInjEntry PgStat_StatInjEntry
void pgstat_drop_inj(const char *name)
#define PGSTAT_KIND_INJECTION
PG_FUNCTION_INFO_V1(injection_points_stats_numcalls)
static PgStat_StatInjEntry * pgstat_fetch_stat_injentry(const char *name)
static bool injection_stats_flush_cb(PgStat_EntryRef *entry_ref, bool nowait)
static const PgStat_KindInfo injection_stats
struct PgStatShared_InjectionPoint PgStatShared_InjectionPoint
Datum injection_points_stats_numcalls(PG_FUNCTION_ARGS)
void * pgstat_fetch_entry(PgStat_Kind kind, Oid dboid, uint64 objid)
Definition: pgstat.c:928
void pgstat_register_kind(PgStat_Kind kind, const PgStat_KindInfo *kind_info)
Definition: pgstat.c:1458
int64 PgStat_Counter
Definition: pgstat.h:120
void pgstat_request_entry_refs_gc(void)
Definition: pgstat_shmem.c:663
bool pgstat_drop_entry(PgStat_Kind kind, Oid dboid, uint64 objid)
Definition: pgstat_shmem.c:909
void pgstat_unlock_entry(PgStat_EntryRef *entry_ref)
Definition: pgstat_shmem.c:638
PgStat_EntryRef * pgstat_get_entry_ref_locked(PgStat_Kind kind, Oid dboid, uint64 objid, bool nowait)
Definition: pgstat_shmem.c:647
bool pgstat_lock_entry(PgStat_EntryRef *entry_ref, bool nowait)
Definition: pgstat_shmem.c:610
uintptr_t Datum
Definition: postgres.h:64
#define InvalidOid
Definition: postgres_ext.h:36
PgStat_StatInjEntry stats
PgStatShared_Common header
PgStatShared_Common * shared_stats
const char *const name
PgStat_Counter numcalls
char * text_to_cstring(const text *t)
Definition: varlena.c:217
const char * name