Commit e67ab045 authored by Kent Overstreet's avatar Kent Overstreet Committed by Kent Overstreet

bcachefs: Fix bch2_btree_node_iter_prev_filter()

bch2_btree_node_iter_prev_filter() tried to be smart about iterating
backwards when skipping over whiteouts/discards - but unfortunately,
doing so can leave the node iterator in an inconsistent state; the sane
solution is to just always iterate backwards one key at a time.

But we compact btree nodes when more than a quarter of the keys are
whiteouts/discards, so the optimization wasn't buying us that much
anyways.
Signed-off-by: default avatarKent Overstreet <kent.overstreet@linux.dev>
parent 9df27940
...@@ -1682,12 +1682,10 @@ void bch2_btree_node_iter_advance(struct btree_node_iter *iter, ...@@ -1682,12 +1682,10 @@ void bch2_btree_node_iter_advance(struct btree_node_iter *iter,
/* /*
* Expensive: * Expensive:
*/ */
struct bkey_packed *bch2_btree_node_iter_prev_filter(struct btree_node_iter *iter, struct bkey_packed *bch2_btree_node_iter_prev_all(struct btree_node_iter *iter,
struct btree *b, struct btree *b)
unsigned min_key_type)
{ {
struct bkey_packed *k, *prev = NULL; struct bkey_packed *k, *prev = NULL;
struct bkey_packed *orig_pos = bch2_btree_node_iter_peek_all(iter, b);
struct btree_node_iter_set *set; struct btree_node_iter_set *set;
struct bset_tree *t; struct bset_tree *t;
unsigned end = 0; unsigned end = 0;
...@@ -1695,9 +1693,8 @@ struct bkey_packed *bch2_btree_node_iter_prev_filter(struct btree_node_iter *ite ...@@ -1695,9 +1693,8 @@ struct bkey_packed *bch2_btree_node_iter_prev_filter(struct btree_node_iter *ite
bch2_btree_node_iter_verify(iter, b); bch2_btree_node_iter_verify(iter, b);
for_each_bset(b, t) { for_each_bset(b, t) {
k = bch2_bkey_prev_filter(b, t, k = bch2_bkey_prev_all(b, t,
bch2_btree_node_iter_bset_pos(iter, b, t), bch2_btree_node_iter_bset_pos(iter, b, t));
min_key_type);
if (k && if (k &&
(!prev || bkey_iter_cmp(b, k, prev) > 0)) { (!prev || bkey_iter_cmp(b, k, prev) > 0)) {
prev = k; prev = k;
...@@ -1706,7 +1703,7 @@ struct bkey_packed *bch2_btree_node_iter_prev_filter(struct btree_node_iter *ite ...@@ -1706,7 +1703,7 @@ struct bkey_packed *bch2_btree_node_iter_prev_filter(struct btree_node_iter *ite
} }
if (!prev) if (!prev)
goto out; return NULL;
/* /*
* We're manually memmoving instead of just calling sort() to ensure the * We're manually memmoving instead of just calling sort() to ensure the
...@@ -1727,18 +1724,20 @@ struct bkey_packed *bch2_btree_node_iter_prev_filter(struct btree_node_iter *ite ...@@ -1727,18 +1724,20 @@ struct bkey_packed *bch2_btree_node_iter_prev_filter(struct btree_node_iter *ite
iter->data[0].k = __btree_node_key_to_offset(b, prev); iter->data[0].k = __btree_node_key_to_offset(b, prev);
iter->data[0].end = end; iter->data[0].end = end;
out:
if (btree_keys_expensive_checks(b)) {
struct btree_node_iter iter2 = *iter;
if (prev) bch2_btree_node_iter_verify(iter, b);
__bch2_btree_node_iter_advance(&iter2, b); return prev;
}
struct bkey_packed *bch2_btree_node_iter_prev_filter(struct btree_node_iter *iter,
struct btree *b,
unsigned min_key_type)
{
struct bkey_packed *prev;
while ((k = bch2_btree_node_iter_peek_all(&iter2, b)) != orig_pos) { do {
BUG_ON(k->type >= min_key_type); prev = bch2_btree_node_iter_prev_all(iter, b);
__bch2_btree_node_iter_advance(&iter2, b); } while (prev && prev->type < min_key_type);
}
}
return prev; return prev;
} }
......
...@@ -543,15 +543,11 @@ bch2_btree_node_iter_next_all(struct btree_node_iter *iter, struct btree *b) ...@@ -543,15 +543,11 @@ bch2_btree_node_iter_next_all(struct btree_node_iter *iter, struct btree *b)
return ret; return ret;
} }
struct bkey_packed *bch2_btree_node_iter_prev_all(struct btree_node_iter *,
struct btree *);
struct bkey_packed *bch2_btree_node_iter_prev_filter(struct btree_node_iter *, struct bkey_packed *bch2_btree_node_iter_prev_filter(struct btree_node_iter *,
struct btree *, unsigned); struct btree *, unsigned);
static inline struct bkey_packed *
bch2_btree_node_iter_prev_all(struct btree_node_iter *iter, struct btree *b)
{
return bch2_btree_node_iter_prev_filter(iter, b, 0);
}
static inline struct bkey_packed * static inline struct bkey_packed *
bch2_btree_node_iter_prev(struct btree_node_iter *iter, struct btree *b) bch2_btree_node_iter_prev(struct btree_node_iter *iter, struct btree *b)
{ {
......
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