summaryrefslogtreecommitdiff
path: root/usr.sbin
diff options
context:
space:
mode:
authorMartin Hedenfal <martinh@cvs.openbsd.org>2010-07-05 21:06:46 +0000
committerMartin Hedenfal <martinh@cvs.openbsd.org>2010-07-05 21:06:46 +0000
commitf59bf720b1c38ff9078ff851a7f42c5f92726ac3 (patch)
treefffa29476f9bed8b8dac6ef18a0d957fc8200f37 /usr.sbin
parentaf23a97690e5aebe4091f192bdc0d1222d23bc80 (diff)
Unbreak writing updates to the btree while having a cursor open on the
affected pages. When a cursor has increased the reference count on a cached page, it is copied before updated. The new pointer was however not passed back to the caller, effectively commiting the old page numbers.
Diffstat (limited to 'usr.sbin')
-rw-r--r--usr.sbin/ldapd/btree.c34
1 files changed, 21 insertions, 13 deletions
diff --git a/usr.sbin/ldapd/btree.c b/usr.sbin/ldapd/btree.c
index 3a869b2b5b5..48e0473bd7b 100644
--- a/usr.sbin/ldapd/btree.c
+++ b/usr.sbin/ldapd/btree.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: btree.c,v 1.23 2010/07/05 17:11:41 martinh Exp $ */
+/* $OpenBSD: btree.c,v 1.24 2010/07/05 21:06:45 martinh Exp $ */
/*
* Copyright (c) 2009, 2010 Martin Hedenfalk <martin@bzero.se>
@@ -147,7 +147,7 @@ static void mpage_flush(struct btree *bt);
static struct mpage *mpage_copy(struct btree *bt, struct mpage *mp);
static void mpage_prune(struct btree *bt);
static void mpage_dirty(struct btree *bt, struct mpage *mp);
-static void mpage_touch(struct btree *bt, struct mpage *mp);
+static struct mpage *mpage_touch(struct btree *bt, struct mpage *mp);
RB_PROTOTYPE(page_cache, mpage, entry, mpage_cmp);
RB_GENERATE(page_cache, mpage, entry, mpage_cmp);
@@ -573,7 +573,7 @@ mpage_dirty(struct btree *bt, struct mpage *mp)
/* Touch a page: make it dirty and re-insert into tree with updated pgno.
*/
-static void
+static struct mpage *
mpage_touch(struct btree *bt, struct mpage *mp)
{
assert(bt != NULL);
@@ -584,8 +584,10 @@ mpage_touch(struct btree *bt, struct mpage *mp)
DPRINTF("touching page %u -> %u", mp->pgno, bt->txn->next_pgno);
if (mp->ref == 0)
mpage_del(bt, mp);
- else
- mp = mpage_copy(bt, mp);
+ else {
+ if ((mp = mpage_copy(bt, mp)) == NULL)
+ return NULL;
+ }
mp->pgno = mp->page->pgno = bt->txn->next_pgno++;
mpage_dirty(bt, mp);
mpage_add(bt, mp);
@@ -595,6 +597,8 @@ mpage_touch(struct btree *bt, struct mpage *mp)
NODEPGNO(NODEPTR(mp->parent,
mp->parent_index)) = mp->pgno;
}
+
+ return mp;
}
static int
@@ -1409,8 +1413,8 @@ btree_search_page_root(struct btree *bt, struct mpage *root, struct btval *key,
if (cursor && cursor_push_page(cursor, mp) == NULL)
return BT_FAIL;
- if (modify)
- mpage_touch(bt, mp);
+ if (modify && (mp = mpage_touch(bt, mp)) == NULL)
+ return BT_FAIL;
}
if (!IS_LEAF(mp)) {
@@ -1476,7 +1480,8 @@ btree_search_page(struct btree *bt, struct btree_txn *txn, struct btval *key,
assert(mp->prefix.len == 0);
if (modify && !mp->dirty) {
- mpage_touch(bt, mp);
+ if ((mp = mpage_touch(bt, mp)) == NULL)
+ return BT_FAIL;
txn->root = mp->pgno;
}
@@ -2224,8 +2229,9 @@ btree_move_node(struct btree *bt, struct mpage *src, indx_t srcindx,
}
/* Mark src and dst as dirty. */
- mpage_touch(bt, src);
- mpage_touch(bt, dst);
+ if ((src = mpage_touch(bt, src)) == NULL ||
+ (dst = mpage_touch(bt, dst)) == NULL)
+ return BT_FAIL;
find_common_prefix(bt, dst);
@@ -2346,7 +2352,8 @@ btree_move_node(struct btree *bt, struct mpage *src, indx_t srcindx,
find_common_prefix(bt, mp);
if (mp->prefix.len != mp_pfxlen) {
DPRINTF("moved branch node has changed prefix");
- mpage_touch(bt, mp);
+ if ((mp = mpage_touch(bt, mp)) == NULL)
+ return BT_FAIL;
if (btree_adjust_prefix(bt, mp,
mp->prefix.len - mp_pfxlen) != BT_SUCCESS)
return BT_FAIL;
@@ -2373,8 +2380,9 @@ btree_merge(struct btree *bt, struct mpage *src, struct mpage *dst)
assert(bt->txn != NULL);
/* Mark src and dst as dirty. */
- mpage_touch(bt, src);
- mpage_touch(bt, dst);
+ if ((src = mpage_touch(bt, src)) == NULL ||
+ (dst = mpage_touch(bt, dst)) == NULL)
+ return BT_FAIL;
find_common_prefix(bt, src);
find_common_prefix(bt, dst);