bcachefs: Tighten up btree locking invariants

New rule is: if a btree path holds any locks it should be holding
precisely the locks wanted (accoringing to path->level and
path->locks_want).

Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
This commit is contained in:
Kent Overstreet 2021-09-04 21:23:11 -04:00 committed by Kent Overstreet
parent 1ae29c1faa
commit 1d3ecd7ea7
3 changed files with 32 additions and 46 deletions

View File

@ -224,7 +224,6 @@ static inline bool btree_path_get_locks(struct btree_trans *trans,
? path->l[l].b->c.lock.state.seq ? path->l[l].b->c.lock.state.seq
: 0); : 0);
fail_idx = l; fail_idx = l;
btree_path_set_dirty(path, BTREE_ITER_NEED_TRAVERSE);
} }
l++; l++;
@ -235,10 +234,14 @@ static inline bool btree_path_get_locks(struct btree_trans *trans,
* can't be relocked so bch2_btree_path_traverse has to walk back up to * can't be relocked so bch2_btree_path_traverse has to walk back up to
* the node that we failed to relock: * the node that we failed to relock:
*/ */
while (fail_idx >= 0) { if (fail_idx >= 0) {
btree_node_unlock(path, fail_idx); __bch2_btree_path_unlock(path);
path->l[fail_idx].b = BTREE_ITER_NO_NODE_GET_LOCKS; btree_path_set_dirty(path, BTREE_ITER_NEED_TRAVERSE);
--fail_idx;
do {
path->l[fail_idx].b = BTREE_ITER_NO_NODE_GET_LOCKS;
--fail_idx;
} while (fail_idx >= 0);
} }
if (path->uptodate == BTREE_ITER_NEED_RELOCK) if (path->uptodate == BTREE_ITER_NEED_RELOCK)
@ -376,14 +379,14 @@ static void bch2_btree_path_verify_locks(struct btree_path *path)
{ {
unsigned l; unsigned l;
for (l = 0; btree_path_node(path, l); l++) { if (!path->nodes_locked) {
if (path->uptodate >= BTREE_ITER_NEED_RELOCK && BUG_ON(path->uptodate == BTREE_ITER_UPTODATE);
!btree_node_locked(path, l)) return;
continue; }
for (l = 0; btree_path_node(path, l); l++)
BUG_ON(btree_lock_want(path, l) != BUG_ON(btree_lock_want(path, l) !=
btree_node_locked_type(path, l)); btree_node_locked_type(path, l));
}
} }
void bch2_trans_verify_locks(struct btree_trans *trans) void bch2_trans_verify_locks(struct btree_trans *trans)
@ -421,6 +424,7 @@ bool bch2_btree_path_relock_intent(struct btree_trans *trans,
is_btree_node(path, l) is_btree_node(path, l)
? path->l[l].b->c.lock.state.seq ? path->l[l].b->c.lock.state.seq
: 0); : 0);
__bch2_btree_path_unlock(path);
btree_path_set_dirty(path, BTREE_ITER_NEED_TRAVERSE); btree_path_set_dirty(path, BTREE_ITER_NEED_TRAVERSE);
btree_trans_restart(trans); btree_trans_restart(trans);
return false; return false;
@ -668,9 +672,6 @@ void bch2_trans_verify_paths(struct btree_trans *trans)
{ {
struct btree_path *path; struct btree_path *path;
if (!bch2_debug_check_iterators)
return;
trans_for_each_path(trans, path) trans_for_each_path(trans, path)
bch2_btree_path_verify(trans, path); bch2_btree_path_verify(trans, path);
} }
@ -1013,7 +1014,7 @@ static void btree_path_verify_new_node(struct btree_trans *trans,
} }
if (!parent_locked) if (!parent_locked)
btree_node_unlock(path, b->c.level + 1); btree_node_unlock(path, plevel);
} }
static inline void __btree_path_level_init(struct btree_path *path, static inline void __btree_path_level_init(struct btree_path *path,
@ -1055,21 +1056,17 @@ static inline void btree_path_level_init(struct btree_trans *trans,
*/ */
void bch2_trans_node_add(struct btree_trans *trans, struct btree *b) void bch2_trans_node_add(struct btree_trans *trans, struct btree *b)
{ {
enum btree_node_locked_type t;
struct btree_path *path; struct btree_path *path;
trans_for_each_path(trans, path) trans_for_each_path(trans, path)
if (!path->cached && if (!path->cached &&
btree_path_pos_in_node(path, b)) { btree_path_pos_in_node(path, b)) {
/* enum btree_node_locked_type t =
* bch2_btree_path_node_drop() has already been called - btree_lock_want(path, b->c.level);
* the old node we're replacing has already been
* unlocked and the pointer invalidated
*/
BUG_ON(btree_node_locked(path, b->c.level));
t = btree_lock_want(path, b->c.level); if (path->nodes_locked &&
if (t != BTREE_NODE_UNLOCKED) { t != BTREE_NODE_UNLOCKED) {
btree_node_unlock(path, b->c.level);
six_lock_increment(&b->c.lock, (enum six_lock_type) t); six_lock_increment(&b->c.lock, (enum six_lock_type) t);
mark_btree_node_locked(trans, path, b->c.level, (enum six_lock_type) t); mark_btree_node_locked(trans, path, b->c.level, (enum six_lock_type) t);
} }
@ -1078,18 +1075,6 @@ void bch2_trans_node_add(struct btree_trans *trans, struct btree *b)
} }
} }
void bch2_trans_node_drop(struct btree_trans *trans, struct btree *b)
{
struct btree_path *path;
unsigned level = b->c.level;
trans_for_each_path(trans, path)
if (path->l[level].b == b) {
btree_node_unlock(path, level);
path->l[level].b = BTREE_ITER_NO_NODE_DROP;
}
}
/* /*
* A btree node has been modified in such a way as to invalidate iterators - fix * A btree node has been modified in such a way as to invalidate iterators - fix
* them: * them:
@ -1392,6 +1377,9 @@ static inline unsigned btree_path_up_until_good_node(struct btree_trans *trans,
{ {
unsigned l = path->level; unsigned l = path->level;
if (!path->nodes_locked)
btree_path_get_locks(trans, path, false, _THIS_IP_);
while (btree_path_node(path, l) && while (btree_path_node(path, l) &&
!btree_path_good_node(trans, path, l, check_pos)) { !btree_path_good_node(trans, path, l, check_pos)) {
btree_node_unlock(path, l); btree_node_unlock(path, l);
@ -1584,14 +1572,12 @@ __bch2_btree_path_set_pos(struct btree_trans *trans,
if (cmp < 0 || if (cmp < 0 ||
!btree_path_advance_to_pos(path, &path->l[l], 8)) !btree_path_advance_to_pos(path, &path->l[l], 8))
__btree_path_level_init(path, l); __btree_path_level_init(path, l);
/* Don't leave it locked if we're not supposed to: */
if (btree_lock_want(path, l) == BTREE_NODE_UNLOCKED)
btree_node_unlock(path, l);
} }
if (l != path->level) if (l != path->level) {
btree_path_set_dirty(path, BTREE_ITER_NEED_TRAVERSE); btree_path_set_dirty(path, BTREE_ITER_NEED_TRAVERSE);
__bch2_btree_path_unlock(path);
}
out: out:
bch2_btree_path_verify(trans, path); bch2_btree_path_verify(trans, path);
#ifdef CONFIG_BCACHEFS_DEBUG #ifdef CONFIG_BCACHEFS_DEBUG
@ -2760,9 +2746,10 @@ void bch2_btree_trans_to_text(struct printbuf *out, struct bch_fs *c)
if (!path->nodes_locked) if (!path->nodes_locked)
continue; continue;
pr_buf(out, " path %u %c %s:", pr_buf(out, " path %u %c l=%u %s:",
path->idx, path->idx,
path->cached ? 'c' : 'b', path->cached ? 'c' : 'b',
path->level,
bch2_btree_ids[path->btree_id]); bch2_btree_ids[path->btree_id]);
bch2_bpos_to_text(out, path->pos); bch2_bpos_to_text(out, path->pos);
pr_buf(out, "\n"); pr_buf(out, "\n");

View File

@ -219,7 +219,6 @@ static inline void bch2_btree_path_downgrade(struct btree_path *path)
void bch2_trans_downgrade(struct btree_trans *); void bch2_trans_downgrade(struct btree_trans *);
void bch2_trans_node_add(struct btree_trans *trans, struct btree *); void bch2_trans_node_add(struct btree_trans *trans, struct btree *);
void bch2_trans_node_drop(struct btree_trans *, struct btree *);
void bch2_trans_node_reinit_iter(struct btree_trans *, struct btree *); void bch2_trans_node_reinit_iter(struct btree_trans *, struct btree *);
int __must_check bch2_btree_iter_traverse(struct btree_iter *); int __must_check bch2_btree_iter_traverse(struct btree_iter *);

View File

@ -1429,7 +1429,6 @@ static void btree_split(struct btree_update *as, struct btree_trans *trans,
/* Successful split, update the path to point to the new nodes: */ /* Successful split, update the path to point to the new nodes: */
six_lock_increment(&b->c.lock, SIX_LOCK_intent); six_lock_increment(&b->c.lock, SIX_LOCK_intent);
bch2_trans_node_drop(trans, b);
if (n3) if (n3)
bch2_trans_node_add(trans, n3); bch2_trans_node_add(trans, n3);
if (n2) if (n2)
@ -1694,14 +1693,16 @@ retry:
bch2_keylist_add(&as->parent_keys, &delete); bch2_keylist_add(&as->parent_keys, &delete);
bch2_keylist_add(&as->parent_keys, &n->key); bch2_keylist_add(&as->parent_keys, &n->key);
bch2_trans_verify_paths(trans);
bch2_btree_insert_node(as, trans, path, parent, &as->parent_keys, flags); bch2_btree_insert_node(as, trans, path, parent, &as->parent_keys, flags);
bch2_trans_verify_paths(trans);
bch2_btree_update_get_open_buckets(as, n); bch2_btree_update_get_open_buckets(as, n);
six_lock_increment(&b->c.lock, SIX_LOCK_intent); six_lock_increment(&b->c.lock, SIX_LOCK_intent);
six_lock_increment(&m->c.lock, SIX_LOCK_intent); six_lock_increment(&m->c.lock, SIX_LOCK_intent);
bch2_trans_node_drop(trans, b);
bch2_trans_node_drop(trans, m);
bch2_trans_node_add(trans, n); bch2_trans_node_add(trans, n);
@ -1798,7 +1799,6 @@ retry:
bch2_btree_update_get_open_buckets(as, n); bch2_btree_update_get_open_buckets(as, n);
six_lock_increment(&b->c.lock, SIX_LOCK_intent); six_lock_increment(&b->c.lock, SIX_LOCK_intent);
bch2_trans_node_drop(trans, b);
bch2_trans_node_add(trans, n); bch2_trans_node_add(trans, n);
bch2_btree_node_free_inmem(trans, b); bch2_btree_node_free_inmem(trans, b);
six_unlock_intent(&n->c.lock); six_unlock_intent(&n->c.lock);