diff options
author | Michael S. Tsirkin <mst@redhat.com> | 2014-10-13 20:10:34 -0400 |
---|---|---|
committer | Rusty Russell <rusty@rustcorp.com.au> | 2014-10-14 19:54:54 -0400 |
commit | 016c98c6fe0c914d12e2e242b2bccde6d6dea54b (patch) | |
tree | 6e4fb0faa50c839cc23857954bf51f9985fb93ea | |
parent | 6fbc198cf623944ab60a1db6d306a4d55cdd820d (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.c | 6 | ||||
-rw-r--r-- | drivers/s390/kvm/kvm_virtio.c | 9 | ||||
-rw-r--r-- | drivers/s390/kvm/virtio_ccw.c | 6 | ||||
-rw-r--r-- | drivers/virtio/virtio.c | 9 | ||||
-rw-r--r-- | drivers/virtio/virtio_mmio.c | 7 | ||||
-rw-r--r-- | drivers/virtio/virtio_pci.c | 6 | ||||
-rw-r--r-- | include/linux/virtio.h | 2 |
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 | } |
240 | EXPORT_SYMBOL_GPL(unregister_virtio_device); | 240 | EXPORT_SYMBOL_GPL(unregister_virtio_device); |
241 | 241 | ||
242 | void 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 | } | ||
249 | EXPORT_SYMBOL_GPL(virtio_config_changed); | ||
250 | |||
242 | static int virtio_init(void) | 251 | static 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) | |||
211 | static irqreturn_t vp_config_changed(int irq, void *opaque) | 211 | static 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 | ||
109 | void virtio_break_device(struct virtio_device *dev); | 109 | void virtio_break_device(struct virtio_device *dev); |
110 | 110 | ||
111 | void 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). |