Commit 10520f43 authored by Robert Haas's avatar Robert Haas

Rethink the delay-checkpoint-end mechanism in the back-branches.

The back-patch of commit bbace569 had
the unfortunate effect of changing the layout of PGPROC in the
back-branches, which could break extensions. This happened because it
changed the delayChkpt from type bool to type int. So, change it back,
and add a new bool delayChkptEnd field instead. The new field should
fall within what used to be padding space within the struct, and so
hopefully won't cause any extensions to break.

Per report from Markus Wanner and discussion with Tom Lane and others.

Patch originally by me, somewhat revised by Markus Wanner per a
suggestion from Michael Paquier. A very similar patch was developed
by Kyotaro Horiguchi, but I failed to see the email in which that was
posted before writing one of my own.

Discussion: http://postgr.es/m/CA+Tgmoao-kUD9c5nG5sub3F7tbo39+cdr8jKaOVEs_1aBWcJ3Q@mail.gmail.com
Discussion: http://postgr.es/m/20220406.164521.17171257901083417.horikyota.ntt@gmail.com
parent df6bbe73
...@@ -3075,8 +3075,8 @@ TruncateMultiXact(MultiXactId newOldestMulti, Oid newOldestMultiDB) ...@@ -3075,8 +3075,8 @@ TruncateMultiXact(MultiXactId newOldestMulti, Oid newOldestMultiDB)
* crash/basebackup, even though the state of the data directory would * crash/basebackup, even though the state of the data directory would
* require it. * require it.
*/ */
Assert((MyProc->delayChkpt & DELAY_CHKPT_START) == 0); Assert(!MyProc->delayChkpt);
MyProc->delayChkpt |= DELAY_CHKPT_START; MyProc->delayChkpt = true;
/* WAL log truncation */ /* WAL log truncation */
WriteMTruncateXlogRec(newOldestMultiDB, WriteMTruncateXlogRec(newOldestMultiDB,
...@@ -3102,7 +3102,7 @@ TruncateMultiXact(MultiXactId newOldestMulti, Oid newOldestMultiDB) ...@@ -3102,7 +3102,7 @@ TruncateMultiXact(MultiXactId newOldestMulti, Oid newOldestMultiDB)
/* Then offsets */ /* Then offsets */
PerformOffsetsTruncation(oldestMulti, newOldestMulti); PerformOffsetsTruncation(oldestMulti, newOldestMulti);
MyProc->delayChkpt &= ~DELAY_CHKPT_START; MyProc->delayChkpt = false;
END_CRIT_SECTION(); END_CRIT_SECTION();
LWLockRelease(MultiXactTruncationLock); LWLockRelease(MultiXactTruncationLock);
......
...@@ -474,8 +474,9 @@ MarkAsPreparingGuts(GlobalTransaction gxact, TransactionId xid, const char *gid, ...@@ -474,8 +474,9 @@ MarkAsPreparingGuts(GlobalTransaction gxact, TransactionId xid, const char *gid,
} }
proc->xid = xid; proc->xid = xid;
Assert(proc->xmin == InvalidTransactionId); Assert(proc->xmin == InvalidTransactionId);
proc->delayChkpt = 0; proc->delayChkpt = false;
proc->statusFlags = 0; proc->statusFlags = 0;
proc->delayChkptEnd = false;
proc->pid = 0; proc->pid = 0;
proc->databaseId = databaseid; proc->databaseId = databaseid;
proc->roleId = owner; proc->roleId = owner;
...@@ -1165,8 +1166,8 @@ EndPrepare(GlobalTransaction gxact) ...@@ -1165,8 +1166,8 @@ EndPrepare(GlobalTransaction gxact)
START_CRIT_SECTION(); START_CRIT_SECTION();
Assert((MyProc->delayChkpt & DELAY_CHKPT_START) == 0); Assert(!MyProc->delayChkpt);
MyProc->delayChkpt |= DELAY_CHKPT_START; MyProc->delayChkpt = true;
XLogBeginInsert(); XLogBeginInsert();
for (record = records.head; record != NULL; record = record->next) for (record = records.head; record != NULL; record = record->next)
...@@ -1209,7 +1210,7 @@ EndPrepare(GlobalTransaction gxact) ...@@ -1209,7 +1210,7 @@ EndPrepare(GlobalTransaction gxact)
* checkpoint starting after this will certainly see the gxact as a * checkpoint starting after this will certainly see the gxact as a
* candidate for fsyncing. * candidate for fsyncing.
*/ */
MyProc->delayChkpt &= ~DELAY_CHKPT_START; MyProc->delayChkpt = false;
/* /*
* Remember that we have this GlobalTransaction entry locked for us. If * Remember that we have this GlobalTransaction entry locked for us. If
...@@ -2276,8 +2277,8 @@ RecordTransactionCommitPrepared(TransactionId xid, ...@@ -2276,8 +2277,8 @@ RecordTransactionCommitPrepared(TransactionId xid,
START_CRIT_SECTION(); START_CRIT_SECTION();
/* See notes in RecordTransactionCommit */ /* See notes in RecordTransactionCommit */
Assert((MyProc->delayChkpt & DELAY_CHKPT_START) == 0); Assert(!MyProc->delayChkpt);
MyProc->delayChkpt |= DELAY_CHKPT_START; MyProc->delayChkpt = true;
/* /*
* Emit the XLOG commit record. Note that we mark 2PC commits as * Emit the XLOG commit record. Note that we mark 2PC commits as
...@@ -2325,7 +2326,7 @@ RecordTransactionCommitPrepared(TransactionId xid, ...@@ -2325,7 +2326,7 @@ RecordTransactionCommitPrepared(TransactionId xid,
TransactionIdCommitTree(xid, nchildren, children); TransactionIdCommitTree(xid, nchildren, children);
/* Checkpoint can proceed now */ /* Checkpoint can proceed now */
MyProc->delayChkpt &= ~DELAY_CHKPT_START; MyProc->delayChkpt = false;
END_CRIT_SECTION(); END_CRIT_SECTION();
......
...@@ -1335,9 +1335,9 @@ RecordTransactionCommit(void) ...@@ -1335,9 +1335,9 @@ RecordTransactionCommit(void)
* This makes checkpoint's determination of which xacts are delayChkpt * This makes checkpoint's determination of which xacts are delayChkpt
* a bit fuzzy, but it doesn't matter. * a bit fuzzy, but it doesn't matter.
*/ */
Assert((MyProc->delayChkpt & DELAY_CHKPT_START) == 0); Assert(!MyProc->delayChkpt);
START_CRIT_SECTION(); START_CRIT_SECTION();
MyProc->delayChkpt |= DELAY_CHKPT_START; MyProc->delayChkpt = true;
SetCurrentTransactionStopTimestamp(); SetCurrentTransactionStopTimestamp();
...@@ -1438,7 +1438,7 @@ RecordTransactionCommit(void) ...@@ -1438,7 +1438,7 @@ RecordTransactionCommit(void)
*/ */
if (markXidCommitted) if (markXidCommitted)
{ {
MyProc->delayChkpt &= ~DELAY_CHKPT_START; MyProc->delayChkpt = false;
END_CRIT_SECTION(); END_CRIT_SECTION();
} }
......
...@@ -9228,27 +9228,25 @@ CreateCheckPoint(int flags) ...@@ -9228,27 +9228,25 @@ CreateCheckPoint(int flags)
* and we will correctly flush the update below. So we cannot miss any * and we will correctly flush the update below. So we cannot miss any
* xacts we need to wait for. * xacts we need to wait for.
*/ */
vxids = GetVirtualXIDsDelayingChkpt(&nvxids, DELAY_CHKPT_START); vxids = GetVirtualXIDsDelayingChkpt(&nvxids);
if (nvxids > 0) if (nvxids > 0)
{ {
do do
{ {
pg_usleep(10000L); /* wait for 10 msec */ pg_usleep(10000L); /* wait for 10 msec */
} while (HaveVirtualXIDsDelayingChkpt(vxids, nvxids, } while (HaveVirtualXIDsDelayingChkpt(vxids, nvxids));
DELAY_CHKPT_START));
} }
pfree(vxids); pfree(vxids);
CheckPointGuts(checkPoint.redo, flags); CheckPointGuts(checkPoint.redo, flags);
vxids = GetVirtualXIDsDelayingChkpt(&nvxids, DELAY_CHKPT_COMPLETE); vxids = GetVirtualXIDsDelayingChkptEnd(&nvxids);
if (nvxids > 0) if (nvxids > 0)
{ {
do do
{ {
pg_usleep(10000L); /* wait for 10 msec */ pg_usleep(10000L); /* wait for 10 msec */
} while (HaveVirtualXIDsDelayingChkpt(vxids, nvxids, } while (HaveVirtualXIDsDelayingChkptEnd(vxids, nvxids));
DELAY_CHKPT_COMPLETE));
} }
pfree(vxids); pfree(vxids);
......
...@@ -925,7 +925,7 @@ XLogSaveBufferForHint(Buffer buffer, bool buffer_std) ...@@ -925,7 +925,7 @@ XLogSaveBufferForHint(Buffer buffer, bool buffer_std)
/* /*
* Ensure no checkpoint can change our view of RedoRecPtr. * Ensure no checkpoint can change our view of RedoRecPtr.
*/ */
Assert((MyProc->delayChkpt & DELAY_CHKPT_START) != 0); Assert(MyProc->delayChkpt);
/* /*
* Update RedoRecPtr so that we can make the right decision * Update RedoRecPtr so that we can make the right decision
......
...@@ -338,8 +338,8 @@ RelationTruncate(Relation rel, BlockNumber nblocks) ...@@ -338,8 +338,8 @@ RelationTruncate(Relation rel, BlockNumber nblocks)
* the blocks to not exist on disk at all, but not for them to have the * the blocks to not exist on disk at all, but not for them to have the
* wrong contents. * wrong contents.
*/ */
Assert((MyProc->delayChkpt & DELAY_CHKPT_COMPLETE) == 0); Assert(!MyProc->delayChkptEnd);
MyProc->delayChkpt |= DELAY_CHKPT_COMPLETE; MyProc->delayChkptEnd = true;
/* /*
* We WAL-log the truncation before actually truncating, which means * We WAL-log the truncation before actually truncating, which means
...@@ -387,7 +387,7 @@ RelationTruncate(Relation rel, BlockNumber nblocks) ...@@ -387,7 +387,7 @@ RelationTruncate(Relation rel, BlockNumber nblocks)
smgrtruncate(rel->rd_smgr, forks, nforks, blocks); smgrtruncate(rel->rd_smgr, forks, nforks, blocks);
/* We've done all the critical work, so checkpoints are OK now. */ /* We've done all the critical work, so checkpoints are OK now. */
MyProc->delayChkpt &= ~DELAY_CHKPT_COMPLETE; MyProc->delayChkptEnd = false;
/* /*
* Update upper-level FSM pages to account for the truncation. This is * Update upper-level FSM pages to account for the truncation. This is
......
...@@ -3946,8 +3946,8 @@ MarkBufferDirtyHint(Buffer buffer, bool buffer_std) ...@@ -3946,8 +3946,8 @@ MarkBufferDirtyHint(Buffer buffer, bool buffer_std)
* essential that CreateCheckpoint waits for virtual transactions * essential that CreateCheckpoint waits for virtual transactions
* rather than full transactionids. * rather than full transactionids.
*/ */
Assert((MyProc->delayChkpt & DELAY_CHKPT_START) == 0); Assert(!MyProc->delayChkpt);
MyProc->delayChkpt |= DELAY_CHKPT_START; MyProc->delayChkpt = true;
delayChkpt = true; delayChkpt = true;
lsn = XLogSaveBufferForHint(buffer, buffer_std); lsn = XLogSaveBufferForHint(buffer, buffer_std);
} }
...@@ -3981,7 +3981,7 @@ MarkBufferDirtyHint(Buffer buffer, bool buffer_std) ...@@ -3981,7 +3981,7 @@ MarkBufferDirtyHint(Buffer buffer, bool buffer_std)
UnlockBufHdr(bufHdr, buf_state); UnlockBufHdr(bufHdr, buf_state);
if (delayChkpt) if (delayChkpt)
MyProc->delayChkpt &= ~DELAY_CHKPT_START; MyProc->delayChkpt = false;
if (dirtied) if (dirtied)
{ {
......
...@@ -330,6 +330,11 @@ static void DisplayXidCache(void); ...@@ -330,6 +330,11 @@ static void DisplayXidCache(void);
#define xc_slow_answer_inc() ((void) 0) #define xc_slow_answer_inc() ((void) 0)
#endif /* XIDCACHE_DEBUG */ #endif /* XIDCACHE_DEBUG */
static VirtualTransactionId *GetVirtualXIDsDelayingChkptGuts(int *nvxids,
int type);
static bool HaveVirtualXIDsDelayingChkptGuts(VirtualTransactionId *vxids,
int nvxids, int type);
/* Primitives for KnownAssignedXids array handling for standby */ /* Primitives for KnownAssignedXids array handling for standby */
static void KnownAssignedXidsCompress(bool force); static void KnownAssignedXidsCompress(bool force);
static void KnownAssignedXidsAdd(TransactionId from_xid, TransactionId to_xid, static void KnownAssignedXidsAdd(TransactionId from_xid, TransactionId to_xid,
...@@ -690,8 +695,9 @@ ProcArrayEndTransaction(PGPROC *proc, TransactionId latestXid) ...@@ -690,8 +695,9 @@ ProcArrayEndTransaction(PGPROC *proc, TransactionId latestXid)
proc->lxid = InvalidLocalTransactionId; proc->lxid = InvalidLocalTransactionId;
proc->xmin = InvalidTransactionId; proc->xmin = InvalidTransactionId;
/* be sure this is cleared in abort */ /* be sure these are cleared in abort */
proc->delayChkpt = 0; proc->delayChkpt = false;
proc->delayChkptEnd = false;
proc->recoveryConflictPending = false; proc->recoveryConflictPending = false;
...@@ -732,8 +738,9 @@ ProcArrayEndTransactionInternal(PGPROC *proc, TransactionId latestXid) ...@@ -732,8 +738,9 @@ ProcArrayEndTransactionInternal(PGPROC *proc, TransactionId latestXid)
proc->lxid = InvalidLocalTransactionId; proc->lxid = InvalidLocalTransactionId;
proc->xmin = InvalidTransactionId; proc->xmin = InvalidTransactionId;
/* be sure this is cleared in abort */ /* be sure these are cleared in abort */
proc->delayChkpt = 0; proc->delayChkpt = false;
proc->delayChkptEnd = false;
proc->recoveryConflictPending = false; proc->recoveryConflictPending = false;
...@@ -3045,26 +3052,28 @@ GetOldestSafeDecodingTransactionId(bool catalogOnly) ...@@ -3045,26 +3052,28 @@ GetOldestSafeDecodingTransactionId(bool catalogOnly)
} }
/* /*
* GetVirtualXIDsDelayingChkpt -- Get the VXIDs of transactions that are * GetVirtualXIDsDelayingChkptGuts -- Get the VXIDs of transactions that are
* delaying checkpoint because they have critical actions in progress. * delaying the start or end of a checkpoint because they have critical
* actions in progress.
* *
* Constructs an array of VXIDs of transactions that are currently in commit * Constructs an array of VXIDs of transactions that are currently in commit
* critical sections, as shown by having specified delayChkpt bits set in their * critical sections, as shown by having delayChkpt or delayChkptEnd set in
* PGPROC. * their PGPROC.
* *
* Returns a palloc'd array that should be freed by the caller. * Returns a palloc'd array that should be freed by the caller.
* *nvxids is the number of valid entries. * *nvxids is the number of valid entries.
* *
* Note that because backends set or clear delayChkpt without holding any lock, * Note that because backends set or clear delayChkpt and delayChkptEnd
* the result is somewhat indeterminate, but we don't really care. Even in * without holding any lock, the result is somewhat indeterminate, but we
* a multiprocessor with delayed writes to shared memory, it should be certain * don't really care. Even in a multiprocessor with delayed writes to
* that setting of delayChkpt will propagate to shared memory when the backend * shared memory, it should be certain that setting of delayChkpt will
* takes a lock, so we cannot fail to see a virtual xact as delayChkpt if * propagate to shared memory when the backend takes a lock, so we cannot
* it's already inserted its commit record. Whether it takes a little while * fail to see a virtual xact as delayChkpt if it's already inserted its
* for clearing of delayChkpt to propagate is unimportant for correctness. * commit record. Whether it takes a little while for clearing of
* delayChkpt to propagate is unimportant for correctness.
*/ */
VirtualTransactionId * static VirtualTransactionId *
GetVirtualXIDsDelayingChkpt(int *nvxids, int type) GetVirtualXIDsDelayingChkptGuts(int *nvxids, int type)
{ {
VirtualTransactionId *vxids; VirtualTransactionId *vxids;
ProcArrayStruct *arrayP = procArray; ProcArrayStruct *arrayP = procArray;
...@@ -3084,7 +3093,8 @@ GetVirtualXIDsDelayingChkpt(int *nvxids, int type) ...@@ -3084,7 +3093,8 @@ GetVirtualXIDsDelayingChkpt(int *nvxids, int type)
int pgprocno = arrayP->pgprocnos[index]; int pgprocno = arrayP->pgprocnos[index];
PGPROC *proc = &allProcs[pgprocno]; PGPROC *proc = &allProcs[pgprocno];
if ((proc->delayChkpt & type) != 0) if (((type & DELAY_CHKPT_START) && proc->delayChkpt) ||
((type & DELAY_CHKPT_COMPLETE) && proc->delayChkptEnd))
{ {
VirtualTransactionId vxid; VirtualTransactionId vxid;
...@@ -3100,6 +3110,26 @@ GetVirtualXIDsDelayingChkpt(int *nvxids, int type) ...@@ -3100,6 +3110,26 @@ GetVirtualXIDsDelayingChkpt(int *nvxids, int type)
return vxids; return vxids;
} }
/*
* GetVirtualXIDsDelayingChkpt - Get the VXIDs of transactions that are
* delaying the start of a checkpoint.
*/
VirtualTransactionId *
GetVirtualXIDsDelayingChkpt(int *nvxids)
{
return GetVirtualXIDsDelayingChkptGuts(nvxids, DELAY_CHKPT_START);
}
/*
* GetVirtualXIDsDelayingChkptEnd - Get the VXIDs of transactions that are
* delaying the end of a checkpoint.
*/
VirtualTransactionId *
GetVirtualXIDsDelayingChkptEnd(int *nvxids)
{
return GetVirtualXIDsDelayingChkptGuts(nvxids, DELAY_CHKPT_COMPLETE);
}
/* /*
* HaveVirtualXIDsDelayingChkpt -- Are any of the specified VXIDs delaying? * HaveVirtualXIDsDelayingChkpt -- Are any of the specified VXIDs delaying?
* *
...@@ -3109,8 +3139,9 @@ GetVirtualXIDsDelayingChkpt(int *nvxids, int type) ...@@ -3109,8 +3139,9 @@ GetVirtualXIDsDelayingChkpt(int *nvxids, int type)
* Note: this is O(N^2) in the number of vxacts that are/were delaying, but * Note: this is O(N^2) in the number of vxacts that are/were delaying, but
* those numbers should be small enough for it not to be a problem. * those numbers should be small enough for it not to be a problem.
*/ */
bool static bool
HaveVirtualXIDsDelayingChkpt(VirtualTransactionId *vxids, int nvxids, int type) HaveVirtualXIDsDelayingChkptGuts(VirtualTransactionId *vxids, int nvxids,
int type)
{ {
bool result = false; bool result = false;
ProcArrayStruct *arrayP = procArray; ProcArrayStruct *arrayP = procArray;
...@@ -3128,7 +3159,8 @@ HaveVirtualXIDsDelayingChkpt(VirtualTransactionId *vxids, int nvxids, int type) ...@@ -3128,7 +3159,8 @@ HaveVirtualXIDsDelayingChkpt(VirtualTransactionId *vxids, int nvxids, int type)
GET_VXID_FROM_PGPROC(vxid, *proc); GET_VXID_FROM_PGPROC(vxid, *proc);
if ((proc->delayChkpt & type) != 0 && if ((((type & DELAY_CHKPT_START) && proc->delayChkpt) ||
((type & DELAY_CHKPT_COMPLETE) && proc->delayChkptEnd)) &&
VirtualTransactionIdIsValid(vxid)) VirtualTransactionIdIsValid(vxid))
{ {
int i; int i;
...@@ -3151,6 +3183,28 @@ HaveVirtualXIDsDelayingChkpt(VirtualTransactionId *vxids, int nvxids, int type) ...@@ -3151,6 +3183,28 @@ HaveVirtualXIDsDelayingChkpt(VirtualTransactionId *vxids, int nvxids, int type)
return result; return result;
} }
/*
* HaveVirtualXIDsDelayingChkpt -- Are any of the specified VXIDs delaying
* the start of a checkpoint?
*/
bool
HaveVirtualXIDsDelayingChkpt(VirtualTransactionId *vxids, int nvxids)
{
return HaveVirtualXIDsDelayingChkptGuts(vxids, nvxids,
DELAY_CHKPT_START);
}
/*
* HaveVirtualXIDsDelayingChkptEnd -- Are any of the specified VXIDs delaying
* the end of a checkpoint?
*/
bool
HaveVirtualXIDsDelayingChkptEnd(VirtualTransactionId *vxids, int nvxids)
{
return HaveVirtualXIDsDelayingChkptGuts(vxids, nvxids,
DELAY_CHKPT_COMPLETE);
}
/* /*
* BackendPidGetProc -- get a backend's PGPROC given its PID * BackendPidGetProc -- get a backend's PGPROC given its PID
* *
......
...@@ -87,36 +87,8 @@ struct XidCache ...@@ -87,36 +87,8 @@ struct XidCache
#define INVALID_PGPROCNO PG_INT32_MAX #define INVALID_PGPROCNO PG_INT32_MAX
/* /*
* Flags for PGPROC.delayChkpt * Flags used only for type of internal functions
* * GetVirtualXIDsDelayingChkptGuts and HaveVirtualXIDsDelayingChkptGuts.
* These flags can be used to delay the start or completion of a checkpoint
* for short periods. A flag is in effect if the corresponding bit is set in
* the PGPROC of any backend.
*
* For our purposes here, a checkpoint has three phases: (1) determine the
* location to which the redo pointer will be moved, (2) write all the
* data durably to disk, and (3) WAL-log the checkpoint.
*
* Setting DELAY_CHKPT_START prevents the system from moving from phase 1
* to phase 2. This is useful when we are performing a WAL-logged modification
* of data that will be flushed to disk in phase 2. By setting this flag
* before writing WAL and clearing it after we've both written WAL and
* performed the corresponding modification, we ensure that if the WAL record
* is inserted prior to the new redo point, the corresponding data changes will
* also be flushed to disk before the checkpoint can complete. (In the
* extremely common case where the data being modified is in shared buffers
* and we acquire an exclusive content lock on the relevant buffers before
* writing WAL, this mechanism is not needed, because phase 2 will block
* until we release the content lock and then flush the modified data to
* disk.)
*
* Setting DELAY_CHKPT_COMPLETE prevents the system from moving from phase 2
* to phase 3. This is useful if we are performing a WAL-logged operation that
* might invalidate buffers, such as relation truncation. In this case, we need
* to ensure that any buffers which were invalidated and thus not flushed by
* the checkpoint are actaully destroyed on disk. Replay can cope with a file
* or block that doesn't exist, but not with a block that has the wrong
* contents.
*/ */
#define DELAY_CHKPT_START (1<<0) #define DELAY_CHKPT_START (1<<0)
#define DELAY_CHKPT_COMPLETE (1<<1) #define DELAY_CHKPT_COMPLETE (1<<1)
...@@ -226,11 +198,12 @@ struct PGPROC ...@@ -226,11 +198,12 @@ struct PGPROC
pg_atomic_uint64 waitStart; /* time at which wait for lock acquisition pg_atomic_uint64 waitStart; /* time at which wait for lock acquisition
* started */ * started */
int delayChkpt; /* for DELAY_CHKPT_* flags */ bool delayChkpt; /* true if this proc delays checkpoint start */
uint8 statusFlags; /* this backend's status flags, see PROC_* uint8 statusFlags; /* this backend's status flags, see PROC_*
* above. mirrored in * above. mirrored in
* ProcGlobal->statusFlags[pgxactoff] */ * ProcGlobal->statusFlags[pgxactoff] */
bool delayChkptEnd; /* true if this proc delays checkpoint end */
/* /*
* Info to allow us to wait for synchronous replication, if needed. * Info to allow us to wait for synchronous replication, if needed.
......
...@@ -59,9 +59,12 @@ extern TransactionId GetOldestActiveTransactionId(void); ...@@ -59,9 +59,12 @@ extern TransactionId GetOldestActiveTransactionId(void);
extern TransactionId GetOldestSafeDecodingTransactionId(bool catalogOnly); extern TransactionId GetOldestSafeDecodingTransactionId(bool catalogOnly);
extern void GetReplicationHorizons(TransactionId *slot_xmin, TransactionId *catalog_xmin); extern void GetReplicationHorizons(TransactionId *slot_xmin, TransactionId *catalog_xmin);
extern VirtualTransactionId *GetVirtualXIDsDelayingChkpt(int *nvxids, int type); extern VirtualTransactionId *GetVirtualXIDsDelayingChkpt(int *nvxids);
extern VirtualTransactionId *GetVirtualXIDsDelayingChkptEnd(int *nvxids);
extern bool HaveVirtualXIDsDelayingChkpt(VirtualTransactionId *vxids, extern bool HaveVirtualXIDsDelayingChkpt(VirtualTransactionId *vxids,
int nvxids, int type); int nvxids);
extern bool HaveVirtualXIDsDelayingChkptEnd(VirtualTransactionId *vxids,
int nvxids);
extern PGPROC *BackendPidGetProc(int pid); extern PGPROC *BackendPidGetProc(int pid);
extern PGPROC *BackendPidGetProcWithLock(int pid); extern PGPROC *BackendPidGetProcWithLock(int pid);
......
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