Commit fb84af8a authored by stephen hemminger's avatar stephen hemminger Committed by David S. Miller

netvsc: fix rtnl deadlock on unregister of vf

With new transparent VF support, it is possible to get a deadlock
when some of the deferred work is running and the unregister_vf
is trying to cancel the work element. The solution is to use
trylock and reschedule (similar to bonding and team device).
Reported-by: default avatarVitaly Kuznetsov <vkuznets@redhat.com>
Fixes: 0c195567 ("netvsc: transparent VF management")
Signed-off-by: default avatarStephen Hemminger <sthemmin@microsoft.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 5f6b4e14
...@@ -1601,7 +1601,11 @@ static void netvsc_vf_setup(struct work_struct *w) ...@@ -1601,7 +1601,11 @@ static void netvsc_vf_setup(struct work_struct *w)
struct net_device *ndev = hv_get_drvdata(ndev_ctx->device_ctx); struct net_device *ndev = hv_get_drvdata(ndev_ctx->device_ctx);
struct net_device *vf_netdev; struct net_device *vf_netdev;
rtnl_lock(); if (!rtnl_trylock()) {
schedule_work(w);
return;
}
vf_netdev = rtnl_dereference(ndev_ctx->vf_netdev); vf_netdev = rtnl_dereference(ndev_ctx->vf_netdev);
if (vf_netdev) if (vf_netdev)
__netvsc_vf_setup(ndev, vf_netdev); __netvsc_vf_setup(ndev, vf_netdev);
...@@ -1655,7 +1659,11 @@ static void netvsc_vf_update(struct work_struct *w) ...@@ -1655,7 +1659,11 @@ static void netvsc_vf_update(struct work_struct *w)
struct net_device *vf_netdev; struct net_device *vf_netdev;
bool vf_is_up; bool vf_is_up;
rtnl_lock(); if (!rtnl_trylock()) {
schedule_work(w);
return;
}
vf_netdev = rtnl_dereference(ndev_ctx->vf_netdev); vf_netdev = rtnl_dereference(ndev_ctx->vf_netdev);
if (!vf_netdev) if (!vf_netdev)
goto unlock; goto unlock;
......
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