struct net_device __rcu *vf_netdev;
        struct netvsc_vf_pcpu_stats __percpu *vf_stats;
        struct delayed_work vf_takeover;
+       struct delayed_work vfns_work;
 
        /* 1: allocated, serial number is valid. 0: not allocated */
        u32 vf_alloc;
        struct netvsc_device_info *saved_netvsc_dev_info;
 };
 
+void netvsc_vfns_work(struct work_struct *w);
+
 /* Azure hosts don't support non-TCP port numbers in hashing for fragmented
  * packets. We can use ethtool to change UDP hash level when necessary.
  */
 
        spin_lock_init(&net_device_ctx->lock);
        INIT_LIST_HEAD(&net_device_ctx->reconfig_events);
        INIT_DELAYED_WORK(&net_device_ctx->vf_takeover, netvsc_vf_setup);
+       INIT_DELAYED_WORK(&net_device_ctx->vfns_work, netvsc_vfns_work);
 
        net_device_ctx->vf_stats
                = netdev_alloc_pcpu_stats(struct netvsc_vf_pcpu_stats);
        cancel_delayed_work_sync(&ndev_ctx->dwork);
 
        rtnl_lock();
+       cancel_delayed_work_sync(&ndev_ctx->vfns_work);
+
        nvdev = rtnl_dereference(ndev_ctx->nvdev);
        if (nvdev) {
                cancel_work_sync(&nvdev->subchan_work);
        cancel_delayed_work_sync(&ndev_ctx->dwork);
 
        rtnl_lock();
+       cancel_delayed_work_sync(&ndev_ctx->vfns_work);
 
        nvdev = rtnl_dereference(ndev_ctx->nvdev);
        if (nvdev == NULL) {
        }
 }
 
+void netvsc_vfns_work(struct work_struct *w)
+{
+       struct net_device_context *ndev_ctx =
+               container_of(w, struct net_device_context, vfns_work.work);
+       struct net_device *ndev;
+
+       if (!rtnl_trylock()) {
+               schedule_delayed_work(&ndev_ctx->vfns_work, 1);
+               return;
+       }
+
+       ndev = hv_get_drvdata(ndev_ctx->device_ctx);
+       if (!ndev)
+               goto out;
+
+       netvsc_event_set_vf_ns(ndev);
+
+out:
+       rtnl_unlock();
+}
+
 /*
  * On Hyper-V, every VF interface is matched with a corresponding
  * synthetic interface. The synthetic interface is presented first
                               unsigned long event, void *ptr)
 {
        struct net_device *event_dev = netdev_notifier_info_to_dev(ptr);
+       struct net_device_context *ndev_ctx;
        int ret = 0;
 
        if (event_dev->netdev_ops == &device_ops && event == NETDEV_REGISTER) {
-               netvsc_event_set_vf_ns(event_dev);
+               ndev_ctx = netdev_priv(event_dev);
+               schedule_delayed_work(&ndev_ctx->vfns_work, 0);
                return NOTIFY_DONE;
        }