Commit 2ec254c0 authored by Kent Overstreet's avatar Kent Overstreet

bcachefs: Ensure bch2_btree_node_lock_write_nofail() never fails

In order for bch2_btree_node_lock_write_nofail() to never produce a
deadlock, we must ensure we're never holding read locks when using it.
Fortunately, it's only used from code paths where any read locks may be
safely dropped.
Signed-off-by: default avatarKent Overstreet <kent.overstreet@linux.dev>
parent 0d7009d7
...@@ -320,6 +320,40 @@ int __bch2_btree_node_lock_write(struct btree_trans *trans, struct btree_path *p ...@@ -320,6 +320,40 @@ int __bch2_btree_node_lock_write(struct btree_trans *trans, struct btree_path *p
return ret; return ret;
} }
void bch2_btree_node_lock_write_nofail(struct btree_trans *trans,
struct btree_path *path,
struct btree_bkey_cached_common *b)
{
struct btree_path *linked;
unsigned i;
int ret;
/*
* XXX BIG FAT NOTICE
*
* Drop all read locks before taking a write lock:
*
* This is a hack, because bch2_btree_node_lock_write_nofail() is a
* hack - but by dropping read locks first, this should never fail, and
* we only use this in code paths where whatever read locks we've
* already taken are no longer needed:
*/
trans_for_each_path(trans, linked) {
if (!linked->nodes_locked)
continue;
for (i = 0; i < BTREE_MAX_DEPTH; i++)
if (btree_node_read_locked(linked, i)) {
btree_node_unlock(trans, linked, i);
btree_path_set_dirty(linked, BTREE_ITER_NEED_RELOCK);
}
}
ret = __btree_node_lock_write(trans, path, b, true);
BUG_ON(ret);
}
/* relock */ /* relock */
static inline bool btree_path_get_locks(struct btree_trans *trans, static inline bool btree_path_get_locks(struct btree_trans *trans,
......
...@@ -290,14 +290,6 @@ static inline int __btree_node_lock_write(struct btree_trans *trans, ...@@ -290,14 +290,6 @@ static inline int __btree_node_lock_write(struct btree_trans *trans,
: __bch2_btree_node_lock_write(trans, path, b, lock_may_not_fail); : __bch2_btree_node_lock_write(trans, path, b, lock_may_not_fail);
} }
static inline void bch2_btree_node_lock_write_nofail(struct btree_trans *trans,
struct btree_path *path,
struct btree_bkey_cached_common *b)
{
int ret = __btree_node_lock_write(trans, path, b, true);
BUG_ON(ret);
}
static inline int __must_check static inline int __must_check
bch2_btree_node_lock_write(struct btree_trans *trans, bch2_btree_node_lock_write(struct btree_trans *trans,
struct btree_path *path, struct btree_path *path,
...@@ -306,6 +298,10 @@ bch2_btree_node_lock_write(struct btree_trans *trans, ...@@ -306,6 +298,10 @@ bch2_btree_node_lock_write(struct btree_trans *trans,
return __btree_node_lock_write(trans, path, b, false); return __btree_node_lock_write(trans, path, b, false);
} }
void bch2_btree_node_lock_write_nofail(struct btree_trans *,
struct btree_path *,
struct btree_bkey_cached_common *);
/* relock: */ /* relock: */
bool bch2_btree_path_relock_norestart(struct btree_trans *, bool bch2_btree_path_relock_norestart(struct btree_trans *,
......
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