On Thu, Oct 13, 2016 at 05:50:55PM +0200, Benjamin Tissoires wrote: > From: Bjorn Andersson <bjorn.andersson@xxxxxxxxxx> > > The attn IRQ is related to the chip, rather than the transport, so move > all handling of interrupts to the core driver. This also makes sure that > there are no races between interrupts and availability of the resources > used by the core driver. > > Signed-off-by: Bjorn Andersson <bjorn.andersson@xxxxxxxxxx> > Signed-off-by: Benjamin Tissoires <benjamin.tissoires@xxxxxxxxxx> Applied, thank you. > > --- > > new in v3 > > changes since Bjorn's submission: > - call disable_irq on remove() > --- > drivers/input/rmi4/rmi_driver.c | 73 +++++++++++++++++++++++++++++++++++++--- > drivers/input/rmi4/rmi_i2c.c | 74 +++-------------------------------------- > drivers/input/rmi4/rmi_spi.c | 72 +++------------------------------------ > include/linux/rmi.h | 7 ++-- > 4 files changed, 83 insertions(+), 143 deletions(-) > > diff --git a/drivers/input/rmi4/rmi_driver.c b/drivers/input/rmi4/rmi_driver.c > index c83bce8..cf780ef 100644 > --- a/drivers/input/rmi4/rmi_driver.c > +++ b/drivers/input/rmi4/rmi_driver.c > @@ -17,6 +17,7 @@ > #include <linux/bitmap.h> > #include <linux/delay.h> > #include <linux/fs.h> > +#include <linux/irq.h> > #include <linux/kconfig.h> > #include <linux/pm.h> > #include <linux/slab.h> > @@ -134,7 +135,7 @@ static void process_one_interrupt(struct rmi_driver_data *data, > } > } > > -int rmi_process_interrupt_requests(struct rmi_device *rmi_dev) > +static int rmi_process_interrupt_requests(struct rmi_device *rmi_dev) > { > struct rmi_driver_data *data = dev_get_drvdata(&rmi_dev->dev); > struct device *dev = &rmi_dev->dev; > @@ -179,7 +180,42 @@ int rmi_process_interrupt_requests(struct rmi_device *rmi_dev) > > return 0; > } > -EXPORT_SYMBOL_GPL(rmi_process_interrupt_requests); > + > +static irqreturn_t rmi_irq_fn(int irq, void *dev_id) > +{ > + struct rmi_device *rmi_dev = dev_id; > + int ret; > + > + ret = rmi_process_interrupt_requests(rmi_dev); > + if (ret) > + rmi_dbg(RMI_DEBUG_CORE, &rmi_dev->dev, > + "Failed to process interrupt request: %d\n", ret); > + > + return IRQ_HANDLED; > +} > + > +static int rmi_irq_init(struct rmi_device *rmi_dev) > +{ > + struct rmi_device_platform_data *pdata = rmi_get_platform_data(rmi_dev); > + int irq_flags = irq_get_trigger_type(pdata->irq); > + int ret; > + > + if (!irq_flags) > + irq_flags = IRQF_TRIGGER_LOW; > + > + ret = devm_request_threaded_irq(&rmi_dev->dev, pdata->irq, NULL, > + rmi_irq_fn, irq_flags | IRQF_ONESHOT, > + dev_name(rmi_dev->xport->dev), > + rmi_dev); > + if (ret < 0) { > + dev_warn(&rmi_dev->dev, "Failed to register interrupt %d\n", > + pdata->irq); > + > + return ret; > + } > + > + return 0; > +} > > static int suspend_one_function(struct rmi_function *fn) > { > @@ -787,8 +823,10 @@ err_put_fn: > return error; > } > > -int rmi_driver_suspend(struct rmi_device *rmi_dev) > +int rmi_driver_suspend(struct rmi_device *rmi_dev, bool enable_wake) > { > + struct rmi_device_platform_data *pdata = rmi_get_platform_data(rmi_dev); > + int irq = pdata->irq; > int retval = 0; > > retval = rmi_suspend_functions(rmi_dev); > @@ -796,14 +834,33 @@ int rmi_driver_suspend(struct rmi_device *rmi_dev) > dev_warn(&rmi_dev->dev, "Failed to suspend functions: %d\n", > retval); > > + disable_irq(irq); > + if (enable_wake && device_may_wakeup(rmi_dev->xport->dev)) { > + retval = enable_irq_wake(irq); > + if (!retval) > + dev_warn(&rmi_dev->dev, > + "Failed to enable irq for wake: %d\n", > + retval); > + } > return retval; > } > EXPORT_SYMBOL_GPL(rmi_driver_suspend); > > -int rmi_driver_resume(struct rmi_device *rmi_dev) > +int rmi_driver_resume(struct rmi_device *rmi_dev, bool clear_wake) > { > + struct rmi_device_platform_data *pdata = rmi_get_platform_data(rmi_dev); > + int irq = pdata->irq; > int retval; > > + enable_irq(irq); > + if (clear_wake && device_may_wakeup(rmi_dev->xport->dev)) { > + retval = disable_irq_wake(irq); > + if (!retval) > + dev_warn(&rmi_dev->dev, > + "Failed to disable irq for wake: %d\n", > + retval); > + } > + > retval = rmi_resume_functions(rmi_dev); > if (retval) > dev_warn(&rmi_dev->dev, "Failed to suspend functions: %d\n", > @@ -816,6 +873,10 @@ EXPORT_SYMBOL_GPL(rmi_driver_resume); > static int rmi_driver_remove(struct device *dev) > { > struct rmi_device *rmi_dev = to_rmi_device(dev); > + struct rmi_device_platform_data *pdata = rmi_get_platform_data(rmi_dev); > + int irq = pdata->irq; > + > + disable_irq(irq); > > rmi_free_function_list(rmi_dev); > > @@ -1004,6 +1065,10 @@ static int rmi_driver_probe(struct device *dev) > } > } > > + retval = rmi_irq_init(rmi_dev); > + if (retval < 0) > + goto err_destroy_functions; > + > if (data->f01_container->dev.driver) > /* Driver already bound, so enable ATTN now. */ > return enable_sensor(rmi_dev); > diff --git a/drivers/input/rmi4/rmi_i2c.c b/drivers/input/rmi4/rmi_i2c.c > index 6f2e0e4..64a5488 100644 > --- a/drivers/input/rmi4/rmi_i2c.c > +++ b/drivers/input/rmi4/rmi_i2c.c > @@ -9,7 +9,6 @@ > > #include <linux/i2c.h> > #include <linux/rmi.h> > -#include <linux/irq.h> > #include <linux/of.h> > #include <linux/delay.h> > #include <linux/regulator/consumer.h> > @@ -35,8 +34,6 @@ struct rmi_i2c_xport { > struct mutex page_mutex; > int page; > > - int irq; > - > u8 *tx_buf; > size_t tx_buf_size; > > @@ -177,42 +174,6 @@ static const struct rmi_transport_ops rmi_i2c_ops = { > .read_block = rmi_i2c_read_block, > }; > > -static irqreturn_t rmi_i2c_irq(int irq, void *dev_id) > -{ > - struct rmi_i2c_xport *rmi_i2c = dev_id; > - struct rmi_device *rmi_dev = rmi_i2c->xport.rmi_dev; > - int ret; > - > - ret = rmi_process_interrupt_requests(rmi_dev); > - if (ret) > - rmi_dbg(RMI_DEBUG_XPORT, &rmi_dev->dev, > - "Failed to process interrupt request: %d\n", ret); > - > - return IRQ_HANDLED; > -} > - > -static int rmi_i2c_init_irq(struct i2c_client *client) > -{ > - struct rmi_i2c_xport *rmi_i2c = i2c_get_clientdata(client); > - int irq_flags = irqd_get_trigger_type(irq_get_irq_data(rmi_i2c->irq)); > - int ret; > - > - if (!irq_flags) > - irq_flags = IRQF_TRIGGER_LOW; > - > - ret = devm_request_threaded_irq(&client->dev, rmi_i2c->irq, NULL, > - rmi_i2c_irq, irq_flags | IRQF_ONESHOT, client->name, > - rmi_i2c); > - if (ret < 0) { > - dev_warn(&client->dev, "Failed to register interrupt %d\n", > - rmi_i2c->irq); > - > - return ret; > - } > - > - return 0; > -} > - > #ifdef CONFIG_OF > static const struct of_device_id rmi_i2c_of_match[] = { > { .compatible = "syna,rmi4-i2c" }, > @@ -240,8 +201,7 @@ static int rmi_i2c_probe(struct i2c_client *client, > if (!client->dev.of_node && client_pdata) > *pdata = *client_pdata; > > - if (client->irq > 0) > - rmi_i2c->irq = client->irq; > + pdata->irq = client->irq; > > rmi_dbg(RMI_DEBUG_XPORT, &client->dev, "Probing %s.\n", > dev_name(&client->dev)); > @@ -295,10 +255,6 @@ static int rmi_i2c_probe(struct i2c_client *client, > return retval; > } > > - retval = rmi_i2c_init_irq(client); > - if (retval < 0) > - return retval; > - > dev_info(&client->dev, "registered rmi i2c driver at %#04x.\n", > client->addr); > return 0; > @@ -322,18 +278,10 @@ static int rmi_i2c_suspend(struct device *dev) > struct rmi_i2c_xport *rmi_i2c = i2c_get_clientdata(client); > int ret; > > - ret = rmi_driver_suspend(rmi_i2c->xport.rmi_dev); > + ret = rmi_driver_suspend(rmi_i2c->xport.rmi_dev, true); > if (ret) > dev_warn(dev, "Failed to resume device: %d\n", ret); > > - disable_irq(rmi_i2c->irq); > - if (device_may_wakeup(&client->dev)) { > - ret = enable_irq_wake(rmi_i2c->irq); > - if (!ret) > - dev_warn(dev, "Failed to enable irq for wake: %d\n", > - ret); > - } > - > regulator_bulk_disable(ARRAY_SIZE(rmi_i2c->supplies), > rmi_i2c->supplies); > > @@ -353,15 +301,7 @@ static int rmi_i2c_resume(struct device *dev) > > msleep(rmi_i2c->startup_delay); > > - enable_irq(rmi_i2c->irq); > - if (device_may_wakeup(&client->dev)) { > - ret = disable_irq_wake(rmi_i2c->irq); > - if (!ret) > - dev_warn(dev, "Failed to disable irq for wake: %d\n", > - ret); > - } > - > - ret = rmi_driver_resume(rmi_i2c->xport.rmi_dev); > + ret = rmi_driver_resume(rmi_i2c->xport.rmi_dev, true); > if (ret) > dev_warn(dev, "Failed to resume device: %d\n", ret); > > @@ -376,12 +316,10 @@ static int rmi_i2c_runtime_suspend(struct device *dev) > struct rmi_i2c_xport *rmi_i2c = i2c_get_clientdata(client); > int ret; > > - ret = rmi_driver_suspend(rmi_i2c->xport.rmi_dev); > + ret = rmi_driver_suspend(rmi_i2c->xport.rmi_dev, false); > if (ret) > dev_warn(dev, "Failed to resume device: %d\n", ret); > > - disable_irq(rmi_i2c->irq); > - > regulator_bulk_disable(ARRAY_SIZE(rmi_i2c->supplies), > rmi_i2c->supplies); > > @@ -401,9 +339,7 @@ static int rmi_i2c_runtime_resume(struct device *dev) > > msleep(rmi_i2c->startup_delay); > > - enable_irq(rmi_i2c->irq); > - > - ret = rmi_driver_resume(rmi_i2c->xport.rmi_dev); > + ret = rmi_driver_resume(rmi_i2c->xport.rmi_dev, false); > if (ret) > dev_warn(dev, "Failed to resume device: %d\n", ret); > > diff --git a/drivers/input/rmi4/rmi_spi.c b/drivers/input/rmi4/rmi_spi.c > index 55bd1b3..f3e9e48 100644 > --- a/drivers/input/rmi4/rmi_spi.c > +++ b/drivers/input/rmi4/rmi_spi.c > @@ -12,7 +12,6 @@ > #include <linux/rmi.h> > #include <linux/slab.h> > #include <linux/spi/spi.h> > -#include <linux/irq.h> > #include <linux/of.h> > #include "rmi_driver.h" > > @@ -44,8 +43,6 @@ struct rmi_spi_xport { > struct mutex page_mutex; > int page; > > - int irq; > - > u8 *rx_buf; > u8 *tx_buf; > int xfer_buf_size; > @@ -326,41 +323,6 @@ static const struct rmi_transport_ops rmi_spi_ops = { > .read_block = rmi_spi_read_block, > }; > > -static irqreturn_t rmi_spi_irq(int irq, void *dev_id) > -{ > - struct rmi_spi_xport *rmi_spi = dev_id; > - struct rmi_device *rmi_dev = rmi_spi->xport.rmi_dev; > - int ret; > - > - ret = rmi_process_interrupt_requests(rmi_dev); > - if (ret) > - rmi_dbg(RMI_DEBUG_XPORT, &rmi_dev->dev, > - "Failed to process interrupt request: %d\n", ret); > - > - return IRQ_HANDLED; > -} > - > -static int rmi_spi_init_irq(struct spi_device *spi) > -{ > - struct rmi_spi_xport *rmi_spi = spi_get_drvdata(spi); > - int irq_flags = irqd_get_trigger_type(irq_get_irq_data(rmi_spi->irq)); > - int ret; > - > - if (!irq_flags) > - irq_flags = IRQF_TRIGGER_LOW; > - > - ret = devm_request_threaded_irq(&spi->dev, rmi_spi->irq, NULL, > - rmi_spi_irq, irq_flags | IRQF_ONESHOT, > - dev_name(&spi->dev), rmi_spi); > - if (ret < 0) { > - dev_warn(&spi->dev, "Failed to register interrupt %d\n", > - rmi_spi->irq); > - return ret; > - } > - > - return 0; > -} > - > #ifdef CONFIG_OF > static int rmi_spi_of_probe(struct spi_device *spi, > struct rmi_device_platform_data *pdata) > @@ -433,8 +395,7 @@ static int rmi_spi_probe(struct spi_device *spi) > return retval; > } > > - if (spi->irq > 0) > - rmi_spi->irq = spi->irq; > + pdata->irq = spi->irq; > > rmi_spi->spi = spi; > mutex_init(&rmi_spi->page_mutex); > @@ -465,10 +426,6 @@ static int rmi_spi_probe(struct spi_device *spi) > return retval; > } > > - retval = rmi_spi_init_irq(spi); > - if (retval < 0) > - return retval; > - > dev_info(&spi->dev, "registered RMI SPI driver\n"); > return 0; > } > @@ -489,17 +446,10 @@ static int rmi_spi_suspend(struct device *dev) > struct rmi_spi_xport *rmi_spi = spi_get_drvdata(spi); > int ret; > > - ret = rmi_driver_suspend(rmi_spi->xport.rmi_dev); > + ret = rmi_driver_suspend(rmi_spi->xport.rmi_dev, true); > if (ret) > dev_warn(dev, "Failed to resume device: %d\n", ret); > > - disable_irq(rmi_spi->irq); > - if (device_may_wakeup(&spi->dev)) { > - ret = enable_irq_wake(rmi_spi->irq); > - if (!ret) > - dev_warn(dev, "Failed to enable irq for wake: %d\n", > - ret); > - } > return ret; > } > > @@ -509,15 +459,7 @@ static int rmi_spi_resume(struct device *dev) > struct rmi_spi_xport *rmi_spi = spi_get_drvdata(spi); > int ret; > > - enable_irq(rmi_spi->irq); > - if (device_may_wakeup(&spi->dev)) { > - ret = disable_irq_wake(rmi_spi->irq); > - if (!ret) > - dev_warn(dev, "Failed to disable irq for wake: %d\n", > - ret); > - } > - > - ret = rmi_driver_resume(rmi_spi->xport.rmi_dev); > + ret = rmi_driver_resume(rmi_spi->xport.rmi_dev, true); > if (ret) > dev_warn(dev, "Failed to resume device: %d\n", ret); > > @@ -532,12 +474,10 @@ static int rmi_spi_runtime_suspend(struct device *dev) > struct rmi_spi_xport *rmi_spi = spi_get_drvdata(spi); > int ret; > > - ret = rmi_driver_suspend(rmi_spi->xport.rmi_dev); > + ret = rmi_driver_suspend(rmi_spi->xport.rmi_dev, false); > if (ret) > dev_warn(dev, "Failed to resume device: %d\n", ret); > > - disable_irq(rmi_spi->irq); > - > return 0; > } > > @@ -547,9 +487,7 @@ static int rmi_spi_runtime_resume(struct device *dev) > struct rmi_spi_xport *rmi_spi = spi_get_drvdata(spi); > int ret; > > - enable_irq(rmi_spi->irq); > - > - ret = rmi_driver_resume(rmi_spi->xport.rmi_dev); > + ret = rmi_driver_resume(rmi_spi->xport.rmi_dev, false); > if (ret) > dev_warn(dev, "Failed to resume device: %d\n", ret); > > diff --git a/include/linux/rmi.h b/include/linux/rmi.h > index e0aca14..5944e6c 100644 > --- a/include/linux/rmi.h > +++ b/include/linux/rmi.h > @@ -204,9 +204,11 @@ struct rmi_device_platform_data_spi { > * @reset_delay_ms - after issuing a reset command to the touch sensor, the > * driver waits a few milliseconds to give the firmware a chance to > * to re-initialize. You can override the default wait period here. > + * @irq: irq associated with the attn gpio line, or negative > */ > struct rmi_device_platform_data { > int reset_delay_ms; > + int irq; > > struct rmi_device_platform_data_spi spi_data; > > @@ -352,8 +354,7 @@ struct rmi_driver_data { > > int rmi_register_transport_device(struct rmi_transport_dev *xport); > void rmi_unregister_transport_device(struct rmi_transport_dev *xport); > -int rmi_process_interrupt_requests(struct rmi_device *rmi_dev); > > -int rmi_driver_suspend(struct rmi_device *rmi_dev); > -int rmi_driver_resume(struct rmi_device *rmi_dev); > +int rmi_driver_suspend(struct rmi_device *rmi_dev, bool enable_wake); > +int rmi_driver_resume(struct rmi_device *rmi_dev, bool clear_wake); > #endif > -- > 2.7.4 > -- Dmitry -- To unsubscribe from this list: send the line "unsubscribe linux-input" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html