/*
* drivers/s390/cio/device.c
* bus driver for ccw devices
- * $Revision: 1.124 $
+ * $Revision: 1.129 $
*
* Copyright (C) 2002 IBM Deutschland Entwicklung GmbH,
* IBM Corporation
if (!cdev)
return -ENODEV;
- if (cdev->private->state == DEV_STATE_NOT_OPER)
- return -ENODEV;
-
/* what we want to pass to /sbin/hotplug */
envp[i++] = buffer;
if ((ret = device_add(dev)))
return ret;
- if ((ret = device_add_files(dev)))
- device_del(dev);
-
+ set_bit(1, &cdev->private->registered);
+ if ((ret = device_add_files(dev))) {
+ if (test_and_clear_bit(1, &cdev->private->registered))
+ device_del(dev);
+ }
return ret;
}
cdev = to_ccwdev(dev);
if ((cdev->private->state == DEV_STATE_DISCONNECTED) &&
(cdev->private->devno == devno) &&
- (!strncmp(cdev->dev.bus_id, sibling->dev.bus_id,
- BUS_ID_SIZE))) {
+ (cdev != sibling)) {
cdev->private->state = DEV_STATE_NOT_OPER;
break;
}
return cdev;
}
+static void
+ccw_device_add_changed(void *data)
+{
+
+ struct ccw_device *cdev;
+
+ cdev = (struct ccw_device *)data;
+ if (device_add(&cdev->dev)) {
+ put_device(&cdev->dev);
+ return;
+ }
+ set_bit(1, &cdev->private->registered);
+ if (device_add_files(&cdev->dev)) {
+ if (test_and_clear_bit(1, &cdev->private->registered))
+ device_unregister(&cdev->dev);
+ }
+}
+
extern int css_get_ssd_info(struct subchannel *sch);
void
} else
need_rename = 0;
device_remove_files(&cdev->dev);
- device_del(&cdev->dev);
+ if (test_and_clear_bit(1, &cdev->private->registered))
+ device_del(&cdev->dev);
if (need_rename)
snprintf (cdev->dev.bus_id, BUS_ID_SIZE, "0.0.%04x",
sch->schib.pmcw.dev);
- if (device_add(&cdev->dev)) {
- put_device(&cdev->dev);
- return;
- }
- if (device_add_files(&cdev->dev))
- device_unregister(&cdev->dev);
+ PREPARE_WORK(&cdev->private->kick_work,
+ ccw_device_add_changed, (void *)cdev);
+ queue_work(ccw_device_work, &cdev->private->kick_work);
}
static void
struct ccw_device *cdev;
struct subchannel *sch;
int ret;
+ unsigned long flags;
cdev = (struct ccw_device *) data;
sch = to_subchannel(cdev->dev.parent);
printk (KERN_WARNING "%s: could not register %s\n",
__func__, cdev->dev.bus_id);
put_device(&cdev->dev);
- sch->dev.driver_data = 0;
+ spin_lock_irqsave(&sch->lock, flags);
+ sch->dev.driver_data = NULL;
+ spin_unlock_irqrestore(&sch->lock, flags);
kfree (cdev->private);
kfree (cdev);
put_device(&sch->dev);
+ if (atomic_dec_and_test(&ccw_device_init_count))
+ wake_up(&ccw_device_init_wq);
return;
}
cdev->private->flags.recog_done = 1;
put_device(&sch->dev);
wake_up(&cdev->private->wait_q);
+ if (atomic_dec_and_test(&ccw_device_init_count))
+ wake_up(&ccw_device_init_wq);
}
void
if (!get_device(&cdev->dev))
break;
sch = to_subchannel(cdev->dev.parent);
- INIT_WORK(&cdev->private->kick_work,
- ccw_device_call_sch_unregister, (void *) cdev);
+ PREPARE_WORK(&cdev->private->kick_work,
+ ccw_device_call_sch_unregister, (void *) cdev);
queue_work(slow_path_wq, &cdev->private->kick_work);
+ if (atomic_dec_and_test(&ccw_device_init_count))
+ wake_up(&ccw_device_init_wq);
break;
case DEV_STATE_BOXED:
/* Device did not respond in time. */
*/
if (!get_device(&cdev->dev))
break;
- INIT_WORK(&cdev->private->kick_work,
- io_subchannel_register, (void *) cdev);
- queue_work(ccw_device_work, &cdev->private->kick_work);
+ PREPARE_WORK(&cdev->private->kick_work,
+ io_subchannel_register, (void *) cdev);
+ queue_work(slow_path_wq, &cdev->private->kick_work);
break;
}
- if (atomic_dec_and_test(&ccw_device_init_count))
- wake_up(&ccw_device_init_wq);
}
static int
struct subchannel *sch;
struct ccw_device *cdev;
int rc;
+ unsigned long flags;
sch = to_subchannel(pdev);
if (sch->dev.driver_data) {
.parent = pdev,
.release = ccw_device_release,
};
+ INIT_LIST_HEAD(&cdev->private->kick_work.entry);
/* Do first half of device_register. */
device_initialize(&cdev->dev);
rc = io_subchannel_recog(cdev, to_subchannel(pdev));
if (rc) {
- sch->dev.driver_data = 0;
+ spin_lock_irqsave(&sch->lock, flags);
+ sch->dev.driver_data = NULL;
+ spin_unlock_irqrestore(&sch->lock, flags);
if (cdev->dev.release)
cdev->dev.release(&cdev->dev);
}
return rc;
}
+static void
+ccw_device_unregister(void *data)
+{
+ struct ccw_device *cdev;
+
+ cdev = (struct ccw_device *)data;
+ if (test_and_clear_bit(1, &cdev->private->registered))
+ device_unregister(&cdev->dev);
+ put_device(&cdev->dev);
+}
+
static int
io_subchannel_remove (struct device *dev)
{
struct ccw_device *cdev;
+ unsigned long flags;
if (!dev->driver_data)
return 0;
cdev = dev->driver_data;
/* Set ccw device to not operational and drop reference. */
+ spin_lock_irqsave(cdev->ccwlock, flags);
+ dev->driver_data = NULL;
cdev->private->state = DEV_STATE_NOT_OPER;
+ spin_unlock_irqrestore(cdev->ccwlock, flags);
/*
- * Careful here. Our ccw device might be yet unregistered when
- * de-registering its subchannel (machine check during device
- * recognition). Better look if the subchannel has children.
+ * Put unregistration on workqueue to avoid livelocks on the css bus
+ * semaphore.
*/
- if (!list_empty(&dev->children))
- device_unregister(&cdev->dev);
- dev->driver_data = NULL;
+ if (get_device(&cdev->dev)) {
+ PREPARE_WORK(&cdev->private->kick_work,
+ ccw_device_unregister, (void *) cdev);
+ queue_work(ccw_device_work, &cdev->private->kick_work);
+ }
return 0;
}