aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMichael S. Tsirkin <mst@redhat.com>2014-10-13 20:10:34 -0400
committerRusty Russell <rusty@rustcorp.com.au>2014-10-14 19:54:54 -0400
commit016c98c6fe0c914d12e2e242b2bccde6d6dea54b (patch)
tree6e4fb0faa50c839cc23857954bf51f9985fb93ea
parent6fbc198cf623944ab60a1db6d306a4d55cdd820d (diff)
virtio: unify config_changed handling
Replace duplicated code in all transports with a single wrapper in virtio.c. The only functional change is in virtio_mmio.c: if a buggy device sends us an interrupt before driver is set, we previously returned IRQ_NONE, now we return IRQ_HANDLED. As this must not happen in practice, this does not look like a big deal. See also commit 3fff0179e33cd7d0a688dab65700c46ad089e934 virtio-pci: do not oops on config change if driver not loaded. for the original motivation behind the driver check. Signed-off-by: Michael S. Tsirkin <mst@redhat.com> Reviewed-by: Cornelia Huck <cornelia.huck@de.ibm.com> Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
-rw-r--r--drivers/misc/mic/card/mic_virtio.c6
-rw-r--r--drivers/s390/kvm/kvm_virtio.c9
-rw-r--r--drivers/s390/kvm/virtio_ccw.c6
-rw-r--r--drivers/virtio/virtio.c9
-rw-r--r--drivers/virtio/virtio_mmio.c7
-rw-r--r--drivers/virtio/virtio_pci.c6
-rw-r--r--include/linux/virtio.h2
7 files changed, 17 insertions, 28 deletions
diff --git a/drivers/misc/mic/card/mic_virtio.c b/drivers/misc/mic/card/mic_virtio.c
index f14b60080c21..e64794730e21 100644
--- a/drivers/misc/mic/card/mic_virtio.c
+++ b/drivers/misc/mic/card/mic_virtio.c
@@ -462,16 +462,12 @@ static void mic_handle_config_change(struct mic_device_desc __iomem *d,
462 struct mic_device_ctrl __iomem *dc 462 struct mic_device_ctrl __iomem *dc
463 = (void __iomem *)d + mic_aligned_desc_size(d); 463 = (void __iomem *)d + mic_aligned_desc_size(d);
464 struct mic_vdev *mvdev = (struct mic_vdev *)ioread64(&dc->vdev); 464 struct mic_vdev *mvdev = (struct mic_vdev *)ioread64(&dc->vdev);
465 struct virtio_driver *drv;
466 465
467 if (ioread8(&dc->config_change) != MIC_VIRTIO_PARAM_CONFIG_CHANGED) 466 if (ioread8(&dc->config_change) != MIC_VIRTIO_PARAM_CONFIG_CHANGED)
468 return; 467 return;
469 468
470 dev_dbg(mdrv->dev, "%s %d\n", __func__, __LINE__); 469 dev_dbg(mdrv->dev, "%s %d\n", __func__, __LINE__);
471 drv = container_of(mvdev->vdev.dev.driver, 470 virtio_config_changed(&mvdev->vdev);
472 struct virtio_driver, driver);
473 if (drv->config_changed)
474 drv->config_changed(&mvdev->vdev);
475 iowrite8(1, &dc->guest_ack); 471 iowrite8(1, &dc->guest_ack);
476} 472}
477 473
diff --git a/drivers/s390/kvm/kvm_virtio.c b/drivers/s390/kvm/kvm_virtio.c
index a1349653c6d9..643129070c51 100644
--- a/drivers/s390/kvm/kvm_virtio.c
+++ b/drivers/s390/kvm/kvm_virtio.c
@@ -406,15 +406,8 @@ static void kvm_extint_handler(struct ext_code ext_code,
406 406
407 switch (param) { 407 switch (param) {
408 case VIRTIO_PARAM_CONFIG_CHANGED: 408 case VIRTIO_PARAM_CONFIG_CHANGED:
409 { 409 virtio_config_changed(vq->vdev);
410 struct virtio_driver *drv;
411 drv = container_of(vq->vdev->dev.driver,
412 struct virtio_driver, driver);
413 if (drv->config_changed)
414 drv->config_changed(vq->vdev);
415
416 break; 410 break;
417 }
418 case VIRTIO_PARAM_DEV_ADD: 411 case VIRTIO_PARAM_DEV_ADD:
419 schedule_work(&hotplug_work); 412 schedule_work(&hotplug_work);
420 break; 413 break;
diff --git a/drivers/s390/kvm/virtio_ccw.c b/drivers/s390/kvm/virtio_ccw.c
index d2c0b442bce5..6cbe6ef3c889 100644
--- a/drivers/s390/kvm/virtio_ccw.c
+++ b/drivers/s390/kvm/virtio_ccw.c
@@ -940,11 +940,7 @@ static void virtio_ccw_int_handler(struct ccw_device *cdev,
940 vring_interrupt(0, vq); 940 vring_interrupt(0, vq);
941 } 941 }
942 if (test_bit(0, &vcdev->indicators2)) { 942 if (test_bit(0, &vcdev->indicators2)) {
943 drv = container_of(vcdev->vdev.dev.driver, 943 virtio_config_changed(&vcdev->vdev);
944 struct virtio_driver, driver);
945
946 if (drv && drv->config_changed)
947 drv->config_changed(&vcdev->vdev);
948 clear_bit(0, &vcdev->indicators2); 944 clear_bit(0, &vcdev->indicators2);
949 } 945 }
950} 946}
diff --git a/drivers/virtio/virtio.c b/drivers/virtio/virtio.c
index fed0ce198ae3..3980687401f6 100644
--- a/drivers/virtio/virtio.c
+++ b/drivers/virtio/virtio.c
@@ -239,6 +239,15 @@ void unregister_virtio_device(struct virtio_device *dev)
239} 239}
240EXPORT_SYMBOL_GPL(unregister_virtio_device); 240EXPORT_SYMBOL_GPL(unregister_virtio_device);
241 241
242void virtio_config_changed(struct virtio_device *dev)
243{
244 struct virtio_driver *drv = drv_to_virtio(dev->dev.driver);
245
246 if (drv && drv->config_changed)
247 drv->config_changed(dev);
248}
249EXPORT_SYMBOL_GPL(virtio_config_changed);
250
242static int virtio_init(void) 251static int virtio_init(void)
243{ 252{
244 if (bus_register(&virtio_bus) != 0) 253 if (bus_register(&virtio_bus) != 0)
diff --git a/drivers/virtio/virtio_mmio.c b/drivers/virtio/virtio_mmio.c
index c600ccfd6922..ef9a1650bb80 100644
--- a/drivers/virtio/virtio_mmio.c
+++ b/drivers/virtio/virtio_mmio.c
@@ -234,8 +234,6 @@ static irqreturn_t vm_interrupt(int irq, void *opaque)
234{ 234{
235 struct virtio_mmio_device *vm_dev = opaque; 235 struct virtio_mmio_device *vm_dev = opaque;
236 struct virtio_mmio_vq_info *info; 236 struct virtio_mmio_vq_info *info;
237 struct virtio_driver *vdrv = container_of(vm_dev->vdev.dev.driver,
238 struct virtio_driver, driver);
239 unsigned long status; 237 unsigned long status;
240 unsigned long flags; 238 unsigned long flags;
241 irqreturn_t ret = IRQ_NONE; 239 irqreturn_t ret = IRQ_NONE;
@@ -244,9 +242,8 @@ static irqreturn_t vm_interrupt(int irq, void *opaque)
244 status = readl(vm_dev->base + VIRTIO_MMIO_INTERRUPT_STATUS); 242 status = readl(vm_dev->base + VIRTIO_MMIO_INTERRUPT_STATUS);
245 writel(status, vm_dev->base + VIRTIO_MMIO_INTERRUPT_ACK); 243 writel(status, vm_dev->base + VIRTIO_MMIO_INTERRUPT_ACK);
246 244
247 if (unlikely(status & VIRTIO_MMIO_INT_CONFIG) 245 if (unlikely(status & VIRTIO_MMIO_INT_CONFIG)) {
248 && vdrv && vdrv->config_changed) { 246 virtio_config_changed(&vm_dev->vdev);
249 vdrv->config_changed(&vm_dev->vdev);
250 ret = IRQ_HANDLED; 247 ret = IRQ_HANDLED;
251 } 248 }
252 249
diff --git a/drivers/virtio/virtio_pci.c b/drivers/virtio/virtio_pci.c
index add40d00dcdb..f39f4e772e6a 100644
--- a/drivers/virtio/virtio_pci.c
+++ b/drivers/virtio/virtio_pci.c
@@ -211,12 +211,8 @@ static bool vp_notify(struct virtqueue *vq)
211static irqreturn_t vp_config_changed(int irq, void *opaque) 211static irqreturn_t vp_config_changed(int irq, void *opaque)
212{ 212{
213 struct virtio_pci_device *vp_dev = opaque; 213 struct virtio_pci_device *vp_dev = opaque;
214 struct virtio_driver *drv;
215 drv = container_of(vp_dev->vdev.dev.driver,
216 struct virtio_driver, driver);
217 214
218 if (drv && drv->config_changed) 215 virtio_config_changed(&vp_dev->vdev);
219 drv->config_changed(&vp_dev->vdev);
220 return IRQ_HANDLED; 216 return IRQ_HANDLED;
221} 217}
222 218
diff --git a/include/linux/virtio.h b/include/linux/virtio.h
index b46671e28de2..3c19bd3189cb 100644
--- a/include/linux/virtio.h
+++ b/include/linux/virtio.h
@@ -108,6 +108,8 @@ void unregister_virtio_device(struct virtio_device *dev);
108 108
109void virtio_break_device(struct virtio_device *dev); 109void virtio_break_device(struct virtio_device *dev);
110 110
111void virtio_config_changed(struct virtio_device *dev);
112
111/** 113/**
112 * virtio_driver - operations for a virtio I/O driver 114 * virtio_driver - operations for a virtio I/O driver
113 * @driver: underlying device driver (populate name and owner). 115 * @driver: underlying device driver (populate name and owner).