Age Owner Branch data TLA Line data Source code
1 : : /*-------------------------------------------------------------------------
2 : : *
3 : : * transam.h
4 : : * postgres transaction access method support code
5 : : *
6 : : *
7 : : * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
8 : : * Portions Copyright (c) 1994, Regents of the University of California
9 : : *
10 : : * src/include/access/transam.h
11 : : *
12 : : *-------------------------------------------------------------------------
13 : : */
14 : : #ifndef TRANSAM_H
15 : : #define TRANSAM_H
16 : :
17 : : #include "access/xlogdefs.h"
18 : :
19 : :
20 : : /* ----------------
21 : : * Special transaction ID values
22 : : *
23 : : * BootstrapTransactionId is the XID for "bootstrap" operations, and
24 : : * FrozenTransactionId is used for very old tuples. Both should
25 : : * always be considered valid.
26 : : *
27 : : * FirstNormalTransactionId is the first "normal" transaction id.
28 : : * Note: if you need to change it, you must change pg_class.h as well.
29 : : * ----------------
30 : : */
31 : : #define InvalidTransactionId ((TransactionId) 0)
32 : : #define BootstrapTransactionId ((TransactionId) 1)
33 : : #define FrozenTransactionId ((TransactionId) 2)
34 : : #define FirstNormalTransactionId ((TransactionId) 3)
35 : : #define MaxTransactionId ((TransactionId) 0xFFFFFFFF)
36 : :
37 : : /* ----------------
38 : : * transaction ID manipulation macros
39 : : * ----------------
40 : : */
41 : : #define TransactionIdIsValid(xid) ((xid) != InvalidTransactionId)
42 : : #define TransactionIdIsNormal(xid) ((xid) >= FirstNormalTransactionId)
43 : : #define TransactionIdEquals(id1, id2) ((id1) == (id2))
44 : : #define TransactionIdStore(xid, dest) (*(dest) = (xid))
45 : : #define StoreInvalidTransactionId(dest) (*(dest) = InvalidTransactionId)
46 : :
47 : : #define EpochFromFullTransactionId(x) ((uint32) ((x).value >> 32))
48 : : #define XidFromFullTransactionId(x) ((uint32) (x).value)
49 : : #define U64FromFullTransactionId(x) ((x).value)
50 : : #define FullTransactionIdEquals(a, b) ((a).value == (b).value)
51 : : #define FullTransactionIdPrecedes(a, b) ((a).value < (b).value)
52 : : #define FullTransactionIdPrecedesOrEquals(a, b) ((a).value <= (b).value)
53 : : #define FullTransactionIdFollows(a, b) ((a).value > (b).value)
54 : : #define FullTransactionIdFollowsOrEquals(a, b) ((a).value >= (b).value)
55 : : #define FullTransactionIdIsValid(x) TransactionIdIsValid(XidFromFullTransactionId(x))
56 : : #define InvalidFullTransactionId FullTransactionIdFromEpochAndXid(0, InvalidTransactionId)
57 : : #define FirstNormalFullTransactionId FullTransactionIdFromEpochAndXid(0, FirstNormalTransactionId)
58 : : #define FullTransactionIdIsNormal(x) FullTransactionIdFollowsOrEquals(x, FirstNormalFullTransactionId)
59 : :
60 : : /*
61 : : * A 64 bit value that contains an epoch and a TransactionId. This is
62 : : * wrapped in a struct to prevent implicit conversion to/from TransactionId.
63 : : * Not all values represent valid normal XIDs.
64 : : */
65 : : typedef struct FullTransactionId
66 : : {
67 : : uint64 value;
68 : : } FullTransactionId;
69 : :
70 : : static inline FullTransactionId
2405 tmunro@postgresql.or 71 :CBC 1756796 : FullTransactionIdFromEpochAndXid(uint32 epoch, TransactionId xid)
72 : : {
73 : : FullTransactionId result;
74 : :
75 : 1756796 : result.value = ((uint64) epoch) << 32 | xid;
76 : :
77 : 1756796 : return result;
78 : : }
79 : :
80 : : static inline FullTransactionId
2029 81 : 11141942 : FullTransactionIdFromU64(uint64 value)
82 : : {
83 : : FullTransactionId result;
84 : :
85 : 11141942 : result.value = value;
86 : :
87 : 11141942 : return result;
88 : : }
89 : :
90 : : /* advance a transaction ID variable, handling wraparound correctly */
91 : : #define TransactionIdAdvance(dest) \
92 : : do { \
93 : : (dest)++; \
94 : : if ((dest) < FirstNormalTransactionId) \
95 : : (dest) = FirstNormalTransactionId; \
96 : : } while(0)
97 : :
98 : : /*
99 : : * Retreat a FullTransactionId variable, stepping over xids that would appear
100 : : * to be special only when viewed as 32bit XIDs.
101 : : */
102 : : static inline void
1903 andres@anarazel.de 103 : 953 : FullTransactionIdRetreat(FullTransactionId *dest)
104 : : {
105 : 953 : dest->value--;
106 : :
107 : : /*
108 : : * In contrast to 32bit XIDs don't step over the "actual" special xids.
109 : : * For 64bit xids these can't be reached as part of a wraparound as they
110 : : * can in the 32bit case.
111 : : */
112 [ + + ]: 953 : if (FullTransactionIdPrecedes(*dest, FirstNormalFullTransactionId))
113 : 100 : return;
114 : :
115 : : /*
116 : : * But we do need to step over XIDs that'd appear special only for 32bit
117 : : * XIDs.
118 : : */
119 [ - + ]: 853 : while (XidFromFullTransactionId(*dest) < FirstNormalTransactionId)
1903 andres@anarazel.de 120 :UBC 0 : dest->value--;
121 : : }
122 : :
123 : : /*
124 : : * Advance a FullTransactionId variable, stepping over xids that would appear
125 : : * to be special only when viewed as 32bit XIDs.
126 : : */
127 : : static inline void
1902 andres@anarazel.de 128 :CBC 459482 : FullTransactionIdAdvance(FullTransactionId *dest)
129 : : {
130 : 459482 : dest->value++;
131 : :
132 : : /* see FullTransactionIdAdvance() */
133 [ - + ]: 459482 : if (FullTransactionIdPrecedes(*dest, FirstNormalFullTransactionId))
1902 andres@anarazel.de 134 :UBC 0 : return;
135 : :
1902 andres@anarazel.de 136 [ - + ]:CBC 459482 : while (XidFromFullTransactionId(*dest) < FirstNormalTransactionId)
1902 andres@anarazel.de 137 :UBC 0 : dest->value++;
138 : : }
139 : :
140 : : /* back up a transaction ID variable, handling wraparound correctly */
141 : : #define TransactionIdRetreat(dest) \
142 : : do { \
143 : : (dest)--; \
144 : : } while ((dest) < FirstNormalTransactionId)
145 : :
146 : : /* compare two XIDs already known to be normal; this is a macro for speed */
147 : : #define NormalTransactionIdPrecedes(id1, id2) \
148 : : (AssertMacro(TransactionIdIsNormal(id1) && TransactionIdIsNormal(id2)), \
149 : : (int32) ((id1) - (id2)) < 0)
150 : :
151 : : /* compare two XIDs already known to be normal; this is a macro for speed */
152 : : #define NormalTransactionIdFollows(id1, id2) \
153 : : (AssertMacro(TransactionIdIsNormal(id1) && TransactionIdIsNormal(id2)), \
154 : : (int32) ((id1) - (id2)) > 0)
155 : :
156 : : /* ----------
157 : : * Object ID (OID) zero is InvalidOid.
158 : : *
159 : : * OIDs 1-9999 are reserved for manual assignment (see .dat files in
160 : : * src/include/catalog/). Of these, 8000-9999 are reserved for
161 : : * development purposes (such as in-progress patches and forks);
162 : : * they should not appear in released versions.
163 : : *
164 : : * OIDs 10000-11999 are reserved for assignment by genbki.pl, for use
165 : : * when the .dat files in src/include/catalog/ do not specify an OID
166 : : * for a catalog entry that requires one. Note that genbki.pl assigns
167 : : * these OIDs independently in each catalog, so they're not guaranteed
168 : : * to be globally unique. Furthermore, the bootstrap backend and
169 : : * initdb's post-bootstrap processing can also assign OIDs in this range.
170 : : * The normal OID-generation logic takes care of any OID conflicts that
171 : : * might arise from that.
172 : : *
173 : : * OIDs 12000-16383 are reserved for unpinned objects created by initdb's
174 : : * post-bootstrap processing. initdb forces the OID generator up to
175 : : * 12000 as soon as it's made the pinned objects it's responsible for.
176 : : *
177 : : * OIDs beginning at 16384 are assigned from the OID generator
178 : : * during normal multiuser operation. (We force the generator up to
179 : : * 16384 as soon as we are in normal operation.)
180 : : *
181 : : * The choices of 8000, 10000 and 12000 are completely arbitrary, and can be
182 : : * moved if we run low on OIDs in any category. Changing the macros below,
183 : : * and updating relevant documentation (see bki.sgml and RELEASE_CHANGES),
184 : : * should be sufficient to do this. Moving the 16384 boundary between
185 : : * initdb-assigned OIDs and user-defined objects would be substantially
186 : : * more painful, however, since some user-defined OIDs will appear in
187 : : * on-disk data; such a change would probably break pg_upgrade.
188 : : *
189 : : * NOTE: if the OID generator wraps around, we skip over OIDs 0-16383
190 : : * and resume with 16384. This minimizes the odds of OID conflict, by not
191 : : * reassigning OIDs that might have been assigned during initdb. Critically,
192 : : * it also ensures that no user-created object will be considered pinned.
193 : : * ----------
194 : : */
195 : : #define FirstGenbkiObjectId 10000
196 : : #define FirstUnpinnedObjectId 12000
197 : : #define FirstNormalObjectId 16384
198 : :
199 : : /*
200 : : * TransamVariables is a data structure in shared memory that is used to track
201 : : * OID and XID assignment state. For largely historical reasons, there is
202 : : * just one struct with different fields that are protected by different
203 : : * LWLocks.
204 : : *
205 : : * Note: xidWrapLimit and oldestXidDB are not "active" values, but are
206 : : * used just to generate useful messages when xidWarnLimit or xidStopLimit
207 : : * are exceeded.
208 : : */
209 : : typedef struct TransamVariablesData
210 : : {
211 : : /*
212 : : * These fields are protected by OidGenLock.
213 : : */
214 : : Oid nextOid; /* next OID to assign */
215 : : uint32 oidCount; /* OIDs available before must do XLOG work */
216 : :
217 : : /*
218 : : * These fields are protected by XidGenLock.
219 : : */
220 : : FullTransactionId nextXid; /* next XID to assign */
221 : :
222 : : TransactionId oldestXid; /* cluster-wide minimum datfrozenxid */
223 : : TransactionId xidVacLimit; /* start forcing autovacuums here */
224 : : TransactionId xidWarnLimit; /* start complaining here */
225 : : TransactionId xidStopLimit; /* refuse to advance nextXid beyond here */
226 : : TransactionId xidWrapLimit; /* where the world ends */
227 : : Oid oldestXidDB; /* database with minimum datfrozenxid */
228 : :
229 : : /*
230 : : * These fields are protected by CommitTsLock
231 : : */
232 : : TransactionId oldestCommitTsXid;
233 : : TransactionId newestCommitTsXid;
234 : :
235 : : /*
236 : : * These fields are protected by ProcArrayLock.
237 : : */
238 : : FullTransactionId latestCompletedXid; /* newest full XID that has
239 : : * committed or aborted */
240 : :
241 : : /*
242 : : * Number of top-level transactions with xids (i.e. which may have
243 : : * modified the database) that completed in some form since the start of
244 : : * the server. This currently is solely used to check whether
245 : : * GetSnapshotData() needs to recompute the contents of the snapshot, or
246 : : * not. There are likely other users of this. Always above 1.
247 : : */
248 : : uint64 xactCompletionCount;
249 : :
250 : : /*
251 : : * These fields are protected by XactTruncationLock
252 : : */
253 : : TransactionId oldestClogXid; /* oldest it's safe to look up in clog */
254 : :
255 : : } TransamVariablesData;
256 : :
257 : :
258 : :
259 : : /*
260 : : * TransactionIdPrecedes --- is id1 logically < id2?
261 : : */
262 : : static inline bool
13 melanieplageman@gmai 263 :GNC 107061101 : TransactionIdPrecedes(TransactionId id1, TransactionId id2)
264 : : {
265 : : /*
266 : : * If either ID is a permanent XID then we can just do unsigned
267 : : * comparison. If both are normal, do a modulo-2^32 comparison.
268 : : */
269 : : int32 diff;
270 : :
271 [ + + + + ]: 107061101 : if (!TransactionIdIsNormal(id1) || !TransactionIdIsNormal(id2))
272 : 37640358 : return (id1 < id2);
273 : :
274 : 69420743 : diff = (int32) (id1 - id2);
275 : 69420743 : return (diff < 0);
276 : : }
277 : :
278 : : /*
279 : : * TransactionIdPrecedesOrEquals --- is id1 logically <= id2?
280 : : */
281 : : static inline bool
282 : 4942905 : TransactionIdPrecedesOrEquals(TransactionId id1, TransactionId id2)
283 : : {
284 : : int32 diff;
285 : :
286 [ + + + + ]: 4942905 : if (!TransactionIdIsNormal(id1) || !TransactionIdIsNormal(id2))
287 : 857 : return (id1 <= id2);
288 : :
289 : 4942048 : diff = (int32) (id1 - id2);
290 : 4942048 : return (diff <= 0);
291 : : }
292 : :
293 : : /*
294 : : * TransactionIdFollows --- is id1 logically > id2?
295 : : */
296 : : static inline bool
297 : 17659003 : TransactionIdFollows(TransactionId id1, TransactionId id2)
298 : : {
299 : : int32 diff;
300 : :
301 [ + + + + ]: 17659003 : if (!TransactionIdIsNormal(id1) || !TransactionIdIsNormal(id2))
302 : 3656613 : return (id1 > id2);
303 : :
304 : 14002390 : diff = (int32) (id1 - id2);
305 : 14002390 : return (diff > 0);
306 : : }
307 : :
308 : : /*
309 : : * TransactionIdFollowsOrEquals --- is id1 logically >= id2?
310 : : */
311 : : static inline bool
312 : 18434446 : TransactionIdFollowsOrEquals(TransactionId id1, TransactionId id2)
313 : : {
314 : : int32 diff;
315 : :
316 [ + + - + ]: 18434446 : if (!TransactionIdIsNormal(id1) || !TransactionIdIsNormal(id2))
317 : 53188 : return (id1 >= id2);
318 : :
319 : 18381258 : diff = (int32) (id1 - id2);
320 : 18381258 : return (diff >= 0);
321 : : }
322 : :
323 : :
324 : : /* ----------------
325 : : * extern declarations
326 : : * ----------------
327 : : */
328 : :
329 : : /* in transam/xact.c */
330 : : extern bool TransactionStartedDuringRecovery(void);
331 : :
332 : : /* in transam/varsup.c */
333 : : extern PGDLLIMPORT TransamVariablesData *TransamVariables;
334 : :
335 : : /*
336 : : * prototypes for functions in transam/transam.c
337 : : */
338 : : extern bool TransactionIdDidCommit(TransactionId transactionId);
339 : : extern bool TransactionIdDidAbort(TransactionId transactionId);
340 : : extern void TransactionIdCommitTree(TransactionId xid, int nxids, TransactionId *xids);
341 : : extern void TransactionIdAsyncCommitTree(TransactionId xid, int nxids, TransactionId *xids, XLogRecPtr lsn);
342 : : extern void TransactionIdAbortTree(TransactionId xid, int nxids, TransactionId *xids);
343 : : extern TransactionId TransactionIdLatest(TransactionId mainxid,
344 : : int nxids, const TransactionId *xids);
345 : : extern XLogRecPtr TransactionIdGetCommitLSN(TransactionId xid);
346 : :
347 : : /* in transam/varsup.c */
348 : : extern Size VarsupShmemSize(void);
349 : : extern void VarsupShmemInit(void);
350 : : extern FullTransactionId GetNewTransactionId(bool isSubXact);
351 : : extern void AdvanceNextFullTransactionIdPastXid(TransactionId xid);
352 : : extern FullTransactionId ReadNextFullTransactionId(void);
353 : : extern void SetTransactionIdLimit(TransactionId oldest_datfrozenxid,
354 : : Oid oldest_datoid);
355 : : extern void AdvanceOldestClogXid(TransactionId oldest_datfrozenxid);
356 : : extern bool ForceTransactionIdLimitUpdate(void);
357 : : extern Oid GetNewObjectId(void);
358 : : extern void StopGeneratingPinnedObjectIds(void);
359 : :
360 : : #ifdef USE_ASSERT_CHECKING
361 : : extern void AssertTransactionIdInAllowableRange(TransactionId xid);
362 : : #else
363 : : #define AssertTransactionIdInAllowableRange(xid) ((void)true)
364 : : #endif
365 : :
366 : : /*
367 : : * Some frontend programs include this header. For compilers that emit static
368 : : * inline functions even when they're unused, that leads to unsatisfied
369 : : * external references; hence hide them with #ifndef FRONTEND.
370 : : */
371 : : #ifndef FRONTEND
372 : :
373 : : /*
374 : : * For callers that just need the XID part of the next transaction ID.
375 : : */
376 : : static inline TransactionId
1715 tmunro@postgresql.or 377 :CBC 29706 : ReadNextTransactionId(void)
378 : : {
2405 379 : 29706 : return XidFromFullTransactionId(ReadNextFullTransactionId());
380 : : }
381 : :
382 : : /* return transaction ID backed up by amount, handling wraparound correctly */
383 : : static inline TransactionId
384 : : TransactionIdRetreatedBy(TransactionId xid, uint32 amount)
385 : : {
386 : : xid -= amount;
387 : :
388 : : while (xid < FirstNormalTransactionId)
389 : : xid--;
390 : :
391 : : return xid;
392 : : }
393 : :
394 : : /* return the older of the two IDs */
395 : : static inline TransactionId
1902 andres@anarazel.de 396 : 2484867 : TransactionIdOlder(TransactionId a, TransactionId b)
397 : : {
398 [ + + ]: 2484867 : if (!TransactionIdIsValid(a))
399 : 610315 : return b;
400 : :
401 [ + + ]: 1874552 : if (!TransactionIdIsValid(b))
402 : 885045 : return a;
403 : :
404 [ + + ]: 989507 : if (TransactionIdPrecedes(a, b))
405 : 238838 : return a;
406 : 750669 : return b;
407 : : }
408 : :
409 : : /* return the older of the two IDs, assuming they're both normal */
410 : : static inline TransactionId
411 : : NormalTransactionIdOlder(TransactionId a, TransactionId b)
412 : : {
413 : : Assert(TransactionIdIsNormal(a));
414 : : Assert(TransactionIdIsNormal(b));
415 : : if (NormalTransactionIdPrecedes(a, b))
416 : : return a;
417 : : return b;
418 : : }
419 : :
420 : : /* return the newer of the two IDs */
421 : : static inline FullTransactionId
422 : 2577510 : FullTransactionIdNewer(FullTransactionId a, FullTransactionId b)
423 : : {
424 [ + + ]: 2577510 : if (!FullTransactionIdIsValid(a))
425 : 45066 : return b;
426 : :
427 [ + + ]: 2532444 : if (!FullTransactionIdIsValid(b))
428 : 46587 : return a;
429 : :
430 [ + + ]: 2485857 : if (FullTransactionIdFollows(a, b))
431 : 1379004 : return a;
432 : 1106853 : return b;
433 : : }
434 : :
435 : : /*
436 : : * Compute FullTransactionId for the given TransactionId, assuming xid was
437 : : * between [oldestXid, nextXid] at the time when TransamVariables->nextXid was
438 : : * nextFullXid. When adding calls, evaluate what prevents xid from preceding
439 : : * oldestXid if SetTransactionIdLimit() runs between the collection of xid and
440 : : * the collection of nextFullXid.
441 : : */
442 : : static inline FullTransactionId
275 noah@leadboat.com 443 : 659 : FullTransactionIdFromAllowableAt(FullTransactionId nextFullXid,
444 : : TransactionId xid)
445 : : {
446 : : uint32 epoch;
447 : :
448 : : /* Special transaction ID. */
449 [ - + ]: 659 : if (!TransactionIdIsNormal(xid))
275 noah@leadboat.com 450 :UBC 0 : return FullTransactionIdFromEpochAndXid(0, xid);
451 : :
275 noah@leadboat.com 452 [ - + ]:CBC 659 : Assert(TransactionIdPrecedesOrEquals(xid,
453 : : XidFromFullTransactionId(nextFullXid)));
454 : :
455 : : /*
456 : : * The 64 bit result must be <= nextFullXid, since nextFullXid hadn't been
457 : : * issued yet when xid was in the past. The xid must therefore be from
458 : : * the epoch of nextFullXid or the epoch before. We know this because we
459 : : * must remove (by freezing) an XID before assigning the XID half an epoch
460 : : * ahead of it.
461 : : *
462 : : * The unlikely() branch hint is dubious. It's perfect for the first 2^32
463 : : * XIDs of a cluster's life. Right at 2^32 XIDs, misprediction shoots to
464 : : * 100%, then improves until perfection returns 2^31 XIDs later. Since
465 : : * current callers pass relatively-recent XIDs, expect >90% prediction
466 : : * accuracy overall. This favors average latency over tail latency.
467 : : */
468 : 659 : epoch = EpochFromFullTransactionId(nextFullXid);
469 [ - + ]: 659 : if (unlikely(xid > XidFromFullTransactionId(nextFullXid)))
470 : : {
275 noah@leadboat.com 471 [ # # ]:UBC 0 : Assert(epoch != 0);
472 : 0 : epoch--;
473 : : }
474 : :
275 noah@leadboat.com 475 :CBC 659 : return FullTransactionIdFromEpochAndXid(epoch, xid);
476 : : }
477 : :
478 : : #endif /* FRONTEND */
479 : :
480 : : #endif /* TRANSAM_H */
|