Re: [PATCH] Input: rmi4 - remove the need for artificial IRQ in case of HID

From: Lyude Paul
Date: Mon Jul 24 2017 - 18:19:06 EST


Hi, could we get this thread going again? I've been hit by this problem
on my new machine recently, Razer Blade Stealth, and it's been making
the touchpad rather difficult to use (jumping cursors being seen, just
like the ones mentioned in the thread).

Additionally, I've been running this patch on my own kernel for a while
now with no issues, it seems to fix the jumping cursor problem
completely. We also just included it downstream in fedora's kernel, so
getting an upstream solution would be nice.

On Tue, 2017-04-04 at 16:05 +0200, Benjamin Tissoires wrote:
> Hi Dmitry,
>
> On Apr 03 2017 or thereabouts, Dmitry Torokhov wrote:
> > Hi Benjamin,
> >
> > On Mon, Apr 03, 2017 at 06:18:21PM +0200, Benjamin Tissoires wrote:
> > > The IRQ from rmi4 may interfere with the one we currently use on
> > > i2c-hid.
> > > Given that there is already a need for an external API from rmi4
> > > to
> > > forward the attention data, we can, in this particular case rely
> > > on a
> > > separate workqueue to prevent cursor jumps.
> >
> > Can you please walk me through how IRQ handler implemented by RMI
> > driver
> > interferes with i2c-hid interrupt and why using a workqueue
> > produces
> > better results.
>
> Unfortunately, I can only guess what is happening as I do not have
> production hardware for these touchpads.
>
> I believe the issue come from the
> local_irq_save()/local_irq_restore()
> calls in rmi_input_event(). The data coming out of the device seems
> clean in terms of coordinates:
> https://bugs.freedesktop.org/attachment.cgi?id=130522
> But the timing must be wrong, which leads libinput into inserting the
> jumps.
>
> I can't find (nor did Andrew) an other explanation, but I'd gladly
> take
> one other if you can find one.
>
> As for walking you through the current process:
> - i2c-hid receives an interrupt telling that data is available
> - i2c-hid read the input register from the device, which contains
> both
> the rmi4 irq mask and the attn_data.
> - hid-core forwards the input event to hid-rmi, which then calls
> local_irq_save(), rmi_set_attn_data(), generic_handle_irq() and
> local_irq_restore().
> - rmi4-core stores the attn_data in a fifo and on an IRQ pull the
> incoming data from the fifo instead of asking the transport driver
> to
> read from the data register
>
> This patch removes the artificial irq handling and should theorically
> be
> equivalent. The results show that they are not :(
>
> >
> > Frankly, I do not like the whole attn_data business. OI wonder if
> > we
> > we could have hidden it in HID transport code.
>
> Yes, ideally, that would be nice, but the issue is that hid-rmi is
> just
> a transport driver and we don't know where the attn_data would be
> stored
> in the device. So we can't hide this in the rmi_read_block operation.
> And given the way you retrieve the data from the device over HID
> (write of an input event, read of one other), it's a long operation,
> which explains why the data is embedded in the interrupt data itself.
>
> Cheers,
> Benjamin
>
> >
> > Thanks.
> >
> > >
> > > Reported-by: Cameron Gutman <aicommander@xxxxxxxxx>
> > > Reported-by: Thorsten Leemhuis <linux@xxxxxxxxxxxxx>
> > > Reported-by: Jason Ekstrand <jason@xxxxxxxxxxxxxx>
> > > Tested-by: Andrew Duggan <aduggan@xxxxxxxxxxxxx>
> > > Signed-off-by: Benjamin Tissoires <benjamin.tissoires@xxxxxxxxxx>
> > > ---
> > >
> > > Hi Dmitry, Jiri,
> > >
> > > This is a regression introduced by my code in v4.11.
> > > I do not think it is worth splitting between HID and input, there
> > > hasn't
> > > been any development in hid-rmi since v4.11-rc1.
> > >
> > > Cheers,
> > > Benjamin
> > >
> > > drivers/hid/hid-rmi.c | 64 ---------------------
> > > drivers/input/rmi4/rmi_driver.c | 122 ++++++++++++++++++++++++
> > > ----------------
> > > include/linux/rmi.h | 1 +
> > > 3 files changed, 75 insertions(+), 112 deletions(-)
> > >
> > > diff --git a/drivers/hid/hid-rmi.c b/drivers/hid/hid-rmi.c
> > > index 5b40c26..4aa882c 100644
> > > --- a/drivers/hid/hid-rmi.c
> > > +++ b/drivers/hid/hid-rmi.c
> > > @@ -316,19 +316,12 @@ static int rmi_input_event(struct
> > > hid_device *hdev, u8 *data, int size)
> > > {
> > > struct rmi_data *hdata = hid_get_drvdata(hdev);
> > > struct rmi_device *rmi_dev = hdata->xport.rmi_dev;
> > > - unsigned long flags;
> > >
> > > if (!(test_bit(RMI_STARTED, &hdata->flags)))
> > > return 0;
> > >
> > > - local_irq_save(flags);
> > > -
> > > rmi_set_attn_data(rmi_dev, data[1], &data[2], size - 2);
> > >
> > > - generic_handle_irq(hdata->rmi_irq);
> > > -
> > > - local_irq_restore(flags);
> > > -
> > > return 1;
> > > }
> > >
> > > @@ -556,56 +549,6 @@ static const struct rmi_transport_ops
> > > hid_rmi_ops = {
> > > .reset = rmi_hid_reset,
> > > };
> > >
> > > -static void rmi_irq_teardown(void *data)
> > > -{
> > > - struct rmi_data *hdata = data;
> > > - struct irq_domain *domain = hdata->domain;
> > > -
> > > - if (!domain)
> > > - return;
> > > -
> > > - irq_dispose_mapping(irq_find_mapping(domain, 0));
> > > -
> > > - irq_domain_remove(domain);
> > > - hdata->domain = NULL;
> > > - hdata->rmi_irq = 0;
> > > -}
> > > -
> > > -static int rmi_irq_map(struct irq_domain *h, unsigned int virq,
> > > - irq_hw_number_t hw_irq_num)
> > > -{
> > > - irq_set_chip_and_handler(virq, &dummy_irq_chip,
> > > handle_simple_irq);
> > > -
> > > - return 0;
> > > -}
> > > -
> > > -static const struct irq_domain_ops rmi_irq_ops = {
> > > - .map = rmi_irq_map,
> > > -};
> > > -
> > > -static int rmi_setup_irq_domain(struct hid_device *hdev)
> > > -{
> > > - struct rmi_data *hdata = hid_get_drvdata(hdev);
> > > - int ret;
> > > -
> > > - hdata->domain = irq_domain_create_linear(hdev-
> > > >dev.fwnode, 1,
> > > - &rmi_irq_ops,
> > > hdata);
> > > - if (!hdata->domain)
> > > - return -ENOMEM;
> > > -
> > > - ret = devm_add_action_or_reset(&hdev->dev,
> > > &rmi_irq_teardown, hdata);
> > > - if (ret)
> > > - return ret;
> > > -
> > > - hdata->rmi_irq = irq_create_mapping(hdata->domain, 0);
> > > - if (hdata->rmi_irq <= 0) {
> > > - hid_err(hdev, "Can't allocate an IRQ\n");
> > > - return hdata->rmi_irq < 0 ? hdata->rmi_irq :
> > > -ENXIO;
> > > - }
> > > -
> > > - return 0;
> > > -}
> > > -
> > > static int rmi_probe(struct hid_device *hdev, const struct
> > > hid_device_id *id)
> > > {
> > > struct rmi_data *data = NULL;
> > > @@ -677,18 +620,11 @@ static int rmi_probe(struct hid_device
> > > *hdev, const struct hid_device_id *id)
> > >
> > > mutex_init(&data->page_mutex);
> > >
> > > - ret = rmi_setup_irq_domain(hdev);
> > > - if (ret) {
> > > - hid_err(hdev, "failed to allocate IRQ
> > > domain\n");
> > > - return ret;
> > > - }
> > > -
> > > if (data->device_flags & RMI_DEVICE_HAS_PHYS_BUTTONS)
> > > rmi_hid_pdata.f30_data.disable = true;
> > >
> > > data->xport.dev = hdev->dev.parent;
> > > data->xport.pdata = rmi_hid_pdata;
> > > - data->xport.pdata.irq = data->rmi_irq;
> > > data->xport.proto_name = "hid";
> > > data->xport.ops = &hid_rmi_ops;
> > >
> > > diff --git a/drivers/input/rmi4/rmi_driver.c
> > > b/drivers/input/rmi4/rmi_driver.c
> > > index d64fc92..5e65cba 100644
> > > --- a/drivers/input/rmi4/rmi_driver.c
> > > +++ b/drivers/input/rmi4/rmi_driver.c
> > > @@ -209,32 +209,46 @@ void rmi_set_attn_data(struct rmi_device
> > > *rmi_dev, unsigned long irq_status,
> > > attn_data.data = fifo_data;
> > >
> > > kfifo_put(&drvdata->attn_fifo, attn_data);
> > > +
> > > + schedule_work(&drvdata->attn_work);
> > > }
> > > EXPORT_SYMBOL_GPL(rmi_set_attn_data);
> > >
> > > -static irqreturn_t rmi_irq_fn(int irq, void *dev_id)
> > > +static void attn_callback(struct work_struct *work)
> > > {
> > > - struct rmi_device *rmi_dev = dev_id;
> > > - struct rmi_driver_data *drvdata =
> > > dev_get_drvdata(&rmi_dev->dev);
> > > + struct rmi_driver_data *drvdata = container_of(work,
> > > + struct
> > > rmi_driver_data,
> > > + attn_wor
> > > k);
> > > struct rmi4_attn_data attn_data = {0};
> > > int ret, count;
> > >
> > > count = kfifo_get(&drvdata->attn_fifo, &attn_data);
> > > - if (count) {
> > > - *(drvdata->irq_status) = attn_data.irq_status;
> > > - drvdata->attn_data = attn_data;
> > > - }
> > > + if (!count)
> > > + return;
> > >
> > > - ret = rmi_process_interrupt_requests(rmi_dev);
> > > + *(drvdata->irq_status) = attn_data.irq_status;
> > > + drvdata->attn_data = attn_data;
> > > +
> > > + ret = rmi_process_interrupt_requests(drvdata->rmi_dev);
> > > if (ret)
> > > - rmi_dbg(RMI_DEBUG_CORE, &rmi_dev->dev,
> > > + rmi_dbg(RMI_DEBUG_CORE, &drvdata->rmi_dev->dev,
> > > "Failed to process interrupt request:
> > > %d\n", ret);
> > >
> > > - if (count)
> > > - kfree(attn_data.data);
> > > + kfree(attn_data.data);
> > >
> > > if (!kfifo_is_empty(&drvdata->attn_fifo))
> > > - return rmi_irq_fn(irq, dev_id);
> > > + schedule_work(&drvdata->attn_work);
> > > +}
> > > +
> > > +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;
> > > }
> > > @@ -242,7 +256,6 @@ static irqreturn_t rmi_irq_fn(int irq, void
> > > *dev_id)
> > > static int rmi_irq_init(struct rmi_device *rmi_dev)
> > > {
> > > struct rmi_device_platform_data *pdata =
> > > rmi_get_platform_data(rmi_dev);
> > > - struct rmi_driver_data *data = dev_get_drvdata(&rmi_dev-
> > > >dev);
> > > int irq_flags = irq_get_trigger_type(pdata->irq);
> > > int ret;
> > >
> > > @@ -260,8 +273,6 @@ static int rmi_irq_init(struct rmi_device
> > > *rmi_dev)
> > > return ret;
> > > }
> > >
> > > - data->enabled = true;
> > > -
> > > return 0;
> > > }
> > >
> > > @@ -910,23 +921,27 @@ void rmi_enable_irq(struct rmi_device
> > > *rmi_dev, bool clear_wake)
> > > if (data->enabled)
> > > goto out;
> > >
> > > - enable_irq(irq);
> > > - data->enabled = true;
> > > - 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);
> > > - }
> > > + if (irq) {
> > > + enable_irq(irq);
> > > + data->enabled = true;
> > > + 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);
> > > + }
> > >
> > > - /*
> > > - * Call rmi_process_interrupt_requests() after enabling
> > > irq,
> > > - * otherwise we may lose interrupt on edge-triggered
> > > systems.
> > > - */
> > > - irq_flags = irq_get_trigger_type(pdata->irq);
> > > - if (irq_flags & IRQ_TYPE_EDGE_BOTH)
> > > - rmi_process_interrupt_requests(rmi_dev);
> > > + /*
> > > + * Call rmi_process_interrupt_requests() after
> > > enabling irq,
> > > + * otherwise we may lose interrupt on edge-
> > > triggered systems.
> > > + */
> > > + irq_flags = irq_get_trigger_type(pdata->irq);
> > > + if (irq_flags & IRQ_TYPE_EDGE_BOTH)
> > > + rmi_process_interrupt_requests(rmi_dev);
> > > + } else {
> > > + data->enabled = true;
> > > + }
> > >
> > > out:
> > > mutex_unlock(&data->enabled_mutex);
> > > @@ -946,20 +961,22 @@ void rmi_disable_irq(struct rmi_device
> > > *rmi_dev, bool enable_wake)
> > > goto out;
> > >
> > > data->enabled = false;
> > > - 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);
> > > - }
> > > -
> > > - /* make sure the fifo is clean */
> > > - while (!kfifo_is_empty(&data->attn_fifo)) {
> > > - count = kfifo_get(&data->attn_fifo, &attn_data);
> > > - if (count)
> > > - kfree(attn_data.data);
> > > + if (irq) {
> > > + 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);
> > > + }
> > > + } else {
> > > + /* make sure the fifo is clean */
> > > + while (!kfifo_is_empty(&data->attn_fifo)) {
> > > + count = kfifo_get(&data->attn_fifo,
> > > &attn_data);
> > > + if (count)
> > > + kfree(attn_data.data);
> > > + }
> > > }
> > >
> > > out:
> > > @@ -998,9 +1015,12 @@ 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_driver_data *data = dev_get_drvdata(&rmi_dev-
> > > >dev);
> > >
> > > rmi_disable_irq(rmi_dev, false);
> > >
> > > + cancel_work_sync(&data->attn_work);
> > > +
> > > rmi_f34_remove_sysfs(rmi_dev);
> > > rmi_free_function_list(rmi_dev);
> > >
> > > @@ -1230,9 +1250,15 @@ static int rmi_driver_probe(struct device
> > > *dev)
> > > }
> > > }
> > >
> > > - retval = rmi_irq_init(rmi_dev);
> > > - if (retval < 0)
> > > - goto err_destroy_functions;
> > > + if (pdata->irq) {
> > > + retval = rmi_irq_init(rmi_dev);
> > > + if (retval < 0)
> > > + goto err_destroy_functions;
> > > + }
> > > +
> > > + data->enabled = true;
> > > +
> > > + INIT_WORK(&data->attn_work, attn_callback);
> > >
> > > if (data->f01_container->dev.driver)
> > > /* Driver already bound, so enable ATTN now. */
> > > diff --git a/include/linux/rmi.h b/include/linux/rmi.h
> > > index 64125443..dc90178 100644
> > > --- a/include/linux/rmi.h
> > > +++ b/include/linux/rmi.h
> > > @@ -364,6 +364,7 @@ struct rmi_driver_data {
> > >
> > > struct rmi4_attn_data attn_data;
> > > DECLARE_KFIFO(attn_fifo, struct rmi4_attn_data, 16);
> > > + struct work_struct attn_work;
> > > };
> > >
> > > int rmi_register_transport_device(struct rmi_transport_dev
> > > *xport);
> > > --
> > > 2.9.3
> > >
> >
> > --
> > 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
--
Cheers,
Lyude