diff options
author | Sebastian Andrzej Siewior <bigeasy@linutronix.de> | 2013-12-19 10:28:29 -0500 |
---|---|---|
committer | Lee Jones <lee.jones@linaro.org> | 2014-01-07 03:41:15 -0500 |
commit | 7e170c6e4f7501bea900aa66b2b27a6ce5001e25 (patch) | |
tree | 76eea8291945b24d8bbbf215243f7239c9728a08 | |
parent | 3466bd2273b81a0a29d0e134ba1c78b64b84f40b (diff) |
mfd: ti_am335x_tscadc: Don't read back REG_SE
The purpose of reg_se_cache has been defeated. It should avoid the
read-back of the register to avoid the latency and the fact that the
bits are reset to 0 after the individual conversation took place.
The reason why this is required like this to work, is that read-back of
the register removes the bits of the ADC so they do not start another
conversation after the register is re-written from the TSC side for the
update.
To avoid the not required read-back I introduce a "set once" variant which
does not update the cache mask. After the conversation completes, the
bit is removed from the SE register anyway and we don't plan a new
conversation "any time soon". The current set function is renamed to
set_cache to distinguish the two operations.
This is a small preparation for a larger sync-rework.
Signed-off-by: Sebastian Andrzej Siewior <bigeasy@linutronix.de>
Acked-by: Dmitry Torokhov <dmitry.torokhov@gmail.com>
Acked-by: Jonathan Cameron <jic23@kernel.org>
Signed-off-by: Lee Jones <lee.jones@linaro.org>
-rw-r--r-- | drivers/iio/adc/ti_am335x_adc.c | 4 | ||||
-rw-r--r-- | drivers/input/touchscreen/ti_am335x_tsc.c | 4 | ||||
-rw-r--r-- | drivers/mfd/ti_am335x_tscadc.c | 16 | ||||
-rw-r--r-- | include/linux/mfd/ti_am335x_tscadc.h | 3 |
4 files changed, 18 insertions, 9 deletions
diff --git a/drivers/iio/adc/ti_am335x_adc.c b/drivers/iio/adc/ti_am335x_adc.c index ce8d03ac900f..95eef8e89979 100644 --- a/drivers/iio/adc/ti_am335x_adc.c +++ b/drivers/iio/adc/ti_am335x_adc.c | |||
@@ -181,7 +181,7 @@ static int tiadc_buffer_postenable(struct iio_dev *indio_dev) | |||
181 | enb |= (get_adc_step_bit(adc_dev, bit) << 1); | 181 | enb |= (get_adc_step_bit(adc_dev, bit) << 1); |
182 | adc_dev->buffer_en_ch_steps = enb; | 182 | adc_dev->buffer_en_ch_steps = enb; |
183 | 183 | ||
184 | am335x_tsc_se_set(adc_dev->mfd_tscadc, enb); | 184 | am335x_tsc_se_set_cache(adc_dev->mfd_tscadc, enb); |
185 | 185 | ||
186 | tiadc_writel(adc_dev, REG_IRQSTATUS, IRQENB_FIFO1THRES | 186 | tiadc_writel(adc_dev, REG_IRQSTATUS, IRQENB_FIFO1THRES |
187 | | IRQENB_FIFO1OVRRUN | IRQENB_FIFO1UNDRFLW); | 187 | | IRQENB_FIFO1OVRRUN | IRQENB_FIFO1UNDRFLW); |
@@ -332,7 +332,7 @@ static int tiadc_read_raw(struct iio_dev *indio_dev, | |||
332 | return -EBUSY; | 332 | return -EBUSY; |
333 | 333 | ||
334 | step_en = get_adc_step_mask(adc_dev); | 334 | step_en = get_adc_step_mask(adc_dev); |
335 | am335x_tsc_se_set(adc_dev->mfd_tscadc, step_en); | 335 | am335x_tsc_se_set_once(adc_dev->mfd_tscadc, step_en); |
336 | 336 | ||
337 | /* Wait for ADC sequencer to complete sampling */ | 337 | /* Wait for ADC sequencer to complete sampling */ |
338 | while (tiadc_readl(adc_dev, REG_ADCFSM) & SEQ_STATUS) { | 338 | while (tiadc_readl(adc_dev, REG_ADCFSM) & SEQ_STATUS) { |
diff --git a/drivers/input/touchscreen/ti_am335x_tsc.c b/drivers/input/touchscreen/ti_am335x_tsc.c index 68beadaabceb..2ca5a7bee04e 100644 --- a/drivers/input/touchscreen/ti_am335x_tsc.c +++ b/drivers/input/touchscreen/ti_am335x_tsc.c | |||
@@ -198,7 +198,7 @@ static void titsc_step_config(struct titsc *ts_dev) | |||
198 | /* The steps1 … end and bit 0 for TS_Charge */ | 198 | /* The steps1 … end and bit 0 for TS_Charge */ |
199 | stepenable = (1 << (end_step + 2)) - 1; | 199 | stepenable = (1 << (end_step + 2)) - 1; |
200 | ts_dev->step_mask = stepenable; | 200 | ts_dev->step_mask = stepenable; |
201 | am335x_tsc_se_set(ts_dev->mfd_tscadc, ts_dev->step_mask); | 201 | am335x_tsc_se_set_cache(ts_dev->mfd_tscadc, ts_dev->step_mask); |
202 | } | 202 | } |
203 | 203 | ||
204 | static void titsc_read_coordinates(struct titsc *ts_dev, | 204 | static void titsc_read_coordinates(struct titsc *ts_dev, |
@@ -322,7 +322,7 @@ static irqreturn_t titsc_irq(int irq, void *dev) | |||
322 | 322 | ||
323 | if (irqclr) { | 323 | if (irqclr) { |
324 | titsc_writel(ts_dev, REG_IRQSTATUS, irqclr); | 324 | titsc_writel(ts_dev, REG_IRQSTATUS, irqclr); |
325 | am335x_tsc_se_set(ts_dev->mfd_tscadc, ts_dev->step_mask); | 325 | am335x_tsc_se_set_cache(ts_dev->mfd_tscadc, ts_dev->step_mask); |
326 | return IRQ_HANDLED; | 326 | return IRQ_HANDLED; |
327 | } | 327 | } |
328 | return IRQ_NONE; | 328 | return IRQ_NONE; |
diff --git a/drivers/mfd/ti_am335x_tscadc.c b/drivers/mfd/ti_am335x_tscadc.c index 67d0eb469a45..cb0c211fc7d8 100644 --- a/drivers/mfd/ti_am335x_tscadc.c +++ b/drivers/mfd/ti_am335x_tscadc.c | |||
@@ -53,24 +53,32 @@ static void am335x_tsc_se_update(struct ti_tscadc_dev *tsadc) | |||
53 | tscadc_writel(tsadc, REG_SE, tsadc->reg_se_cache); | 53 | tscadc_writel(tsadc, REG_SE, tsadc->reg_se_cache); |
54 | } | 54 | } |
55 | 55 | ||
56 | void am335x_tsc_se_set(struct ti_tscadc_dev *tsadc, u32 val) | 56 | void am335x_tsc_se_set_cache(struct ti_tscadc_dev *tsadc, u32 val) |
57 | { | 57 | { |
58 | unsigned long flags; | 58 | unsigned long flags; |
59 | 59 | ||
60 | spin_lock_irqsave(&tsadc->reg_lock, flags); | 60 | spin_lock_irqsave(&tsadc->reg_lock, flags); |
61 | tsadc->reg_se_cache = tscadc_readl(tsadc, REG_SE); | ||
62 | tsadc->reg_se_cache |= val; | 61 | tsadc->reg_se_cache |= val; |
63 | am335x_tsc_se_update(tsadc); | 62 | am335x_tsc_se_update(tsadc); |
64 | spin_unlock_irqrestore(&tsadc->reg_lock, flags); | 63 | spin_unlock_irqrestore(&tsadc->reg_lock, flags); |
65 | } | 64 | } |
66 | EXPORT_SYMBOL_GPL(am335x_tsc_se_set); | 65 | EXPORT_SYMBOL_GPL(am335x_tsc_se_set_cache); |
66 | |||
67 | void am335x_tsc_se_set_once(struct ti_tscadc_dev *tsadc, u32 val) | ||
68 | { | ||
69 | unsigned long flags; | ||
70 | |||
71 | spin_lock_irqsave(&tsadc->reg_lock, flags); | ||
72 | tscadc_writel(tsadc, REG_SE, tsadc->reg_se_cache | val); | ||
73 | spin_unlock_irqrestore(&tsadc->reg_lock, flags); | ||
74 | } | ||
75 | EXPORT_SYMBOL_GPL(am335x_tsc_se_set_once); | ||
67 | 76 | ||
68 | void am335x_tsc_se_clr(struct ti_tscadc_dev *tsadc, u32 val) | 77 | void am335x_tsc_se_clr(struct ti_tscadc_dev *tsadc, u32 val) |
69 | { | 78 | { |
70 | unsigned long flags; | 79 | unsigned long flags; |
71 | 80 | ||
72 | spin_lock_irqsave(&tsadc->reg_lock, flags); | 81 | spin_lock_irqsave(&tsadc->reg_lock, flags); |
73 | tsadc->reg_se_cache = tscadc_readl(tsadc, REG_SE); | ||
74 | tsadc->reg_se_cache &= ~val; | 82 | tsadc->reg_se_cache &= ~val; |
75 | am335x_tsc_se_update(tsadc); | 83 | am335x_tsc_se_update(tsadc); |
76 | spin_unlock_irqrestore(&tsadc->reg_lock, flags); | 84 | spin_unlock_irqrestore(&tsadc->reg_lock, flags); |
diff --git a/include/linux/mfd/ti_am335x_tscadc.h b/include/linux/mfd/ti_am335x_tscadc.h index 1fe72199e670..2fa9c0613da4 100644 --- a/include/linux/mfd/ti_am335x_tscadc.h +++ b/include/linux/mfd/ti_am335x_tscadc.h | |||
@@ -176,7 +176,8 @@ static inline struct ti_tscadc_dev *ti_tscadc_dev_get(struct platform_device *p) | |||
176 | return *tscadc_dev; | 176 | return *tscadc_dev; |
177 | } | 177 | } |
178 | 178 | ||
179 | void am335x_tsc_se_set(struct ti_tscadc_dev *tsadc, u32 val); | 179 | void am335x_tsc_se_set_cache(struct ti_tscadc_dev *tsadc, u32 val); |
180 | void am335x_tsc_se_set_once(struct ti_tscadc_dev *tsadc, u32 val); | ||
180 | void am335x_tsc_se_clr(struct ti_tscadc_dev *tsadc, u32 val); | 181 | void am335x_tsc_se_clr(struct ti_tscadc_dev *tsadc, u32 val); |
181 | 182 | ||
182 | #endif | 183 | #endif |