Line data Source code
1 : /*-------------------------------------------------------------------------
2 : *
3 : * xactdesc.c
4 : * rmgr descriptor routines for access/transam/xact.c
5 : *
6 : * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
7 : * Portions Copyright (c) 1994, Regents of the University of California
8 : *
9 : *
10 : * IDENTIFICATION
11 : * src/backend/access/rmgrdesc/xactdesc.c
12 : *
13 : *-------------------------------------------------------------------------
14 : */
15 : #include "postgres.h"
16 :
17 : #include "access/transam.h"
18 : #include "access/xact.h"
19 : #include "replication/origin.h"
20 : #include "storage/sinval.h"
21 : #include "storage/standbydefs.h"
22 : #include "utils/timestamp.h"
23 :
24 : /*
25 : * Parse the WAL format of an xact commit and abort records into an easier to
26 : * understand format.
27 : *
28 : * These routines are in xactdesc.c because they're accessed in backend (when
29 : * replaying WAL) and frontend (pg_waldump) code. This file is the only xact
30 : * specific one shared between both. They're complicated enough that
31 : * duplication would be bothersome.
32 : */
33 :
34 : void
35 48920 : ParseCommitRecord(uint8 info, xl_xact_commit *xlrec, xl_xact_parsed_commit *parsed)
36 : {
37 48920 : char *data = ((char *) xlrec) + MinSizeOfXactCommit;
38 :
39 48920 : memset(parsed, 0, sizeof(*parsed));
40 :
41 48920 : parsed->xinfo = 0; /* default, if no XLOG_XACT_HAS_INFO is
42 : * present */
43 :
44 48920 : parsed->xact_time = xlrec->xact_time;
45 :
46 48920 : if (info & XLOG_XACT_HAS_INFO)
47 : {
48 35006 : xl_xact_xinfo *xl_xinfo = (xl_xact_xinfo *) data;
49 :
50 35006 : parsed->xinfo = xl_xinfo->xinfo;
51 :
52 35006 : data += sizeof(xl_xact_xinfo);
53 : }
54 :
55 48920 : if (parsed->xinfo & XACT_XINFO_HAS_DBINFO)
56 : {
57 34684 : xl_xact_dbinfo *xl_dbinfo = (xl_xact_dbinfo *) data;
58 :
59 34684 : parsed->dbId = xl_dbinfo->dbId;
60 34684 : parsed->tsId = xl_dbinfo->tsId;
61 :
62 34684 : data += sizeof(xl_xact_dbinfo);
63 : }
64 :
65 48920 : if (parsed->xinfo & XACT_XINFO_HAS_SUBXACTS)
66 : {
67 504 : xl_xact_subxacts *xl_subxacts = (xl_xact_subxacts *) data;
68 :
69 504 : parsed->nsubxacts = xl_subxacts->nsubxacts;
70 504 : parsed->subxacts = xl_subxacts->subxacts;
71 :
72 504 : data += MinSizeOfXactSubxacts;
73 504 : data += parsed->nsubxacts * sizeof(TransactionId);
74 : }
75 :
76 48920 : if (parsed->xinfo & XACT_XINFO_HAS_RELFILELOCATORS)
77 : {
78 4262 : xl_xact_relfilelocators *xl_rellocators = (xl_xact_relfilelocators *) data;
79 :
80 4262 : parsed->nrels = xl_rellocators->nrels;
81 4262 : parsed->xlocators = xl_rellocators->xlocators;
82 :
83 4262 : data += MinSizeOfXactRelfileLocators;
84 4262 : data += xl_rellocators->nrels * sizeof(RelFileLocator);
85 : }
86 :
87 48920 : if (parsed->xinfo & XACT_XINFO_HAS_DROPPED_STATS)
88 : {
89 5368 : xl_xact_stats_items *xl_drops = (xl_xact_stats_items *) data;
90 :
91 5368 : parsed->nstats = xl_drops->nitems;
92 5368 : parsed->stats = xl_drops->items;
93 :
94 5368 : data += MinSizeOfXactStatsItems;
95 5368 : data += xl_drops->nitems * sizeof(xl_xact_stats_item);
96 : }
97 :
98 48920 : if (parsed->xinfo & XACT_XINFO_HAS_INVALS)
99 : {
100 31050 : xl_xact_invals *xl_invals = (xl_xact_invals *) data;
101 :
102 31050 : parsed->nmsgs = xl_invals->nmsgs;
103 31050 : parsed->msgs = xl_invals->msgs;
104 :
105 31050 : data += MinSizeOfXactInvals;
106 31050 : data += xl_invals->nmsgs * sizeof(SharedInvalidationMessage);
107 : }
108 :
109 48920 : if (parsed->xinfo & XACT_XINFO_HAS_TWOPHASE)
110 : {
111 382 : xl_xact_twophase *xl_twophase = (xl_xact_twophase *) data;
112 :
113 382 : parsed->twophase_xid = xl_twophase->xid;
114 :
115 382 : data += sizeof(xl_xact_twophase);
116 :
117 382 : if (parsed->xinfo & XACT_XINFO_HAS_GID)
118 : {
119 214 : strlcpy(parsed->twophase_gid, data, sizeof(parsed->twophase_gid));
120 214 : data += strlen(data) + 1;
121 : }
122 : }
123 :
124 : /* Note: no alignment is guaranteed after this point */
125 :
126 48920 : if (parsed->xinfo & XACT_XINFO_HAS_ORIGIN)
127 : {
128 : xl_xact_origin xl_origin;
129 :
130 : /* no alignment is guaranteed, so copy onto stack */
131 176 : memcpy(&xl_origin, data, sizeof(xl_origin));
132 :
133 176 : parsed->origin_lsn = xl_origin.origin_lsn;
134 176 : parsed->origin_timestamp = xl_origin.origin_timestamp;
135 :
136 176 : data += sizeof(xl_xact_origin);
137 : }
138 48920 : }
139 :
140 : void
141 3632 : ParseAbortRecord(uint8 info, xl_xact_abort *xlrec, xl_xact_parsed_abort *parsed)
142 : {
143 3632 : char *data = ((char *) xlrec) + MinSizeOfXactAbort;
144 :
145 3632 : memset(parsed, 0, sizeof(*parsed));
146 :
147 3632 : parsed->xinfo = 0; /* default, if no XLOG_XACT_HAS_INFO is
148 : * present */
149 :
150 3632 : parsed->xact_time = xlrec->xact_time;
151 :
152 3632 : if (info & XLOG_XACT_HAS_INFO)
153 : {
154 2328 : xl_xact_xinfo *xl_xinfo = (xl_xact_xinfo *) data;
155 :
156 2328 : parsed->xinfo = xl_xinfo->xinfo;
157 :
158 2328 : data += sizeof(xl_xact_xinfo);
159 : }
160 :
161 3632 : if (parsed->xinfo & XACT_XINFO_HAS_DBINFO)
162 : {
163 86 : xl_xact_dbinfo *xl_dbinfo = (xl_xact_dbinfo *) data;
164 :
165 86 : parsed->dbId = xl_dbinfo->dbId;
166 86 : parsed->tsId = xl_dbinfo->tsId;
167 :
168 86 : data += sizeof(xl_xact_dbinfo);
169 : }
170 :
171 3632 : if (parsed->xinfo & XACT_XINFO_HAS_SUBXACTS)
172 : {
173 46 : xl_xact_subxacts *xl_subxacts = (xl_xact_subxacts *) data;
174 :
175 46 : parsed->nsubxacts = xl_subxacts->nsubxacts;
176 46 : parsed->subxacts = xl_subxacts->subxacts;
177 :
178 46 : data += MinSizeOfXactSubxacts;
179 46 : data += parsed->nsubxacts * sizeof(TransactionId);
180 : }
181 :
182 3632 : if (parsed->xinfo & XACT_XINFO_HAS_RELFILELOCATORS)
183 : {
184 610 : xl_xact_relfilelocators *xl_rellocator = (xl_xact_relfilelocators *) data;
185 :
186 610 : parsed->nrels = xl_rellocator->nrels;
187 610 : parsed->xlocators = xl_rellocator->xlocators;
188 :
189 610 : data += MinSizeOfXactRelfileLocators;
190 610 : data += xl_rellocator->nrels * sizeof(RelFileLocator);
191 : }
192 :
193 3632 : if (parsed->xinfo & XACT_XINFO_HAS_DROPPED_STATS)
194 : {
195 784 : xl_xact_stats_items *xl_drops = (xl_xact_stats_items *) data;
196 :
197 784 : parsed->nstats = xl_drops->nitems;
198 784 : parsed->stats = xl_drops->items;
199 :
200 784 : data += MinSizeOfXactStatsItems;
201 784 : data += xl_drops->nitems * sizeof(xl_xact_stats_item);
202 : }
203 :
204 3632 : if (parsed->xinfo & XACT_XINFO_HAS_TWOPHASE)
205 : {
206 162 : xl_xact_twophase *xl_twophase = (xl_xact_twophase *) data;
207 :
208 162 : parsed->twophase_xid = xl_twophase->xid;
209 :
210 162 : data += sizeof(xl_xact_twophase);
211 :
212 162 : if (parsed->xinfo & XACT_XINFO_HAS_GID)
213 : {
214 86 : strlcpy(parsed->twophase_gid, data, sizeof(parsed->twophase_gid));
215 86 : data += strlen(data) + 1;
216 : }
217 : }
218 :
219 : /* Note: no alignment is guaranteed after this point */
220 :
221 3632 : if (parsed->xinfo & XACT_XINFO_HAS_ORIGIN)
222 : {
223 : xl_xact_origin xl_origin;
224 :
225 : /* no alignment is guaranteed, so copy onto stack */
226 18 : memcpy(&xl_origin, data, sizeof(xl_origin));
227 :
228 18 : parsed->origin_lsn = xl_origin.origin_lsn;
229 18 : parsed->origin_timestamp = xl_origin.origin_timestamp;
230 :
231 18 : data += sizeof(xl_xact_origin);
232 : }
233 3632 : }
234 :
235 : /*
236 : * ParsePrepareRecord
237 : */
238 : void
239 312 : ParsePrepareRecord(uint8 info, xl_xact_prepare *xlrec, xl_xact_parsed_prepare *parsed)
240 : {
241 : char *bufptr;
242 :
243 312 : bufptr = ((char *) xlrec) + MAXALIGN(sizeof(xl_xact_prepare));
244 :
245 312 : memset(parsed, 0, sizeof(*parsed));
246 :
247 312 : parsed->xact_time = xlrec->prepared_at;
248 312 : parsed->origin_lsn = xlrec->origin_lsn;
249 312 : parsed->origin_timestamp = xlrec->origin_timestamp;
250 312 : parsed->twophase_xid = xlrec->xid;
251 312 : parsed->dbId = xlrec->database;
252 312 : parsed->nsubxacts = xlrec->nsubxacts;
253 312 : parsed->nrels = xlrec->ncommitrels;
254 312 : parsed->nabortrels = xlrec->nabortrels;
255 312 : parsed->nmsgs = xlrec->ninvalmsgs;
256 :
257 312 : strncpy(parsed->twophase_gid, bufptr, xlrec->gidlen);
258 312 : bufptr += MAXALIGN(xlrec->gidlen);
259 :
260 312 : parsed->subxacts = (TransactionId *) bufptr;
261 312 : bufptr += MAXALIGN(xlrec->nsubxacts * sizeof(TransactionId));
262 :
263 312 : parsed->xlocators = (RelFileLocator *) bufptr;
264 312 : bufptr += MAXALIGN(xlrec->ncommitrels * sizeof(RelFileLocator));
265 :
266 312 : parsed->abortlocators = (RelFileLocator *) bufptr;
267 312 : bufptr += MAXALIGN(xlrec->nabortrels * sizeof(RelFileLocator));
268 :
269 312 : parsed->stats = (xl_xact_stats_item *) bufptr;
270 312 : bufptr += MAXALIGN(xlrec->ncommitstats * sizeof(xl_xact_stats_item));
271 :
272 312 : parsed->abortstats = (xl_xact_stats_item *) bufptr;
273 312 : bufptr += MAXALIGN(xlrec->nabortstats * sizeof(xl_xact_stats_item));
274 :
275 312 : parsed->msgs = (SharedInvalidationMessage *) bufptr;
276 312 : bufptr += MAXALIGN(xlrec->ninvalmsgs * sizeof(SharedInvalidationMessage));
277 312 : }
278 :
279 : static void
280 48 : xact_desc_relations(StringInfo buf, char *label, int nrels,
281 : RelFileLocator *xlocators)
282 : {
283 : int i;
284 :
285 48 : if (nrels > 0)
286 : {
287 6 : appendStringInfo(buf, "; %s:", label);
288 22 : for (i = 0; i < nrels; i++)
289 : {
290 16 : char *path = relpathperm(xlocators[i], MAIN_FORKNUM);
291 :
292 16 : appendStringInfo(buf, " %s", path);
293 16 : pfree(path);
294 : }
295 : }
296 48 : }
297 :
298 : static void
299 48 : xact_desc_subxacts(StringInfo buf, int nsubxacts, TransactionId *subxacts)
300 : {
301 : int i;
302 :
303 48 : if (nsubxacts > 0)
304 : {
305 0 : appendStringInfoString(buf, "; subxacts:");
306 0 : for (i = 0; i < nsubxacts; i++)
307 0 : appendStringInfo(buf, " %u", subxacts[i]);
308 : }
309 48 : }
310 :
311 : static void
312 48 : xact_desc_stats(StringInfo buf, const char *label,
313 : int ndropped, xl_xact_stats_item *dropped_stats)
314 : {
315 : int i;
316 :
317 48 : if (ndropped > 0)
318 : {
319 12 : appendStringInfo(buf, "; %sdropped stats:", label);
320 30 : for (i = 0; i < ndropped; i++)
321 : {
322 18 : uint64 objid =
323 18 : ((uint64) dropped_stats[i].objid_hi) << 32 | dropped_stats[i].objid_lo;
324 :
325 18 : appendStringInfo(buf, " %d/%u/%llu",
326 18 : dropped_stats[i].kind,
327 18 : dropped_stats[i].dboid,
328 : (unsigned long long) objid);
329 : }
330 : }
331 48 : }
332 :
333 : static void
334 48 : xact_desc_commit(StringInfo buf, uint8 info, xl_xact_commit *xlrec, RepOriginId origin_id)
335 : {
336 : xl_xact_parsed_commit parsed;
337 :
338 48 : ParseCommitRecord(info, xlrec, &parsed);
339 :
340 : /* If this is a prepared xact, show the xid of the original xact */
341 48 : if (TransactionIdIsValid(parsed.twophase_xid))
342 0 : appendStringInfo(buf, "%u: ", parsed.twophase_xid);
343 :
344 48 : appendStringInfoString(buf, timestamptz_to_str(xlrec->xact_time));
345 :
346 48 : xact_desc_relations(buf, "rels", parsed.nrels, parsed.xlocators);
347 48 : xact_desc_subxacts(buf, parsed.nsubxacts, parsed.subxacts);
348 48 : xact_desc_stats(buf, "", parsed.nstats, parsed.stats);
349 :
350 48 : standby_desc_invalidations(buf, parsed.nmsgs, parsed.msgs, parsed.dbId,
351 : parsed.tsId,
352 48 : XactCompletionRelcacheInitFileInval(parsed.xinfo));
353 :
354 48 : if (XactCompletionApplyFeedback(parsed.xinfo))
355 0 : appendStringInfoString(buf, "; apply_feedback");
356 :
357 48 : if (XactCompletionForceSyncCommit(parsed.xinfo))
358 22 : appendStringInfoString(buf, "; sync");
359 :
360 48 : if (parsed.xinfo & XACT_XINFO_HAS_ORIGIN)
361 : {
362 0 : appendStringInfo(buf, "; origin: node %u, lsn %X/%X, at %s",
363 : origin_id,
364 0 : LSN_FORMAT_ARGS(parsed.origin_lsn),
365 : timestamptz_to_str(parsed.origin_timestamp));
366 : }
367 48 : }
368 :
369 : static void
370 0 : xact_desc_abort(StringInfo buf, uint8 info, xl_xact_abort *xlrec, RepOriginId origin_id)
371 : {
372 : xl_xact_parsed_abort parsed;
373 :
374 0 : ParseAbortRecord(info, xlrec, &parsed);
375 :
376 : /* If this is a prepared xact, show the xid of the original xact */
377 0 : if (TransactionIdIsValid(parsed.twophase_xid))
378 0 : appendStringInfo(buf, "%u: ", parsed.twophase_xid);
379 :
380 0 : appendStringInfoString(buf, timestamptz_to_str(xlrec->xact_time));
381 :
382 0 : xact_desc_relations(buf, "rels", parsed.nrels, parsed.xlocators);
383 0 : xact_desc_subxacts(buf, parsed.nsubxacts, parsed.subxacts);
384 :
385 0 : if (parsed.xinfo & XACT_XINFO_HAS_ORIGIN)
386 : {
387 0 : appendStringInfo(buf, "; origin: node %u, lsn %X/%X, at %s",
388 : origin_id,
389 0 : LSN_FORMAT_ARGS(parsed.origin_lsn),
390 : timestamptz_to_str(parsed.origin_timestamp));
391 : }
392 :
393 0 : xact_desc_stats(buf, "", parsed.nstats, parsed.stats);
394 0 : }
395 :
396 : static void
397 0 : xact_desc_prepare(StringInfo buf, uint8 info, xl_xact_prepare *xlrec, RepOriginId origin_id)
398 : {
399 : xl_xact_parsed_prepare parsed;
400 :
401 0 : ParsePrepareRecord(info, xlrec, &parsed);
402 :
403 0 : appendStringInfo(buf, "gid %s: ", parsed.twophase_gid);
404 0 : appendStringInfoString(buf, timestamptz_to_str(parsed.xact_time));
405 :
406 0 : xact_desc_relations(buf, "rels(commit)", parsed.nrels, parsed.xlocators);
407 0 : xact_desc_relations(buf, "rels(abort)", parsed.nabortrels,
408 : parsed.abortlocators);
409 0 : xact_desc_stats(buf, "commit ", parsed.nstats, parsed.stats);
410 0 : xact_desc_stats(buf, "abort ", parsed.nabortstats, parsed.abortstats);
411 0 : xact_desc_subxacts(buf, parsed.nsubxacts, parsed.subxacts);
412 :
413 0 : standby_desc_invalidations(buf, parsed.nmsgs, parsed.msgs, parsed.dbId,
414 0 : parsed.tsId, xlrec->initfileinval);
415 :
416 : /*
417 : * Check if the replication origin has been set in this record in the same
418 : * way as PrepareRedoAdd().
419 : */
420 0 : if (origin_id != InvalidRepOriginId)
421 0 : appendStringInfo(buf, "; origin: node %u, lsn %X/%X, at %s",
422 : origin_id,
423 0 : LSN_FORMAT_ARGS(parsed.origin_lsn),
424 : timestamptz_to_str(parsed.origin_timestamp));
425 0 : }
426 :
427 : static void
428 0 : xact_desc_assignment(StringInfo buf, xl_xact_assignment *xlrec)
429 : {
430 : int i;
431 :
432 0 : appendStringInfoString(buf, "subxacts:");
433 :
434 0 : for (i = 0; i < xlrec->nsubxacts; i++)
435 0 : appendStringInfo(buf, " %u", xlrec->xsub[i]);
436 0 : }
437 :
438 : void
439 48 : xact_desc(StringInfo buf, XLogReaderState *record)
440 : {
441 48 : char *rec = XLogRecGetData(record);
442 48 : uint8 info = XLogRecGetInfo(record) & XLOG_XACT_OPMASK;
443 :
444 48 : if (info == XLOG_XACT_COMMIT || info == XLOG_XACT_COMMIT_PREPARED)
445 48 : {
446 48 : xl_xact_commit *xlrec = (xl_xact_commit *) rec;
447 :
448 48 : xact_desc_commit(buf, XLogRecGetInfo(record), xlrec,
449 48 : XLogRecGetOrigin(record));
450 : }
451 0 : else if (info == XLOG_XACT_ABORT || info == XLOG_XACT_ABORT_PREPARED)
452 0 : {
453 0 : xl_xact_abort *xlrec = (xl_xact_abort *) rec;
454 :
455 0 : xact_desc_abort(buf, XLogRecGetInfo(record), xlrec,
456 0 : XLogRecGetOrigin(record));
457 : }
458 0 : else if (info == XLOG_XACT_PREPARE)
459 : {
460 0 : xl_xact_prepare *xlrec = (xl_xact_prepare *) rec;
461 :
462 0 : xact_desc_prepare(buf, XLogRecGetInfo(record), xlrec,
463 0 : XLogRecGetOrigin(record));
464 : }
465 0 : else if (info == XLOG_XACT_ASSIGNMENT)
466 : {
467 0 : xl_xact_assignment *xlrec = (xl_xact_assignment *) rec;
468 :
469 : /*
470 : * Note that we ignore the WAL record's xid, since we're more
471 : * interested in the top-level xid that issued the record and which
472 : * xids are being reported here.
473 : */
474 0 : appendStringInfo(buf, "xtop %u: ", xlrec->xtop);
475 0 : xact_desc_assignment(buf, xlrec);
476 : }
477 0 : else if (info == XLOG_XACT_INVALIDATIONS)
478 : {
479 0 : xl_xact_invals *xlrec = (xl_xact_invals *) rec;
480 :
481 0 : standby_desc_invalidations(buf, xlrec->nmsgs, xlrec->msgs, InvalidOid,
482 : InvalidOid, false);
483 : }
484 48 : }
485 :
486 : const char *
487 48 : xact_identify(uint8 info)
488 : {
489 48 : const char *id = NULL;
490 :
491 48 : switch (info & XLOG_XACT_OPMASK)
492 : {
493 48 : case XLOG_XACT_COMMIT:
494 48 : id = "COMMIT";
495 48 : break;
496 0 : case XLOG_XACT_PREPARE:
497 0 : id = "PREPARE";
498 0 : break;
499 0 : case XLOG_XACT_ABORT:
500 0 : id = "ABORT";
501 0 : break;
502 0 : case XLOG_XACT_COMMIT_PREPARED:
503 0 : id = "COMMIT_PREPARED";
504 0 : break;
505 0 : case XLOG_XACT_ABORT_PREPARED:
506 0 : id = "ABORT_PREPARED";
507 0 : break;
508 0 : case XLOG_XACT_ASSIGNMENT:
509 0 : id = "ASSIGNMENT";
510 0 : break;
511 0 : case XLOG_XACT_INVALIDATIONS:
512 0 : id = "INVALIDATION";
513 0 : break;
514 : }
515 :
516 48 : return id;
517 : }
|