LCOV - code coverage report
Current view: top level - src/include/access - transam.h (source / functions) Coverage Total Hit
Test: Code coverage Lines: 88.8 % 80 71
Test Date: 2026-01-26 10:56:24 Functions: 100.0 % 12 12
Legend: Lines:     hit not hit
Branches: + taken - not taken # not executed
Branches: 70.5 % 44 31

             Branch data     Line data    Source code
       1                 :             : /*-------------------------------------------------------------------------
       2                 :             :  *
       3                 :             :  * transam.h
       4                 :             :  *        postgres transaction access method support code
       5                 :             :  *
       6                 :             :  *
       7                 :             :  * Portions Copyright (c) 1996-2026, 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
      71                 :      289923 : FullTransactionIdFromEpochAndXid(uint32 epoch, TransactionId xid)
      72                 :             : {
      73                 :             :         FullTransactionId result;
      74                 :             : 
      75                 :      289923 :         result.value = ((uint64) epoch) << 32 | xid;
      76                 :             : 
      77                 :      289923 :         return result;
      78                 :             : }
      79                 :             : 
      80                 :             : static inline FullTransactionId
      81                 :         961 : FullTransactionIdFromU64(uint64 value)
      82                 :             : {
      83                 :             :         FullTransactionId result;
      84                 :             : 
      85                 :         961 :         result.value = value;
      86                 :             : 
      87                 :         961 :         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
     103                 :           4 : FullTransactionIdRetreat(FullTransactionId *dest)
     104                 :             : {
     105                 :           4 :         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         [ +  + ]:           4 :         if (FullTransactionIdPrecedes(*dest, FirstNormalFullTransactionId))
     113                 :           2 :                 return;
     114                 :             : 
     115                 :             :         /*
     116                 :             :          * But we do need to step over XIDs that'd appear special only for 32bit
     117                 :             :          * XIDs.
     118                 :             :          */
     119         [ +  - ]:           2 :         while (XidFromFullTransactionId(*dest) < FirstNormalTransactionId)
     120                 :           0 :                 dest->value--;
     121                 :           4 : }
     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
     128                 :       92492 : FullTransactionIdAdvance(FullTransactionId *dest)
     129                 :             : {
     130                 :       92492 :         dest->value++;
     131                 :             : 
     132                 :             :         /* see FullTransactionIdAdvance() */
     133         [ -  + ]:       92492 :         if (FullTransactionIdPrecedes(*dest, FirstNormalFullTransactionId))
     134                 :           0 :                 return;
     135                 :             : 
     136         [ +  - ]:       92492 :         while (XidFromFullTransactionId(*dest) < FirstNormalTransactionId)
     137                 :           0 :                 dest->value++;
     138                 :       92492 : }
     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
     263                 :    21909301 : 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                 :    21909301 :         int32           diff;
     270                 :             : 
     271   [ +  +  -  + ]:    21909301 :         if (!TransactionIdIsNormal(id1) || !TransactionIdIsNormal(id2))
     272                 :     1538628 :                 return (id1 < id2);
     273                 :             : 
     274                 :    20370673 :         diff = (int32) (id1 - id2);
     275                 :    20370673 :         return (diff < 0);
     276                 :    21909301 : }
     277                 :             : 
     278                 :             : /*
     279                 :             :  * TransactionIdPrecedesOrEquals --- is id1 logically <= id2?
     280                 :             :  */
     281                 :             : static inline bool
     282                 :     1818768 : TransactionIdPrecedesOrEquals(TransactionId id1, TransactionId id2)
     283                 :             : {
     284                 :     1818768 :         int32           diff;
     285                 :             : 
     286   [ +  -  -  + ]:     1818768 :         if (!TransactionIdIsNormal(id1) || !TransactionIdIsNormal(id2))
     287                 :           0 :                 return (id1 <= id2);
     288                 :             : 
     289                 :     1818768 :         diff = (int32) (id1 - id2);
     290                 :     1818768 :         return (diff <= 0);
     291                 :     1818768 : }
     292                 :             : 
     293                 :             : /*
     294                 :             :  * TransactionIdFollows --- is id1 logically > id2?
     295                 :             :  */
     296                 :             : static inline bool
     297                 :     2847247 : TransactionIdFollows(TransactionId id1, TransactionId id2)
     298                 :             : {
     299                 :     2847247 :         int32           diff;
     300                 :             : 
     301   [ +  +  +  + ]:     2847247 :         if (!TransactionIdIsNormal(id1) || !TransactionIdIsNormal(id2))
     302                 :      170570 :                 return (id1 > id2);
     303                 :             : 
     304                 :     2676677 :         diff = (int32) (id1 - id2);
     305                 :     2676677 :         return (diff > 0);
     306                 :     2847247 : }
     307                 :             : 
     308                 :             : /*
     309                 :             :  * TransactionIdFollowsOrEquals --- is id1 logically >= id2?
     310                 :             :  */
     311                 :             : static inline bool
     312                 :     4807257 : TransactionIdFollowsOrEquals(TransactionId id1, TransactionId id2)
     313                 :             : {
     314                 :     4807257 :         int32           diff;
     315                 :             : 
     316   [ +  -  -  + ]:     4807257 :         if (!TransactionIdIsNormal(id1) || !TransactionIdIsNormal(id2))
     317                 :           0 :                 return (id1 >= id2);
     318                 :             : 
     319                 :     4807257 :         diff = (int32) (id1 - id2);
     320                 :     4807257 :         return (diff >= 0);
     321                 :     4807257 : }
     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
     377                 :        1725 : ReadNextTransactionId(void)
     378                 :             : {
     379                 :        1725 :         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
     396                 :      500677 : TransactionIdOlder(TransactionId a, TransactionId b)
     397                 :             : {
     398         [ +  + ]:      500677 :         if (!TransactionIdIsValid(a))
     399                 :      142910 :                 return b;
     400                 :             : 
     401         [ +  + ]:      357767 :         if (!TransactionIdIsValid(b))
     402                 :      166479 :                 return a;
     403                 :             : 
     404         [ +  + ]:      191288 :         if (TransactionIdPrecedes(a, b))
     405                 :       58308 :                 return a;
     406                 :      132980 :         return b;
     407                 :      500677 : }
     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                 :      553758 : FullTransactionIdNewer(FullTransactionId a, FullTransactionId b)
     423                 :             : {
     424         [ +  + ]:      553758 :         if (!FullTransactionIdIsValid(a))
     425                 :        2394 :                 return b;
     426                 :             : 
     427         [ +  + ]:      551364 :         if (!FullTransactionIdIsValid(b))
     428                 :        2397 :                 return a;
     429                 :             : 
     430         [ +  + ]:      548967 :         if (FullTransactionIdFollows(a, b))
     431                 :      305514 :                 return a;
     432                 :      243453 :         return b;
     433                 :      553758 : }
     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
     443                 :          20 : FullTransactionIdFromAllowableAt(FullTransactionId nextFullXid,
     444                 :             :                                                                  TransactionId xid)
     445                 :             : {
     446                 :          20 :         uint32          epoch;
     447                 :             : 
     448                 :             :         /* Special transaction ID. */
     449         [ -  + ]:          20 :         if (!TransactionIdIsNormal(xid))
     450                 :           0 :                 return FullTransactionIdFromEpochAndXid(0, xid);
     451                 :             : 
     452         [ +  - ]:          20 :         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                 :          20 :         epoch = EpochFromFullTransactionId(nextFullXid);
     469         [ +  - ]:          20 :         if (unlikely(xid > XidFromFullTransactionId(nextFullXid)))
     470                 :             :         {
     471         [ #  # ]:           0 :                 Assert(epoch != 0);
     472                 :           0 :                 epoch--;
     473                 :           0 :         }
     474                 :             : 
     475                 :          20 :         return FullTransactionIdFromEpochAndXid(epoch, xid);
     476                 :          20 : }
     477                 :             : 
     478                 :             : #endif                                                  /* FRONTEND */
     479                 :             : 
     480                 :             : #endif                                                  /* TRANSAM_H */
        

Generated by: LCOV version 2.3.2-1