On Thu, Oct 13, 2016 at 05:50:55PM +0200, Benjamin Tissoires wrote:
> From: Bjorn Andersson <[email protected]>
> 
> 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 <[email protected]>
> Signed-off-by: Benjamin Tissoires <[email protected]>

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

Reply via email to