mirror of
https://github.com/isar/libmdbx.git
synced 2025-01-04 20:24:13 +08:00
mdbx: add checks for unexpected LEAF2-pages.
The fourth case of https://github.com/erthink/libmdbx/issues/217.
This commit is contained in:
parent
68aef96f0a
commit
e7336e1d5e
50
src/core.c
50
src/core.c
@ -13666,7 +13666,11 @@ skip:
|
|||||||
return MDBX_CORRUPTED;
|
return MDBX_CORRUPTED;
|
||||||
|
|
||||||
if (IS_LEAF2(mp)) {
|
if (IS_LEAF2(mp)) {
|
||||||
if (likely(key)) {
|
if (!MDBX_DISABLE_PAGECHECKS && unlikely((mc->mc_flags & C_SUB) == 0)) {
|
||||||
|
mdbx_error("unexpected LEAF2-page %" PRIaPGNO "for non-dupsort cursor",
|
||||||
|
mp->mp_pgno);
|
||||||
|
return MDBX_CORRUPTED;
|
||||||
|
} else if (likely(key)) {
|
||||||
key->iov_len = mc->mc_db->md_xsize;
|
key->iov_len = mc->mc_db->md_xsize;
|
||||||
key->iov_base = page_leaf2key(mp, mc->mc_ki[mc->mc_top], key->iov_len);
|
key->iov_base = page_leaf2key(mp, mc->mc_ki[mc->mc_top], key->iov_len);
|
||||||
}
|
}
|
||||||
@ -13756,7 +13760,11 @@ static int mdbx_cursor_prev(MDBX_cursor *mc, MDBX_val *key, MDBX_val *data,
|
|||||||
return MDBX_CORRUPTED;
|
return MDBX_CORRUPTED;
|
||||||
|
|
||||||
if (IS_LEAF2(mp)) {
|
if (IS_LEAF2(mp)) {
|
||||||
if (likely(key)) {
|
if (!MDBX_DISABLE_PAGECHECKS && unlikely((mc->mc_flags & C_SUB) == 0)) {
|
||||||
|
mdbx_error("unexpected LEAF2-page %" PRIaPGNO "for non-dupsort cursor",
|
||||||
|
mp->mp_pgno);
|
||||||
|
return MDBX_CORRUPTED;
|
||||||
|
} else if (likely(key)) {
|
||||||
key->iov_len = mc->mc_db->md_xsize;
|
key->iov_len = mc->mc_db->md_xsize;
|
||||||
key->iov_base = page_leaf2key(mp, mc->mc_ki[mc->mc_top], key->iov_len);
|
key->iov_base = page_leaf2key(mp, mc->mc_ki[mc->mc_top], key->iov_len);
|
||||||
}
|
}
|
||||||
@ -13974,11 +13982,17 @@ got_node:
|
|||||||
mc->mc_flags &= ~C_EOF;
|
mc->mc_flags &= ~C_EOF;
|
||||||
|
|
||||||
if (IS_LEAF2(mp)) {
|
if (IS_LEAF2(mp)) {
|
||||||
|
if (!MDBX_DISABLE_PAGECHECKS && unlikely((mc->mc_flags & C_SUB) == 0)) {
|
||||||
|
mdbx_error("unexpected LEAF2-page %" PRIaPGNO "for non-dupsort cursor",
|
||||||
|
mp->mp_pgno);
|
||||||
|
ret.err = MDBX_CORRUPTED;
|
||||||
|
} else {
|
||||||
if (op == MDBX_SET_RANGE || op == MDBX_SET_KEY) {
|
if (op == MDBX_SET_RANGE || op == MDBX_SET_KEY) {
|
||||||
key->iov_len = mc->mc_db->md_xsize;
|
key->iov_len = mc->mc_db->md_xsize;
|
||||||
key->iov_base = page_leaf2key(mp, mc->mc_ki[mc->mc_top], key->iov_len);
|
key->iov_base = page_leaf2key(mp, mc->mc_ki[mc->mc_top], key->iov_len);
|
||||||
}
|
}
|
||||||
ret.err = MDBX_SUCCESS;
|
ret.err = MDBX_SUCCESS;
|
||||||
|
}
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -14081,7 +14095,11 @@ static int mdbx_cursor_first(MDBX_cursor *mc, MDBX_val *key, MDBX_val *data) {
|
|||||||
mc->mc_ki[mc->mc_top] = 0;
|
mc->mc_ki[mc->mc_top] = 0;
|
||||||
|
|
||||||
if (IS_LEAF2(mc->mc_pg[mc->mc_top])) {
|
if (IS_LEAF2(mc->mc_pg[mc->mc_top])) {
|
||||||
if (likely(key)) {
|
if (!MDBX_DISABLE_PAGECHECKS && unlikely((mc->mc_flags & C_SUB) == 0)) {
|
||||||
|
mdbx_error("unexpected LEAF2-page %" PRIaPGNO "for non-dupsort cursor",
|
||||||
|
mc->mc_pg[mc->mc_top]->mp_pgno);
|
||||||
|
return MDBX_CORRUPTED;
|
||||||
|
} else if (likely(key)) {
|
||||||
key->iov_len = mc->mc_db->md_xsize;
|
key->iov_len = mc->mc_db->md_xsize;
|
||||||
key->iov_base = page_leaf2key(mc->mc_pg[mc->mc_top], 0, key->iov_len);
|
key->iov_base = page_leaf2key(mc->mc_pg[mc->mc_top], 0, key->iov_len);
|
||||||
}
|
}
|
||||||
@ -14128,7 +14146,11 @@ static int mdbx_cursor_last(MDBX_cursor *mc, MDBX_val *key, MDBX_val *data) {
|
|||||||
mc->mc_flags |= C_INITIALIZED | C_EOF;
|
mc->mc_flags |= C_INITIALIZED | C_EOF;
|
||||||
|
|
||||||
if (IS_LEAF2(mc->mc_pg[mc->mc_top])) {
|
if (IS_LEAF2(mc->mc_pg[mc->mc_top])) {
|
||||||
if (likely(key)) {
|
if (!MDBX_DISABLE_PAGECHECKS && unlikely((mc->mc_flags & C_SUB) == 0)) {
|
||||||
|
mdbx_error("unexpected LEAF2-page %" PRIaPGNO "for non-dupsort cursor",
|
||||||
|
mc->mc_pg[mc->mc_top]->mp_pgno);
|
||||||
|
return MDBX_CORRUPTED;
|
||||||
|
} else if (likely(key)) {
|
||||||
key->iov_len = mc->mc_db->md_xsize;
|
key->iov_len = mc->mc_db->md_xsize;
|
||||||
key->iov_base = page_leaf2key(mc->mc_pg[mc->mc_top],
|
key->iov_base = page_leaf2key(mc->mc_pg[mc->mc_top],
|
||||||
mc->mc_ki[mc->mc_top], key->iov_len);
|
mc->mc_ki[mc->mc_top], key->iov_len);
|
||||||
@ -14186,6 +14208,11 @@ int mdbx_cursor_get(MDBX_cursor *mc, MDBX_val *key, MDBX_val *data,
|
|||||||
|
|
||||||
rc = MDBX_SUCCESS;
|
rc = MDBX_SUCCESS;
|
||||||
if (IS_LEAF2(mp)) {
|
if (IS_LEAF2(mp)) {
|
||||||
|
if (!MDBX_DISABLE_PAGECHECKS && unlikely((mc->mc_flags & C_SUB) == 0)) {
|
||||||
|
mdbx_error("unexpected LEAF2-page %" PRIaPGNO "for non-dupsort cursor",
|
||||||
|
mp->mp_pgno);
|
||||||
|
return MDBX_CORRUPTED;
|
||||||
|
}
|
||||||
key->iov_len = mc->mc_db->md_xsize;
|
key->iov_len = mc->mc_db->md_xsize;
|
||||||
key->iov_base = page_leaf2key(mp, mc->mc_ki[mc->mc_top], key->iov_len);
|
key->iov_base = page_leaf2key(mp, mc->mc_ki[mc->mc_top], key->iov_len);
|
||||||
} else {
|
} else {
|
||||||
@ -15239,8 +15266,14 @@ int mdbx_cursor_del(MDBX_cursor *mc, MDBX_put_flags_t flags) {
|
|||||||
MDBX_page *mp = mc->mc_pg[mc->mc_top];
|
MDBX_page *mp = mc->mc_pg[mc->mc_top];
|
||||||
if (!MDBX_DISABLE_PAGECHECKS && unlikely(!IS_LEAF(mp)))
|
if (!MDBX_DISABLE_PAGECHECKS && unlikely(!IS_LEAF(mp)))
|
||||||
return MDBX_CORRUPTED;
|
return MDBX_CORRUPTED;
|
||||||
if (IS_LEAF2(mp))
|
if (IS_LEAF2(mp)) {
|
||||||
|
if (!MDBX_DISABLE_PAGECHECKS && unlikely((mc->mc_flags & C_SUB) == 0)) {
|
||||||
|
mdbx_error("unexpected LEAF2-page %" PRIaPGNO "for non-dupsort cursor",
|
||||||
|
mp->mp_pgno);
|
||||||
|
return MDBX_CORRUPTED;
|
||||||
|
}
|
||||||
goto del_key;
|
goto del_key;
|
||||||
|
}
|
||||||
|
|
||||||
MDBX_node *node = page_node(mp, mc->mc_ki[mc->mc_top]);
|
MDBX_node *node = page_node(mp, mc->mc_ki[mc->mc_top]);
|
||||||
if (F_ISSET(node_flags(node), F_DUPDATA)) {
|
if (F_ISSET(node_flags(node), F_DUPDATA)) {
|
||||||
@ -17065,6 +17098,10 @@ __cold static int mdbx_page_check(MDBX_cursor *const mc,
|
|||||||
if (unlikely(mp->mp_pgno + mp->mp_pages > mc->mc_txn->mt_next_pgno))
|
if (unlikely(mp->mp_pgno + mp->mp_pages > mc->mc_txn->mt_next_pgno))
|
||||||
return bad_page(mp, "overflow page beyond (%u) next-pgno\n",
|
return bad_page(mp, "overflow page beyond (%u) next-pgno\n",
|
||||||
mp->mp_pgno + mp->mp_pages);
|
mp->mp_pgno + mp->mp_pages);
|
||||||
|
if (unlikely(mc->mc_db->md_flags & MDBX_DUPSORT))
|
||||||
|
return bad_page(mp,
|
||||||
|
"unexpected overflow-page for dupsort db (flags 0x%x)\n",
|
||||||
|
mc->mc_db->md_flags);
|
||||||
return MDBX_SUCCESS;
|
return MDBX_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -17073,6 +17110,9 @@ __cold static int mdbx_page_check(MDBX_cursor *const mc,
|
|||||||
if (unlikely(nkeys < 2 && IS_BRANCH(mp)))
|
if (unlikely(nkeys < 2 && IS_BRANCH(mp)))
|
||||||
rc = bad_page(mp, "branch-page nkey (%u) < 2\n", nkeys);
|
rc = bad_page(mp, "branch-page nkey (%u) < 2\n", nkeys);
|
||||||
}
|
}
|
||||||
|
if (IS_LEAF2(mp) && unlikely((mc->mc_flags & C_SUB) == 0))
|
||||||
|
rc = bad_page(mp, "unexpected leaf2-page (db flags 0x%x)\n",
|
||||||
|
mc->mc_db->md_flags);
|
||||||
|
|
||||||
MDBX_val here, prev = {0, 0};
|
MDBX_val here, prev = {0, 0};
|
||||||
for (unsigned i = 0; i < nkeys; ++i) {
|
for (unsigned i = 0; i < nkeys; ++i) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user