Commit 1f42d35a authored by Alvaro Herrera's avatar Alvaro Herrera

BRIN: Handle concurrent desummarization properly

If a page range is desummarized at just the right time concurrently with
an index walk, BRIN would raise an error indicating index corruption.
This is scary and unhelpful; silently returning that the page range is
not summarized is sufficient reaction.

This bug was introduced by commit 975ad4e6 as additional protection
against a bug whose actual fix was elsewhere.  Backpatch equally.
Reported-By: default avatarAnastasia Lubennikova <a.lubennikova@postgrespro.ru>
Diagnosed-By: default avatarAlexander Lakhin <exclusion@gmail.com>
Discussion: https://postgr.es/m/2588667e-d07d-7e10-74e2-7e1e46194491@postgrespro.ru
Backpatch: 9.5 - master
parent 3546cf8a
...@@ -282,10 +282,17 @@ brinGetTupleForHeapBlock(BrinRevmap *revmap, BlockNumber heapBlk, ...@@ -282,10 +282,17 @@ brinGetTupleForHeapBlock(BrinRevmap *revmap, BlockNumber heapBlk,
/* If we land on a revmap page, start over */ /* If we land on a revmap page, start over */
if (BRIN_IS_REGULAR_PAGE(page)) if (BRIN_IS_REGULAR_PAGE(page))
{ {
/*
* If the offset number is greater than what's in the page, it's
* possible that the range was desummarized concurrently. Just
* return NULL to handle that case.
*/
if (*off > PageGetMaxOffsetNumber(page)) if (*off > PageGetMaxOffsetNumber(page))
ereport(ERROR, {
(errcode(ERRCODE_INDEX_CORRUPTED), LockBuffer(*buf, BUFFER_LOCK_UNLOCK);
errmsg_internal("corrupted BRIN index: inconsistent range map"))); return NULL;
}
lp = PageGetItemId(page, *off); lp = PageGetItemId(page, *off);
if (ItemIdIsUsed(lp)) if (ItemIdIsUsed(lp))
{ {
......
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