xref: /openbmc/linux/drivers/input/keyboard/tc3589x-keypad.c (revision df2634f43f5106947f3735a0b61a6527a4b278cd)
1 /*
2  * Copyright (C) ST-Ericsson SA 2010
3  *
4  * Author: Jayeeta Banerjee <jayeeta.banerjee@stericsson.com>
5  * Author: Sundar Iyer <sundar.iyer@stericsson.com>
6  *
7  * License Terms: GNU General Public License, version 2
8  *
9  * TC35893 MFD Keypad Controller driver
10  */
11 
12 #include <linux/module.h>
13 #include <linux/init.h>
14 #include <linux/interrupt.h>
15 #include <linux/input.h>
16 #include <linux/platform_device.h>
17 #include <linux/input/matrix_keypad.h>
18 #include <linux/i2c.h>
19 #include <linux/slab.h>
20 #include <linux/mfd/tc3589x.h>
21 
22 /* Maximum supported keypad matrix row/columns size */
23 #define TC3589x_MAX_KPROW               8
24 #define TC3589x_MAX_KPCOL               12
25 
26 /* keypad related Constants */
27 #define TC3589x_MAX_DEBOUNCE_SETTLE     0xFF
28 #define DEDICATED_KEY_VAL		0xFF
29 
30 /* Pull up/down masks */
31 #define TC3589x_NO_PULL_MASK		0x0
32 #define TC3589x_PULL_DOWN_MASK		0x1
33 #define TC3589x_PULL_UP_MASK		0x2
34 #define TC3589x_PULLUP_ALL_MASK		0xAA
35 #define TC3589x_IO_PULL_VAL(index, mask)	((mask)<<((index)%4)*2))
36 
37 /* Bit masks for IOCFG register */
38 #define IOCFG_BALLCFG		0x01
39 #define IOCFG_IG		0x08
40 
41 #define KP_EVCODE_COL_MASK	0x0F
42 #define KP_EVCODE_ROW_MASK	0x70
43 #define KP_RELEASE_EVT_MASK	0x80
44 
45 #define KP_ROW_SHIFT		4
46 
47 #define KP_NO_VALID_KEY_MASK	0x7F
48 
49 /* bit masks for RESTCTRL register */
50 #define TC3589x_KBDRST		0x2
51 #define TC3589x_IRQRST		0x10
52 #define TC3589x_RESET_ALL	0x1B
53 
54 /* KBDMFS register bit mask */
55 #define TC3589x_KBDMFS_EN	0x1
56 
57 /* CLKEN register bitmask */
58 #define KPD_CLK_EN		0x1
59 
60 /* RSTINTCLR register bit mask */
61 #define IRQ_CLEAR		0x1
62 
63 /* bit masks for keyboard interrupts*/
64 #define TC3589x_EVT_LOSS_INT	0x8
65 #define TC3589x_EVT_INT		0x4
66 #define TC3589x_KBD_LOSS_INT	0x2
67 #define TC3589x_KBD_INT		0x1
68 
69 /* bit masks for keyboard interrupt clear*/
70 #define TC3589x_EVT_INT_CLR	0x2
71 #define TC3589x_KBD_INT_CLR	0x1
72 
73 #define TC3589x_KBD_KEYMAP_SIZE     64
74 
75 /**
76  * struct tc_keypad - data structure used by keypad driver
77  * @input:      pointer to input device object
78  * @board:      keypad platform device
79  * @krow:	number of rows
80  * @kcol:	number of coloumns
81  * @keymap:     matrix scan code table for keycodes
82  */
83 struct tc_keypad {
84 	struct tc3589x *tc3589x;
85 	struct input_dev *input;
86 	const struct tc3589x_keypad_platform_data *board;
87 	unsigned int krow;
88 	unsigned int kcol;
89 	unsigned short keymap[TC3589x_KBD_KEYMAP_SIZE];
90 	bool keypad_stopped;
91 };
92 
93 static int __devinit tc3589x_keypad_init_key_hardware(struct tc_keypad *keypad)
94 {
95 	int ret;
96 	struct tc3589x *tc3589x = keypad->tc3589x;
97 	u8 settle_time = keypad->board->settle_time;
98 	u8 dbounce_period = keypad->board->debounce_period;
99 	u8 rows = keypad->board->krow & 0xf;	/* mask out the nibble */
100 	u8 column = keypad->board->kcol & 0xf;	/* mask out the nibble */
101 
102 	/* validate platform configurations */
103 	if (keypad->board->kcol > TC3589x_MAX_KPCOL ||
104 	    keypad->board->krow > TC3589x_MAX_KPROW ||
105 	    keypad->board->debounce_period > TC3589x_MAX_DEBOUNCE_SETTLE ||
106 	    keypad->board->settle_time > TC3589x_MAX_DEBOUNCE_SETTLE)
107 		return -EINVAL;
108 
109 	/* configure KBDSIZE 4 LSbits for cols and 4 MSbits for rows */
110 	ret = tc3589x_reg_write(tc3589x, TC3589x_KBDSIZE,
111 			(rows << KP_ROW_SHIFT) | column);
112 	if (ret < 0)
113 		return ret;
114 
115 	/* configure dedicated key config, no dedicated key selected */
116 	ret = tc3589x_reg_write(tc3589x, TC3589x_KBCFG_LSB, DEDICATED_KEY_VAL);
117 	if (ret < 0)
118 		return ret;
119 
120 	ret = tc3589x_reg_write(tc3589x, TC3589x_KBCFG_MSB, DEDICATED_KEY_VAL);
121 	if (ret < 0)
122 		return ret;
123 
124 	/* Configure settle time */
125 	ret = tc3589x_reg_write(tc3589x, TC3589x_KBDSETTLE_REG, settle_time);
126 	if (ret < 0)
127 		return ret;
128 
129 	/* Configure debounce time */
130 	ret = tc3589x_reg_write(tc3589x, TC3589x_KBDBOUNCE, dbounce_period);
131 	if (ret < 0)
132 		return ret;
133 
134 	/* Start of initialise keypad GPIOs */
135 	ret = tc3589x_set_bits(tc3589x, TC3589x_IOCFG, 0x0, IOCFG_IG);
136 	if (ret < 0)
137 		return ret;
138 
139 	/* Configure pull-up resistors for all row GPIOs */
140 	ret = tc3589x_reg_write(tc3589x, TC3589x_IOPULLCFG0_LSB,
141 					TC3589x_PULLUP_ALL_MASK);
142 	if (ret < 0)
143 		return ret;
144 
145 	ret = tc3589x_reg_write(tc3589x, TC3589x_IOPULLCFG0_MSB,
146 					TC3589x_PULLUP_ALL_MASK);
147 	if (ret < 0)
148 		return ret;
149 
150 	/* Configure pull-up resistors for all column GPIOs */
151 	ret = tc3589x_reg_write(tc3589x, TC3589x_IOPULLCFG1_LSB,
152 			TC3589x_PULLUP_ALL_MASK);
153 	if (ret < 0)
154 		return ret;
155 
156 	ret = tc3589x_reg_write(tc3589x, TC3589x_IOPULLCFG1_MSB,
157 			TC3589x_PULLUP_ALL_MASK);
158 	if (ret < 0)
159 		return ret;
160 
161 	ret = tc3589x_reg_write(tc3589x, TC3589x_IOPULLCFG2_LSB,
162 			TC3589x_PULLUP_ALL_MASK);
163 
164 	return ret;
165 }
166 
167 #define TC35893_DATA_REGS		4
168 #define TC35893_KEYCODE_FIFO_EMPTY	0x7f
169 #define TC35893_KEYCODE_FIFO_CLEAR	0xff
170 #define TC35893_KEYPAD_ROW_SHIFT	0x3
171 
172 static irqreturn_t tc3589x_keypad_irq(int irq, void *dev)
173 {
174 	struct tc_keypad *keypad = dev;
175 	struct tc3589x *tc3589x = keypad->tc3589x;
176 	u8 i, row_index, col_index, kbd_code, up;
177 	u8 code;
178 
179 	for (i = 0; i < TC35893_DATA_REGS * 2; i++) {
180 		kbd_code = tc3589x_reg_read(tc3589x, TC3589x_EVTCODE_FIFO);
181 
182 		/* loop till fifo is empty and no more keys are pressed */
183 		if (kbd_code == TC35893_KEYCODE_FIFO_EMPTY ||
184 				kbd_code == TC35893_KEYCODE_FIFO_CLEAR)
185 			continue;
186 
187 		/* valid key is found */
188 		col_index = kbd_code & KP_EVCODE_COL_MASK;
189 		row_index = (kbd_code & KP_EVCODE_ROW_MASK) >> KP_ROW_SHIFT;
190 		code = MATRIX_SCAN_CODE(row_index, col_index,
191 						TC35893_KEYPAD_ROW_SHIFT);
192 		up = kbd_code & KP_RELEASE_EVT_MASK;
193 
194 		input_event(keypad->input, EV_MSC, MSC_SCAN, code);
195 		input_report_key(keypad->input, keypad->keymap[code], !up);
196 		input_sync(keypad->input);
197 	}
198 
199 	/* clear IRQ */
200 	tc3589x_set_bits(tc3589x, TC3589x_KBDIC,
201 			0x0, TC3589x_EVT_INT_CLR | TC3589x_KBD_INT_CLR);
202 	/* enable IRQ */
203 	tc3589x_set_bits(tc3589x, TC3589x_KBDMSK,
204 			0x0, TC3589x_EVT_LOSS_INT | TC3589x_EVT_INT);
205 
206 	return IRQ_HANDLED;
207 }
208 
209 static int tc3589x_keypad_enable(struct tc_keypad *keypad)
210 {
211 	struct tc3589x *tc3589x = keypad->tc3589x;
212 	int ret;
213 
214 	/* pull the keypad module out of reset */
215 	ret = tc3589x_set_bits(tc3589x, TC3589x_RSTCTRL, TC3589x_KBDRST, 0x0);
216 	if (ret < 0)
217 		return ret;
218 
219 	/* configure KBDMFS */
220 	ret = tc3589x_set_bits(tc3589x, TC3589x_KBDMFS, 0x0, TC3589x_KBDMFS_EN);
221 	if (ret < 0)
222 		return ret;
223 
224 	/* enable the keypad clock */
225 	ret = tc3589x_set_bits(tc3589x, TC3589x_CLKEN, 0x0, KPD_CLK_EN);
226 	if (ret < 0)
227 		return ret;
228 
229 	/* clear pending IRQs */
230 	ret =  tc3589x_set_bits(tc3589x, TC3589x_RSTINTCLR, 0x0, 0x1);
231 	if (ret < 0)
232 		return ret;
233 
234 	/* enable the IRQs */
235 	ret = tc3589x_set_bits(tc3589x, TC3589x_KBDMSK, 0x0,
236 					TC3589x_EVT_LOSS_INT | TC3589x_EVT_INT);
237 	if (ret < 0)
238 		return ret;
239 
240 	keypad->keypad_stopped = false;
241 
242 	return ret;
243 }
244 
245 static int tc3589x_keypad_disable(struct tc_keypad *keypad)
246 {
247 	struct tc3589x *tc3589x = keypad->tc3589x;
248 	int ret;
249 
250 	/* clear IRQ */
251 	ret = tc3589x_set_bits(tc3589x, TC3589x_KBDIC,
252 			0x0, TC3589x_EVT_INT_CLR | TC3589x_KBD_INT_CLR);
253 	if (ret < 0)
254 		return ret;
255 
256 	/* disable all interrupts */
257 	ret = tc3589x_set_bits(tc3589x, TC3589x_KBDMSK,
258 			~(TC3589x_EVT_LOSS_INT | TC3589x_EVT_INT), 0x0);
259 	if (ret < 0)
260 		return ret;
261 
262 	/* disable the keypad module */
263 	ret = tc3589x_set_bits(tc3589x, TC3589x_CLKEN, 0x1, 0x0);
264 	if (ret < 0)
265 		return ret;
266 
267 	/* put the keypad module into reset */
268 	ret = tc3589x_set_bits(tc3589x, TC3589x_RSTCTRL, TC3589x_KBDRST, 0x1);
269 
270 	keypad->keypad_stopped = true;
271 
272 	return ret;
273 }
274 
275 static int tc3589x_keypad_open(struct input_dev *input)
276 {
277 	int error;
278 	struct tc_keypad *keypad = input_get_drvdata(input);
279 
280 	/* enable the keypad module */
281 	error = tc3589x_keypad_enable(keypad);
282 	if (error < 0) {
283 		dev_err(&input->dev, "failed to enable keypad module\n");
284 		return error;
285 	}
286 
287 	error = tc3589x_keypad_init_key_hardware(keypad);
288 	if (error < 0) {
289 		dev_err(&input->dev, "failed to configure keypad module\n");
290 		return error;
291 	}
292 
293 	return 0;
294 }
295 
296 static void tc3589x_keypad_close(struct input_dev *input)
297 {
298 	struct tc_keypad *keypad = input_get_drvdata(input);
299 
300 	/* disable the keypad module */
301 	tc3589x_keypad_disable(keypad);
302 }
303 
304 static int __devinit tc3589x_keypad_probe(struct platform_device *pdev)
305 {
306 	struct tc3589x *tc3589x = dev_get_drvdata(pdev->dev.parent);
307 	struct tc_keypad *keypad;
308 	struct input_dev *input;
309 	const struct tc3589x_keypad_platform_data *plat;
310 	int error, irq;
311 
312 	plat = tc3589x->pdata->keypad;
313 	if (!plat) {
314 		dev_err(&pdev->dev, "invalid keypad platform data\n");
315 		return -EINVAL;
316 	}
317 
318 	irq = platform_get_irq(pdev, 0);
319 	if (irq < 0)
320 		return irq;
321 
322 	keypad = kzalloc(sizeof(struct tc_keypad), GFP_KERNEL);
323 	input = input_allocate_device();
324 	if (!keypad || !input) {
325 		dev_err(&pdev->dev, "failed to allocate keypad memory\n");
326 		error = -ENOMEM;
327 		goto err_free_mem;
328 	}
329 
330 	keypad->board = plat;
331 	keypad->input = input;
332 	keypad->tc3589x = tc3589x;
333 
334 	input->id.bustype = BUS_I2C;
335 	input->name = pdev->name;
336 	input->dev.parent = &pdev->dev;
337 
338 	input->keycode = keypad->keymap;
339 	input->keycodesize = sizeof(keypad->keymap[0]);
340 	input->keycodemax = ARRAY_SIZE(keypad->keymap);
341 
342 	input->open = tc3589x_keypad_open;
343 	input->close = tc3589x_keypad_close;
344 
345 	input_set_drvdata(input, keypad);
346 
347 	input_set_capability(input, EV_MSC, MSC_SCAN);
348 
349 	__set_bit(EV_KEY, input->evbit);
350 	if (!plat->no_autorepeat)
351 		__set_bit(EV_REP, input->evbit);
352 
353 	matrix_keypad_build_keymap(plat->keymap_data, 0x3,
354 			input->keycode, input->keybit);
355 
356 	error = request_threaded_irq(irq, NULL,
357 			tc3589x_keypad_irq, plat->irqtype,
358 			"tc3589x-keypad", keypad);
359 	if (error < 0) {
360 		dev_err(&pdev->dev,
361 				"Could not allocate irq %d,error %d\n",
362 				irq, error);
363 		goto err_free_mem;
364 	}
365 
366 	error = input_register_device(input);
367 	if (error) {
368 		dev_err(&pdev->dev, "Could not register input device\n");
369 		goto err_free_irq;
370 	}
371 
372 	/* let platform decide if keypad is a wakeup source or not */
373 	device_init_wakeup(&pdev->dev, plat->enable_wakeup);
374 	device_set_wakeup_capable(&pdev->dev, plat->enable_wakeup);
375 
376 	platform_set_drvdata(pdev, keypad);
377 
378 	return 0;
379 
380 err_free_irq:
381 	free_irq(irq, keypad);
382 err_free_mem:
383 	input_free_device(input);
384 	kfree(keypad);
385 	return error;
386 }
387 
388 static int __devexit tc3589x_keypad_remove(struct platform_device *pdev)
389 {
390 	struct tc_keypad *keypad = platform_get_drvdata(pdev);
391 	int irq = platform_get_irq(pdev, 0);
392 
393 	if (!keypad->keypad_stopped)
394 		tc3589x_keypad_disable(keypad);
395 
396 	free_irq(irq, keypad);
397 
398 	input_unregister_device(keypad->input);
399 
400 	kfree(keypad);
401 
402 	return 0;
403 }
404 
405 #ifdef CONFIG_PM
406 static int tc3589x_keypad_suspend(struct device *dev)
407 {
408 	struct platform_device *pdev = to_platform_device(dev);
409 	struct tc_keypad *keypad = platform_get_drvdata(pdev);
410 	int irq = platform_get_irq(pdev, 0);
411 
412 	/* keypad is already off; we do nothing */
413 	if (keypad->keypad_stopped)
414 		return 0;
415 
416 	/* if device is not a wakeup source, disable it for powersave */
417 	if (!device_may_wakeup(&pdev->dev))
418 		tc3589x_keypad_disable(keypad);
419 	else
420 		enable_irq_wake(irq);
421 
422 	return 0;
423 }
424 
425 static int tc3589x_keypad_resume(struct device *dev)
426 {
427 	struct platform_device *pdev = to_platform_device(dev);
428 	struct tc_keypad *keypad = platform_get_drvdata(pdev);
429 	int irq = platform_get_irq(pdev, 0);
430 
431 	if (!keypad->keypad_stopped)
432 		return 0;
433 
434 	/* enable the device to resume normal operations */
435 	if (!device_may_wakeup(&pdev->dev))
436 		tc3589x_keypad_enable(keypad);
437 	else
438 		disable_irq_wake(irq);
439 
440 	return 0;
441 }
442 
443 static const SIMPLE_DEV_PM_OPS(tc3589x_keypad_dev_pm_ops,
444 			       tc3589x_keypad_suspend, tc3589x_keypad_resume);
445 #endif
446 
447 static struct platform_driver tc3589x_keypad_driver = {
448 	.driver.name  = "tc3589x-keypad",
449 	.driver.owner = THIS_MODULE,
450 #ifdef CONFIG_PM
451 	.driver.pm = &tc3589x_keypad_dev_pm_ops,
452 #endif
453 	.probe = tc3589x_keypad_probe,
454 	.remove = __devexit_p(tc3589x_keypad_remove),
455 };
456 
457 static int __init tc3589x_keypad_init(void)
458 {
459 	return platform_driver_register(&tc3589x_keypad_driver);
460 }
461 module_init(tc3589x_keypad_init);
462 
463 static void __exit tc3589x_keypad_exit(void)
464 {
465 	return platform_driver_unregister(&tc3589x_keypad_driver);
466 }
467 module_exit(tc3589x_keypad_exit);
468 
469 MODULE_LICENSE("GPL v2");
470 MODULE_AUTHOR("Jayeeta Banerjee/Sundar Iyer");
471 MODULE_DESCRIPTION("TC35893 Keypad Driver");
472 MODULE_ALIAS("platform:tc3589x-keypad");
473