#include <linux/cdev.h>
 #include <linux/slab.h>
 #include <linux/module.h>
+#include <linux/kobject.h>
 
 #include <linux/uaccess.h>
 #include <asm/cio.h>
 #include <asm/ccwdev.h>
 #include <asm/debug.h>
 #include <asm/diag.h>
+#include <asm/scsw.h>
 
 #include "vmur.h"
 
 
 static DEFINE_MUTEX(vmur_mutex);
 
+static void ur_uevent(struct work_struct *ws);
+
 /*
  * Allocation, freeing, getting and putting of urdev structures
  *
        ccw_device_get_id(cdev, &urd->dev_id);
        mutex_init(&urd->io_mutex);
        init_waitqueue_head(&urd->wait);
+       INIT_WORK(&urd->uevent_work, ur_uevent);
        spin_lock_init(&urd->open_lock);
        refcount_set(&urd->ref_count,  1);
        urd->cdev = cdev;
        return rc;
 }
 
+static void ur_uevent(struct work_struct *ws)
+{
+       struct urdev *urd = container_of(ws, struct urdev, uevent_work);
+       char *envp[] = {
+               "EVENT=unsol_de",       /* Unsolicited device-end interrupt */
+               NULL
+       };
+
+       kobject_uevent_env(&urd->cdev->dev.kobj, KOBJ_CHANGE, envp);
+       urdev_put(urd);
+}
+
 /*
  * ur interrupt handler, called from the ccw_device layer
  */
                      intparm, irb->scsw.cmd.cstat, irb->scsw.cmd.dstat,
                      irb->scsw.cmd.count);
        }
+       urd = dev_get_drvdata(&cdev->dev);
        if (!intparm) {
                TRACE("ur_int_handler: unsolicited interrupt\n");
+
+               if (scsw_dstat(&irb->scsw) & DEV_STAT_DEV_END) {
+                       /*
+                        * Userspace might be interested in a transition to
+                        * device-ready state.
+                        */
+                       urdev_get(urd);
+                       schedule_work(&urd->uevent_work);
+               }
+
                return;
        }
-       urd = dev_get_drvdata(&cdev->dev);
        /* On special conditions irb is an error pointer */
        if (IS_ERR(irb))
                urd->io_request_rc = PTR_ERR(irb);
                rc = -EBUSY;
                goto fail_urdev_put;
        }
+       if (cancel_work_sync(&urd->uevent_work)) {
+               /* Work not run yet - need to release reference here */
+               urdev_put(urd);
+       }
        device_destroy(vmur_class, urd->char_device->dev);
        cdev_del(urd->char_device);
        urd->char_device = NULL;
 
 #define _VMUR_H_
 
 #include <linux/refcount.h>
+#include <linux/workqueue.h>
 
 #define DEV_CLASS_UR_I 0x20 /* diag210 unit record input device class */
 #define DEV_CLASS_UR_O 0x10 /* diag210 unit record output device class */
        wait_queue_head_t wait;         /* wait queue to serialize open */
        int open_flag;                  /* "urdev is open" flag */
        spinlock_t open_lock;           /* serialize critical sections */
+       struct work_struct uevent_work; /* work to send uevent */
 };
 
 /*