mirror of
https://git.postgresql.org/git/postgresql.git
synced 2024-12-21 08:29:39 +08:00
Fix corner case in cleanup of transactions using SSI.
When the only remaining active transactions are READ ONLY, we do a "partial cleanup" of committed transactions because certain types of conflicts aren't possible anymore. For committed r/w transactions, we release the SIREAD locks but keep the SERIALIZABLEXACT. However, for committed r/o transactions, we can go further and release the SERIALIZABLEXACT too. The problem was with the latter case: we were returning the SERIALIZABLEXACT to the free list without removing it from the finished list. The only real change in the patch is the SHMQueueDelete line, but I also reworked some of the surrounding code to make it obvious that r/o and r/w transactions are handled differently -- the existing code felt a bit too clever. Dan Ports
This commit is contained in:
parent
2106c55ac8
commit
326b922e8b
@ -3528,10 +3528,29 @@ ClearOldPredicateLocks(void)
|
||||
else if (finishedSxact->commitSeqNo > PredXact->HavePartialClearedThrough
|
||||
&& finishedSxact->commitSeqNo <= PredXact->CanPartialClearThrough)
|
||||
{
|
||||
/*
|
||||
* Any active transactions that took their snapshot before this
|
||||
* transaction committed are read-only, so we can clear part of
|
||||
* its state.
|
||||
*/
|
||||
LWLockRelease(SerializableXactHashLock);
|
||||
ReleaseOneSerializableXact(finishedSxact,
|
||||
!SxactIsReadOnly(finishedSxact),
|
||||
false);
|
||||
|
||||
if (SxactIsReadOnly(finishedSxact))
|
||||
{
|
||||
/* A read-only transaction can be removed entirely */
|
||||
SHMQueueDelete(&(finishedSxact->finishedLink));
|
||||
ReleaseOneSerializableXact(finishedSxact, false, false);
|
||||
}
|
||||
else
|
||||
{
|
||||
/*
|
||||
* A read-write transaction can only be partially
|
||||
* cleared. We need to keep the SERIALIZABLEXACT but
|
||||
* can release the SIREAD locks and conflicts in.
|
||||
*/
|
||||
ReleaseOneSerializableXact(finishedSxact, true, false);
|
||||
}
|
||||
|
||||
PredXact->HavePartialClearedThrough = finishedSxact->commitSeqNo;
|
||||
LWLockAcquire(SerializableXactHashLock, LW_SHARED);
|
||||
}
|
||||
@ -3637,6 +3656,7 @@ ReleaseOneSerializableXact(SERIALIZABLEXACT *sxact, bool partial,
|
||||
|
||||
Assert(sxact != NULL);
|
||||
Assert(SxactIsRolledBack(sxact) || SxactIsCommitted(sxact));
|
||||
Assert(partial || !SxactIsOnFinishedList(sxact));
|
||||
Assert(LWLockHeldByMe(SerializableFinishedListLock));
|
||||
|
||||
/*
|
||||
|
Loading…
Reference in New Issue
Block a user