#include <linux/kernel.h>
 #include <linux/sched.h>
 #include <linux/wait.h>
+#include <linux/delay.h>
 #include <linux/mm.h>
 #include <linux/slab.h>
 #include <linux/list.h>
        void (*message_handler)(struct vmbus_channel_message_header *msg);
 };
 
+struct vmbus_rescind_work {
+       struct work_struct work;
+       struct vmbus_channel *channel;
+};
 
 /**
  * vmbus_prep_negotiate_resp() - Create default response for Hyper-V Negotiate message
 
 EXPORT_SYMBOL_GPL(vmbus_prep_negotiate_resp);
 
-static void vmbus_process_device_unregister(struct work_struct *work)
-{
-       struct device *dev;
-       struct vmbus_channel *channel = container_of(work,
-                                                       struct vmbus_channel,
-                                                       work);
-
-       dev = get_device(&channel->device_obj->device);
-       if (dev) {
-               vmbus_device_unregister(channel->device_obj);
-               put_device(dev);
-       }
-}
-
 static void vmbus_sc_creation_cb(struct work_struct *work)
 {
        struct vmbus_channel *newchannel = container_of(work,
        queue_work(vmbus_connection.work_queue, &channel->work);
 }
 
+static void process_rescind_fn(struct work_struct *work)
+{
+       struct vmbus_rescind_work *rc_work;
+       struct vmbus_channel *channel;
+       struct device *dev;
+
+       rc_work = container_of(work, struct vmbus_rescind_work, work);
+       channel = rc_work->channel;
+
+       /*
+        * We have already acquired a reference on the channel
+        * and so it cannot vanish underneath us.
+        * It is possible (while very unlikely) that we may
+        * get here while the processing of the initial offer
+        * is still not complete. Deal with this situation by
+        * just waiting until the channel is in the correct state.
+        */
+
+       while (channel->work.func != release_channel)
+               msleep(1000);
+
+       if (channel->device_obj) {
+               dev = get_device(&channel->device_obj->device);
+               if (dev) {
+                       vmbus_device_unregister(channel->device_obj);
+                       put_device(dev);
+               }
+       } else {
+               hv_process_channel_removal(channel,
+                                          channel->offermsg.child_relid);
+       }
+       kfree(work);
+}
+
 static void percpu_channel_enq(void *arg)
 {
        struct vmbus_channel *channel = arg;
        }
 }
 
+static void vmbus_do_device_register(struct work_struct *work)
+{
+       struct hv_device *device_obj;
+       int ret;
+       unsigned long flags;
+       struct vmbus_channel *newchannel = container_of(work,
+                                                    struct vmbus_channel,
+                                                    work);
+
+       ret = vmbus_device_register(newchannel->device_obj);
+       if (ret != 0) {
+               pr_err("unable to add child device object (relid %d)\n",
+                       newchannel->offermsg.child_relid);
+               spin_lock_irqsave(&vmbus_connection.channel_lock, flags);
+               list_del(&newchannel->listentry);
+               device_obj = newchannel->device_obj;
+               newchannel->device_obj = NULL;
+               spin_unlock_irqrestore(&vmbus_connection.channel_lock, flags);
+
+               if (newchannel->target_cpu != get_cpu()) {
+                       put_cpu();
+                       smp_call_function_single(newchannel->target_cpu,
+                                        percpu_channel_deq, newchannel, true);
+               } else {
+                       percpu_channel_deq(newchannel);
+                       put_cpu();
+               }
+
+               kfree(device_obj);
+               if (!newchannel->rescind) {
+                       free_channel(newchannel);
+                       return;
+               }
+       }
+       /*
+        * The next state for this channel is to be freed.
+        */
+       INIT_WORK(&newchannel->work, release_channel);
+}
+
 /*
  * vmbus_process_offer - Process the offer by creating a channel/device
  * associated with this offer
        struct vmbus_channel *channel;
        bool fnew = true;
        bool enq = false;
-       int ret;
        unsigned long flags;
 
        /* Make sure this is a new offer */
         * Add the new device to the bus. This will kick off device-driver
         * binding which eventually invokes the device driver's AddDevice()
         * method.
+        * Invoke this call on the per-channel work context.
+        * Until we return from this function, rescind offer message
+        * cannot be processed as we are running on the global message
+        * handling work.
         */
-       ret = vmbus_device_register(newchannel->device_obj);
-       if (ret != 0) {
-               pr_err("unable to add child device object (relid %d)\n",
-                          newchannel->offermsg.child_relid);
-
-               kfree(newchannel->device_obj);
-               goto err_deq_chan;
-       }
-
+       INIT_WORK(&newchannel->work, vmbus_do_device_register);
+       queue_work(newchannel->controlwq, &newchannel->work);
        return;
 
 err_deq_chan:
 {
        struct vmbus_channel_rescind_offer *rescind;
        struct vmbus_channel *channel;
-       unsigned long flags;
+       struct vmbus_rescind_work *rc_work;
 
        rescind = (struct vmbus_channel_rescind_offer *)hdr;
-       channel = relid2channel(rescind->child_relid);
+       channel = relid2channel(rescind->child_relid, true);
 
        if (channel == NULL) {
                hv_process_channel_removal(NULL, rescind->child_relid);
                return;
        }
 
-       spin_lock_irqsave(&channel->lock, flags);
-       channel->rescind = true;
-       spin_unlock_irqrestore(&channel->lock, flags);
-
-       if (channel->device_obj) {
-               /*
-                * We will have to unregister this device from the
-                * driver core. Do this in the per-channel work context.
-                * Note that we are currently executing on the global
-                * workq for handling messages from the host.
-                */
-               INIT_WORK(&channel->work, vmbus_process_device_unregister);
-               queue_work(channel->controlwq, &channel->work);
-       } else {
-               hv_process_channel_removal(channel,
-                                          channel->offermsg.child_relid);
+       /*
+        * We have acquired a reference on the channel and have posted
+        * the rescind state. Perform further cleanup in a work context
+        * that is different from the global work context in which
+        * we process messages from the host (we are currently executing
+        * on that global context.
+        */
+       rc_work = kzalloc(sizeof(struct vmbus_rescind_work), GFP_KERNEL);
+       if (!rc_work) {
+               pr_err("Unable to allocate memory for rescind processing ");
+               return;
        }
+       rc_work->channel = channel;
+       INIT_WORK(&rc_work->work, process_rescind_fn);
+       schedule_work(&rc_work->work);
 }
 
 /*