| 
									
										
										
										
											2010-03-19 22:18:15 -07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Driver for a keypad w/16 buttons connected to a PCF8574 I2C I/O expander | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright 2005-2008 Analog Devices Inc. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Licensed under the GPL-2 or later. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <linux/module.h>
 | 
					
						
							|  |  |  | #include <linux/input.h>
 | 
					
						
							|  |  |  | #include <linux/interrupt.h>
 | 
					
						
							|  |  |  | #include <linux/i2c.h>
 | 
					
						
							|  |  |  | #include <linux/slab.h>
 | 
					
						
							|  |  |  | #include <linux/workqueue.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define DRV_NAME "pcf8574_keypad"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const unsigned char pcf8574_kp_btncode[] = { | 
					
						
							|  |  |  | 	[0] = KEY_RESERVED, | 
					
						
							|  |  |  | 	[1] = KEY_ENTER, | 
					
						
							|  |  |  | 	[2] = KEY_BACKSLASH, | 
					
						
							|  |  |  | 	[3] = KEY_0, | 
					
						
							|  |  |  | 	[4] = KEY_RIGHTBRACE, | 
					
						
							|  |  |  | 	[5] = KEY_C, | 
					
						
							|  |  |  | 	[6] = KEY_9, | 
					
						
							|  |  |  | 	[7] = KEY_8, | 
					
						
							|  |  |  | 	[8] = KEY_7, | 
					
						
							|  |  |  | 	[9] = KEY_B, | 
					
						
							|  |  |  | 	[10] = KEY_6, | 
					
						
							|  |  |  | 	[11] = KEY_5, | 
					
						
							|  |  |  | 	[12] = KEY_4, | 
					
						
							|  |  |  | 	[13] = KEY_A, | 
					
						
							|  |  |  | 	[14] = KEY_3, | 
					
						
							|  |  |  | 	[15] = KEY_2, | 
					
						
							|  |  |  | 	[16] = KEY_1 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct kp_data { | 
					
						
							|  |  |  | 	unsigned short btncode[ARRAY_SIZE(pcf8574_kp_btncode)]; | 
					
						
							|  |  |  | 	struct input_dev *idev; | 
					
						
							|  |  |  | 	struct i2c_client *client; | 
					
						
							|  |  |  | 	char name[64]; | 
					
						
							|  |  |  | 	char phys[32]; | 
					
						
							|  |  |  | 	unsigned char laststate; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static short read_state(struct kp_data *lp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned char x, y, a, b; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	i2c_smbus_write_byte(lp->client, 240); | 
					
						
							|  |  |  | 	x = 0xF & (~(i2c_smbus_read_byte(lp->client) >> 4)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	i2c_smbus_write_byte(lp->client, 15); | 
					
						
							|  |  |  | 	y = 0xF & (~i2c_smbus_read_byte(lp->client)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (a = 0; x > 0; a++) | 
					
						
							|  |  |  | 		x = x >> 1; | 
					
						
							|  |  |  | 	for (b = 0; y > 0; b++) | 
					
						
							|  |  |  | 		y = y >> 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ((a - 1) * 4) + b; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static irqreturn_t pcf8574_kp_irq_handler(int irq, void *dev_id) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct kp_data *lp = dev_id; | 
					
						
							|  |  |  | 	unsigned char nextstate = read_state(lp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (lp->laststate != nextstate) { | 
					
						
							| 
									
										
										
										
											2010-06-05 00:34:08 -07:00
										 |  |  | 		int key_down = nextstate < ARRAY_SIZE(lp->btncode); | 
					
						
							| 
									
										
										
										
											2010-03-19 22:18:15 -07:00
										 |  |  | 		unsigned short keycode = key_down ? | 
					
						
							|  |  |  | 			lp->btncode[nextstate] : lp->btncode[lp->laststate]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		input_report_key(lp->idev, keycode, key_down); | 
					
						
							|  |  |  | 		input_sync(lp->idev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		lp->laststate = nextstate; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return IRQ_HANDLED; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-23 21:38:25 -08:00
										 |  |  | static int pcf8574_kp_probe(struct i2c_client *client, const struct i2c_device_id *id) | 
					
						
							| 
									
										
										
										
											2010-03-19 22:18:15 -07:00
										 |  |  | { | 
					
						
							|  |  |  | 	int i, ret; | 
					
						
							|  |  |  | 	struct input_dev *idev; | 
					
						
							|  |  |  | 	struct kp_data *lp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (i2c_smbus_write_byte(client, 240) < 0) { | 
					
						
							|  |  |  | 		dev_err(&client->dev, "probe: write fail\n"); | 
					
						
							|  |  |  | 		return -ENODEV; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	lp = kzalloc(sizeof(*lp), GFP_KERNEL); | 
					
						
							|  |  |  | 	if (!lp) | 
					
						
							|  |  |  | 		return -ENOMEM; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	idev = input_allocate_device(); | 
					
						
							|  |  |  | 	if (!idev) { | 
					
						
							|  |  |  | 		dev_err(&client->dev, "Can't allocate input device\n"); | 
					
						
							|  |  |  | 		ret = -ENOMEM; | 
					
						
							|  |  |  | 		goto fail_allocate; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	lp->idev = idev; | 
					
						
							|  |  |  | 	lp->client = client; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	idev->evbit[0] = BIT_MASK(EV_KEY); | 
					
						
							|  |  |  | 	idev->keycode = lp->btncode; | 
					
						
							|  |  |  | 	idev->keycodesize = sizeof(lp->btncode[0]); | 
					
						
							|  |  |  | 	idev->keycodemax = ARRAY_SIZE(lp->btncode); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (i = 0; i < ARRAY_SIZE(pcf8574_kp_btncode); i++) { | 
					
						
							| 
									
										
										
										
											2013-11-23 10:06:36 -08:00
										 |  |  | 		if (lp->btncode[i] <= KEY_MAX) { | 
					
						
							|  |  |  | 			lp->btncode[i] = pcf8574_kp_btncode[i]; | 
					
						
							|  |  |  | 			__set_bit(lp->btncode[i], idev->keybit); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-03-19 22:18:15 -07:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-11-23 10:06:36 -08:00
										 |  |  | 	__clear_bit(KEY_RESERVED, idev->keybit); | 
					
						
							| 
									
										
										
										
											2010-03-19 22:18:15 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	sprintf(lp->name, DRV_NAME); | 
					
						
							|  |  |  | 	sprintf(lp->phys, "kp_data/input0"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	idev->name = lp->name; | 
					
						
							|  |  |  | 	idev->phys = lp->phys; | 
					
						
							|  |  |  | 	idev->id.bustype = BUS_I2C; | 
					
						
							|  |  |  | 	idev->id.vendor = 0x0001; | 
					
						
							|  |  |  | 	idev->id.product = 0x0001; | 
					
						
							|  |  |  | 	idev->id.version = 0x0100; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	lp->laststate = read_state(lp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ret = request_threaded_irq(client->irq, NULL, pcf8574_kp_irq_handler, | 
					
						
							|  |  |  | 				   IRQF_TRIGGER_LOW | IRQF_ONESHOT, | 
					
						
							|  |  |  | 				   DRV_NAME, lp); | 
					
						
							|  |  |  | 	if (ret) { | 
					
						
							|  |  |  | 		dev_err(&client->dev, "IRQ %d is not free\n", client->irq); | 
					
						
							| 
									
										
										
										
											2010-11-10 23:59:20 -08:00
										 |  |  | 		goto fail_free_device; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ret = input_register_device(idev); | 
					
						
							|  |  |  | 	if (ret) { | 
					
						
							|  |  |  | 		dev_err(&client->dev, "input_register_device() failed\n"); | 
					
						
							|  |  |  | 		goto fail_free_irq; | 
					
						
							| 
									
										
										
										
											2010-03-19 22:18:15 -07:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	i2c_set_clientdata(client, lp); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-10 23:59:20 -08:00
										 |  |  |  fail_free_irq: | 
					
						
							|  |  |  | 	free_irq(client->irq, lp); | 
					
						
							|  |  |  |  fail_free_device: | 
					
						
							| 
									
										
										
										
											2010-03-19 22:18:15 -07:00
										 |  |  | 	input_free_device(idev); | 
					
						
							|  |  |  |  fail_allocate: | 
					
						
							|  |  |  | 	kfree(lp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-23 21:50:47 -08:00
										 |  |  | static int pcf8574_kp_remove(struct i2c_client *client) | 
					
						
							| 
									
										
										
										
											2010-03-19 22:18:15 -07:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct kp_data *lp = i2c_get_clientdata(client); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	free_irq(client->irq, lp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	input_unregister_device(lp->idev); | 
					
						
							|  |  |  | 	kfree(lp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef CONFIG_PM
 | 
					
						
							| 
									
										
										
										
											2010-11-11 01:03:00 -08:00
										 |  |  | static int pcf8574_kp_resume(struct device *dev) | 
					
						
							| 
									
										
										
										
											2010-03-19 22:18:15 -07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-11-11 01:03:00 -08:00
										 |  |  | 	struct i2c_client *client = to_i2c_client(dev); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-19 22:18:15 -07:00
										 |  |  | 	enable_irq(client->irq); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-11 01:03:00 -08:00
										 |  |  | static int pcf8574_kp_suspend(struct device *dev) | 
					
						
							| 
									
										
										
										
											2010-03-19 22:18:15 -07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-11-11 01:03:00 -08:00
										 |  |  | 	struct i2c_client *client = to_i2c_client(dev); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-19 22:18:15 -07:00
										 |  |  | 	disable_irq(client->irq); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-11-11 01:03:00 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | static const struct dev_pm_ops pcf8574_kp_pm_ops = { | 
					
						
							|  |  |  | 	.suspend	= pcf8574_kp_suspend, | 
					
						
							|  |  |  | 	.resume		= pcf8574_kp_resume, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-19 22:18:15 -07:00
										 |  |  | #else
 | 
					
						
							|  |  |  | # define pcf8574_kp_resume  NULL
 | 
					
						
							|  |  |  | # define pcf8574_kp_suspend NULL
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct i2c_device_id pcf8574_kp_id[] = { | 
					
						
							|  |  |  | 	{ DRV_NAME, 0 }, | 
					
						
							|  |  |  | 	{ } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | MODULE_DEVICE_TABLE(i2c, pcf8574_kp_id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct i2c_driver pcf8574_kp_driver = { | 
					
						
							|  |  |  | 	.driver = { | 
					
						
							|  |  |  | 		.name  = DRV_NAME, | 
					
						
							|  |  |  | 		.owner = THIS_MODULE, | 
					
						
							| 
									
										
										
										
											2010-11-11 01:03:00 -08:00
										 |  |  | #ifdef CONFIG_PM
 | 
					
						
							|  |  |  | 		.pm = &pcf8574_kp_pm_ops, | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-03-19 22:18:15 -07:00
										 |  |  | 	}, | 
					
						
							|  |  |  | 	.probe    = pcf8574_kp_probe, | 
					
						
							| 
									
										
										
										
											2012-11-23 21:27:39 -08:00
										 |  |  | 	.remove   = pcf8574_kp_remove, | 
					
						
							| 
									
										
										
										
											2010-03-19 22:18:15 -07:00
										 |  |  | 	.id_table = pcf8574_kp_id, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-16 23:05:41 -07:00
										 |  |  | module_i2c_driver(pcf8574_kp_driver); | 
					
						
							| 
									
										
										
										
											2010-03-19 22:18:15 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | MODULE_AUTHOR("Michael Hennerich"); | 
					
						
							|  |  |  | MODULE_DESCRIPTION("Keypad input driver for 16 keys connected to PCF8574"); | 
					
						
							|  |  |  | MODULE_LICENSE("GPL"); |