[media] rc: Postpone ISR registration
An early registration of an ISR was causing a crash to several users (for example, with the ite-cir driver: http://bugs.launchpad.net/bugs/972723). The reason was that IRQs were being triggered before a driver initialisation was completed. This patch fixes this by moving the invocation to request_irq() and to request_region() to a later stage on the driver probe function. Cc: <stable@vger.kernel.org> Signed-off-by: Luis Henriques <luis.henriques@canonical.com> Acked-by: Jarod Wilson <jarod@redhat.com> Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
This commit is contained in:
parent
9967232f1b
commit
9ef449c6b3
5 changed files with 93 additions and 93 deletions
|
@ -1018,22 +1018,6 @@ static int ene_probe(struct pnp_dev *pnp_dev, const struct pnp_device_id *id)
|
||||||
|
|
||||||
spin_lock_init(&dev->hw_lock);
|
spin_lock_init(&dev->hw_lock);
|
||||||
|
|
||||||
/* claim the resources */
|
|
||||||
error = -EBUSY;
|
|
||||||
dev->hw_io = pnp_port_start(pnp_dev, 0);
|
|
||||||
if (!request_region(dev->hw_io, ENE_IO_SIZE, ENE_DRIVER_NAME)) {
|
|
||||||
dev->hw_io = -1;
|
|
||||||
dev->irq = -1;
|
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
|
|
||||||
dev->irq = pnp_irq(pnp_dev, 0);
|
|
||||||
if (request_irq(dev->irq, ene_isr,
|
|
||||||
IRQF_SHARED, ENE_DRIVER_NAME, (void *)dev)) {
|
|
||||||
dev->irq = -1;
|
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
|
|
||||||
pnp_set_drvdata(pnp_dev, dev);
|
pnp_set_drvdata(pnp_dev, dev);
|
||||||
dev->pnp_dev = pnp_dev;
|
dev->pnp_dev = pnp_dev;
|
||||||
|
|
||||||
|
@ -1086,6 +1070,22 @@ static int ene_probe(struct pnp_dev *pnp_dev, const struct pnp_device_id *id)
|
||||||
device_set_wakeup_capable(&pnp_dev->dev, true);
|
device_set_wakeup_capable(&pnp_dev->dev, true);
|
||||||
device_set_wakeup_enable(&pnp_dev->dev, true);
|
device_set_wakeup_enable(&pnp_dev->dev, true);
|
||||||
|
|
||||||
|
/* claim the resources */
|
||||||
|
error = -EBUSY;
|
||||||
|
dev->hw_io = pnp_port_start(pnp_dev, 0);
|
||||||
|
if (!request_region(dev->hw_io, ENE_IO_SIZE, ENE_DRIVER_NAME)) {
|
||||||
|
dev->hw_io = -1;
|
||||||
|
dev->irq = -1;
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
|
dev->irq = pnp_irq(pnp_dev, 0);
|
||||||
|
if (request_irq(dev->irq, ene_isr,
|
||||||
|
IRQF_SHARED, ENE_DRIVER_NAME, (void *)dev)) {
|
||||||
|
dev->irq = -1;
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
error = rc_register_device(rdev);
|
error = rc_register_device(rdev);
|
||||||
if (error < 0)
|
if (error < 0)
|
||||||
goto error;
|
goto error;
|
||||||
|
|
|
@ -514,16 +514,6 @@ static int fintek_probe(struct pnp_dev *pdev, const struct pnp_device_id *dev_id
|
||||||
|
|
||||||
spin_lock_init(&fintek->fintek_lock);
|
spin_lock_init(&fintek->fintek_lock);
|
||||||
|
|
||||||
ret = -EBUSY;
|
|
||||||
/* now claim resources */
|
|
||||||
if (!request_region(fintek->cir_addr,
|
|
||||||
fintek->cir_port_len, FINTEK_DRIVER_NAME))
|
|
||||||
goto failure;
|
|
||||||
|
|
||||||
if (request_irq(fintek->cir_irq, fintek_cir_isr, IRQF_SHARED,
|
|
||||||
FINTEK_DRIVER_NAME, (void *)fintek))
|
|
||||||
goto failure;
|
|
||||||
|
|
||||||
pnp_set_drvdata(pdev, fintek);
|
pnp_set_drvdata(pdev, fintek);
|
||||||
fintek->pdev = pdev;
|
fintek->pdev = pdev;
|
||||||
|
|
||||||
|
@ -558,6 +548,16 @@ static int fintek_probe(struct pnp_dev *pdev, const struct pnp_device_id *dev_id
|
||||||
/* rx resolution is hardwired to 50us atm, 1, 25, 100 also possible */
|
/* rx resolution is hardwired to 50us atm, 1, 25, 100 also possible */
|
||||||
rdev->rx_resolution = US_TO_NS(CIR_SAMPLE_PERIOD);
|
rdev->rx_resolution = US_TO_NS(CIR_SAMPLE_PERIOD);
|
||||||
|
|
||||||
|
ret = -EBUSY;
|
||||||
|
/* now claim resources */
|
||||||
|
if (!request_region(fintek->cir_addr,
|
||||||
|
fintek->cir_port_len, FINTEK_DRIVER_NAME))
|
||||||
|
goto failure;
|
||||||
|
|
||||||
|
if (request_irq(fintek->cir_irq, fintek_cir_isr, IRQF_SHARED,
|
||||||
|
FINTEK_DRIVER_NAME, (void *)fintek))
|
||||||
|
goto failure;
|
||||||
|
|
||||||
ret = rc_register_device(rdev);
|
ret = rc_register_device(rdev);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto failure;
|
goto failure;
|
||||||
|
|
|
@ -1515,16 +1515,6 @@ static int ite_probe(struct pnp_dev *pdev, const struct pnp_device_id
|
||||||
/* initialize raw event */
|
/* initialize raw event */
|
||||||
init_ir_raw_event(&itdev->rawir);
|
init_ir_raw_event(&itdev->rawir);
|
||||||
|
|
||||||
ret = -EBUSY;
|
|
||||||
/* now claim resources */
|
|
||||||
if (!request_region(itdev->cir_addr,
|
|
||||||
dev_desc->io_region_size, ITE_DRIVER_NAME))
|
|
||||||
goto failure;
|
|
||||||
|
|
||||||
if (request_irq(itdev->cir_irq, ite_cir_isr, IRQF_SHARED,
|
|
||||||
ITE_DRIVER_NAME, (void *)itdev))
|
|
||||||
goto failure;
|
|
||||||
|
|
||||||
/* set driver data into the pnp device */
|
/* set driver data into the pnp device */
|
||||||
pnp_set_drvdata(pdev, itdev);
|
pnp_set_drvdata(pdev, itdev);
|
||||||
itdev->pdev = pdev;
|
itdev->pdev = pdev;
|
||||||
|
@ -1600,6 +1590,16 @@ static int ite_probe(struct pnp_dev *pdev, const struct pnp_device_id
|
||||||
rdev->driver_name = ITE_DRIVER_NAME;
|
rdev->driver_name = ITE_DRIVER_NAME;
|
||||||
rdev->map_name = RC_MAP_RC6_MCE;
|
rdev->map_name = RC_MAP_RC6_MCE;
|
||||||
|
|
||||||
|
ret = -EBUSY;
|
||||||
|
/* now claim resources */
|
||||||
|
if (!request_region(itdev->cir_addr,
|
||||||
|
dev_desc->io_region_size, ITE_DRIVER_NAME))
|
||||||
|
goto failure;
|
||||||
|
|
||||||
|
if (request_irq(itdev->cir_irq, ite_cir_isr, IRQF_SHARED,
|
||||||
|
ITE_DRIVER_NAME, (void *)itdev))
|
||||||
|
goto failure;
|
||||||
|
|
||||||
ret = rc_register_device(rdev);
|
ret = rc_register_device(rdev);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto failure;
|
goto failure;
|
||||||
|
|
|
@ -1021,24 +1021,6 @@ static int nvt_probe(struct pnp_dev *pdev, const struct pnp_device_id *dev_id)
|
||||||
spin_lock_init(&nvt->nvt_lock);
|
spin_lock_init(&nvt->nvt_lock);
|
||||||
spin_lock_init(&nvt->tx.lock);
|
spin_lock_init(&nvt->tx.lock);
|
||||||
|
|
||||||
ret = -EBUSY;
|
|
||||||
/* now claim resources */
|
|
||||||
if (!request_region(nvt->cir_addr,
|
|
||||||
CIR_IOREG_LENGTH, NVT_DRIVER_NAME))
|
|
||||||
goto failure;
|
|
||||||
|
|
||||||
if (request_irq(nvt->cir_irq, nvt_cir_isr, IRQF_SHARED,
|
|
||||||
NVT_DRIVER_NAME, (void *)nvt))
|
|
||||||
goto failure;
|
|
||||||
|
|
||||||
if (!request_region(nvt->cir_wake_addr,
|
|
||||||
CIR_IOREG_LENGTH, NVT_DRIVER_NAME))
|
|
||||||
goto failure;
|
|
||||||
|
|
||||||
if (request_irq(nvt->cir_wake_irq, nvt_cir_wake_isr, IRQF_SHARED,
|
|
||||||
NVT_DRIVER_NAME, (void *)nvt))
|
|
||||||
goto failure;
|
|
||||||
|
|
||||||
pnp_set_drvdata(pdev, nvt);
|
pnp_set_drvdata(pdev, nvt);
|
||||||
nvt->pdev = pdev;
|
nvt->pdev = pdev;
|
||||||
|
|
||||||
|
@ -1085,6 +1067,24 @@ static int nvt_probe(struct pnp_dev *pdev, const struct pnp_device_id *dev_id)
|
||||||
rdev->tx_resolution = XYZ;
|
rdev->tx_resolution = XYZ;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
ret = -EBUSY;
|
||||||
|
/* now claim resources */
|
||||||
|
if (!request_region(nvt->cir_addr,
|
||||||
|
CIR_IOREG_LENGTH, NVT_DRIVER_NAME))
|
||||||
|
goto failure;
|
||||||
|
|
||||||
|
if (request_irq(nvt->cir_irq, nvt_cir_isr, IRQF_SHARED,
|
||||||
|
NVT_DRIVER_NAME, (void *)nvt))
|
||||||
|
goto failure;
|
||||||
|
|
||||||
|
if (!request_region(nvt->cir_wake_addr,
|
||||||
|
CIR_IOREG_LENGTH, NVT_DRIVER_NAME))
|
||||||
|
goto failure;
|
||||||
|
|
||||||
|
if (request_irq(nvt->cir_wake_irq, nvt_cir_wake_isr, IRQF_SHARED,
|
||||||
|
NVT_DRIVER_NAME, (void *)nvt))
|
||||||
|
goto failure;
|
||||||
|
|
||||||
ret = rc_register_device(rdev);
|
ret = rc_register_device(rdev);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto failure;
|
goto failure;
|
||||||
|
|
|
@ -991,39 +991,10 @@ wbcir_probe(struct pnp_dev *device, const struct pnp_device_id *dev_id)
|
||||||
"(w: 0x%lX, e: 0x%lX, s: 0x%lX, i: %u)\n",
|
"(w: 0x%lX, e: 0x%lX, s: 0x%lX, i: %u)\n",
|
||||||
data->wbase, data->ebase, data->sbase, data->irq);
|
data->wbase, data->ebase, data->sbase, data->irq);
|
||||||
|
|
||||||
if (!request_region(data->wbase, WAKEUP_IOMEM_LEN, DRVNAME)) {
|
|
||||||
dev_err(dev, "Region 0x%lx-0x%lx already in use!\n",
|
|
||||||
data->wbase, data->wbase + WAKEUP_IOMEM_LEN - 1);
|
|
||||||
err = -EBUSY;
|
|
||||||
goto exit_free_data;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!request_region(data->ebase, EHFUNC_IOMEM_LEN, DRVNAME)) {
|
|
||||||
dev_err(dev, "Region 0x%lx-0x%lx already in use!\n",
|
|
||||||
data->ebase, data->ebase + EHFUNC_IOMEM_LEN - 1);
|
|
||||||
err = -EBUSY;
|
|
||||||
goto exit_release_wbase;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!request_region(data->sbase, SP_IOMEM_LEN, DRVNAME)) {
|
|
||||||
dev_err(dev, "Region 0x%lx-0x%lx already in use!\n",
|
|
||||||
data->sbase, data->sbase + SP_IOMEM_LEN - 1);
|
|
||||||
err = -EBUSY;
|
|
||||||
goto exit_release_ebase;
|
|
||||||
}
|
|
||||||
|
|
||||||
err = request_irq(data->irq, wbcir_irq_handler,
|
|
||||||
IRQF_DISABLED, DRVNAME, device);
|
|
||||||
if (err) {
|
|
||||||
dev_err(dev, "Failed to claim IRQ %u\n", data->irq);
|
|
||||||
err = -EBUSY;
|
|
||||||
goto exit_release_sbase;
|
|
||||||
}
|
|
||||||
|
|
||||||
led_trigger_register_simple("cir-tx", &data->txtrigger);
|
led_trigger_register_simple("cir-tx", &data->txtrigger);
|
||||||
if (!data->txtrigger) {
|
if (!data->txtrigger) {
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
goto exit_free_irq;
|
goto exit_free_data;
|
||||||
}
|
}
|
||||||
|
|
||||||
led_trigger_register_simple("cir-rx", &data->rxtrigger);
|
led_trigger_register_simple("cir-rx", &data->rxtrigger);
|
||||||
|
@ -1062,9 +1033,38 @@ wbcir_probe(struct pnp_dev *device, const struct pnp_device_id *dev_id)
|
||||||
data->dev->priv = data;
|
data->dev->priv = data;
|
||||||
data->dev->dev.parent = &device->dev;
|
data->dev->dev.parent = &device->dev;
|
||||||
|
|
||||||
|
if (!request_region(data->wbase, WAKEUP_IOMEM_LEN, DRVNAME)) {
|
||||||
|
dev_err(dev, "Region 0x%lx-0x%lx already in use!\n",
|
||||||
|
data->wbase, data->wbase + WAKEUP_IOMEM_LEN - 1);
|
||||||
|
err = -EBUSY;
|
||||||
|
goto exit_free_rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!request_region(data->ebase, EHFUNC_IOMEM_LEN, DRVNAME)) {
|
||||||
|
dev_err(dev, "Region 0x%lx-0x%lx already in use!\n",
|
||||||
|
data->ebase, data->ebase + EHFUNC_IOMEM_LEN - 1);
|
||||||
|
err = -EBUSY;
|
||||||
|
goto exit_release_wbase;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!request_region(data->sbase, SP_IOMEM_LEN, DRVNAME)) {
|
||||||
|
dev_err(dev, "Region 0x%lx-0x%lx already in use!\n",
|
||||||
|
data->sbase, data->sbase + SP_IOMEM_LEN - 1);
|
||||||
|
err = -EBUSY;
|
||||||
|
goto exit_release_ebase;
|
||||||
|
}
|
||||||
|
|
||||||
|
err = request_irq(data->irq, wbcir_irq_handler,
|
||||||
|
IRQF_DISABLED, DRVNAME, device);
|
||||||
|
if (err) {
|
||||||
|
dev_err(dev, "Failed to claim IRQ %u\n", data->irq);
|
||||||
|
err = -EBUSY;
|
||||||
|
goto exit_release_sbase;
|
||||||
|
}
|
||||||
|
|
||||||
err = rc_register_device(data->dev);
|
err = rc_register_device(data->dev);
|
||||||
if (err)
|
if (err)
|
||||||
goto exit_free_rc;
|
goto exit_free_irq;
|
||||||
|
|
||||||
device_init_wakeup(&device->dev, 1);
|
device_init_wakeup(&device->dev, 1);
|
||||||
|
|
||||||
|
@ -1072,14 +1072,6 @@ wbcir_probe(struct pnp_dev *device, const struct pnp_device_id *dev_id)
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
exit_free_rc:
|
|
||||||
rc_free_device(data->dev);
|
|
||||||
exit_unregister_led:
|
|
||||||
led_classdev_unregister(&data->led);
|
|
||||||
exit_unregister_rxtrigger:
|
|
||||||
led_trigger_unregister_simple(data->rxtrigger);
|
|
||||||
exit_unregister_txtrigger:
|
|
||||||
led_trigger_unregister_simple(data->txtrigger);
|
|
||||||
exit_free_irq:
|
exit_free_irq:
|
||||||
free_irq(data->irq, device);
|
free_irq(data->irq, device);
|
||||||
exit_release_sbase:
|
exit_release_sbase:
|
||||||
|
@ -1088,6 +1080,14 @@ wbcir_probe(struct pnp_dev *device, const struct pnp_device_id *dev_id)
|
||||||
release_region(data->ebase, EHFUNC_IOMEM_LEN);
|
release_region(data->ebase, EHFUNC_IOMEM_LEN);
|
||||||
exit_release_wbase:
|
exit_release_wbase:
|
||||||
release_region(data->wbase, WAKEUP_IOMEM_LEN);
|
release_region(data->wbase, WAKEUP_IOMEM_LEN);
|
||||||
|
exit_free_rc:
|
||||||
|
rc_free_device(data->dev);
|
||||||
|
exit_unregister_led:
|
||||||
|
led_classdev_unregister(&data->led);
|
||||||
|
exit_unregister_rxtrigger:
|
||||||
|
led_trigger_unregister_simple(data->rxtrigger);
|
||||||
|
exit_unregister_txtrigger:
|
||||||
|
led_trigger_unregister_simple(data->txtrigger);
|
||||||
exit_free_data:
|
exit_free_data:
|
||||||
kfree(data);
|
kfree(data);
|
||||||
pnp_set_drvdata(device, NULL);
|
pnp_set_drvdata(device, NULL);
|
||||||
|
|
Loading…
Reference in a new issue