diff options
author | Dmitry Torokhov <dmitry.torokhov@gmail.com> | 2010-09-05 15:20:24 -0400 |
---|---|---|
committer | Dmitry Torokhov <dmitry.torokhov@gmail.com> | 2010-09-05 15:20:24 -0400 |
commit | 92944c1c943647c7eb39c081feeecdd169f68ee4 (patch) | |
tree | 1412deee46f6be19cf9b5a65a8d76ed4553866a8 | |
parent | 77686517977e77d101c8a7b397717df00a88922b (diff) | |
parent | d4f5f937c3dfdea9f49c3cdeea8813b0d9f7715a (diff) |
Merge branch 'mrst-touchscreen' into next
Conflicts:
drivers/input/touchscreen/Makefile
-rw-r--r-- | drivers/input/touchscreen/Kconfig | 12 | ||||
-rw-r--r-- | drivers/input/touchscreen/Makefile | 1 | ||||
-rw-r--r-- | drivers/input/touchscreen/intel-mid-touch.c | 687 | ||||
-rw-r--r-- | drivers/staging/Kconfig | 2 | ||||
-rw-r--r-- | drivers/staging/Makefile | 1 | ||||
-rw-r--r-- | drivers/staging/mrst-touchscreen/Kconfig | 7 | ||||
-rw-r--r-- | drivers/staging/mrst-touchscreen/Makefile | 3 | ||||
-rw-r--r-- | drivers/staging/mrst-touchscreen/TODO | 2 | ||||
-rw-r--r-- | drivers/staging/mrst-touchscreen/intel-mid-touch.c | 864 |
9 files changed, 700 insertions, 879 deletions
diff --git a/drivers/input/touchscreen/Kconfig b/drivers/input/touchscreen/Kconfig index 400c99de0bb3..d59feb7b90f6 100644 --- a/drivers/input/touchscreen/Kconfig +++ b/drivers/input/touchscreen/Kconfig | |||
@@ -260,6 +260,18 @@ config TOUCHSCREEN_INEXIO | |||
260 | To compile this driver as a module, choose M here: the | 260 | To compile this driver as a module, choose M here: the |
261 | module will be called inexio. | 261 | module will be called inexio. |
262 | 262 | ||
263 | config TOUCHSCREEN_INTEL_MID | ||
264 | tristate "Intel MID platform resistive touchscreen" | ||
265 | depends on INTEL_SCU_IPC | ||
266 | help | ||
267 | Say Y here if you have a Intel MID based touchscreen in | ||
268 | your system. | ||
269 | |||
270 | If unsure, say N. | ||
271 | |||
272 | To compile this driver as a module, choose M here: the | ||
273 | module will be called intel_mid_touch. | ||
274 | |||
263 | config TOUCHSCREEN_MK712 | 275 | config TOUCHSCREEN_MK712 |
264 | tristate "ICS MicroClock MK712 touchscreen" | 276 | tristate "ICS MicroClock MK712 touchscreen" |
265 | help | 277 | help |
diff --git a/drivers/input/touchscreen/Makefile b/drivers/input/touchscreen/Makefile index 22e2d598dc69..f1bc8a416824 100644 --- a/drivers/input/touchscreen/Makefile +++ b/drivers/input/touchscreen/Makefile | |||
@@ -23,6 +23,7 @@ obj-$(CONFIG_TOUCHSCREEN_EETI) += eeti_ts.o | |||
23 | obj-$(CONFIG_TOUCHSCREEN_ELO) += elo.o | 23 | obj-$(CONFIG_TOUCHSCREEN_ELO) += elo.o |
24 | obj-$(CONFIG_TOUCHSCREEN_FUJITSU) += fujitsu_ts.o | 24 | obj-$(CONFIG_TOUCHSCREEN_FUJITSU) += fujitsu_ts.o |
25 | obj-$(CONFIG_TOUCHSCREEN_INEXIO) += inexio.o | 25 | obj-$(CONFIG_TOUCHSCREEN_INEXIO) += inexio.o |
26 | obj-$(CONFIG_TOUCHSCREEN_INTEL_MID) += intel-mid-touch.o | ||
26 | obj-$(CONFIG_TOUCHSCREEN_LPC32XX) += lpc32xx_ts.o | 27 | obj-$(CONFIG_TOUCHSCREEN_LPC32XX) += lpc32xx_ts.o |
27 | obj-$(CONFIG_TOUCHSCREEN_MC13783) += mc13783_ts.o | 28 | obj-$(CONFIG_TOUCHSCREEN_MC13783) += mc13783_ts.o |
28 | obj-$(CONFIG_TOUCHSCREEN_MCS5000) += mcs5000_ts.o | 29 | obj-$(CONFIG_TOUCHSCREEN_MCS5000) += mcs5000_ts.o |
diff --git a/drivers/input/touchscreen/intel-mid-touch.c b/drivers/input/touchscreen/intel-mid-touch.c new file mode 100644 index 000000000000..c0307b22d86f --- /dev/null +++ b/drivers/input/touchscreen/intel-mid-touch.c | |||
@@ -0,0 +1,687 @@ | |||
1 | /* | ||
2 | * Intel MID Resistive Touch Screen Driver | ||
3 | * | ||
4 | * Copyright (C) 2008 Intel Corp | ||
5 | * | ||
6 | * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||
7 | * | ||
8 | * This program is free software; you can redistribute it and/or modify | ||
9 | * it under the terms of the GNU General Public License as published by | ||
10 | * the Free Software Foundation; version 2 of the License. | ||
11 | * | ||
12 | * This program is distributed in the hope that it will be useful, but | ||
13 | * WITHOUT ANY WARRANTY; without even the implied warranty of | ||
14 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | ||
15 | * General Public License for more details. | ||
16 | * | ||
17 | * You should have received a copy of the GNU General Public License along | ||
18 | * with this program; if not, write to the Free Software Foundation, Inc., | ||
19 | * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA. | ||
20 | * | ||
21 | * Questions/Comments/Bug fixes to Sreedhara (sreedhara.ds@intel.com) | ||
22 | * Ramesh Agarwal (ramesh.agarwal@intel.com) | ||
23 | * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||
24 | * | ||
25 | * TODO: | ||
26 | * review conversion of r/m/w sequences | ||
27 | */ | ||
28 | |||
29 | #include <linux/module.h> | ||
30 | #include <linux/init.h> | ||
31 | #include <linux/input.h> | ||
32 | #include <linux/interrupt.h> | ||
33 | #include <linux/err.h> | ||
34 | #include <linux/param.h> | ||
35 | #include <linux/slab.h> | ||
36 | #include <linux/platform_device.h> | ||
37 | #include <linux/irq.h> | ||
38 | #include <linux/delay.h> | ||
39 | #include <asm/intel_scu_ipc.h> | ||
40 | |||
41 | /* PMIC Interrupt registers */ | ||
42 | #define PMIC_REG_ID1 0x00 /* PMIC ID1 register */ | ||
43 | |||
44 | /* PMIC Interrupt registers */ | ||
45 | #define PMIC_REG_INT 0x04 /* PMIC interrupt register */ | ||
46 | #define PMIC_REG_MINT 0x05 /* PMIC interrupt mask register */ | ||
47 | |||
48 | /* ADC Interrupt registers */ | ||
49 | #define PMIC_REG_ADCINT 0x5F /* ADC interrupt register */ | ||
50 | #define PMIC_REG_MADCINT 0x60 /* ADC interrupt mask register */ | ||
51 | |||
52 | /* ADC Control registers */ | ||
53 | #define PMIC_REG_ADCCNTL1 0x61 /* ADC control register */ | ||
54 | |||
55 | /* ADC Channel Selection registers */ | ||
56 | #define PMICADDR0 0xA4 | ||
57 | #define END_OF_CHANNEL 0x1F | ||
58 | |||
59 | /* ADC Result register */ | ||
60 | #define PMIC_REG_ADCSNS0H 0x64 | ||
61 | |||
62 | /* ADC channels for touch screen */ | ||
63 | #define MRST_TS_CHAN10 0xA /* Touch screen X+ connection */ | ||
64 | #define MRST_TS_CHAN11 0xB /* Touch screen X- connection */ | ||
65 | #define MRST_TS_CHAN12 0xC /* Touch screen Y+ connection */ | ||
66 | #define MRST_TS_CHAN13 0xD /* Touch screen Y- connection */ | ||
67 | |||
68 | /* Touch screen channel BIAS constants */ | ||
69 | #define MRST_XBIAS 0x20 | ||
70 | #define MRST_YBIAS 0x40 | ||
71 | #define MRST_ZBIAS 0x80 | ||
72 | |||
73 | /* Touch screen coordinates */ | ||
74 | #define MRST_X_MIN 10 | ||
75 | #define MRST_X_MAX 1024 | ||
76 | #define MRST_X_FUZZ 5 | ||
77 | #define MRST_Y_MIN 10 | ||
78 | #define MRST_Y_MAX 1024 | ||
79 | #define MRST_Y_FUZZ 5 | ||
80 | #define MRST_PRESSURE_MIN 0 | ||
81 | #define MRST_PRESSURE_NOMINAL 50 | ||
82 | #define MRST_PRESSURE_MAX 100 | ||
83 | |||
84 | #define WAIT_ADC_COMPLETION 10 /* msec */ | ||
85 | |||
86 | /* PMIC ADC round robin delays */ | ||
87 | #define ADC_LOOP_DELAY0 0x0 /* Continuous loop */ | ||
88 | #define ADC_LOOP_DELAY1 0x1 /* 4.5 ms approximate */ | ||
89 | |||
90 | /* PMIC Vendor Identifiers */ | ||
91 | #define PMIC_VENDOR_FS 0 /* PMIC vendor FreeScale */ | ||
92 | #define PMIC_VENDOR_MAXIM 1 /* PMIC vendor MAXIM */ | ||
93 | #define PMIC_VENDOR_NEC 2 /* PMIC vendor NEC */ | ||
94 | #define MRSTOUCH_MAX_CHANNELS 32 /* Maximum ADC channels */ | ||
95 | |||
96 | /* Touch screen device structure */ | ||
97 | struct mrstouch_dev { | ||
98 | struct device *dev; /* device associated with touch screen */ | ||
99 | struct input_dev *input; | ||
100 | char phys[32]; | ||
101 | u16 asr; /* Address selection register */ | ||
102 | int irq; | ||
103 | unsigned int vendor; /* PMIC vendor */ | ||
104 | unsigned int rev; /* PMIC revision */ | ||
105 | |||
106 | int (*read_prepare)(struct mrstouch_dev *tsdev); | ||
107 | int (*read)(struct mrstouch_dev *tsdev, u16 *x, u16 *y, u16 *z); | ||
108 | int (*read_finish)(struct mrstouch_dev *tsdev); | ||
109 | }; | ||
110 | |||
111 | |||
112 | /*************************** NEC and Maxim Interface ************************/ | ||
113 | |||
114 | static int mrstouch_nec_adc_read_prepare(struct mrstouch_dev *tsdev) | ||
115 | { | ||
116 | return intel_scu_ipc_update_register(PMIC_REG_MADCINT, 0, 0x20); | ||
117 | } | ||
118 | |||
119 | /* | ||
120 | * Enables PENDET interrupt. | ||
121 | */ | ||
122 | static int mrstouch_nec_adc_read_finish(struct mrstouch_dev *tsdev) | ||
123 | { | ||
124 | int err; | ||
125 | |||
126 | err = intel_scu_ipc_update_register(PMIC_REG_MADCINT, 0x20, 0x20); | ||
127 | if (!err) | ||
128 | err = intel_scu_ipc_update_register(PMIC_REG_ADCCNTL1, 0, 0x05); | ||
129 | |||
130 | return err; | ||
131 | } | ||
132 | |||
133 | /* | ||
134 | * Reads PMIC ADC touch screen result | ||
135 | * Reads ADC storage registers for higher 7 and lower 3 bits and | ||
136 | * converts the two readings into a single value and turns off gain bit | ||
137 | */ | ||
138 | static int mrstouch_ts_chan_read(u16 offset, u16 chan, u16 *vp, u16 *vm) | ||
139 | { | ||
140 | int err; | ||
141 | u16 result; | ||
142 | u32 res; | ||
143 | |||
144 | result = PMIC_REG_ADCSNS0H + offset; | ||
145 | |||
146 | if (chan == MRST_TS_CHAN12) | ||
147 | result += 4; | ||
148 | |||
149 | err = intel_scu_ipc_ioread32(result, &res); | ||
150 | if (err) | ||
151 | return err; | ||
152 | |||
153 | /* Mash the bits up */ | ||
154 | |||
155 | *vp = (res & 0xFF) << 3; /* Highest 7 bits */ | ||
156 | *vp |= (res >> 8) & 0x07; /* Lower 3 bits */ | ||
157 | *vp &= 0x3FF; | ||
158 | |||
159 | res >>= 16; | ||
160 | |||
161 | *vm = (res & 0xFF) << 3; /* Highest 7 bits */ | ||
162 | *vm |= (res >> 8) & 0x07; /* Lower 3 bits */ | ||
163 | *vm &= 0x3FF; | ||
164 | |||
165 | return 0; | ||
166 | } | ||
167 | |||
168 | /* | ||
169 | * Enables X, Y and Z bias values | ||
170 | * Enables YPYM for X channels and XPXM for Y channels | ||
171 | */ | ||
172 | static int mrstouch_ts_bias_set(uint offset, uint bias) | ||
173 | { | ||
174 | int count; | ||
175 | u16 chan, start; | ||
176 | u16 reg[4]; | ||
177 | u8 data[4]; | ||
178 | |||
179 | chan = PMICADDR0 + offset; | ||
180 | start = MRST_TS_CHAN10; | ||
181 | |||
182 | for (count = 0; count <= 3; count++) { | ||
183 | reg[count] = chan++; | ||
184 | data[count] = bias | (start + count); | ||
185 | } | ||
186 | |||
187 | return intel_scu_ipc_writev(reg, data, 4); | ||
188 | } | ||
189 | |||
190 | /* To read touch screen channel values */ | ||
191 | static int mrstouch_nec_adc_read(struct mrstouch_dev *tsdev, | ||
192 | u16 *x, u16 *y, u16 *z) | ||
193 | { | ||
194 | int err; | ||
195 | u16 xm, ym, zm; | ||
196 | |||
197 | /* configure Y bias for X channels */ | ||
198 | err = mrstouch_ts_bias_set(tsdev->asr, MRST_YBIAS); | ||
199 | if (err) | ||
200 | goto ipc_error; | ||
201 | |||
202 | msleep(WAIT_ADC_COMPLETION); | ||
203 | |||
204 | /* read x+ and x- channels */ | ||
205 | err = mrstouch_ts_chan_read(tsdev->asr, MRST_TS_CHAN10, x, &xm); | ||
206 | if (err) | ||
207 | goto ipc_error; | ||
208 | |||
209 | /* configure x bias for y channels */ | ||
210 | err = mrstouch_ts_bias_set(tsdev->asr, MRST_XBIAS); | ||
211 | if (err) | ||
212 | goto ipc_error; | ||
213 | |||
214 | msleep(WAIT_ADC_COMPLETION); | ||
215 | |||
216 | /* read y+ and y- channels */ | ||
217 | err = mrstouch_ts_chan_read(tsdev->asr, MRST_TS_CHAN12, y, &ym); | ||
218 | if (err) | ||
219 | goto ipc_error; | ||
220 | |||
221 | /* configure z bias for x and y channels */ | ||
222 | err = mrstouch_ts_bias_set(tsdev->asr, MRST_ZBIAS); | ||
223 | if (err) | ||
224 | goto ipc_error; | ||
225 | |||
226 | msleep(WAIT_ADC_COMPLETION); | ||
227 | |||
228 | /* read z+ and z- channels */ | ||
229 | err = mrstouch_ts_chan_read(tsdev->asr, MRST_TS_CHAN10, z, &zm); | ||
230 | if (err) | ||
231 | goto ipc_error; | ||
232 | |||
233 | return 0; | ||
234 | |||
235 | ipc_error: | ||
236 | dev_err(tsdev->dev, "ipc error during adc read\n"); | ||
237 | return err; | ||
238 | } | ||
239 | |||
240 | |||
241 | /*************************** Freescale Interface ************************/ | ||
242 | |||
243 | static int mrstouch_fs_adc_read_prepare(struct mrstouch_dev *tsdev) | ||
244 | { | ||
245 | int err, count; | ||
246 | u16 chan; | ||
247 | u16 reg[5]; | ||
248 | u8 data[5]; | ||
249 | |||
250 | /* Stop the ADC */ | ||
251 | err = intel_scu_ipc_update_register(PMIC_REG_MADCINT, 0x00, 0x02); | ||
252 | if (err) | ||
253 | goto ipc_error; | ||
254 | |||
255 | chan = PMICADDR0 + tsdev->asr; | ||
256 | |||
257 | /* Set X BIAS */ | ||
258 | for (count = 0; count <= 3; count++) { | ||
259 | reg[count] = chan++; | ||
260 | data[count] = 0x2A; | ||
261 | } | ||
262 | reg[count] = chan++; /* Dummy */ | ||
263 | data[count] = 0; | ||
264 | |||
265 | err = intel_scu_ipc_writev(reg, data, 5); | ||
266 | if (err) | ||
267 | goto ipc_error; | ||
268 | |||
269 | msleep(WAIT_ADC_COMPLETION); | ||
270 | |||
271 | /* Set Y BIAS */ | ||
272 | for (count = 0; count <= 3; count++) { | ||
273 | reg[count] = chan++; | ||
274 | data[count] = 0x4A; | ||
275 | } | ||
276 | reg[count] = chan++; /* Dummy */ | ||
277 | data[count] = 0; | ||
278 | |||
279 | err = intel_scu_ipc_writev(reg, data, 5); | ||
280 | if (err) | ||
281 | goto ipc_error; | ||
282 | |||
283 | msleep(WAIT_ADC_COMPLETION); | ||
284 | |||
285 | /* Set Z BIAS */ | ||
286 | err = intel_scu_ipc_iowrite32(chan + 2, 0x8A8A8A8A); | ||
287 | if (err) | ||
288 | goto ipc_error; | ||
289 | |||
290 | msleep(WAIT_ADC_COMPLETION); | ||
291 | |||
292 | return 0; | ||
293 | |||
294 | ipc_error: | ||
295 | dev_err(tsdev->dev, "ipc error during %s\n", __func__); | ||
296 | return err; | ||
297 | } | ||
298 | |||
299 | static int mrstouch_fs_adc_read(struct mrstouch_dev *tsdev, | ||
300 | u16 *x, u16 *y, u16 *z) | ||
301 | { | ||
302 | int err; | ||
303 | u16 result; | ||
304 | u16 reg[4]; | ||
305 | u8 data[4]; | ||
306 | |||
307 | result = PMIC_REG_ADCSNS0H + tsdev->asr; | ||
308 | |||
309 | reg[0] = result + 4; | ||
310 | reg[1] = result + 5; | ||
311 | reg[2] = result + 16; | ||
312 | reg[3] = result + 17; | ||
313 | |||
314 | err = intel_scu_ipc_readv(reg, data, 4); | ||
315 | if (err) | ||
316 | goto ipc_error; | ||
317 | |||
318 | *x = data[0] << 3; /* Higher 7 bits */ | ||
319 | *x |= data[1] & 0x7; /* Lower 3 bits */ | ||
320 | *x &= 0x3FF; | ||
321 | |||
322 | *y = data[2] << 3; /* Higher 7 bits */ | ||
323 | *y |= data[3] & 0x7; /* Lower 3 bits */ | ||
324 | *y &= 0x3FF; | ||
325 | |||
326 | /* Read Z value */ | ||
327 | reg[0] = result + 28; | ||
328 | reg[1] = result + 29; | ||
329 | |||
330 | err = intel_scu_ipc_readv(reg, data, 4); | ||
331 | if (err) | ||
332 | goto ipc_error; | ||
333 | |||
334 | *z = data[0] << 3; /* Higher 7 bits */ | ||
335 | *z |= data[1] & 0x7; /* Lower 3 bits */ | ||
336 | *z &= 0x3FF; | ||
337 | |||
338 | return 0; | ||
339 | |||
340 | ipc_error: | ||
341 | dev_err(tsdev->dev, "ipc error during %s\n", __func__); | ||
342 | return err; | ||
343 | } | ||
344 | |||
345 | static int mrstouch_fs_adc_read_finish(struct mrstouch_dev *tsdev) | ||
346 | { | ||
347 | int err, count; | ||
348 | u16 chan; | ||
349 | u16 reg[5]; | ||
350 | u8 data[5]; | ||
351 | |||
352 | /* Clear all TS channels */ | ||
353 | chan = PMICADDR0 + tsdev->asr; | ||
354 | for (count = 0; count <= 4; count++) { | ||
355 | reg[count] = chan++; | ||
356 | data[count] = 0; | ||
357 | } | ||
358 | err = intel_scu_ipc_writev(reg, data, 5); | ||
359 | if (err) | ||
360 | goto ipc_error; | ||
361 | |||
362 | for (count = 0; count <= 4; count++) { | ||
363 | reg[count] = chan++; | ||
364 | data[count] = 0; | ||
365 | } | ||
366 | err = intel_scu_ipc_writev(reg, data, 5); | ||
367 | if (err) | ||
368 | goto ipc_error; | ||
369 | |||
370 | err = intel_scu_ipc_iowrite32(chan + 2, 0x00000000); | ||
371 | if (err) | ||
372 | goto ipc_error; | ||
373 | |||
374 | /* Start ADC */ | ||
375 | err = intel_scu_ipc_update_register(PMIC_REG_MADCINT, 0x02, 0x02); | ||
376 | if (err) | ||
377 | goto ipc_error; | ||
378 | |||
379 | return 0; | ||
380 | |||
381 | ipc_error: | ||
382 | dev_err(tsdev->dev, "ipc error during %s\n", __func__); | ||
383 | return err; | ||
384 | } | ||
385 | |||
386 | static void mrstouch_report_event(struct input_dev *input, | ||
387 | unsigned int x, unsigned int y, unsigned int z) | ||
388 | { | ||
389 | if (z > MRST_PRESSURE_NOMINAL) { | ||
390 | /* Pen touched, report button touch and coordinates */ | ||
391 | input_report_key(input, BTN_TOUCH, 1); | ||
392 | input_report_abs(input, ABS_X, x); | ||
393 | input_report_abs(input, ABS_Y, y); | ||
394 | } else { | ||
395 | input_report_key(input, BTN_TOUCH, 0); | ||
396 | } | ||
397 | |||
398 | input_report_abs(input, ABS_PRESSURE, z); | ||
399 | input_sync(input); | ||
400 | } | ||
401 | |||
402 | /* PENDET interrupt handler */ | ||
403 | static irqreturn_t mrstouch_pendet_irq(int irq, void *dev_id) | ||
404 | { | ||
405 | struct mrstouch_dev *tsdev = dev_id; | ||
406 | u16 x, y, z; | ||
407 | |||
408 | /* | ||
409 | * Should we lower thread priority? Probably not, since we are | ||
410 | * not spinning but sleeping... | ||
411 | */ | ||
412 | |||
413 | if (tsdev->read_prepare(tsdev)) | ||
414 | goto out; | ||
415 | |||
416 | do { | ||
417 | if (tsdev->read(tsdev, &x, &y, &z)) | ||
418 | break; | ||
419 | |||
420 | mrstouch_report_event(tsdev->input, x, y, z); | ||
421 | } while (z > MRST_PRESSURE_NOMINAL); | ||
422 | |||
423 | tsdev->read_finish(tsdev); | ||
424 | |||
425 | out: | ||
426 | return IRQ_HANDLED; | ||
427 | } | ||
428 | |||
429 | /* Utility to read PMIC ID */ | ||
430 | static int __devinit mrstouch_read_pmic_id(uint *vendor, uint *rev) | ||
431 | { | ||
432 | int err; | ||
433 | u8 r; | ||
434 | |||
435 | err = intel_scu_ipc_ioread8(PMIC_REG_ID1, &r); | ||
436 | if (err) | ||
437 | return err; | ||
438 | |||
439 | *vendor = r & 0x7; | ||
440 | *rev = (r >> 3) & 0x7; | ||
441 | |||
442 | return 0; | ||
443 | } | ||
444 | |||
445 | /* | ||
446 | * Parse ADC channels to find end of the channel configured by other ADC user | ||
447 | * NEC and MAXIM requires 4 channels and FreeScale needs 18 channels | ||
448 | */ | ||
449 | static int __devinit mrstouch_chan_parse(struct mrstouch_dev *tsdev) | ||
450 | { | ||
451 | int err, i, found; | ||
452 | u8 r8; | ||
453 | |||
454 | found = -1; | ||
455 | |||
456 | for (i = 0; i < MRSTOUCH_MAX_CHANNELS; i++) { | ||
457 | if (found >= 0) | ||
458 | break; | ||
459 | |||
460 | err = intel_scu_ipc_ioread8(PMICADDR0 + i, &r8); | ||
461 | if (err) | ||
462 | return err; | ||
463 | |||
464 | if (r8 == END_OF_CHANNEL) { | ||
465 | found = i; | ||
466 | break; | ||
467 | } | ||
468 | } | ||
469 | if (found < 0) | ||
470 | return 0; | ||
471 | |||
472 | if (tsdev->vendor == PMIC_VENDOR_FS) { | ||
473 | if (found && found > (MRSTOUCH_MAX_CHANNELS - 18)) | ||
474 | return -ENOSPC; | ||
475 | } else { | ||
476 | if (found && found > (MRSTOUCH_MAX_CHANNELS - 4)) | ||
477 | return -ENOSPC; | ||
478 | } | ||
479 | return found; | ||
480 | } | ||
481 | |||
482 | |||
483 | /* | ||
484 | * Writes touch screen channels to ADC address selection registers | ||
485 | */ | ||
486 | static int __devinit mrstouch_ts_chan_set(uint offset) | ||
487 | { | ||
488 | u16 chan; | ||
489 | |||
490 | int ret, count; | ||
491 | |||
492 | chan = PMICADDR0 + offset; | ||
493 | for (count = 0; count <= 3; count++) { | ||
494 | ret = intel_scu_ipc_iowrite8(chan++, MRST_TS_CHAN10 + count); | ||
495 | if (ret) | ||
496 | return ret; | ||
497 | } | ||
498 | return intel_scu_ipc_iowrite8(chan++, END_OF_CHANNEL); | ||
499 | } | ||
500 | |||
501 | /* Initialize ADC */ | ||
502 | static int __devinit mrstouch_adc_init(struct mrstouch_dev *tsdev) | ||
503 | { | ||
504 | int err, start; | ||
505 | u8 ra, rm; | ||
506 | |||
507 | err = mrstouch_read_pmic_id(&tsdev->vendor, &tsdev->rev); | ||
508 | if (err) { | ||
509 | dev_err(tsdev->dev, "Unable to read PMIC id\n"); | ||
510 | return err; | ||
511 | } | ||
512 | |||
513 | switch (tsdev->vendor) { | ||
514 | case PMIC_VENDOR_NEC: | ||
515 | case PMIC_VENDOR_MAXIM: | ||
516 | tsdev->read_prepare = mrstouch_nec_adc_read_prepare; | ||
517 | tsdev->read = mrstouch_nec_adc_read; | ||
518 | tsdev->read_finish = mrstouch_nec_adc_read_finish; | ||
519 | break; | ||
520 | |||
521 | case PMIC_VENDOR_FS: | ||
522 | tsdev->read_prepare = mrstouch_fs_adc_read_prepare; | ||
523 | tsdev->read = mrstouch_fs_adc_read; | ||
524 | tsdev->read_finish = mrstouch_fs_adc_read_finish; | ||
525 | break; | ||
526 | |||
527 | default: | ||
528 | dev_err(tsdev->dev, | ||
529 | "Unsupported touchscreen: %d\n", tsdev->vendor); | ||
530 | return -ENXIO; | ||
531 | } | ||
532 | |||
533 | start = mrstouch_chan_parse(tsdev); | ||
534 | if (start < 0) { | ||
535 | dev_err(tsdev->dev, "Unable to parse channels\n"); | ||
536 | return start; | ||
537 | } | ||
538 | |||
539 | tsdev->asr = start; | ||
540 | |||
541 | /* | ||
542 | * ADC power on, start, enable PENDET and set loop delay | ||
543 | * ADC loop delay is set to 4.5 ms approximately | ||
544 | * Loop delay more than this results in jitter in adc readings | ||
545 | * Setting loop delay to 0 (continous loop) in MAXIM stops PENDET | ||
546 | * interrupt generation sometimes. | ||
547 | */ | ||
548 | |||
549 | if (tsdev->vendor == PMIC_VENDOR_FS) { | ||
550 | ra = 0xE0 | ADC_LOOP_DELAY0; | ||
551 | rm = 0x5; | ||
552 | } else { | ||
553 | /* NEC and MAXIm not consistent with loop delay 0 */ | ||
554 | ra = 0xE0 | ADC_LOOP_DELAY1; | ||
555 | rm = 0x0; | ||
556 | |||
557 | /* configure touch screen channels */ | ||
558 | err = mrstouch_ts_chan_set(tsdev->asr); | ||
559 | if (err) | ||
560 | return err; | ||
561 | } | ||
562 | |||
563 | err = intel_scu_ipc_update_register(PMIC_REG_ADCCNTL1, ra, 0xE7); | ||
564 | if (err) | ||
565 | return err; | ||
566 | |||
567 | err = intel_scu_ipc_update_register(PMIC_REG_MADCINT, rm, 0x03); | ||
568 | if (err) | ||
569 | return err; | ||
570 | |||
571 | return 0; | ||
572 | } | ||
573 | |||
574 | |||
575 | /* Probe function for touch screen driver */ | ||
576 | static int __devinit mrstouch_probe(struct platform_device *pdev) | ||
577 | { | ||
578 | struct mrstouch_dev *tsdev; | ||
579 | struct input_dev *input; | ||
580 | int err; | ||
581 | int irq; | ||
582 | |||
583 | irq = platform_get_irq(pdev, 0); | ||
584 | if (irq < 0) { | ||
585 | dev_err(&pdev->dev, "no interrupt assigned\n"); | ||
586 | return -EINVAL; | ||
587 | } | ||
588 | |||
589 | tsdev = kzalloc(sizeof(struct mrstouch_dev), GFP_KERNEL); | ||
590 | input = input_allocate_device(); | ||
591 | if (!tsdev || !input) { | ||
592 | dev_err(&pdev->dev, "unable to allocate memory\n"); | ||
593 | err = -ENOMEM; | ||
594 | goto err_free_mem; | ||
595 | } | ||
596 | |||
597 | tsdev->dev = &pdev->dev; | ||
598 | tsdev->input = input; | ||
599 | tsdev->irq = irq; | ||
600 | |||
601 | snprintf(tsdev->phys, sizeof(tsdev->phys), | ||
602 | "%s/input0", dev_name(tsdev->dev)); | ||
603 | |||
604 | err = mrstouch_adc_init(tsdev); | ||
605 | if (err) { | ||
606 | dev_err(&pdev->dev, "ADC initialization failed\n"); | ||
607 | goto err_free_mem; | ||
608 | } | ||
609 | |||
610 | input->name = "mrst_touchscreen"; | ||
611 | input->phys = tsdev->phys; | ||
612 | input->dev.parent = tsdev->dev; | ||
613 | |||
614 | input->id.vendor = tsdev->vendor; | ||
615 | input->id.version = tsdev->rev; | ||
616 | |||
617 | input->evbit[0] = BIT_MASK(EV_KEY) | BIT_MASK(EV_ABS); | ||
618 | input->keybit[BIT_WORD(BTN_TOUCH)] = BIT_MASK(BTN_TOUCH); | ||
619 | |||
620 | input_set_abs_params(tsdev->input, ABS_X, | ||
621 | MRST_X_MIN, MRST_X_MAX, MRST_X_FUZZ, 0); | ||
622 | input_set_abs_params(tsdev->input, ABS_Y, | ||
623 | MRST_Y_MIN, MRST_Y_MAX, MRST_Y_FUZZ, 0); | ||
624 | input_set_abs_params(tsdev->input, ABS_PRESSURE, | ||
625 | MRST_PRESSURE_MIN, MRST_PRESSURE_MAX, 0, 0); | ||
626 | |||
627 | err = request_threaded_irq(tsdev->irq, NULL, mrstouch_pendet_irq, | ||
628 | 0, "mrstouch", tsdev); | ||
629 | if (err) { | ||
630 | dev_err(tsdev->dev, "unable to allocate irq\n"); | ||
631 | goto err_free_mem; | ||
632 | } | ||
633 | |||
634 | err = input_register_device(tsdev->input); | ||
635 | if (err) { | ||
636 | dev_err(tsdev->dev, "unable to register input device\n"); | ||
637 | goto err_free_irq; | ||
638 | } | ||
639 | |||
640 | platform_set_drvdata(pdev, tsdev); | ||
641 | return 0; | ||
642 | |||
643 | err_free_irq: | ||
644 | free_irq(tsdev->irq, tsdev); | ||
645 | err_free_mem: | ||
646 | input_free_device(input); | ||
647 | kfree(tsdev); | ||
648 | return err; | ||
649 | } | ||
650 | |||
651 | static int __devexit mrstouch_remove(struct platform_device *pdev) | ||
652 | { | ||
653 | struct mrstouch_dev *tsdev = platform_get_drvdata(pdev); | ||
654 | |||
655 | free_irq(tsdev->irq, tsdev); | ||
656 | input_unregister_device(tsdev->input); | ||
657 | kfree(tsdev); | ||
658 | |||
659 | platform_set_drvdata(pdev, NULL); | ||
660 | |||
661 | return 0; | ||
662 | } | ||
663 | |||
664 | static struct platform_driver mrstouch_driver = { | ||
665 | .driver = { | ||
666 | .name = "pmic_touch", | ||
667 | .owner = THIS_MODULE, | ||
668 | }, | ||
669 | .probe = mrstouch_probe, | ||
670 | .remove = __devexit_p(mrstouch_remove), | ||
671 | }; | ||
672 | |||
673 | static int __init mrstouch_init(void) | ||
674 | { | ||
675 | return platform_driver_register(&mrstouch_driver); | ||
676 | } | ||
677 | module_init(mrstouch_init); | ||
678 | |||
679 | static void __exit mrstouch_exit(void) | ||
680 | { | ||
681 | platform_driver_unregister(&mrstouch_driver); | ||
682 | } | ||
683 | module_exit(mrstouch_exit); | ||
684 | |||
685 | MODULE_AUTHOR("Sreedhara Murthy. D.S, sreedhara.ds@intel.com"); | ||
686 | MODULE_DESCRIPTION("Intel Moorestown Resistive Touch Screen Driver"); | ||
687 | MODULE_LICENSE("GPL"); | ||
diff --git a/drivers/staging/Kconfig b/drivers/staging/Kconfig index 4a7a7a7f11b6..5766aea69f00 100644 --- a/drivers/staging/Kconfig +++ b/drivers/staging/Kconfig | |||
@@ -141,8 +141,6 @@ source "drivers/staging/adis16255/Kconfig" | |||
141 | 141 | ||
142 | source "drivers/staging/xgifb/Kconfig" | 142 | source "drivers/staging/xgifb/Kconfig" |
143 | 143 | ||
144 | source "drivers/staging/mrst-touchscreen/Kconfig" | ||
145 | |||
146 | source "drivers/staging/msm/Kconfig" | 144 | source "drivers/staging/msm/Kconfig" |
147 | 145 | ||
148 | source "drivers/staging/lirc/Kconfig" | 146 | source "drivers/staging/lirc/Kconfig" |
diff --git a/drivers/staging/Makefile b/drivers/staging/Makefile index ca5c03eb3ce3..bbe62e0dc2ff 100644 --- a/drivers/staging/Makefile +++ b/drivers/staging/Makefile | |||
@@ -52,7 +52,6 @@ obj-$(CONFIG_CXT1E1) += cxt1e1/ | |||
52 | obj-$(CONFIG_TI_ST) += ti-st/ | 52 | obj-$(CONFIG_TI_ST) += ti-st/ |
53 | obj-$(CONFIG_ADIS16255) += adis16255/ | 53 | obj-$(CONFIG_ADIS16255) += adis16255/ |
54 | obj-$(CONFIG_FB_XGI) += xgifb/ | 54 | obj-$(CONFIG_FB_XGI) += xgifb/ |
55 | obj-$(CONFIG_TOUCHSCREEN_MRSTOUCH) += mrst-touchscreen/ | ||
56 | obj-$(CONFIG_MSM_STAGING) += msm/ | 55 | obj-$(CONFIG_MSM_STAGING) += msm/ |
57 | obj-$(CONFIG_EASYCAP) += easycap/ | 56 | obj-$(CONFIG_EASYCAP) += easycap/ |
58 | obj-$(CONFIG_SOLO6X10) += solo6x10/ | 57 | obj-$(CONFIG_SOLO6X10) += solo6x10/ |
diff --git a/drivers/staging/mrst-touchscreen/Kconfig b/drivers/staging/mrst-touchscreen/Kconfig deleted file mode 100644 index c2af49217084..000000000000 --- a/drivers/staging/mrst-touchscreen/Kconfig +++ /dev/null | |||
@@ -1,7 +0,0 @@ | |||
1 | config TOUCHSCREEN_INTEL_MID | ||
2 | tristate "Intel MID platform resistive touchscreen" | ||
3 | depends on INTEL_SCU_IPC | ||
4 | default y | ||
5 | help | ||
6 | Say Y here if you have a Intel MID based touchscreen | ||
7 | If unsure, say N. | ||
diff --git a/drivers/staging/mrst-touchscreen/Makefile b/drivers/staging/mrst-touchscreen/Makefile deleted file mode 100644 index 2d638b0d70bf..000000000000 --- a/drivers/staging/mrst-touchscreen/Makefile +++ /dev/null | |||
@@ -1,3 +0,0 @@ | |||
1 | obj-$(CONFIG_TOUCHSCREEN_INTEL_MID) := intel_mid_touch.o | ||
2 | |||
3 | |||
diff --git a/drivers/staging/mrst-touchscreen/TODO b/drivers/staging/mrst-touchscreen/TODO deleted file mode 100644 index 7157028d634a..000000000000 --- a/drivers/staging/mrst-touchscreen/TODO +++ /dev/null | |||
@@ -1,2 +0,0 @@ | |||
1 | - Move the driver to not think it is SPI (requires fixing some of the SFI | ||
2 | and firmware side) | ||
diff --git a/drivers/staging/mrst-touchscreen/intel-mid-touch.c b/drivers/staging/mrst-touchscreen/intel-mid-touch.c deleted file mode 100644 index abba22f921be..000000000000 --- a/drivers/staging/mrst-touchscreen/intel-mid-touch.c +++ /dev/null | |||
@@ -1,864 +0,0 @@ | |||
1 | /* | ||
2 | * intel_mid_touch.c - Intel MID Resistive Touch Screen Driver | ||
3 | * | ||
4 | * Copyright (C) 2008 Intel Corp | ||
5 | * | ||
6 | * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||
7 | * | ||
8 | * This program is free software; you can redistribute it and/or modify | ||
9 | * it under the terms of the GNU General Public License as published by | ||
10 | * the Free Software Foundation; version 2 of the License. | ||
11 | * | ||
12 | * This program is distributed in the hope that it will be useful, but | ||
13 | * WITHOUT ANY WARRANTY; without even the implied warranty of | ||
14 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | ||
15 | * General Public License for more details. | ||
16 | * | ||
17 | * You should have received a copy of the GNU General Public License along | ||
18 | * with this program; ifnot, write to the Free Software Foundation, Inc., | ||
19 | * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA. | ||
20 | * | ||
21 | * Questions/Comments/Bug fixes to Sreedhara (sreedhara.ds@intel.com) | ||
22 | * Ramesh Agarwal (ramesh.agarwal@intel.com) | ||
23 | * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||
24 | * | ||
25 | * TODO: | ||
26 | * kill off mrstouch_debug eventually | ||
27 | * review conversion of r/m/w sequences | ||
28 | * Replace interrupt mutex abuse | ||
29 | * Kill of mrstouchdevp pointer | ||
30 | * | ||
31 | */ | ||
32 | |||
33 | #include <linux/module.h> | ||
34 | #include <linux/init.h> | ||
35 | #include <linux/input.h> | ||
36 | #include <linux/interrupt.h> | ||
37 | #include <linux/err.h> | ||
38 | #include <linux/param.h> | ||
39 | #include <linux/spi/spi.h> | ||
40 | #include <linux/irq.h> | ||
41 | #include <linux/delay.h> | ||
42 | #include <linux/kthread.h> | ||
43 | #include <asm/intel_scu_ipc.h> | ||
44 | |||
45 | |||
46 | #if defined(MRSTOUCH_DEBUG) | ||
47 | #define mrstouch_debug(fmt, args...)\ | ||
48 | do { \ | ||
49 | printk(KERN_DEBUG "\n[MRSTOUCH(%d)] - ", __LINE__); \ | ||
50 | printk(KERN_DEBUG fmt, ##args); \ | ||
51 | } while (0); | ||
52 | #else | ||
53 | #define mrstouch_debug(fmt, args...) | ||
54 | #endif | ||
55 | |||
56 | /* PMIC Interrupt registers */ | ||
57 | #define PMIC_REG_ID1 0x00 /*PMIC ID1 register */ | ||
58 | |||
59 | /* PMIC Interrupt registers */ | ||
60 | #define PMIC_REG_INT 0x04 /*PMIC interrupt register */ | ||
61 | #define PMIC_REG_MINT 0x05 /*PMIC interrupt mask register */ | ||
62 | |||
63 | /* ADC Interrupt registers */ | ||
64 | #define PMIC_REG_ADCINT 0x5F /*ADC interrupt register */ | ||
65 | #define PMIC_REG_MADCINT 0x60 /*ADC interrupt mask register */ | ||
66 | |||
67 | /* ADC Control registers */ | ||
68 | #define PMIC_REG_ADCCNTL1 0x61 /*ADC control register */ | ||
69 | |||
70 | /* ADC Channel Selection registers */ | ||
71 | #define PMICADDR0 0xA4 | ||
72 | #define END_OF_CHANNEL 0x1F | ||
73 | |||
74 | /* ADC Result register */ | ||
75 | #define PMIC_REG_ADCSNS0H 0x64 | ||
76 | |||
77 | /* ADC channels for touch screen */ | ||
78 | #define MRST_TS_CHAN10 0xA /* Touch screen X+ connection */ | ||
79 | #define MRST_TS_CHAN11 0xB /* Touch screen X- connection */ | ||
80 | #define MRST_TS_CHAN12 0xC /* Touch screen Y+ connection */ | ||
81 | #define MRST_TS_CHAN13 0xD /* Touch screen Y- connection */ | ||
82 | |||
83 | /* Touch screen coordinate constants */ | ||
84 | #define TOUCH_PRESSURE 50 | ||
85 | #define TOUCH_PRESSURE_FS 100 | ||
86 | |||
87 | #define XMOVE_LIMIT 5 | ||
88 | #define YMOVE_LIMIT 5 | ||
89 | #define XYMOVE_CNT 3 | ||
90 | |||
91 | #define MAX_10BIT ((1<<10)-1) | ||
92 | |||
93 | /* Touch screen channel BIAS constants */ | ||
94 | #define XBIAS 0x20 | ||
95 | #define YBIAS 0x40 | ||
96 | #define ZBIAS 0x80 | ||
97 | |||
98 | /* Touch screen coordinates */ | ||
99 | #define MIN_X 10 | ||
100 | #define MAX_X 1024 | ||
101 | #define MIN_Y 10 | ||
102 | #define MAX_Y 1024 | ||
103 | #define WAIT_ADC_COMPLETION 10 | ||
104 | |||
105 | /* PMIC ADC round robin delays */ | ||
106 | #define ADC_LOOP_DELAY0 0x0 /* Continuous loop */ | ||
107 | #define ADC_LOOP_DELAY1 0x1 /* 4.5 ms approximate */ | ||
108 | |||
109 | /* PMIC Vendor Identifiers */ | ||
110 | #define PMIC_VENDOR_FS 0 /* PMIC vendor FreeScale */ | ||
111 | #define PMIC_VENDOR_MAXIM 1 /* PMIC vendor MAXIM */ | ||
112 | #define PMIC_VENDOR_NEC 2 /* PMIC vendor NEC */ | ||
113 | #define MRSTOUCH_MAX_CHANNELS 32 /* Maximum ADC channels */ | ||
114 | |||
115 | /* Touch screen device structure */ | ||
116 | struct mrstouch_dev { | ||
117 | struct spi_device *spi; /* SPI device associated with touch screen */ | ||
118 | struct input_dev *input; /* input device for touchscreen*/ | ||
119 | char phys[32]; /* Device name */ | ||
120 | struct task_struct *pendet_thrd; /* PENDET interrupt handler */ | ||
121 | struct mutex lock; /* Sync between interrupt and PENDET handler */ | ||
122 | bool busy; /* Busy flag */ | ||
123 | u16 asr; /* Address selection register */ | ||
124 | int irq; /* Touch screen IRQ # */ | ||
125 | uint vendor; /* PMIC vendor */ | ||
126 | uint rev; /* PMIC revision */ | ||
127 | bool suspended; /* Device suspended status */ | ||
128 | bool disabled; /* Device disabled status */ | ||
129 | u16 x; /* X coordinate */ | ||
130 | u16 y; /* Y coordinate */ | ||
131 | bool pendown; /* PEN position */ | ||
132 | } ; | ||
133 | |||
134 | |||
135 | /* Global Pointer to Touch screen device */ | ||
136 | static struct mrstouch_dev *mrstouchdevp; | ||
137 | |||
138 | /* Utility to read PMIC ID */ | ||
139 | static int mrstouch_pmic_id(uint *vendor, uint *rev) | ||
140 | { | ||
141 | int err; | ||
142 | u8 r; | ||
143 | |||
144 | err = intel_scu_ipc_ioread8(PMIC_REG_ID1, &r); | ||
145 | if (err) | ||
146 | return err; | ||
147 | |||
148 | *vendor = r & 0x7; | ||
149 | *rev = (r >> 3) & 0x7; | ||
150 | |||
151 | return 0; | ||
152 | } | ||
153 | |||
154 | /* | ||
155 | * Parse ADC channels to find end of the channel configured by other ADC user | ||
156 | * NEC and MAXIM requires 4 channels and FreeScale needs 18 channels | ||
157 | */ | ||
158 | static int mrstouch_chan_parse(struct mrstouch_dev *tsdev) | ||
159 | { | ||
160 | int err, i, j, found; | ||
161 | u32 r32; | ||
162 | |||
163 | found = -1; | ||
164 | |||
165 | for (i = 0; i < MRSTOUCH_MAX_CHANNELS; i++) { | ||
166 | if (found >= 0) | ||
167 | break; | ||
168 | |||
169 | err = intel_scu_ipc_ioread32(PMICADDR0, &r32); | ||
170 | if (err) | ||
171 | return err; | ||
172 | |||
173 | for (j = 0; j < 32; j+= 8) { | ||
174 | if (((r32 >> j) & 0xFF) == END_OF_CHANNEL) { | ||
175 | found = i; | ||
176 | break; | ||
177 | } | ||
178 | } | ||
179 | } | ||
180 | if (found < 0) | ||
181 | return 0; | ||
182 | |||
183 | if (tsdev->vendor == PMIC_VENDOR_FS) { | ||
184 | if (found && found > (MRSTOUCH_MAX_CHANNELS - 18)) | ||
185 | return -ENOSPC; | ||
186 | } else { | ||
187 | if (found && found > (MRSTOUCH_MAX_CHANNELS - 4)) | ||
188 | return -ENOSPC; | ||
189 | } | ||
190 | return found; | ||
191 | } | ||
192 | |||
193 | /* Utility to enable/disable pendet. | ||
194 | * pendet set to true enables PENDET interrupt | ||
195 | * pendet set to false disables PENDET interrupt | ||
196 | * Also clears RND mask bit | ||
197 | */ | ||
198 | static int pendet_enable(struct mrstouch_dev *tsdev, bool pendet) | ||
199 | { | ||
200 | u16 reg; | ||
201 | u8 r; | ||
202 | u8 pendet_enabled = 0; | ||
203 | int retry = 0; | ||
204 | int err; | ||
205 | |||
206 | err = intel_scu_ipc_ioread16(PMIC_REG_MADCINT, ®); | ||
207 | if (err) | ||
208 | return err; | ||
209 | |||
210 | if (pendet) { | ||
211 | reg &= ~0x0005; | ||
212 | reg |= 0x2000; /* Enable pendet */ | ||
213 | } else | ||
214 | reg &= 0xDFFF; /* Disable pendet */ | ||
215 | |||
216 | /* Set MADCINT and update ADCCNTL1 (next reg byte) */ | ||
217 | err = intel_scu_ipc_iowrite16(PMIC_REG_MADCINT, reg); | ||
218 | if (!pendet || err) | ||
219 | return err; | ||
220 | |||
221 | /* | ||
222 | * Sometimes even after the register write succeeds | ||
223 | * the PMIC register value is not updated. Retry few iterations | ||
224 | * to enable pendet. | ||
225 | */ | ||
226 | |||
227 | err = intel_scu_ipc_ioread8(PMIC_REG_ADCCNTL1, &r); | ||
228 | pendet_enabled = (r >> 5) & 0x01; | ||
229 | |||
230 | retry = 0; | ||
231 | while (!err && !pendet_enabled) { | ||
232 | retry++; | ||
233 | msleep(10); | ||
234 | err = intel_scu_ipc_iowrite8(PMIC_REG_ADCCNTL1, reg >> 8); | ||
235 | if (err) | ||
236 | break; | ||
237 | err = intel_scu_ipc_ioread8(PMIC_REG_ADCCNTL1, &r); | ||
238 | if (err == 0) | ||
239 | pendet_enabled = (r >> 5) & 0x01; | ||
240 | if (retry >= 10) { | ||
241 | dev_err(&tsdev->spi->dev, "Touch screen disabled.\n"); | ||
242 | return -EIO; | ||
243 | } | ||
244 | } | ||
245 | return 0; | ||
246 | } | ||
247 | |||
248 | /* To read PMIC ADC touch screen result | ||
249 | * Reads ADC storage registers for higher 7 and lower 3 bits | ||
250 | * converts the two readings to single value and turns off gain bit | ||
251 | */ | ||
252 | static int mrstouch_ts_chan_read(u16 offset, u16 chan, u16 *vp, u16 *vm) | ||
253 | { | ||
254 | int err; | ||
255 | u16 result; | ||
256 | u32 res; | ||
257 | |||
258 | result = PMIC_REG_ADCSNS0H + offset; | ||
259 | |||
260 | if (chan == MRST_TS_CHAN12) | ||
261 | result += 4; | ||
262 | |||
263 | err = intel_scu_ipc_ioread32(result, &res); | ||
264 | if (err) | ||
265 | return err; | ||
266 | |||
267 | /* Mash the bits up */ | ||
268 | |||
269 | *vp = (res & 0xFF) << 3; /* Highest 7 bits */ | ||
270 | *vp |= (res >> 8) & 0x07; /* Lower 3 bits */ | ||
271 | *vp &= 0x3FF; | ||
272 | |||
273 | res >>= 16; | ||
274 | |||
275 | *vm = (res & 0xFF) << 3; /* Highest 7 bits */ | ||
276 | *vm |= (res >> 8) & 0x07; /* Lower 3 bits */ | ||
277 | *vm &= 0x3FF; | ||
278 | |||
279 | return 0; | ||
280 | } | ||
281 | |||
282 | /* To configure touch screen channels | ||
283 | * Writes touch screen channels to ADC address selection registers | ||
284 | */ | ||
285 | static int mrstouch_ts_chan_set(uint offset) | ||
286 | { | ||
287 | int count; | ||
288 | u16 chan; | ||
289 | u16 reg[5]; | ||
290 | u8 data[5]; | ||
291 | |||
292 | chan = PMICADDR0 + offset; | ||
293 | for (count = 0; count <= 3; count++) { | ||
294 | reg[count] = chan++; | ||
295 | data[count] = MRST_TS_CHAN10 + count; | ||
296 | } | ||
297 | reg[count] = chan; | ||
298 | data[count] = END_OF_CHANNEL; | ||
299 | |||
300 | return intel_scu_ipc_writev(reg, data, 5); | ||
301 | } | ||
302 | |||
303 | /* Initialize ADC */ | ||
304 | static int mrstouch_adc_init(struct mrstouch_dev *tsdev) | ||
305 | { | ||
306 | int err, start; | ||
307 | u8 ra, rm; | ||
308 | |||
309 | err = mrstouch_pmic_id(&tsdev->vendor, &tsdev->rev); | ||
310 | if (err) { | ||
311 | dev_err(&tsdev->spi->dev, "Unable to read PMIC id\n"); | ||
312 | return err; | ||
313 | } | ||
314 | |||
315 | start = mrstouch_chan_parse(tsdev); | ||
316 | if (start < 0) { | ||
317 | dev_err(&tsdev->spi->dev, "Unable to parse channels\n"); | ||
318 | return start; | ||
319 | } | ||
320 | |||
321 | tsdev->asr = start; | ||
322 | |||
323 | mrstouch_debug("Channel offset(%d): 0x%X\n", tsdev->asr, tsdev->vendor); | ||
324 | |||
325 | /* ADC power on, start, enable PENDET and set loop delay | ||
326 | * ADC loop delay is set to 4.5 ms approximately | ||
327 | * Loop delay more than this results in jitter in adc readings | ||
328 | * Setting loop delay to 0 (continous loop) in MAXIM stops PENDET | ||
329 | * interrupt generation sometimes. | ||
330 | */ | ||
331 | |||
332 | if (tsdev->vendor == PMIC_VENDOR_FS) { | ||
333 | ra = 0xE0 | ADC_LOOP_DELAY0; | ||
334 | rm = 0x5; | ||
335 | } else { | ||
336 | /* NEC and MAXIm not consistent with loop delay 0 */ | ||
337 | ra = 0xE0 | ADC_LOOP_DELAY1; | ||
338 | rm = 0x0; | ||
339 | |||
340 | /* configure touch screen channels */ | ||
341 | err = mrstouch_ts_chan_set(tsdev->asr); | ||
342 | if (err) | ||
343 | return err; | ||
344 | } | ||
345 | err = intel_scu_ipc_update_register(PMIC_REG_ADCCNTL1, ra, 0xE7); | ||
346 | if (err == 0) | ||
347 | err = intel_scu_ipc_update_register(PMIC_REG_MADCINT, rm, 0x03); | ||
348 | return err; | ||
349 | } | ||
350 | |||
351 | /* Reports x,y coordinates to event subsystem */ | ||
352 | static void mrstouch_report_xy(struct mrstouch_dev *tsdev, u16 x, u16 y, u16 z) | ||
353 | { | ||
354 | int xdiff, ydiff; | ||
355 | |||
356 | if (tsdev->pendown && z <= TOUCH_PRESSURE) { | ||
357 | /* Pen removed, report button release */ | ||
358 | mrstouch_debug("BTN REL(%d)", z); | ||
359 | input_report_key(tsdev->input, BTN_TOUCH, 0); | ||
360 | tsdev->pendown = false; | ||
361 | } | ||
362 | |||
363 | xdiff = abs(x - tsdev->x); | ||
364 | ydiff = abs(y - tsdev->y); | ||
365 | |||
366 | /* | ||
367 | if x and y values changes for XYMOVE_CNT readings it is considered | ||
368 | as stylus is moving. This is required to differentiate between stylus | ||
369 | movement and jitter | ||
370 | */ | ||
371 | if (x < MIN_X || x > MAX_X || y < MIN_Y || y > MAX_Y) { | ||
372 | /* Spurious values, release button if touched and return */ | ||
373 | if (tsdev->pendown) { | ||
374 | mrstouch_debug("BTN REL(%d)", z); | ||
375 | input_report_key(tsdev->input, BTN_TOUCH, 0); | ||
376 | tsdev->pendown = false; | ||
377 | } | ||
378 | return; | ||
379 | } else if (xdiff >= XMOVE_LIMIT || ydiff >= YMOVE_LIMIT) { | ||
380 | tsdev->x = x; | ||
381 | tsdev->y = y; | ||
382 | |||
383 | input_report_abs(tsdev->input, ABS_X, x); | ||
384 | input_report_abs(tsdev->input, ABS_Y, y); | ||
385 | input_sync(tsdev->input); | ||
386 | } | ||
387 | |||
388 | |||
389 | if (!tsdev->pendown && z > TOUCH_PRESSURE) { | ||
390 | /* Pen touched, report button touch */ | ||
391 | mrstouch_debug("BTN TCH(%d, %d, %d)", x, y, z); | ||
392 | input_report_key(tsdev->input, BTN_TOUCH, 1); | ||
393 | tsdev->pendown = true; | ||
394 | } | ||
395 | } | ||
396 | |||
397 | |||
398 | /* Utility to start ADC, used by freescale handler */ | ||
399 | static int pendet_mask(void) | ||
400 | { | ||
401 | return intel_scu_ipc_update_register(PMIC_REG_MADCINT, 0x02, 0x02); | ||
402 | } | ||
403 | |||
404 | /* Utility to stop ADC, used by freescale handler */ | ||
405 | static int pendet_umask(void) | ||
406 | { | ||
407 | return intel_scu_ipc_update_register(PMIC_REG_MADCINT, 0x00, 0x02); | ||
408 | } | ||
409 | |||
410 | /* Utility to read ADC, used by freescale handler */ | ||
411 | static int mrstouch_pmic_fs_adc_read(struct mrstouch_dev *tsdev) | ||
412 | { | ||
413 | int err; | ||
414 | u16 x, y, z, result; | ||
415 | u16 reg[4]; | ||
416 | u8 data[4]; | ||
417 | |||
418 | result = PMIC_REG_ADCSNS0H + tsdev->asr; | ||
419 | |||
420 | reg[0] = result + 4; | ||
421 | reg[1] = result + 5; | ||
422 | reg[2] = result + 16; | ||
423 | reg[3] = result + 17; | ||
424 | |||
425 | err = intel_scu_ipc_readv(reg, data, 4); | ||
426 | if (err) | ||
427 | goto ipc_error; | ||
428 | |||
429 | x = data[0] << 3; /* Higher 7 bits */ | ||
430 | x |= data[1] & 0x7; /* Lower 3 bits */ | ||
431 | x &= 0x3FF; | ||
432 | |||
433 | y = data[2] << 3; /* Higher 7 bits */ | ||
434 | y |= data[3] & 0x7; /* Lower 3 bits */ | ||
435 | y &= 0x3FF; | ||
436 | |||
437 | /* Read Z value */ | ||
438 | reg[0] = result + 28; | ||
439 | reg[1] = result + 29; | ||
440 | |||
441 | err = intel_scu_ipc_readv(reg, data, 4); | ||
442 | if (err) | ||
443 | goto ipc_error; | ||
444 | |||
445 | z = data[0] << 3; /* Higher 7 bits */ | ||
446 | z |= data[1] & 0x7; /* Lower 3 bits */ | ||
447 | z &= 0x3FF; | ||
448 | |||
449 | #if defined(MRSTOUCH_PRINT_XYZP) | ||
450 | mrstouch_debug("X: %d, Y: %d, Z: %d", x, y, z); | ||
451 | #endif | ||
452 | |||
453 | if (z >= TOUCH_PRESSURE_FS) { | ||
454 | mrstouch_report_xy(tsdev, x, y, TOUCH_PRESSURE - 1); /* Pen Removed */ | ||
455 | return TOUCH_PRESSURE - 1; | ||
456 | } else { | ||
457 | mrstouch_report_xy(tsdev, x, y, TOUCH_PRESSURE + 1); /* Pen Touched */ | ||
458 | return TOUCH_PRESSURE + 1; | ||
459 | } | ||
460 | |||
461 | return 0; | ||
462 | |||
463 | ipc_error: | ||
464 | dev_err(&tsdev->spi->dev, "ipc error during fs_adc read\n"); | ||
465 | return err; | ||
466 | } | ||
467 | |||
468 | /* To handle free scale pmic pendet interrupt */ | ||
469 | static int pmic0_pendet(void *dev_id) | ||
470 | { | ||
471 | int err, count; | ||
472 | u16 chan; | ||
473 | unsigned int touched; | ||
474 | struct mrstouch_dev *tsdev = (struct mrstouch_dev *)dev_id; | ||
475 | u16 reg[5]; | ||
476 | u8 data[5]; | ||
477 | |||
478 | chan = PMICADDR0 + tsdev->asr; | ||
479 | |||
480 | /* Set X BIAS */ | ||
481 | for (count = 0; count <= 3; count++) { | ||
482 | reg[count] = chan++; | ||
483 | data[count] = 0x2A; | ||
484 | } | ||
485 | reg[count] = chan++; /* Dummy */ | ||
486 | data[count] = 0; | ||
487 | |||
488 | err = intel_scu_ipc_writev(reg, data, 5); | ||
489 | if (err) | ||
490 | goto ipc_error; | ||
491 | |||
492 | msleep(WAIT_ADC_COMPLETION); | ||
493 | |||
494 | /* Set Y BIAS */ | ||
495 | for (count = 0; count <= 3; count++) { | ||
496 | reg[count] = chan++; | ||
497 | data[count] = 0x4A; | ||
498 | } | ||
499 | reg[count] = chan++; /* Dummy */ | ||
500 | data[count] = 0; | ||
501 | |||
502 | err = intel_scu_ipc_writev(reg, data, 5); | ||
503 | if (err) | ||
504 | goto ipc_error; | ||
505 | |||
506 | msleep(WAIT_ADC_COMPLETION); | ||
507 | |||
508 | /* Set Z BIAS */ | ||
509 | err = intel_scu_ipc_iowrite32(chan + 2, 0x8A8A8A8A); | ||
510 | if (err) | ||
511 | goto ipc_error; | ||
512 | |||
513 | msleep(WAIT_ADC_COMPLETION); | ||
514 | |||
515 | /*Read touch screen channels till pen removed | ||
516 | * Freescale reports constant value of z for all points | ||
517 | * z is high when screen is not touched and low when touched | ||
518 | * Map high z value to not touched and low z value to pen touched | ||
519 | */ | ||
520 | touched = mrstouch_pmic_fs_adc_read(tsdev); | ||
521 | while (touched > TOUCH_PRESSURE) { | ||
522 | touched = mrstouch_pmic_fs_adc_read(tsdev); | ||
523 | msleep(WAIT_ADC_COMPLETION); | ||
524 | } | ||
525 | |||
526 | /* Clear all TS channels */ | ||
527 | chan = PMICADDR0 + tsdev->asr; | ||
528 | for (count = 0; count <= 4; count++) { | ||
529 | reg[count] = chan++; | ||
530 | data[count] = 0; | ||
531 | } | ||
532 | err = intel_scu_ipc_writev(reg, data, 5); | ||
533 | if (err) | ||
534 | goto ipc_error; | ||
535 | |||
536 | for (count = 0; count <= 4; count++) { | ||
537 | reg[count] = chan++; | ||
538 | data[count] = 0; | ||
539 | } | ||
540 | err = intel_scu_ipc_writev(reg, data, 5); | ||
541 | if (err) | ||
542 | goto ipc_error; | ||
543 | |||
544 | err = intel_scu_ipc_iowrite32(chan + 2, 0x00000000); | ||
545 | if (err) | ||
546 | goto ipc_error; | ||
547 | |||
548 | return 0; | ||
549 | |||
550 | ipc_error: | ||
551 | dev_err(&tsdev->spi->dev, "ipc error during pendet\n"); | ||
552 | return err; | ||
553 | } | ||
554 | |||
555 | |||
556 | /* To enable X, Y and Z bias values | ||
557 | * Enables YPYM for X channels and XPXM for Y channels | ||
558 | */ | ||
559 | static int mrstouch_ts_bias_set(uint offset, uint bias) | ||
560 | { | ||
561 | int count; | ||
562 | u16 chan, start; | ||
563 | u16 reg[4]; | ||
564 | u8 data[4]; | ||
565 | |||
566 | chan = PMICADDR0 + offset; | ||
567 | start = MRST_TS_CHAN10; | ||
568 | |||
569 | for (count = 0; count <= 3; count++) { | ||
570 | reg[count] = chan++; | ||
571 | data[count] = bias | (start + count); | ||
572 | } | ||
573 | return intel_scu_ipc_writev(reg, data, 4); | ||
574 | } | ||
575 | |||
576 | /* To read touch screen channel values */ | ||
577 | static int mrstouch_adc_read(struct mrstouch_dev *tsdev) | ||
578 | { | ||
579 | int err; | ||
580 | u16 xp, xm, yp, ym, zp, zm; | ||
581 | |||
582 | /* configure Y bias for X channels */ | ||
583 | err = mrstouch_ts_bias_set(tsdev->asr, YBIAS); | ||
584 | if (err) | ||
585 | goto ipc_error; | ||
586 | |||
587 | msleep(WAIT_ADC_COMPLETION); | ||
588 | |||
589 | /* read x+ and x- channels */ | ||
590 | err = mrstouch_ts_chan_read(tsdev->asr, MRST_TS_CHAN10, &xp, &xm); | ||
591 | if (err) | ||
592 | goto ipc_error; | ||
593 | |||
594 | /* configure x bias for y channels */ | ||
595 | err = mrstouch_ts_bias_set(tsdev->asr, XBIAS); | ||
596 | if (err) | ||
597 | goto ipc_error; | ||
598 | |||
599 | msleep(WAIT_ADC_COMPLETION); | ||
600 | |||
601 | /* read y+ and y- channels */ | ||
602 | err = mrstouch_ts_chan_read(tsdev->asr, MRST_TS_CHAN12, &yp, &ym); | ||
603 | if (err) | ||
604 | goto ipc_error; | ||
605 | |||
606 | /* configure z bias for x and y channels */ | ||
607 | err = mrstouch_ts_bias_set(tsdev->asr, ZBIAS); | ||
608 | if (err) | ||
609 | goto ipc_error; | ||
610 | |||
611 | msleep(WAIT_ADC_COMPLETION); | ||
612 | |||
613 | /* read z+ and z- channels */ | ||
614 | err = mrstouch_ts_chan_read(tsdev->asr, MRST_TS_CHAN10, &zp, &zm); | ||
615 | if (err) | ||
616 | goto ipc_error; | ||
617 | |||
618 | #if defined(MRSTOUCH_PRINT_XYZP) | ||
619 | printk(KERN_INFO "X+: %d, Y+: %d, Z+: %d\n", xp, yp, zp); | ||
620 | #endif | ||
621 | |||
622 | #if defined(MRSTOUCH_PRINT_XYZM) | ||
623 | printk(KERN_INFO "X-: %d, Y-: %d, Z-: %d\n", xm, ym, zm); | ||
624 | #endif | ||
625 | |||
626 | mrstouch_report_xy(tsdev, xp, yp, zp); /* report x and y to eventX */ | ||
627 | |||
628 | return zp; | ||
629 | |||
630 | ipc_error: | ||
631 | dev_err(&tsdev->spi->dev, "ipc error during adc read\n"); | ||
632 | return err; | ||
633 | } | ||
634 | |||
635 | /* PENDET interrupt handler function for NEC and MAXIM */ | ||
636 | static void pmic12_pendet(void *data) | ||
637 | { | ||
638 | unsigned int touched; | ||
639 | struct mrstouch_dev *tsdev = (struct mrstouch_dev *)data; | ||
640 | |||
641 | /* read touch screen channels till pen removed */ | ||
642 | do { | ||
643 | touched = mrstouch_adc_read(tsdev); | ||
644 | } while (touched > TOUCH_PRESSURE); | ||
645 | } | ||
646 | |||
647 | /* Handler to process PENDET interrupt */ | ||
648 | int mrstouch_pendet(void *data) | ||
649 | { | ||
650 | struct mrstouch_dev *tsdev = (struct mrstouch_dev *)data; | ||
651 | while (1) { | ||
652 | /* Wait for PENDET interrupt */ | ||
653 | if (mutex_lock_interruptible(&tsdev->lock)) { | ||
654 | msleep(WAIT_ADC_COMPLETION); | ||
655 | continue; | ||
656 | } | ||
657 | |||
658 | if (tsdev->busy) | ||
659 | return 0; | ||
660 | |||
661 | tsdev->busy = true; | ||
662 | |||
663 | if (tsdev->vendor == PMIC_VENDOR_NEC || | ||
664 | tsdev->vendor == PMIC_VENDOR_MAXIM) { | ||
665 | /* PENDET must be disabled in NEC before reading ADC */ | ||
666 | pendet_enable(tsdev,false); /* Disbale PENDET */ | ||
667 | pmic12_pendet(tsdev); | ||
668 | pendet_enable(tsdev, true); /*Enable PENDET */ | ||
669 | } else if (tsdev->vendor == PMIC_VENDOR_FS) { | ||
670 | pendet_umask(); /* Stop ADC */ | ||
671 | pmic0_pendet(tsdev); | ||
672 | pendet_mask(); /* Stop ADC */ | ||
673 | } else | ||
674 | dev_err(&tsdev->spi->dev, "Unsupported touchscreen: %d\n", | ||
675 | tsdev->vendor); | ||
676 | |||
677 | tsdev->busy = false; | ||
678 | |||
679 | } | ||
680 | return 0; | ||
681 | } | ||
682 | |||
683 | /* PENDET interrupt handler */ | ||
684 | static irqreturn_t pendet_intr_handler(int irq, void *handle) | ||
685 | { | ||
686 | struct mrstouch_dev *tsdev = (struct mrstouch_dev *)handle; | ||
687 | |||
688 | mutex_unlock(&tsdev->lock); | ||
689 | return IRQ_HANDLED; | ||
690 | } | ||
691 | |||
692 | /* Intializes input device and registers with input subsystem */ | ||
693 | static int ts_input_dev_init(struct mrstouch_dev *tsdev, struct spi_device *spi) | ||
694 | { | ||
695 | int err = 0; | ||
696 | |||
697 | mrstouch_debug("%s", __func__); | ||
698 | |||
699 | tsdev->input = input_allocate_device(); | ||
700 | if (!tsdev->input) { | ||
701 | dev_err(&tsdev->spi->dev, "Unable to allocate input device.\n"); | ||
702 | return -EINVAL; | ||
703 | } | ||
704 | |||
705 | tsdev->input->name = "mrst_touchscreen"; | ||
706 | snprintf(tsdev->phys, sizeof(tsdev->phys), | ||
707 | "%s/input0", dev_name(&spi->dev)); | ||
708 | tsdev->input->phys = tsdev->phys; | ||
709 | tsdev->input->dev.parent = &spi->dev; | ||
710 | |||
711 | tsdev->input->id.vendor = tsdev->vendor; | ||
712 | tsdev->input->id.version = tsdev->rev; | ||
713 | |||
714 | tsdev->input->evbit[0] = BIT_MASK(EV_KEY) | BIT_MASK(EV_ABS); | ||
715 | tsdev->input->keybit[BIT_WORD(BTN_TOUCH)] = BIT_MASK(BTN_TOUCH); | ||
716 | |||
717 | input_set_abs_params(tsdev->input, ABS_X, MIN_X, MIN_Y, 0, 0); | ||
718 | input_set_abs_params(tsdev->input, ABS_Y, MIN_X, MIN_Y, 0, 0); | ||
719 | |||
720 | err = input_register_device(tsdev->input); | ||
721 | if (err) { | ||
722 | dev_err(&tsdev->spi->dev, "unable to register input device\n"); | ||
723 | input_free_device(tsdev->input); | ||
724 | return err; | ||
725 | } | ||
726 | |||
727 | mrstouch_debug("%s", "mrstouch initialized"); | ||
728 | |||
729 | return 0; | ||
730 | |||
731 | } | ||
732 | |||
733 | /* Probe function for touch screen driver */ | ||
734 | static int __devinit mrstouch_probe(struct spi_device *mrstouch_spi) | ||
735 | { | ||
736 | int err; | ||
737 | unsigned int myirq; | ||
738 | struct mrstouch_dev *tsdev; | ||
739 | |||
740 | mrstouch_debug("%s(%p)", __func__, mrstouch_spi); | ||
741 | |||
742 | mrstouchdevp = NULL; | ||
743 | myirq = mrstouch_spi->irq; | ||
744 | |||
745 | if (!mrstouch_spi->irq) { | ||
746 | dev_err(&mrstouch_spi->dev, "no interrupt assigned\n"); | ||
747 | return -EINVAL; | ||
748 | } | ||
749 | |||
750 | tsdev = kzalloc(sizeof(struct mrstouch_dev), GFP_KERNEL); | ||
751 | if (!tsdev) { | ||
752 | dev_err(&mrstouch_spi->dev, "unable to allocate memory\n"); | ||
753 | return -ENOMEM; | ||
754 | } | ||
755 | |||
756 | tsdev->irq = myirq; | ||
757 | mrstouchdevp = tsdev; | ||
758 | |||
759 | err = mrstouch_adc_init(tsdev); | ||
760 | if (err) { | ||
761 | dev_err(&mrstouch_spi->dev, "ADC init failed\n"); | ||
762 | goto mrstouch_err_free_mem; | ||
763 | } | ||
764 | |||
765 | dev_set_drvdata(&mrstouch_spi->dev, tsdev); | ||
766 | tsdev->spi = mrstouch_spi; | ||
767 | |||
768 | err = ts_input_dev_init(tsdev, mrstouch_spi); | ||
769 | if (err) { | ||
770 | dev_err(&tsdev->spi->dev, "ts_input_dev_init failed"); | ||
771 | goto mrstouch_err_free_mem; | ||
772 | } | ||
773 | |||
774 | mutex_init(&tsdev->lock); | ||
775 | mutex_lock(&tsdev->lock) | ||
776 | |||
777 | mrstouch_debug("Requesting IRQ-%d", myirq); | ||
778 | err = request_irq(myirq, pendet_intr_handler, | ||
779 | 0, "mrstouch", tsdev); | ||
780 | if (err) { | ||
781 | dev_err(&tsdev->spi->dev, "unable to allocate irq\n"); | ||
782 | goto mrstouch_err_free_mem; | ||
783 | } | ||
784 | |||
785 | tsdev->pendet_thrd = kthread_run(mrstouch_pendet, | ||
786 | (void *)tsdev, "pendet handler"); | ||
787 | if (IS_ERR(tsdev->pendet_thrd)) { | ||
788 | dev_err(&tsdev->spi->dev, "kthread_run failed\n"); | ||
789 | err = PTR_ERR(tsdev->pendet_thrd); | ||
790 | goto mrstouch_err_free_mem; | ||
791 | } | ||
792 | mrstouch_debug("%s", "Driver initialized"); | ||
793 | return 0; | ||
794 | |||
795 | mrstouch_err_free_mem: | ||
796 | kfree(tsdev); | ||
797 | return err; | ||
798 | } | ||
799 | |||
800 | static int mrstouch_suspend(struct spi_device *spi, pm_message_t msg) | ||
801 | { | ||
802 | mrstouch_debug("%s", __func__); | ||
803 | mrstouchdevp->suspended = 1; | ||
804 | return 0; | ||
805 | } | ||
806 | |||
807 | static int mrstouch_resume(struct spi_device *spi) | ||
808 | { | ||
809 | mrstouch_debug("%s", __func__); | ||
810 | mrstouchdevp->suspended = 0; | ||
811 | return 0; | ||
812 | } | ||
813 | |||
814 | static int mrstouch_remove(struct spi_device *spi) | ||
815 | { | ||
816 | mrstouch_debug("%s", __func__); | ||
817 | free_irq(mrstouchdevp->irq, mrstouchdevp); | ||
818 | input_unregister_device(mrstouchdevp->input); | ||
819 | input_free_device(mrstouchdevp->input); | ||
820 | if (mrstouchdevp->pendet_thrd) | ||
821 | kthread_stop(mrstouchdevp->pendet_thrd); | ||
822 | kfree(mrstouchdevp); | ||
823 | return 0; | ||
824 | } | ||
825 | |||
826 | static struct spi_driver mrstouch_driver = { | ||
827 | .driver = { | ||
828 | .name = "pmic_touch", | ||
829 | .bus = &spi_bus_type, | ||
830 | .owner = THIS_MODULE, | ||
831 | }, | ||
832 | .probe = mrstouch_probe, | ||
833 | .suspend = mrstouch_suspend, | ||
834 | .resume = mrstouch_resume, | ||
835 | .remove = mrstouch_remove, | ||
836 | }; | ||
837 | |||
838 | static int __init mrstouch_module_init(void) | ||
839 | { | ||
840 | int err; | ||
841 | |||
842 | mrstouch_debug("%s", __func__); | ||
843 | err = spi_register_driver(&mrstouch_driver); | ||
844 | if (err) { | ||
845 | mrstouch_debug("%s(%d)", "SPI PENDET failed", err); | ||
846 | return -1; | ||
847 | } | ||
848 | |||
849 | return 0; | ||
850 | } | ||
851 | |||
852 | static void __exit mrstouch_module_exit(void) | ||
853 | { | ||
854 | mrstouch_debug("%s", __func__); | ||
855 | spi_unregister_driver(&mrstouch_driver); | ||
856 | return; | ||
857 | } | ||
858 | |||
859 | module_init(mrstouch_module_init); | ||
860 | module_exit(mrstouch_module_exit); | ||
861 | |||
862 | MODULE_AUTHOR("Sreedhara Murthy. D.S, sreedhara.ds@intel.com"); | ||
863 | MODULE_DESCRIPTION("Intel Moorestown Resistive Touch Screen Driver"); | ||
864 | MODULE_LICENSE("GPL"); | ||