LCOV - code coverage report
Current view: top level - src/backend/access/transam - xact.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 1596 1836 86.9 %
Date: 2025-01-18 04:15:08 Functions: 102 109 93.6 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * xact.c
       4             :  *    top level transaction system support routines
       5             :  *
       6             :  * See src/backend/access/transam/README for more information.
       7             :  *
       8             :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
       9             :  * Portions Copyright (c) 1994, Regents of the University of California
      10             :  *
      11             :  *
      12             :  * IDENTIFICATION
      13             :  *    src/backend/access/transam/xact.c
      14             :  *
      15             :  *-------------------------------------------------------------------------
      16             :  */
      17             : 
      18             : #include "postgres.h"
      19             : 
      20             : #include <time.h>
      21             : #include <unistd.h>
      22             : 
      23             : #include "access/commit_ts.h"
      24             : #include "access/multixact.h"
      25             : #include "access/parallel.h"
      26             : #include "access/subtrans.h"
      27             : #include "access/transam.h"
      28             : #include "access/twophase.h"
      29             : #include "access/xact.h"
      30             : #include "access/xlog.h"
      31             : #include "access/xloginsert.h"
      32             : #include "access/xlogrecovery.h"
      33             : #include "access/xlogutils.h"
      34             : #include "catalog/index.h"
      35             : #include "catalog/namespace.h"
      36             : #include "catalog/pg_enum.h"
      37             : #include "catalog/storage.h"
      38             : #include "commands/async.h"
      39             : #include "commands/tablecmds.h"
      40             : #include "commands/trigger.h"
      41             : #include "common/pg_prng.h"
      42             : #include "executor/spi.h"
      43             : #include "libpq/be-fsstubs.h"
      44             : #include "libpq/pqsignal.h"
      45             : #include "miscadmin.h"
      46             : #include "pg_trace.h"
      47             : #include "pgstat.h"
      48             : #include "replication/logical.h"
      49             : #include "replication/logicallauncher.h"
      50             : #include "replication/logicalworker.h"
      51             : #include "replication/origin.h"
      52             : #include "replication/snapbuild.h"
      53             : #include "replication/syncrep.h"
      54             : #include "storage/condition_variable.h"
      55             : #include "storage/fd.h"
      56             : #include "storage/lmgr.h"
      57             : #include "storage/md.h"
      58             : #include "storage/predicate.h"
      59             : #include "storage/proc.h"
      60             : #include "storage/procarray.h"
      61             : #include "storage/sinvaladt.h"
      62             : #include "storage/smgr.h"
      63             : #include "utils/builtins.h"
      64             : #include "utils/combocid.h"
      65             : #include "utils/guc.h"
      66             : #include "utils/inval.h"
      67             : #include "utils/memutils.h"
      68             : #include "utils/relmapper.h"
      69             : #include "utils/snapmgr.h"
      70             : #include "utils/timeout.h"
      71             : #include "utils/timestamp.h"
      72             : #include "utils/typcache.h"
      73             : 
      74             : /*
      75             :  *  User-tweakable parameters
      76             :  */
      77             : int         DefaultXactIsoLevel = XACT_READ_COMMITTED;
      78             : int         XactIsoLevel = XACT_READ_COMMITTED;
      79             : 
      80             : bool        DefaultXactReadOnly = false;
      81             : bool        XactReadOnly;
      82             : 
      83             : bool        DefaultXactDeferrable = false;
      84             : bool        XactDeferrable;
      85             : 
      86             : int         synchronous_commit = SYNCHRONOUS_COMMIT_ON;
      87             : 
      88             : /*
      89             :  * CheckXidAlive is a xid value pointing to a possibly ongoing (sub)
      90             :  * transaction.  Currently, it is used in logical decoding.  It's possible
      91             :  * that such transactions can get aborted while the decoding is ongoing in
      92             :  * which case we skip decoding that particular transaction.  To ensure that we
      93             :  * check whether the CheckXidAlive is aborted after fetching the tuple from
      94             :  * system tables.  We also ensure that during logical decoding we never
      95             :  * directly access the tableam or heap APIs because we are checking for the
      96             :  * concurrent aborts only in systable_* APIs.
      97             :  */
      98             : TransactionId CheckXidAlive = InvalidTransactionId;
      99             : bool        bsysscan = false;
     100             : 
     101             : /*
     102             :  * When running as a parallel worker, we place only a single
     103             :  * TransactionStateData on the parallel worker's state stack, and the XID
     104             :  * reflected there will be that of the *innermost* currently-active
     105             :  * subtransaction in the backend that initiated parallelism.  However,
     106             :  * GetTopTransactionId() and TransactionIdIsCurrentTransactionId()
     107             :  * need to return the same answers in the parallel worker as they would have
     108             :  * in the user backend, so we need some additional bookkeeping.
     109             :  *
     110             :  * XactTopFullTransactionId stores the XID of our toplevel transaction, which
     111             :  * will be the same as TopTransactionStateData.fullTransactionId in an
     112             :  * ordinary backend; but in a parallel backend, which does not have the entire
     113             :  * transaction state, it will instead be copied from the backend that started
     114             :  * the parallel operation.
     115             :  *
     116             :  * nParallelCurrentXids will be 0 and ParallelCurrentXids NULL in an ordinary
     117             :  * backend, but in a parallel backend, nParallelCurrentXids will contain the
     118             :  * number of XIDs that need to be considered current, and ParallelCurrentXids
     119             :  * will contain the XIDs themselves.  This includes all XIDs that were current
     120             :  * or sub-committed in the parent at the time the parallel operation began.
     121             :  * The XIDs are stored sorted in numerical order (not logical order) to make
     122             :  * lookups as fast as possible.
     123             :  */
     124             : static FullTransactionId XactTopFullTransactionId = {InvalidTransactionId};
     125             : static int  nParallelCurrentXids = 0;
     126             : static TransactionId *ParallelCurrentXids;
     127             : 
     128             : /*
     129             :  * Miscellaneous flag bits to record events which occur on the top level
     130             :  * transaction. These flags are only persisted in MyXactFlags and are intended
     131             :  * so we remember to do certain things later on in the transaction. This is
     132             :  * globally accessible, so can be set from anywhere in the code that requires
     133             :  * recording flags.
     134             :  */
     135             : int         MyXactFlags;
     136             : 
     137             : /*
     138             :  *  transaction states - transaction state from server perspective
     139             :  */
     140             : typedef enum TransState
     141             : {
     142             :     TRANS_DEFAULT,              /* idle */
     143             :     TRANS_START,                /* transaction starting */
     144             :     TRANS_INPROGRESS,           /* inside a valid transaction */
     145             :     TRANS_COMMIT,               /* commit in progress */
     146             :     TRANS_ABORT,                /* abort in progress */
     147             :     TRANS_PREPARE,              /* prepare in progress */
     148             : } TransState;
     149             : 
     150             : /*
     151             :  *  transaction block states - transaction state of client queries
     152             :  *
     153             :  * Note: the subtransaction states are used only for non-topmost
     154             :  * transactions; the others appear only in the topmost transaction.
     155             :  */
     156             : typedef enum TBlockState
     157             : {
     158             :     /* not-in-transaction-block states */
     159             :     TBLOCK_DEFAULT,             /* idle */
     160             :     TBLOCK_STARTED,             /* running single-query transaction */
     161             : 
     162             :     /* transaction block states */
     163             :     TBLOCK_BEGIN,               /* starting transaction block */
     164             :     TBLOCK_INPROGRESS,          /* live transaction */
     165             :     TBLOCK_IMPLICIT_INPROGRESS, /* live transaction after implicit BEGIN */
     166             :     TBLOCK_PARALLEL_INPROGRESS, /* live transaction inside parallel worker */
     167             :     TBLOCK_END,                 /* COMMIT received */
     168             :     TBLOCK_ABORT,               /* failed xact, awaiting ROLLBACK */
     169             :     TBLOCK_ABORT_END,           /* failed xact, ROLLBACK received */
     170             :     TBLOCK_ABORT_PENDING,       /* live xact, ROLLBACK received */
     171             :     TBLOCK_PREPARE,             /* live xact, PREPARE received */
     172             : 
     173             :     /* subtransaction states */
     174             :     TBLOCK_SUBBEGIN,            /* starting a subtransaction */
     175             :     TBLOCK_SUBINPROGRESS,       /* live subtransaction */
     176             :     TBLOCK_SUBRELEASE,          /* RELEASE received */
     177             :     TBLOCK_SUBCOMMIT,           /* COMMIT received while TBLOCK_SUBINPROGRESS */
     178             :     TBLOCK_SUBABORT,            /* failed subxact, awaiting ROLLBACK */
     179             :     TBLOCK_SUBABORT_END,        /* failed subxact, ROLLBACK received */
     180             :     TBLOCK_SUBABORT_PENDING,    /* live subxact, ROLLBACK received */
     181             :     TBLOCK_SUBRESTART,          /* live subxact, ROLLBACK TO received */
     182             :     TBLOCK_SUBABORT_RESTART,    /* failed subxact, ROLLBACK TO received */
     183             : } TBlockState;
     184             : 
     185             : /*
     186             :  *  transaction state structure
     187             :  *
     188             :  * Note: parallelModeLevel counts the number of unmatched EnterParallelMode
     189             :  * calls done at this transaction level.  parallelChildXact is true if any
     190             :  * upper transaction level has nonzero parallelModeLevel.
     191             :  */
     192             : typedef struct TransactionStateData
     193             : {
     194             :     FullTransactionId fullTransactionId;    /* my FullTransactionId */
     195             :     SubTransactionId subTransactionId;  /* my subxact ID */
     196             :     char       *name;           /* savepoint name, if any */
     197             :     int         savepointLevel; /* savepoint level */
     198             :     TransState  state;          /* low-level state */
     199             :     TBlockState blockState;     /* high-level state */
     200             :     int         nestingLevel;   /* transaction nesting depth */
     201             :     int         gucNestLevel;   /* GUC context nesting depth */
     202             :     MemoryContext curTransactionContext;    /* my xact-lifetime context */
     203             :     ResourceOwner curTransactionOwner;  /* my query resources */
     204             :     MemoryContext priorContext; /* CurrentMemoryContext before xact started */
     205             :     TransactionId *childXids;   /* subcommitted child XIDs, in XID order */
     206             :     int         nChildXids;     /* # of subcommitted child XIDs */
     207             :     int         maxChildXids;   /* allocated size of childXids[] */
     208             :     Oid         prevUser;       /* previous CurrentUserId setting */
     209             :     int         prevSecContext; /* previous SecurityRestrictionContext */
     210             :     bool        prevXactReadOnly;   /* entry-time xact r/o state */
     211             :     bool        startedInRecovery;  /* did we start in recovery? */
     212             :     bool        didLogXid;      /* has xid been included in WAL record? */
     213             :     int         parallelModeLevel;  /* Enter/ExitParallelMode counter */
     214             :     bool        parallelChildXact;  /* is any parent transaction parallel? */
     215             :     bool        chain;          /* start a new block after this one */
     216             :     bool        topXidLogged;   /* for a subxact: is top-level XID logged? */
     217             :     struct TransactionStateData *parent;    /* back link to parent */
     218             : } TransactionStateData;
     219             : 
     220             : typedef TransactionStateData *TransactionState;
     221             : 
     222             : /*
     223             :  * Serialized representation used to transmit transaction state to parallel
     224             :  * workers through shared memory.
     225             :  */
     226             : typedef struct SerializedTransactionState
     227             : {
     228             :     int         xactIsoLevel;
     229             :     bool        xactDeferrable;
     230             :     FullTransactionId topFullTransactionId;
     231             :     FullTransactionId currentFullTransactionId;
     232             :     CommandId   currentCommandId;
     233             :     int         nParallelCurrentXids;
     234             :     TransactionId parallelCurrentXids[FLEXIBLE_ARRAY_MEMBER];
     235             : } SerializedTransactionState;
     236             : 
     237             : /* The size of SerializedTransactionState, not including the final array. */
     238             : #define SerializedTransactionStateHeaderSize \
     239             :     offsetof(SerializedTransactionState, parallelCurrentXids)
     240             : 
     241             : /*
     242             :  * CurrentTransactionState always points to the current transaction state
     243             :  * block.  It will point to TopTransactionStateData when not in a
     244             :  * transaction at all, or when in a top-level transaction.
     245             :  */
     246             : static TransactionStateData TopTransactionStateData = {
     247             :     .state = TRANS_DEFAULT,
     248             :     .blockState = TBLOCK_DEFAULT,
     249             :     .topXidLogged = false,
     250             : };
     251             : 
     252             : /*
     253             :  * unreportedXids holds XIDs of all subtransactions that have not yet been
     254             :  * reported in an XLOG_XACT_ASSIGNMENT record.
     255             :  */
     256             : static int  nUnreportedXids;
     257             : static TransactionId unreportedXids[PGPROC_MAX_CACHED_SUBXIDS];
     258             : 
     259             : static TransactionState CurrentTransactionState = &TopTransactionStateData;
     260             : 
     261             : /*
     262             :  * The subtransaction ID and command ID assignment counters are global
     263             :  * to a whole transaction, so we do not keep them in the state stack.
     264             :  */
     265             : static SubTransactionId currentSubTransactionId;
     266             : static CommandId currentCommandId;
     267             : static bool currentCommandIdUsed;
     268             : 
     269             : /*
     270             :  * xactStartTimestamp is the value of transaction_timestamp().
     271             :  * stmtStartTimestamp is the value of statement_timestamp().
     272             :  * xactStopTimestamp is the time at which we log a commit / abort WAL record,
     273             :  * or if that was skipped, the time of the first subsequent
     274             :  * GetCurrentTransactionStopTimestamp() call.
     275             :  *
     276             :  * These do not change as we enter and exit subtransactions, so we don't
     277             :  * keep them inside the TransactionState stack.
     278             :  */
     279             : static TimestampTz xactStartTimestamp;
     280             : static TimestampTz stmtStartTimestamp;
     281             : static TimestampTz xactStopTimestamp;
     282             : 
     283             : /*
     284             :  * GID to be used for preparing the current transaction.  This is also
     285             :  * global to a whole transaction, so we don't keep it in the state stack.
     286             :  */
     287             : static char *prepareGID;
     288             : 
     289             : /*
     290             :  * Some commands want to force synchronous commit.
     291             :  */
     292             : static bool forceSyncCommit = false;
     293             : 
     294             : /* Flag for logging statements in a transaction. */
     295             : bool        xact_is_sampled = false;
     296             : 
     297             : /*
     298             :  * Private context for transaction-abort work --- we reserve space for this
     299             :  * at startup to ensure that AbortTransaction and AbortSubTransaction can work
     300             :  * when we've run out of memory.
     301             :  */
     302             : static MemoryContext TransactionAbortContext = NULL;
     303             : 
     304             : /*
     305             :  * List of add-on start- and end-of-xact callbacks
     306             :  */
     307             : typedef struct XactCallbackItem
     308             : {
     309             :     struct XactCallbackItem *next;
     310             :     XactCallback callback;
     311             :     void       *arg;
     312             : } XactCallbackItem;
     313             : 
     314             : static XactCallbackItem *Xact_callbacks = NULL;
     315             : 
     316             : /*
     317             :  * List of add-on start- and end-of-subxact callbacks
     318             :  */
     319             : typedef struct SubXactCallbackItem
     320             : {
     321             :     struct SubXactCallbackItem *next;
     322             :     SubXactCallback callback;
     323             :     void       *arg;
     324             : } SubXactCallbackItem;
     325             : 
     326             : static SubXactCallbackItem *SubXact_callbacks = NULL;
     327             : 
     328             : 
     329             : /* local function prototypes */
     330             : static void AssignTransactionId(TransactionState s);
     331             : static void AbortTransaction(void);
     332             : static void AtAbort_Memory(void);
     333             : static void AtCleanup_Memory(void);
     334             : static void AtAbort_ResourceOwner(void);
     335             : static void AtCCI_LocalCache(void);
     336             : static void AtCommit_Memory(void);
     337             : static void AtStart_Cache(void);
     338             : static void AtStart_Memory(void);
     339             : static void AtStart_ResourceOwner(void);
     340             : static void CallXactCallbacks(XactEvent event);
     341             : static void CallSubXactCallbacks(SubXactEvent event,
     342             :                                  SubTransactionId mySubid,
     343             :                                  SubTransactionId parentSubid);
     344             : static void CleanupTransaction(void);
     345             : static void CheckTransactionBlock(bool isTopLevel, bool throwError,
     346             :                                   const char *stmtType);
     347             : static void CommitTransaction(void);
     348             : static TransactionId RecordTransactionAbort(bool isSubXact);
     349             : static void StartTransaction(void);
     350             : 
     351             : static bool CommitTransactionCommandInternal(void);
     352             : static bool AbortCurrentTransactionInternal(void);
     353             : 
     354             : static void StartSubTransaction(void);
     355             : static void CommitSubTransaction(void);
     356             : static void AbortSubTransaction(void);
     357             : static void CleanupSubTransaction(void);
     358             : static void PushTransaction(void);
     359             : static void PopTransaction(void);
     360             : 
     361             : static void AtSubAbort_Memory(void);
     362             : static void AtSubCleanup_Memory(void);
     363             : static void AtSubAbort_ResourceOwner(void);
     364             : static void AtSubCommit_Memory(void);
     365             : static void AtSubStart_Memory(void);
     366             : static void AtSubStart_ResourceOwner(void);
     367             : 
     368             : static void ShowTransactionState(const char *str);
     369             : static void ShowTransactionStateRec(const char *str, TransactionState s);
     370             : static const char *BlockStateAsString(TBlockState blockState);
     371             : static const char *TransStateAsString(TransState state);
     372             : 
     373             : 
     374             : /* ----------------------------------------------------------------
     375             :  *  transaction state accessors
     376             :  * ----------------------------------------------------------------
     377             :  */
     378             : 
     379             : /*
     380             :  *  IsTransactionState
     381             :  *
     382             :  *  This returns true if we are inside a valid transaction; that is,
     383             :  *  it is safe to initiate database access, take heavyweight locks, etc.
     384             :  */
     385             : bool
     386     2012360 : IsTransactionState(void)
     387             : {
     388     2012360 :     TransactionState s = CurrentTransactionState;
     389             : 
     390             :     /*
     391             :      * TRANS_DEFAULT and TRANS_ABORT are obviously unsafe states.  However, we
     392             :      * also reject the startup/shutdown states TRANS_START, TRANS_COMMIT,
     393             :      * TRANS_PREPARE since it might be too soon or too late within those
     394             :      * transition states to do anything interesting.  Hence, the only "valid"
     395             :      * state is TRANS_INPROGRESS.
     396             :      */
     397     2012360 :     return (s->state == TRANS_INPROGRESS);
     398             : }
     399             : 
     400             : /*
     401             :  *  IsAbortedTransactionBlockState
     402             :  *
     403             :  *  This returns true if we are within an aborted transaction block.
     404             :  */
     405             : bool
     406     1386862 : IsAbortedTransactionBlockState(void)
     407             : {
     408     1386862 :     TransactionState s = CurrentTransactionState;
     409             : 
     410     1386862 :     if (s->blockState == TBLOCK_ABORT ||
     411     1383924 :         s->blockState == TBLOCK_SUBABORT)
     412        3502 :         return true;
     413             : 
     414     1383360 :     return false;
     415             : }
     416             : 
     417             : 
     418             : /*
     419             :  *  GetTopTransactionId
     420             :  *
     421             :  * This will return the XID of the main transaction, assigning one if
     422             :  * it's not yet set.  Be careful to call this only inside a valid xact.
     423             :  */
     424             : TransactionId
     425       56826 : GetTopTransactionId(void)
     426             : {
     427       56826 :     if (!FullTransactionIdIsValid(XactTopFullTransactionId))
     428        1108 :         AssignTransactionId(&TopTransactionStateData);
     429       56826 :     return XidFromFullTransactionId(XactTopFullTransactionId);
     430             : }
     431             : 
     432             : /*
     433             :  *  GetTopTransactionIdIfAny
     434             :  *
     435             :  * This will return the XID of the main transaction, if one is assigned.
     436             :  * It will return InvalidTransactionId if we are not currently inside a
     437             :  * transaction, or inside a transaction that hasn't yet been assigned an XID.
     438             :  */
     439             : TransactionId
     440    86186204 : GetTopTransactionIdIfAny(void)
     441             : {
     442    86186204 :     return XidFromFullTransactionId(XactTopFullTransactionId);
     443             : }
     444             : 
     445             : /*
     446             :  *  GetCurrentTransactionId
     447             :  *
     448             :  * This will return the XID of the current transaction (main or sub
     449             :  * transaction), assigning one if it's not yet set.  Be careful to call this
     450             :  * only inside a valid xact.
     451             :  */
     452             : TransactionId
     453    20223564 : GetCurrentTransactionId(void)
     454             : {
     455    20223564 :     TransactionState s = CurrentTransactionState;
     456             : 
     457    20223564 :     if (!FullTransactionIdIsValid(s->fullTransactionId))
     458      245084 :         AssignTransactionId(s);
     459    20223552 :     return XidFromFullTransactionId(s->fullTransactionId);
     460             : }
     461             : 
     462             : /*
     463             :  *  GetCurrentTransactionIdIfAny
     464             :  *
     465             :  * This will return the XID of the current sub xact, if one is assigned.
     466             :  * It will return InvalidTransactionId if we are not currently inside a
     467             :  * transaction, or inside a transaction that hasn't been assigned an XID yet.
     468             :  */
     469             : TransactionId
     470    27786568 : GetCurrentTransactionIdIfAny(void)
     471             : {
     472    27786568 :     return XidFromFullTransactionId(CurrentTransactionState->fullTransactionId);
     473             : }
     474             : 
     475             : /*
     476             :  *  GetTopFullTransactionId
     477             :  *
     478             :  * This will return the FullTransactionId of the main transaction, assigning
     479             :  * one if it's not yet set.  Be careful to call this only inside a valid xact.
     480             :  */
     481             : FullTransactionId
     482        5714 : GetTopFullTransactionId(void)
     483             : {
     484        5714 :     if (!FullTransactionIdIsValid(XactTopFullTransactionId))
     485        4144 :         AssignTransactionId(&TopTransactionStateData);
     486        5714 :     return XactTopFullTransactionId;
     487             : }
     488             : 
     489             : /*
     490             :  *  GetTopFullTransactionIdIfAny
     491             :  *
     492             :  * This will return the FullTransactionId of the main transaction, if one is
     493             :  * assigned.  It will return InvalidFullTransactionId if we are not currently
     494             :  * inside a transaction, or inside a transaction that hasn't yet been assigned
     495             :  * one.
     496             :  */
     497             : FullTransactionId
     498          24 : GetTopFullTransactionIdIfAny(void)
     499             : {
     500          24 :     return XactTopFullTransactionId;
     501             : }
     502             : 
     503             : /*
     504             :  *  GetCurrentFullTransactionId
     505             :  *
     506             :  * This will return the FullTransactionId of the current transaction (main or
     507             :  * sub transaction), assigning one if it's not yet set.  Be careful to call
     508             :  * this only inside a valid xact.
     509             :  */
     510             : FullTransactionId
     511           0 : GetCurrentFullTransactionId(void)
     512             : {
     513           0 :     TransactionState s = CurrentTransactionState;
     514             : 
     515           0 :     if (!FullTransactionIdIsValid(s->fullTransactionId))
     516           0 :         AssignTransactionId(s);
     517           0 :     return s->fullTransactionId;
     518             : }
     519             : 
     520             : /*
     521             :  *  GetCurrentFullTransactionIdIfAny
     522             :  *
     523             :  * This will return the FullTransactionId of the current sub xact, if one is
     524             :  * assigned.  It will return InvalidFullTransactionId if we are not currently
     525             :  * inside a transaction, or inside a transaction that hasn't been assigned one
     526             :  * yet.
     527             :  */
     528             : FullTransactionId
     529           0 : GetCurrentFullTransactionIdIfAny(void)
     530             : {
     531           0 :     return CurrentTransactionState->fullTransactionId;
     532             : }
     533             : 
     534             : /*
     535             :  *  MarkCurrentTransactionIdLoggedIfAny
     536             :  *
     537             :  * Remember that the current xid - if it is assigned - now has been wal logged.
     538             :  */
     539             : void
     540    27712164 : MarkCurrentTransactionIdLoggedIfAny(void)
     541             : {
     542    27712164 :     if (FullTransactionIdIsValid(CurrentTransactionState->fullTransactionId))
     543    27229614 :         CurrentTransactionState->didLogXid = true;
     544    27712164 : }
     545             : 
     546             : /*
     547             :  * IsSubxactTopXidLogPending
     548             :  *
     549             :  * This is used to decide whether we need to WAL log the top-level XID for
     550             :  * operation in a subtransaction.  We require that for logical decoding, see
     551             :  * LogicalDecodingProcessRecord.
     552             :  *
     553             :  * This returns true if wal_level >= logical and we are inside a valid
     554             :  * subtransaction, for which the assignment was not yet written to any WAL
     555             :  * record.
     556             :  */
     557             : bool
     558    27725996 : IsSubxactTopXidLogPending(void)
     559             : {
     560             :     /* check whether it is already logged */
     561    27725996 :     if (CurrentTransactionState->topXidLogged)
     562      204254 :         return false;
     563             : 
     564             :     /* wal_level has to be logical */
     565    27521742 :     if (!XLogLogicalInfoActive())
     566    26475780 :         return false;
     567             : 
     568             :     /* we need to be in a transaction state */
     569     1045962 :     if (!IsTransactionState())
     570        7130 :         return false;
     571             : 
     572             :     /* it has to be a subtransaction */
     573     1038832 :     if (!IsSubTransaction())
     574     1038376 :         return false;
     575             : 
     576             :     /* the subtransaction has to have a XID assigned */
     577         456 :     if (!TransactionIdIsValid(GetCurrentTransactionIdIfAny()))
     578          14 :         return false;
     579             : 
     580         442 :     return true;
     581             : }
     582             : 
     583             : /*
     584             :  * MarkSubxactTopXidLogged
     585             :  *
     586             :  * Remember that the top transaction id for the current subtransaction is WAL
     587             :  * logged now.
     588             :  */
     589             : void
     590         438 : MarkSubxactTopXidLogged(void)
     591             : {
     592             :     Assert(IsSubxactTopXidLogPending());
     593             : 
     594         438 :     CurrentTransactionState->topXidLogged = true;
     595         438 : }
     596             : 
     597             : /*
     598             :  *  GetStableLatestTransactionId
     599             :  *
     600             :  * Get the transaction's XID if it has one, else read the next-to-be-assigned
     601             :  * XID.  Once we have a value, return that same value for the remainder of the
     602             :  * current transaction.  This is meant to provide the reference point for the
     603             :  * age(xid) function, but might be useful for other maintenance tasks as well.
     604             :  */
     605             : TransactionId
     606         288 : GetStableLatestTransactionId(void)
     607             : {
     608             :     static LocalTransactionId lxid = InvalidLocalTransactionId;
     609             :     static TransactionId stablexid = InvalidTransactionId;
     610             : 
     611         288 :     if (lxid != MyProc->vxid.lxid)
     612             :     {
     613          24 :         lxid = MyProc->vxid.lxid;
     614          24 :         stablexid = GetTopTransactionIdIfAny();
     615          24 :         if (!TransactionIdIsValid(stablexid))
     616          24 :             stablexid = ReadNextTransactionId();
     617             :     }
     618             : 
     619             :     Assert(TransactionIdIsValid(stablexid));
     620             : 
     621         288 :     return stablexid;
     622             : }
     623             : 
     624             : /*
     625             :  * AssignTransactionId
     626             :  *
     627             :  * Assigns a new permanent FullTransactionId to the given TransactionState.
     628             :  * We do not assign XIDs to transactions until/unless this is called.
     629             :  * Also, any parent TransactionStates that don't yet have XIDs are assigned
     630             :  * one; this maintains the invariant that a child transaction has an XID
     631             :  * following its parent's.
     632             :  */
     633             : static void
     634      252752 : AssignTransactionId(TransactionState s)
     635             : {
     636      252752 :     bool        isSubXact = (s->parent != NULL);
     637             :     ResourceOwner currentOwner;
     638      252752 :     bool        log_unknown_top = false;
     639             : 
     640             :     /* Assert that caller didn't screw up */
     641             :     Assert(!FullTransactionIdIsValid(s->fullTransactionId));
     642             :     Assert(s->state == TRANS_INPROGRESS);
     643             : 
     644             :     /*
     645             :      * Workers synchronize transaction state at the beginning of each parallel
     646             :      * operation, so we can't account for new XIDs at this point.
     647             :      */
     648      252752 :     if (IsInParallelMode() || IsParallelWorker())
     649           0 :         ereport(ERROR,
     650             :                 (errcode(ERRCODE_INVALID_TRANSACTION_STATE),
     651             :                  errmsg("cannot assign transaction IDs during a parallel operation")));
     652             : 
     653             :     /*
     654             :      * Ensure parent(s) have XIDs, so that a child always has an XID later
     655             :      * than its parent.  Mustn't recurse here, or we might get a stack
     656             :      * overflow if we're at the bottom of a huge stack of subtransactions none
     657             :      * of which have XIDs yet.
     658             :      */
     659      252752 :     if (isSubXact && !FullTransactionIdIsValid(s->parent->fullTransactionId))
     660             :     {
     661        1278 :         TransactionState p = s->parent;
     662             :         TransactionState *parents;
     663        1278 :         size_t      parentOffset = 0;
     664             : 
     665        1278 :         parents = palloc(sizeof(TransactionState) * s->nestingLevel);
     666        3694 :         while (p != NULL && !FullTransactionIdIsValid(p->fullTransactionId))
     667             :         {
     668        2416 :             parents[parentOffset++] = p;
     669        2416 :             p = p->parent;
     670             :         }
     671             : 
     672             :         /*
     673             :          * This is technically a recursive call, but the recursion will never
     674             :          * be more than one layer deep.
     675             :          */
     676        3694 :         while (parentOffset != 0)
     677        2416 :             AssignTransactionId(parents[--parentOffset]);
     678             : 
     679        1278 :         pfree(parents);
     680             :     }
     681             : 
     682             :     /*
     683             :      * When wal_level=logical, guarantee that a subtransaction's xid can only
     684             :      * be seen in the WAL stream if its toplevel xid has been logged before.
     685             :      * If necessary we log an xact_assignment record with fewer than
     686             :      * PGPROC_MAX_CACHED_SUBXIDS. Note that it is fine if didLogXid isn't set
     687             :      * for a transaction even though it appears in a WAL record, we just might
     688             :      * superfluously log something. That can happen when an xid is included
     689             :      * somewhere inside a wal record, but not in XLogRecord->xl_xid, like in
     690             :      * xl_standby_locks.
     691             :      */
     692      252752 :     if (isSubXact && XLogLogicalInfoActive() &&
     693         600 :         !TopTransactionStateData.didLogXid)
     694          48 :         log_unknown_top = true;
     695             : 
     696             :     /*
     697             :      * Generate a new FullTransactionId and record its xid in PGPROC and
     698             :      * pg_subtrans.
     699             :      *
     700             :      * NB: we must make the subtrans entry BEFORE the Xid appears anywhere in
     701             :      * shared storage other than PGPROC; because if there's no room for it in
     702             :      * PGPROC, the subtrans entry is needed to ensure that other backends see
     703             :      * the Xid as "running".  See GetNewTransactionId.
     704             :      */
     705      252752 :     s->fullTransactionId = GetNewTransactionId(isSubXact);
     706      252740 :     if (!isSubXact)
     707      243968 :         XactTopFullTransactionId = s->fullTransactionId;
     708             : 
     709      252740 :     if (isSubXact)
     710        8772 :         SubTransSetParent(XidFromFullTransactionId(s->fullTransactionId),
     711        8772 :                           XidFromFullTransactionId(s->parent->fullTransactionId));
     712             : 
     713             :     /*
     714             :      * If it's a top-level transaction, the predicate locking system needs to
     715             :      * be told about it too.
     716             :      */
     717      252740 :     if (!isSubXact)
     718      243968 :         RegisterPredicateLockingXid(XidFromFullTransactionId(s->fullTransactionId));
     719             : 
     720             :     /*
     721             :      * Acquire lock on the transaction XID.  (We assume this cannot block.) We
     722             :      * have to ensure that the lock is assigned to the transaction's own
     723             :      * ResourceOwner.
     724             :      */
     725      252740 :     currentOwner = CurrentResourceOwner;
     726      252740 :     CurrentResourceOwner = s->curTransactionOwner;
     727             : 
     728      252740 :     XactLockTableInsert(XidFromFullTransactionId(s->fullTransactionId));
     729             : 
     730      252740 :     CurrentResourceOwner = currentOwner;
     731             : 
     732             :     /*
     733             :      * Every PGPROC_MAX_CACHED_SUBXIDS assigned transaction ids within each
     734             :      * top-level transaction we issue a WAL record for the assignment. We
     735             :      * include the top-level xid and all the subxids that have not yet been
     736             :      * reported using XLOG_XACT_ASSIGNMENT records.
     737             :      *
     738             :      * This is required to limit the amount of shared memory required in a hot
     739             :      * standby server to keep track of in-progress XIDs. See notes for
     740             :      * RecordKnownAssignedTransactionIds().
     741             :      *
     742             :      * We don't keep track of the immediate parent of each subxid, only the
     743             :      * top-level transaction that each subxact belongs to. This is correct in
     744             :      * recovery only because aborted subtransactions are separately WAL
     745             :      * logged.
     746             :      *
     747             :      * This is correct even for the case where several levels above us didn't
     748             :      * have an xid assigned as we recursed up to them beforehand.
     749             :      */
     750      252740 :     if (isSubXact && XLogStandbyInfoActive())
     751             :     {
     752        7656 :         unreportedXids[nUnreportedXids] = XidFromFullTransactionId(s->fullTransactionId);
     753        7656 :         nUnreportedXids++;
     754             : 
     755             :         /*
     756             :          * ensure this test matches similar one in
     757             :          * RecoverPreparedTransactions()
     758             :          */
     759        7656 :         if (nUnreportedXids >= PGPROC_MAX_CACHED_SUBXIDS ||
     760             :             log_unknown_top)
     761             :         {
     762             :             xl_xact_assignment xlrec;
     763             : 
     764             :             /*
     765             :              * xtop is always set by now because we recurse up transaction
     766             :              * stack to the highest unassigned xid and then come back down
     767             :              */
     768         132 :             xlrec.xtop = GetTopTransactionId();
     769             :             Assert(TransactionIdIsValid(xlrec.xtop));
     770         132 :             xlrec.nsubxacts = nUnreportedXids;
     771             : 
     772         132 :             XLogBeginInsert();
     773         132 :             XLogRegisterData((char *) &xlrec, MinSizeOfXactAssignment);
     774         132 :             XLogRegisterData((char *) unreportedXids,
     775             :                              nUnreportedXids * sizeof(TransactionId));
     776             : 
     777         132 :             (void) XLogInsert(RM_XACT_ID, XLOG_XACT_ASSIGNMENT);
     778             : 
     779         132 :             nUnreportedXids = 0;
     780             :             /* mark top, not current xact as having been logged */
     781         132 :             TopTransactionStateData.didLogXid = true;
     782             :         }
     783             :     }
     784      252740 : }
     785             : 
     786             : /*
     787             :  *  GetCurrentSubTransactionId
     788             :  */
     789             : SubTransactionId
     790    17645478 : GetCurrentSubTransactionId(void)
     791             : {
     792    17645478 :     TransactionState s = CurrentTransactionState;
     793             : 
     794    17645478 :     return s->subTransactionId;
     795             : }
     796             : 
     797             : /*
     798             :  *  SubTransactionIsActive
     799             :  *
     800             :  * Test if the specified subxact ID is still active.  Note caller is
     801             :  * responsible for checking whether this ID is relevant to the current xact.
     802             :  */
     803             : bool
     804       92458 : SubTransactionIsActive(SubTransactionId subxid)
     805             : {
     806             :     TransactionState s;
     807             : 
     808       92470 :     for (s = CurrentTransactionState; s != NULL; s = s->parent)
     809             :     {
     810       92470 :         if (s->state == TRANS_ABORT)
     811           0 :             continue;
     812       92470 :         if (s->subTransactionId == subxid)
     813       92458 :             return true;
     814             :     }
     815           0 :     return false;
     816             : }
     817             : 
     818             : 
     819             : /*
     820             :  *  GetCurrentCommandId
     821             :  *
     822             :  * "used" must be true if the caller intends to use the command ID to mark
     823             :  * inserted/updated/deleted tuples.  false means the ID is being fetched
     824             :  * for read-only purposes (ie, as a snapshot validity cutoff).  See
     825             :  * CommandCounterIncrement() for discussion.
     826             :  */
     827             : CommandId
     828    10653732 : GetCurrentCommandId(bool used)
     829             : {
     830             :     /* this is global to a transaction, not subtransaction-local */
     831    10653732 :     if (used)
     832             :     {
     833             :         /*
     834             :          * Forbid setting currentCommandIdUsed in a parallel worker, because
     835             :          * we have no provision for communicating this back to the leader.  We
     836             :          * could relax this restriction when currentCommandIdUsed was already
     837             :          * true at the start of the parallel operation.
     838             :          */
     839     6333252 :         if (IsParallelWorker())
     840           0 :             ereport(ERROR,
     841             :                     (errcode(ERRCODE_INVALID_TRANSACTION_STATE),
     842             :                      errmsg("cannot modify data in a parallel worker")));
     843             : 
     844     6333252 :         currentCommandIdUsed = true;
     845             :     }
     846    10653732 :     return currentCommandId;
     847             : }
     848             : 
     849             : /*
     850             :  *  SetParallelStartTimestamps
     851             :  *
     852             :  * In a parallel worker, we should inherit the parent transaction's
     853             :  * timestamps rather than setting our own.  The parallel worker
     854             :  * infrastructure must call this to provide those values before
     855             :  * calling StartTransaction() or SetCurrentStatementStartTimestamp().
     856             :  */
     857             : void
     858        2712 : SetParallelStartTimestamps(TimestampTz xact_ts, TimestampTz stmt_ts)
     859             : {
     860             :     Assert(IsParallelWorker());
     861        2712 :     xactStartTimestamp = xact_ts;
     862        2712 :     stmtStartTimestamp = stmt_ts;
     863        2712 : }
     864             : 
     865             : /*
     866             :  *  GetCurrentTransactionStartTimestamp
     867             :  */
     868             : TimestampTz
     869       85848 : GetCurrentTransactionStartTimestamp(void)
     870             : {
     871       85848 :     return xactStartTimestamp;
     872             : }
     873             : 
     874             : /*
     875             :  *  GetCurrentStatementStartTimestamp
     876             :  */
     877             : TimestampTz
     878     2115572 : GetCurrentStatementStartTimestamp(void)
     879             : {
     880     2115572 :     return stmtStartTimestamp;
     881             : }
     882             : 
     883             : /*
     884             :  *  GetCurrentTransactionStopTimestamp
     885             :  *
     886             :  * If the transaction stop time hasn't already been set, which can happen if
     887             :  * we decided we don't need to log an XLOG record, set xactStopTimestamp.
     888             :  */
     889             : TimestampTz
     890     1827082 : GetCurrentTransactionStopTimestamp(void)
     891             : {
     892     1827082 :     TransactionState s PG_USED_FOR_ASSERTS_ONLY = CurrentTransactionState;
     893             : 
     894             :     /* should only be called after commit / abort processing */
     895             :     Assert(s->state == TRANS_DEFAULT ||
     896             :            s->state == TRANS_COMMIT ||
     897             :            s->state == TRANS_ABORT ||
     898             :            s->state == TRANS_PREPARE);
     899             : 
     900     1827082 :     if (xactStopTimestamp == 0)
     901      536548 :         xactStopTimestamp = GetCurrentTimestamp();
     902             : 
     903     1827082 :     return xactStopTimestamp;
     904             : }
     905             : 
     906             : /*
     907             :  *  SetCurrentStatementStartTimestamp
     908             :  *
     909             :  * In a parallel worker, this should already have been provided by a call
     910             :  * to SetParallelStartTimestamps().
     911             :  */
     912             : void
     913     1107614 : SetCurrentStatementStartTimestamp(void)
     914             : {
     915     1107614 :     if (!IsParallelWorker())
     916     1104902 :         stmtStartTimestamp = GetCurrentTimestamp();
     917             :     else
     918             :         Assert(stmtStartTimestamp != 0);
     919     1107614 : }
     920             : 
     921             : /*
     922             :  *  GetCurrentTransactionNestLevel
     923             :  *
     924             :  * Note: this will return zero when not inside any transaction, one when
     925             :  * inside a top-level transaction, etc.
     926             :  */
     927             : int
     928    32020674 : GetCurrentTransactionNestLevel(void)
     929             : {
     930    32020674 :     TransactionState s = CurrentTransactionState;
     931             : 
     932    32020674 :     return s->nestingLevel;
     933             : }
     934             : 
     935             : 
     936             : /*
     937             :  *  TransactionIdIsCurrentTransactionId
     938             :  */
     939             : bool
     940    86342416 : TransactionIdIsCurrentTransactionId(TransactionId xid)
     941             : {
     942             :     TransactionState s;
     943             : 
     944             :     /*
     945             :      * We always say that BootstrapTransactionId is "not my transaction ID"
     946             :      * even when it is (ie, during bootstrap).  Along with the fact that
     947             :      * transam.c always treats BootstrapTransactionId as already committed,
     948             :      * this causes the heapam_visibility.c routines to see all tuples as
     949             :      * committed, which is what we need during bootstrap.  (Bootstrap mode
     950             :      * only inserts tuples, it never updates or deletes them, so all tuples
     951             :      * can be presumed good immediately.)
     952             :      *
     953             :      * Likewise, InvalidTransactionId and FrozenTransactionId are certainly
     954             :      * not my transaction ID, so we can just return "false" immediately for
     955             :      * any non-normal XID.
     956             :      */
     957    86342416 :     if (!TransactionIdIsNormal(xid))
     958     1141606 :         return false;
     959             : 
     960    85200810 :     if (TransactionIdEquals(xid, GetTopTransactionIdIfAny()))
     961    54244056 :         return true;
     962             : 
     963             :     /*
     964             :      * In parallel workers, the XIDs we must consider as current are stored in
     965             :      * ParallelCurrentXids rather than the transaction-state stack.  Note that
     966             :      * the XIDs in this array are sorted numerically rather than according to
     967             :      * transactionIdPrecedes order.
     968             :      */
     969    30956754 :     if (nParallelCurrentXids > 0)
     970             :     {
     971             :         int         low,
     972             :                     high;
     973             : 
     974     3748710 :         low = 0;
     975     3748710 :         high = nParallelCurrentXids - 1;
     976    14707890 :         while (low <= high)
     977             :         {
     978             :             int         middle;
     979             :             TransactionId probe;
     980             : 
     981    14564502 :             middle = low + (high - low) / 2;
     982    14564502 :             probe = ParallelCurrentXids[middle];
     983    14564502 :             if (probe == xid)
     984     3605322 :                 return true;
     985    10959180 :             else if (probe < xid)
     986    10815804 :                 low = middle + 1;
     987             :             else
     988      143376 :                 high = middle - 1;
     989             :         }
     990      143388 :         return false;
     991             :     }
     992             : 
     993             :     /*
     994             :      * We will return true for the Xid of the current subtransaction, any of
     995             :      * its subcommitted children, any of its parents, or any of their
     996             :      * previously subcommitted children.  However, a transaction being aborted
     997             :      * is no longer "current", even though it may still have an entry on the
     998             :      * state stack.
     999             :      */
    1000    54428132 :     for (s = CurrentTransactionState; s != NULL; s = s->parent)
    1001             :     {
    1002             :         int         low,
    1003             :                     high;
    1004             : 
    1005    27292038 :         if (s->state == TRANS_ABORT)
    1006           0 :             continue;
    1007    27292038 :         if (!FullTransactionIdIsValid(s->fullTransactionId))
    1008    20783082 :             continue;           /* it can't have any child XIDs either */
    1009     6508956 :         if (TransactionIdEquals(xid, XidFromFullTransactionId(s->fullTransactionId)))
    1010       68876 :             return true;
    1011             :         /* As the childXids array is ordered, we can use binary search */
    1012     6440080 :         low = 0;
    1013     6440080 :         high = s->nChildXids - 1;
    1014     6441818 :         while (low <= high)
    1015             :         {
    1016             :             int         middle;
    1017             :             TransactionId probe;
    1018             : 
    1019        4812 :             middle = low + (high - low) / 2;
    1020        4812 :             probe = s->childXids[middle];
    1021        4812 :             if (TransactionIdEquals(probe, xid))
    1022        3074 :                 return true;
    1023        1738 :             else if (TransactionIdPrecedes(probe, xid))
    1024        1606 :                 low = middle + 1;
    1025             :             else
    1026         132 :                 high = middle - 1;
    1027             :         }
    1028             :     }
    1029             : 
    1030    27136094 :     return false;
    1031             : }
    1032             : 
    1033             : /*
    1034             :  *  TransactionStartedDuringRecovery
    1035             :  *
    1036             :  * Returns true if the current transaction started while recovery was still
    1037             :  * in progress. Recovery might have ended since so RecoveryInProgress() might
    1038             :  * return false already.
    1039             :  */
    1040             : bool
    1041    12781866 : TransactionStartedDuringRecovery(void)
    1042             : {
    1043    12781866 :     return CurrentTransactionState->startedInRecovery;
    1044             : }
    1045             : 
    1046             : /*
    1047             :  *  EnterParallelMode
    1048             :  */
    1049             : void
    1050        6280 : EnterParallelMode(void)
    1051             : {
    1052        6280 :     TransactionState s = CurrentTransactionState;
    1053             : 
    1054             :     Assert(s->parallelModeLevel >= 0);
    1055             : 
    1056        6280 :     ++s->parallelModeLevel;
    1057        6280 : }
    1058             : 
    1059             : /*
    1060             :  *  ExitParallelMode
    1061             :  */
    1062             : void
    1063        3556 : ExitParallelMode(void)
    1064             : {
    1065        3556 :     TransactionState s = CurrentTransactionState;
    1066             : 
    1067             :     Assert(s->parallelModeLevel > 0);
    1068             :     Assert(s->parallelModeLevel > 1 || s->parallelChildXact ||
    1069             :            !ParallelContextActive());
    1070             : 
    1071        3556 :     --s->parallelModeLevel;
    1072        3556 : }
    1073             : 
    1074             : /*
    1075             :  *  IsInParallelMode
    1076             :  *
    1077             :  * Are we in a parallel operation, as either the leader or a worker?  Check
    1078             :  * this to prohibit operations that change backend-local state expected to
    1079             :  * match across all workers.  Mere caches usually don't require such a
    1080             :  * restriction.  State modified in a strict push/pop fashion, such as the
    1081             :  * active snapshot stack, is often fine.
    1082             :  *
    1083             :  * We say we are in parallel mode if we are in a subxact of a transaction
    1084             :  * that's initiated a parallel operation; for most purposes that context
    1085             :  * has all the same restrictions.
    1086             :  */
    1087             : bool
    1088    57232512 : IsInParallelMode(void)
    1089             : {
    1090    57232512 :     TransactionState s = CurrentTransactionState;
    1091             : 
    1092    57232512 :     return s->parallelModeLevel != 0 || s->parallelChildXact;
    1093             : }
    1094             : 
    1095             : /*
    1096             :  *  CommandCounterIncrement
    1097             :  */
    1098             : void
    1099     2068286 : CommandCounterIncrement(void)
    1100             : {
    1101             :     /*
    1102             :      * If the current value of the command counter hasn't been "used" to mark
    1103             :      * tuples, we need not increment it, since there's no need to distinguish
    1104             :      * a read-only command from others.  This helps postpone command counter
    1105             :      * overflow, and keeps no-op CommandCounterIncrement operations cheap.
    1106             :      */
    1107     2068286 :     if (currentCommandIdUsed)
    1108             :     {
    1109             :         /*
    1110             :          * Workers synchronize transaction state at the beginning of each
    1111             :          * parallel operation, so we can't account for new commands after that
    1112             :          * point.
    1113             :          */
    1114     1070094 :         if (IsInParallelMode() || IsParallelWorker())
    1115           0 :             ereport(ERROR,
    1116             :                     (errcode(ERRCODE_INVALID_TRANSACTION_STATE),
    1117             :                      errmsg("cannot start commands during a parallel operation")));
    1118             : 
    1119     1070094 :         currentCommandId += 1;
    1120     1070094 :         if (currentCommandId == InvalidCommandId)
    1121             :         {
    1122           0 :             currentCommandId -= 1;
    1123           0 :             ereport(ERROR,
    1124             :                     (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
    1125             :                      errmsg("cannot have more than 2^32-2 commands in a transaction")));
    1126             :         }
    1127     1070094 :         currentCommandIdUsed = false;
    1128             : 
    1129             :         /* Propagate new command ID into static snapshots */
    1130     1070094 :         SnapshotSetCommandId(currentCommandId);
    1131             : 
    1132             :         /*
    1133             :          * Make any catalog changes done by the just-completed command visible
    1134             :          * in the local syscache.  We obviously don't need to do this after a
    1135             :          * read-only command.  (But see hacks in inval.c to make real sure we
    1136             :          * don't think a command that queued inval messages was read-only.)
    1137             :          */
    1138     1070094 :         AtCCI_LocalCache();
    1139             :     }
    1140     2068280 : }
    1141             : 
    1142             : /*
    1143             :  * ForceSyncCommit
    1144             :  *
    1145             :  * Interface routine to allow commands to force a synchronous commit of the
    1146             :  * current top-level transaction.  Currently, two-phase commit does not
    1147             :  * persist and restore this variable.  So long as all callers use
    1148             :  * PreventInTransactionBlock(), that omission has no consequences.
    1149             :  */
    1150             : void
    1151         866 : ForceSyncCommit(void)
    1152             : {
    1153         866 :     forceSyncCommit = true;
    1154         866 : }
    1155             : 
    1156             : 
    1157             : /* ----------------------------------------------------------------
    1158             :  *                      StartTransaction stuff
    1159             :  * ----------------------------------------------------------------
    1160             :  */
    1161             : 
    1162             : /*
    1163             :  *  AtStart_Cache
    1164             :  */
    1165             : static void
    1166      785102 : AtStart_Cache(void)
    1167             : {
    1168      785102 :     AcceptInvalidationMessages();
    1169      785102 : }
    1170             : 
    1171             : /*
    1172             :  *  AtStart_Memory
    1173             :  */
    1174             : static void
    1175      785102 : AtStart_Memory(void)
    1176             : {
    1177      785102 :     TransactionState s = CurrentTransactionState;
    1178             : 
    1179             :     /*
    1180             :      * Remember the memory context that was active prior to transaction start.
    1181             :      */
    1182      785102 :     s->priorContext = CurrentMemoryContext;
    1183             : 
    1184             :     /*
    1185             :      * If this is the first time through, create a private context for
    1186             :      * AbortTransaction to work in.  By reserving some space now, we can
    1187             :      * insulate AbortTransaction from out-of-memory scenarios.  Like
    1188             :      * ErrorContext, we set it up with slow growth rate and a nonzero minimum
    1189             :      * size, so that space will be reserved immediately.
    1190             :      */
    1191      785102 :     if (TransactionAbortContext == NULL)
    1192       30116 :         TransactionAbortContext =
    1193       30116 :             AllocSetContextCreate(TopMemoryContext,
    1194             :                                   "TransactionAbortContext",
    1195             :                                   32 * 1024,
    1196             :                                   32 * 1024,
    1197             :                                   32 * 1024);
    1198             : 
    1199             :     /*
    1200             :      * Likewise, if this is the first time through, create a top-level context
    1201             :      * for transaction-local data.  This context will be reset at transaction
    1202             :      * end, and then re-used in later transactions.
    1203             :      */
    1204      785102 :     if (TopTransactionContext == NULL)
    1205       30116 :         TopTransactionContext =
    1206       30116 :             AllocSetContextCreate(TopMemoryContext,
    1207             :                                   "TopTransactionContext",
    1208             :                                   ALLOCSET_DEFAULT_SIZES);
    1209             : 
    1210             :     /*
    1211             :      * In a top-level transaction, CurTransactionContext is the same as
    1212             :      * TopTransactionContext.
    1213             :      */
    1214      785102 :     CurTransactionContext = TopTransactionContext;
    1215      785102 :     s->curTransactionContext = CurTransactionContext;
    1216             : 
    1217             :     /* Make the CurTransactionContext active. */
    1218      785102 :     MemoryContextSwitchTo(CurTransactionContext);
    1219      785102 : }
    1220             : 
    1221             : /*
    1222             :  *  AtStart_ResourceOwner
    1223             :  */
    1224             : static void
    1225      785102 : AtStart_ResourceOwner(void)
    1226             : {
    1227      785102 :     TransactionState s = CurrentTransactionState;
    1228             : 
    1229             :     /*
    1230             :      * We shouldn't have a transaction resource owner already.
    1231             :      */
    1232             :     Assert(TopTransactionResourceOwner == NULL);
    1233             : 
    1234             :     /*
    1235             :      * Create a toplevel resource owner for the transaction.
    1236             :      */
    1237      785102 :     s->curTransactionOwner = ResourceOwnerCreate(NULL, "TopTransaction");
    1238             : 
    1239      785102 :     TopTransactionResourceOwner = s->curTransactionOwner;
    1240      785102 :     CurTransactionResourceOwner = s->curTransactionOwner;
    1241      785102 :     CurrentResourceOwner = s->curTransactionOwner;
    1242      785102 : }
    1243             : 
    1244             : /* ----------------------------------------------------------------
    1245             :  *                      StartSubTransaction stuff
    1246             :  * ----------------------------------------------------------------
    1247             :  */
    1248             : 
    1249             : /*
    1250             :  * AtSubStart_Memory
    1251             :  */
    1252             : static void
    1253       20040 : AtSubStart_Memory(void)
    1254             : {
    1255       20040 :     TransactionState s = CurrentTransactionState;
    1256             : 
    1257             :     Assert(CurTransactionContext != NULL);
    1258             : 
    1259             :     /*
    1260             :      * Remember the context that was active prior to subtransaction start.
    1261             :      */
    1262       20040 :     s->priorContext = CurrentMemoryContext;
    1263             : 
    1264             :     /*
    1265             :      * Create a CurTransactionContext, which will be used to hold data that
    1266             :      * survives subtransaction commit but disappears on subtransaction abort.
    1267             :      * We make it a child of the immediate parent's CurTransactionContext.
    1268             :      */
    1269       20040 :     CurTransactionContext = AllocSetContextCreate(CurTransactionContext,
    1270             :                                                   "CurTransactionContext",
    1271             :                                                   ALLOCSET_DEFAULT_SIZES);
    1272       20040 :     s->curTransactionContext = CurTransactionContext;
    1273             : 
    1274             :     /* Make the CurTransactionContext active. */
    1275       20040 :     MemoryContextSwitchTo(CurTransactionContext);
    1276       20040 : }
    1277             : 
    1278             : /*
    1279             :  * AtSubStart_ResourceOwner
    1280             :  */
    1281             : static void
    1282       20040 : AtSubStart_ResourceOwner(void)
    1283             : {
    1284       20040 :     TransactionState s = CurrentTransactionState;
    1285             : 
    1286             :     Assert(s->parent != NULL);
    1287             : 
    1288             :     /*
    1289             :      * Create a resource owner for the subtransaction.  We make it a child of
    1290             :      * the immediate parent's resource owner.
    1291             :      */
    1292       20040 :     s->curTransactionOwner =
    1293       20040 :         ResourceOwnerCreate(s->parent->curTransactionOwner,
    1294             :                             "SubTransaction");
    1295             : 
    1296       20040 :     CurTransactionResourceOwner = s->curTransactionOwner;
    1297       20040 :     CurrentResourceOwner = s->curTransactionOwner;
    1298       20040 : }
    1299             : 
    1300             : /* ----------------------------------------------------------------
    1301             :  *                      CommitTransaction stuff
    1302             :  * ----------------------------------------------------------------
    1303             :  */
    1304             : 
    1305             : /*
    1306             :  *  RecordTransactionCommit
    1307             :  *
    1308             :  * Returns latest XID among xact and its children, or InvalidTransactionId
    1309             :  * if the xact has no XID.  (We compute that here just because it's easier.)
    1310             :  *
    1311             :  * If you change this function, see RecordTransactionCommitPrepared also.
    1312             :  */
    1313             : static TransactionId
    1314      734544 : RecordTransactionCommit(void)
    1315             : {
    1316      734544 :     TransactionId xid = GetTopTransactionIdIfAny();
    1317      734544 :     bool        markXidCommitted = TransactionIdIsValid(xid);
    1318      734544 :     TransactionId latestXid = InvalidTransactionId;
    1319             :     int         nrels;
    1320             :     RelFileLocator *rels;
    1321             :     int         nchildren;
    1322             :     TransactionId *children;
    1323      734544 :     int         ndroppedstats = 0;
    1324      734544 :     xl_xact_stats_item *droppedstats = NULL;
    1325      734544 :     int         nmsgs = 0;
    1326      734544 :     SharedInvalidationMessage *invalMessages = NULL;
    1327      734544 :     bool        RelcacheInitFileInval = false;
    1328             :     bool        wrote_xlog;
    1329             : 
    1330             :     /*
    1331             :      * Log pending invalidations for logical decoding of in-progress
    1332             :      * transactions.  Normally for DDLs, we log this at each command end,
    1333             :      * however, for certain cases where we directly update the system table
    1334             :      * without a transaction block, the invalidations are not logged till this
    1335             :      * time.
    1336             :      */
    1337      734544 :     if (XLogLogicalInfoActive())
    1338       22810 :         LogLogicalInvalidations();
    1339             : 
    1340             :     /* Get data needed for commit record */
    1341      734544 :     nrels = smgrGetPendingDeletes(true, &rels);
    1342      734544 :     nchildren = xactGetCommittedChildren(&children);
    1343      734544 :     ndroppedstats = pgstat_get_transactional_drops(true, &droppedstats);
    1344      734544 :     if (XLogStandbyInfoActive())
    1345      372274 :         nmsgs = xactGetCommittedInvalidationMessages(&invalMessages,
    1346             :                                                      &RelcacheInitFileInval);
    1347      734544 :     wrote_xlog = (XactLastRecEnd != 0);
    1348             : 
    1349             :     /*
    1350             :      * If we haven't been assigned an XID yet, we neither can, nor do we want
    1351             :      * to write a COMMIT record.
    1352             :      */
    1353      734544 :     if (!markXidCommitted)
    1354             :     {
    1355             :         /*
    1356             :          * We expect that every RelationDropStorage is followed by a catalog
    1357             :          * update, and hence XID assignment, so we shouldn't get here with any
    1358             :          * pending deletes. Same is true for dropping stats.
    1359             :          *
    1360             :          * Use a real test not just an Assert to check this, since it's a bit
    1361             :          * fragile.
    1362             :          */
    1363      501160 :         if (nrels != 0 || ndroppedstats != 0)
    1364           0 :             elog(ERROR, "cannot commit a transaction that deleted files but has no xid");
    1365             : 
    1366             :         /* Can't have child XIDs either; AssignTransactionId enforces this */
    1367             :         Assert(nchildren == 0);
    1368             : 
    1369             :         /*
    1370             :          * Transactions without an assigned xid can contain invalidation
    1371             :          * messages.  While inplace updates do this, this is not known to be
    1372             :          * necessary; see comment at inplace CacheInvalidateHeapTuple().
    1373             :          * Extensions might still rely on this capability, and standbys may
    1374             :          * need to process those invals.  We can't emit a commit record
    1375             :          * without an xid, and we don't want to force assigning an xid,
    1376             :          * because that'd be problematic for e.g. vacuum.  Hence we emit a
    1377             :          * bespoke record for the invalidations. We don't want to use that in
    1378             :          * case a commit record is emitted, so they happen synchronously with
    1379             :          * commits (besides not wanting to emit more WAL records).
    1380             :          *
    1381             :          * XXX Every known use of this capability is a defect.  Since an XID
    1382             :          * isn't controlling visibility of the change that prompted invals,
    1383             :          * other sessions need the inval even if this transactions aborts.
    1384             :          *
    1385             :          * ON COMMIT DELETE ROWS does a nontransactional index_build(), which
    1386             :          * queues a relcache inval, including in transactions without an xid
    1387             :          * that had read the (empty) table.  Standbys don't need any ON COMMIT
    1388             :          * DELETE ROWS invals, but we've not done the work to withhold them.
    1389             :          */
    1390      501160 :         if (nmsgs != 0)
    1391             :         {
    1392       15724 :             LogStandbyInvalidations(nmsgs, invalMessages,
    1393             :                                     RelcacheInitFileInval);
    1394       15724 :             wrote_xlog = true;  /* not strictly necessary */
    1395             :         }
    1396             : 
    1397             :         /*
    1398             :          * If we didn't create XLOG entries, we're done here; otherwise we
    1399             :          * should trigger flushing those entries the same as a commit record
    1400             :          * would.  This will primarily happen for HOT pruning and the like; we
    1401             :          * want these to be flushed to disk in due time.
    1402             :          */
    1403      501160 :         if (!wrote_xlog)
    1404      436180 :             goto cleanup;
    1405             :     }
    1406             :     else
    1407             :     {
    1408             :         bool        replorigin;
    1409             : 
    1410             :         /*
    1411             :          * Are we using the replication origins feature?  Or, in other words,
    1412             :          * are we replaying remote actions?
    1413             :          */
    1414      235318 :         replorigin = (replorigin_session_origin != InvalidRepOriginId &&
    1415        1934 :                       replorigin_session_origin != DoNotReplicateId);
    1416             : 
    1417             :         /*
    1418             :          * Mark ourselves as within our "commit critical section".  This
    1419             :          * forces any concurrent checkpoint to wait until we've updated
    1420             :          * pg_xact.  Without this, it is possible for the checkpoint to set
    1421             :          * REDO after the XLOG record but fail to flush the pg_xact update to
    1422             :          * disk, leading to loss of the transaction commit if the system
    1423             :          * crashes a little later.
    1424             :          *
    1425             :          * Note: we could, but don't bother to, set this flag in
    1426             :          * RecordTransactionAbort.  That's because loss of a transaction abort
    1427             :          * is noncritical; the presumption would be that it aborted, anyway.
    1428             :          *
    1429             :          * It's safe to change the delayChkptFlags flag of our own backend
    1430             :          * without holding the ProcArrayLock, since we're the only one
    1431             :          * modifying it.  This makes checkpoint's determination of which xacts
    1432             :          * are delaying the checkpoint a bit fuzzy, but it doesn't matter.
    1433             :          */
    1434             :         Assert((MyProc->delayChkptFlags & DELAY_CHKPT_START) == 0);
    1435      233384 :         START_CRIT_SECTION();
    1436      233384 :         MyProc->delayChkptFlags |= DELAY_CHKPT_START;
    1437             : 
    1438             :         /*
    1439             :          * Insert the commit XLOG record.
    1440             :          */
    1441      233384 :         XactLogCommitRecord(GetCurrentTransactionStopTimestamp(),
    1442             :                             nchildren, children, nrels, rels,
    1443             :                             ndroppedstats, droppedstats,
    1444             :                             nmsgs, invalMessages,
    1445             :                             RelcacheInitFileInval,
    1446             :                             MyXactFlags,
    1447             :                             InvalidTransactionId, NULL /* plain commit */ );
    1448             : 
    1449      233384 :         if (replorigin)
    1450             :             /* Move LSNs forward for this replication origin */
    1451        1934 :             replorigin_session_advance(replorigin_session_origin_lsn,
    1452             :                                        XactLastRecEnd);
    1453             : 
    1454             :         /*
    1455             :          * Record commit timestamp.  The value comes from plain commit
    1456             :          * timestamp if there's no replication origin; otherwise, the
    1457             :          * timestamp was already set in replorigin_session_origin_timestamp by
    1458             :          * replication.
    1459             :          *
    1460             :          * We don't need to WAL-log anything here, as the commit record
    1461             :          * written above already contains the data.
    1462             :          */
    1463             : 
    1464      233384 :         if (!replorigin || replorigin_session_origin_timestamp == 0)
    1465      231642 :             replorigin_session_origin_timestamp = GetCurrentTransactionStopTimestamp();
    1466             : 
    1467      233384 :         TransactionTreeSetCommitTsData(xid, nchildren, children,
    1468             :                                        replorigin_session_origin_timestamp,
    1469             :                                        replorigin_session_origin);
    1470             :     }
    1471             : 
    1472             :     /*
    1473             :      * Check if we want to commit asynchronously.  We can allow the XLOG flush
    1474             :      * to happen asynchronously if synchronous_commit=off, or if the current
    1475             :      * transaction has not performed any WAL-logged operation or didn't assign
    1476             :      * an xid.  The transaction can end up not writing any WAL, even if it has
    1477             :      * an xid, if it only wrote to temporary and/or unlogged tables.  It can
    1478             :      * end up having written WAL without an xid if it did HOT pruning.  In
    1479             :      * case of a crash, the loss of such a transaction will be irrelevant;
    1480             :      * temp tables will be lost anyway, unlogged tables will be truncated and
    1481             :      * HOT pruning will be done again later. (Given the foregoing, you might
    1482             :      * think that it would be unnecessary to emit the XLOG record at all in
    1483             :      * this case, but we don't currently try to do that.  It would certainly
    1484             :      * cause problems at least in Hot Standby mode, where the
    1485             :      * KnownAssignedXids machinery requires tracking every XID assignment.  It
    1486             :      * might be OK to skip it only when wal_level < replica, but for now we
    1487             :      * don't.)
    1488             :      *
    1489             :      * However, if we're doing cleanup of any non-temp rels or committing any
    1490             :      * command that wanted to force sync commit, then we must flush XLOG
    1491             :      * immediately.  (We must not allow asynchronous commit if there are any
    1492             :      * non-temp tables to be deleted, because we might delete the files before
    1493             :      * the COMMIT record is flushed to disk.  We do allow asynchronous commit
    1494             :      * if all to-be-deleted tables are temporary though, since they are lost
    1495             :      * anyway if we crash.)
    1496             :      */
    1497      298364 :     if ((wrote_xlog && markXidCommitted &&
    1498      298364 :          synchronous_commit > SYNCHRONOUS_COMMIT_OFF) ||
    1499       76102 :         forceSyncCommit || nrels > 0)
    1500             :     {
    1501      222302 :         XLogFlush(XactLastRecEnd);
    1502             : 
    1503             :         /*
    1504             :          * Now we may update the CLOG, if we wrote a COMMIT record above
    1505             :          */
    1506      222302 :         if (markXidCommitted)
    1507      222302 :             TransactionIdCommitTree(xid, nchildren, children);
    1508             :     }
    1509             :     else
    1510             :     {
    1511             :         /*
    1512             :          * Asynchronous commit case:
    1513             :          *
    1514             :          * This enables possible committed transaction loss in the case of a
    1515             :          * postmaster crash because WAL buffers are left unwritten. Ideally we
    1516             :          * could issue the WAL write without the fsync, but some
    1517             :          * wal_sync_methods do not allow separate write/fsync.
    1518             :          *
    1519             :          * Report the latest async commit LSN, so that the WAL writer knows to
    1520             :          * flush this commit.
    1521             :          */
    1522       76062 :         XLogSetAsyncXactLSN(XactLastRecEnd);
    1523             : 
    1524             :         /*
    1525             :          * We must not immediately update the CLOG, since we didn't flush the
    1526             :          * XLOG. Instead, we store the LSN up to which the XLOG must be
    1527             :          * flushed before the CLOG may be updated.
    1528             :          */
    1529       76062 :         if (markXidCommitted)
    1530       11082 :             TransactionIdAsyncCommitTree(xid, nchildren, children, XactLastRecEnd);
    1531             :     }
    1532             : 
    1533             :     /*
    1534             :      * If we entered a commit critical section, leave it now, and let
    1535             :      * checkpoints proceed.
    1536             :      */
    1537      298364 :     if (markXidCommitted)
    1538             :     {
    1539      233384 :         MyProc->delayChkptFlags &= ~DELAY_CHKPT_START;
    1540      233384 :         END_CRIT_SECTION();
    1541             :     }
    1542             : 
    1543             :     /* Compute latestXid while we have the child XIDs handy */
    1544      298364 :     latestXid = TransactionIdLatest(xid, nchildren, children);
    1545             : 
    1546             :     /*
    1547             :      * Wait for synchronous replication, if required. Similar to the decision
    1548             :      * above about using committing asynchronously we only want to wait if
    1549             :      * this backend assigned an xid and wrote WAL.  No need to wait if an xid
    1550             :      * was assigned due to temporary/unlogged tables or due to HOT pruning.
    1551             :      *
    1552             :      * Note that at this stage we have marked clog, but still show as running
    1553             :      * in the procarray and continue to hold locks.
    1554             :      */
    1555      298364 :     if (wrote_xlog && markXidCommitted)
    1556      225514 :         SyncRepWaitForLSN(XactLastRecEnd, true);
    1557             : 
    1558             :     /* remember end of last commit record */
    1559      298364 :     XactLastCommitEnd = XactLastRecEnd;
    1560             : 
    1561             :     /* Reset XactLastRecEnd until the next transaction writes something */
    1562      298364 :     XactLastRecEnd = 0;
    1563      734544 : cleanup:
    1564             :     /* Clean up local data */
    1565      734544 :     if (rels)
    1566       17682 :         pfree(rels);
    1567      734544 :     if (ndroppedstats)
    1568       20352 :         pfree(droppedstats);
    1569             : 
    1570      734544 :     return latestXid;
    1571             : }
    1572             : 
    1573             : 
    1574             : /*
    1575             :  *  AtCCI_LocalCache
    1576             :  */
    1577             : static void
    1578     1070094 : AtCCI_LocalCache(void)
    1579             : {
    1580             :     /*
    1581             :      * Make any pending relation map changes visible.  We must do this before
    1582             :      * processing local sinval messages, so that the map changes will get
    1583             :      * reflected into the relcache when relcache invals are processed.
    1584             :      */
    1585     1070094 :     AtCCI_RelationMap();
    1586             : 
    1587             :     /*
    1588             :      * Make catalog changes visible to me for the next command.
    1589             :      */
    1590     1070094 :     CommandEndInvalidationMessages();
    1591     1070088 : }
    1592             : 
    1593             : /*
    1594             :  *  AtCommit_Memory
    1595             :  */
    1596             : static void
    1597      738000 : AtCommit_Memory(void)
    1598             : {
    1599      738000 :     TransactionState s = CurrentTransactionState;
    1600             : 
    1601             :     /*
    1602             :      * Return to the memory context that was current before we started the
    1603             :      * transaction.  (In principle, this could not be any of the contexts we
    1604             :      * are about to delete.  If it somehow is, assertions in mcxt.c will
    1605             :      * complain.)
    1606             :      */
    1607      738000 :     MemoryContextSwitchTo(s->priorContext);
    1608             : 
    1609             :     /*
    1610             :      * Release all transaction-local memory.  TopTransactionContext survives
    1611             :      * but becomes empty; any sub-contexts go away.
    1612             :      */
    1613             :     Assert(TopTransactionContext != NULL);
    1614      738000 :     MemoryContextReset(TopTransactionContext);
    1615             : 
    1616             :     /*
    1617             :      * Clear these pointers as a pro-forma matter.  (Notionally, while
    1618             :      * TopTransactionContext still exists, it's currently not associated with
    1619             :      * this TransactionState struct.)
    1620             :      */
    1621      738000 :     CurTransactionContext = NULL;
    1622      738000 :     s->curTransactionContext = NULL;
    1623      738000 : }
    1624             : 
    1625             : /* ----------------------------------------------------------------
    1626             :  *                      CommitSubTransaction stuff
    1627             :  * ----------------------------------------------------------------
    1628             :  */
    1629             : 
    1630             : /*
    1631             :  * AtSubCommit_Memory
    1632             :  */
    1633             : static void
    1634       10772 : AtSubCommit_Memory(void)
    1635             : {
    1636       10772 :     TransactionState s = CurrentTransactionState;
    1637             : 
    1638             :     Assert(s->parent != NULL);
    1639             : 
    1640             :     /* Return to parent transaction level's memory context. */
    1641       10772 :     CurTransactionContext = s->parent->curTransactionContext;
    1642       10772 :     MemoryContextSwitchTo(CurTransactionContext);
    1643             : 
    1644             :     /*
    1645             :      * Ordinarily we cannot throw away the child's CurTransactionContext,
    1646             :      * since the data it contains will be needed at upper commit.  However, if
    1647             :      * there isn't actually anything in it, we can throw it away.  This avoids
    1648             :      * a small memory leak in the common case of "trivial" subxacts.
    1649             :      */
    1650       10772 :     if (MemoryContextIsEmpty(s->curTransactionContext))
    1651             :     {
    1652       10754 :         MemoryContextDelete(s->curTransactionContext);
    1653       10754 :         s->curTransactionContext = NULL;
    1654             :     }
    1655       10772 : }
    1656             : 
    1657             : /*
    1658             :  * AtSubCommit_childXids
    1659             :  *
    1660             :  * Pass my own XID and my child XIDs up to my parent as committed children.
    1661             :  */
    1662             : static void
    1663        7474 : AtSubCommit_childXids(void)
    1664             : {
    1665        7474 :     TransactionState s = CurrentTransactionState;
    1666             :     int         new_nChildXids;
    1667             : 
    1668             :     Assert(s->parent != NULL);
    1669             : 
    1670             :     /*
    1671             :      * The parent childXids array will need to hold my XID and all my
    1672             :      * childXids, in addition to the XIDs already there.
    1673             :      */
    1674        7474 :     new_nChildXids = s->parent->nChildXids + s->nChildXids + 1;
    1675             : 
    1676             :     /* Allocate or enlarge the parent array if necessary */
    1677        7474 :     if (s->parent->maxChildXids < new_nChildXids)
    1678             :     {
    1679             :         int         new_maxChildXids;
    1680             :         TransactionId *new_childXids;
    1681             : 
    1682             :         /*
    1683             :          * Make it 2x what's needed right now, to avoid having to enlarge it
    1684             :          * repeatedly. But we can't go above MaxAllocSize.  (The latter limit
    1685             :          * is what ensures that we don't need to worry about integer overflow
    1686             :          * here or in the calculation of new_nChildXids.)
    1687             :          */
    1688        3406 :         new_maxChildXids = Min(new_nChildXids * 2,
    1689             :                                (int) (MaxAllocSize / sizeof(TransactionId)));
    1690             : 
    1691        3406 :         if (new_maxChildXids < new_nChildXids)
    1692           0 :             ereport(ERROR,
    1693             :                     (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
    1694             :                      errmsg("maximum number of committed subtransactions (%d) exceeded",
    1695             :                             (int) (MaxAllocSize / sizeof(TransactionId)))));
    1696             : 
    1697             :         /*
    1698             :          * We keep the child-XID arrays in TopTransactionContext; this avoids
    1699             :          * setting up child-transaction contexts for what might be just a few
    1700             :          * bytes of grandchild XIDs.
    1701             :          */
    1702        3406 :         if (s->parent->childXids == NULL)
    1703             :             new_childXids =
    1704        3322 :                 MemoryContextAlloc(TopTransactionContext,
    1705             :                                    new_maxChildXids * sizeof(TransactionId));
    1706             :         else
    1707          84 :             new_childXids = repalloc(s->parent->childXids,
    1708             :                                      new_maxChildXids * sizeof(TransactionId));
    1709             : 
    1710        3406 :         s->parent->childXids = new_childXids;
    1711        3406 :         s->parent->maxChildXids = new_maxChildXids;
    1712             :     }
    1713             : 
    1714             :     /*
    1715             :      * Copy all my XIDs to parent's array.
    1716             :      *
    1717             :      * Note: We rely on the fact that the XID of a child always follows that
    1718             :      * of its parent.  By copying the XID of this subtransaction before the
    1719             :      * XIDs of its children, we ensure that the array stays ordered. Likewise,
    1720             :      * all XIDs already in the array belong to subtransactions started and
    1721             :      * subcommitted before us, so their XIDs must precede ours.
    1722             :      */
    1723        7474 :     s->parent->childXids[s->parent->nChildXids] = XidFromFullTransactionId(s->fullTransactionId);
    1724             : 
    1725        7474 :     if (s->nChildXids > 0)
    1726        2016 :         memcpy(&s->parent->childXids[s->parent->nChildXids + 1],
    1727        2016 :                s->childXids,
    1728        2016 :                s->nChildXids * sizeof(TransactionId));
    1729             : 
    1730        7474 :     s->parent->nChildXids = new_nChildXids;
    1731             : 
    1732             :     /* Release child's array to avoid leakage */
    1733        7474 :     if (s->childXids != NULL)
    1734        2016 :         pfree(s->childXids);
    1735             :     /* We must reset these to avoid double-free if fail later in commit */
    1736        7474 :     s->childXids = NULL;
    1737        7474 :     s->nChildXids = 0;
    1738        7474 :     s->maxChildXids = 0;
    1739        7474 : }
    1740             : 
    1741             : /* ----------------------------------------------------------------
    1742             :  *                      AbortTransaction stuff
    1743             :  * ----------------------------------------------------------------
    1744             :  */
    1745             : 
    1746             : /*
    1747             :  *  RecordTransactionAbort
    1748             :  *
    1749             :  * Returns latest XID among xact and its children, or InvalidTransactionId
    1750             :  * if the xact has no XID.  (We compute that here just because it's easier.)
    1751             :  */
    1752             : static TransactionId
    1753       56358 : RecordTransactionAbort(bool isSubXact)
    1754             : {
    1755       56358 :     TransactionId xid = GetCurrentTransactionIdIfAny();
    1756             :     TransactionId latestXid;
    1757             :     int         nrels;
    1758             :     RelFileLocator *rels;
    1759       56358 :     int         ndroppedstats = 0;
    1760       56358 :     xl_xact_stats_item *droppedstats = NULL;
    1761             :     int         nchildren;
    1762             :     TransactionId *children;
    1763             :     TimestampTz xact_time;
    1764             :     bool        replorigin;
    1765             : 
    1766             :     /*
    1767             :      * If we haven't been assigned an XID, nobody will care whether we aborted
    1768             :      * or not.  Hence, we're done in that case.  It does not matter if we have
    1769             :      * rels to delete (note that this routine is not responsible for actually
    1770             :      * deleting 'em).  We cannot have any child XIDs, either.
    1771             :      */
    1772       56358 :     if (!TransactionIdIsValid(xid))
    1773             :     {
    1774             :         /* Reset XactLastRecEnd until the next transaction writes something */
    1775       45232 :         if (!isSubXact)
    1776       37262 :             XactLastRecEnd = 0;
    1777       45232 :         return InvalidTransactionId;
    1778             :     }
    1779             : 
    1780             :     /*
    1781             :      * We have a valid XID, so we should write an ABORT record for it.
    1782             :      *
    1783             :      * We do not flush XLOG to disk here, since the default assumption after a
    1784             :      * crash would be that we aborted, anyway.  For the same reason, we don't
    1785             :      * need to worry about interlocking against checkpoint start.
    1786             :      */
    1787             : 
    1788             :     /*
    1789             :      * Check that we haven't aborted halfway through RecordTransactionCommit.
    1790             :      */
    1791       11126 :     if (TransactionIdDidCommit(xid))
    1792           0 :         elog(PANIC, "cannot abort transaction %u, it was already committed",
    1793             :              xid);
    1794             : 
    1795             :     /*
    1796             :      * Are we using the replication origins feature?  Or, in other words, are
    1797             :      * we replaying remote actions?
    1798             :      */
    1799       11172 :     replorigin = (replorigin_session_origin != InvalidRepOriginId &&
    1800          46 :                   replorigin_session_origin != DoNotReplicateId);
    1801             : 
    1802             :     /* Fetch the data we need for the abort record */
    1803       11126 :     nrels = smgrGetPendingDeletes(false, &rels);
    1804       11126 :     nchildren = xactGetCommittedChildren(&children);
    1805       11126 :     ndroppedstats = pgstat_get_transactional_drops(false, &droppedstats);
    1806             : 
    1807             :     /* XXX do we really need a critical section here? */
    1808       11126 :     START_CRIT_SECTION();
    1809             : 
    1810             :     /* Write the ABORT record */
    1811       11126 :     if (isSubXact)
    1812        1298 :         xact_time = GetCurrentTimestamp();
    1813             :     else
    1814             :     {
    1815        9828 :         xact_time = GetCurrentTransactionStopTimestamp();
    1816             :     }
    1817             : 
    1818       11126 :     XactLogAbortRecord(xact_time,
    1819             :                        nchildren, children,
    1820             :                        nrels, rels,
    1821             :                        ndroppedstats, droppedstats,
    1822             :                        MyXactFlags, InvalidTransactionId,
    1823             :                        NULL);
    1824             : 
    1825       11126 :     if (replorigin)
    1826             :         /* Move LSNs forward for this replication origin */
    1827          46 :         replorigin_session_advance(replorigin_session_origin_lsn,
    1828             :                                    XactLastRecEnd);
    1829             : 
    1830             :     /*
    1831             :      * Report the latest async abort LSN, so that the WAL writer knows to
    1832             :      * flush this abort. There's nothing to be gained by delaying this, since
    1833             :      * WALWriter may as well do this when it can. This is important with
    1834             :      * streaming replication because if we don't flush WAL regularly we will
    1835             :      * find that large aborts leave us with a long backlog for when commits
    1836             :      * occur after the abort, increasing our window of data loss should
    1837             :      * problems occur at that point.
    1838             :      */
    1839       11126 :     if (!isSubXact)
    1840        9828 :         XLogSetAsyncXactLSN(XactLastRecEnd);
    1841             : 
    1842             :     /*
    1843             :      * Mark the transaction aborted in clog.  This is not absolutely necessary
    1844             :      * but we may as well do it while we are here; also, in the subxact case
    1845             :      * it is helpful because XactLockTableWait makes use of it to avoid
    1846             :      * waiting for already-aborted subtransactions.  It is OK to do it without
    1847             :      * having flushed the ABORT record to disk, because in event of a crash
    1848             :      * we'd be assumed to have aborted anyway.
    1849             :      */
    1850       11126 :     TransactionIdAbortTree(xid, nchildren, children);
    1851             : 
    1852       11126 :     END_CRIT_SECTION();
    1853             : 
    1854             :     /* Compute latestXid while we have the child XIDs handy */
    1855       11126 :     latestXid = TransactionIdLatest(xid, nchildren, children);
    1856             : 
    1857             :     /*
    1858             :      * If we're aborting a subtransaction, we can immediately remove failed
    1859             :      * XIDs from PGPROC's cache of running child XIDs.  We do that here for
    1860             :      * subxacts, because we already have the child XID array at hand.  For
    1861             :      * main xacts, the equivalent happens just after this function returns.
    1862             :      */
    1863       11126 :     if (isSubXact)
    1864        1298 :         XidCacheRemoveRunningXids(xid, nchildren, children, latestXid);
    1865             : 
    1866             :     /* Reset XactLastRecEnd until the next transaction writes something */
    1867       11126 :     if (!isSubXact)
    1868        9828 :         XactLastRecEnd = 0;
    1869             : 
    1870             :     /* And clean up local data */
    1871       11126 :     if (rels)
    1872        1798 :         pfree(rels);
    1873       11126 :     if (ndroppedstats)
    1874        2502 :         pfree(droppedstats);
    1875             : 
    1876       11126 :     return latestXid;
    1877             : }
    1878             : 
    1879             : /*
    1880             :  *  AtAbort_Memory
    1881             :  */
    1882             : static void
    1883       77920 : AtAbort_Memory(void)
    1884             : {
    1885             :     /*
    1886             :      * Switch into TransactionAbortContext, which should have some free space
    1887             :      * even if nothing else does.  We'll work in this context until we've
    1888             :      * finished cleaning up.
    1889             :      *
    1890             :      * It is barely possible to get here when we've not been able to create
    1891             :      * TransactionAbortContext yet; if so use TopMemoryContext.
    1892             :      */
    1893       77920 :     if (TransactionAbortContext != NULL)
    1894       77920 :         MemoryContextSwitchTo(TransactionAbortContext);
    1895             :     else
    1896           0 :         MemoryContextSwitchTo(TopMemoryContext);
    1897       77920 : }
    1898             : 
    1899             : /*
    1900             :  * AtSubAbort_Memory
    1901             :  */
    1902             : static void
    1903        9268 : AtSubAbort_Memory(void)
    1904             : {
    1905             :     Assert(TransactionAbortContext != NULL);
    1906             : 
    1907        9268 :     MemoryContextSwitchTo(TransactionAbortContext);
    1908        9268 : }
    1909             : 
    1910             : 
    1911             : /*
    1912             :  *  AtAbort_ResourceOwner
    1913             :  */
    1914             : static void
    1915       47102 : AtAbort_ResourceOwner(void)
    1916             : {
    1917             :     /*
    1918             :      * Make sure we have a valid ResourceOwner, if possible (else it will be
    1919             :      * NULL, which is OK)
    1920             :      */
    1921       47102 :     CurrentResourceOwner = TopTransactionResourceOwner;
    1922       47102 : }
    1923             : 
    1924             : /*
    1925             :  * AtSubAbort_ResourceOwner
    1926             :  */
    1927             : static void
    1928        9268 : AtSubAbort_ResourceOwner(void)
    1929             : {
    1930        9268 :     TransactionState s = CurrentTransactionState;
    1931             : 
    1932             :     /* Make sure we have a valid ResourceOwner */
    1933        9268 :     CurrentResourceOwner = s->curTransactionOwner;
    1934        9268 : }
    1935             : 
    1936             : 
    1937             : /*
    1938             :  * AtSubAbort_childXids
    1939             :  */
    1940             : static void
    1941        1298 : AtSubAbort_childXids(void)
    1942             : {
    1943        1298 :     TransactionState s = CurrentTransactionState;
    1944             : 
    1945             :     /*
    1946             :      * We keep the child-XID arrays in TopTransactionContext (see
    1947             :      * AtSubCommit_childXids).  This means we'd better free the array
    1948             :      * explicitly at abort to avoid leakage.
    1949             :      */
    1950        1298 :     if (s->childXids != NULL)
    1951          50 :         pfree(s->childXids);
    1952        1298 :     s->childXids = NULL;
    1953        1298 :     s->nChildXids = 0;
    1954        1298 :     s->maxChildXids = 0;
    1955             : 
    1956             :     /*
    1957             :      * We could prune the unreportedXids array here. But we don't bother. That
    1958             :      * would potentially reduce number of XLOG_XACT_ASSIGNMENT records but it
    1959             :      * would likely introduce more CPU time into the more common paths, so we
    1960             :      * choose not to do that.
    1961             :      */
    1962        1298 : }
    1963             : 
    1964             : /* ----------------------------------------------------------------
    1965             :  *                      CleanupTransaction stuff
    1966             :  * ----------------------------------------------------------------
    1967             :  */
    1968             : 
    1969             : /*
    1970             :  *  AtCleanup_Memory
    1971             :  */
    1972             : static void
    1973       47102 : AtCleanup_Memory(void)
    1974             : {
    1975       47102 :     TransactionState s = CurrentTransactionState;
    1976             : 
    1977             :     /* Should be at top level */
    1978             :     Assert(s->parent == NULL);
    1979             : 
    1980             :     /*
    1981             :      * Return to the memory context that was current before we started the
    1982             :      * transaction.  (In principle, this could not be any of the contexts we
    1983             :      * are about to delete.  If it somehow is, assertions in mcxt.c will
    1984             :      * complain.)
    1985             :      */
    1986       47102 :     MemoryContextSwitchTo(s->priorContext);
    1987             : 
    1988             :     /*
    1989             :      * Clear the special abort context for next time.
    1990             :      */
    1991       47102 :     if (TransactionAbortContext != NULL)
    1992       47102 :         MemoryContextReset(TransactionAbortContext);
    1993             : 
    1994             :     /*
    1995             :      * Release all transaction-local memory, the same as in AtCommit_Memory,
    1996             :      * except we must cope with the possibility that we didn't get as far as
    1997             :      * creating TopTransactionContext.
    1998             :      */
    1999       47102 :     if (TopTransactionContext != NULL)
    2000       47102 :         MemoryContextReset(TopTransactionContext);
    2001             : 
    2002             :     /*
    2003             :      * Clear these pointers as a pro-forma matter.  (Notionally, while
    2004             :      * TopTransactionContext still exists, it's currently not associated with
    2005             :      * this TransactionState struct.)
    2006             :      */
    2007       47102 :     CurTransactionContext = NULL;
    2008       47102 :     s->curTransactionContext = NULL;
    2009       47102 : }
    2010             : 
    2011             : 
    2012             : /* ----------------------------------------------------------------
    2013             :  *                      CleanupSubTransaction stuff
    2014             :  * ----------------------------------------------------------------
    2015             :  */
    2016             : 
    2017             : /*
    2018             :  * AtSubCleanup_Memory
    2019             :  */
    2020             : static void
    2021        9268 : AtSubCleanup_Memory(void)
    2022             : {
    2023        9268 :     TransactionState s = CurrentTransactionState;
    2024             : 
    2025             :     Assert(s->parent != NULL);
    2026             : 
    2027             :     /*
    2028             :      * Return to the memory context that was current before we started the
    2029             :      * subtransaction.  (In principle, this could not be any of the contexts
    2030             :      * we are about to delete.  If it somehow is, assertions in mcxt.c will
    2031             :      * complain.)
    2032             :      */
    2033        9268 :     MemoryContextSwitchTo(s->priorContext);
    2034             : 
    2035             :     /* Update CurTransactionContext (might not be same as priorContext) */
    2036        9268 :     CurTransactionContext = s->parent->curTransactionContext;
    2037             : 
    2038             :     /*
    2039             :      * Clear the special abort context for next time.
    2040             :      */
    2041        9268 :     if (TransactionAbortContext != NULL)
    2042        9268 :         MemoryContextReset(TransactionAbortContext);
    2043             : 
    2044             :     /*
    2045             :      * Delete the subxact local memory contexts. Its CurTransactionContext can
    2046             :      * go too (note this also kills CurTransactionContexts from any children
    2047             :      * of the subxact).
    2048             :      */
    2049        9268 :     if (s->curTransactionContext)
    2050        9268 :         MemoryContextDelete(s->curTransactionContext);
    2051        9268 :     s->curTransactionContext = NULL;
    2052        9268 : }
    2053             : 
    2054             : /* ----------------------------------------------------------------
    2055             :  *                      interface routines
    2056             :  * ----------------------------------------------------------------
    2057             :  */
    2058             : 
    2059             : /*
    2060             :  *  StartTransaction
    2061             :  */
    2062             : static void
    2063      785102 : StartTransaction(void)
    2064             : {
    2065             :     TransactionState s;
    2066             :     VirtualTransactionId vxid;
    2067             : 
    2068             :     /*
    2069             :      * Let's just make sure the state stack is empty
    2070             :      */
    2071      785102 :     s = &TopTransactionStateData;
    2072      785102 :     CurrentTransactionState = s;
    2073             : 
    2074             :     Assert(!FullTransactionIdIsValid(XactTopFullTransactionId));
    2075             : 
    2076             :     /* check the current transaction state */
    2077             :     Assert(s->state == TRANS_DEFAULT);
    2078             : 
    2079             :     /*
    2080             :      * Set the current transaction state information appropriately during
    2081             :      * start processing.  Note that once the transaction status is switched
    2082             :      * this process cannot fail until the user ID and the security context
    2083             :      * flags are fetched below.
    2084             :      */
    2085      785102 :     s->state = TRANS_START;
    2086      785102 :     s->fullTransactionId = InvalidFullTransactionId; /* until assigned */
    2087             : 
    2088             :     /* Determine if statements are logged in this transaction */
    2089      785102 :     xact_is_sampled = log_xact_sample_rate != 0 &&
    2090           0 :         (log_xact_sample_rate == 1 ||
    2091           0 :          pg_prng_double(&pg_global_prng_state) <= log_xact_sample_rate);
    2092             : 
    2093             :     /*
    2094             :      * initialize current transaction state fields
    2095             :      *
    2096             :      * note: prevXactReadOnly is not used at the outermost level
    2097             :      */
    2098      785102 :     s->nestingLevel = 1;
    2099      785102 :     s->gucNestLevel = 1;
    2100      785102 :     s->childXids = NULL;
    2101      785102 :     s->nChildXids = 0;
    2102      785102 :     s->maxChildXids = 0;
    2103             : 
    2104             :     /*
    2105             :      * Once the current user ID and the security context flags are fetched,
    2106             :      * both will be properly reset even if transaction startup fails.
    2107             :      */
    2108      785102 :     GetUserIdAndSecContext(&s->prevUser, &s->prevSecContext);
    2109             : 
    2110             :     /* SecurityRestrictionContext should never be set outside a transaction */
    2111             :     Assert(s->prevSecContext == 0);
    2112             : 
    2113             :     /*
    2114             :      * Make sure we've reset xact state variables
    2115             :      *
    2116             :      * If recovery is still in progress, mark this transaction as read-only.
    2117             :      * We have lower level defences in XLogInsert and elsewhere to stop us
    2118             :      * from modifying data during recovery, but this gives the normal
    2119             :      * indication to the user that the transaction is read-only.
    2120             :      */
    2121      785102 :     if (RecoveryInProgress())
    2122             :     {
    2123        3004 :         s->startedInRecovery = true;
    2124        3004 :         XactReadOnly = true;
    2125             :     }
    2126             :     else
    2127             :     {
    2128      782098 :         s->startedInRecovery = false;
    2129      782098 :         XactReadOnly = DefaultXactReadOnly;
    2130             :     }
    2131      785102 :     XactDeferrable = DefaultXactDeferrable;
    2132      785102 :     XactIsoLevel = DefaultXactIsoLevel;
    2133      785102 :     forceSyncCommit = false;
    2134      785102 :     MyXactFlags = 0;
    2135             : 
    2136             :     /*
    2137             :      * reinitialize within-transaction counters
    2138             :      */
    2139      785102 :     s->subTransactionId = TopSubTransactionId;
    2140      785102 :     currentSubTransactionId = TopSubTransactionId;
    2141      785102 :     currentCommandId = FirstCommandId;
    2142      785102 :     currentCommandIdUsed = false;
    2143             : 
    2144             :     /*
    2145             :      * initialize reported xid accounting
    2146             :      */
    2147      785102 :     nUnreportedXids = 0;
    2148      785102 :     s->didLogXid = false;
    2149             : 
    2150             :     /*
    2151             :      * must initialize resource-management stuff first
    2152             :      */
    2153      785102 :     AtStart_Memory();
    2154      785102 :     AtStart_ResourceOwner();
    2155             : 
    2156             :     /*
    2157             :      * Assign a new LocalTransactionId, and combine it with the proc number to
    2158             :      * form a virtual transaction id.
    2159             :      */
    2160      785102 :     vxid.procNumber = MyProcNumber;
    2161      785102 :     vxid.localTransactionId = GetNextLocalTransactionId();
    2162             : 
    2163             :     /*
    2164             :      * Lock the virtual transaction id before we announce it in the proc array
    2165             :      */
    2166      785102 :     VirtualXactLockTableInsert(vxid);
    2167             : 
    2168             :     /*
    2169             :      * Advertise it in the proc array.  We assume assignment of
    2170             :      * localTransactionId is atomic, and the proc number should be set
    2171             :      * already.
    2172             :      */
    2173             :     Assert(MyProc->vxid.procNumber == vxid.procNumber);
    2174      785102 :     MyProc->vxid.lxid = vxid.localTransactionId;
    2175             : 
    2176             :     TRACE_POSTGRESQL_TRANSACTION_START(vxid.localTransactionId);
    2177             : 
    2178             :     /*
    2179             :      * set transaction_timestamp() (a/k/a now()).  Normally, we want this to
    2180             :      * be the same as the first command's statement_timestamp(), so don't do a
    2181             :      * fresh GetCurrentTimestamp() call (which'd be expensive anyway).  But
    2182             :      * for transactions started inside procedures (i.e., nonatomic SPI
    2183             :      * contexts), we do need to advance the timestamp.  Also, in a parallel
    2184             :      * worker, the timestamp should already have been provided by a call to
    2185             :      * SetParallelStartTimestamps().
    2186             :      */
    2187      785102 :     if (!IsParallelWorker())
    2188             :     {
    2189      776966 :         if (!SPI_inside_nonatomic_context())
    2190      772552 :             xactStartTimestamp = stmtStartTimestamp;
    2191             :         else
    2192        4414 :             xactStartTimestamp = GetCurrentTimestamp();
    2193             :     }
    2194             :     else
    2195             :         Assert(xactStartTimestamp != 0);
    2196      785102 :     pgstat_report_xact_timestamp(xactStartTimestamp);
    2197             :     /* Mark xactStopTimestamp as unset. */
    2198      785102 :     xactStopTimestamp = 0;
    2199             : 
    2200             :     /*
    2201             :      * initialize other subsystems for new transaction
    2202             :      */
    2203      785102 :     AtStart_GUC();
    2204      785102 :     AtStart_Cache();
    2205      785102 :     AfterTriggerBeginXact();
    2206             : 
    2207             :     /*
    2208             :      * done with start processing, set current transaction state to "in
    2209             :      * progress"
    2210             :      */
    2211      785102 :     s->state = TRANS_INPROGRESS;
    2212             : 
    2213             :     /* Schedule transaction timeout */
    2214      785102 :     if (TransactionTimeout > 0)
    2215           2 :         enable_timeout_after(TRANSACTION_TIMEOUT, TransactionTimeout);
    2216             : 
    2217      785102 :     ShowTransactionState("StartTransaction");
    2218      785102 : }
    2219             : 
    2220             : 
    2221             : /*
    2222             :  *  CommitTransaction
    2223             :  *
    2224             :  * NB: if you change this routine, better look at PrepareTransaction too!
    2225             :  */
    2226             : static void
    2227      737708 : CommitTransaction(void)
    2228             : {
    2229      737708 :     TransactionState s = CurrentTransactionState;
    2230             :     TransactionId latestXid;
    2231             :     bool        is_parallel_worker;
    2232             : 
    2233      737708 :     is_parallel_worker = (s->blockState == TBLOCK_PARALLEL_INPROGRESS);
    2234             : 
    2235             :     /* Enforce parallel mode restrictions during parallel worker commit. */
    2236      737708 :     if (is_parallel_worker)
    2237        2700 :         EnterParallelMode();
    2238             : 
    2239      737708 :     ShowTransactionState("CommitTransaction");
    2240             : 
    2241             :     /*
    2242             :      * check the current transaction state
    2243             :      */
    2244      737708 :     if (s->state != TRANS_INPROGRESS)
    2245           0 :         elog(WARNING, "CommitTransaction while in %s state",
    2246             :              TransStateAsString(s->state));
    2247             :     Assert(s->parent == NULL);
    2248             : 
    2249             :     /*
    2250             :      * Do pre-commit processing that involves calling user-defined code, such
    2251             :      * as triggers.  SECURITY_RESTRICTED_OPERATION contexts must not queue an
    2252             :      * action that would run here, because that would bypass the sandbox.
    2253             :      * Since closing cursors could queue trigger actions, triggers could open
    2254             :      * cursors, etc, we have to keep looping until there's nothing left to do.
    2255             :      */
    2256             :     for (;;)
    2257             :     {
    2258             :         /*
    2259             :          * Fire all currently pending deferred triggers.
    2260             :          */
    2261      748424 :         AfterTriggerFireDeferred();
    2262             : 
    2263             :         /*
    2264             :          * Close open portals (converting holdable ones into static portals).
    2265             :          * If there weren't any, we are done ... otherwise loop back to check
    2266             :          * if they queued deferred triggers.  Lather, rinse, repeat.
    2267             :          */
    2268      748276 :         if (!PreCommit_Portals(false))
    2269      737560 :             break;
    2270             :     }
    2271             : 
    2272             :     /*
    2273             :      * The remaining actions cannot call any user-defined code, so it's safe
    2274             :      * to start shutting down within-transaction services.  But note that most
    2275             :      * of this stuff could still throw an error, which would switch us into
    2276             :      * the transaction-abort path.
    2277             :      */
    2278             : 
    2279      737560 :     CallXactCallbacks(is_parallel_worker ? XACT_EVENT_PARALLEL_PRE_COMMIT
    2280             :                       : XACT_EVENT_PRE_COMMIT);
    2281             : 
    2282             :     /*
    2283             :      * If this xact has started any unfinished parallel operation, clean up
    2284             :      * its workers, warning about leaked resources.  (But we don't actually
    2285             :      * reset parallelModeLevel till entering TRANS_COMMIT, a bit below.  This
    2286             :      * keeps parallel mode restrictions active as long as possible in a
    2287             :      * parallel worker.)
    2288             :      */
    2289      737560 :     AtEOXact_Parallel(true);
    2290      737560 :     if (is_parallel_worker)
    2291             :     {
    2292        2700 :         if (s->parallelModeLevel != 1)
    2293           0 :             elog(WARNING, "parallelModeLevel is %d not 1 at end of parallel worker transaction",
    2294             :                  s->parallelModeLevel);
    2295             :     }
    2296             :     else
    2297             :     {
    2298      734860 :         if (s->parallelModeLevel != 0)
    2299           0 :             elog(WARNING, "parallelModeLevel is %d not 0 at end of transaction",
    2300             :                  s->parallelModeLevel);
    2301             :     }
    2302             : 
    2303             :     /* Shut down the deferred-trigger manager */
    2304      737560 :     AfterTriggerEndXact(true);
    2305             : 
    2306             :     /*
    2307             :      * Let ON COMMIT management do its thing (must happen after closing
    2308             :      * cursors, to avoid dangling-reference problems)
    2309             :      */
    2310      737560 :     PreCommit_on_commit_actions();
    2311             : 
    2312             :     /*
    2313             :      * Synchronize files that are created and not WAL-logged during this
    2314             :      * transaction. This must happen before AtEOXact_RelationMap(), so that we
    2315             :      * don't see committed-but-broken files after a crash.
    2316             :      */
    2317      737554 :     smgrDoPendingSyncs(true, is_parallel_worker);
    2318             : 
    2319             :     /* close large objects before lower-level cleanup */
    2320      737554 :     AtEOXact_LargeObject(true);
    2321             : 
    2322             :     /*
    2323             :      * Insert notifications sent by NOTIFY commands into the queue.  This
    2324             :      * should be late in the pre-commit sequence to minimize time spent
    2325             :      * holding the notify-insertion lock.  However, this could result in
    2326             :      * creating a snapshot, so we must do it before serializable cleanup.
    2327             :      */
    2328      737554 :     PreCommit_Notify();
    2329             : 
    2330             :     /*
    2331             :      * Mark serializable transaction as complete for predicate locking
    2332             :      * purposes.  This should be done as late as we can put it and still allow
    2333             :      * errors to be raised for failure patterns found at commit.  This is not
    2334             :      * appropriate in a parallel worker however, because we aren't committing
    2335             :      * the leader's transaction and its serializable state will live on.
    2336             :      */
    2337      737554 :     if (!is_parallel_worker)
    2338      734854 :         PreCommit_CheckForSerializationFailure();
    2339             : 
    2340             :     /* Prevent cancel/die interrupt while cleaning up */
    2341      737244 :     HOLD_INTERRUPTS();
    2342             : 
    2343             :     /* Commit updates to the relation map --- do this as late as possible */
    2344      737244 :     AtEOXact_RelationMap(true, is_parallel_worker);
    2345             : 
    2346             :     /*
    2347             :      * set the current transaction state information appropriately during
    2348             :      * commit processing
    2349             :      */
    2350      737244 :     s->state = TRANS_COMMIT;
    2351      737244 :     s->parallelModeLevel = 0;
    2352      737244 :     s->parallelChildXact = false;    /* should be false already */
    2353             : 
    2354             :     /* Disable transaction timeout */
    2355      737244 :     if (TransactionTimeout > 0)
    2356           2 :         disable_timeout(TRANSACTION_TIMEOUT, false);
    2357             : 
    2358      737244 :     if (!is_parallel_worker)
    2359             :     {
    2360             :         /*
    2361             :          * We need to mark our XIDs as committed in pg_xact.  This is where we
    2362             :          * durably commit.
    2363             :          */
    2364      734544 :         latestXid = RecordTransactionCommit();
    2365             :     }
    2366             :     else
    2367             :     {
    2368             :         /*
    2369             :          * We must not mark our XID committed; the parallel leader is
    2370             :          * responsible for that.
    2371             :          */
    2372        2700 :         latestXid = InvalidTransactionId;
    2373             : 
    2374             :         /*
    2375             :          * Make sure the leader will know about any WAL we wrote before it
    2376             :          * commits.
    2377             :          */
    2378        2700 :         ParallelWorkerReportLastRecEnd(XactLastRecEnd);
    2379             :     }
    2380             : 
    2381             :     TRACE_POSTGRESQL_TRANSACTION_COMMIT(MyProc->vxid.lxid);
    2382             : 
    2383             :     /*
    2384             :      * Let others know about no transaction in progress by me. Note that this
    2385             :      * must be done _before_ releasing locks we hold and _after_
    2386             :      * RecordTransactionCommit.
    2387             :      */
    2388      737244 :     ProcArrayEndTransaction(MyProc, latestXid);
    2389             : 
    2390             :     /*
    2391             :      * This is all post-commit cleanup.  Note that if an error is raised here,
    2392             :      * it's too late to abort the transaction.  This should be just
    2393             :      * noncritical resource releasing.
    2394             :      *
    2395             :      * The ordering of operations is not entirely random.  The idea is:
    2396             :      * release resources visible to other backends (eg, files, buffer pins);
    2397             :      * then release locks; then release backend-local resources. We want to
    2398             :      * release locks at the point where any backend waiting for us will see
    2399             :      * our transaction as being fully cleaned up.
    2400             :      *
    2401             :      * Resources that can be associated with individual queries are handled by
    2402             :      * the ResourceOwner mechanism.  The other calls here are for backend-wide
    2403             :      * state.
    2404             :      */
    2405             : 
    2406      737244 :     CallXactCallbacks(is_parallel_worker ? XACT_EVENT_PARALLEL_COMMIT
    2407             :                       : XACT_EVENT_COMMIT);
    2408             : 
    2409      737244 :     CurrentResourceOwner = NULL;
    2410      737244 :     ResourceOwnerRelease(TopTransactionResourceOwner,
    2411             :                          RESOURCE_RELEASE_BEFORE_LOCKS,
    2412             :                          true, true);
    2413             : 
    2414             :     /* Check we've released all buffer pins */
    2415      737244 :     AtEOXact_Buffers(true);
    2416             : 
    2417             :     /* Clean up the relation cache */
    2418      737244 :     AtEOXact_RelationCache(true);
    2419             : 
    2420             :     /* Clean up the type cache */
    2421      737244 :     AtEOXact_TypeCache();
    2422             : 
    2423             :     /*
    2424             :      * Make catalog changes visible to all backends.  This has to happen after
    2425             :      * relcache references are dropped (see comments for
    2426             :      * AtEOXact_RelationCache), but before locks are released (if anyone is
    2427             :      * waiting for lock on a relation we've modified, we want them to know
    2428             :      * about the catalog change before they start using the relation).
    2429             :      */
    2430      737244 :     AtEOXact_Inval(true);
    2431             : 
    2432      737244 :     AtEOXact_MultiXact();
    2433             : 
    2434      737244 :     ResourceOwnerRelease(TopTransactionResourceOwner,
    2435             :                          RESOURCE_RELEASE_LOCKS,
    2436             :                          true, true);
    2437      737244 :     ResourceOwnerRelease(TopTransactionResourceOwner,
    2438             :                          RESOURCE_RELEASE_AFTER_LOCKS,
    2439             :                          true, true);
    2440             : 
    2441             :     /*
    2442             :      * Likewise, dropping of files deleted during the transaction is best done
    2443             :      * after releasing relcache and buffer pins.  (This is not strictly
    2444             :      * necessary during commit, since such pins should have been released
    2445             :      * already, but this ordering is definitely critical during abort.)  Since
    2446             :      * this may take many seconds, also delay until after releasing locks.
    2447             :      * Other backends will observe the attendant catalog changes and not
    2448             :      * attempt to access affected files.
    2449             :      */
    2450      737244 :     smgrDoPendingDeletes(true);
    2451             : 
    2452             :     /*
    2453             :      * Send out notification signals to other backends (and do other
    2454             :      * post-commit NOTIFY cleanup).  This must not happen until after our
    2455             :      * transaction is fully done from the viewpoint of other backends.
    2456             :      */
    2457      737244 :     AtCommit_Notify();
    2458             : 
    2459             :     /*
    2460             :      * Everything after this should be purely internal-to-this-backend
    2461             :      * cleanup.
    2462             :      */
    2463      737244 :     AtEOXact_GUC(true, 1);
    2464      737244 :     AtEOXact_SPI(true);
    2465      737244 :     AtEOXact_Enum();
    2466      737244 :     AtEOXact_on_commit_actions(true);
    2467      737244 :     AtEOXact_Namespace(true, is_parallel_worker);
    2468      737244 :     AtEOXact_SMgr();
    2469      737244 :     AtEOXact_Files(true);
    2470      737244 :     AtEOXact_ComboCid();
    2471      737244 :     AtEOXact_HashTables(true);
    2472      737244 :     AtEOXact_PgStat(true, is_parallel_worker);
    2473      737244 :     AtEOXact_Snapshot(true, false);
    2474      737244 :     AtEOXact_ApplyLauncher(true);
    2475      737244 :     AtEOXact_LogicalRepWorkers(true);
    2476      737244 :     pgstat_report_xact_timestamp(0);
    2477             : 
    2478      737244 :     ResourceOwnerDelete(TopTransactionResourceOwner);
    2479      737244 :     s->curTransactionOwner = NULL;
    2480      737244 :     CurTransactionResourceOwner = NULL;
    2481      737244 :     TopTransactionResourceOwner = NULL;
    2482             : 
    2483      737244 :     AtCommit_Memory();
    2484             : 
    2485      737244 :     s->fullTransactionId = InvalidFullTransactionId;
    2486      737244 :     s->subTransactionId = InvalidSubTransactionId;
    2487      737244 :     s->nestingLevel = 0;
    2488      737244 :     s->gucNestLevel = 0;
    2489      737244 :     s->childXids = NULL;
    2490      737244 :     s->nChildXids = 0;
    2491      737244 :     s->maxChildXids = 0;
    2492             : 
    2493      737244 :     XactTopFullTransactionId = InvalidFullTransactionId;
    2494      737244 :     nParallelCurrentXids = 0;
    2495             : 
    2496             :     /*
    2497             :      * done with commit processing, set current transaction state back to
    2498             :      * default
    2499             :      */
    2500      737244 :     s->state = TRANS_DEFAULT;
    2501             : 
    2502      737244 :     RESUME_INTERRUPTS();
    2503      737244 : }
    2504             : 
    2505             : 
    2506             : /*
    2507             :  *  PrepareTransaction
    2508             :  *
    2509             :  * NB: if you change this routine, better look at CommitTransaction too!
    2510             :  */
    2511             : static void
    2512         854 : PrepareTransaction(void)
    2513             : {
    2514         854 :     TransactionState s = CurrentTransactionState;
    2515         854 :     TransactionId xid = GetCurrentTransactionId();
    2516             :     GlobalTransaction gxact;
    2517             :     TimestampTz prepared_at;
    2518             : 
    2519             :     Assert(!IsInParallelMode());
    2520             : 
    2521         854 :     ShowTransactionState("PrepareTransaction");
    2522             : 
    2523             :     /*
    2524             :      * check the current transaction state
    2525             :      */
    2526         854 :     if (s->state != TRANS_INPROGRESS)
    2527           0 :         elog(WARNING, "PrepareTransaction while in %s state",
    2528             :              TransStateAsString(s->state));
    2529             :     Assert(s->parent == NULL);
    2530             : 
    2531             :     /*
    2532             :      * Do pre-commit processing that involves calling user-defined code, such
    2533             :      * as triggers.  Since closing cursors could queue trigger actions,
    2534             :      * triggers could open cursors, etc, we have to keep looping until there's
    2535             :      * nothing left to do.
    2536             :      */
    2537             :     for (;;)
    2538             :     {
    2539             :         /*
    2540             :          * Fire all currently pending deferred triggers.
    2541             :          */
    2542         860 :         AfterTriggerFireDeferred();
    2543             : 
    2544             :         /*
    2545             :          * Close open portals (converting holdable ones into static portals).
    2546             :          * If there weren't any, we are done ... otherwise loop back to check
    2547             :          * if they queued deferred triggers.  Lather, rinse, repeat.
    2548             :          */
    2549         860 :         if (!PreCommit_Portals(true))
    2550         854 :             break;
    2551             :     }
    2552             : 
    2553         854 :     CallXactCallbacks(XACT_EVENT_PRE_PREPARE);
    2554             : 
    2555             :     /*
    2556             :      * The remaining actions cannot call any user-defined code, so it's safe
    2557             :      * to start shutting down within-transaction services.  But note that most
    2558             :      * of this stuff could still throw an error, which would switch us into
    2559             :      * the transaction-abort path.
    2560             :      */
    2561             : 
    2562             :     /* Shut down the deferred-trigger manager */
    2563         852 :     AfterTriggerEndXact(true);
    2564             : 
    2565             :     /*
    2566             :      * Let ON COMMIT management do its thing (must happen after closing
    2567             :      * cursors, to avoid dangling-reference problems)
    2568             :      */
    2569         852 :     PreCommit_on_commit_actions();
    2570             : 
    2571             :     /*
    2572             :      * Synchronize files that are created and not WAL-logged during this
    2573             :      * transaction. This must happen before EndPrepare(), so that we don't see
    2574             :      * committed-but-broken files after a crash and COMMIT PREPARED.
    2575             :      */
    2576         852 :     smgrDoPendingSyncs(true, false);
    2577             : 
    2578             :     /* close large objects before lower-level cleanup */
    2579         852 :     AtEOXact_LargeObject(true);
    2580             : 
    2581             :     /* NOTIFY requires no work at this point */
    2582             : 
    2583             :     /*
    2584             :      * Mark serializable transaction as complete for predicate locking
    2585             :      * purposes.  This should be done as late as we can put it and still allow
    2586             :      * errors to be raised for failure patterns found at commit.
    2587             :      */
    2588         852 :     PreCommit_CheckForSerializationFailure();
    2589             : 
    2590             :     /*
    2591             :      * Don't allow PREPARE TRANSACTION if we've accessed a temporary table in
    2592             :      * this transaction.  Having the prepared xact hold locks on another
    2593             :      * backend's temp table seems a bad idea --- for instance it would prevent
    2594             :      * the backend from exiting.  There are other problems too, such as how to
    2595             :      * clean up the source backend's local buffers and ON COMMIT state if the
    2596             :      * prepared xact includes a DROP of a temp table.
    2597             :      *
    2598             :      * Other objects types, like functions, operators or extensions, share the
    2599             :      * same restriction as they should not be created, locked or dropped as
    2600             :      * this can mess up with this session or even a follow-up session trying
    2601             :      * to use the same temporary namespace.
    2602             :      *
    2603             :      * We must check this after executing any ON COMMIT actions, because they
    2604             :      * might still access a temp relation.
    2605             :      *
    2606             :      * XXX In principle this could be relaxed to allow some useful special
    2607             :      * cases, such as a temp table created and dropped all within the
    2608             :      * transaction.  That seems to require much more bookkeeping though.
    2609             :      */
    2610         852 :     if ((MyXactFlags & XACT_FLAGS_ACCESSEDTEMPNAMESPACE))
    2611          68 :         ereport(ERROR,
    2612             :                 (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
    2613             :                  errmsg("cannot PREPARE a transaction that has operated on temporary objects")));
    2614             : 
    2615             :     /*
    2616             :      * Likewise, don't allow PREPARE after pg_export_snapshot.  This could be
    2617             :      * supported if we added cleanup logic to twophase.c, but for now it
    2618             :      * doesn't seem worth the trouble.
    2619             :      */
    2620         784 :     if (XactHasExportedSnapshots())
    2621           0 :         ereport(ERROR,
    2622             :                 (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
    2623             :                  errmsg("cannot PREPARE a transaction that has exported snapshots")));
    2624             : 
    2625             :     /* Prevent cancel/die interrupt while cleaning up */
    2626         784 :     HOLD_INTERRUPTS();
    2627             : 
    2628             :     /*
    2629             :      * set the current transaction state information appropriately during
    2630             :      * prepare processing
    2631             :      */
    2632         784 :     s->state = TRANS_PREPARE;
    2633             : 
    2634             :     /* Disable transaction timeout */
    2635         784 :     if (TransactionTimeout > 0)
    2636           0 :         disable_timeout(TRANSACTION_TIMEOUT, false);
    2637             : 
    2638         784 :     prepared_at = GetCurrentTimestamp();
    2639             : 
    2640             :     /*
    2641             :      * Reserve the GID for this transaction. This could fail if the requested
    2642             :      * GID is invalid or already in use.
    2643             :      */
    2644         784 :     gxact = MarkAsPreparing(xid, prepareGID, prepared_at,
    2645             :                             GetUserId(), MyDatabaseId);
    2646         760 :     prepareGID = NULL;
    2647             : 
    2648             :     /*
    2649             :      * Collect data for the 2PC state file.  Note that in general, no actual
    2650             :      * state change should happen in the called modules during this step,
    2651             :      * since it's still possible to fail before commit, and in that case we
    2652             :      * want transaction abort to be able to clean up.  (In particular, the
    2653             :      * AtPrepare routines may error out if they find cases they cannot
    2654             :      * handle.)  State cleanup should happen in the PostPrepare routines
    2655             :      * below.  However, some modules can go ahead and clear state here because
    2656             :      * they wouldn't do anything with it during abort anyway.
    2657             :      *
    2658             :      * Note: because the 2PC state file records will be replayed in the same
    2659             :      * order they are made, the order of these calls has to match the order in
    2660             :      * which we want things to happen during COMMIT PREPARED or ROLLBACK
    2661             :      * PREPARED; in particular, pay attention to whether things should happen
    2662             :      * before or after releasing the transaction's locks.
    2663             :      */
    2664         760 :     StartPrepare(gxact);
    2665             : 
    2666         760 :     AtPrepare_Notify();
    2667         760 :     AtPrepare_Locks();
    2668         756 :     AtPrepare_PredicateLocks();
    2669         756 :     AtPrepare_PgStat();
    2670         756 :     AtPrepare_MultiXact();
    2671         756 :     AtPrepare_RelationMap();
    2672             : 
    2673             :     /*
    2674             :      * Here is where we really truly prepare.
    2675             :      *
    2676             :      * We have to record transaction prepares even if we didn't make any
    2677             :      * updates, because the transaction manager might get confused if we lose
    2678             :      * a global transaction.
    2679             :      */
    2680         756 :     EndPrepare(gxact);
    2681             : 
    2682             :     /*
    2683             :      * Now we clean up backend-internal state and release internal resources.
    2684             :      */
    2685             : 
    2686             :     /* Reset XactLastRecEnd until the next transaction writes something */
    2687         756 :     XactLastRecEnd = 0;
    2688             : 
    2689             :     /*
    2690             :      * Transfer our locks to a dummy PGPROC.  This has to be done before
    2691             :      * ProcArrayClearTransaction().  Otherwise, a GetLockConflicts() would
    2692             :      * conclude "xact already committed or aborted" for our locks.
    2693             :      */
    2694         756 :     PostPrepare_Locks(xid);
    2695             : 
    2696             :     /*
    2697             :      * Let others know about no transaction in progress by me.  This has to be
    2698             :      * done *after* the prepared transaction has been marked valid, else
    2699             :      * someone may think it is unlocked and recyclable.
    2700             :      */
    2701         756 :     ProcArrayClearTransaction(MyProc);
    2702             : 
    2703             :     /*
    2704             :      * In normal commit-processing, this is all non-critical post-transaction
    2705             :      * cleanup.  When the transaction is prepared, however, it's important
    2706             :      * that the locks and other per-backend resources are transferred to the
    2707             :      * prepared transaction's PGPROC entry.  Note that if an error is raised
    2708             :      * here, it's too late to abort the transaction. XXX: This probably should
    2709             :      * be in a critical section, to force a PANIC if any of this fails, but
    2710             :      * that cure could be worse than the disease.
    2711             :      */
    2712             : 
    2713         756 :     CallXactCallbacks(XACT_EVENT_PREPARE);
    2714             : 
    2715         756 :     ResourceOwnerRelease(TopTransactionResourceOwner,
    2716             :                          RESOURCE_RELEASE_BEFORE_LOCKS,
    2717             :                          true, true);
    2718             : 
    2719             :     /* Check we've released all buffer pins */
    2720         756 :     AtEOXact_Buffers(true);
    2721             : 
    2722             :     /* Clean up the relation cache */
    2723         756 :     AtEOXact_RelationCache(true);
    2724             : 
    2725             :     /* Clean up the type cache */
    2726         756 :     AtEOXact_TypeCache();
    2727             : 
    2728             :     /* notify doesn't need a postprepare call */
    2729             : 
    2730         756 :     PostPrepare_PgStat();
    2731             : 
    2732         756 :     PostPrepare_Inval();
    2733             : 
    2734         756 :     PostPrepare_smgr();
    2735             : 
    2736         756 :     PostPrepare_MultiXact(xid);
    2737             : 
    2738         756 :     PostPrepare_PredicateLocks(xid);
    2739             : 
    2740         756 :     ResourceOwnerRelease(TopTransactionResourceOwner,
    2741             :                          RESOURCE_RELEASE_LOCKS,
    2742             :                          true, true);
    2743         756 :     ResourceOwnerRelease(TopTransactionResourceOwner,
    2744             :                          RESOURCE_RELEASE_AFTER_LOCKS,
    2745             :                          true, true);
    2746             : 
    2747             :     /*
    2748             :      * Allow another backend to finish the transaction.  After
    2749             :      * PostPrepare_Twophase(), the transaction is completely detached from our
    2750             :      * backend.  The rest is just non-critical cleanup of backend-local state.
    2751             :      */
    2752         756 :     PostPrepare_Twophase();
    2753             : 
    2754             :     /* PREPARE acts the same as COMMIT as far as GUC is concerned */
    2755         756 :     AtEOXact_GUC(true, 1);
    2756         756 :     AtEOXact_SPI(true);
    2757         756 :     AtEOXact_Enum();
    2758         756 :     AtEOXact_on_commit_actions(true);
    2759         756 :     AtEOXact_Namespace(true, false);
    2760         756 :     AtEOXact_SMgr();
    2761         756 :     AtEOXact_Files(true);
    2762         756 :     AtEOXact_ComboCid();
    2763         756 :     AtEOXact_HashTables(true);
    2764             :     /* don't call AtEOXact_PgStat here; we fixed pgstat state above */
    2765         756 :     AtEOXact_Snapshot(true, true);
    2766             :     /* we treat PREPARE as ROLLBACK so far as waking workers goes */
    2767         756 :     AtEOXact_ApplyLauncher(false);
    2768         756 :     AtEOXact_LogicalRepWorkers(false);
    2769         756 :     pgstat_report_xact_timestamp(0);
    2770             : 
    2771         756 :     CurrentResourceOwner = NULL;
    2772         756 :     ResourceOwnerDelete(TopTransactionResourceOwner);
    2773         756 :     s->curTransactionOwner = NULL;
    2774         756 :     CurTransactionResourceOwner = NULL;
    2775         756 :     TopTransactionResourceOwner = NULL;
    2776             : 
    2777         756 :     AtCommit_Memory();
    2778             : 
    2779         756 :     s->fullTransactionId = InvalidFullTransactionId;
    2780         756 :     s->subTransactionId = InvalidSubTransactionId;
    2781         756 :     s->nestingLevel = 0;
    2782         756 :     s->gucNestLevel = 0;
    2783         756 :     s->childXids = NULL;
    2784         756 :     s->nChildXids = 0;
    2785         756 :     s->maxChildXids = 0;
    2786             : 
    2787         756 :     XactTopFullTransactionId = InvalidFullTransactionId;
    2788         756 :     nParallelCurrentXids = 0;
    2789             : 
    2790             :     /*
    2791             :      * done with 1st phase commit processing, set current transaction state
    2792             :      * back to default
    2793             :      */
    2794         756 :     s->state = TRANS_DEFAULT;
    2795             : 
    2796         756 :     RESUME_INTERRUPTS();
    2797         756 : }
    2798             : 
    2799             : 
    2800             : /*
    2801             :  *  AbortTransaction
    2802             :  */
    2803             : static void
    2804       47102 : AbortTransaction(void)
    2805             : {
    2806       47102 :     TransactionState s = CurrentTransactionState;
    2807             :     TransactionId latestXid;
    2808             :     bool        is_parallel_worker;
    2809             : 
    2810             :     /* Prevent cancel/die interrupt while cleaning up */
    2811       47102 :     HOLD_INTERRUPTS();
    2812             : 
    2813             :     /* Disable transaction timeout */
    2814       47102 :     if (TransactionTimeout > 0)
    2815           2 :         disable_timeout(TRANSACTION_TIMEOUT, false);
    2816             : 
    2817             :     /* Make sure we have a valid memory context and resource owner */
    2818       47102 :     AtAbort_Memory();
    2819       47102 :     AtAbort_ResourceOwner();
    2820             : 
    2821             :     /*
    2822             :      * Release any LW locks we might be holding as quickly as possible.
    2823             :      * (Regular locks, however, must be held till we finish aborting.)
    2824             :      * Releasing LW locks is critical since we might try to grab them again
    2825             :      * while cleaning up!
    2826             :      */
    2827       47102 :     LWLockReleaseAll();
    2828             : 
    2829             :     /* Clear wait information and command progress indicator */
    2830       47102 :     pgstat_report_wait_end();
    2831       47102 :     pgstat_progress_end_command();
    2832             : 
    2833             :     /* Clean up buffer content locks, too */
    2834       47102 :     UnlockBuffers();
    2835             : 
    2836             :     /* Reset WAL record construction state */
    2837       47102 :     XLogResetInsertion();
    2838             : 
    2839             :     /* Cancel condition variable sleep */
    2840       47102 :     ConditionVariableCancelSleep();
    2841             : 
    2842             :     /*
    2843             :      * Also clean up any open wait for lock, since the lock manager will choke
    2844             :      * if we try to wait for another lock before doing this.
    2845             :      */
    2846       47102 :     LockErrorCleanup();
    2847             : 
    2848             :     /*
    2849             :      * If any timeout events are still active, make sure the timeout interrupt
    2850             :      * is scheduled.  This covers possible loss of a timeout interrupt due to
    2851             :      * longjmp'ing out of the SIGINT handler (see notes in handle_sig_alarm).
    2852             :      * We delay this till after LockErrorCleanup so that we don't uselessly
    2853             :      * reschedule lock or deadlock check timeouts.
    2854             :      */
    2855       47102 :     reschedule_timeouts();
    2856             : 
    2857             :     /*
    2858             :      * Re-enable signals, in case we got here by longjmp'ing out of a signal
    2859             :      * handler.  We do this fairly early in the sequence so that the timeout
    2860             :      * infrastructure will be functional if needed while aborting.
    2861             :      */
    2862       47102 :     sigprocmask(SIG_SETMASK, &UnBlockSig, NULL);
    2863             : 
    2864             :     /*
    2865             :      * check the current transaction state
    2866             :      */
    2867       47102 :     is_parallel_worker = (s->blockState == TBLOCK_PARALLEL_INPROGRESS);
    2868       47102 :     if (s->state != TRANS_INPROGRESS && s->state != TRANS_PREPARE)
    2869           0 :         elog(WARNING, "AbortTransaction while in %s state",
    2870             :              TransStateAsString(s->state));
    2871             :     Assert(s->parent == NULL);
    2872             : 
    2873             :     /*
    2874             :      * set the current transaction state information appropriately during the
    2875             :      * abort processing
    2876             :      */
    2877       47102 :     s->state = TRANS_ABORT;
    2878             : 
    2879             :     /*
    2880             :      * Reset user ID which might have been changed transiently.  We need this
    2881             :      * to clean up in case control escaped out of a SECURITY DEFINER function
    2882             :      * or other local change of CurrentUserId; therefore, the prior value of
    2883             :      * SecurityRestrictionContext also needs to be restored.
    2884             :      *
    2885             :      * (Note: it is not necessary to restore session authorization or role
    2886             :      * settings here because those can only be changed via GUC, and GUC will
    2887             :      * take care of rolling them back if need be.)
    2888             :      */
    2889       47102 :     SetUserIdAndSecContext(s->prevUser, s->prevSecContext);
    2890             : 
    2891             :     /* Forget about any active REINDEX. */
    2892       47102 :     ResetReindexState(s->nestingLevel);
    2893             : 
    2894             :     /* Reset logical streaming state. */
    2895       47102 :     ResetLogicalStreamingState();
    2896             : 
    2897             :     /* Reset snapshot export state. */
    2898       47102 :     SnapBuildResetExportedSnapshotState();
    2899             : 
    2900             :     /*
    2901             :      * If this xact has started any unfinished parallel operation, clean up
    2902             :      * its workers and exit parallel mode.  Don't warn about leaked resources.
    2903             :      */
    2904       47102 :     AtEOXact_Parallel(false);
    2905       47102 :     s->parallelModeLevel = 0;
    2906       47102 :     s->parallelChildXact = false;    /* should be false already */
    2907             : 
    2908             :     /*
    2909             :      * do abort processing
    2910             :      */
    2911       47102 :     AfterTriggerEndXact(false); /* 'false' means it's abort */
    2912       47102 :     AtAbort_Portals();
    2913       47102 :     smgrDoPendingSyncs(false, is_parallel_worker);
    2914       47102 :     AtEOXact_LargeObject(false);
    2915       47102 :     AtAbort_Notify();
    2916       47102 :     AtEOXact_RelationMap(false, is_parallel_worker);
    2917       47102 :     AtAbort_Twophase();
    2918             : 
    2919             :     /*
    2920             :      * Advertise the fact that we aborted in pg_xact (assuming that we got as
    2921             :      * far as assigning an XID to advertise).  But if we're inside a parallel
    2922             :      * worker, skip this; the user backend must be the one to write the abort
    2923             :      * record.
    2924             :      */
    2925       47102 :     if (!is_parallel_worker)
    2926       47090 :         latestXid = RecordTransactionAbort(false);
    2927             :     else
    2928             :     {
    2929          12 :         latestXid = InvalidTransactionId;
    2930             : 
    2931             :         /*
    2932             :          * Since the parallel leader won't get our value of XactLastRecEnd in
    2933             :          * this case, we nudge WAL-writer ourselves in this case.  See related
    2934             :          * comments in RecordTransactionAbort for why this matters.
    2935             :          */
    2936          12 :         XLogSetAsyncXactLSN(XactLastRecEnd);
    2937             :     }
    2938             : 
    2939             :     TRACE_POSTGRESQL_TRANSACTION_ABORT(MyProc->vxid.lxid);
    2940             : 
    2941             :     /*
    2942             :      * Let others know about no transaction in progress by me. Note that this
    2943             :      * must be done _before_ releasing locks we hold and _after_
    2944             :      * RecordTransactionAbort.
    2945             :      */
    2946       47102 :     ProcArrayEndTransaction(MyProc, latestXid);
    2947             : 
    2948             :     /*
    2949             :      * Post-abort cleanup.  See notes in CommitTransaction() concerning
    2950             :      * ordering.  We can skip all of it if the transaction failed before
    2951             :      * creating a resource owner.
    2952             :      */
    2953       47102 :     if (TopTransactionResourceOwner != NULL)
    2954             :     {
    2955       47102 :         if (is_parallel_worker)
    2956          12 :             CallXactCallbacks(XACT_EVENT_PARALLEL_ABORT);
    2957             :         else
    2958       47090 :             CallXactCallbacks(XACT_EVENT_ABORT);
    2959             : 
    2960       47102 :         ResourceOwnerRelease(TopTransactionResourceOwner,
    2961             :                              RESOURCE_RELEASE_BEFORE_LOCKS,
    2962             :                              false, true);
    2963       47102 :         AtEOXact_Buffers(false);
    2964       47102 :         AtEOXact_RelationCache(false);
    2965       47102 :         AtEOXact_TypeCache();
    2966       47102 :         AtEOXact_Inval(false);
    2967       47102 :         AtEOXact_MultiXact();
    2968       47102 :         ResourceOwnerRelease(TopTransactionResourceOwner,
    2969             :                              RESOURCE_RELEASE_LOCKS,
    2970             :                              false, true);
    2971       47102 :         ResourceOwnerRelease(TopTransactionResourceOwner,
    2972             :                              RESOURCE_RELEASE_AFTER_LOCKS,
    2973             :                              false, true);
    2974       47102 :         smgrDoPendingDeletes(false);
    2975             : 
    2976       47102 :         AtEOXact_GUC(false, 1);
    2977       47102 :         AtEOXact_SPI(false);
    2978       47102 :         AtEOXact_Enum();
    2979       47102 :         AtEOXact_on_commit_actions(false);
    2980       47102 :         AtEOXact_Namespace(false, is_parallel_worker);
    2981       47102 :         AtEOXact_SMgr();
    2982       47102 :         AtEOXact_Files(false);
    2983       47102 :         AtEOXact_ComboCid();
    2984       47102 :         AtEOXact_HashTables(false);
    2985       47102 :         AtEOXact_PgStat(false, is_parallel_worker);
    2986       47102 :         AtEOXact_ApplyLauncher(false);
    2987       47102 :         AtEOXact_LogicalRepWorkers(false);
    2988       47102 :         pgstat_report_xact_timestamp(0);
    2989             :     }
    2990             : 
    2991             :     /*
    2992             :      * State remains TRANS_ABORT until CleanupTransaction().
    2993             :      */
    2994       47102 :     RESUME_INTERRUPTS();
    2995       47102 : }
    2996             : 
    2997             : /*
    2998             :  *  CleanupTransaction
    2999             :  */
    3000             : static void
    3001       47102 : CleanupTransaction(void)
    3002             : {
    3003       47102 :     TransactionState s = CurrentTransactionState;
    3004             : 
    3005             :     /*
    3006             :      * State should still be TRANS_ABORT from AbortTransaction().
    3007             :      */
    3008       47102 :     if (s->state != TRANS_ABORT)
    3009           0 :         elog(FATAL, "CleanupTransaction: unexpected state %s",
    3010             :              TransStateAsString(s->state));
    3011             : 
    3012             :     /*
    3013             :      * do abort cleanup processing
    3014             :      */
    3015       47102 :     AtCleanup_Portals();        /* now safe to release portal memory */
    3016       47102 :     AtEOXact_Snapshot(false, true); /* and release the transaction's snapshots */
    3017             : 
    3018       47102 :     CurrentResourceOwner = NULL;    /* and resource owner */
    3019       47102 :     if (TopTransactionResourceOwner)
    3020       47102 :         ResourceOwnerDelete(TopTransactionResourceOwner);
    3021       47102 :     s->curTransactionOwner = NULL;
    3022       47102 :     CurTransactionResourceOwner = NULL;
    3023       47102 :     TopTransactionResourceOwner = NULL;
    3024             : 
    3025       47102 :     AtCleanup_Memory();         /* and transaction memory */
    3026             : 
    3027       47102 :     s->fullTransactionId = InvalidFullTransactionId;
    3028       47102 :     s->subTransactionId = InvalidSubTransactionId;
    3029       47102 :     s->nestingLevel = 0;
    3030       47102 :     s->gucNestLevel = 0;
    3031       47102 :     s->childXids = NULL;
    3032       47102 :     s->nChildXids = 0;
    3033       47102 :     s->maxChildXids = 0;
    3034       47102 :     s->parallelModeLevel = 0;
    3035       47102 :     s->parallelChildXact = false;
    3036             : 
    3037       47102 :     XactTopFullTransactionId = InvalidFullTransactionId;
    3038       47102 :     nParallelCurrentXids = 0;
    3039             : 
    3040             :     /*
    3041             :      * done with abort processing, set current transaction state back to
    3042             :      * default
    3043             :      */
    3044       47102 :     s->state = TRANS_DEFAULT;
    3045       47102 : }
    3046             : 
    3047             : /*
    3048             :  *  StartTransactionCommand
    3049             :  */
    3050             : void
    3051      942928 : StartTransactionCommand(void)
    3052             : {
    3053      942928 :     TransactionState s = CurrentTransactionState;
    3054             : 
    3055      942928 :     switch (s->blockState)
    3056             :     {
    3057             :             /*
    3058             :              * if we aren't in a transaction block, we just do our usual start
    3059             :              * transaction.
    3060             :              */
    3061      782330 :         case TBLOCK_DEFAULT:
    3062      782330 :             StartTransaction();
    3063      782330 :             s->blockState = TBLOCK_STARTED;
    3064      782330 :             break;
    3065             : 
    3066             :             /*
    3067             :              * We are somewhere in a transaction block or subtransaction and
    3068             :              * about to start a new command.  For now we do nothing, but
    3069             :              * someday we may do command-local resource initialization. (Note
    3070             :              * that any needed CommandCounterIncrement was done by the
    3071             :              * previous CommitTransactionCommand.)
    3072             :              */
    3073      158854 :         case TBLOCK_INPROGRESS:
    3074             :         case TBLOCK_IMPLICIT_INPROGRESS:
    3075             :         case TBLOCK_SUBINPROGRESS:
    3076      158854 :             break;
    3077             : 
    3078             :             /*
    3079             :              * Here we are in a failed transaction block (one of the commands
    3080             :              * caused an abort) so we do nothing but remain in the abort
    3081             :              * state.  Eventually we will get a ROLLBACK command which will
    3082             :              * get us out of this state.  (It is up to other code to ensure
    3083             :              * that no commands other than ROLLBACK will be processed in these
    3084             :              * states.)
    3085             :              */
    3086        1744 :         case TBLOCK_ABORT:
    3087             :         case TBLOCK_SUBABORT:
    3088        1744 :             break;
    3089             : 
    3090             :             /* These cases are invalid. */
    3091           0 :         case TBLOCK_STARTED:
    3092             :         case TBLOCK_BEGIN:
    3093             :         case TBLOCK_PARALLEL_INPROGRESS:
    3094             :         case TBLOCK_SUBBEGIN:
    3095             :         case TBLOCK_END:
    3096             :         case TBLOCK_SUBRELEASE:
    3097             :         case TBLOCK_SUBCOMMIT:
    3098             :         case TBLOCK_ABORT_END:
    3099             :         case TBLOCK_SUBABORT_END:
    3100             :         case TBLOCK_ABORT_PENDING:
    3101             :         case TBLOCK_SUBABORT_PENDING:
    3102             :         case TBLOCK_SUBRESTART:
    3103             :         case TBLOCK_SUBABORT_RESTART:
    3104             :         case TBLOCK_PREPARE:
    3105           0 :             elog(ERROR, "StartTransactionCommand: unexpected state %s",
    3106             :                  BlockStateAsString(s->blockState));
    3107             :             break;
    3108             :     }
    3109             : 
    3110             :     /*
    3111             :      * We must switch to CurTransactionContext before returning. This is
    3112             :      * already done if we called StartTransaction, otherwise not.
    3113             :      */
    3114      942928 :     Assert(CurTransactionContext != NULL);
    3115      942928 :     MemoryContextSwitchTo(CurTransactionContext);
    3116      942928 : }
    3117             : 
    3118             : 
    3119             : /*
    3120             :  * Simple system for saving and restoring transaction characteristics
    3121             :  * (isolation level, read only, deferrable).  We need this for transaction
    3122             :  * chaining, so that we can set the characteristics of the new transaction to
    3123             :  * be the same as the previous one.  (We need something like this because the
    3124             :  * GUC system resets the characteristics at transaction end, so for example
    3125             :  * just skipping the reset in StartTransaction() won't work.)
    3126             :  */
    3127             : void
    3128      899196 : SaveTransactionCharacteristics(SavedTransactionCharacteristics *s)
    3129             : {
    3130      899196 :     s->save_XactIsoLevel = XactIsoLevel;
    3131      899196 :     s->save_XactReadOnly = XactReadOnly;
    3132      899196 :     s->save_XactDeferrable = XactDeferrable;
    3133      899196 : }
    3134             : 
    3135             : void
    3136          68 : RestoreTransactionCharacteristics(const SavedTransactionCharacteristics *s)
    3137             : {
    3138          68 :     XactIsoLevel = s->save_XactIsoLevel;
    3139          68 :     XactReadOnly = s->save_XactReadOnly;
    3140          68 :     XactDeferrable = s->save_XactDeferrable;
    3141          68 : }
    3142             : 
    3143             : /*
    3144             :  *  CommitTransactionCommand -- a wrapper function handling the
    3145             :  *      loop over subtransactions to avoid a potentially dangerous recursion
    3146             :  *      in CommitTransactionCommandInternal().
    3147             :  */
    3148             : void
    3149      898764 : CommitTransactionCommand(void)
    3150             : {
    3151             :     /*
    3152             :      * Repeatedly call CommitTransactionCommandInternal() until all the work
    3153             :      * is done.
    3154             :      */
    3155      899188 :     while (!CommitTransactionCommandInternal())
    3156             :     {
    3157             :     }
    3158      898202 : }
    3159             : 
    3160             : /*
    3161             :  *  CommitTransactionCommandInternal - a function doing an iteration of work
    3162             :  *      regarding handling the commit transaction command.  In the case of
    3163             :  *      subtransactions more than one iterations could be required.  Returns
    3164             :  *      true when no more iterations required, false otherwise.
    3165             :  */
    3166             : static bool
    3167      899188 : CommitTransactionCommandInternal(void)
    3168             : {
    3169      899188 :     TransactionState s = CurrentTransactionState;
    3170             :     SavedTransactionCharacteristics savetc;
    3171             : 
    3172             :     /* Must save in case we need to restore below */
    3173      899188 :     SaveTransactionCharacteristics(&savetc);
    3174             : 
    3175      899188 :     switch (s->blockState)
    3176             :     {
    3177             :             /*
    3178             :              * These shouldn't happen.  TBLOCK_DEFAULT means the previous
    3179             :              * StartTransactionCommand didn't set the STARTED state
    3180             :              * appropriately, while TBLOCK_PARALLEL_INPROGRESS should be ended
    3181             :              * by EndParallelWorkerTransaction(), not this function.
    3182             :              */
    3183           0 :         case TBLOCK_DEFAULT:
    3184             :         case TBLOCK_PARALLEL_INPROGRESS:
    3185           0 :             elog(FATAL, "CommitTransactionCommand: unexpected state %s",
    3186             :                  BlockStateAsString(s->blockState));
    3187             :             break;
    3188             : 
    3189             :             /*
    3190             :              * If we aren't in a transaction block, just do our usual
    3191             :              * transaction commit, and return to the idle state.
    3192             :              */
    3193      723718 :         case TBLOCK_STARTED:
    3194      723718 :             CommitTransaction();
    3195      723698 :             s->blockState = TBLOCK_DEFAULT;
    3196      723698 :             break;
    3197             : 
    3198             :             /*
    3199             :              * We are completing a "BEGIN TRANSACTION" command, so we change
    3200             :              * to the "transaction block in progress" state and return.  (We
    3201             :              * assume the BEGIN did nothing to the database, so we need no
    3202             :              * CommandCounterIncrement.)
    3203             :              */
    3204       16008 :         case TBLOCK_BEGIN:
    3205       16008 :             s->blockState = TBLOCK_INPROGRESS;
    3206       16008 :             break;
    3207             : 
    3208             :             /*
    3209             :              * This is the case when we have finished executing a command
    3210             :              * someplace within a transaction block.  We increment the command
    3211             :              * counter and return.
    3212             :              */
    3213      123086 :         case TBLOCK_INPROGRESS:
    3214             :         case TBLOCK_IMPLICIT_INPROGRESS:
    3215             :         case TBLOCK_SUBINPROGRESS:
    3216      123086 :             CommandCounterIncrement();
    3217      123086 :             break;
    3218             : 
    3219             :             /*
    3220             :              * We are completing a "COMMIT" command.  Do it and return to the
    3221             :              * idle state.
    3222             :              */
    3223       10550 :         case TBLOCK_END:
    3224       10550 :             CommitTransaction();
    3225       10132 :             s->blockState = TBLOCK_DEFAULT;
    3226       10132 :             if (s->chain)
    3227             :             {
    3228          12 :                 StartTransaction();
    3229          12 :                 s->blockState = TBLOCK_INPROGRESS;
    3230          12 :                 s->chain = false;
    3231          12 :                 RestoreTransactionCharacteristics(&savetc);
    3232             :             }
    3233       10132 :             break;
    3234             : 
    3235             :             /*
    3236             :              * Here we are in the middle of a transaction block but one of the
    3237             :              * commands caused an abort so we do nothing but remain in the
    3238             :              * abort state.  Eventually we will get a ROLLBACK command.
    3239             :              */
    3240          18 :         case TBLOCK_ABORT:
    3241             :         case TBLOCK_SUBABORT:
    3242          18 :             break;
    3243             : 
    3244             :             /*
    3245             :              * Here we were in an aborted transaction block and we just got
    3246             :              * the ROLLBACK command from the user, so clean up the
    3247             :              * already-aborted transaction and return to the idle state.
    3248             :              */
    3249        1360 :         case TBLOCK_ABORT_END:
    3250        1360 :             CleanupTransaction();
    3251        1360 :             s->blockState = TBLOCK_DEFAULT;
    3252        1360 :             if (s->chain)
    3253             :             {
    3254          12 :                 StartTransaction();
    3255          12 :                 s->blockState = TBLOCK_INPROGRESS;
    3256          12 :                 s->chain = false;
    3257          12 :                 RestoreTransactionCharacteristics(&savetc);
    3258             :             }
    3259        1360 :             break;
    3260             : 
    3261             :             /*
    3262             :              * Here we were in a perfectly good transaction block but the user
    3263             :              * told us to ROLLBACK anyway.  We have to abort the transaction
    3264             :              * and then clean up.
    3265             :              */
    3266        2114 :         case TBLOCK_ABORT_PENDING:
    3267        2114 :             AbortTransaction();
    3268        2114 :             CleanupTransaction();
    3269        2114 :             s->blockState = TBLOCK_DEFAULT;
    3270        2114 :             if (s->chain)
    3271             :             {
    3272          18 :                 StartTransaction();
    3273          18 :                 s->blockState = TBLOCK_INPROGRESS;
    3274          18 :                 s->chain = false;
    3275          18 :                 RestoreTransactionCharacteristics(&savetc);
    3276             :             }
    3277        2114 :             break;
    3278             : 
    3279             :             /*
    3280             :              * We are completing a "PREPARE TRANSACTION" command.  Do it and
    3281             :              * return to the idle state.
    3282             :              */
    3283         548 :         case TBLOCK_PREPARE:
    3284         548 :             PrepareTransaction();
    3285         452 :             s->blockState = TBLOCK_DEFAULT;
    3286         452 :             break;
    3287             : 
    3288             :             /*
    3289             :              * The user issued a SAVEPOINT inside a transaction block. Start a
    3290             :              * subtransaction.  (DefineSavepoint already did PushTransaction,
    3291             :              * so as to have someplace to put the SUBBEGIN state.)
    3292             :              */
    3293       19328 :         case TBLOCK_SUBBEGIN:
    3294       19328 :             StartSubTransaction();
    3295       19328 :             s->blockState = TBLOCK_SUBINPROGRESS;
    3296       19328 :             break;
    3297             : 
    3298             :             /*
    3299             :              * The user issued a RELEASE command, so we end the current
    3300             :              * subtransaction and return to the parent transaction. The parent
    3301             :              * might be ended too, so repeat till we find an INPROGRESS
    3302             :              * transaction or subtransaction.
    3303             :              */
    3304         172 :         case TBLOCK_SUBRELEASE:
    3305             :             do
    3306             :             {
    3307         448 :                 CommitSubTransaction();
    3308         448 :                 s = CurrentTransactionState;    /* changed by pop */
    3309         448 :             } while (s->blockState == TBLOCK_SUBRELEASE);
    3310             : 
    3311             :             Assert(s->blockState == TBLOCK_INPROGRESS ||
    3312             :                    s->blockState == TBLOCK_SUBINPROGRESS);
    3313         276 :             break;
    3314             : 
    3315             :             /*
    3316             :              * The user issued a COMMIT, so we end the current subtransaction
    3317             :              * hierarchy and perform final commit. We do this by rolling up
    3318             :              * any subtransactions into their parent, which leads to O(N^2)
    3319             :              * operations with respect to resource owners - this isn't that
    3320             :              * bad until we approach a thousands of savepoints but is
    3321             :              * necessary for correctness should after triggers create new
    3322             :              * resource owners.
    3323             :              */
    3324         134 :         case TBLOCK_SUBCOMMIT:
    3325             :             do
    3326             :             {
    3327        1180 :                 CommitSubTransaction();
    3328        1180 :                 s = CurrentTransactionState;    /* changed by pop */
    3329        1180 :             } while (s->blockState == TBLOCK_SUBCOMMIT);
    3330             :             /* If we had a COMMIT command, finish off the main xact too */
    3331        1046 :             if (s->blockState == TBLOCK_END)
    3332             :             {
    3333             :                 Assert(s->parent == NULL);
    3334         740 :                 CommitTransaction();
    3335         714 :                 s->blockState = TBLOCK_DEFAULT;
    3336         714 :                 if (s->chain)
    3337             :                 {
    3338          18 :                     StartTransaction();
    3339          18 :                     s->blockState = TBLOCK_INPROGRESS;
    3340          18 :                     s->chain = false;
    3341          18 :                     RestoreTransactionCharacteristics(&savetc);
    3342             :                 }
    3343             :             }
    3344         306 :             else if (s->blockState == TBLOCK_PREPARE)
    3345             :             {
    3346             :                 Assert(s->parent == NULL);
    3347         306 :                 PrepareTransaction();
    3348         304 :                 s->blockState = TBLOCK_DEFAULT;
    3349             :             }
    3350             :             else
    3351           0 :                 elog(ERROR, "CommitTransactionCommand: unexpected state %s",
    3352             :                      BlockStateAsString(s->blockState));
    3353        1018 :             break;
    3354             : 
    3355             :             /*
    3356             :              * The current already-failed subtransaction is ending due to a
    3357             :              * ROLLBACK or ROLLBACK TO command, so pop it and recursively
    3358             :              * examine the parent (which could be in any of several states).
    3359             :              * As we need to examine the parent, return false to request the
    3360             :              * caller to do the next iteration.
    3361             :              */
    3362          80 :         case TBLOCK_SUBABORT_END:
    3363          80 :             CleanupSubTransaction();
    3364          80 :             return false;
    3365             : 
    3366             :             /*
    3367             :              * As above, but it's not dead yet, so abort first.
    3368             :              */
    3369         344 :         case TBLOCK_SUBABORT_PENDING:
    3370         344 :             AbortSubTransaction();
    3371         344 :             CleanupSubTransaction();
    3372         344 :             return false;
    3373             : 
    3374             :             /*
    3375             :              * The current subtransaction is the target of a ROLLBACK TO
    3376             :              * command.  Abort and pop it, then start a new subtransaction
    3377             :              * with the same name.
    3378             :              */
    3379         520 :         case TBLOCK_SUBRESTART:
    3380             :             {
    3381             :                 char       *name;
    3382             :                 int         savepointLevel;
    3383             : 
    3384             :                 /* save name and keep Cleanup from freeing it */
    3385         520 :                 name = s->name;
    3386         520 :                 s->name = NULL;
    3387         520 :                 savepointLevel = s->savepointLevel;
    3388             : 
    3389         520 :                 AbortSubTransaction();
    3390         520 :                 CleanupSubTransaction();
    3391             : 
    3392         520 :                 DefineSavepoint(NULL);
    3393         520 :                 s = CurrentTransactionState;    /* changed by push */
    3394         520 :                 s->name = name;
    3395         520 :                 s->savepointLevel = savepointLevel;
    3396             : 
    3397             :                 /* This is the same as TBLOCK_SUBBEGIN case */
    3398             :                 Assert(s->blockState == TBLOCK_SUBBEGIN);
    3399         520 :                 StartSubTransaction();
    3400         520 :                 s->blockState = TBLOCK_SUBINPROGRESS;
    3401             :             }
    3402         520 :             break;
    3403             : 
    3404             :             /*
    3405             :              * Same as above, but the subtransaction had already failed, so we
    3406             :              * don't need AbortSubTransaction.
    3407             :              */
    3408         192 :         case TBLOCK_SUBABORT_RESTART:
    3409             :             {
    3410             :                 char       *name;
    3411             :                 int         savepointLevel;
    3412             : 
    3413             :                 /* save name and keep Cleanup from freeing it */
    3414         192 :                 name = s->name;
    3415         192 :                 s->name = NULL;
    3416         192 :                 savepointLevel = s->savepointLevel;
    3417             : 
    3418         192 :                 CleanupSubTransaction();
    3419             : 
    3420         192 :                 DefineSavepoint(NULL);
    3421         192 :                 s = CurrentTransactionState;    /* changed by push */
    3422         192 :                 s->name = name;
    3423         192 :                 s->savepointLevel = savepointLevel;
    3424             : 
    3425             :                 /* This is the same as TBLOCK_SUBBEGIN case */
    3426             :                 Assert(s->blockState == TBLOCK_SUBBEGIN);
    3427         192 :                 StartSubTransaction();
    3428         192 :                 s->blockState = TBLOCK_SUBINPROGRESS;
    3429             :             }
    3430         192 :             break;
    3431             :     }
    3432             : 
    3433             :     /* Done, no more iterations required */
    3434      898202 :     return true;
    3435             : }
    3436             : 
    3437             : /*
    3438             :  *  AbortCurrentTransaction -- a wrapper function handling the
    3439             :  *      loop over subtransactions to avoid potentially dangerous recursion in
    3440             :  *      AbortCurrentTransactionInternal().
    3441             :  */
    3442             : void
    3443       46408 : AbortCurrentTransaction(void)
    3444             : {
    3445             :     /*
    3446             :      * Repeatedly call AbortCurrentTransactionInternal() until all the work is
    3447             :      * done.
    3448             :      */
    3449       46408 :     while (!AbortCurrentTransactionInternal())
    3450             :     {
    3451             :     }
    3452       46408 : }
    3453             : 
    3454             : /*
    3455             :  *  AbortCurrentTransactionInternal - a function doing an iteration of work
    3456             :  *      regarding handling the current transaction abort.  In the case of
    3457             :  *      subtransactions more than one iterations could be required.  Returns
    3458             :  *      true when no more iterations required, false otherwise.
    3459             :  */
    3460             : static bool
    3461       46408 : AbortCurrentTransactionInternal(void)
    3462             : {
    3463       46408 :     TransactionState s = CurrentTransactionState;
    3464             : 
    3465       46408 :     switch (s->blockState)
    3466             :     {
    3467         102 :         case TBLOCK_DEFAULT:
    3468         102 :             if (s->state == TRANS_DEFAULT)
    3469             :             {
    3470             :                 /* we are idle, so nothing to do */
    3471             :             }
    3472             :             else
    3473             :             {
    3474             :                 /*
    3475             :                  * We can get here after an error during transaction start
    3476             :                  * (state will be TRANS_START).  Need to clean up the
    3477             :                  * incompletely started transaction.  First, adjust the
    3478             :                  * low-level state to suppress warning message from
    3479             :                  * AbortTransaction.
    3480             :                  */
    3481           0 :                 if (s->state == TRANS_START)
    3482           0 :                     s->state = TRANS_INPROGRESS;
    3483           0 :                 AbortTransaction();
    3484           0 :                 CleanupTransaction();
    3485             :             }
    3486         102 :             break;
    3487             : 
    3488             :             /*
    3489             :              * If we aren't in a transaction block, we just do the basic abort
    3490             :              * & cleanup transaction.  For this purpose, we treat an implicit
    3491             :              * transaction block as if it were a simple statement.
    3492             :              */
    3493       42202 :         case TBLOCK_STARTED:
    3494             :         case TBLOCK_IMPLICIT_INPROGRESS:
    3495       42202 :             AbortTransaction();
    3496       42202 :             CleanupTransaction();
    3497       42202 :             s->blockState = TBLOCK_DEFAULT;
    3498       42202 :             break;
    3499             : 
    3500             :             /*
    3501             :              * If we are in TBLOCK_BEGIN it means something screwed up right
    3502             :              * after reading "BEGIN TRANSACTION".  We assume that the user
    3503             :              * will interpret the error as meaning the BEGIN failed to get him
    3504             :              * into a transaction block, so we should abort and return to idle
    3505             :              * state.
    3506             :              */
    3507           0 :         case TBLOCK_BEGIN:
    3508           0 :             AbortTransaction();
    3509           0 :             CleanupTransaction();
    3510           0 :             s->blockState = TBLOCK_DEFAULT;
    3511           0 :             break;
    3512             : 
    3513             :             /*
    3514             :              * We are somewhere in a transaction block and we've gotten a
    3515             :              * failure, so we abort the transaction and set up the persistent
    3516             :              * ABORT state.  We will stay in ABORT until we get a ROLLBACK.
    3517             :              */
    3518        1382 :         case TBLOCK_INPROGRESS:
    3519             :         case TBLOCK_PARALLEL_INPROGRESS:
    3520        1382 :             AbortTransaction();
    3521        1382 :             s->blockState = TBLOCK_ABORT;
    3522             :             /* CleanupTransaction happens when we exit TBLOCK_ABORT_END */
    3523        1382 :             break;
    3524             : 
    3525             :             /*
    3526             :              * Here, we failed while trying to COMMIT.  Clean up the
    3527             :              * transaction and return to idle state (we do not want to stay in
    3528             :              * the transaction).
    3529             :              */
    3530         444 :         case TBLOCK_END:
    3531         444 :             AbortTransaction();
    3532         444 :             CleanupTransaction();
    3533         444 :             s->blockState = TBLOCK_DEFAULT;
    3534         444 :             break;
    3535             : 
    3536             :             /*
    3537             :              * Here, we are already in an aborted transaction state and are
    3538             :              * waiting for a ROLLBACK, but for some reason we failed again! So
    3539             :              * we just remain in the abort state.
    3540             :              */
    3541          96 :         case TBLOCK_ABORT:
    3542             :         case TBLOCK_SUBABORT:
    3543          96 :             break;
    3544             : 
    3545             :             /*
    3546             :              * We are in a failed transaction and we got the ROLLBACK command.
    3547             :              * We have already aborted, we just need to cleanup and go to idle
    3548             :              * state.
    3549             :              */
    3550           0 :         case TBLOCK_ABORT_END:
    3551           0 :             CleanupTransaction();
    3552           0 :             s->blockState = TBLOCK_DEFAULT;
    3553           0 :             break;
    3554             : 
    3555             :             /*
    3556             :              * We are in a live transaction and we got a ROLLBACK command.
    3557             :              * Abort, cleanup, go to idle state.
    3558             :              */
    3559           0 :         case TBLOCK_ABORT_PENDING:
    3560           0 :             AbortTransaction();
    3561           0 :             CleanupTransaction();
    3562           0 :             s->blockState = TBLOCK_DEFAULT;
    3563           0 :             break;
    3564             : 
    3565             :             /*
    3566             :              * Here, we failed while trying to PREPARE.  Clean up the
    3567             :              * transaction and return to idle state (we do not want to stay in
    3568             :              * the transaction).
    3569             :              */
    3570          96 :         case TBLOCK_PREPARE:
    3571          96 :             AbortTransaction();
    3572          96 :             CleanupTransaction();
    3573          96 :             s->blockState = TBLOCK_DEFAULT;
    3574          96 :             break;
    3575             : 
    3576             :             /*
    3577             :              * We got an error inside a subtransaction.  Abort just the
    3578             :              * subtransaction, and go to the persistent SUBABORT state until
    3579             :              * we get ROLLBACK.
    3580             :              */
    3581        2086 :         case TBLOCK_SUBINPROGRESS:
    3582        2086 :             AbortSubTransaction();
    3583        2086 :             s->blockState = TBLOCK_SUBABORT;
    3584        2086 :             break;
    3585             : 
    3586             :             /*
    3587             :              * If we failed while trying to create a subtransaction, clean up
    3588             :              * the broken subtransaction and abort the parent.  The same
    3589             :              * applies if we get a failure while ending a subtransaction.  As
    3590             :              * we need to abort the parent, return false to request the caller
    3591             :              * to do the next iteration.
    3592             :              */
    3593           0 :         case TBLOCK_SUBBEGIN:
    3594             :         case TBLOCK_SUBRELEASE:
    3595             :         case TBLOCK_SUBCOMMIT:
    3596             :         case TBLOCK_SUBABORT_PENDING:
    3597             :         case TBLOCK_SUBRESTART:
    3598           0 :             AbortSubTransaction();
    3599           0 :             CleanupSubTransaction();
    3600           0 :             return false;
    3601             : 
    3602             :             /*
    3603             :              * Same as above, except the Abort() was already done.
    3604             :              */
    3605           0 :         case TBLOCK_SUBABORT_END:
    3606             :         case TBLOCK_SUBABORT_RESTART:
    3607           0 :             CleanupSubTransaction();
    3608           0 :             return false;
    3609             :     }
    3610             : 
    3611             :     /* Done, no more iterations required */
    3612       46408 :     return true;
    3613             : }
    3614             : 
    3615             : /*
    3616             :  *  PreventInTransactionBlock
    3617             :  *
    3618             :  *  This routine is to be called by statements that must not run inside
    3619             :  *  a transaction block, typically because they have non-rollback-able
    3620             :  *  side effects or do internal commits.
    3621             :  *
    3622             :  *  If this routine completes successfully, then the calling statement is
    3623             :  *  guaranteed that if it completes without error, its results will be
    3624             :  *  committed immediately.
    3625             :  *
    3626             :  *  If we have already started a transaction block, issue an error; also issue
    3627             :  *  an error if we appear to be running inside a user-defined function (which
    3628             :  *  could issue more commands and possibly cause a failure after the statement
    3629             :  *  completes).  Subtransactions are verboten too.
    3630             :  *
    3631             :  *  We must also set XACT_FLAGS_NEEDIMMEDIATECOMMIT in MyXactFlags, to ensure
    3632             :  *  that postgres.c follows through by committing after the statement is done.
    3633             :  *
    3634             :  *  isTopLevel: passed down from ProcessUtility to determine whether we are
    3635             :  *  inside a function.  (We will always fail if this is false, but it's
    3636             :  *  convenient to centralize the check here instead of making callers do it.)
    3637             :  *  stmtType: statement type name, for error messages.
    3638             :  */
    3639             : void
    3640      105180 : PreventInTransactionBlock(bool isTopLevel, const char *stmtType)
    3641             : {
    3642             :     /*
    3643             :      * xact block already started?
    3644             :      */
    3645      105180 :     if (IsTransactionBlock())
    3646         106 :         ereport(ERROR,
    3647             :                 (errcode(ERRCODE_ACTIVE_SQL_TRANSACTION),
    3648             :         /* translator: %s represents an SQL statement name */
    3649             :                  errmsg("%s cannot run inside a transaction block",
    3650             :                         stmtType)));
    3651             : 
    3652             :     /*
    3653             :      * subtransaction?
    3654             :      */
    3655      105074 :     if (IsSubTransaction())
    3656           0 :         ereport(ERROR,
    3657             :                 (errcode(ERRCODE_ACTIVE_SQL_TRANSACTION),
    3658             :         /* translator: %s represents an SQL statement name */
    3659             :                  errmsg("%s cannot run inside a subtransaction",
    3660             :                         stmtType)));
    3661             : 
    3662             :     /*
    3663             :      * inside a function call?
    3664             :      */
    3665      105074 :     if (!isTopLevel)
    3666           6 :         ereport(ERROR,
    3667             :                 (errcode(ERRCODE_ACTIVE_SQL_TRANSACTION),
    3668             :         /* translator: %s represents an SQL statement name */
    3669             :                  errmsg("%s cannot be executed from a function", stmtType)));
    3670             : 
    3671             :     /* If we got past IsTransactionBlock test, should be in default state */
    3672      105068 :     if (CurrentTransactionState->blockState != TBLOCK_DEFAULT &&
    3673      103434 :         CurrentTransactionState->blockState != TBLOCK_STARTED)
    3674           0 :         elog(FATAL, "cannot prevent transaction chain");
    3675             : 
    3676             :     /* All okay.  Set the flag to make sure the right thing happens later. */
    3677      105068 :     MyXactFlags |= XACT_FLAGS_NEEDIMMEDIATECOMMIT;
    3678      105068 : }
    3679             : 
    3680             : /*
    3681             :  *  WarnNoTransactionBlock
    3682             :  *  RequireTransactionBlock
    3683             :  *
    3684             :  *  These two functions allow for warnings or errors if a command is executed
    3685             :  *  outside of a transaction block.  This is useful for commands that have no
    3686             :  *  effects that persist past transaction end (and so calling them outside a
    3687             :  *  transaction block is presumably an error).  DECLARE CURSOR is an example.
    3688             :  *  While top-level transaction control commands (BEGIN/COMMIT/ABORT) and SET
    3689             :  *  that have no effect issue warnings, all other no-effect commands generate
    3690             :  *  errors.
    3691             :  *
    3692             :  *  If we appear to be running inside a user-defined function, we do not
    3693             :  *  issue anything, since the function could issue more commands that make
    3694             :  *  use of the current statement's results.  Likewise subtransactions.
    3695             :  *  Thus these are inverses for PreventInTransactionBlock.
    3696             :  *
    3697             :  *  isTopLevel: passed down from ProcessUtility to determine whether we are
    3698             :  *  inside a function.
    3699             :  *  stmtType: statement type name, for warning or error messages.
    3700             :  */
    3701             : void
    3702        1926 : WarnNoTransactionBlock(bool isTopLevel, const char *stmtType)
    3703             : {
    3704        1926 :     CheckTransactionBlock(isTopLevel, false, stmtType);
    3705        1926 : }
    3706             : 
    3707             : void
    3708        6752 : RequireTransactionBlock(bool isTopLevel, const char *stmtType)
    3709             : {
    3710        6752 :     CheckTransactionBlock(isTopLevel, true, stmtType);
    3711        6728 : }
    3712             : 
    3713             : /*
    3714             :  * This is the implementation of the above two.
    3715             :  */
    3716             : static void
    3717        8678 : CheckTransactionBlock(bool isTopLevel, bool throwError, const char *stmtType)
    3718             : {
    3719             :     /*
    3720             :      * xact block already started?
    3721             :      */
    3722        8678 :     if (IsTransactionBlock())
    3723        8522 :         return;
    3724             : 
    3725             :     /*
    3726             :      * subtransaction?
    3727             :      */
    3728         156 :     if (IsSubTransaction())
    3729           0 :         return;
    3730             : 
    3731             :     /*
    3732             :      * inside a function call?
    3733             :      */
    3734         156 :     if (!isTopLevel)
    3735         116 :         return;
    3736             : 
    3737          40 :     ereport(throwError ? ERROR : WARNING,
    3738             :             (errcode(ERRCODE_NO_ACTIVE_SQL_TRANSACTION),
    3739             :     /* translator: %s represents an SQL statement name */
    3740             :              errmsg("%s can only be used in transaction blocks",
    3741             :                     stmtType)));
    3742             : }
    3743             : 
    3744             : /*
    3745             :  *  IsInTransactionBlock
    3746             :  *
    3747             :  *  This routine is for statements that need to behave differently inside
    3748             :  *  a transaction block than when running as single commands.  ANALYZE is
    3749             :  *  currently the only example.
    3750             :  *
    3751             :  *  If this routine returns "false", then the calling statement is allowed
    3752             :  *  to perform internal transaction-commit-and-start cycles; there is not a
    3753             :  *  risk of messing up any transaction already in progress.  (Note that this
    3754             :  *  is not the identical guarantee provided by PreventInTransactionBlock,
    3755             :  *  since we will not force a post-statement commit.)
    3756             :  *
    3757             :  *  isTopLevel: passed down from ProcessUtility to determine whether we are
    3758             :  *  inside a function.
    3759             :  */
    3760             : bool
    3761        4736 : IsInTransactionBlock(bool isTopLevel)
    3762             : {
    3763             :     /*
    3764             :      * Return true on same conditions that would make
    3765             :      * PreventInTransactionBlock error out
    3766             :      */
    3767        4736 :     if (IsTransactionBlock())
    3768         138 :         return true;
    3769             : 
    3770        4598 :     if (IsSubTransaction())
    3771           0 :         return true;
    3772             : 
    3773        4598 :     if (!isTopLevel)
    3774         112 :         return true;
    3775             : 
    3776        4486 :     if (CurrentTransactionState->blockState != TBLOCK_DEFAULT &&
    3777        4486 :         CurrentTransactionState->blockState != TBLOCK_STARTED)
    3778           0 :         return true;
    3779             : 
    3780        4486 :     return false;
    3781             : }
    3782             : 
    3783             : 
    3784             : /*
    3785             :  * Register or deregister callback functions for start- and end-of-xact
    3786             :  * operations.
    3787             :  *
    3788             :  * These functions are intended for use by dynamically loaded modules.
    3789             :  * For built-in modules we generally just hardwire the appropriate calls
    3790             :  * (mainly because it's easier to control the order that way, where needed).
    3791             :  *
    3792             :  * At transaction end, the callback occurs post-commit or post-abort, so the
    3793             :  * callback functions can only do noncritical cleanup.
    3794             :  */
    3795             : void
    3796        3510 : RegisterXactCallback(XactCallback callback, void *arg)
    3797             : {
    3798             :     XactCallbackItem *item;
    3799             : 
    3800             :     item = (XactCallbackItem *)
    3801        3510 :         MemoryContextAlloc(TopMemoryContext, sizeof(XactCallbackItem));
    3802        3510 :     item->callback = callback;
    3803        3510 :     item->arg = arg;
    3804        3510 :     item->next = Xact_callbacks;
    3805        3510 :     Xact_callbacks = item;
    3806        3510 : }
    3807             : 
    3808             : void
    3809           0 : UnregisterXactCallback(XactCallback callback, void *arg)
    3810             : {
    3811             :     XactCallbackItem *item;
    3812             :     XactCallbackItem *prev;
    3813             : 
    3814           0 :     prev = NULL;
    3815           0 :     for (item = Xact_callbacks; item; prev = item, item = item->next)
    3816             :     {
    3817           0 :         if (item->callback == callback && item->arg == arg)
    3818             :         {
    3819           0 :             if (prev)
    3820           0 :                 prev->next = item->next;
    3821             :             else
    3822           0 :                 Xact_callbacks = item->next;
    3823           0 :             pfree(item);
    3824           0 :             break;
    3825             :         }
    3826             :     }
    3827           0 : }
    3828             : 
    3829             : static void
    3830     1523516 : CallXactCallbacks(XactEvent event)
    3831             : {
    3832             :     XactCallbackItem *item;
    3833             :     XactCallbackItem *next;
    3834             : 
    3835     1810978 :     for (item = Xact_callbacks; item; item = next)
    3836             :     {
    3837             :         /* allow callbacks to unregister themselves when called */
    3838      287464 :         next = item->next;
    3839      287464 :         item->callback(event, item->arg);
    3840             :     }
    3841     1523514 : }
    3842             : 
    3843             : 
    3844             : /*
    3845             :  * Register or deregister callback functions for start- and end-of-subxact
    3846             :  * operations.
    3847             :  *
    3848             :  * Pretty much same as above, but for subtransaction events.
    3849             :  *
    3850             :  * At subtransaction end, the callback occurs post-subcommit or post-subabort,
    3851             :  * so the callback functions can only do noncritical cleanup.  At
    3852             :  * subtransaction start, the callback is called when the subtransaction has
    3853             :  * finished initializing.
    3854             :  */
    3855             : void
    3856        3510 : RegisterSubXactCallback(SubXactCallback callback, void *arg)
    3857             : {
    3858             :     SubXactCallbackItem *item;
    3859             : 
    3860             :     item = (SubXactCallbackItem *)
    3861        3510 :         MemoryContextAlloc(TopMemoryContext, sizeof(SubXactCallbackItem));
    3862        3510 :     item->callback = callback;
    3863        3510 :     item->arg = arg;
    3864        3510 :     item->next = SubXact_callbacks;
    3865        3510 :     SubXact_callbacks = item;
    3866        3510 : }
    3867             : 
    3868             : void
    3869           0 : UnregisterSubXactCallback(SubXactCallback callback, void *arg)
    3870             : {
    3871             :     SubXactCallbackItem *item;
    3872             :     SubXactCallbackItem *prev;
    3873             : 
    3874           0 :     prev = NULL;
    3875           0 :     for (item = SubXact_callbacks; item; prev = item, item = item->next)
    3876             :     {
    3877           0 :         if (item->callback == callback && item->arg == arg)
    3878             :         {
    3879           0 :             if (prev)
    3880           0 :                 prev->next = item->next;
    3881             :             else
    3882           0 :                 SubXact_callbacks = item->next;
    3883           0 :             pfree(item);
    3884           0 :             break;
    3885             :         }
    3886             :     }
    3887           0 : }
    3888             : 
    3889             : static void
    3890       50852 : CallSubXactCallbacks(SubXactEvent event,
    3891             :                      SubTransactionId mySubid,
    3892             :                      SubTransactionId parentSubid)
    3893             : {
    3894             :     SubXactCallbackItem *item;
    3895             :     SubXactCallbackItem *next;
    3896             : 
    3897       90412 :     for (item = SubXact_callbacks; item; item = next)
    3898             :     {
    3899             :         /* allow callbacks to unregister themselves when called */
    3900       39560 :         next = item->next;
    3901       39560 :         item->callback(event, mySubid, parentSubid, item->arg);
    3902             :     }
    3903       50852 : }
    3904             : 
    3905             : 
    3906             : /* ----------------------------------------------------------------
    3907             :  *                     transaction block support
    3908             :  * ----------------------------------------------------------------
    3909             :  */
    3910             : 
    3911             : /*
    3912             :  *  BeginTransactionBlock
    3913             :  *      This executes a BEGIN command.
    3914             :  */
    3915             : void
    3916       16008 : BeginTransactionBlock(void)
    3917             : {
    3918       16008 :     TransactionState s = CurrentTransactionState;
    3919             : 
    3920       16008 :     switch (s->blockState)
    3921             :     {
    3922             :             /*
    3923             :              * We are not inside a transaction block, so allow one to begin.
    3924             :              */
    3925       15052 :         case TBLOCK_STARTED:
    3926       15052 :             s->blockState = TBLOCK_BEGIN;
    3927       15052 :             break;
    3928             : 
    3929             :             /*
    3930             :              * BEGIN converts an implicit transaction block to a regular one.
    3931             :              * (Note that we allow this even if we've already done some
    3932             :              * commands, which is a bit odd but matches historical practice.)
    3933             :              */
    3934         956 :         case TBLOCK_IMPLICIT_INPROGRESS:
    3935         956 :             s->blockState = TBLOCK_BEGIN;
    3936         956 :             break;
    3937             : 
    3938             :             /*
    3939             :              * Already a transaction block in progress.
    3940             :              */
    3941           0 :         case TBLOCK_INPROGRESS:
    3942             :         case TBLOCK_PARALLEL_INPROGRESS:
    3943             :         case TBLOCK_SUBINPROGRESS:
    3944             :         case TBLOCK_ABORT:
    3945             :         case TBLOCK_SUBABORT:
    3946           0 :             ereport(WARNING,
    3947             :                     (errcode(ERRCODE_ACTIVE_SQL_TRANSACTION),
    3948             :                      errmsg("there is already a transaction in progress")));
    3949           0 :             break;
    3950             : 
    3951             :             /* These cases are invalid. */
    3952           0 :         case TBLOCK_DEFAULT:
    3953             :         case TBLOCK_BEGIN:
    3954             :         case TBLOCK_SUBBEGIN:
    3955             :         case TBLOCK_END:
    3956             :         case TBLOCK_SUBRELEASE:
    3957             :         case TBLOCK_SUBCOMMIT:
    3958             :         case TBLOCK_ABORT_END:
    3959             :         case TBLOCK_SUBABORT_END:
    3960             :         case TBLOCK_ABORT_PENDING:
    3961             :         case TBLOCK_SUBABORT_PENDING:
    3962             :         case TBLOCK_SUBRESTART:
    3963             :         case TBLOCK_SUBABORT_RESTART:
    3964             :         case TBLOCK_PREPARE:
    3965           0 :             elog(FATAL, "BeginTransactionBlock: unexpected state %s",
    3966             :                  BlockStateAsString(s->blockState));
    3967             :             break;
    3968             :     }
    3969       16008 : }
    3970             : 
    3971             : /*
    3972             :  *  PrepareTransactionBlock
    3973             :  *      This executes a PREPARE command.
    3974             :  *
    3975             :  * Since PREPARE may actually do a ROLLBACK, the result indicates what
    3976             :  * happened: true for PREPARE, false for ROLLBACK.
    3977             :  *
    3978             :  * Note that we don't actually do anything here except change blockState.
    3979             :  * The real work will be done in the upcoming PrepareTransaction().
    3980             :  * We do it this way because it's not convenient to change memory context,
    3981             :  * resource owner, etc while executing inside a Portal.
    3982             :  */
    3983             : bool
    3984         858 : PrepareTransactionBlock(const char *gid)
    3985             : {
    3986             :     TransactionState s;
    3987             :     bool        result;
    3988             : 
    3989             :     /* Set up to commit the current transaction */
    3990         858 :     result = EndTransactionBlock(false);
    3991             : 
    3992             :     /* If successful, change outer tblock state to PREPARE */
    3993         858 :     if (result)
    3994             :     {
    3995         854 :         s = CurrentTransactionState;
    3996             : 
    3997        1174 :         while (s->parent != NULL)
    3998         320 :             s = s->parent;
    3999             : 
    4000         854 :         if (s->blockState == TBLOCK_END)
    4001             :         {
    4002             :             /* Save GID where PrepareTransaction can find it again */
    4003         854 :             prepareGID = MemoryContextStrdup(TopTransactionContext, gid);
    4004             : 
    4005         854 :             s->blockState = TBLOCK_PREPARE;
    4006             :         }
    4007             :         else
    4008             :         {
    4009             :             /*
    4010             :              * ignore case where we are not in a transaction;
    4011             :              * EndTransactionBlock already issued a warning.
    4012             :              */
    4013             :             Assert(s->blockState == TBLOCK_STARTED ||
    4014             :                    s->blockState == TBLOCK_IMPLICIT_INPROGRESS);
    4015             :             /* Don't send back a PREPARE result tag... */
    4016           0 :             result = false;
    4017             :         }
    4018             :     }
    4019             : 
    4020         858 :     return result;
    4021             : }
    4022             : 
    4023             : /*
    4024             :  *  EndTransactionBlock
    4025             :  *      This executes a COMMIT command.
    4026             :  *
    4027             :  * Since COMMIT may actually do a ROLLBACK, the result indicates what
    4028             :  * happened: true for COMMIT, false for ROLLBACK.
    4029             :  *
    4030             :  * Note that we don't actually do anything here except change blockState.
    4031             :  * The real work will be done in the upcoming CommitTransactionCommand().
    4032             :  * We do it this way because it's not convenient to change memory context,
    4033             :  * resource owner, etc while executing inside a Portal.
    4034             :  */
    4035             : bool
    4036       12946 : EndTransactionBlock(bool chain)
    4037             : {
    4038       12946 :     TransactionState s = CurrentTransactionState;
    4039       12946 :     bool        result = false;
    4040             : 
    4041       12946 :     switch (s->blockState)
    4042             :     {
    4043             :             /*
    4044             :              * We are in a transaction block, so tell CommitTransactionCommand
    4045             :              * to COMMIT.
    4046             :              */
    4047       11074 :         case TBLOCK_INPROGRESS:
    4048       11074 :             s->blockState = TBLOCK_END;
    4049       11074 :             result = true;
    4050       11074 :             break;
    4051             : 
    4052             :             /*
    4053             :              * We are in an implicit transaction block.  If AND CHAIN was
    4054             :              * specified, error.  Otherwise commit, but issue a warning
    4055             :              * because there was no explicit BEGIN before this.
    4056             :              */
    4057          48 :         case TBLOCK_IMPLICIT_INPROGRESS:
    4058          48 :             if (chain)
    4059          24 :                 ereport(ERROR,
    4060             :                         (errcode(ERRCODE_NO_ACTIVE_SQL_TRANSACTION),
    4061             :                 /* translator: %s represents an SQL statement name */
    4062             :                          errmsg("%s can only be used in transaction blocks",
    4063             :                                 "COMMIT AND CHAIN")));
    4064             :             else
    4065          24 :                 ereport(WARNING,
    4066             :                         (errcode(ERRCODE_NO_ACTIVE_SQL_TRANSACTION),
    4067             :                          errmsg("there is no transaction in progress")));
    4068          24 :             s->blockState = TBLOCK_END;
    4069          24 :             result = true;
    4070          24 :             break;
    4071             : 
    4072             :             /*
    4073             :              * We are in a failed transaction block.  Tell
    4074             :              * CommitTransactionCommand it's time to exit the block.
    4075             :              */
    4076         694 :         case TBLOCK_ABORT:
    4077         694 :             s->blockState = TBLOCK_ABORT_END;
    4078         694 :             break;
    4079             : 
    4080             :             /*
    4081             :              * We are in a live subtransaction block.  Set up to subcommit all
    4082             :              * open subtransactions and then commit the main transaction.
    4083             :              */
    4084        1046 :         case TBLOCK_SUBINPROGRESS:
    4085        2226 :             while (s->parent != NULL)
    4086             :             {
    4087        1180 :                 if (s->blockState == TBLOCK_SUBINPROGRESS)
    4088        1180 :                     s->blockState = TBLOCK_SUBCOMMIT;
    4089             :                 else
    4090           0 :                     elog(FATAL, "EndTransactionBlock: unexpected state %s",
    4091             :                          BlockStateAsString(s->blockState));
    4092        1180 :                 s = s->parent;
    4093             :             }
    4094        1046 :             if (s->blockState == TBLOCK_INPROGRESS)
    4095        1046 :                 s->blockState = TBLOCK_END;
    4096             :             else
    4097           0 :                 elog(FATAL, "EndTransactionBlock: unexpected state %s",
    4098             :                      BlockStateAsString(s->blockState));
    4099        1046 :             result = true;
    4100        1046 :             break;
    4101             : 
    4102             :             /*
    4103             :              * Here we are inside an aborted subtransaction.  Treat the COMMIT
    4104             :              * as ROLLBACK: set up to abort everything and exit the main
    4105             :              * transaction.
    4106             :              */
    4107          60 :         case TBLOCK_SUBABORT:
    4108         120 :             while (s->parent != NULL)
    4109             :             {
    4110          60 :                 if (s->blockState == TBLOCK_SUBINPROGRESS)
    4111           0 :                     s->blockState = TBLOCK_SUBABORT_PENDING;
    4112          60 :                 else if (s->blockState == TBLOCK_SUBABORT)
    4113          60 :                     s->blockState = TBLOCK_SUBABORT_END;
    4114             :                 else
    4115           0 :                     elog(FATAL, "EndTransactionBlock: unexpected state %s",
    4116             :                          BlockStateAsString(s->blockState));
    4117          60 :                 s = s->parent;
    4118             :             }
    4119          60 :             if (s->blockState == TBLOCK_INPROGRESS)
    4120          60 :                 s->blockState = TBLOCK_ABORT_PENDING;
    4121           0 :             else if (s->blockState == TBLOCK_ABORT)
    4122           0 :                 s->blockState = TBLOCK_ABORT_END;
    4123             :             else
    4124           0 :                 elog(FATAL, "EndTransactionBlock: unexpected state %s",
    4125             :                      BlockStateAsString(s->blockState));
    4126          60 :             break;
    4127             : 
    4128             :             /*
    4129             :              * The user issued COMMIT when not inside a transaction.  For
    4130             :              * COMMIT without CHAIN, issue a WARNING, staying in
    4131             :              * TBLOCK_STARTED state.  The upcoming call to
    4132             :              * CommitTransactionCommand() will then close the transaction and
    4133             :              * put us back into the default state.  For COMMIT AND CHAIN,
    4134             :              * error.
    4135             :              */
    4136          24 :         case TBLOCK_STARTED:
    4137          24 :             if (chain)
    4138           6 :                 ereport(ERROR,
    4139             :                         (errcode(ERRCODE_NO_ACTIVE_SQL_TRANSACTION),
    4140             :                 /* translator: %s represents an SQL statement name */
    4141             :                          errmsg("%s can only be used in transaction blocks",
    4142             :                                 "COMMIT AND CHAIN")));
    4143             :             else
    4144          18 :                 ereport(WARNING,
    4145             :                         (errcode(ERRCODE_NO_ACTIVE_SQL_TRANSACTION),
    4146             :                          errmsg("there is no transaction in progress")));
    4147          18 :             result = true;
    4148          18 :             break;
    4149             : 
    4150             :             /*
    4151             :              * The user issued a COMMIT that somehow ran inside a parallel
    4152             :              * worker.  We can't cope with that.
    4153             :              */
    4154           0 :         case TBLOCK_PARALLEL_INPROGRESS:
    4155           0 :             ereport(FATAL,
    4156             :                     (errcode(ERRCODE_INVALID_TRANSACTION_STATE),
    4157             :                      errmsg("cannot commit during a parallel operation")));
    4158             :             break;
    4159             : 
    4160             :             /* These cases are invalid. */
    4161           0 :         case TBLOCK_DEFAULT:
    4162             :         case TBLOCK_BEGIN:
    4163             :         case TBLOCK_SUBBEGIN:
    4164             :         case TBLOCK_END:
    4165             :         case TBLOCK_SUBRELEASE:
    4166             :         case TBLOCK_SUBCOMMIT:
    4167             :         case TBLOCK_ABORT_END:
    4168             :         case TBLOCK_SUBABORT_END:
    4169             :         case TBLOCK_ABORT_PENDING:
    4170             :         case TBLOCK_SUBABORT_PENDING:
    4171             :         case TBLOCK_SUBRESTART:
    4172             :         case TBLOCK_SUBABORT_RESTART:
    4173             :         case TBLOCK_PREPARE:
    4174           0 :             elog(FATAL, "EndTransactionBlock: unexpected state %s",
    4175             :                  BlockStateAsString(s->blockState));
    4176             :             break;
    4177             :     }
    4178             : 
    4179       12916 :     Assert(s->blockState == TBLOCK_STARTED ||
    4180             :            s->blockState == TBLOCK_END ||
    4181             :            s->blockState == TBLOCK_ABORT_END ||
    4182             :            s->blockState == TBLOCK_ABORT_PENDING);
    4183             : 
    4184       12916 :     s->chain = chain;
    4185             : 
    4186       12916 :     return result;
    4187             : }
    4188             : 
    4189             : /*
    4190             :  *  UserAbortTransactionBlock
    4191             :  *      This executes a ROLLBACK command.
    4192             :  *
    4193             :  * As above, we don't actually do anything here except change blockState.
    4194             :  */
    4195             : void
    4196        2750 : UserAbortTransactionBlock(bool chain)
    4197             : {
    4198        2750 :     TransactionState s = CurrentTransactionState;
    4199             : 
    4200        2750 :     switch (s->blockState)
    4201             :     {
    4202             :             /*
    4203             :              * We are inside a transaction block and we got a ROLLBACK command
    4204             :              * from the user, so tell CommitTransactionCommand to abort and
    4205             :              * exit the transaction block.
    4206             :              */
    4207        1906 :         case TBLOCK_INPROGRESS:
    4208        1906 :             s->blockState = TBLOCK_ABORT_PENDING;
    4209        1906 :             break;
    4210             : 
    4211             :             /*
    4212             :              * We are inside a failed transaction block and we got a ROLLBACK
    4213             :              * command from the user.  Abort processing is already done, so
    4214             :              * CommitTransactionCommand just has to cleanup and go back to
    4215             :              * idle state.
    4216             :              */
    4217         666 :         case TBLOCK_ABORT:
    4218         666 :             s->blockState = TBLOCK_ABORT_END;
    4219         666 :             break;
    4220             : 
    4221             :             /*
    4222             :              * We are inside a subtransaction.  Mark everything up to top
    4223             :              * level as exitable.
    4224             :              */
    4225         100 :         case TBLOCK_SUBINPROGRESS:
    4226             :         case TBLOCK_SUBABORT:
    4227         412 :             while (s->parent != NULL)
    4228             :             {
    4229         312 :                 if (s->blockState == TBLOCK_SUBINPROGRESS)
    4230         292 :                     s->blockState = TBLOCK_SUBABORT_PENDING;
    4231          20 :                 else if (s->blockState == TBLOCK_SUBABORT)
    4232          20 :                     s->blockState = TBLOCK_SUBABORT_END;
    4233             :                 else
    4234           0 :                     elog(FATAL, "UserAbortTransactionBlock: unexpected state %s",
    4235             :                          BlockStateAsString(s->blockState));
    4236         312 :                 s = s->parent;
    4237             :             }
    4238         100 :             if (s->blockState == TBLOCK_INPROGRESS)
    4239         100 :                 s->blockState = TBLOCK_ABORT_PENDING;
    4240           0 :             else if (s->blockState == TBLOCK_ABORT)
    4241           0 :                 s->blockState = TBLOCK_ABORT_END;
    4242             :             else
    4243           0 :                 elog(FATAL, "UserAbortTransactionBlock: unexpected state %s",
    4244             :                      BlockStateAsString(s->blockState));
    4245         100 :             break;
    4246             : 
    4247             :             /*
    4248             :              * The user issued ABORT when not inside a transaction.  For
    4249             :              * ROLLBACK without CHAIN, issue a WARNING and go to abort state.
    4250             :              * The upcoming call to CommitTransactionCommand() will then put
    4251             :              * us back into the default state.  For ROLLBACK AND CHAIN, error.
    4252             :              *
    4253             :              * We do the same thing with ABORT inside an implicit transaction,
    4254             :              * although in this case we might be rolling back actual database
    4255             :              * state changes.  (It's debatable whether we should issue a
    4256             :              * WARNING in this case, but we have done so historically.)
    4257             :              */
    4258          78 :         case TBLOCK_STARTED:
    4259             :         case TBLOCK_IMPLICIT_INPROGRESS:
    4260          78 :             if (chain)
    4261          30 :                 ereport(ERROR,
    4262             :                         (errcode(ERRCODE_NO_ACTIVE_SQL_TRANSACTION),
    4263             :                 /* translator: %s represents an SQL statement name */
    4264             :                          errmsg("%s can only be used in transaction blocks",
    4265             :                                 "ROLLBACK AND CHAIN")));
    4266             :             else
    4267          48 :                 ereport(WARNING,
    4268             :                         (errcode(ERRCODE_NO_ACTIVE_SQL_TRANSACTION),
    4269             :                          errmsg("there is no transaction in progress")));
    4270          48 :             s->blockState = TBLOCK_ABORT_PENDING;
    4271          48 :             break;
    4272             : 
    4273             :             /*
    4274             :              * The user issued an ABORT that somehow ran inside a parallel
    4275             :              * worker.  We can't cope with that.
    4276             :              */
    4277           0 :         case TBLOCK_PARALLEL_INPROGRESS:
    4278           0 :             ereport(FATAL,
    4279             :                     (errcode(ERRCODE_INVALID_TRANSACTION_STATE),
    4280             :                      errmsg("cannot abort during a parallel operation")));
    4281             :             break;
    4282             : 
    4283             :             /* These cases are invalid. */
    4284           0 :         case TBLOCK_DEFAULT:
    4285             :         case TBLOCK_BEGIN:
    4286             :         case TBLOCK_SUBBEGIN:
    4287             :         case TBLOCK_END:
    4288             :         case TBLOCK_SUBRELEASE:
    4289             :         case TBLOCK_SUBCOMMIT:
    4290             :         case TBLOCK_ABORT_END:
    4291             :         case TBLOCK_SUBABORT_END:
    4292             :         case TBLOCK_ABORT_PENDING:
    4293             :         case TBLOCK_SUBABORT_PENDING:
    4294             :         case TBLOCK_SUBRESTART:
    4295             :         case TBLOCK_SUBABORT_RESTART:
    4296             :         case TBLOCK_PREPARE:
    4297           0 :             elog(FATAL, "UserAbortTransactionBlock: unexpected state %s",
    4298             :                  BlockStateAsString(s->blockState));
    4299             :             break;
    4300             :     }
    4301             : 
    4302        2720 :     Assert(s->blockState == TBLOCK_ABORT_END ||
    4303             :            s->blockState == TBLOCK_ABORT_PENDING);
    4304             : 
    4305        2720 :     s->chain = chain;
    4306        2720 : }
    4307             : 
    4308             : /*
    4309             :  * BeginImplicitTransactionBlock
    4310             :  *      Start an implicit transaction block if we're not already in one.
    4311             :  *
    4312             :  * Unlike BeginTransactionBlock, this is called directly from the main loop
    4313             :  * in postgres.c, not within a Portal.  So we can just change blockState
    4314             :  * without a lot of ceremony.  We do not expect caller to do
    4315             :  * CommitTransactionCommand/StartTransactionCommand.
    4316             :  */
    4317             : void
    4318       81698 : BeginImplicitTransactionBlock(void)
    4319             : {
    4320       81698 :     TransactionState s = CurrentTransactionState;
    4321             : 
    4322             :     /*
    4323             :      * If we are in STARTED state (that is, no transaction block is open),
    4324             :      * switch to IMPLICIT_INPROGRESS state, creating an implicit transaction
    4325             :      * block.
    4326             :      *
    4327             :      * For caller convenience, we consider all other transaction states as
    4328             :      * legal here; otherwise the caller would need its own state check, which
    4329             :      * seems rather pointless.
    4330             :      */
    4331       81698 :     if (s->blockState == TBLOCK_STARTED)
    4332        9424 :         s->blockState = TBLOCK_IMPLICIT_INPROGRESS;
    4333       81698 : }
    4334             : 
    4335             : /*
    4336             :  * EndImplicitTransactionBlock
    4337             :  *      End an implicit transaction block, if we're in one.
    4338             :  *
    4339             :  * Like EndTransactionBlock, we just make any needed blockState change here.
    4340             :  * The real work will be done in the upcoming CommitTransactionCommand().
    4341             :  */
    4342             : void
    4343       33658 : EndImplicitTransactionBlock(void)
    4344             : {
    4345       33658 :     TransactionState s = CurrentTransactionState;
    4346             : 
    4347             :     /*
    4348             :      * If we are in IMPLICIT_INPROGRESS state, switch back to STARTED state,
    4349             :      * allowing CommitTransactionCommand to commit whatever happened during
    4350             :      * the implicit transaction block as though it were a single statement.
    4351             :      *
    4352             :      * For caller convenience, we consider all other transaction states as
    4353             :      * legal here; otherwise the caller would need its own state check, which
    4354             :      * seems rather pointless.
    4355             :      */
    4356       33658 :     if (s->blockState == TBLOCK_IMPLICIT_INPROGRESS)
    4357        8308 :         s->blockState = TBLOCK_STARTED;
    4358       33658 : }
    4359             : 
    4360             : /*
    4361             :  * DefineSavepoint
    4362             :  *      This executes a SAVEPOINT command.
    4363             :  */
    4364             : void
    4365        2776 : DefineSavepoint(const char *name)
    4366             : {
    4367        2776 :     TransactionState s = CurrentTransactionState;
    4368             : 
    4369             :     /*
    4370             :      * Workers synchronize transaction state at the beginning of each parallel
    4371             :      * operation, so we can't account for new subtransactions after that
    4372             :      * point.  (Note that this check will certainly error out if s->blockState
    4373             :      * is TBLOCK_PARALLEL_INPROGRESS, so we can treat that as an invalid case
    4374             :      * below.)
    4375             :      */
    4376        2776 :     if (IsInParallelMode() || IsParallelWorker())
    4377           0 :         ereport(ERROR,
    4378             :                 (errcode(ERRCODE_INVALID_TRANSACTION_STATE),
    4379             :                  errmsg("cannot define savepoints during a parallel operation")));
    4380             : 
    4381        2776 :     switch (s->blockState)
    4382             :     {
    4383        2764 :         case TBLOCK_INPROGRESS:
    4384             :         case TBLOCK_SUBINPROGRESS:
    4385             :             /* Normal subtransaction start */
    4386        2764 :             PushTransaction();
    4387        2764 :             s = CurrentTransactionState;    /* changed by push */
    4388             : 
    4389             :             /*
    4390             :              * Savepoint names, like the TransactionState block itself, live
    4391             :              * in TopTransactionContext.
    4392             :              */
    4393        2764 :             if (name)
    4394        2052 :                 s->name = MemoryContextStrdup(TopTransactionContext, name);
    4395        2764 :             break;
    4396             : 
    4397             :             /*
    4398             :              * We disallow savepoint commands in implicit transaction blocks.
    4399             :              * There would be no great difficulty in allowing them so far as
    4400             :              * this module is concerned, but a savepoint seems inconsistent
    4401             :              * with exec_simple_query's behavior of abandoning the whole query
    4402             :              * string upon error.  Also, the point of an implicit transaction
    4403             :              * block (as opposed to a regular one) is to automatically close
    4404             :              * after an error, so it's hard to see how a savepoint would fit
    4405             :              * into that.
    4406             :              *
    4407             :              * The error messages for this are phrased as if there were no
    4408             :              * active transaction block at all, which is historical but
    4409             :              * perhaps could be improved.
    4410             :              */
    4411          12 :         case TBLOCK_IMPLICIT_INPROGRESS:
    4412          12 :             ereport(ERROR,
    4413             :                     (errcode(ERRCODE_NO_ACTIVE_SQL_TRANSACTION),
    4414             :             /* translator: %s represents an SQL statement name */
    4415             :                      errmsg("%s can only be used in transaction blocks",
    4416             :                             "SAVEPOINT")));
    4417             :             break;
    4418             : 
    4419             :             /* These cases are invalid. */
    4420           0 :         case TBLOCK_DEFAULT:
    4421             :         case TBLOCK_STARTED:
    4422             :         case TBLOCK_BEGIN:
    4423             :         case TBLOCK_PARALLEL_INPROGRESS:
    4424             :         case TBLOCK_SUBBEGIN:
    4425             :         case TBLOCK_END:
    4426             :         case TBLOCK_SUBRELEASE:
    4427             :         case TBLOCK_SUBCOMMIT:
    4428             :         case TBLOCK_ABORT:
    4429             :         case TBLOCK_SUBABORT:
    4430             :         case TBLOCK_ABORT_END:
    4431             :         case TBLOCK_SUBABORT_END:
    4432             :         case TBLOCK_ABORT_PENDING:
    4433             :         case TBLOCK_SUBABORT_PENDING:
    4434             :         case TBLOCK_SUBRESTART:
    4435             :         case TBLOCK_SUBABORT_RESTART:
    4436             :         case TBLOCK_PREPARE:
    4437           0 :             elog(FATAL, "DefineSavepoint: unexpected state %s",
    4438             :                  BlockStateAsString(s->blockState));
    4439             :             break;
    4440             :     }
    4441        2764 : }
    4442             : 
    4443             : /*
    4444             :  * ReleaseSavepoint
    4445             :  *      This executes a RELEASE command.
    4446             :  *
    4447             :  * As above, we don't actually do anything here except change blockState.
    4448             :  */
    4449             : void
    4450         282 : ReleaseSavepoint(const char *name)
    4451             : {
    4452         282 :     TransactionState s = CurrentTransactionState;
    4453             :     TransactionState target,
    4454             :                 xact;
    4455             : 
    4456             :     /*
    4457             :      * Workers synchronize transaction state at the beginning of each parallel
    4458             :      * operation, so we can't account for transaction state change after that
    4459             :      * point.  (Note that this check will certainly error out if s->blockState
    4460             :      * is TBLOCK_PARALLEL_INPROGRESS, so we can treat that as an invalid case
    4461             :      * below.)
    4462             :      */
    4463         282 :     if (IsInParallelMode() || IsParallelWorker())
    4464           0 :         ereport(ERROR,
    4465             :                 (errcode(ERRCODE_INVALID_TRANSACTION_STATE),
    4466             :                  errmsg("cannot release savepoints during a parallel operation")));
    4467             : 
    4468         282 :     switch (s->blockState)
    4469             :     {
    4470             :             /*
    4471             :              * We can't release a savepoint if there is no savepoint defined.
    4472             :              */
    4473           0 :         case TBLOCK_INPROGRESS:
    4474           0 :             ereport(ERROR,
    4475             :                     (errcode(ERRCODE_S_E_INVALID_SPECIFICATION),
    4476             :                      errmsg("savepoint \"%s\" does not exist", name)));
    4477             :             break;
    4478             : 
    4479           6 :         case TBLOCK_IMPLICIT_INPROGRESS:
    4480             :             /* See comment about implicit transactions in DefineSavepoint */
    4481           6 :             ereport(ERROR,
    4482             :                     (errcode(ERRCODE_NO_ACTIVE_SQL_TRANSACTION),
    4483             :             /* translator: %s represents an SQL statement name */
    4484             :                      errmsg("%s can only be used in transaction blocks",
    4485             :                             "RELEASE SAVEPOINT")));
    4486             :             break;
    4487             : 
    4488             :             /*
    4489             :              * We are in a non-aborted subtransaction.  This is the only valid
    4490             :              * case.
    4491             :              */
    4492         276 :         case TBLOCK_SUBINPROGRESS:
    4493         276 :             break;
    4494             : 
    4495             :             /* These cases are invalid. */
    4496           0 :         case TBLOCK_DEFAULT:
    4497             :         case TBLOCK_STARTED:
    4498             :         case TBLOCK_BEGIN:
    4499             :         case TBLOCK_PARALLEL_INPROGRESS:
    4500             :         case TBLOCK_SUBBEGIN:
    4501             :         case TBLOCK_END:
    4502             :         case TBLOCK_SUBRELEASE:
    4503             :         case TBLOCK_SUBCOMMIT:
    4504             :         case TBLOCK_ABORT:
    4505             :         case TBLOCK_SUBABORT:
    4506             :         case TBLOCK_ABORT_END:
    4507             :         case TBLOCK_SUBABORT_END:
    4508             :         case TBLOCK_ABORT_PENDING:
    4509             :         case TBLOCK_SUBABORT_PENDING:
    4510             :         case TBLOCK_SUBRESTART:
    4511             :         case TBLOCK_SUBABORT_RESTART:
    4512             :         case TBLOCK_PREPARE:
    4513           0 :             elog(FATAL, "ReleaseSavepoint: unexpected state %s",
    4514             :                  BlockStateAsString(s->blockState));
    4515             :             break;
    4516             :     }
    4517             : 
    4518         448 :     for (target = s; PointerIsValid(target); target = target->parent)
    4519             :     {
    4520         448 :         if (PointerIsValid(target->name) && strcmp(target->name, name) == 0)
    4521         276 :             break;
    4522             :     }
    4523             : 
    4524         276 :     if (!PointerIsValid(target))
    4525           0 :         ereport(ERROR,
    4526             :                 (errcode(ERRCODE_S_E_INVALID_SPECIFICATION),
    4527             :                  errmsg("savepoint \"%s\" does not exist", name)));
    4528             : 
    4529             :     /* disallow crossing savepoint level boundaries */
    4530         276 :     if (target->savepointLevel != s->savepointLevel)
    4531           0 :         ereport(ERROR,
    4532             :                 (errcode(ERRCODE_S_E_INVALID_SPECIFICATION),
    4533             :                  errmsg("savepoint \"%s\" does not exist within current savepoint level", name)));
    4534             : 
    4535             :     /*
    4536             :      * Mark "commit pending" all subtransactions up to the target
    4537             :      * subtransaction.  The actual commits will happen when control gets to
    4538             :      * CommitTransactionCommand.
    4539             :      */
    4540         276 :     xact = CurrentTransactionState;
    4541             :     for (;;)
    4542             :     {
    4543         172 :         Assert(xact->blockState == TBLOCK_SUBINPROGRESS);
    4544         448 :         xact->blockState = TBLOCK_SUBRELEASE;
    4545         448 :         if (xact == target)
    4546         276 :             break;
    4547         172 :         xact = xact->parent;
    4548             :         Assert(PointerIsValid(xact));
    4549             :     }
    4550         276 : }
    4551             : 
    4552             : /*
    4553             :  * RollbackToSavepoint
    4554             :  *      This executes a ROLLBACK TO <savepoint> command.
    4555             :  *
    4556             :  * As above, we don't actually do anything here except change blockState.
    4557             :  */
    4558             : void
    4559         724 : RollbackToSavepoint(const char *name)
    4560             : {
    4561         724 :     TransactionState s = CurrentTransactionState;
    4562             :     TransactionState target,
    4563             :                 xact;
    4564             : 
    4565             :     /*
    4566             :      * Workers synchronize transaction state at the beginning of each parallel
    4567             :      * operation, so we can't account for transaction state change after that
    4568             :      * point.  (Note that this check will certainly error out if s->blockState
    4569             :      * is TBLOCK_PARALLEL_INPROGRESS, so we can treat that as an invalid case
    4570             :      * below.)
    4571             :      */
    4572         724 :     if (IsInParallelMode() || IsParallelWorker())
    4573           0 :         ereport(ERROR,
    4574             :                 (errcode(ERRCODE_INVALID_TRANSACTION_STATE),
    4575             :                  errmsg("cannot rollback to savepoints during a parallel operation")));
    4576             : 
    4577         724 :     switch (s->blockState)
    4578             :     {
    4579             :             /*
    4580             :              * We can't rollback to a savepoint if there is no savepoint
    4581             :              * defined.
    4582             :              */
    4583           6 :         case TBLOCK_INPROGRESS:
    4584             :         case TBLOCK_ABORT:
    4585           6 :             ereport(ERROR,
    4586             :                     (errcode(ERRCODE_S_E_INVALID_SPECIFICATION),
    4587             :                      errmsg("savepoint \"%s\" does not exist", name)));
    4588             :             break;
    4589             : 
    4590           6 :         case TBLOCK_IMPLICIT_INPROGRESS:
    4591             :             /* See comment about implicit transactions in DefineSavepoint */
    4592           6 :             ereport(ERROR,
    4593             :                     (errcode(ERRCODE_NO_ACTIVE_SQL_TRANSACTION),
    4594             :             /* translator: %s represents an SQL statement name */
    4595             :                      errmsg("%s can only be used in transaction blocks",
    4596             :                             "ROLLBACK TO SAVEPOINT")));
    4597             :             break;
    4598             : 
    4599             :             /*
    4600             :              * There is at least one savepoint, so proceed.
    4601             :              */
    4602         712 :         case TBLOCK_SUBINPROGRESS:
    4603             :         case TBLOCK_SUBABORT:
    4604         712 :             break;
    4605             : 
    4606             :             /* These cases are invalid. */
    4607           0 :         case TBLOCK_DEFAULT:
    4608             :         case TBLOCK_STARTED:
    4609             :         case TBLOCK_BEGIN:
    4610             :         case TBLOCK_PARALLEL_INPROGRESS:
    4611             :         case TBLOCK_SUBBEGIN:
    4612             :         case TBLOCK_END:
    4613             :         case TBLOCK_SUBRELEASE:
    4614             :         case TBLOCK_SUBCOMMIT:
    4615             :         case TBLOCK_ABORT_END:
    4616             :         case TBLOCK_SUBABORT_END:
    4617             :         case TBLOCK_ABORT_PENDING:
    4618             :         case TBLOCK_SUBABORT_PENDING:
    4619             :         case TBLOCK_SUBRESTART:
    4620             :         case TBLOCK_SUBABORT_RESTART:
    4621             :         case TBLOCK_PREPARE:
    4622           0 :             elog(FATAL, "RollbackToSavepoint: unexpected state %s",
    4623             :                  BlockStateAsString(s->blockState));
    4624             :             break;
    4625             :     }
    4626             : 
    4627         764 :     for (target = s; PointerIsValid(target); target = target->parent)
    4628             :     {
    4629         764 :         if (PointerIsValid(target->name) && strcmp(target->name, name) == 0)
    4630         712 :             break;
    4631             :     }
    4632             : 
    4633         712 :     if (!PointerIsValid(target))
    4634           0 :         ereport(ERROR,
    4635             :                 (errcode(ERRCODE_S_E_INVALID_SPECIFICATION),
    4636             :                  errmsg("savepoint \"%s\" does not exist", name)));
    4637             : 
    4638             :     /* disallow crossing savepoint level boundaries */
    4639         712 :     if (target->savepointLevel != s->savepointLevel)
    4640           0 :         ereport(ERROR,
    4641             :                 (errcode(ERRCODE_S_E_INVALID_SPECIFICATION),
    4642             :                  errmsg("savepoint \"%s\" does not exist within current savepoint level", name)));
    4643             : 
    4644             :     /*
    4645             :      * Mark "abort pending" all subtransactions up to the target
    4646             :      * subtransaction.  The actual aborts will happen when control gets to
    4647             :      * CommitTransactionCommand.
    4648             :      */
    4649         712 :     xact = CurrentTransactionState;
    4650             :     for (;;)
    4651             :     {
    4652         764 :         if (xact == target)
    4653         712 :             break;
    4654          52 :         if (xact->blockState == TBLOCK_SUBINPROGRESS)
    4655          52 :             xact->blockState = TBLOCK_SUBABORT_PENDING;
    4656           0 :         else if (xact->blockState == TBLOCK_SUBABORT)
    4657           0 :             xact->blockState = TBLOCK_SUBABORT_END;
    4658             :         else
    4659           0 :             elog(FATAL, "RollbackToSavepoint: unexpected state %s",
    4660             :                  BlockStateAsString(xact->blockState));
    4661          52 :         xact = xact->parent;
    4662             :         Assert(PointerIsValid(xact));
    4663             :     }
    4664             : 
    4665             :     /* And mark the target as "restart pending" */
    4666         712 :     if (xact->blockState == TBLOCK_SUBINPROGRESS)
    4667         520 :         xact->blockState = TBLOCK_SUBRESTART;
    4668         192 :     else if (xact->blockState == TBLOCK_SUBABORT)
    4669         192 :         xact->blockState = TBLOCK_SUBABORT_RESTART;
    4670             :     else
    4671           0 :         elog(FATAL, "RollbackToSavepoint: unexpected state %s",
    4672             :              BlockStateAsString(xact->blockState));
    4673         712 : }
    4674             : 
    4675             : /*
    4676             :  * BeginInternalSubTransaction
    4677             :  *      This is the same as DefineSavepoint except it allows TBLOCK_STARTED,
    4678             :  *      TBLOCK_IMPLICIT_INPROGRESS, TBLOCK_PARALLEL_INPROGRESS, TBLOCK_END,
    4679             :  *      and TBLOCK_PREPARE states, and therefore it can safely be used in
    4680             :  *      functions that might be called when not inside a BEGIN block or when
    4681             :  *      running deferred triggers at COMMIT/PREPARE time.  Also, it
    4682             :  *      automatically does CommitTransactionCommand/StartTransactionCommand
    4683             :  *      instead of expecting the caller to do it.
    4684             :  */
    4685             : void
    4686       17276 : BeginInternalSubTransaction(const char *name)
    4687             : {
    4688       17276 :     TransactionState s = CurrentTransactionState;
    4689       17276 :     bool        save_ExitOnAnyError = ExitOnAnyError;
    4690             : 
    4691             :     /*
    4692             :      * Errors within this function are improbable, but if one does happen we
    4693             :      * force a FATAL exit.  Callers generally aren't prepared to handle losing
    4694             :      * control, and moreover our transaction state is probably corrupted if we
    4695             :      * fail partway through; so an ordinary ERROR longjmp isn't okay.
    4696             :      */
    4697       17276 :     ExitOnAnyError = true;
    4698             : 
    4699             :     /*
    4700             :      * We do not check for parallel mode here.  It's permissible to start and
    4701             :      * end "internal" subtransactions while in parallel mode, so long as no
    4702             :      * new XIDs or command IDs are assigned.  Enforcement of that occurs in
    4703             :      * AssignTransactionId() and CommandCounterIncrement().
    4704             :      */
    4705             : 
    4706       17276 :     switch (s->blockState)
    4707             :     {
    4708       17276 :         case TBLOCK_STARTED:
    4709             :         case TBLOCK_INPROGRESS:
    4710             :         case TBLOCK_IMPLICIT_INPROGRESS:
    4711             :         case TBLOCK_PARALLEL_INPROGRESS:
    4712             :         case TBLOCK_END:
    4713             :         case TBLOCK_PREPARE:
    4714             :         case TBLOCK_SUBINPROGRESS:
    4715             :             /* Normal subtransaction start */
    4716       17276 :             PushTransaction();
    4717       17276 :             s = CurrentTransactionState;    /* changed by push */
    4718             : 
    4719             :             /*
    4720             :              * Savepoint names, like the TransactionState block itself, live
    4721             :              * in TopTransactionContext.
    4722             :              */
    4723       17276 :             if (name)
    4724        1814 :                 s->name = MemoryContextStrdup(TopTransactionContext, name);
    4725       17276 :             break;
    4726             : 
    4727             :             /* These cases are invalid. */
    4728           0 :         case TBLOCK_DEFAULT:
    4729             :         case TBLOCK_BEGIN:
    4730             :         case TBLOCK_SUBBEGIN:
    4731             :         case TBLOCK_SUBRELEASE:
    4732             :         case TBLOCK_SUBCOMMIT:
    4733             :         case TBLOCK_ABORT:
    4734             :         case TBLOCK_SUBABORT:
    4735             :         case TBLOCK_ABORT_END:
    4736             :         case TBLOCK_SUBABORT_END:
    4737             :         case TBLOCK_ABORT_PENDING:
    4738             :         case TBLOCK_SUBABORT_PENDING:
    4739             :         case TBLOCK_SUBRESTART:
    4740             :         case TBLOCK_SUBABORT_RESTART:
    4741           0 :             elog(FATAL, "BeginInternalSubTransaction: unexpected state %s",
    4742             :                  BlockStateAsString(s->blockState));
    4743             :             break;
    4744             :     }
    4745             : 
    4746       17276 :     CommitTransactionCommand();
    4747       17276 :     StartTransactionCommand();
    4748             : 
    4749       17276 :     ExitOnAnyError = save_ExitOnAnyError;
    4750       17276 : }
    4751             : 
    4752             : /*
    4753             :  * ReleaseCurrentSubTransaction
    4754             :  *
    4755             :  * RELEASE (ie, commit) the innermost subtransaction, regardless of its
    4756             :  * savepoint name (if any).
    4757             :  * NB: do NOT use CommitTransactionCommand/StartTransactionCommand with this.
    4758             :  */
    4759             : void
    4760        9144 : ReleaseCurrentSubTransaction(void)
    4761             : {
    4762        9144 :     TransactionState s = CurrentTransactionState;
    4763             : 
    4764             :     /*
    4765             :      * We do not check for parallel mode here.  It's permissible to start and
    4766             :      * end "internal" subtransactions while in parallel mode, so long as no
    4767             :      * new XIDs or command IDs are assigned.
    4768             :      */
    4769             : 
    4770        9144 :     if (s->blockState != TBLOCK_SUBINPROGRESS)
    4771           0 :         elog(ERROR, "ReleaseCurrentSubTransaction: unexpected state %s",
    4772             :              BlockStateAsString(s->blockState));
    4773             :     Assert(s->state == TRANS_INPROGRESS);
    4774        9144 :     MemoryContextSwitchTo(CurTransactionContext);
    4775        9144 :     CommitSubTransaction();
    4776        9144 :     s = CurrentTransactionState;    /* changed by pop */
    4777             :     Assert(s->state == TRANS_INPROGRESS);
    4778        9144 : }
    4779             : 
    4780             : /*
    4781             :  * RollbackAndReleaseCurrentSubTransaction
    4782             :  *
    4783             :  * ROLLBACK and RELEASE (ie, abort) the innermost subtransaction, regardless
    4784             :  * of its savepoint name (if any).
    4785             :  * NB: do NOT use CommitTransactionCommand/StartTransactionCommand with this.
    4786             :  */
    4787             : void
    4788        8132 : RollbackAndReleaseCurrentSubTransaction(void)
    4789             : {
    4790        8132 :     TransactionState s = CurrentTransactionState;
    4791             : 
    4792             :     /*
    4793             :      * We do not check for parallel mode here.  It's permissible to start and
    4794             :      * end "internal" subtransactions while in parallel mode, so long as no
    4795             :      * new XIDs or command IDs are assigned.
    4796             :      */
    4797             : 
    4798        8132 :     switch (s->blockState)
    4799             :     {
    4800             :             /* Must be in a subtransaction */
    4801        8132 :         case TBLOCK_SUBINPROGRESS:
    4802             :         case TBLOCK_SUBABORT:
    4803        8132 :             break;
    4804             : 
    4805             :             /* These cases are invalid. */
    4806           0 :         case TBLOCK_DEFAULT:
    4807             :         case TBLOCK_STARTED:
    4808             :         case TBLOCK_BEGIN:
    4809             :         case TBLOCK_IMPLICIT_INPROGRESS:
    4810             :         case TBLOCK_PARALLEL_INPROGRESS:
    4811             :         case TBLOCK_SUBBEGIN:
    4812             :         case TBLOCK_INPROGRESS:
    4813             :         case TBLOCK_END:
    4814             :         case TBLOCK_SUBRELEASE:
    4815             :         case TBLOCK_SUBCOMMIT:
    4816             :         case TBLOCK_ABORT:
    4817             :         case TBLOCK_ABORT_END:
    4818             :         case TBLOCK_SUBABORT_END:
    4819             :         case TBLOCK_ABORT_PENDING:
    4820             :         case TBLOCK_SUBABORT_PENDING:
    4821             :         case TBLOCK_SUBRESTART:
    4822             :         case TBLOCK_SUBABORT_RESTART:
    4823             :         case TBLOCK_PREPARE:
    4824           0 :             elog(FATAL, "RollbackAndReleaseCurrentSubTransaction: unexpected state %s",
    4825             :                  BlockStateAsString(s->blockState));
    4826             :             break;
    4827             :     }
    4828             : 
    4829             :     /*
    4830             :      * Abort the current subtransaction, if needed.
    4831             :      */
    4832        8132 :     if (s->blockState == TBLOCK_SUBINPROGRESS)
    4833        6318 :         AbortSubTransaction();
    4834             : 
    4835             :     /* And clean it up, too */
    4836        8132 :     CleanupSubTransaction();
    4837             : 
    4838        8132 :     s = CurrentTransactionState;    /* changed by pop */
    4839             :     Assert(s->blockState == TBLOCK_SUBINPROGRESS ||
    4840             :            s->blockState == TBLOCK_INPROGRESS ||
    4841             :            s->blockState == TBLOCK_IMPLICIT_INPROGRESS ||
    4842             :            s->blockState == TBLOCK_PARALLEL_INPROGRESS ||
    4843             :            s->blockState == TBLOCK_STARTED);
    4844        8132 : }
    4845             : 
    4846             : /*
    4847             :  *  AbortOutOfAnyTransaction
    4848             :  *
    4849             :  *  This routine is provided for error recovery purposes.  It aborts any
    4850             :  *  active transaction or transaction block, leaving the system in a known
    4851             :  *  idle state.
    4852             :  */
    4853             : void
    4854       30820 : AbortOutOfAnyTransaction(void)
    4855             : {
    4856       30820 :     TransactionState s = CurrentTransactionState;
    4857             : 
    4858             :     /* Ensure we're not running in a doomed memory context */
    4859       30820 :     AtAbort_Memory();
    4860             : 
    4861             :     /*
    4862             :      * Get out of any transaction or nested transaction
    4863             :      */
    4864             :     do
    4865             :     {
    4866       30820 :         switch (s->blockState)
    4867             :         {
    4868       29934 :             case TBLOCK_DEFAULT:
    4869       29934 :                 if (s->state == TRANS_DEFAULT)
    4870             :                 {
    4871             :                     /* Not in a transaction, do nothing */
    4872             :                 }
    4873             :                 else
    4874             :                 {
    4875             :                     /*
    4876             :                      * We can get here after an error during transaction start
    4877             :                      * (state will be TRANS_START).  Need to clean up the
    4878             :                      * incompletely started transaction.  First, adjust the
    4879             :                      * low-level state to suppress warning message from
    4880             :                      * AbortTransaction.
    4881             :                      */
    4882           0 :                     if (s->state == TRANS_START)
    4883           0 :                         s->state = TRANS_INPROGRESS;
    4884           0 :                     AbortTransaction();
    4885           0 :                     CleanupTransaction();
    4886             :                 }
    4887       29934 :                 break;
    4888         864 :             case TBLOCK_STARTED:
    4889             :             case TBLOCK_BEGIN:
    4890             :             case TBLOCK_INPROGRESS:
    4891             :             case TBLOCK_IMPLICIT_INPROGRESS:
    4892             :             case TBLOCK_PARALLEL_INPROGRESS:
    4893             :             case TBLOCK_END:
    4894             :             case TBLOCK_ABORT_PENDING:
    4895             :             case TBLOCK_PREPARE:
    4896             :                 /* In a transaction, so clean up */
    4897         864 :                 AbortTransaction();
    4898         864 :                 CleanupTransaction();
    4899         864 :                 s->blockState = TBLOCK_DEFAULT;
    4900         864 :                 break;
    4901          22 :             case TBLOCK_ABORT:
    4902             :             case TBLOCK_ABORT_END:
    4903             : 
    4904             :                 /*
    4905             :                  * AbortTransaction is already done, still need Cleanup.
    4906             :                  * However, if we failed partway through running ROLLBACK,
    4907             :                  * there will be an active portal running that command, which
    4908             :                  * we need to shut down before doing CleanupTransaction.
    4909             :                  */
    4910          22 :                 AtAbort_Portals();
    4911          22 :                 CleanupTransaction();
    4912          22 :                 s->blockState = TBLOCK_DEFAULT;
    4913          22 :                 break;
    4914             : 
    4915             :                 /*
    4916             :                  * In a subtransaction, so clean it up and abort parent too
    4917             :                  */
    4918           0 :             case TBLOCK_SUBBEGIN:
    4919             :             case TBLOCK_SUBINPROGRESS:
    4920             :             case TBLOCK_SUBRELEASE:
    4921             :             case TBLOCK_SUBCOMMIT:
    4922             :             case TBLOCK_SUBABORT_PENDING:
    4923             :             case TBLOCK_SUBRESTART:
    4924           0 :                 AbortSubTransaction();
    4925           0 :                 CleanupSubTransaction();
    4926           0 :                 s = CurrentTransactionState;    /* changed by pop */
    4927           0 :                 break;
    4928             : 
    4929           0 :             case TBLOCK_SUBABORT:
    4930             :             case TBLOCK_SUBABORT_END:
    4931             :             case TBLOCK_SUBABORT_RESTART:
    4932             :                 /* As above, but AbortSubTransaction already done */
    4933           0 :                 if (s->curTransactionOwner)
    4934             :                 {
    4935             :                     /* As in TBLOCK_ABORT, might have a live portal to zap */
    4936           0 :                     AtSubAbort_Portals(s->subTransactionId,
    4937           0 :                                        s->parent->subTransactionId,
    4938             :                                        s->curTransactionOwner,
    4939           0 :                                        s->parent->curTransactionOwner);
    4940             :                 }
    4941           0 :                 CleanupSubTransaction();
    4942           0 :                 s = CurrentTransactionState;    /* changed by pop */
    4943           0 :                 break;
    4944             :         }
    4945       30820 :     } while (s->blockState != TBLOCK_DEFAULT);
    4946             : 
    4947             :     /* Should be out of all subxacts now */
    4948             :     Assert(s->parent == NULL);
    4949             : 
    4950             :     /*
    4951             :      * Revert to TopMemoryContext, to ensure we exit in a well-defined state
    4952             :      * whether there were any transactions to close or not.  (Callers that
    4953             :      * don't intend to exit soon should switch to some other context to avoid
    4954             :      * long-term memory leaks.)
    4955             :      */
    4956       30820 :     MemoryContextSwitchTo(TopMemoryContext);
    4957       30820 : }
    4958             : 
    4959             : /*
    4960             :  * IsTransactionBlock --- are we within a transaction block?
    4961             :  */
    4962             : bool
    4963      445178 : IsTransactionBlock(void)
    4964             : {
    4965      445178 :     TransactionState s = CurrentTransactionState;
    4966             : 
    4967      445178 :     if (s->blockState == TBLOCK_DEFAULT || s->blockState == TBLOCK_STARTED)
    4968      338600 :         return false;
    4969             : 
    4970      106578 :     return true;
    4971             : }
    4972             : 
    4973             : /*
    4974             :  * IsTransactionOrTransactionBlock --- are we within either a transaction
    4975             :  * or a transaction block?  (The backend is only really "idle" when this
    4976             :  * returns false.)
    4977             :  *
    4978             :  * This should match up with IsTransactionBlock and IsTransactionState.
    4979             :  */
    4980             : bool
    4981      667684 : IsTransactionOrTransactionBlock(void)
    4982             : {
    4983      667684 :     TransactionState s = CurrentTransactionState;
    4984             : 
    4985      667684 :     if (s->blockState == TBLOCK_DEFAULT)
    4986      522808 :         return false;
    4987             : 
    4988      144876 :     return true;
    4989             : }
    4990             : 
    4991             : /*
    4992             :  * TransactionBlockStatusCode - return status code to send in ReadyForQuery
    4993             :  */
    4994             : char
    4995      596434 : TransactionBlockStatusCode(void)
    4996             : {
    4997      596434 :     TransactionState s = CurrentTransactionState;
    4998             : 
    4999      596434 :     switch (s->blockState)
    5000             :     {
    5001      453306 :         case TBLOCK_DEFAULT:
    5002             :         case TBLOCK_STARTED:
    5003      453306 :             return 'I';         /* idle --- not in transaction */
    5004      141362 :         case TBLOCK_BEGIN:
    5005             :         case TBLOCK_SUBBEGIN:
    5006             :         case TBLOCK_INPROGRESS:
    5007             :         case TBLOCK_IMPLICIT_INPROGRESS:
    5008             :         case TBLOCK_PARALLEL_INPROGRESS:
    5009             :         case TBLOCK_SUBINPROGRESS:
    5010             :         case TBLOCK_END:
    5011             :         case TBLOCK_SUBRELEASE:
    5012             :         case TBLOCK_SUBCOMMIT:
    5013             :         case TBLOCK_PREPARE:
    5014      141362 :             return 'T';         /* in transaction */
    5015        1766 :         case TBLOCK_ABORT:
    5016             :         case TBLOCK_SUBABORT:
    5017             :         case TBLOCK_ABORT_END:
    5018             :         case TBLOCK_SUBABORT_END:
    5019             :         case TBLOCK_ABORT_PENDING:
    5020             :         case TBLOCK_SUBABORT_PENDING:
    5021             :         case TBLOCK_SUBRESTART:
    5022             :         case TBLOCK_SUBABORT_RESTART:
    5023        1766 :             return 'E';         /* in failed transaction */
    5024             :     }
    5025             : 
    5026             :     /* should never get here */
    5027           0 :     elog(FATAL, "invalid transaction block state: %s",
    5028             :          BlockStateAsString(s->blockState));
    5029             :     return 0;                   /* keep compiler quiet */
    5030             : }
    5031             : 
    5032             : /*
    5033             :  * IsSubTransaction
    5034             :  */
    5035             : bool
    5036     1171012 : IsSubTransaction(void)
    5037             : {
    5038     1171012 :     TransactionState s = CurrentTransactionState;
    5039             : 
    5040     1171012 :     if (s->nestingLevel >= 2)
    5041         486 :         return true;
    5042             : 
    5043     1170526 :     return false;
    5044             : }
    5045             : 
    5046             : /*
    5047             :  * StartSubTransaction
    5048             :  *
    5049             :  * If you're wondering why this is separate from PushTransaction: it's because
    5050             :  * we can't conveniently do this stuff right inside DefineSavepoint.  The
    5051             :  * SAVEPOINT utility command will be executed inside a Portal, and if we
    5052             :  * muck with CurrentMemoryContext or CurrentResourceOwner then exit from
    5053             :  * the Portal will undo those settings.  So we make DefineSavepoint just
    5054             :  * push a dummy transaction block, and when control returns to the main
    5055             :  * idle loop, CommitTransactionCommand will be called, and we'll come here
    5056             :  * to finish starting the subtransaction.
    5057             :  */
    5058             : static void
    5059       20040 : StartSubTransaction(void)
    5060             : {
    5061       20040 :     TransactionState s = CurrentTransactionState;
    5062             : 
    5063       20040 :     if (s->state != TRANS_DEFAULT)
    5064           0 :         elog(WARNING, "StartSubTransaction while in %s state",
    5065             :              TransStateAsString(s->state));
    5066             : 
    5067       20040 :     s->state = TRANS_START;
    5068             : 
    5069             :     /*
    5070             :      * Initialize subsystems for new subtransaction
    5071             :      *
    5072             :      * must initialize resource-management stuff first
    5073             :      */
    5074       20040 :     AtSubStart_Memory();
    5075       20040 :     AtSubStart_ResourceOwner();
    5076       20040 :     AfterTriggerBeginSubXact();
    5077             : 
    5078       20040 :     s->state = TRANS_INPROGRESS;
    5079             : 
    5080             :     /*
    5081             :      * Call start-of-subxact callbacks
    5082             :      */
    5083       20040 :     CallSubXactCallbacks(SUBXACT_EVENT_START_SUB, s->subTransactionId,
    5084       20040 :                          s->parent->subTransactionId);
    5085             : 
    5086       20040 :     ShowTransactionState("StartSubTransaction");
    5087       20040 : }
    5088             : 
    5089             : /*
    5090             :  * CommitSubTransaction
    5091             :  *
    5092             :  *  The caller has to make sure to always reassign CurrentTransactionState
    5093             :  *  if it has a local pointer to it after calling this function.
    5094             :  */
    5095             : static void
    5096       10772 : CommitSubTransaction(void)
    5097             : {
    5098       10772 :     TransactionState s = CurrentTransactionState;
    5099             : 
    5100       10772 :     ShowTransactionState("CommitSubTransaction");
    5101             : 
    5102       10772 :     if (s->state != TRANS_INPROGRESS)
    5103           0 :         elog(WARNING, "CommitSubTransaction while in %s state",
    5104             :              TransStateAsString(s->state));
    5105             : 
    5106             :     /* Pre-commit processing goes here */
    5107             : 
    5108       10772 :     CallSubXactCallbacks(SUBXACT_EVENT_PRE_COMMIT_SUB, s->subTransactionId,
    5109       10772 :                          s->parent->subTransactionId);
    5110             : 
    5111             :     /*
    5112             :      * If this subxact has started any unfinished parallel operation, clean up
    5113             :      * its workers and exit parallel mode.  Warn about leaked resources.
    5114             :      */
    5115       10772 :     AtEOSubXact_Parallel(true, s->subTransactionId);
    5116       10772 :     if (s->parallelModeLevel != 0)
    5117             :     {
    5118           0 :         elog(WARNING, "parallelModeLevel is %d not 0 at end of subtransaction",
    5119             :              s->parallelModeLevel);
    5120           0 :         s->parallelModeLevel = 0;
    5121             :     }
    5122             : 
    5123             :     /* Do the actual "commit", such as it is */
    5124       10772 :     s->state = TRANS_COMMIT;
    5125             : 
    5126             :     /* Must CCI to ensure commands of subtransaction are seen as done */
    5127       10772 :     CommandCounterIncrement();
    5128             : 
    5129             :     /*
    5130             :      * Prior to 8.4 we marked subcommit in clog at this point.  We now only
    5131             :      * perform that step, if required, as part of the atomic update of the
    5132             :      * whole transaction tree at top level commit or abort.
    5133             :      */
    5134             : 
    5135             :     /* Post-commit cleanup */
    5136       10772 :     if (FullTransactionIdIsValid(s->fullTransactionId))
    5137        7474 :         AtSubCommit_childXids();
    5138       10772 :     AfterTriggerEndSubXact(true);
    5139       10772 :     AtSubCommit_Portals(s->subTransactionId,
    5140       10772 :                         s->parent->subTransactionId,
    5141       10772 :                         s->parent->nestingLevel,
    5142       10772 :                         s->parent->curTransactionOwner);
    5143       10772 :     AtEOSubXact_LargeObject(true, s->subTransactionId,
    5144       10772 :                             s->parent->subTransactionId);
    5145       10772 :     AtSubCommit_Notify();
    5146             : 
    5147       10772 :     CallSubXactCallbacks(SUBXACT_EVENT_COMMIT_SUB, s->subTransactionId,
    5148       10772 :                          s->parent->subTransactionId);
    5149             : 
    5150       10772 :     ResourceOwnerRelease(s->curTransactionOwner,
    5151             :                          RESOURCE_RELEASE_BEFORE_LOCKS,
    5152             :                          true, false);
    5153       10772 :     AtEOSubXact_RelationCache(true, s->subTransactionId,
    5154       10772 :                               s->parent->subTransactionId);
    5155       10772 :     AtEOSubXact_TypeCache();
    5156       10772 :     AtEOSubXact_Inval(true);
    5157       10772 :     AtSubCommit_smgr();
    5158             : 
    5159             :     /*
    5160             :      * The only lock we actually release here is the subtransaction XID lock.
    5161             :      */
    5162       10772 :     CurrentResourceOwner = s->curTransactionOwner;
    5163       10772 :     if (FullTransactionIdIsValid(s->fullTransactionId))
    5164        7474 :         XactLockTableDelete(XidFromFullTransactionId(s->fullTransactionId));
    5165             : 
    5166             :     /*
    5167             :      * Other locks should get transferred to their parent resource owner.
    5168             :      */
    5169       10772 :     ResourceOwnerRelease(s->curTransactionOwner,
    5170             :                          RESOURCE_RELEASE_LOCKS,
    5171             :                          true, false);
    5172       10772 :     ResourceOwnerRelease(s->curTransactionOwner,
    5173             :                          RESOURCE_RELEASE_AFTER_LOCKS,
    5174             :                          true, false);
    5175             : 
    5176       10772 :     AtEOXact_GUC(true, s->gucNestLevel);
    5177       10772 :     AtEOSubXact_SPI(true, s->subTransactionId);
    5178       10772 :     AtEOSubXact_on_commit_actions(true, s->subTransactionId,
    5179       10772 :                                   s->parent->subTransactionId);
    5180       10772 :     AtEOSubXact_Namespace(true, s->subTransactionId,
    5181       10772 :                           s->parent->subTransactionId);
    5182       10772 :     AtEOSubXact_Files(true, s->subTransactionId,
    5183       10772 :                       s->parent->subTransactionId);
    5184       10772 :     AtEOSubXact_HashTables(true, s->nestingLevel);
    5185       10772 :     AtEOSubXact_PgStat(true, s->nestingLevel);
    5186       10772 :     AtSubCommit_Snapshot(s->nestingLevel);
    5187             : 
    5188             :     /*
    5189             :      * We need to restore the upper transaction's read-only state, in case the
    5190             :      * upper is read-write while the child is read-only; GUC will incorrectly
    5191             :      * think it should leave the child state in place.
    5192             :      */
    5193       10772 :     XactReadOnly = s->prevXactReadOnly;
    5194             : 
    5195       10772 :     CurrentResourceOwner = s->parent->curTransactionOwner;
    5196       10772 :     CurTransactionResourceOwner = s->parent->curTransactionOwner;
    5197       10772 :     ResourceOwnerDelete(s->curTransactionOwner);
    5198       10772 :     s->curTransactionOwner = NULL;
    5199             : 
    5200       10772 :     AtSubCommit_Memory();
    5201             : 
    5202       10772 :     s->state = TRANS_DEFAULT;
    5203             : 
    5204       10772 :     PopTransaction();
    5205       10772 : }
    5206             : 
    5207             : /*
    5208             :  * AbortSubTransaction
    5209             :  */
    5210             : static void
    5211        9268 : AbortSubTransaction(void)
    5212             : {
    5213        9268 :     TransactionState s = CurrentTransactionState;
    5214             : 
    5215             :     /* Prevent cancel/die interrupt while cleaning up */
    5216        9268 :     HOLD_INTERRUPTS();
    5217             : 
    5218             :     /* Make sure we have a valid memory context and resource owner */
    5219        9268 :     AtSubAbort_Memory();
    5220        9268 :     AtSubAbort_ResourceOwner();
    5221             : 
    5222             :     /*
    5223             :      * Release any LW locks we might be holding as quickly as possible.
    5224             :      * (Regular locks, however, must be held till we finish aborting.)
    5225             :      * Releasing LW locks is critical since we might try to grab them again
    5226             :      * while cleaning up!
    5227             :      *
    5228             :      * FIXME This may be incorrect --- Are there some locks we should keep?
    5229             :      * Buffer locks, for example?  I don't think so but I'm not sure.
    5230             :      */
    5231        9268 :     LWLockReleaseAll();
    5232             : 
    5233        9268 :     pgstat_report_wait_end();
    5234        9268 :     pgstat_progress_end_command();
    5235        9268 :     UnlockBuffers();
    5236             : 
    5237             :     /* Reset WAL record construction state */
    5238        9268 :     XLogResetInsertion();
    5239             : 
    5240             :     /* Cancel condition variable sleep */
    5241        9268 :     ConditionVariableCancelSleep();
    5242             : 
    5243             :     /*
    5244             :      * Also clean up any open wait for lock, since the lock manager will choke
    5245             :      * if we try to wait for another lock before doing this.
    5246             :      */
    5247        9268 :     LockErrorCleanup();
    5248             : 
    5249             :     /*
    5250             :      * If any timeout events are still active, make sure the timeout interrupt
    5251             :      * is scheduled.  This covers possible loss of a timeout interrupt due to
    5252             :      * longjmp'ing out of the SIGINT handler (see notes in handle_sig_alarm).
    5253             :      * We delay this till after LockErrorCleanup so that we don't uselessly
    5254             :      * reschedule lock or deadlock check timeouts.
    5255             :      */
    5256        9268 :     reschedule_timeouts();
    5257             : 
    5258             :     /*
    5259             :      * Re-enable signals, in case we got here by longjmp'ing out of a signal
    5260             :      * handler.  We do this fairly early in the sequence so that the timeout
    5261             :      * infrastructure will be functional if needed while aborting.
    5262             :      */
    5263        9268 :     sigprocmask(SIG_SETMASK, &UnBlockSig, NULL);
    5264             : 
    5265             :     /*
    5266             :      * check the current transaction state
    5267             :      */
    5268        9268 :     ShowTransactionState("AbortSubTransaction");
    5269             : 
    5270        9268 :     if (s->state != TRANS_INPROGRESS)
    5271           0 :         elog(WARNING, "AbortSubTransaction while in %s state",
    5272             :              TransStateAsString(s->state));
    5273             : 
    5274        9268 :     s->state = TRANS_ABORT;
    5275             : 
    5276             :     /*
    5277             :      * Reset user ID which might have been changed transiently.  (See notes in
    5278             :      * AbortTransaction.)
    5279             :      */
    5280        9268 :     SetUserIdAndSecContext(s->prevUser, s->prevSecContext);
    5281             : 
    5282             :     /* Forget about any active REINDEX. */
    5283        9268 :     ResetReindexState(s->nestingLevel);
    5284             : 
    5285             :     /* Reset logical streaming state. */
    5286        9268 :     ResetLogicalStreamingState();
    5287             : 
    5288             :     /*
    5289             :      * No need for SnapBuildResetExportedSnapshotState() here, snapshot
    5290             :      * exports are not supported in subtransactions.
    5291             :      */
    5292             : 
    5293             :     /*
    5294             :      * If this subxact has started any unfinished parallel operation, clean up
    5295             :      * its workers and exit parallel mode.  Don't warn about leaked resources.
    5296             :      */
    5297        9268 :     AtEOSubXact_Parallel(false, s->subTransactionId);
    5298        9268 :     s->parallelModeLevel = 0;
    5299             : 
    5300             :     /*
    5301             :      * We can skip all this stuff if the subxact failed before creating a
    5302             :      * ResourceOwner...
    5303             :      */
    5304        9268 :     if (s->curTransactionOwner)
    5305             :     {
    5306        9268 :         AfterTriggerEndSubXact(false);
    5307        9268 :         AtSubAbort_Portals(s->subTransactionId,
    5308        9268 :                            s->parent->subTransactionId,
    5309             :                            s->curTransactionOwner,
    5310        9268 :                            s->parent->curTransactionOwner);
    5311        9268 :         AtEOSubXact_LargeObject(false, s->subTransactionId,
    5312        9268 :                                 s->parent->subTransactionId);
    5313        9268 :         AtSubAbort_Notify();
    5314             : 
    5315             :         /* Advertise the fact that we aborted in pg_xact. */
    5316        9268 :         (void) RecordTransactionAbort(true);
    5317             : 
    5318             :         /* Post-abort cleanup */
    5319        9268 :         if (FullTransactionIdIsValid(s->fullTransactionId))
    5320        1298 :             AtSubAbort_childXids();
    5321             : 
    5322        9268 :         CallSubXactCallbacks(SUBXACT_EVENT_ABORT_SUB, s->subTransactionId,
    5323        9268 :                              s->parent->subTransactionId);
    5324             : 
    5325        9268 :         ResourceOwnerRelease(s->curTransactionOwner,
    5326             :                              RESOURCE_RELEASE_BEFORE_LOCKS,
    5327             :                              false, false);
    5328             : 
    5329        9268 :         AtEOSubXact_RelationCache(false, s->subTransactionId,
    5330        9268 :                                   s->parent->subTransactionId);
    5331        9268 :         AtEOSubXact_TypeCache();
    5332        9268 :         AtEOSubXact_Inval(false);
    5333        9268 :         ResourceOwnerRelease(s->curTransactionOwner,
    5334             :                              RESOURCE_RELEASE_LOCKS,
    5335             :                              false, false);
    5336        9268 :         ResourceOwnerRelease(s->curTransactionOwner,
    5337             :                              RESOURCE_RELEASE_AFTER_LOCKS,
    5338             :                              false, false);
    5339        9268 :         AtSubAbort_smgr();
    5340             : 
    5341        9268 :         AtEOXact_GUC(false, s->gucNestLevel);
    5342        9268 :         AtEOSubXact_SPI(false, s->subTransactionId);
    5343        9268 :         AtEOSubXact_on_commit_actions(false, s->subTransactionId,
    5344        9268 :                                       s->parent->subTransactionId);
    5345        9268 :         AtEOSubXact_Namespace(false, s->subTransactionId,
    5346        9268 :                               s->parent->subTransactionId);
    5347        9268 :         AtEOSubXact_Files(false, s->subTransactionId,
    5348        9268 :                           s->parent->subTransactionId);
    5349        9268 :         AtEOSubXact_HashTables(false, s->nestingLevel);
    5350        9268 :         AtEOSubXact_PgStat(false, s->nestingLevel);
    5351        9268 :         AtSubAbort_Snapshot(s->nestingLevel);
    5352             :     }
    5353             : 
    5354             :     /*
    5355             :      * Restore the upper transaction's read-only state, too.  This should be
    5356             :      * redundant with GUC's cleanup but we may as well do it for consistency
    5357             :      * with the commit case.
    5358             :      */
    5359        9268 :     XactReadOnly = s->prevXactReadOnly;
    5360             : 
    5361        9268 :     RESUME_INTERRUPTS();
    5362        9268 : }
    5363             : 
    5364             : /*
    5365             :  * CleanupSubTransaction
    5366             :  *
    5367             :  *  The caller has to make sure to always reassign CurrentTransactionState
    5368             :  *  if it has a local pointer to it after calling this function.
    5369             :  */
    5370             : static void
    5371        9268 : CleanupSubTransaction(void)
    5372             : {
    5373        9268 :     TransactionState s = CurrentTransactionState;
    5374             : 
    5375        9268 :     ShowTransactionState("CleanupSubTransaction");
    5376             : 
    5377        9268 :     if (s->state != TRANS_ABORT)
    5378           0 :         elog(WARNING, "CleanupSubTransaction while in %s state",
    5379             :              TransStateAsString(s->state));
    5380             : 
    5381        9268 :     AtSubCleanup_Portals(s->subTransactionId);
    5382             : 
    5383        9268 :     CurrentResourceOwner = s->parent->curTransactionOwner;
    5384        9268 :     CurTransactionResourceOwner = s->parent->curTransactionOwner;
    5385        9268 :     if (s->curTransactionOwner)
    5386        9268 :         ResourceOwnerDelete(s->curTransactionOwner);
    5387        9268 :     s->curTransactionOwner = NULL;
    5388             : 
    5389        9268 :     AtSubCleanup_Memory();
    5390             : 
    5391        9268 :     s->state = TRANS_DEFAULT;
    5392             : 
    5393        9268 :     PopTransaction();
    5394        9268 : }
    5395             : 
    5396             : /*
    5397             :  * PushTransaction
    5398             :  *      Create transaction state stack entry for a subtransaction
    5399             :  *
    5400             :  *  The caller has to make sure to always reassign CurrentTransactionState
    5401             :  *  if it has a local pointer to it after calling this function.
    5402             :  */
    5403             : static void
    5404       20040 : PushTransaction(void)
    5405             : {
    5406       20040 :     TransactionState p = CurrentTransactionState;
    5407             :     TransactionState s;
    5408             : 
    5409             :     /*
    5410             :      * We keep subtransaction state nodes in TopTransactionContext.
    5411             :      */
    5412             :     s = (TransactionState)
    5413       20040 :         MemoryContextAllocZero(TopTransactionContext,
    5414             :                                sizeof(TransactionStateData));
    5415             : 
    5416             :     /*
    5417             :      * Assign a subtransaction ID, watching out for counter wraparound.
    5418             :      */
    5419       20040 :     currentSubTransactionId += 1;
    5420       20040 :     if (currentSubTransactionId == InvalidSubTransactionId)
    5421             :     {
    5422           0 :         currentSubTransactionId -= 1;
    5423           0 :         pfree(s);
    5424           0 :         ereport(ERROR,
    5425             :                 (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
    5426             :                  errmsg("cannot have more than 2^32-1 subtransactions in a transaction")));
    5427             :     }
    5428             : 
    5429             :     /*
    5430             :      * We can now stack a minimally valid subtransaction without fear of
    5431             :      * failure.
    5432             :      */
    5433       20040 :     s->fullTransactionId = InvalidFullTransactionId; /* until assigned */
    5434       20040 :     s->subTransactionId = currentSubTransactionId;
    5435       20040 :     s->parent = p;
    5436       20040 :     s->nestingLevel = p->nestingLevel + 1;
    5437       20040 :     s->gucNestLevel = NewGUCNestLevel();
    5438       20040 :     s->savepointLevel = p->savepointLevel;
    5439       20040 :     s->state = TRANS_DEFAULT;
    5440       20040 :     s->blockState = TBLOCK_SUBBEGIN;
    5441       20040 :     GetUserIdAndSecContext(&s->prevUser, &s->prevSecContext);
    5442       20040 :     s->prevXactReadOnly = XactReadOnly;
    5443       20040 :     s->startedInRecovery = p->startedInRecovery;
    5444       20040 :     s->parallelModeLevel = 0;
    5445       20040 :     s->parallelChildXact = (p->parallelModeLevel != 0 || p->parallelChildXact);
    5446       20040 :     s->topXidLogged = false;
    5447             : 
    5448       20040 :     CurrentTransactionState = s;
    5449             : 
    5450             :     /*
    5451             :      * AbortSubTransaction and CleanupSubTransaction have to be able to cope
    5452             :      * with the subtransaction from here on out; in particular they should not
    5453             :      * assume that it necessarily has a transaction context, resource owner,
    5454             :      * or XID.
    5455             :      */
    5456       20040 : }
    5457             : 
    5458             : /*
    5459             :  * PopTransaction
    5460             :  *      Pop back to parent transaction state
    5461             :  *
    5462             :  *  The caller has to make sure to always reassign CurrentTransactionState
    5463             :  *  if it has a local pointer to it after calling this function.
    5464             :  */
    5465             : static void
    5466       20040 : PopTransaction(void)
    5467             : {
    5468       20040 :     TransactionState s = CurrentTransactionState;
    5469             : 
    5470       20040 :     if (s->state != TRANS_DEFAULT)
    5471           0 :         elog(WARNING, "PopTransaction while in %s state",
    5472             :              TransStateAsString(s->state));
    5473             : 
    5474       20040 :     if (s->parent == NULL)
    5475           0 :         elog(FATAL, "PopTransaction with no parent");
    5476             : 
    5477       20040 :     CurrentTransactionState = s->parent;
    5478             : 
    5479             :     /* Let's just make sure CurTransactionContext is good */
    5480       20040 :     CurTransactionContext = s->parent->curTransactionContext;
    5481       20040 :     MemoryContextSwitchTo(CurTransactionContext);
    5482             : 
    5483             :     /* Ditto for ResourceOwner links */
    5484       20040 :     CurTransactionResourceOwner = s->parent->curTransactionOwner;
    5485       20040 :     CurrentResourceOwner = s->parent->curTransactionOwner;
    5486             : 
    5487             :     /* Free the old child structure */
    5488       20040 :     if (s->name)
    5489        3866 :         pfree(s->name);
    5490       20040 :     pfree(s);
    5491       20040 : }
    5492             : 
    5493             : /*
    5494             :  * EstimateTransactionStateSpace
    5495             :  *      Estimate the amount of space that will be needed by
    5496             :  *      SerializeTransactionState.  It would be OK to overestimate slightly,
    5497             :  *      but it's simple for us to work out the precise value, so we do.
    5498             :  */
    5499             : Size
    5500         892 : EstimateTransactionStateSpace(void)
    5501             : {
    5502             :     TransactionState s;
    5503         892 :     Size        nxids = 0;
    5504         892 :     Size        size = SerializedTransactionStateHeaderSize;
    5505             : 
    5506        4322 :     for (s = CurrentTransactionState; s != NULL; s = s->parent)
    5507             :     {
    5508        3430 :         if (FullTransactionIdIsValid(s->fullTransactionId))
    5509        1976 :             nxids = add_size(nxids, 1);
    5510        3430 :         nxids = add_size(nxids, s->nChildXids);
    5511             :     }
    5512             : 
    5513         892 :     return add_size(size, mul_size(sizeof(TransactionId), nxids));
    5514             : }
    5515             : 
    5516             : /*
    5517             :  * SerializeTransactionState
    5518             :  *      Write out relevant details of our transaction state that will be
    5519             :  *      needed by a parallel worker.
    5520             :  *
    5521             :  * We need to save and restore XactDeferrable, XactIsoLevel, and the XIDs
    5522             :  * associated with this transaction.  These are serialized into a
    5523             :  * caller-supplied buffer big enough to hold the number of bytes reported by
    5524             :  * EstimateTransactionStateSpace().  We emit the XIDs in sorted order for the
    5525             :  * convenience of the receiving process.
    5526             :  */
    5527             : void
    5528         892 : SerializeTransactionState(Size maxsize, char *start_address)
    5529             : {
    5530             :     TransactionState s;
    5531         892 :     Size        nxids = 0;
    5532         892 :     Size        i = 0;
    5533             :     TransactionId *workspace;
    5534             :     SerializedTransactionState *result;
    5535             : 
    5536         892 :     result = (SerializedTransactionState *) start_address;
    5537             : 
    5538         892 :     result->xactIsoLevel = XactIsoLevel;
    5539         892 :     result->xactDeferrable = XactDeferrable;
    5540         892 :     result->topFullTransactionId = XactTopFullTransactionId;
    5541         892 :     result->currentFullTransactionId =
    5542         892 :         CurrentTransactionState->fullTransactionId;
    5543         892 :     result->currentCommandId = currentCommandId;
    5544             : 
    5545             :     /*
    5546             :      * If we're running in a parallel worker and launching a parallel worker
    5547             :      * of our own, we can just pass along the information that was passed to
    5548             :      * us.
    5549             :      */
    5550         892 :     if (nParallelCurrentXids > 0)
    5551             :     {
    5552           0 :         result->nParallelCurrentXids = nParallelCurrentXids;
    5553           0 :         memcpy(&result->parallelCurrentXids[0], ParallelCurrentXids,
    5554             :                nParallelCurrentXids * sizeof(TransactionId));
    5555           0 :         return;
    5556             :     }
    5557             : 
    5558             :     /*
    5559             :      * OK, we need to generate a sorted list of XIDs that our workers should
    5560             :      * view as current.  First, figure out how many there are.
    5561             :      */
    5562        4322 :     for (s = CurrentTransactionState; s != NULL; s = s->parent)
    5563             :     {
    5564        3430 :         if (FullTransactionIdIsValid(s->fullTransactionId))
    5565        1976 :             nxids = add_size(nxids, 1);
    5566        3430 :         nxids = add_size(nxids, s->nChildXids);
    5567             :     }
    5568             :     Assert(SerializedTransactionStateHeaderSize + nxids * sizeof(TransactionId)
    5569             :            <= maxsize);
    5570             : 
    5571             :     /* Copy them to our scratch space. */
    5572         892 :     workspace = palloc(nxids * sizeof(TransactionId));
    5573        4322 :     for (s = CurrentTransactionState; s != NULL; s = s->parent)
    5574             :     {
    5575        3430 :         if (FullTransactionIdIsValid(s->fullTransactionId))
    5576        1976 :             workspace[i++] = XidFromFullTransactionId(s->fullTransactionId);
    5577        3430 :         if (s->nChildXids > 0)
    5578           0 :             memcpy(&workspace[i], s->childXids,
    5579           0 :                    s->nChildXids * sizeof(TransactionId));
    5580        3430 :         i += s->nChildXids;
    5581             :     }
    5582             :     Assert(i == nxids);
    5583             : 
    5584             :     /* Sort them. */
    5585         892 :     qsort(workspace, nxids, sizeof(TransactionId), xidComparator);
    5586             : 
    5587             :     /* Copy data into output area. */
    5588         892 :     result->nParallelCurrentXids = nxids;
    5589         892 :     memcpy(&result->parallelCurrentXids[0], workspace,
    5590             :            nxids * sizeof(TransactionId));
    5591             : }
    5592             : 
    5593             : /*
    5594             :  * StartParallelWorkerTransaction
    5595             :  *      Start a parallel worker transaction, restoring the relevant
    5596             :  *      transaction state serialized by SerializeTransactionState.
    5597             :  */
    5598             : void
    5599        2712 : StartParallelWorkerTransaction(char *tstatespace)
    5600             : {
    5601             :     SerializedTransactionState *tstate;
    5602             : 
    5603             :     Assert(CurrentTransactionState->blockState == TBLOCK_DEFAULT);
    5604        2712 :     StartTransaction();
    5605             : 
    5606        2712 :     tstate = (SerializedTransactionState *) tstatespace;
    5607        2712 :     XactIsoLevel = tstate->xactIsoLevel;
    5608        2712 :     XactDeferrable = tstate->xactDeferrable;
    5609        2712 :     XactTopFullTransactionId = tstate->topFullTransactionId;
    5610        2712 :     CurrentTransactionState->fullTransactionId =
    5611             :         tstate->currentFullTransactionId;
    5612        2712 :     currentCommandId = tstate->currentCommandId;
    5613        2712 :     nParallelCurrentXids = tstate->nParallelCurrentXids;
    5614        2712 :     ParallelCurrentXids = &tstate->parallelCurrentXids[0];
    5615             : 
    5616        2712 :     CurrentTransactionState->blockState = TBLOCK_PARALLEL_INPROGRESS;
    5617        2712 : }
    5618             : 
    5619             : /*
    5620             :  * EndParallelWorkerTransaction
    5621             :  *      End a parallel worker transaction.
    5622             :  */
    5623             : void
    5624        2700 : EndParallelWorkerTransaction(void)
    5625             : {
    5626             :     Assert(CurrentTransactionState->blockState == TBLOCK_PARALLEL_INPROGRESS);
    5627        2700 :     CommitTransaction();
    5628        2700 :     CurrentTransactionState->blockState = TBLOCK_DEFAULT;
    5629        2700 : }
    5630             : 
    5631             : /*
    5632             :  * ShowTransactionState
    5633             :  *      Debug support
    5634             :  */
    5635             : static void
    5636     1573020 : ShowTransactionState(const char *str)
    5637             : {
    5638             :     /* skip work if message will definitely not be printed */
    5639     1573020 :     if (message_level_is_interesting(DEBUG5))
    5640           0 :         ShowTransactionStateRec(str, CurrentTransactionState);
    5641     1573020 : }
    5642             : 
    5643             : /*
    5644             :  * ShowTransactionStateRec
    5645             :  *      Recursive subroutine for ShowTransactionState
    5646             :  */
    5647             : static void
    5648           0 : ShowTransactionStateRec(const char *str, TransactionState s)
    5649             : {
    5650             :     StringInfoData buf;
    5651             : 
    5652           0 :     if (s->parent)
    5653             :     {
    5654             :         /*
    5655             :          * Since this function recurses, it could be driven to stack overflow.
    5656             :          * This is just a debugging aid, so we can leave out some details
    5657             :          * instead of erroring out with check_stack_depth().
    5658             :          */
    5659           0 :         if (stack_is_too_deep())
    5660           0 :             ereport(DEBUG5,
    5661             :                     (errmsg_internal("%s(%d): parent omitted to avoid stack overflow",
    5662             :                                      str, s->nestingLevel)));
    5663             :         else
    5664           0 :             ShowTransactionStateRec(str, s->parent);
    5665             :     }
    5666             : 
    5667           0 :     initStringInfo(&buf);
    5668           0 :     if (s->nChildXids > 0)
    5669             :     {
    5670             :         int         i;
    5671             : 
    5672           0 :         appendStringInfo(&buf, ", children: %u", s->childXids[0]);
    5673           0 :         for (i = 1; i < s->nChildXids; i++)
    5674           0 :             appendStringInfo(&buf, " %u", s->childXids[i]);
    5675             :     }
    5676           0 :     ereport(DEBUG5,
    5677             :             (errmsg_internal("%s(%d) name: %s; blockState: %s; state: %s, xid/subid/cid: %u/%u/%u%s%s",
    5678             :                              str, s->nestingLevel,
    5679             :                              PointerIsValid(s->name) ? s->name : "unnamed",
    5680             :                              BlockStateAsString(s->blockState),
    5681             :                              TransStateAsString(s->state),
    5682             :                              (unsigned int) XidFromFullTransactionId(s->fullTransactionId),
    5683             :                              (unsigned int) s->subTransactionId,
    5684             :                              (unsigned int) currentCommandId,
    5685             :                              currentCommandIdUsed ? " (used)" : "",
    5686             :                              buf.data)));
    5687           0 :     pfree(buf.data);
    5688           0 : }
    5689             : 
    5690             : /*
    5691             :  * BlockStateAsString
    5692             :  *      Debug support
    5693             :  */
    5694             : static const char *
    5695           0 : BlockStateAsString(TBlockState blockState)
    5696             : {
    5697           0 :     switch (blockState)
    5698             :     {
    5699           0 :         case TBLOCK_DEFAULT:
    5700           0 :             return "DEFAULT";
    5701           0 :         case TBLOCK_STARTED:
    5702           0 :             return "STARTED";
    5703           0 :         case TBLOCK_BEGIN:
    5704           0 :             return "BEGIN";
    5705           0 :         case TBLOCK_INPROGRESS:
    5706           0 :             return "INPROGRESS";
    5707           0 :         case TBLOCK_IMPLICIT_INPROGRESS:
    5708           0 :             return "IMPLICIT_INPROGRESS";
    5709           0 :         case TBLOCK_PARALLEL_INPROGRESS:
    5710           0 :             return "PARALLEL_INPROGRESS";
    5711           0 :         case TBLOCK_END:
    5712           0 :             return "END";
    5713           0 :         case TBLOCK_ABORT:
    5714           0 :             return "ABORT";
    5715           0 :         case TBLOCK_ABORT_END:
    5716           0 :             return "ABORT_END";
    5717           0 :         case TBLOCK_ABORT_PENDING:
    5718           0 :             return "ABORT_PENDING";
    5719           0 :         case TBLOCK_PREPARE:
    5720           0 :             return "PREPARE";
    5721           0 :         case TBLOCK_SUBBEGIN:
    5722           0 :             return "SUBBEGIN";
    5723           0 :         case TBLOCK_SUBINPROGRESS:
    5724           0 :             return "SUBINPROGRESS";
    5725           0 :         case TBLOCK_SUBRELEASE:
    5726           0 :             return "SUBRELEASE";
    5727           0 :         case TBLOCK_SUBCOMMIT:
    5728           0 :             return "SUBCOMMIT";
    5729           0 :         case TBLOCK_SUBABORT:
    5730           0 :             return "SUBABORT";
    5731           0 :         case TBLOCK_SUBABORT_END:
    5732           0 :             return "SUBABORT_END";
    5733           0 :         case TBLOCK_SUBABORT_PENDING:
    5734           0 :             return "SUBABORT_PENDING";
    5735           0 :         case TBLOCK_SUBRESTART:
    5736           0 :             return "SUBRESTART";
    5737           0 :         case TBLOCK_SUBABORT_RESTART:
    5738           0 :             return "SUBABORT_RESTART";
    5739             :     }
    5740           0 :     return "UNRECOGNIZED";
    5741             : }
    5742             : 
    5743             : /*
    5744             :  * TransStateAsString
    5745             :  *      Debug support
    5746             :  */
    5747             : static const char *
    5748           0 : TransStateAsString(TransState state)
    5749             : {
    5750           0 :     switch (state)
    5751             :     {
    5752           0 :         case TRANS_DEFAULT:
    5753           0 :             return "DEFAULT";
    5754           0 :         case TRANS_START:
    5755           0 :             return "START";
    5756           0 :         case TRANS_INPROGRESS:
    5757           0 :             return "INPROGRESS";
    5758           0 :         case TRANS_COMMIT:
    5759           0 :             return "COMMIT";
    5760           0 :         case TRANS_ABORT:
    5761           0 :             return "ABORT";
    5762           0 :         case TRANS_PREPARE:
    5763           0 :             return "PREPARE";
    5764             :     }
    5765           0 :     return "UNRECOGNIZED";
    5766             : }
    5767             : 
    5768             : /*
    5769             :  * xactGetCommittedChildren
    5770             :  *
    5771             :  * Gets the list of committed children of the current transaction.  The return
    5772             :  * value is the number of child transactions.  *ptr is set to point to an
    5773             :  * array of TransactionIds.  The array is allocated in TopTransactionContext;
    5774             :  * the caller should *not* pfree() it (this is a change from pre-8.4 code!).
    5775             :  * If there are no subxacts, *ptr is set to NULL.
    5776             :  */
    5777             : int
    5778      746450 : xactGetCommittedChildren(TransactionId **ptr)
    5779             : {
    5780      746450 :     TransactionState s = CurrentTransactionState;
    5781             : 
    5782      746450 :     if (s->nChildXids == 0)
    5783      745144 :         *ptr = NULL;
    5784             :     else
    5785        1306 :         *ptr = s->childXids;
    5786             : 
    5787      746450 :     return s->nChildXids;
    5788             : }
    5789             : 
    5790             : /*
    5791             :  *  XLOG support routines
    5792             :  */
    5793             : 
    5794             : 
    5795             : /*
    5796             :  * Log the commit record for a plain or twophase transaction commit.
    5797             :  *
    5798             :  * A 2pc commit will be emitted when twophase_xid is valid, a plain one
    5799             :  * otherwise.
    5800             :  */
    5801             : XLogRecPtr
    5802      234076 : XactLogCommitRecord(TimestampTz commit_time,
    5803             :                     int nsubxacts, TransactionId *subxacts,
    5804             :                     int nrels, RelFileLocator *rels,
    5805             :                     int ndroppedstats, xl_xact_stats_item *droppedstats,
    5806             :                     int nmsgs, SharedInvalidationMessage *msgs,
    5807             :                     bool relcacheInval,
    5808             :                     int xactflags, TransactionId twophase_xid,
    5809             :                     const char *twophase_gid)
    5810             : {
    5811             :     xl_xact_commit xlrec;
    5812             :     xl_xact_xinfo xl_xinfo;
    5813             :     xl_xact_dbinfo xl_dbinfo;
    5814             :     xl_xact_subxacts xl_subxacts;
    5815             :     xl_xact_relfilelocators xl_relfilelocators;
    5816             :     xl_xact_stats_items xl_dropped_stats;
    5817             :     xl_xact_invals xl_invals;
    5818             :     xl_xact_twophase xl_twophase;
    5819             :     xl_xact_origin xl_origin;
    5820             :     uint8       info;
    5821             : 
    5822             :     Assert(CritSectionCount > 0);
    5823             : 
    5824      234076 :     xl_xinfo.xinfo = 0;
    5825             : 
    5826             :     /* decide between a plain and 2pc commit */
    5827      234076 :     if (!TransactionIdIsValid(twophase_xid))
    5828      233384 :         info = XLOG_XACT_COMMIT;
    5829             :     else
    5830         692 :         info = XLOG_XACT_COMMIT_PREPARED;
    5831             : 
    5832             :     /* First figure out and collect all the information needed */
    5833             : 
    5834      234076 :     xlrec.xact_time = commit_time;
    5835             : 
    5836      234076 :     if (relcacheInval)
    5837        6692 :         xl_xinfo.xinfo |= XACT_COMPLETION_UPDATE_RELCACHE_FILE;
    5838      234076 :     if (forceSyncCommit)
    5839         866 :         xl_xinfo.xinfo |= XACT_COMPLETION_FORCE_SYNC_COMMIT;
    5840      234076 :     if ((xactflags & XACT_FLAGS_ACQUIREDACCESSEXCLUSIVELOCK))
    5841       70432 :         xl_xinfo.xinfo |= XACT_XINFO_HAS_AE_LOCKS;
    5842             : 
    5843             :     /*
    5844             :      * Check if the caller would like to ask standbys for immediate feedback
    5845             :      * once this commit is applied.
    5846             :      */
    5847      234076 :     if (synchronous_commit >= SYNCHRONOUS_COMMIT_REMOTE_APPLY)
    5848           4 :         xl_xinfo.xinfo |= XACT_COMPLETION_APPLY_FEEDBACK;
    5849             : 
    5850             :     /*
    5851             :      * Relcache invalidations requires information about the current database
    5852             :      * and so does logical decoding.
    5853             :      */
    5854      234076 :     if (nmsgs > 0 || XLogLogicalInfoActive())
    5855             :     {
    5856      137762 :         xl_xinfo.xinfo |= XACT_XINFO_HAS_DBINFO;
    5857      137762 :         xl_dbinfo.dbId = MyDatabaseId;
    5858      137762 :         xl_dbinfo.tsId = MyDatabaseTableSpace;
    5859             :     }
    5860             : 
    5861      234076 :     if (nsubxacts > 0)
    5862             :     {
    5863        1116 :         xl_xinfo.xinfo |= XACT_XINFO_HAS_SUBXACTS;
    5864        1116 :         xl_subxacts.nsubxacts = nsubxacts;
    5865             :     }
    5866             : 
    5867      234076 :     if (nrels > 0)
    5868             :     {
    5869       17692 :         xl_xinfo.xinfo |= XACT_XINFO_HAS_RELFILELOCATORS;
    5870       17692 :         xl_relfilelocators.nrels = nrels;
    5871       17692 :         info |= XLR_SPECIAL_REL_UPDATE;
    5872             :     }
    5873             : 
    5874      234076 :     if (ndroppedstats > 0)
    5875             :     {
    5876       20362 :         xl_xinfo.xinfo |= XACT_XINFO_HAS_DROPPED_STATS;
    5877       20362 :         xl_dropped_stats.nitems = ndroppedstats;
    5878             :     }
    5879             : 
    5880      234076 :     if (nmsgs > 0)
    5881             :     {
    5882      136108 :         xl_xinfo.xinfo |= XACT_XINFO_HAS_INVALS;
    5883      136108 :         xl_invals.nmsgs = nmsgs;
    5884             :     }
    5885             : 
    5886      234076 :     if (TransactionIdIsValid(twophase_xid))
    5887             :     {
    5888         692 :         xl_xinfo.xinfo |= XACT_XINFO_HAS_TWOPHASE;
    5889         692 :         xl_twophase.xid = twophase_xid;
    5890             :         Assert(twophase_gid != NULL);
    5891             : 
    5892         692 :         if (XLogLogicalInfoActive())
    5893          74 :             xl_xinfo.xinfo |= XACT_XINFO_HAS_GID;
    5894             :     }
    5895             : 
    5896             :     /* dump transaction origin information */
    5897      234076 :     if (replorigin_session_origin != InvalidRepOriginId)
    5898             :     {
    5899        1976 :         xl_xinfo.xinfo |= XACT_XINFO_HAS_ORIGIN;
    5900             : 
    5901        1976 :         xl_origin.origin_lsn = replorigin_session_origin_lsn;
    5902        1976 :         xl_origin.origin_timestamp = replorigin_session_origin_timestamp;
    5903             :     }
    5904             : 
    5905      234076 :     if (xl_xinfo.xinfo != 0)
    5906      148654 :         info |= XLOG_XACT_HAS_INFO;
    5907             : 
    5908             :     /* Then include all the collected data into the commit record. */
    5909             : 
    5910      234076 :     XLogBeginInsert();
    5911             : 
    5912      234076 :     XLogRegisterData((char *) (&xlrec), sizeof(xl_xact_commit));
    5913             : 
    5914      234076 :     if (xl_xinfo.xinfo != 0)
    5915      148654 :         XLogRegisterData((char *) (&xl_xinfo.xinfo), sizeof(xl_xinfo.xinfo));
    5916             : 
    5917      234076 :     if (xl_xinfo.xinfo & XACT_XINFO_HAS_DBINFO)
    5918      137762 :         XLogRegisterData((char *) (&xl_dbinfo), sizeof(xl_dbinfo));
    5919             : 
    5920      234076 :     if (xl_xinfo.xinfo & XACT_XINFO_HAS_SUBXACTS)
    5921             :     {
    5922        1116 :         XLogRegisterData((char *) (&xl_subxacts),
    5923             :                          MinSizeOfXactSubxacts);
    5924        1116 :         XLogRegisterData((char *) subxacts,
    5925             :                          nsubxacts * sizeof(TransactionId));
    5926             :     }
    5927             : 
    5928      234076 :     if (xl_xinfo.xinfo & XACT_XINFO_HAS_RELFILELOCATORS)
    5929             :     {
    5930       17692 :         XLogRegisterData((char *) (&xl_relfilelocators),
    5931             :                          MinSizeOfXactRelfileLocators);
    5932       17692 :         XLogRegisterData((char *) rels,
    5933             :                          nrels * sizeof(RelFileLocator));
    5934             :     }
    5935             : 
    5936      234076 :     if (xl_xinfo.xinfo & XACT_XINFO_HAS_DROPPED_STATS)
    5937             :     {
    5938       20362 :         XLogRegisterData((char *) (&xl_dropped_stats),
    5939             :                          MinSizeOfXactStatsItems);
    5940       20362 :         XLogRegisterData((char *) droppedstats,
    5941             :                          ndroppedstats * sizeof(xl_xact_stats_item));
    5942             :     }
    5943             : 
    5944      234076 :     if (xl_xinfo.xinfo & XACT_XINFO_HAS_INVALS)
    5945             :     {
    5946      136108 :         XLogRegisterData((char *) (&xl_invals), MinSizeOfXactInvals);
    5947      136108 :         XLogRegisterData((char *) msgs,
    5948             :                          nmsgs * sizeof(SharedInvalidationMessage));
    5949             :     }
    5950             : 
    5951      234076 :     if (xl_xinfo.xinfo & XACT_XINFO_HAS_TWOPHASE)
    5952             :     {
    5953         692 :         XLogRegisterData((char *) (&xl_twophase), sizeof(xl_xact_twophase));
    5954         692 :         if (xl_xinfo.xinfo & XACT_XINFO_HAS_GID)
    5955          74 :             XLogRegisterData(twophase_gid, strlen(twophase_gid) + 1);
    5956             :     }
    5957             : 
    5958      234076 :     if (xl_xinfo.xinfo & XACT_XINFO_HAS_ORIGIN)
    5959        1976 :         XLogRegisterData((char *) (&xl_origin), sizeof(xl_xact_origin));
    5960             : 
    5961             :     /* we allow filtering by xacts */
    5962      234076 :     XLogSetRecordFlags(XLOG_INCLUDE_ORIGIN);
    5963             : 
    5964      234076 :     return XLogInsert(RM_XACT_ID, info);
    5965             : }
    5966             : 
    5967             : /*
    5968             :  * Log the commit record for a plain or twophase transaction abort.
    5969             :  *
    5970             :  * A 2pc abort will be emitted when twophase_xid is valid, a plain one
    5971             :  * otherwise.
    5972             :  */
    5973             : XLogRecPtr
    5974       11204 : XactLogAbortRecord(TimestampTz abort_time,
    5975             :                    int nsubxacts, TransactionId *subxacts,
    5976             :                    int nrels, RelFileLocator *rels,
    5977             :                    int ndroppedstats, xl_xact_stats_item *droppedstats,
    5978             :                    int xactflags, TransactionId twophase_xid,
    5979             :                    const char *twophase_gid)
    5980             : {
    5981             :     xl_xact_abort xlrec;
    5982             :     xl_xact_xinfo xl_xinfo;
    5983             :     xl_xact_subxacts xl_subxacts;
    5984             :     xl_xact_relfilelocators xl_relfilelocators;
    5985             :     xl_xact_stats_items xl_dropped_stats;
    5986             :     xl_xact_twophase xl_twophase;
    5987             :     xl_xact_dbinfo xl_dbinfo;
    5988             :     xl_xact_origin xl_origin;
    5989             : 
    5990             :     uint8       info;
    5991             : 
    5992             :     Assert(CritSectionCount > 0);
    5993             : 
    5994       11204 :     xl_xinfo.xinfo = 0;
    5995             : 
    5996             :     /* decide between a plain and 2pc abort */
    5997       11204 :     if (!TransactionIdIsValid(twophase_xid))
    5998       11126 :         info = XLOG_XACT_ABORT;
    5999             :     else
    6000          78 :         info = XLOG_XACT_ABORT_PREPARED;
    6001             : 
    6002             : 
    6003             :     /* First figure out and collect all the information needed */
    6004             : 
    6005       11204 :     xlrec.xact_time = abort_time;
    6006             : 
    6007       11204 :     if ((xactflags & XACT_FLAGS_ACQUIREDACCESSEXCLUSIVELOCK))
    6008        4462 :         xl_xinfo.xinfo |= XACT_XINFO_HAS_AE_LOCKS;
    6009             : 
    6010       11204 :     if (nsubxacts > 0)
    6011             :     {
    6012         202 :         xl_xinfo.xinfo |= XACT_XINFO_HAS_SUBXACTS;
    6013         202 :         xl_subxacts.nsubxacts = nsubxacts;
    6014             :     }
    6015             : 
    6016       11204 :     if (nrels > 0)
    6017             :     {
    6018        1806 :         xl_xinfo.xinfo |= XACT_XINFO_HAS_RELFILELOCATORS;
    6019        1806 :         xl_relfilelocators.nrels = nrels;
    6020        1806 :         info |= XLR_SPECIAL_REL_UPDATE;
    6021             :     }
    6022             : 
    6023       11204 :     if (ndroppedstats > 0)
    6024             :     {
    6025        2506 :         xl_xinfo.xinfo |= XACT_XINFO_HAS_DROPPED_STATS;
    6026        2506 :         xl_dropped_stats.nitems = ndroppedstats;
    6027             :     }
    6028             : 
    6029       11204 :     if (TransactionIdIsValid(twophase_xid))
    6030             :     {
    6031          78 :         xl_xinfo.xinfo |= XACT_XINFO_HAS_TWOPHASE;
    6032          78 :         xl_twophase.xid = twophase_xid;
    6033             :         Assert(twophase_gid != NULL);
    6034             : 
    6035          78 :         if (XLogLogicalInfoActive())
    6036          24 :             xl_xinfo.xinfo |= XACT_XINFO_HAS_GID;
    6037             :     }
    6038             : 
    6039       11204 :     if (TransactionIdIsValid(twophase_xid) && XLogLogicalInfoActive())
    6040             :     {
    6041          24 :         xl_xinfo.xinfo |= XACT_XINFO_HAS_DBINFO;
    6042          24 :         xl_dbinfo.dbId = MyDatabaseId;
    6043          24 :         xl_dbinfo.tsId = MyDatabaseTableSpace;
    6044             :     }
    6045             : 
    6046             :     /*
    6047             :      * Dump transaction origin information. We need this during recovery to
    6048             :      * update the replication origin progress.
    6049             :      */
    6050       11204 :     if (replorigin_session_origin != InvalidRepOriginId)
    6051             :     {
    6052          58 :         xl_xinfo.xinfo |= XACT_XINFO_HAS_ORIGIN;
    6053             : 
    6054          58 :         xl_origin.origin_lsn = replorigin_session_origin_lsn;
    6055          58 :         xl_origin.origin_timestamp = replorigin_session_origin_timestamp;
    6056             :     }
    6057             : 
    6058       11204 :     if (xl_xinfo.xinfo != 0)
    6059        5610 :         info |= XLOG_XACT_HAS_INFO;
    6060             : 
    6061             :     /* Then include all the collected data into the abort record. */
    6062             : 
    6063       11204 :     XLogBeginInsert();
    6064             : 
    6065       11204 :     XLogRegisterData((char *) (&xlrec), MinSizeOfXactAbort);
    6066             : 
    6067       11204 :     if (xl_xinfo.xinfo != 0)
    6068        5610 :         XLogRegisterData((char *) (&xl_xinfo), sizeof(xl_xinfo));
    6069             : 
    6070       11204 :     if (xl_xinfo.xinfo & XACT_XINFO_HAS_DBINFO)
    6071          24 :         XLogRegisterData((char *) (&xl_dbinfo), sizeof(xl_dbinfo));
    6072             : 
    6073       11204 :     if (xl_xinfo.xinfo & XACT_XINFO_HAS_SUBXACTS)
    6074             :     {
    6075         202 :         XLogRegisterData((char *) (&xl_subxacts),
    6076             :                          MinSizeOfXactSubxacts);
    6077         202 :         XLogRegisterData((char *) subxacts,
    6078             :                          nsubxacts * sizeof(TransactionId));
    6079             :     }
    6080             : 
    6081       11204 :     if (xl_xinfo.xinfo & XACT_XINFO_HAS_RELFILELOCATORS)
    6082             :     {
    6083        1806 :         XLogRegisterData((char *) (&xl_relfilelocators),
    6084             :                          MinSizeOfXactRelfileLocators);
    6085        1806 :         XLogRegisterData((char *) rels,
    6086             :                          nrels * sizeof(RelFileLocator));
    6087             :     }
    6088             : 
    6089       11204 :     if (xl_xinfo.xinfo & XACT_XINFO_HAS_DROPPED_STATS)
    6090             :     {
    6091        2506 :         XLogRegisterData((char *) (&xl_dropped_stats),
    6092             :                          MinSizeOfXactStatsItems);
    6093        2506 :         XLogRegisterData((char *) droppedstats,
    6094             :                          ndroppedstats * sizeof(xl_xact_stats_item));
    6095             :     }
    6096             : 
    6097       11204 :     if (xl_xinfo.xinfo & XACT_XINFO_HAS_TWOPHASE)
    6098             :     {
    6099          78 :         XLogRegisterData((char *) (&xl_twophase), sizeof(xl_xact_twophase));
    6100          78 :         if (xl_xinfo.xinfo & XACT_XINFO_HAS_GID)
    6101          24 :             XLogRegisterData(twophase_gid, strlen(twophase_gid) + 1);
    6102             :     }
    6103             : 
    6104       11204 :     if (xl_xinfo.xinfo & XACT_XINFO_HAS_ORIGIN)
    6105          58 :         XLogRegisterData((char *) (&xl_origin), sizeof(xl_xact_origin));
    6106             : 
    6107             :     /* Include the replication origin */
    6108       11204 :     XLogSetRecordFlags(XLOG_INCLUDE_ORIGIN);
    6109             : 
    6110       11204 :     return XLogInsert(RM_XACT_ID, info);
    6111             : }
    6112             : 
    6113             : /*
    6114             :  * Before 9.0 this was a fairly short function, but now it performs many
    6115             :  * actions for which the order of execution is critical.
    6116             :  */
    6117             : static void
    6118       41686 : xact_redo_commit(xl_xact_parsed_commit *parsed,
    6119             :                  TransactionId xid,
    6120             :                  XLogRecPtr lsn,
    6121             :                  RepOriginId origin_id)
    6122             : {
    6123             :     TransactionId max_xid;
    6124             :     TimestampTz commit_time;
    6125             : 
    6126             :     Assert(TransactionIdIsValid(xid));
    6127             : 
    6128       41686 :     max_xid = TransactionIdLatest(xid, parsed->nsubxacts, parsed->subxacts);
    6129             : 
    6130             :     /* Make sure nextXid is beyond any XID mentioned in the record. */
    6131       41686 :     AdvanceNextFullTransactionIdPastXid(max_xid);
    6132             : 
    6133             :     Assert(((parsed->xinfo & XACT_XINFO_HAS_ORIGIN) == 0) ==
    6134             :            (origin_id == InvalidRepOriginId));
    6135             : 
    6136       41686 :     if (parsed->xinfo & XACT_XINFO_HAS_ORIGIN)
    6137          40 :         commit_time = parsed->origin_timestamp;
    6138             :     else
    6139       41646 :         commit_time = parsed->xact_time;
    6140             : 
    6141             :     /* Set the transaction commit timestamp and metadata */
    6142       41686 :     TransactionTreeSetCommitTsData(xid, parsed->nsubxacts, parsed->subxacts,
    6143             :                                    commit_time, origin_id);
    6144             : 
    6145       41686 :     if (standbyState == STANDBY_DISABLED)
    6146             :     {
    6147             :         /*
    6148             :          * Mark the transaction committed in pg_xact.
    6149             :          */
    6150        4452 :         TransactionIdCommitTree(xid, parsed->nsubxacts, parsed->subxacts);
    6151             :     }
    6152             :     else
    6153             :     {
    6154             :         /*
    6155             :          * If a transaction completion record arrives that has as-yet
    6156             :          * unobserved subtransactions then this will not have been fully
    6157             :          * handled by the call to RecordKnownAssignedTransactionIds() in the
    6158             :          * main recovery loop in xlog.c. So we need to do bookkeeping again to
    6159             :          * cover that case. This is confusing and it is easy to think this
    6160             :          * call is irrelevant, which has happened three times in development
    6161             :          * already. Leave it in.
    6162             :          */
    6163       37234 :         RecordKnownAssignedTransactionIds(max_xid);
    6164             : 
    6165             :         /*
    6166             :          * Mark the transaction committed in pg_xact. We use async commit
    6167             :          * protocol during recovery to provide information on database
    6168             :          * consistency for when users try to set hint bits. It is important
    6169             :          * that we do not set hint bits until the minRecoveryPoint is past
    6170             :          * this commit record. This ensures that if we crash we don't see hint
    6171             :          * bits set on changes made by transactions that haven't yet
    6172             :          * recovered. It's unlikely but it's good to be safe.
    6173             :          */
    6174       37234 :         TransactionIdAsyncCommitTree(xid, parsed->nsubxacts, parsed->subxacts, lsn);
    6175             : 
    6176             :         /*
    6177             :          * We must mark clog before we update the ProcArray.
    6178             :          */
    6179       37234 :         ExpireTreeKnownAssignedTransactionIds(xid, parsed->nsubxacts, parsed->subxacts, max_xid);
    6180             : 
    6181             :         /*
    6182             :          * Send any cache invalidations attached to the commit. We must
    6183             :          * maintain the same order of invalidation then release locks as
    6184             :          * occurs in CommitTransaction().
    6185             :          */
    6186       37234 :         ProcessCommittedInvalidationMessages(parsed->msgs, parsed->nmsgs,
    6187       37234 :                                              XactCompletionRelcacheInitFileInval(parsed->xinfo),
    6188             :                                              parsed->dbId, parsed->tsId);
    6189             : 
    6190             :         /*
    6191             :          * Release locks, if any. We do this for both two phase and normal one
    6192             :          * phase transactions. In effect we are ignoring the prepare phase and
    6193             :          * just going straight to lock release.
    6194             :          */
    6195       37234 :         if (parsed->xinfo & XACT_XINFO_HAS_AE_LOCKS)
    6196       17926 :             StandbyReleaseLockTree(xid, parsed->nsubxacts, parsed->subxacts);
    6197             :     }
    6198             : 
    6199       41686 :     if (parsed->xinfo & XACT_XINFO_HAS_ORIGIN)
    6200             :     {
    6201             :         /* recover apply progress */
    6202          40 :         replorigin_advance(origin_id, parsed->origin_lsn, lsn,
    6203             :                            false /* backward */ , false /* WAL */ );
    6204             :     }
    6205             : 
    6206             :     /* Make sure files supposed to be dropped are dropped */
    6207       41686 :     if (parsed->nrels > 0)
    6208             :     {
    6209             :         /*
    6210             :          * First update minimum recovery point to cover this WAL record. Once
    6211             :          * a relation is deleted, there's no going back. The buffer manager
    6212             :          * enforces the WAL-first rule for normal updates to relation files,
    6213             :          * so that the minimum recovery point is always updated before the
    6214             :          * corresponding change in the data file is flushed to disk, but we
    6215             :          * have to do the same here since we're bypassing the buffer manager.
    6216             :          *
    6217             :          * Doing this before deleting the files means that if a deletion fails
    6218             :          * for some reason, you cannot start up the system even after restart,
    6219             :          * until you fix the underlying situation so that the deletion will
    6220             :          * succeed. Alternatively, we could update the minimum recovery point
    6221             :          * after deletion, but that would leave a small window where the
    6222             :          * WAL-first rule would be violated.
    6223             :          */
    6224        3838 :         XLogFlush(lsn);
    6225             : 
    6226             :         /* Make sure files supposed to be dropped are dropped */
    6227        3838 :         DropRelationFiles(parsed->xlocators, parsed->nrels, true);
    6228             :     }
    6229             : 
    6230       41686 :     if (parsed->nstats > 0)
    6231             :     {
    6232             :         /* see equivalent call for relations above */
    6233        5014 :         XLogFlush(lsn);
    6234             : 
    6235        5014 :         pgstat_execute_transactional_drops(parsed->nstats, parsed->stats, true);
    6236             :     }
    6237             : 
    6238             :     /*
    6239             :      * We issue an XLogFlush() for the same reason we emit ForceSyncCommit()
    6240             :      * in normal operation. For example, in CREATE DATABASE, we copy all files
    6241             :      * from the template database, and then commit the transaction. If we
    6242             :      * crash after all the files have been copied but before the commit, you
    6243             :      * have files in the data directory without an entry in pg_database. To
    6244             :      * minimize the window for that, we use ForceSyncCommit() to rush the
    6245             :      * commit record to disk as quick as possible. We have the same window
    6246             :      * during recovery, and forcing an XLogFlush() (which updates
    6247             :      * minRecoveryPoint during recovery) helps to reduce that problem window,
    6248             :      * for any user that requested ForceSyncCommit().
    6249             :      */
    6250       41686 :     if (XactCompletionForceSyncCommit(parsed->xinfo))
    6251          94 :         XLogFlush(lsn);
    6252             : 
    6253             :     /*
    6254             :      * If asked by the primary (because someone is waiting for a synchronous
    6255             :      * commit = remote_apply), we will need to ask walreceiver to send a reply
    6256             :      * immediately.
    6257             :      */
    6258       41686 :     if (XactCompletionApplyFeedback(parsed->xinfo))
    6259           4 :         XLogRequestWalReceiverReply();
    6260       41686 : }
    6261             : 
    6262             : /*
    6263             :  * Be careful with the order of execution, as with xact_redo_commit().
    6264             :  * The two functions are similar but differ in key places.
    6265             :  *
    6266             :  * Note also that an abort can be for a subtransaction and its children,
    6267             :  * not just for a top level abort. That means we have to consider
    6268             :  * topxid != xid, whereas in commit we would find topxid == xid always
    6269             :  * because subtransaction commit is never WAL logged.
    6270             :  */
    6271             : static void
    6272        3358 : xact_redo_abort(xl_xact_parsed_abort *parsed, TransactionId xid,
    6273             :                 XLogRecPtr lsn, RepOriginId origin_id)
    6274             : {
    6275             :     TransactionId max_xid;
    6276             : 
    6277             :     Assert(TransactionIdIsValid(xid));
    6278             : 
    6279             :     /* Make sure nextXid is beyond any XID mentioned in the record. */
    6280        3358 :     max_xid = TransactionIdLatest(xid,
    6281             :                                   parsed->nsubxacts,
    6282        3358 :                                   parsed->subxacts);
    6283        3358 :     AdvanceNextFullTransactionIdPastXid(max_xid);
    6284             : 
    6285        3358 :     if (standbyState == STANDBY_DISABLED)
    6286             :     {
    6287             :         /* Mark the transaction aborted in pg_xact, no need for async stuff */
    6288          40 :         TransactionIdAbortTree(xid, parsed->nsubxacts, parsed->subxacts);
    6289             :     }
    6290             :     else
    6291             :     {
    6292             :         /*
    6293             :          * If a transaction completion record arrives that has as-yet
    6294             :          * unobserved subtransactions then this will not have been fully
    6295             :          * handled by the call to RecordKnownAssignedTransactionIds() in the
    6296             :          * main recovery loop in xlog.c. So we need to do bookkeeping again to
    6297             :          * cover that case. This is confusing and it is easy to think this
    6298             :          * call is irrelevant, which has happened three times in development
    6299             :          * already. Leave it in.
    6300             :          */
    6301        3318 :         RecordKnownAssignedTransactionIds(max_xid);
    6302             : 
    6303             :         /* Mark the transaction aborted in pg_xact, no need for async stuff */
    6304        3318 :         TransactionIdAbortTree(xid, parsed->nsubxacts, parsed->subxacts);
    6305             : 
    6306             :         /*
    6307             :          * We must update the ProcArray after we have marked clog.
    6308             :          */
    6309        3318 :         ExpireTreeKnownAssignedTransactionIds(xid, parsed->nsubxacts, parsed->subxacts, max_xid);
    6310             : 
    6311             :         /*
    6312             :          * There are no invalidation messages to send or undo.
    6313             :          */
    6314             : 
    6315             :         /*
    6316             :          * Release locks, if any. There are no invalidations to send.
    6317             :          */
    6318        3318 :         if (parsed->xinfo & XACT_XINFO_HAS_AE_LOCKS)
    6319        2022 :             StandbyReleaseLockTree(xid, parsed->nsubxacts, parsed->subxacts);
    6320             :     }
    6321             : 
    6322        3358 :     if (parsed->xinfo & XACT_XINFO_HAS_ORIGIN)
    6323             :     {
    6324             :         /* recover apply progress */
    6325          10 :         replorigin_advance(origin_id, parsed->origin_lsn, lsn,
    6326             :                            false /* backward */ , false /* WAL */ );
    6327             :     }
    6328             : 
    6329             :     /* Make sure files supposed to be dropped are dropped */
    6330        3358 :     if (parsed->nrels > 0)
    6331             :     {
    6332             :         /*
    6333             :          * See comments about update of minimum recovery point on truncation,
    6334             :          * in xact_redo_commit().
    6335             :          */
    6336         568 :         XLogFlush(lsn);
    6337             : 
    6338         568 :         DropRelationFiles(parsed->xlocators, parsed->nrels, true);
    6339             :     }
    6340             : 
    6341        3358 :     if (parsed->nstats > 0)
    6342             :     {
    6343             :         /* see equivalent call for relations above */
    6344         764 :         XLogFlush(lsn);
    6345             : 
    6346         764 :         pgstat_execute_transactional_drops(parsed->nstats, parsed->stats, true);
    6347             :     }
    6348        3358 : }
    6349             : 
    6350             : void
    6351       45582 : xact_redo(XLogReaderState *record)
    6352             : {
    6353       45582 :     uint8       info = XLogRecGetInfo(record) & XLOG_XACT_OPMASK;
    6354             : 
    6355             :     /* Backup blocks are not used in xact records */
    6356             :     Assert(!XLogRecHasAnyBlockRefs(record));
    6357             : 
    6358       45582 :     if (info == XLOG_XACT_COMMIT)
    6359             :     {
    6360       41598 :         xl_xact_commit *xlrec = (xl_xact_commit *) XLogRecGetData(record);
    6361             :         xl_xact_parsed_commit parsed;
    6362             : 
    6363       41598 :         ParseCommitRecord(XLogRecGetInfo(record), xlrec, &parsed);
    6364       41598 :         xact_redo_commit(&parsed, XLogRecGetXid(record),
    6365       41598 :                          record->EndRecPtr, XLogRecGetOrigin(record));
    6366             :     }
    6367        3984 :     else if (info == XLOG_XACT_COMMIT_PREPARED)
    6368             :     {
    6369          88 :         xl_xact_commit *xlrec = (xl_xact_commit *) XLogRecGetData(record);
    6370             :         xl_xact_parsed_commit parsed;
    6371             : 
    6372          88 :         ParseCommitRecord(XLogRecGetInfo(record), xlrec, &parsed);
    6373          88 :         xact_redo_commit(&parsed, parsed.twophase_xid,
    6374          88 :                          record->EndRecPtr, XLogRecGetOrigin(record));
    6375             : 
    6376             :         /* Delete TwoPhaseState gxact entry and/or 2PC file. */
    6377          88 :         LWLockAcquire(TwoPhaseStateLock, LW_EXCLUSIVE);
    6378          88 :         PrepareRedoRemove(parsed.twophase_xid, false);
    6379          88 :         LWLockRelease(TwoPhaseStateLock);
    6380             :     }
    6381        3896 :     else if (info == XLOG_XACT_ABORT)
    6382             :     {
    6383        3316 :         xl_xact_abort *xlrec = (xl_xact_abort *) XLogRecGetData(record);
    6384             :         xl_xact_parsed_abort parsed;
    6385             : 
    6386        3316 :         ParseAbortRecord(XLogRecGetInfo(record), xlrec, &parsed);
    6387        3316 :         xact_redo_abort(&parsed, XLogRecGetXid(record),
    6388        3316 :                         record->EndRecPtr, XLogRecGetOrigin(record));
    6389             :     }
    6390         580 :     else if (info == XLOG_XACT_ABORT_PREPARED)
    6391             :     {
    6392          42 :         xl_xact_abort *xlrec = (xl_xact_abort *) XLogRecGetData(record);
    6393             :         xl_xact_parsed_abort parsed;
    6394             : 
    6395          42 :         ParseAbortRecord(XLogRecGetInfo(record), xlrec, &parsed);
    6396          42 :         xact_redo_abort(&parsed, parsed.twophase_xid,
    6397          42 :                         record->EndRecPtr, XLogRecGetOrigin(record));
    6398             : 
    6399             :         /* Delete TwoPhaseState gxact entry and/or 2PC file. */
    6400          42 :         LWLockAcquire(TwoPhaseStateLock, LW_EXCLUSIVE);
    6401          42 :         PrepareRedoRemove(parsed.twophase_xid, false);
    6402          42 :         LWLockRelease(TwoPhaseStateLock);
    6403             :     }
    6404         538 :     else if (info == XLOG_XACT_PREPARE)
    6405             :     {
    6406             :         /*
    6407             :          * Store xid and start/end pointers of the WAL record in TwoPhaseState
    6408             :          * gxact entry.
    6409             :          */
    6410         150 :         LWLockAcquire(TwoPhaseStateLock, LW_EXCLUSIVE);
    6411         150 :         PrepareRedoAdd(XLogRecGetData(record),
    6412             :                        record->ReadRecPtr,
    6413             :                        record->EndRecPtr,
    6414         150 :                        XLogRecGetOrigin(record));
    6415         150 :         LWLockRelease(TwoPhaseStateLock);
    6416             :     }
    6417         388 :     else if (info == XLOG_XACT_ASSIGNMENT)
    6418             :     {
    6419          42 :         xl_xact_assignment *xlrec = (xl_xact_assignment *) XLogRecGetData(record);
    6420             : 
    6421          42 :         if (standbyState >= STANDBY_INITIALIZED)
    6422          42 :             ProcArrayApplyXidAssignment(xlrec->xtop,
    6423          42 :                                         xlrec->nsubxacts, xlrec->xsub);
    6424             :     }
    6425         346 :     else if (info == XLOG_XACT_INVALIDATIONS)
    6426             :     {
    6427             :         /*
    6428             :          * XXX we do ignore this for now, what matters are invalidations
    6429             :          * written into the commit record.
    6430             :          */
    6431             :     }
    6432             :     else
    6433           0 :         elog(PANIC, "xact_redo: unknown op code %u", info);
    6434       45582 : }

Generated by: LCOV version 1.14