#include <linux/shmem_fs.h>
 #include <linux/khugepaged.h>
 #include <linux/rcupdate_trace.h>
+#include <linux/workqueue.h>
 
 #include <linux/uprobes.h>
 
        struct list_head        pending_list;
        struct list_head        consumers;
        struct inode            *inode;         /* Also hold a ref to inode */
-       struct rcu_head         rcu;
+       union {
+               struct rcu_head         rcu;
+               struct work_struct      work;
+       };
        loff_t                  offset;
        loff_t                  ref_ctr_offset;
        unsigned long           flags;          /* "unsigned long" so bitops work */
        kfree(uprobe);
 }
 
-static void put_uprobe(struct uprobe *uprobe)
+static void uprobe_free_deferred(struct work_struct *work)
 {
-       if (!refcount_dec_and_test(&uprobe->ref))
-               return;
+       struct uprobe *uprobe = container_of(work, struct uprobe, work);
 
        write_lock(&uprobes_treelock);
 
        call_rcu_tasks_trace(&uprobe->rcu, uprobe_free_rcu);
 }
 
+static void put_uprobe(struct uprobe *uprobe)
+{
+       if (!refcount_dec_and_test(&uprobe->ref))
+               return;
+
+       INIT_WORK(&uprobe->work, uprobe_free_deferred);
+       schedule_work(&uprobe->work);
+}
+
 static __always_inline
 int uprobe_cmp(const struct inode *l_inode, const loff_t l_offset,
               const struct uprobe *r)