Commit 27846f02 authored by Alvaro Herrera's avatar Alvaro Herrera

Optimize locking a tuple already locked by another subxact

Locking and updating the same tuple repeatedly led to some strange
multixacts being created which had several subtransactions of the same
parent transaction holding locks of the same strength.  However,
once a subxact of the current transaction holds a lock of a given
strength, it's not necessary to acquire the same lock again.  This made
some coding patterns much slower than required.

The fix is twofold.  First we change HeapTupleSatisfiesUpdate to return
HeapTupleBeingUpdated for the case where the current transaction is
already a single-xid locker for the given tuple; it used to return
HeapTupleMayBeUpdated for that case.  The new logic is simpler, and the
change to pgrowlocks is a testament to that: previously we needed to
check for the single-xid locker separately in a very ugly way.  That
test is simpler now.

As fallout from the HTSU change, some of its callers need to be amended
so that tuple-locked-by-own-transaction is taken into account in the
BeingUpdated case rather than the MayBeUpdated case.  For many of them
there is no difference; but heap_delete() and heap_update now check
explicitely and do not grab tuple lock in that case.

The HTSU change also means that routine MultiXactHasRunningRemoteMembers
introduced in commit 11ac4c73 is no longer necessary and can be
removed; the case that used to require it is now handled naturally as
result of the changes to heap_delete and heap_update.

The second part of the fix to the performance issue is to adjust
heap_lock_tuple to avoid the slowness:

1. Previously we checked for the case that our own transaction already
held a strong enough lock and returned MayBeUpdated, but only in the
multixact case.  Now we do it for the plain Xid case as well, which
saves having to LockTuple.

2. If the current transaction is the only locker of the tuple (but with
a lock not as strong as what we need; otherwise it would have been
caught in the check mentioned above), we can skip sleeping on the
multixact, and instead go straight to create an updated multixact with
the additional lock strength.

3. Most importantly, make sure that both the single-xid-locker case and
the multixact-locker case optimization are applied always.  We do this
by checking both in a single place, rather than them appearing in two
separate portions of the routine -- something that is made possible by
the HeapTupleSatisfiesUpdate API change.  Previously we would only check
for the single-xid case when HTSU returned MayBeUpdated, and only
checked for the multixact case when HTSU returned BeingUpdated.  This
was at odds with what HTSU actually returned in one case: if our own
transaction was locker in a multixact, it returned MayBeUpdated, so the
optimization never applied.  This is what led to the large multixacts in
the first place.

Per bug report #8470 by Oskari Saarenmaa.
parent 8a0d34e4
...@@ -136,14 +136,9 @@ pgrowlocks(PG_FUNCTION_ARGS) ...@@ -136,14 +136,9 @@ pgrowlocks(PG_FUNCTION_ARGS)
infomask = tuple->t_data->t_infomask; infomask = tuple->t_data->t_infomask;
/* /*
* a tuple is locked if HTSU returns BeingUpdated, and if it returns * A tuple is locked if HTSU returns BeingUpdated.
* MayBeUpdated but the Xmax is valid and pointing at us.
*/ */
if (htsu == HeapTupleBeingUpdated || if (htsu == HeapTupleBeingUpdated)
(htsu == HeapTupleMayBeUpdated &&
!(infomask & HEAP_XMAX_INVALID) &&
!(infomask & HEAP_XMAX_IS_MULTI) &&
(xmax == GetCurrentTransactionIdIfAny())))
{ {
char **values; char **values;
......
This diff is collapsed.
...@@ -533,7 +533,7 @@ MultiXactIdIsRunning(MultiXactId multi, bool isLockOnly) ...@@ -533,7 +533,7 @@ MultiXactIdIsRunning(MultiXactId multi, bool isLockOnly)
*/ */
nmembers = GetMultiXactIdMembers(multi, &members, false, isLockOnly); nmembers = GetMultiXactIdMembers(multi, &members, false, isLockOnly);
if (nmembers < 0) if (nmembers <= 0)
{ {
debug_elog2(DEBUG2, "IsRunning: no members"); debug_elog2(DEBUG2, "IsRunning: no members");
return false; return false;
...@@ -1337,39 +1337,6 @@ retry: ...@@ -1337,39 +1337,6 @@ retry:
return truelength; return truelength;
} }
/*
* MultiXactHasRunningRemoteMembers
* Does the given multixact have still-live members from
* transactions other than our own?
*/
bool
MultiXactHasRunningRemoteMembers(MultiXactId multi)
{
MultiXactMember *members;
int nmembers;
int i;
nmembers = GetMultiXactIdMembers(multi, &members, true, false);
if (nmembers <= 0)
return false;
for (i = 0; i < nmembers; i++)
{
/* not interested in our own members */
if (TransactionIdIsCurrentTransactionId(members[i].xid))
continue;
if (TransactionIdIsInProgress(members[i].xid))
{
pfree(members);
return true;
}
}
pfree(members);
return false;
}
/* /*
* mxactMemberComparator * mxactMemberComparator
* qsort comparison function for MultiXactMember * qsort comparison function for MultiXactMember
......
...@@ -514,20 +514,20 @@ HeapTupleSatisfiesUpdate(HeapTuple htup, CommandId curcid, ...@@ -514,20 +514,20 @@ HeapTupleSatisfiesUpdate(HeapTuple htup, CommandId curcid,
if (tuple->t_infomask & HEAP_XMAX_IS_MULTI) if (tuple->t_infomask & HEAP_XMAX_IS_MULTI)
{ {
if (MultiXactHasRunningRemoteMembers(xmax)) if (MultiXactIdIsRunning(xmax, true))
return HeapTupleBeingUpdated; return HeapTupleBeingUpdated;
else else
return HeapTupleMayBeUpdated; return HeapTupleMayBeUpdated;
} }
/* if locker is gone, all's well */ /*
* If the locker is gone, then there is nothing of interest
* left in this Xmax; otherwise, report the tuple as
* locked/updated.
*/
if (!TransactionIdIsInProgress(xmax)) if (!TransactionIdIsInProgress(xmax))
return HeapTupleMayBeUpdated; return HeapTupleMayBeUpdated;
if (!TransactionIdIsCurrentTransactionId(xmax))
return HeapTupleBeingUpdated; return HeapTupleBeingUpdated;
else
return HeapTupleMayBeUpdated;
} }
if (tuple->t_infomask & HEAP_XMAX_IS_MULTI) if (tuple->t_infomask & HEAP_XMAX_IS_MULTI)
...@@ -539,10 +539,11 @@ HeapTupleSatisfiesUpdate(HeapTuple htup, CommandId curcid, ...@@ -539,10 +539,11 @@ HeapTupleSatisfiesUpdate(HeapTuple htup, CommandId curcid,
/* not LOCKED_ONLY, so it has to have an xmax */ /* not LOCKED_ONLY, so it has to have an xmax */
Assert(TransactionIdIsValid(xmax)); Assert(TransactionIdIsValid(xmax));
/* updating subtransaction must have aborted */ /* deleting subtransaction must have aborted */
if (!TransactionIdIsCurrentTransactionId(xmax)) if (!TransactionIdIsCurrentTransactionId(xmax))
{ {
if (MultiXactHasRunningRemoteMembers(HeapTupleHeaderGetRawXmax(tuple))) if (MultiXactIdIsRunning(HeapTupleHeaderGetRawXmax(tuple),
false))
return HeapTupleBeingUpdated; return HeapTupleBeingUpdated;
return HeapTupleMayBeUpdated; return HeapTupleMayBeUpdated;
} }
...@@ -664,7 +665,7 @@ HeapTupleSatisfiesUpdate(HeapTuple htup, CommandId curcid, ...@@ -664,7 +665,7 @@ HeapTupleSatisfiesUpdate(HeapTuple htup, CommandId curcid,
if (TransactionIdIsCurrentTransactionId(HeapTupleHeaderGetRawXmax(tuple))) if (TransactionIdIsCurrentTransactionId(HeapTupleHeaderGetRawXmax(tuple)))
{ {
if (HEAP_XMAX_IS_LOCKED_ONLY(tuple->t_infomask)) if (HEAP_XMAX_IS_LOCKED_ONLY(tuple->t_infomask))
return HeapTupleMayBeUpdated; return HeapTupleBeingUpdated;
if (HeapTupleHeaderGetCmax(tuple) >= curcid) if (HeapTupleHeaderGetCmax(tuple) >= curcid)
return HeapTupleSelfUpdated; /* updated after scan started */ return HeapTupleSelfUpdated; /* updated after scan started */
else else
......
...@@ -96,7 +96,6 @@ extern bool MultiXactIdIsRunning(MultiXactId multi, bool isLockOnly); ...@@ -96,7 +96,6 @@ extern bool MultiXactIdIsRunning(MultiXactId multi, bool isLockOnly);
extern void MultiXactIdSetOldestMember(void); extern void MultiXactIdSetOldestMember(void);
extern int GetMultiXactIdMembers(MultiXactId multi, MultiXactMember **xids, extern int GetMultiXactIdMembers(MultiXactId multi, MultiXactMember **xids,
bool allow_old, bool isLockOnly); bool allow_old, bool isLockOnly);
extern bool MultiXactHasRunningRemoteMembers(MultiXactId multi);
extern bool MultiXactIdPrecedes(MultiXactId multi1, MultiXactId multi2); extern bool MultiXactIdPrecedes(MultiXactId multi1, MultiXactId multi2);
extern bool MultiXactIdPrecedesOrEquals(MultiXactId multi1, extern bool MultiXactIdPrecedesOrEquals(MultiXactId multi1,
MultiXactId multi2); MultiXactId multi2);
......
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