kfree(area);
 }
 
-static inline void __vfree_deferred(const void *addr)
-{
-       /*
-        * Use raw_cpu_ptr() because this can be called from preemptible
-        * context. Preemption is absolutely fine here, because the llist_add()
-        * implementation is lockless, so it works even if we are adding to
-        * another cpu's list. schedule_work() should be fine with this too.
-        */
-       struct vfree_deferred *p = raw_cpu_ptr(&vfree_deferred);
-
-       if (llist_add((struct llist_node *)addr, &p->list))
-               schedule_work(&p->wq);
-}
-
 /**
  * vfree_atomic - release memory allocated by vmalloc()
  * @addr:        memory base address
  */
 void vfree_atomic(const void *addr)
 {
-       BUG_ON(in_nmi());
+       struct vfree_deferred *p = raw_cpu_ptr(&vfree_deferred);
 
+       BUG_ON(in_nmi());
        kmemleak_free(addr);
 
-       if (!addr)
-               return;
-       __vfree_deferred(addr);
+       /*
+        * Use raw_cpu_ptr() because this can be called from preemptible
+        * context. Preemption is absolutely fine here, because the llist_add()
+        * implementation is lockless, so it works even if we are adding to
+        * another cpu's list. schedule_work() should be fine with this too.
+        */
+       if (addr && llist_add((struct llist_node *)addr, &p->list))
+               schedule_work(&p->wq);
 }
 
 /**
  */
 void vfree(const void *addr)
 {
-       BUG_ON(in_nmi());
+       if (unlikely(in_interrupt())) {
+               vfree_atomic(addr);
+               return;
+       }
 
+       BUG_ON(in_nmi());
        kmemleak_free(addr);
+       might_sleep();
 
-       might_sleep_if(!in_interrupt());
-
-       if (!addr)
-               return;
-       if (unlikely(in_interrupt()))
-               __vfree_deferred(addr);
-       else
+       if (addr)
                __vunmap(addr, 1);
 }
 EXPORT_SYMBOL(vfree);