linux-input.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
From: "Richard Röjfors" <richard.rojfors.ext@mocean-labs.com>
To: Dmitry Torokhov <dmitry.torokhov@gmail.com>
Cc: linux-input@vger.kernel.org,
	Linux Kernel Mailing List <linux-kernel@vger.kernel.org>,
	kwangwoo.lee@gmail.com,
	Thierry Reding <thierry.reding@avionic-design.de>,
	Trilok Soni <soni.trilok@gmail.com>,
	Andrew Morton <akpm@linux-foundation.org>
Subject: Re: [PATCH 1/2] tsc2007: remove HR timer
Date: Tue, 14 Jul 2009 10:47:53 +0200	[thread overview]
Message-ID: <4A5C4639.1070003@mocean-labs.com> (raw)
In-Reply-To: <20090714044957.GD2822@dtor-d630.eng.vmware.com>

On 7/14/09 6:59 AM, Dmitry Torokhov wrote:
> Hi Richard,
>
> On Tue, Jun 23, 2009 at 01:54:48PM +0200, Richard Röjfors wrote:
>> This patch removes the HR timer, since it's bad to do synchronous I2C
>> in the HR timer callback context. The new implementation makes use
>> of the global workqueue. The work is scheduled every 5ms when polling
>> rather than 5 us.
>>
>> Signed-off-by: Richard Röjfors<richard.rojfors.ext@mocean-labs.com>
>> ---
>> Index: linux-2.6.30/drivers/input/touchscreen/tsc2007.c
>> ===================================================================
>> --- linux-2.6.30/drivers/input/touchscreen/tsc2007.c	(revision 932)
>> +++ linux-2.6.30/drivers/input/touchscreen/tsc2007.c	(revision 943)
>> @@ -21,15 +21,13 @@
>>    */
>>
>>   #include<linux/module.h>
>> -#include<linux/hrtimer.h>
>>   #include<linux/slab.h>
>>   #include<linux/input.h>
>>   #include<linux/interrupt.h>
>>   #include<linux/i2c.h>
>>   #include<linux/i2c/tsc2007.h>
>>
>> -#define TS_POLL_DELAY	(10 * 1000)	/* ns delay before the first sample */
>> -#define TS_POLL_PERIOD	(5 * 1000)	/* ns delay between samples */
>> +#define TS_POLL_PERIOD	msecs_to_jiffies(5) /* ms delay between samples */
>>
>>   #define TSC2007_MEASURE_TEMP0		(0x0<<  4)
>>   #define TSC2007_MEASURE_AUX		(0x2<<  4)
>> @@ -70,13 +68,11 @@
>>   struct tsc2007 {
>>   	struct input_dev	*input;
>>   	char			phys[32];
>> -	struct hrtimer		timer;
>> +	struct delayed_work	work;
>>   	struct ts_event		tc;
>>
>>   	struct i2c_client	*client;
>>
>> -	spinlock_t		lock;
>> -
>>   	u16			model;
>>   	u16			x_plate_ohms;
>>
>> @@ -142,8 +138,7 @@
>>   	if (rt>  MAX_12BIT) {
>>   		dev_dbg(&ts->client->dev, "ignored pressure %d\n", rt);
>>
>> -		hrtimer_start(&ts->timer, ktime_set(0, TS_POLL_PERIOD),
>> -			      HRTIMER_MODE_REL);
>> +		schedule_delayed_work(&ts->work, TS_POLL_PERIOD);
>>   		return;
>>   	}
>>
>> @@ -153,7 +148,7 @@
>>   	 * in some cases may not even settle at the expected value.
>>   	 *
>>   	 * The only safe way to check for the pen up condition is in the
>> -	 * timer by reading the pen signal state (it's a GPIO _and_ IRQ).
>> +	 * work function by reading the pen signal state (it's a GPIO and IRQ).
>>   	 */
>>   	if (rt) {
>>   		struct input_dev *input = ts->input;
>> @@ -175,8 +170,7 @@
>>   			x, y, rt);
>>   	}
>>
>> -	hrtimer_start(&ts->timer, ktime_set(0, TS_POLL_PERIOD),
>> -			HRTIMER_MODE_REL);
>> +	schedule_delayed_work(&ts->work, TS_POLL_PERIOD);
>>   }
>>
>>   static int tsc2007_read_values(struct tsc2007 *tsc)
>> @@ -197,13 +191,10 @@
>>   	return 0;
>>   }
>>
>> -static enum hrtimer_restart tsc2007_timer(struct hrtimer *handle)
>> +static void tsc2007_work(struct work_struct *work)
>>   {
>> -	struct tsc2007 *ts = container_of(handle, struct tsc2007, timer);
>> -	unsigned long flags;
>> -
>> -	spin_lock_irqsave(&ts->lock, flags);
>> -
>> +	struct tsc2007 *ts =
>> +		container_of(to_delayed_work(work), struct tsc2007, work);
>>   	if (unlikely(!ts->get_pendown_state()&&  ts->pendown)) {
>>   		struct input_dev *input = ts->input;
>>
>> @@ -222,30 +213,20 @@
>>   		tsc2007_read_values(ts);
>>   		tsc2007_send_event(ts);
>>   	}
>> -
>> -	spin_unlock_irqrestore(&ts->lock, flags);
>> -
>> -	return HRTIMER_NORESTART;
>>   }
>>
>>   static irqreturn_t tsc2007_irq(int irq, void *handle)
>>   {
>>   	struct tsc2007 *ts = handle;
>> -	unsigned long flags;
>>
>> -	spin_lock_irqsave(&ts->lock, flags);
>> -
>>   	if (likely(ts->get_pendown_state())) {
>>   		disable_irq_nosync(ts->irq);
>> -		hrtimer_start(&ts->timer, ktime_set(0, TS_POLL_DELAY),
>> -					HRTIMER_MODE_REL);
>> +		schedule_delayed_work(&ts->work, 0);
>
> Originally we scheduled reading with timy delay to let the device
> settle, why don't we schedule with 1ms delay?

Good point, scheduling of the work queue takes some time, but of course 
that varies.

>
>>   	}
>>
>>   	if (ts->clear_penirq)
>>   		ts->clear_penirq();
>>
>> -	spin_unlock_irqrestore(&ts->lock, flags);
>> -
>>   	return IRQ_HANDLED;
>>   }
>>
>> @@ -278,11 +259,6 @@
>>
>>   	ts->input = input_dev;
>>
>> -	hrtimer_init(&ts->timer, CLOCK_MONOTONIC, HRTIMER_MODE_REL);
>> -	ts->timer.function = tsc2007_timer;
>> -
>> -	spin_lock_init(&ts->lock);
>> -
>>   	ts->model             = pdata->model;
>>   	ts->x_plate_ohms      = pdata->x_plate_ohms;
>>   	ts->get_pendown_state = pdata->get_pendown_state;
>> @@ -308,6 +284,8 @@
>>
>>   	ts->irq = client->irq;
>>
>> +	INIT_DELAYED_WORK(&ts->work, tsc2007_work);
>> +
>>   	err = request_irq(ts->irq, tsc2007_irq, 0,
>>   			client->dev.driver->name, ts);
>>   	if (err<  0) {
>> @@ -325,7 +303,6 @@
>>
>>    err_free_irq:
>>   	free_irq(ts->irq, ts);
>> -	hrtimer_cancel(&ts->timer);
>
> Why don't we cancel work here?

We should.

>
>>    err_free_mem:
>>   	input_free_device(input_dev);
>>   	kfree(ts);
>> @@ -337,11 +314,13 @@
>>   	struct tsc2007	*ts = i2c_get_clientdata(client);
>>   	struct tsc2007_platform_data *pdata;
>>
>> +	/* cancel any work */
>> +	cancel_delayed_work(&ts->work);
>> +
>>   	pdata = client->dev.platform_data;
>>   	pdata->exit_platform_hw();
>>
>
> So what happens if IRQ is raised here and work is scheduled again?
>
>>   	free_irq(ts->irq, ts);
>> -	hrtimer_cancel(&ts->timer);
>>   	input_unregister_device(ts->input);
>>   	kfree(ts);
>>
>
> Could you please try the patch below and see if the touchscreen still
> works for you? It should be applied on top of your patch.

I have modified my code accordingly to most of your changes. I don't 
have any platform callbacks in my platform, so make sure those are 
optional. Will try it out!

--Richard
--
To unsubscribe from this list: send the line "unsubscribe linux-input" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

  parent reply	other threads:[~2009-07-14  8:47 UTC|newest]

Thread overview: 10+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2009-06-23 11:54 [PATCH 1/2] tsc2007: remove HR timer Richard Röjfors
2009-06-25 21:20 ` Andrew Morton
2009-07-09 16:51   ` Richard Röjfors
2009-07-14  4:59 ` Dmitry Torokhov
2009-07-14  7:08   ` Thierry Reding
2009-07-14  8:00     ` Richard Röjfors
2009-07-14  8:21       ` Dmitry Torokhov
2009-07-14  8:21     ` Dmitry Torokhov
2009-07-14  8:47   ` Richard Röjfors [this message]
2009-07-24 14:39   ` Richard Röjfors

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=4A5C4639.1070003@mocean-labs.com \
    --to=richard.rojfors.ext@mocean-labs.com \
    --cc=akpm@linux-foundation.org \
    --cc=dmitry.torokhov@gmail.com \
    --cc=kwangwoo.lee@gmail.com \
    --cc=linux-input@vger.kernel.org \
    --cc=linux-kernel@vger.kernel.org \
    --cc=soni.trilok@gmail.com \
    --cc=thierry.reding@avionic-design.de \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).