Commit ad308058 authored by Thomas Munro's avatar Thomas Munro

Use FullTransactionId for the transaction stack.

Provide GetTopFullTransactionId() and GetCurrentFullTransactionId().
The intended users of these interfaces are access methods that use
xids for visibility checks but don't want to have to go back and
"freeze" existing references some time later before the 32 bit xid
counter wraps around.

Use a new struct to serialize the transaction state for parallel
query, because FullTransactionId doesn't fit into the previous
serialization scheme very well.

Author: Thomas Munro
Reviewed-by: Heikki Linnakangas
Discussion: https://postgr.es/m/CAA4eK1%2BMv%2Bmb0HFfWM9Srtc6MVe160WFurXV68iAFMcagRZ0dQ%40mail.gmail.com
parent 2fc7af5e
...@@ -35,7 +35,8 @@ VariableCache ShmemVariableCache = NULL; ...@@ -35,7 +35,8 @@ VariableCache ShmemVariableCache = NULL;
/* /*
* Allocate the next XID for a new transaction or subtransaction. * Allocate the next FullTransactionId for a new transaction or
* subtransaction.
* *
* The new XID is also stored into MyPgXact before returning. * The new XID is also stored into MyPgXact before returning.
* *
...@@ -44,9 +45,10 @@ VariableCache ShmemVariableCache = NULL; ...@@ -44,9 +45,10 @@ VariableCache ShmemVariableCache = NULL;
* does something. So it is safe to do a database lookup if we want to * does something. So it is safe to do a database lookup if we want to
* issue a warning about XID wrap. * issue a warning about XID wrap.
*/ */
TransactionId FullTransactionId
GetNewTransactionId(bool isSubXact) GetNewTransactionId(bool isSubXact)
{ {
FullTransactionId full_xid;
TransactionId xid; TransactionId xid;
/* /*
...@@ -64,7 +66,7 @@ GetNewTransactionId(bool isSubXact) ...@@ -64,7 +66,7 @@ GetNewTransactionId(bool isSubXact)
{ {
Assert(!isSubXact); Assert(!isSubXact);
MyPgXact->xid = BootstrapTransactionId; MyPgXact->xid = BootstrapTransactionId;
return BootstrapTransactionId; return FullTransactionIdFromEpochAndXid(0, BootstrapTransactionId);
} }
/* safety check, we should never get this far in a HS standby */ /* safety check, we should never get this far in a HS standby */
...@@ -73,7 +75,8 @@ GetNewTransactionId(bool isSubXact) ...@@ -73,7 +75,8 @@ GetNewTransactionId(bool isSubXact)
LWLockAcquire(XidGenLock, LW_EXCLUSIVE); LWLockAcquire(XidGenLock, LW_EXCLUSIVE);
xid = XidFromFullTransactionId(ShmemVariableCache->nextFullXid); full_xid = ShmemVariableCache->nextFullXid;
xid = XidFromFullTransactionId(full_xid);
/*---------- /*----------
* Check to see if it's safe to assign another XID. This protects against * Check to see if it's safe to assign another XID. This protects against
...@@ -232,7 +235,7 @@ GetNewTransactionId(bool isSubXact) ...@@ -232,7 +235,7 @@ GetNewTransactionId(bool isSubXact)
LWLockRelease(XidGenLock); LWLockRelease(XidGenLock);
return xid; return full_xid;
} }
/* /*
......
...@@ -105,7 +105,7 @@ int synchronous_commit = SYNCHRONOUS_COMMIT_ON; ...@@ -105,7 +105,7 @@ int synchronous_commit = SYNCHRONOUS_COMMIT_ON;
* The XIDs are stored sorted in numerical order (not logical order) to make * The XIDs are stored sorted in numerical order (not logical order) to make
* lookups as fast as possible. * lookups as fast as possible.
*/ */
TransactionId XactTopTransactionId = InvalidTransactionId; FullTransactionId XactTopFullTransactionId = {InvalidTransactionId};
int nParallelCurrentXids = 0; int nParallelCurrentXids = 0;
TransactionId *ParallelCurrentXids; TransactionId *ParallelCurrentXids;
...@@ -171,7 +171,7 @@ typedef enum TBlockState ...@@ -171,7 +171,7 @@ typedef enum TBlockState
*/ */
typedef struct TransactionStateData typedef struct TransactionStateData
{ {
TransactionId transactionId; /* my XID, or Invalid if none */ FullTransactionId fullTransactionId; /* my FullTransactionId */
SubTransactionId subTransactionId; /* my subxact ID */ SubTransactionId subTransactionId; /* my subxact ID */
char *name; /* savepoint name, if any */ char *name; /* savepoint name, if any */
int savepointLevel; /* savepoint level */ int savepointLevel; /* savepoint level */
...@@ -196,6 +196,25 @@ typedef struct TransactionStateData ...@@ -196,6 +196,25 @@ typedef struct TransactionStateData
typedef TransactionStateData *TransactionState; typedef TransactionStateData *TransactionState;
/*
* Serialized representation used to transmit transaction state to parallel
* workers though shared memory.
*/
typedef struct SerializedTransactionState
{
int xactIsoLevel;
bool xactDeferrable;
FullTransactionId topFullTransactionId;
FullTransactionId currentFullTransactionId;
CommandId currentCommandId;
int nParallelCurrentXids;
TransactionId parallelCurrentXids[FLEXIBLE_ARRAY_MEMBER];
} SerializedTransactionState;
/* The size of SerializedTransactionState, not including the final array. */
#define SerializedTransactionStateHeaderSize \
offsetof(SerializedTransactionState, parallelCurrentXids)
/* /*
* CurrentTransactionState always points to the current transaction state * CurrentTransactionState always points to the current transaction state
* block. It will point to TopTransactionStateData when not in a * block. It will point to TopTransactionStateData when not in a
...@@ -372,9 +391,9 @@ IsAbortedTransactionBlockState(void) ...@@ -372,9 +391,9 @@ IsAbortedTransactionBlockState(void)
TransactionId TransactionId
GetTopTransactionId(void) GetTopTransactionId(void)
{ {
if (!TransactionIdIsValid(XactTopTransactionId)) if (!FullTransactionIdIsValid(XactTopFullTransactionId))
AssignTransactionId(&TopTransactionStateData); AssignTransactionId(&TopTransactionStateData);
return XactTopTransactionId; return XidFromFullTransactionId(XactTopFullTransactionId);
} }
/* /*
...@@ -387,7 +406,7 @@ GetTopTransactionId(void) ...@@ -387,7 +406,7 @@ GetTopTransactionId(void)
TransactionId TransactionId
GetTopTransactionIdIfAny(void) GetTopTransactionIdIfAny(void)
{ {
return XactTopTransactionId; return XidFromFullTransactionId(XactTopFullTransactionId);
} }
/* /*
...@@ -402,9 +421,9 @@ GetCurrentTransactionId(void) ...@@ -402,9 +421,9 @@ GetCurrentTransactionId(void)
{ {
TransactionState s = CurrentTransactionState; TransactionState s = CurrentTransactionState;
if (!TransactionIdIsValid(s->transactionId)) if (!FullTransactionIdIsValid(s->fullTransactionId))
AssignTransactionId(s); AssignTransactionId(s);
return s->transactionId; return XidFromFullTransactionId(s->fullTransactionId);
} }
/* /*
...@@ -417,7 +436,66 @@ GetCurrentTransactionId(void) ...@@ -417,7 +436,66 @@ GetCurrentTransactionId(void)
TransactionId TransactionId
GetCurrentTransactionIdIfAny(void) GetCurrentTransactionIdIfAny(void)
{ {
return CurrentTransactionState->transactionId; return XidFromFullTransactionId(CurrentTransactionState->fullTransactionId);
}
/*
* GetTopFullTransactionId
*
* This will return the FullTransactionId of the main transaction, assigning
* one if it's not yet set. Be careful to call this only inside a valid xact.
*/
FullTransactionId
GetTopFullTransactionId(void)
{
if (!FullTransactionIdIsValid(XactTopFullTransactionId))
AssignTransactionId(&TopTransactionStateData);
return XactTopFullTransactionId;
}
/*
* GetTopFullTransactionIdIfAny
*
* This will return the FullTransactionId of the main transaction, if one is
* assigned. It will return InvalidFullTransactionId if we are not currently
* inside a transaction, or inside a transaction that hasn't yet been assigned
* one.
*/
FullTransactionId
GetTopFullTransactionIdIfAny(void)
{
return XactTopFullTransactionId;
}
/*
* GetCurrentFullTransactionId
*
* This will return the FullTransactionId of the current transaction (main or
* sub transaction), assigning one if it's not yet set. Be careful to call
* this only inside a valid xact.
*/
FullTransactionId
GetCurrentFullTransactionId(void)
{
TransactionState s = CurrentTransactionState;
if (!FullTransactionIdIsValid(s->fullTransactionId))
AssignTransactionId(s);
return s->fullTransactionId;
}
/*
* GetCurrentFullTransactionIdIfAny
*
* This will return the FullTransactionId of the current sub xact, if one is
* assigned. It will return InvalidFullTransactionId if we are not currently
* inside a transaction, or inside a transaction that hasn't been assigned one
* yet.
*/
FullTransactionId
GetCurrentFullTransactionIdIfAny(void)
{
return CurrentTransactionState->fullTransactionId;
} }
/* /*
...@@ -428,7 +506,7 @@ GetCurrentTransactionIdIfAny(void) ...@@ -428,7 +506,7 @@ GetCurrentTransactionIdIfAny(void)
void void
MarkCurrentTransactionIdLoggedIfAny(void) MarkCurrentTransactionIdLoggedIfAny(void)
{ {
if (TransactionIdIsValid(CurrentTransactionState->transactionId)) if (FullTransactionIdIsValid(CurrentTransactionState->fullTransactionId))
CurrentTransactionState->didLogXid = true; CurrentTransactionState->didLogXid = true;
} }
...@@ -463,7 +541,7 @@ GetStableLatestTransactionId(void) ...@@ -463,7 +541,7 @@ GetStableLatestTransactionId(void)
/* /*
* AssignTransactionId * AssignTransactionId
* *
* Assigns a new permanent XID to the given TransactionState. * Assigns a new permanent FullTransactionId to the given TransactionState.
* We do not assign XIDs to transactions until/unless this is called. * We do not assign XIDs to transactions until/unless this is called.
* Also, any parent TransactionStates that don't yet have XIDs are assigned * Also, any parent TransactionStates that don't yet have XIDs are assigned
* one; this maintains the invariant that a child transaction has an XID * one; this maintains the invariant that a child transaction has an XID
...@@ -477,7 +555,7 @@ AssignTransactionId(TransactionState s) ...@@ -477,7 +555,7 @@ AssignTransactionId(TransactionState s)
bool log_unknown_top = false; bool log_unknown_top = false;
/* Assert that caller didn't screw up */ /* Assert that caller didn't screw up */
Assert(!TransactionIdIsValid(s->transactionId)); Assert(!FullTransactionIdIsValid(s->fullTransactionId));
Assert(s->state == TRANS_INPROGRESS); Assert(s->state == TRANS_INPROGRESS);
/* /*
...@@ -493,14 +571,14 @@ AssignTransactionId(TransactionState s) ...@@ -493,14 +571,14 @@ AssignTransactionId(TransactionState s)
* if we're at the bottom of a huge stack of subtransactions none of which * if we're at the bottom of a huge stack of subtransactions none of which
* have XIDs yet. * have XIDs yet.
*/ */
if (isSubXact && !TransactionIdIsValid(s->parent->transactionId)) if (isSubXact && !FullTransactionIdIsValid(s->parent->fullTransactionId))
{ {
TransactionState p = s->parent; TransactionState p = s->parent;
TransactionState *parents; TransactionState *parents;
size_t parentOffset = 0; size_t parentOffset = 0;
parents = palloc(sizeof(TransactionState) * s->nestingLevel); parents = palloc(sizeof(TransactionState) * s->nestingLevel);
while (p != NULL && !TransactionIdIsValid(p->transactionId)) while (p != NULL && !FullTransactionIdIsValid(p->fullTransactionId))
{ {
parents[parentOffset++] = p; parents[parentOffset++] = p;
p = p->parent; p = p->parent;
...@@ -531,26 +609,28 @@ AssignTransactionId(TransactionState s) ...@@ -531,26 +609,28 @@ AssignTransactionId(TransactionState s)
log_unknown_top = true; log_unknown_top = true;
/* /*
* Generate a new Xid and record it in PG_PROC and pg_subtrans. * Generate a new FullTransactionId and record its xid in PG_PROC and
* pg_subtrans.
* *
* NB: we must make the subtrans entry BEFORE the Xid appears anywhere in * NB: we must make the subtrans entry BEFORE the Xid appears anywhere in
* shared storage other than PG_PROC; because if there's no room for it in * shared storage other than PG_PROC; because if there's no room for it in
* PG_PROC, the subtrans entry is needed to ensure that other backends see * PG_PROC, the subtrans entry is needed to ensure that other backends see
* the Xid as "running". See GetNewTransactionId. * the Xid as "running". See GetNewTransactionId.
*/ */
s->transactionId = GetNewTransactionId(isSubXact); s->fullTransactionId = GetNewTransactionId(isSubXact);
if (!isSubXact) if (!isSubXact)
XactTopTransactionId = s->transactionId; XactTopFullTransactionId = s->fullTransactionId;
if (isSubXact) if (isSubXact)
SubTransSetParent(s->transactionId, s->parent->transactionId); SubTransSetParent(XidFromFullTransactionId(s->fullTransactionId),
XidFromFullTransactionId(s->parent->fullTransactionId));
/* /*
* If it's a top-level transaction, the predicate locking system needs to * If it's a top-level transaction, the predicate locking system needs to
* be told about it too. * be told about it too.
*/ */
if (!isSubXact) if (!isSubXact)
RegisterPredicateLockingXid(s->transactionId); RegisterPredicateLockingXid(XidFromFullTransactionId(s->fullTransactionId));
/* /*
* Acquire lock on the transaction XID. (We assume this cannot block.) We * Acquire lock on the transaction XID. (We assume this cannot block.) We
...@@ -560,7 +640,7 @@ AssignTransactionId(TransactionState s) ...@@ -560,7 +640,7 @@ AssignTransactionId(TransactionState s)
currentOwner = CurrentResourceOwner; currentOwner = CurrentResourceOwner;
CurrentResourceOwner = s->curTransactionOwner; CurrentResourceOwner = s->curTransactionOwner;
XactLockTableInsert(s->transactionId); XactLockTableInsert(XidFromFullTransactionId(s->fullTransactionId));
CurrentResourceOwner = currentOwner; CurrentResourceOwner = currentOwner;
...@@ -584,7 +664,7 @@ AssignTransactionId(TransactionState s) ...@@ -584,7 +664,7 @@ AssignTransactionId(TransactionState s)
*/ */
if (isSubXact && XLogStandbyInfoActive()) if (isSubXact && XLogStandbyInfoActive())
{ {
unreportedXids[nUnreportedXids] = s->transactionId; unreportedXids[nUnreportedXids] = XidFromFullTransactionId(s->fullTransactionId);
nUnreportedXids++; nUnreportedXids++;
/* /*
...@@ -832,9 +912,9 @@ TransactionIdIsCurrentTransactionId(TransactionId xid) ...@@ -832,9 +912,9 @@ TransactionIdIsCurrentTransactionId(TransactionId xid)
if (s->state == TRANS_ABORT) if (s->state == TRANS_ABORT)
continue; continue;
if (!TransactionIdIsValid(s->transactionId)) if (!FullTransactionIdIsValid(s->fullTransactionId))
continue; /* it can't have any child XIDs either */ continue; /* it can't have any child XIDs either */
if (TransactionIdEquals(xid, s->transactionId)) if (TransactionIdEquals(xid, XidFromFullTransactionId(s->fullTransactionId)))
return true; return true;
/* As the childXids array is ordered, we can use binary search */ /* As the childXids array is ordered, we can use binary search */
low = 0; low = 0;
...@@ -1495,7 +1575,7 @@ AtSubCommit_childXids(void) ...@@ -1495,7 +1575,7 @@ AtSubCommit_childXids(void)
* all XIDs already in the array belong to subtransactions started and * all XIDs already in the array belong to subtransactions started and
* subcommitted before us, so their XIDs must precede ours. * subcommitted before us, so their XIDs must precede ours.
*/ */
s->parent->childXids[s->parent->nChildXids] = s->transactionId; s->parent->childXids[s->parent->nChildXids] = XidFromFullTransactionId(s->fullTransactionId);
if (s->nChildXids > 0) if (s->nChildXids > 0)
memcpy(&s->parent->childXids[s->parent->nChildXids + 1], memcpy(&s->parent->childXids[s->parent->nChildXids + 1],
...@@ -1809,7 +1889,7 @@ StartTransaction(void) ...@@ -1809,7 +1889,7 @@ StartTransaction(void)
s = &TopTransactionStateData; s = &TopTransactionStateData;
CurrentTransactionState = s; CurrentTransactionState = s;
Assert(XactTopTransactionId == InvalidTransactionId); Assert(!FullTransactionIdIsValid(XactTopFullTransactionId));
/* check the current transaction state */ /* check the current transaction state */
Assert(s->state == TRANS_DEFAULT); Assert(s->state == TRANS_DEFAULT);
...@@ -1821,7 +1901,7 @@ StartTransaction(void) ...@@ -1821,7 +1901,7 @@ StartTransaction(void)
* flags are fetched below. * flags are fetched below.
*/ */
s->state = TRANS_START; s->state = TRANS_START;
s->transactionId = InvalidTransactionId; /* until assigned */ s->fullTransactionId = InvalidFullTransactionId; /* until assigned */
/* /*
* initialize current transaction state fields * initialize current transaction state fields
...@@ -2165,7 +2245,7 @@ CommitTransaction(void) ...@@ -2165,7 +2245,7 @@ CommitTransaction(void)
AtCommit_Memory(); AtCommit_Memory();
s->transactionId = InvalidTransactionId; s->fullTransactionId = InvalidFullTransactionId;
s->subTransactionId = InvalidSubTransactionId; s->subTransactionId = InvalidSubTransactionId;
s->nestingLevel = 0; s->nestingLevel = 0;
s->gucNestLevel = 0; s->gucNestLevel = 0;
...@@ -2173,7 +2253,7 @@ CommitTransaction(void) ...@@ -2173,7 +2253,7 @@ CommitTransaction(void)
s->nChildXids = 0; s->nChildXids = 0;
s->maxChildXids = 0; s->maxChildXids = 0;
XactTopTransactionId = InvalidTransactionId; XactTopFullTransactionId = InvalidFullTransactionId;
nParallelCurrentXids = 0; nParallelCurrentXids = 0;
/* /*
...@@ -2448,7 +2528,7 @@ PrepareTransaction(void) ...@@ -2448,7 +2528,7 @@ PrepareTransaction(void)
AtCommit_Memory(); AtCommit_Memory();
s->transactionId = InvalidTransactionId; s->fullTransactionId = InvalidFullTransactionId;
s->subTransactionId = InvalidSubTransactionId; s->subTransactionId = InvalidSubTransactionId;
s->nestingLevel = 0; s->nestingLevel = 0;
s->gucNestLevel = 0; s->gucNestLevel = 0;
...@@ -2456,7 +2536,7 @@ PrepareTransaction(void) ...@@ -2456,7 +2536,7 @@ PrepareTransaction(void)
s->nChildXids = 0; s->nChildXids = 0;
s->maxChildXids = 0; s->maxChildXids = 0;
XactTopTransactionId = InvalidTransactionId; XactTopFullTransactionId = InvalidFullTransactionId;
nParallelCurrentXids = 0; nParallelCurrentXids = 0;
/* /*
...@@ -2686,7 +2766,7 @@ CleanupTransaction(void) ...@@ -2686,7 +2766,7 @@ CleanupTransaction(void)
AtCleanup_Memory(); /* and transaction memory */ AtCleanup_Memory(); /* and transaction memory */
s->transactionId = InvalidTransactionId; s->fullTransactionId = InvalidFullTransactionId;
s->subTransactionId = InvalidSubTransactionId; s->subTransactionId = InvalidSubTransactionId;
s->nestingLevel = 0; s->nestingLevel = 0;
s->gucNestLevel = 0; s->gucNestLevel = 0;
...@@ -2695,7 +2775,7 @@ CleanupTransaction(void) ...@@ -2695,7 +2775,7 @@ CleanupTransaction(void)
s->maxChildXids = 0; s->maxChildXids = 0;
s->parallelModeLevel = 0; s->parallelModeLevel = 0;
XactTopTransactionId = InvalidTransactionId; XactTopFullTransactionId = InvalidFullTransactionId;
nParallelCurrentXids = 0; nParallelCurrentXids = 0;
/* /*
...@@ -4693,7 +4773,7 @@ CommitSubTransaction(void) ...@@ -4693,7 +4773,7 @@ CommitSubTransaction(void)
*/ */
/* Post-commit cleanup */ /* Post-commit cleanup */
if (TransactionIdIsValid(s->transactionId)) if (FullTransactionIdIsValid(s->fullTransactionId))
AtSubCommit_childXids(); AtSubCommit_childXids();
AfterTriggerEndSubXact(true); AfterTriggerEndSubXact(true);
AtSubCommit_Portals(s->subTransactionId, AtSubCommit_Portals(s->subTransactionId,
...@@ -4718,8 +4798,8 @@ CommitSubTransaction(void) ...@@ -4718,8 +4798,8 @@ CommitSubTransaction(void)
* The only lock we actually release here is the subtransaction XID lock. * The only lock we actually release here is the subtransaction XID lock.
*/ */
CurrentResourceOwner = s->curTransactionOwner; CurrentResourceOwner = s->curTransactionOwner;
if (TransactionIdIsValid(s->transactionId)) if (FullTransactionIdIsValid(s->fullTransactionId))
XactLockTableDelete(s->transactionId); XactLockTableDelete(XidFromFullTransactionId(s->fullTransactionId));
/* /*
* Other locks should get transferred to their parent resource owner. * Other locks should get transferred to their parent resource owner.
...@@ -4872,7 +4952,7 @@ AbortSubTransaction(void) ...@@ -4872,7 +4952,7 @@ AbortSubTransaction(void)
(void) RecordTransactionAbort(true); (void) RecordTransactionAbort(true);
/* Post-abort cleanup */ /* Post-abort cleanup */
if (TransactionIdIsValid(s->transactionId)) if (FullTransactionIdIsValid(s->fullTransactionId))
AtSubAbort_childXids(); AtSubAbort_childXids();
CallSubXactCallbacks(SUBXACT_EVENT_ABORT_SUB, s->subTransactionId, CallSubXactCallbacks(SUBXACT_EVENT_ABORT_SUB, s->subTransactionId,
...@@ -4985,7 +5065,7 @@ PushTransaction(void) ...@@ -4985,7 +5065,7 @@ PushTransaction(void)
* We can now stack a minimally valid subtransaction without fear of * We can now stack a minimally valid subtransaction without fear of
* failure. * failure.
*/ */
s->transactionId = InvalidTransactionId; /* until assigned */ s->fullTransactionId = InvalidFullTransactionId; /* until assigned */
s->subTransactionId = currentSubTransactionId; s->subTransactionId = currentSubTransactionId;
s->parent = p; s->parent = p;
s->nestingLevel = p->nestingLevel + 1; s->nestingLevel = p->nestingLevel + 1;
...@@ -5052,18 +5132,17 @@ Size ...@@ -5052,18 +5132,17 @@ Size
EstimateTransactionStateSpace(void) EstimateTransactionStateSpace(void)
{ {
TransactionState s; TransactionState s;
Size nxids = 6; /* iso level, deferrable, top & current XID, Size nxids = 0;
* command counter, XID count */ Size size = SerializedTransactionStateHeaderSize;
for (s = CurrentTransactionState; s != NULL; s = s->parent) for (s = CurrentTransactionState; s != NULL; s = s->parent)
{ {
if (TransactionIdIsValid(s->transactionId)) if (FullTransactionIdIsValid(s->fullTransactionId))
nxids = add_size(nxids, 1); nxids = add_size(nxids, 1);
nxids = add_size(nxids, s->nChildXids); nxids = add_size(nxids, s->nChildXids);
} }
nxids = add_size(nxids, nParallelCurrentXids); return add_size(size, sizeof(SerializedTransactionState) * nxids);
return mul_size(nxids, sizeof(TransactionId));
} }
/* /*
...@@ -5072,14 +5151,10 @@ EstimateTransactionStateSpace(void) ...@@ -5072,14 +5151,10 @@ EstimateTransactionStateSpace(void)
* needed by a parallel worker. * needed by a parallel worker.
* *
* We need to save and restore XactDeferrable, XactIsoLevel, and the XIDs * We need to save and restore XactDeferrable, XactIsoLevel, and the XIDs
* associated with this transaction. The first eight bytes of the result * associated with this transaction. These are serialized into a
* contain XactDeferrable and XactIsoLevel; the next twelve bytes contain the * caller-supplied buffer big enough to hold the number of bytes reported by
* XID of the top-level transaction, the XID of the current transaction * EstimateTransactionStateSpace(). We emit the XIDs in sorted order for the
* (or, in each case, InvalidTransactionId if none), and the current command * convenience of the receiving process.
* counter. After that, the next 4 bytes contain a count of how many
* additional XIDs follow; this is followed by all of those XIDs one after
* another. We emit the XIDs in sorted order for the convenience of the
* receiving process.
*/ */
void void
SerializeTransactionState(Size maxsize, char *start_address) SerializeTransactionState(Size maxsize, char *start_address)
...@@ -5087,16 +5162,17 @@ SerializeTransactionState(Size maxsize, char *start_address) ...@@ -5087,16 +5162,17 @@ SerializeTransactionState(Size maxsize, char *start_address)
TransactionState s; TransactionState s;
Size nxids = 0; Size nxids = 0;
Size i = 0; Size i = 0;
Size c = 0;
TransactionId *workspace; TransactionId *workspace;
TransactionId *result = (TransactionId *) start_address; SerializedTransactionState *result;
result = (SerializedTransactionState *) start_address;
result[c++] = (TransactionId) XactIsoLevel; result->xactIsoLevel = XactIsoLevel;
result[c++] = (TransactionId) XactDeferrable; result->xactDeferrable = XactDeferrable;
result[c++] = XactTopTransactionId; result->topFullTransactionId = XactTopFullTransactionId;
result[c++] = CurrentTransactionState->transactionId; result->currentFullTransactionId =
result[c++] = (TransactionId) currentCommandId; CurrentTransactionState->fullTransactionId;
Assert(maxsize >= c * sizeof(TransactionId)); result->currentCommandId = currentCommandId;
/* /*
* If we're running in a parallel worker and launching a parallel worker * If we're running in a parallel worker and launching a parallel worker
...@@ -5105,9 +5181,8 @@ SerializeTransactionState(Size maxsize, char *start_address) ...@@ -5105,9 +5181,8 @@ SerializeTransactionState(Size maxsize, char *start_address)
*/ */
if (nParallelCurrentXids > 0) if (nParallelCurrentXids > 0)
{ {
result[c++] = nParallelCurrentXids; result->nParallelCurrentXids = nParallelCurrentXids;
Assert(maxsize >= (nParallelCurrentXids + c) * sizeof(TransactionId)); memcpy(&result->parallelCurrentXids[0], ParallelCurrentXids,
memcpy(&result[c], ParallelCurrentXids,
nParallelCurrentXids * sizeof(TransactionId)); nParallelCurrentXids * sizeof(TransactionId));
return; return;
} }
...@@ -5118,18 +5193,19 @@ SerializeTransactionState(Size maxsize, char *start_address) ...@@ -5118,18 +5193,19 @@ SerializeTransactionState(Size maxsize, char *start_address)
*/ */
for (s = CurrentTransactionState; s != NULL; s = s->parent) for (s = CurrentTransactionState; s != NULL; s = s->parent)
{ {
if (TransactionIdIsValid(s->transactionId)) if (FullTransactionIdIsValid(s->fullTransactionId))
nxids = add_size(nxids, 1); nxids = add_size(nxids, 1);
nxids = add_size(nxids, s->nChildXids); nxids = add_size(nxids, s->nChildXids);
} }
Assert((c + 1 + nxids) * sizeof(TransactionId) <= maxsize); Assert(SerializedTransactionStateHeaderSize + nxids * sizeof(TransactionId)
<= maxsize);
/* Copy them to our scratch space. */ /* Copy them to our scratch space. */
workspace = palloc(nxids * sizeof(TransactionId)); workspace = palloc(nxids * sizeof(TransactionId));
for (s = CurrentTransactionState; s != NULL; s = s->parent) for (s = CurrentTransactionState; s != NULL; s = s->parent)
{ {
if (TransactionIdIsValid(s->transactionId)) if (FullTransactionIdIsValid(s->fullTransactionId))
workspace[i++] = s->transactionId; workspace[i++] = XidFromFullTransactionId(s->fullTransactionId);
memcpy(&workspace[i], s->childXids, memcpy(&workspace[i], s->childXids,
s->nChildXids * sizeof(TransactionId)); s->nChildXids * sizeof(TransactionId));
i += s->nChildXids; i += s->nChildXids;
...@@ -5140,8 +5216,9 @@ SerializeTransactionState(Size maxsize, char *start_address) ...@@ -5140,8 +5216,9 @@ SerializeTransactionState(Size maxsize, char *start_address)
qsort(workspace, nxids, sizeof(TransactionId), xidComparator); qsort(workspace, nxids, sizeof(TransactionId), xidComparator);
/* Copy data into output area. */ /* Copy data into output area. */
result[c++] = (TransactionId) nxids; result->nParallelCurrentXids = nxids;
memcpy(&result[c], workspace, nxids * sizeof(TransactionId)); memcpy(&result->parallelCurrentXids[0], workspace,
nxids * sizeof(TransactionId));
} }
/* /*
...@@ -5152,18 +5229,20 @@ SerializeTransactionState(Size maxsize, char *start_address) ...@@ -5152,18 +5229,20 @@ SerializeTransactionState(Size maxsize, char *start_address)
void void
StartParallelWorkerTransaction(char *tstatespace) StartParallelWorkerTransaction(char *tstatespace)
{ {
TransactionId *tstate = (TransactionId *) tstatespace; SerializedTransactionState *tstate;
Assert(CurrentTransactionState->blockState == TBLOCK_DEFAULT); Assert(CurrentTransactionState->blockState == TBLOCK_DEFAULT);
StartTransaction(); StartTransaction();
XactIsoLevel = (int) tstate[0]; tstate = (SerializedTransactionState *) tstatespace;
XactDeferrable = (bool) tstate[1]; XactIsoLevel = tstate->xactIsoLevel;
XactTopTransactionId = tstate[2]; XactDeferrable = tstate->xactDeferrable;
CurrentTransactionState->transactionId = tstate[3]; XactTopFullTransactionId = tstate->topFullTransactionId;
currentCommandId = tstate[4]; CurrentTransactionState->fullTransactionId =
nParallelCurrentXids = (int) tstate[5]; tstate->currentFullTransactionId;
ParallelCurrentXids = &tstate[6]; currentCommandId = tstate->currentCommandId;
nParallelCurrentXids = tstate->nParallelCurrentXids;
ParallelCurrentXids = &tstate->parallelCurrentXids[0];
CurrentTransactionState->blockState = TBLOCK_PARALLEL_INPROGRESS; CurrentTransactionState->blockState = TBLOCK_PARALLEL_INPROGRESS;
} }
...@@ -5222,7 +5301,7 @@ ShowTransactionStateRec(const char *str, TransactionState s) ...@@ -5222,7 +5301,7 @@ ShowTransactionStateRec(const char *str, TransactionState s)
PointerIsValid(s->name) ? s->name : "unnamed", PointerIsValid(s->name) ? s->name : "unnamed",
BlockStateAsString(s->blockState), BlockStateAsString(s->blockState),
TransStateAsString(s->state), TransStateAsString(s->state),
(unsigned int) s->transactionId, (unsigned int) XidFromFullTransactionId(s->fullTransactionId),
(unsigned int) s->subTransactionId, (unsigned int) s->subTransactionId,
(unsigned int) currentCommandId, (unsigned int) currentCommandId,
currentCommandIdUsed ? " (used)" : "", currentCommandIdUsed ? " (used)" : "",
......
...@@ -49,6 +49,7 @@ ...@@ -49,6 +49,7 @@
#define U64FromFullTransactionId(x) ((x).value) #define U64FromFullTransactionId(x) ((x).value)
#define FullTransactionIdPrecedes(a, b) ((a).value < (b).value) #define FullTransactionIdPrecedes(a, b) ((a).value < (b).value)
#define FullTransactionIdIsValid(x) TransactionIdIsValid(XidFromFullTransactionId(x)) #define FullTransactionIdIsValid(x) TransactionIdIsValid(XidFromFullTransactionId(x))
#define InvalidFullTransactionId FullTransactionIdFromEpochAndXid(0, InvalidTransactionId)
/* /*
* A 64 bit value that contains an epoch and a TransactionId. This is * A 64 bit value that contains an epoch and a TransactionId. This is
...@@ -221,7 +222,7 @@ extern TransactionId TransactionIdLatest(TransactionId mainxid, ...@@ -221,7 +222,7 @@ extern TransactionId TransactionIdLatest(TransactionId mainxid,
extern XLogRecPtr TransactionIdGetCommitLSN(TransactionId xid); extern XLogRecPtr TransactionIdGetCommitLSN(TransactionId xid);
/* in transam/varsup.c */ /* in transam/varsup.c */
extern TransactionId GetNewTransactionId(bool isSubXact); extern FullTransactionId GetNewTransactionId(bool isSubXact);
extern void AdvanceNextFullTransactionIdPastXid(TransactionId xid); extern void AdvanceNextFullTransactionIdPastXid(TransactionId xid);
extern FullTransactionId ReadNextFullTransactionId(void); extern FullTransactionId ReadNextFullTransactionId(void);
extern void SetTransactionIdLimit(TransactionId oldest_datfrozenxid, extern void SetTransactionIdLimit(TransactionId oldest_datfrozenxid,
......
...@@ -14,6 +14,7 @@ ...@@ -14,6 +14,7 @@
#ifndef XACT_H #ifndef XACT_H
#define XACT_H #define XACT_H
#include "access/transam.h"
#include "access/xlogreader.h" #include "access/xlogreader.h"
#include "lib/stringinfo.h" #include "lib/stringinfo.h"
#include "nodes/pg_list.h" #include "nodes/pg_list.h"
...@@ -355,6 +356,10 @@ extern TransactionId GetCurrentTransactionId(void); ...@@ -355,6 +356,10 @@ extern TransactionId GetCurrentTransactionId(void);
extern TransactionId GetCurrentTransactionIdIfAny(void); extern TransactionId GetCurrentTransactionIdIfAny(void);
extern TransactionId GetStableLatestTransactionId(void); extern TransactionId GetStableLatestTransactionId(void);
extern SubTransactionId GetCurrentSubTransactionId(void); extern SubTransactionId GetCurrentSubTransactionId(void);
extern FullTransactionId GetTopFullTransactionId(void);
extern FullTransactionId GetTopFullTransactionIdIfAny(void);
extern FullTransactionId GetCurrentFullTransactionId(void);
extern FullTransactionId GetCurrentFullTransactionIdIfAny(void);
extern void MarkCurrentTransactionIdLoggedIfAny(void); extern void MarkCurrentTransactionIdLoggedIfAny(void);
extern bool SubTransactionIsActive(SubTransactionId subxid); extern bool SubTransactionIsActive(SubTransactionId subxid);
extern CommandId GetCurrentCommandId(bool used); extern CommandId GetCurrentCommandId(bool used);
......
...@@ -2107,6 +2107,7 @@ SeqTableData ...@@ -2107,6 +2107,7 @@ SeqTableData
SerCommitSeqNo SerCommitSeqNo
SerializedReindexState SerializedReindexState
SerializedSnapshotData SerializedSnapshotData
SerializedTransactionState
Session Session
SessionBackupState SessionBackupState
SetConstraintState SetConstraintState
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment