aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorHeikki Linnakangas <heikki.linnakangas@iki.fi>2014-11-17 18:42:04 +0200
committerHeikki Linnakangas <heikki.linnakangas@iki.fi>2014-11-17 18:45:46 +0200
commitc73669c0e0168923e3f9e787beec980f55af2bd8 (patch)
tree28d7663c466d8aad15ca3d577f2981d641f6d878 /src
parent0f9692b40d1292f1b2640f026561908fd37b7407 (diff)
downloadpostgresql-c73669c0e0168923e3f9e787beec980f55af2bd8.tar.gz
postgresql-c73669c0e0168923e3f9e787beec980f55af2bd8.zip
Fix WAL-logging of B-tree "unlink halfdead page" operation.
There was some confusion on how to record the case that the operation unlinks the last non-leaf page in the branch being deleted. _bt_unlink_halfdead_page set the "topdead" field in the WAL record to the leaf page, but the redo routine assumed that it would be an invalid block number in that case. This commit fixes _bt_unlink_halfdead_page to do what the redo routine expected. This code is new in 9.4, so backpatch there.
Diffstat (limited to 'src')
-rw-r--r--src/backend/access/nbtree/nbtpage.c13
1 files changed, 6 insertions, 7 deletions
diff --git a/src/backend/access/nbtree/nbtpage.c b/src/backend/access/nbtree/nbtpage.c
index 6093215c43d..ea95ce6e1ec 100644
--- a/src/backend/access/nbtree/nbtpage.c
+++ b/src/backend/access/nbtree/nbtpage.c
@@ -1567,7 +1567,6 @@ _bt_unlink_halfdead_page(Relation rel, Buffer leafbuf, bool *rightsib_empty)
int targetlevel;
ItemPointer leafhikey;
BlockNumber nextchild;
- BlockNumber topblkno;
page = BufferGetPage(leafbuf);
opaque = (BTPageOpaque) PageGetSpecialPointer(page);
@@ -1591,11 +1590,10 @@ _bt_unlink_halfdead_page(Relation rel, Buffer leafbuf, bool *rightsib_empty)
*/
if (ItemPointerIsValid(leafhikey))
{
- topblkno = ItemPointerGetBlockNumber(leafhikey);
- target = topblkno;
+ target = ItemPointerGetBlockNumber(leafhikey);
/* fetch the block number of the topmost parent's left sibling */
- buf = _bt_getbuf(rel, topblkno, BT_READ);
+ buf = _bt_getbuf(rel, target, BT_READ);
page = BufferGetPage(buf);
opaque = (BTPageOpaque) PageGetSpecialPointer(page);
leftsib = opaque->btpo_prev;
@@ -1609,7 +1607,6 @@ _bt_unlink_halfdead_page(Relation rel, Buffer leafbuf, bool *rightsib_empty)
}
else
{
- topblkno = InvalidBlockNumber;
target = leafblkno;
buf = leafbuf;
@@ -1694,9 +1691,11 @@ _bt_unlink_halfdead_page(Relation rel, Buffer leafbuf, bool *rightsib_empty)
elog(ERROR, "half-dead page changed status unexpectedly in block %u of index \"%s\"",
target, RelationGetRelationName(rel));
- /* remember the next child down in the branch. */
+ /* remember the next non-leaf child down in the branch. */
itemid = PageGetItemId(page, P_FIRSTDATAKEY(opaque));
nextchild = ItemPointerGetBlockNumber(&((IndexTuple) PageGetItem(page, itemid))->t_tid);
+ if (nextchild == leafblkno)
+ nextchild = InvalidBlockNumber;
}
/*
@@ -1782,7 +1781,7 @@ _bt_unlink_halfdead_page(Relation rel, Buffer leafbuf, bool *rightsib_empty)
*/
if (target != leafblkno)
{
- if (nextchild == leafblkno)
+ if (nextchild == InvalidBlockNumber)
ItemPointerSetInvalid(leafhikey);
else
ItemPointerSet(leafhikey, nextchild, P_HIKEY);