]> granicus.if.org Git - postgresql/commit
Fix improper abort during update chain locking
authorAlvaro Herrera <alvherre@alvh.no-ip.org>
Thu, 5 Dec 2013 20:47:51 +0000 (17:47 -0300)
committerAlvaro Herrera <alvherre@alvh.no-ip.org>
Thu, 5 Dec 2013 20:47:51 +0000 (17:47 -0300)
commit2dcc48c35af5305fba0d8cb5e31fa0c25f52d13f
treeb71461f7857695ebd0b241ed39f74e9b2b218798
parent2a6e1a55454bababea78af2db2f95c6eb7dd8ed5
Fix improper abort during update chain locking

In 247c76a98909, I added some code to do fine-grained checking of
MultiXact status of locking/updating transactions when traversing an
update chain.  There was a thinko in that patch which would have the
traversing abort, that is return HeapTupleUpdated, when the other
transaction is a committed lock-only.  In this case we should ignore it
and return success instead.  Of course, in the case where there is a
committed update, HeapTupleUpdated is the correct return value.

A user-visible symptom of this bug is that in REPEATABLE READ and
SERIALIZABLE transaction isolation modes spurious serializability errors
can occur:
  ERROR:  could not serialize access due to concurrent update

In order for this to happen, there needs to be a tuple that's key-share-
locked and also updated, and the update must abort; a subsequent
transaction trying to acquire a new lock on that tuple would abort with
the above error.  The reason is that the initial FOR KEY SHARE is seen
as committed by the new locking transaction, which triggers this bug.
(If the UPDATE commits, then the serialization error is correctly
reported.)

When running a query in READ COMMITTED mode, what happens is that the
locking is aborted by the HeapTupleUpdated return value, then
EvalPlanQual fetches the newest version of the tuple, which is then the
only version that gets locked.  (The second time the tuple is checked
there is no misbehavior on the committed lock-only, because it's not
checked by the code that traverses update chains; so no bug.) Only the
newest version of the tuple is locked, not older ones, but this is
harmless.

The isolation test added by this commit illustrates the desired
behavior, including the proper serialization errors that get thrown.

Backpatch to 9.3.
src/backend/access/heap/heapam.c
src/test/isolation/expected/multixact-no-forget.out [new file with mode: 0644]
src/test/isolation/expected/multixact-no-forget_1.out [new file with mode: 0644]
src/test/isolation/isolation_schedule
src/test/isolation/specs/multixact-no-forget.spec [new file with mode: 0644]