xref: /linux/drivers/input/touchscreen/sx8654.c (revision ca55b2fef3a9373fcfc30f82fd26bc7fccbda732)
1 /*
2  * Driver for Semtech SX8654 I2C touchscreen controller.
3  *
4  * Copyright (c) 2015 Armadeus Systems
5  *	Sébastien Szymanski <sebastien.szymanski@armadeus.com>
6  *
7  * Using code from:
8  *  - sx865x.c
9  *	Copyright (c) 2013 U-MoBo Srl
10  *	Pierluigi Passaro <p.passaro@u-mobo.com>
11  *  - sx8650.c
12  *      Copyright (c) 2009 Wayne Roberts
13  *  - tsc2007.c
14  *      Copyright (c) 2008 Kwangwoo Lee
15  *  - ads7846.c
16  *      Copyright (c) 2005 David Brownell
17  *      Copyright (c) 2006 Nokia Corporation
18  *  - corgi_ts.c
19  *      Copyright (C) 2004-2005 Richard Purdie
20  *  - omap_ts.[hc], ads7846.h, ts_osk.c
21  *      Copyright (C) 2002 MontaVista Software
22  *      Copyright (C) 2004 Texas Instruments
23  *      Copyright (C) 2005 Dirk Behme
24  *
25  *  This program is free software; you can redistribute it and/or modify
26  *  it under the terms of the GNU General Public License version 2 as
27  *  published by the Free Software Foundation.
28  */
29 
30 #include <linux/input.h>
31 #include <linux/module.h>
32 #include <linux/of.h>
33 #include <linux/i2c.h>
34 #include <linux/interrupt.h>
35 #include <linux/irq.h>
36 
37 /* register addresses */
38 #define I2C_REG_TOUCH0			0x00
39 #define I2C_REG_TOUCH1			0x01
40 #define I2C_REG_CHANMASK		0x04
41 #define I2C_REG_IRQMASK			0x22
42 #define I2C_REG_IRQSRC			0x23
43 #define I2C_REG_SOFTRESET		0x3f
44 
45 /* commands */
46 #define CMD_READ_REGISTER		0x40
47 #define CMD_MANUAL			0xc0
48 #define CMD_PENTRG			0xe0
49 
50 /* value for I2C_REG_SOFTRESET */
51 #define SOFTRESET_VALUE			0xde
52 
53 /* bits for I2C_REG_IRQSRC */
54 #define IRQ_PENTOUCH_TOUCHCONVDONE	0x08
55 #define IRQ_PENRELEASE			0x04
56 
57 /* bits for RegTouch1 */
58 #define CONDIRQ				0x20
59 #define FILT_7SA			0x03
60 
61 /* bits for I2C_REG_CHANMASK */
62 #define CONV_X				0x80
63 #define CONV_Y				0x40
64 
65 /* coordinates rate: higher nibble of CTRL0 register */
66 #define RATE_MANUAL			0x00
67 #define RATE_5000CPS			0xf0
68 
69 /* power delay: lower nibble of CTRL0 register */
70 #define POWDLY_1_1MS			0x0b
71 
72 #define MAX_12BIT			((1 << 12) - 1)
73 
74 struct sx8654 {
75 	struct input_dev *input;
76 	struct i2c_client *client;
77 };
78 
79 static irqreturn_t sx8654_irq(int irq, void *handle)
80 {
81 	struct sx8654 *sx8654 = handle;
82 	int irqsrc;
83 	u8 data[4];
84 	unsigned int x, y;
85 	int retval;
86 
87 	irqsrc = i2c_smbus_read_byte_data(sx8654->client,
88 					  CMD_READ_REGISTER | I2C_REG_IRQSRC);
89 	dev_dbg(&sx8654->client->dev, "irqsrc = 0x%x", irqsrc);
90 
91 	if (irqsrc < 0)
92 		goto out;
93 
94 	if (irqsrc & IRQ_PENRELEASE) {
95 		dev_dbg(&sx8654->client->dev, "pen release interrupt");
96 
97 		input_report_key(sx8654->input, BTN_TOUCH, 0);
98 		input_sync(sx8654->input);
99 	}
100 
101 	if (irqsrc & IRQ_PENTOUCH_TOUCHCONVDONE) {
102 		dev_dbg(&sx8654->client->dev, "pen touch interrupt");
103 
104 		retval = i2c_master_recv(sx8654->client, data, sizeof(data));
105 		if (retval != sizeof(data))
106 			goto out;
107 
108 		/* invalid data */
109 		if (unlikely(data[0] & 0x80 || data[2] & 0x80))
110 			goto out;
111 
112 		x = ((data[0] & 0xf) << 8) | (data[1]);
113 		y = ((data[2] & 0xf) << 8) | (data[3]);
114 
115 		input_report_abs(sx8654->input, ABS_X, x);
116 		input_report_abs(sx8654->input, ABS_Y, y);
117 		input_report_key(sx8654->input, BTN_TOUCH, 1);
118 		input_sync(sx8654->input);
119 
120 		dev_dbg(&sx8654->client->dev, "point(%4d,%4d)\n", x, y);
121 	}
122 
123 out:
124 	return IRQ_HANDLED;
125 }
126 
127 static int sx8654_open(struct input_dev *dev)
128 {
129 	struct sx8654 *sx8654 = input_get_drvdata(dev);
130 	struct i2c_client *client = sx8654->client;
131 	int error;
132 
133 	/* enable pen trigger mode */
134 	error = i2c_smbus_write_byte_data(client, I2C_REG_TOUCH0,
135 					  RATE_5000CPS | POWDLY_1_1MS);
136 	if (error) {
137 		dev_err(&client->dev, "writing to I2C_REG_TOUCH0 failed");
138 		return error;
139 	}
140 
141 	error = i2c_smbus_write_byte(client, CMD_PENTRG);
142 	if (error) {
143 		dev_err(&client->dev, "writing command CMD_PENTRG failed");
144 		return error;
145 	}
146 
147 	enable_irq(client->irq);
148 
149 	return 0;
150 }
151 
152 static void sx8654_close(struct input_dev *dev)
153 {
154 	struct sx8654 *sx8654 = input_get_drvdata(dev);
155 	struct i2c_client *client = sx8654->client;
156 	int error;
157 
158 	disable_irq(client->irq);
159 
160 	/* enable manual mode mode */
161 	error = i2c_smbus_write_byte(client, CMD_MANUAL);
162 	if (error) {
163 		dev_err(&client->dev, "writing command CMD_MANUAL failed");
164 		return;
165 	}
166 
167 	error = i2c_smbus_write_byte_data(client, I2C_REG_TOUCH0, 0);
168 	if (error) {
169 		dev_err(&client->dev, "writing to I2C_REG_TOUCH0 failed");
170 		return;
171 	}
172 }
173 
174 static int sx8654_probe(struct i2c_client *client,
175 			const struct i2c_device_id *id)
176 {
177 	struct sx8654 *sx8654;
178 	struct input_dev *input;
179 	int error;
180 
181 	if (!i2c_check_functionality(client->adapter,
182 				     I2C_FUNC_SMBUS_READ_WORD_DATA))
183 		return -ENXIO;
184 
185 	sx8654 = devm_kzalloc(&client->dev, sizeof(*sx8654), GFP_KERNEL);
186 	if (!sx8654)
187 		return -ENOMEM;
188 
189 	input = devm_input_allocate_device(&client->dev);
190 	if (!input)
191 		return -ENOMEM;
192 
193 	input->name = "SX8654 I2C Touchscreen";
194 	input->id.bustype = BUS_I2C;
195 	input->dev.parent = &client->dev;
196 	input->open = sx8654_open;
197 	input->close = sx8654_close;
198 
199 	__set_bit(INPUT_PROP_DIRECT, input->propbit);
200 	input_set_capability(input, EV_KEY, BTN_TOUCH);
201 	input_set_abs_params(input, ABS_X, 0, MAX_12BIT, 0, 0);
202 	input_set_abs_params(input, ABS_Y, 0, MAX_12BIT, 0, 0);
203 
204 	sx8654->client = client;
205 	sx8654->input = input;
206 
207 	input_set_drvdata(sx8654->input, sx8654);
208 
209 	error = i2c_smbus_write_byte_data(client, I2C_REG_SOFTRESET,
210 					  SOFTRESET_VALUE);
211 	if (error) {
212 		dev_err(&client->dev, "writing softreset value failed");
213 		return error;
214 	}
215 
216 	error = i2c_smbus_write_byte_data(client, I2C_REG_CHANMASK,
217 					  CONV_X | CONV_Y);
218 	if (error) {
219 		dev_err(&client->dev, "writing to I2C_REG_CHANMASK failed");
220 		return error;
221 	}
222 
223 	error = i2c_smbus_write_byte_data(client, I2C_REG_IRQMASK,
224 					  IRQ_PENTOUCH_TOUCHCONVDONE |
225 						IRQ_PENRELEASE);
226 	if (error) {
227 		dev_err(&client->dev, "writing to I2C_REG_IRQMASK failed");
228 		return error;
229 	}
230 
231 	error = i2c_smbus_write_byte_data(client, I2C_REG_TOUCH1,
232 					  CONDIRQ | FILT_7SA);
233 	if (error) {
234 		dev_err(&client->dev, "writing to I2C_REG_TOUCH1 failed");
235 		return error;
236 	}
237 
238 	error = devm_request_threaded_irq(&client->dev, client->irq,
239 					  NULL, sx8654_irq,
240 					  IRQF_TRIGGER_FALLING | IRQF_ONESHOT,
241 					  client->name, sx8654);
242 	if (error) {
243 		dev_err(&client->dev,
244 			"Failed to enable IRQ %d, error: %d\n",
245 			client->irq, error);
246 		return error;
247 	}
248 
249 	/* Disable the IRQ, we'll enable it in sx8654_open() */
250 	disable_irq(client->irq);
251 
252 	error = input_register_device(sx8654->input);
253 	if (error)
254 		return error;
255 
256 	i2c_set_clientdata(client, sx8654);
257 	return 0;
258 }
259 
260 #ifdef CONFIG_OF
261 static const struct of_device_id sx8654_of_match[] = {
262 	{ .compatible = "semtech,sx8654", },
263 	{ },
264 };
265 MODULE_DEVICE_TABLE(of, sx8654_of_match);
266 #endif
267 
268 static const struct i2c_device_id sx8654_id_table[] = {
269 	{ "semtech_sx8654", 0 },
270 	{ },
271 };
272 MODULE_DEVICE_TABLE(i2c, sx8654_id_table);
273 
274 static struct i2c_driver sx8654_driver = {
275 	.driver = {
276 		.name = "sx8654",
277 		.of_match_table = of_match_ptr(sx8654_of_match),
278 	},
279 	.id_table = sx8654_id_table,
280 	.probe = sx8654_probe,
281 };
282 module_i2c_driver(sx8654_driver);
283 
284 MODULE_AUTHOR("Sébastien Szymanski <sebastien.szymanski@armadeus.com>");
285 MODULE_DESCRIPTION("Semtech SX8654 I2C Touchscreen Driver");
286 MODULE_LICENSE("GPL");
287