mirror of https://gitee.com/openkylin/linux.git
btrfs: fix deadlock in delayed_ref_async_start
"Btrfs: track transid for delayed ref flushing" was deadlocking on btrfs_attach_transaction because its not safe to call from the async delayed ref start code. This commit brings back btrfs_join_transaction instead and checks for a blocked commit. Signed-off-by: Josef Bacik <jbacik@fb.com> Signed-off-by: Chris Mason <clm@fb.com>
This commit is contained in:
parent
31b9655f43
commit
0f873eca82
|
@ -2851,16 +2851,13 @@ static void delayed_ref_async_start(struct btrfs_work *work)
|
||||||
|
|
||||||
async = container_of(work, struct async_delayed_refs, work);
|
async = container_of(work, struct async_delayed_refs, work);
|
||||||
|
|
||||||
trans = btrfs_attach_transaction(async->root);
|
/* if the commit is already started, we don't need to wait here */
|
||||||
if (IS_ERR(trans)) {
|
if (btrfs_transaction_blocked(async->root->fs_info))
|
||||||
if (PTR_ERR(trans) != -ENOENT)
|
|
||||||
async->error = PTR_ERR(trans);
|
|
||||||
goto done;
|
goto done;
|
||||||
}
|
|
||||||
|
|
||||||
/* Don't bother flushing if we got into a different transaction */
|
trans = btrfs_join_transaction(async->root);
|
||||||
if (trans->transid != async->transid) {
|
if (IS_ERR(trans)) {
|
||||||
btrfs_end_transaction(trans, async->root);
|
async->error = PTR_ERR(trans);
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2869,10 +2866,15 @@ static void delayed_ref_async_start(struct btrfs_work *work)
|
||||||
* wait on delayed refs
|
* wait on delayed refs
|
||||||
*/
|
*/
|
||||||
trans->sync = true;
|
trans->sync = true;
|
||||||
|
|
||||||
|
/* Don't bother flushing if we got into a different transaction */
|
||||||
|
if (trans->transid > async->transid)
|
||||||
|
goto end;
|
||||||
|
|
||||||
ret = btrfs_run_delayed_refs(trans, async->root, async->count);
|
ret = btrfs_run_delayed_refs(trans, async->root, async->count);
|
||||||
if (ret)
|
if (ret)
|
||||||
async->error = ret;
|
async->error = ret;
|
||||||
|
end:
|
||||||
ret = btrfs_end_transaction(trans, async->root);
|
ret = btrfs_end_transaction(trans, async->root);
|
||||||
if (ret && !async->error)
|
if (ret && !async->error)
|
||||||
async->error = ret;
|
async->error = ret;
|
||||||
|
|
Loading…
Reference in New Issue