blob: 8b08ac9ff14e0204efc80c127827349c48e672f5 [file] [log] [blame]
Andrew Huangab45aab2013-06-27 10:06:56 +00001/* drivers/input/touchscreen/gt9xx.c
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002 *
3 * Copyright (c) 2013, The Linux Foundation. All rights reserved.
4 *
5 * Linux Foundation chooses to take subject only to the GPLv2 license
6 * terms, and distributes only under these terms.
7 *
Andrew Huangab45aab2013-06-27 10:06:56 +00008 * 2010 - 2013 Goodix Technology.
Bingzhe Cai83caeea2013-07-16 17:57:37 +08009 *
Andrew Huangab45aab2013-06-27 10:06:56 +000010 * This program is free software; you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License as published by
12 * the Free Software Foundation; either version 2 of the License, or
13 * (at your option) any later version.
Bingzhe Cai83caeea2013-07-16 17:57:37 +080014 *
15 * This program is distributed in the hope that it will be a reference
16 * to you, when you are integrating the GOODiX's CTP IC into your system,
17 * but WITHOUT ANY WARRANTY; without even the implied warranty of
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Andrew Huangab45aab2013-06-27 10:06:56 +000019 * General Public License for more details.
Bingzhe Cai83caeea2013-07-16 17:57:37 +080020 *
Andrew Huangab45aab2013-06-27 10:06:56 +000021 * Version: 1.8
22 * Authors: andrew@goodix.com, meta@goodix.com
23 * Release Date: 2013/04/25
24 * Revision record:
Bingzhe Cai83caeea2013-07-16 17:57:37 +080025 * V1.0:
26 * first Release. By Andrew, 2012/08/31
Andrew Huangab45aab2013-06-27 10:06:56 +000027 * V1.2:
Bingzhe Cai83caeea2013-07-16 17:57:37 +080028 * modify gtp_reset_guitar,slot report,tracking_id & 0x0F.
29 * By Andrew, 2012/10/15
Andrew Huangab45aab2013-06-27 10:06:56 +000030 * V1.4:
31 * modify gt9xx_update.c. By Andrew, 2012/12/12
Bingzhe Cai83caeea2013-07-16 17:57:37 +080032 * V1.6:
Andrew Huangab45aab2013-06-27 10:06:56 +000033 * 1. new heartbeat/esd_protect mechanism(add external watchdog)
Bingzhe Cai83caeea2013-07-16 17:57:37 +080034 * 2. doze mode, sliding wakeup
35 * 3. 3 more cfg_group(GT9 Sensor_ID: 0~5)
Andrew Huangab45aab2013-06-27 10:06:56 +000036 * 3. config length verification
37 * 4. names & comments
38 * By Meta, 2013/03/11
39 * V1.8:
Bingzhe Cai83caeea2013-07-16 17:57:37 +080040 * 1. pen/stylus identification
Andrew Huangab45aab2013-06-27 10:06:56 +000041 * 2. read double check & fixed config support
42 * 2. new esd & slide wakeup optimization
43 * By Meta, 2013/06/08
44 */
45
Bingzhe Caic68e6112013-07-22 00:12:48 +080046#include <linux/regulator/consumer.h>
Andrew Huangab45aab2013-06-27 10:06:56 +000047#include "gt9xx.h"
48
Bingzhe Caid0361172013-07-18 19:22:11 +080049#include <linux/of_gpio.h>
50
Bingzhe Cai83caeea2013-07-16 17:57:37 +080051#include <linux/input/mt.h>
Andrew Huangab45aab2013-06-27 10:06:56 +000052
Bingzhe Cai6beb0112013-08-27 00:27:26 +080053#define GOODIX_DEV_NAME "Goodix-CTP"
Bingzhe Cai83caeea2013-07-16 17:57:37 +080054#define CFG_MAX_TOUCH_POINTS 5
55#define GOODIX_COORDS_ARR_SIZE 4
56#define MAX_BUTTONS 4
57
58/* HIGH: 0x28/0x29, LOW: 0xBA/0xBB */
59#define GTP_I2C_ADDRESS_HIGH 0x14
60#define GTP_I2C_ADDRESS_LOW 0x5D
61#define CFG_GROUP_LEN(p_cfg_grp) (sizeof(p_cfg_grp) / sizeof(p_cfg_grp[0]))
62
Bingzhe Caic68e6112013-07-22 00:12:48 +080063#define GOODIX_VTG_MIN_UV 2600000
64#define GOODIX_VTG_MAX_UV 3300000
65#define GOODIX_I2C_VTG_MIN_UV 1800000
66#define GOODIX_I2C_VTG_MAX_UV 1800000
67#define GOODIX_VDD_LOAD_MIN_UA 0
68#define GOODIX_VDD_LOAD_MAX_UA 10000
69#define GOODIX_VIO_LOAD_MIN_UA 0
70#define GOODIX_VIO_LOAD_MAX_UA 10000
71
Bingzhe Cai83caeea2013-07-16 17:57:37 +080072#define RESET_DELAY_T3_US 200 /* T3: > 100us */
73#define RESET_DELAY_T4 20 /* T4: > 5ms */
74
Bingzhe Cai15d14462013-09-11 18:52:17 +080075#define PHY_BUF_SIZE 32
76#define PROP_NAME_SIZE 24
Bingzhe Cai83caeea2013-07-16 17:57:37 +080077
78#define GTP_MAX_TOUCH 5
79#define GTP_ESD_CHECK_CIRCLE_MS 2000
Andrew Huangab45aab2013-06-27 10:06:56 +000080
81#if GTP_HAVE_TOUCH_KEY
Bingzhe Cai83caeea2013-07-16 17:57:37 +080082static const u16 touch_key_array[] = {KEY_MENU, KEY_HOMEPAGE, KEY_BACK};
83#define GTP_MAX_KEY_NUM (sizeof(touch_key_array)/sizeof(touch_key_array[0]))
84
Andrew Huangab45aab2013-06-27 10:06:56 +000085#if GTP_DEBUG_ON
Bingzhe Cai83caeea2013-07-16 17:57:37 +080086static const int key_codes[] = {
87 KEY_HOME, KEY_BACK, KEY_MENU, KEY_SEARCH
88};
89static const char *const key_names[] = {
90 "Key_Home", "Key_Back", "Key_Menu", "Key_Search"
91};
Andrew Huangab45aab2013-06-27 10:06:56 +000092#endif
Andrew Huangab45aab2013-06-27 10:06:56 +000093#endif
94
Bingzhe Cai83caeea2013-07-16 17:57:37 +080095static void gtp_reset_guitar(struct goodix_ts_data *ts, int ms);
96static void gtp_int_sync(struct goodix_ts_data *ts, int ms);
97static int gtp_i2c_test(struct i2c_client *client);
Andrew Huangab45aab2013-06-27 10:06:56 +000098
Bingzhe Cai7d49bb32013-07-22 18:23:20 +080099#if defined(CONFIG_FB)
100static int fb_notifier_callback(struct notifier_block *self,
101 unsigned long event, void *data);
102#elif defined(CONFIG_HAS_EARLYSUSPEND)
Andrew Huangab45aab2013-06-27 10:06:56 +0000103static void goodix_ts_early_suspend(struct early_suspend *h);
104static void goodix_ts_late_resume(struct early_suspend *h);
105#endif
Andrew Huangab45aab2013-06-27 10:06:56 +0000106
107#if GTP_ESD_PROTECT
108static struct delayed_work gtp_esd_check_work;
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800109static struct workqueue_struct *gtp_esd_check_workqueue;
110static void gtp_esd_check_func(struct work_struct *work);
111static int gtp_init_ext_watchdog(struct i2c_client *client);
112struct i2c_client *i2c_connect_client;
Andrew Huangab45aab2013-06-27 10:06:56 +0000113#endif
114
Andrew Huangab45aab2013-06-27 10:06:56 +0000115#if GTP_SLIDE_WAKEUP
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800116enum doze_status {
117 DOZE_DISABLED = 0,
118 DOZE_ENABLED = 1,
119 DOZE_WAKEUP = 2,
120};
121static enum doze_status = DOZE_DISABLED;
Andrew Huangab45aab2013-06-27 10:06:56 +0000122static s8 gtp_enter_doze(struct goodix_ts_data *ts);
123#endif
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800124bool init_done;
125static u8 chip_gt9xxs; /* true if ic is gt9xxs, like gt915s */
126u8 grp_cfg_version;
Andrew Huangab45aab2013-06-27 10:06:56 +0000127
128/*******************************************************
129Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800130 Read data from the i2c slave device.
Andrew Huangab45aab2013-06-27 10:06:56 +0000131Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800132 client: i2c device.
133 buf[0~1]: read start address.
134 buf[2~len-1]: read data buffer.
135 len: GTP_ADDR_LENGTH + read bytes count
Andrew Huangab45aab2013-06-27 10:06:56 +0000136Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800137 numbers of i2c_msgs to transfer:
138 2: succeed, otherwise: failed
Andrew Huangab45aab2013-06-27 10:06:56 +0000139*********************************************************/
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800140int gtp_i2c_read(struct i2c_client *client, u8 *buf, int len)
Andrew Huangab45aab2013-06-27 10:06:56 +0000141{
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800142 struct goodix_ts_data *ts = i2c_get_clientdata(client);
143 struct i2c_msg msgs[2];
144 int ret = -EIO;
145 int retries = 0;
Andrew Huangab45aab2013-06-27 10:06:56 +0000146
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800147 msgs[0].flags = !I2C_M_RD;
148 msgs[0].addr = client->addr;
149 msgs[0].len = GTP_ADDR_LENGTH;
150 msgs[0].buf = &buf[0];
Andrew Huangab45aab2013-06-27 10:06:56 +0000151
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800152 msgs[1].flags = I2C_M_RD;
153 msgs[1].addr = client->addr;
154 msgs[1].len = len - GTP_ADDR_LENGTH;
155 msgs[1].buf = &buf[GTP_ADDR_LENGTH];
156
157 while (retries < 5) {
158 ret = i2c_transfer(client->adapter, msgs, 2);
159 if (ret == 2)
160 break;
161 retries++;
162 }
163 if (retries >= 5) {
164#if GTP_SLIDE_WAKEUP
165 /* reset chip would quit doze mode */
166 if (DOZE_ENABLED == doze_status)
167 return ret;
168#endif
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800169 if (init_done)
170 gtp_reset_guitar(ts, 10);
171 else
172 dev_warn(&client->dev,
173 "<GTP> gtp_reset_guitar exit init_done=%d:\n",
174 init_done);
175 }
176 return ret;
Andrew Huangab45aab2013-06-27 10:06:56 +0000177}
178
179/*******************************************************
180Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800181 Write data to the i2c slave device.
Andrew Huangab45aab2013-06-27 10:06:56 +0000182Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800183 client: i2c device.
184 buf[0~1]: write start address.
185 buf[2~len-1]: data buffer
186 len: GTP_ADDR_LENGTH + write bytes count
Andrew Huangab45aab2013-06-27 10:06:56 +0000187Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800188 numbers of i2c_msgs to transfer:
189 1: succeed, otherwise: failed
Andrew Huangab45aab2013-06-27 10:06:56 +0000190*********************************************************/
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800191int gtp_i2c_write(struct i2c_client *client, u8 *buf, int len)
Andrew Huangab45aab2013-06-27 10:06:56 +0000192{
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800193 struct goodix_ts_data *ts = i2c_get_clientdata(client);
194 struct i2c_msg msg;
195 int ret = -EIO;
196 int retries = 0;
Andrew Huangab45aab2013-06-27 10:06:56 +0000197
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800198 msg.flags = !I2C_M_RD;
199 msg.addr = client->addr;
200 msg.len = len;
201 msg.buf = buf;
Andrew Huangab45aab2013-06-27 10:06:56 +0000202
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800203 while (retries < 5) {
204 ret = i2c_transfer(client->adapter, &msg, 1);
205 if (ret == 1)
206 break;
207 retries++;
208 }
209 if ((retries >= 5)) {
210#if GTP_SLIDE_WAKEUP
211 if (DOZE_ENABLED == doze_status)
212 return ret;
213#endif
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800214 if (init_done)
215 gtp_reset_guitar(ts, 10);
216 else
217 dev_warn(&client->dev,
218 "<GTP> gtp_reset_guitar exit init_done=%d:\n",
219 init_done);
220 }
221 return ret;
Andrew Huangab45aab2013-06-27 10:06:56 +0000222}
223/*******************************************************
224Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800225 i2c read twice, compare the results
Andrew Huangab45aab2013-06-27 10:06:56 +0000226Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800227 client: i2c device
228 addr: operate address
229 rxbuf: read data to store, if compare successful
230 len: bytes to read
Andrew Huangab45aab2013-06-27 10:06:56 +0000231Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800232 FAIL: read failed
233 SUCCESS: read successful
Andrew Huangab45aab2013-06-27 10:06:56 +0000234*********************************************************/
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800235int gtp_i2c_read_dbl_check(struct i2c_client *client,
236 u16 addr, u8 *rxbuf, int len)
Andrew Huangab45aab2013-06-27 10:06:56 +0000237{
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800238 u8 buf[16] = {0};
239 u8 confirm_buf[16] = {0};
240 u8 retry = 0;
241
242 while (retry++ < 3) {
243 memset(buf, 0xAA, 16);
244 buf[0] = (u8)(addr >> 8);
245 buf[1] = (u8)(addr & 0xFF);
246 gtp_i2c_read(client, buf, len + 2);
247
248 memset(confirm_buf, 0xAB, 16);
249 confirm_buf[0] = (u8)(addr >> 8);
250 confirm_buf[1] = (u8)(addr & 0xFF);
251 gtp_i2c_read(client, confirm_buf, len + 2);
252
253 if (!memcmp(buf, confirm_buf, len + 2))
254 break;
255 }
256 if (retry < 3) {
257 memcpy(rxbuf, confirm_buf + 2, len);
258 return SUCCESS;
259 } else {
260 dev_err(&client->dev,
261 "i2c read 0x%04X, %d bytes, double check failed!",
262 addr, len);
263 return FAIL;
264 }
Andrew Huangab45aab2013-06-27 10:06:56 +0000265}
266
267/*******************************************************
268Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800269 Send config data.
Andrew Huangab45aab2013-06-27 10:06:56 +0000270Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800271 client: i2c device.
Andrew Huangab45aab2013-06-27 10:06:56 +0000272Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800273 result of i2c write operation.
274 > 0: succeed, otherwise: failed
Andrew Huangab45aab2013-06-27 10:06:56 +0000275*********************************************************/
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800276static int gtp_send_cfg(struct goodix_ts_data *ts)
Andrew Huangab45aab2013-06-27 10:06:56 +0000277{
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800278 int ret;
Andrew Huangab45aab2013-06-27 10:06:56 +0000279#if GTP_DRIVER_SEND_CFG
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800280 int retry = 0;
281
282 if (ts->fixed_cfg) {
283 dev_dbg(&ts->client->dev,
284 "Ic fixed config, no config sent!");
285 ret = 2;
286 } else {
287 for (retry = 0; retry < 5; retry++) {
288 ret = gtp_i2c_write(ts->client,
289 ts->config_data,
290 GTP_CONFIG_MAX_LENGTH + GTP_ADDR_LENGTH);
291 if (ret > 0)
292 break;
293 }
294 }
Andrew Huangab45aab2013-06-27 10:06:56 +0000295#endif
296
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800297 return ret;
Andrew Huangab45aab2013-06-27 10:06:56 +0000298}
299
300/*******************************************************
301Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800302 Disable irq function
Andrew Huangab45aab2013-06-27 10:06:56 +0000303Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800304 ts: goodix i2c_client private data
Andrew Huangab45aab2013-06-27 10:06:56 +0000305Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800306 None.
Andrew Huangab45aab2013-06-27 10:06:56 +0000307*********************************************************/
308void gtp_irq_disable(struct goodix_ts_data *ts)
309{
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800310 unsigned long irqflags;
Andrew Huangab45aab2013-06-27 10:06:56 +0000311
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800312 spin_lock_irqsave(&ts->irq_lock, irqflags);
313 if (!ts->irq_is_disabled) {
314 ts->irq_is_disabled = true;
315 disable_irq_nosync(ts->client->irq);
316 }
317 spin_unlock_irqrestore(&ts->irq_lock, irqflags);
Andrew Huangab45aab2013-06-27 10:06:56 +0000318}
319
320/*******************************************************
321Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800322 Enable irq function
Andrew Huangab45aab2013-06-27 10:06:56 +0000323Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800324 ts: goodix i2c_client private data
Andrew Huangab45aab2013-06-27 10:06:56 +0000325Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800326 None.
Andrew Huangab45aab2013-06-27 10:06:56 +0000327*********************************************************/
328void gtp_irq_enable(struct goodix_ts_data *ts)
329{
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800330 unsigned long irqflags = 0;
Andrew Huangab45aab2013-06-27 10:06:56 +0000331
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800332 spin_lock_irqsave(&ts->irq_lock, irqflags);
333 if (ts->irq_is_disabled) {
334 enable_irq(ts->client->irq);
335 ts->irq_is_disabled = false;
336 }
337 spin_unlock_irqrestore(&ts->irq_lock, irqflags);
Andrew Huangab45aab2013-06-27 10:06:56 +0000338}
339
Andrew Huangab45aab2013-06-27 10:06:56 +0000340/*******************************************************
341Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800342 Report touch point event
Andrew Huangab45aab2013-06-27 10:06:56 +0000343Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800344 ts: goodix i2c_client private data
345 id: trackId
346 x: input x coordinate
347 y: input y coordinate
348 w: input pressure
Andrew Huangab45aab2013-06-27 10:06:56 +0000349Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800350 None.
Andrew Huangab45aab2013-06-27 10:06:56 +0000351*********************************************************/
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800352static void gtp_touch_down(struct goodix_ts_data *ts, int id, int x, int y,
353 int w)
Andrew Huangab45aab2013-06-27 10:06:56 +0000354{
355#if GTP_CHANGE_X2Y
Shantanu Jain7dc2b472013-09-27 11:55:09 +0530356 swap(x, y);
Andrew Huangab45aab2013-06-27 10:06:56 +0000357#endif
358
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800359 input_mt_slot(ts->input_dev, id);
Bingzhe Cai6beb0112013-08-27 00:27:26 +0800360 input_mt_report_slot_state(ts->input_dev, MT_TOOL_FINGER, true);
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800361 input_report_abs(ts->input_dev, ABS_MT_POSITION_X, x);
362 input_report_abs(ts->input_dev, ABS_MT_POSITION_Y, y);
363 input_report_abs(ts->input_dev, ABS_MT_TOUCH_MAJOR, w);
364 input_report_abs(ts->input_dev, ABS_MT_WIDTH_MAJOR, w);
Andrew Huangab45aab2013-06-27 10:06:56 +0000365}
366
367/*******************************************************
368Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800369 Report touch release event
Andrew Huangab45aab2013-06-27 10:06:56 +0000370Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800371 ts: goodix i2c_client private data
Andrew Huangab45aab2013-06-27 10:06:56 +0000372Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800373 None.
Andrew Huangab45aab2013-06-27 10:06:56 +0000374*********************************************************/
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800375static void gtp_touch_up(struct goodix_ts_data *ts, int id)
Andrew Huangab45aab2013-06-27 10:06:56 +0000376{
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800377 input_mt_slot(ts->input_dev, id);
Bingzhe Cai6beb0112013-08-27 00:27:26 +0800378 input_mt_report_slot_state(ts->input_dev, MT_TOOL_FINGER, false);
Andrew Huangab45aab2013-06-27 10:06:56 +0000379}
380
381
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800382
Andrew Huangab45aab2013-06-27 10:06:56 +0000383/*******************************************************
384Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800385 Goodix touchscreen work function
Andrew Huangab45aab2013-06-27 10:06:56 +0000386Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800387 work: work struct of goodix_workqueue
Andrew Huangab45aab2013-06-27 10:06:56 +0000388Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800389 None.
Andrew Huangab45aab2013-06-27 10:06:56 +0000390*********************************************************/
391static void goodix_ts_work_func(struct work_struct *work)
392{
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800393 u8 end_cmd[3] = { GTP_READ_COOR_ADDR >> 8,
394 GTP_READ_COOR_ADDR & 0xFF, 0};
395 u8 point_data[2 + 1 + 8 * GTP_MAX_TOUCH + 1] = {
396 GTP_READ_COOR_ADDR >> 8,
397 GTP_READ_COOR_ADDR & 0xFF};
398 u8 touch_num = 0;
399 u8 finger = 0;
400 static u16 pre_touch;
401 static u8 pre_key;
Andrew Huangab45aab2013-06-27 10:06:56 +0000402#if GTP_WITH_PEN
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800403 static u8 pre_pen;
Andrew Huangab45aab2013-06-27 10:06:56 +0000404#endif
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800405 u8 key_value = 0;
406 u8 *coor_data = NULL;
407 s32 input_x = 0;
408 s32 input_y = 0;
409 s32 input_w = 0;
410 s32 id = 0;
411 s32 i = 0;
412 int ret = -1;
413 struct goodix_ts_data *ts = NULL;
Andrew Huangab45aab2013-06-27 10:06:56 +0000414
415#if GTP_SLIDE_WAKEUP
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800416 u8 doze_buf[3] = {0x81, 0x4B};
Andrew Huangab45aab2013-06-27 10:06:56 +0000417#endif
418
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800419 ts = container_of(work, struct goodix_ts_data, work);
420#ifdef CONFIG_GT9XX_TOUCHPANEL_UPDATE
421 if (ts->enter_update)
422 return;
423#endif
424
Andrew Huangab45aab2013-06-27 10:06:56 +0000425#if GTP_SLIDE_WAKEUP
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800426 if (DOZE_ENABLED == doze_status) {
427 ret = gtp_i2c_read(ts->client, doze_buf, 3);
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800428 if (ret > 0) {
429 if (doze_buf[2] == 0xAA) {
430 dev_dbg(&ts->client->dev,
431 "Slide(0xAA) To Light up the screen!");
432 doze_status = DOZE_WAKEUP;
433 input_report_key(
434 ts->input_dev, KEY_POWER, 1);
435 input_sync(ts->input_dev);
436 input_report_key(
437 ts->input_dev, KEY_POWER, 0);
438 input_sync(ts->input_dev);
439 /* clear 0x814B */
440 doze_buf[2] = 0x00;
441 gtp_i2c_write(ts->client, doze_buf, 3);
442 } else if (doze_buf[2] == 0xBB) {
443 dev_dbg(&ts->client->dev,
444 "Slide(0xBB) To Light up the screen!");
445 doze_status = DOZE_WAKEUP;
446 input_report_key(ts->input_dev, KEY_POWER, 1);
447 input_sync(ts->input_dev);
448 input_report_key(ts->input_dev, KEY_POWER, 0);
449 input_sync(ts->input_dev);
450 /* clear 0x814B*/
451 doze_buf[2] = 0x00;
452 gtp_i2c_write(ts->client, doze_buf, 3);
453 } else if (0xC0 == (doze_buf[2] & 0xC0)) {
454 dev_dbg(&ts->client->dev,
455 "double click to light up the screen!");
456 doze_status = DOZE_WAKEUP;
457 input_report_key(ts->input_dev, KEY_POWER, 1);
458 input_sync(ts->input_dev);
459 input_report_key(ts->input_dev, KEY_POWER, 0);
460 input_sync(ts->input_dev);
461 /* clear 0x814B */
462 doze_buf[2] = 0x00;
463 gtp_i2c_write(ts->client, doze_buf, 3);
464 } else {
465 gtp_enter_doze(ts);
466 }
467 }
468 if (ts->use_irq)
469 gtp_irq_enable(ts);
470
471 return;
472 }
Andrew Huangab45aab2013-06-27 10:06:56 +0000473#endif
474
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800475 ret = gtp_i2c_read(ts->client, point_data, 12);
476 if (ret < 0) {
477 dev_err(&ts->client->dev,
478 "I2C transfer error. errno:%d\n ", ret);
479 goto exit_work_func;
480 }
Andrew Huangab45aab2013-06-27 10:06:56 +0000481
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800482 finger = point_data[GTP_ADDR_LENGTH];
483 if ((finger & 0x80) == 0)
484 goto exit_work_func;
Andrew Huangab45aab2013-06-27 10:06:56 +0000485
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800486 touch_num = finger & 0x0f;
487 if (touch_num > GTP_MAX_TOUCH)
488 goto exit_work_func;
Andrew Huangab45aab2013-06-27 10:06:56 +0000489
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800490 if (touch_num > 1) {
491 u8 buf[8 * GTP_MAX_TOUCH] = { (GTP_READ_COOR_ADDR + 10) >> 8,
492 (GTP_READ_COOR_ADDR + 10) & 0xff };
Andrew Huangab45aab2013-06-27 10:06:56 +0000493
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800494 ret = gtp_i2c_read(ts->client, buf,
495 2 + 8 * (touch_num - 1));
496 memcpy(&point_data[12], &buf[2], 8 * (touch_num - 1));
497 }
Andrew Huangab45aab2013-06-27 10:06:56 +0000498
499#if GTP_HAVE_TOUCH_KEY
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800500 key_value = point_data[3 + 8 * touch_num];
Andrew Huangab45aab2013-06-27 10:06:56 +0000501
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800502 if (key_value || pre_key) {
503 for (i = 0; i < GTP_MAX_KEY_NUM; i++) {
Shantanu Jain7dc2b472013-09-27 11:55:09 +0530504
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800505#if GTP_DEBUG_ON
506 for (ret = 0; ret < 4; ++ret) {
507 if (key_codes[ret] == touch_key_array[i]) {
508 GTP_DEBUG("Key: %s %s",
509 key_names[ret],
510 (key_value & (0x01 << i))
511 ? "Down" : "Up");
512 break;
513 }
514 }
515#endif
516
517 input_report_key(ts->input_dev,
518 touch_key_array[i], key_value & (0x01<<i));
519 }
520 touch_num = 0;
521 pre_touch = 0;
522 }
523#endif
524 pre_key = key_value;
525
Andrew Huangab45aab2013-06-27 10:06:56 +0000526#if GTP_WITH_PEN
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800527 if (pre_pen && (touch_num == 0)) {
Shantanu Jain7dc2b472013-09-27 11:55:09 +0530528 dev_dbg(&ts->client->dev, "Pen touch UP(Slot)!");
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800529 input_report_key(ts->input_dev, BTN_TOOL_PEN, 0);
530 input_mt_slot(ts->input_dev, 5);
531 input_report_abs(ts->input_dev, ABS_MT_TRACKING_ID, -1);
532 pre_pen = 0;
533 }
Andrew Huangab45aab2013-06-27 10:06:56 +0000534#endif
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800535 if (pre_touch || touch_num) {
536 s32 pos = 0;
537 u16 touch_index = 0;
Andrew Huangab45aab2013-06-27 10:06:56 +0000538
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800539 coor_data = &point_data[3];
540 if (touch_num) {
541 id = coor_data[pos] & 0x0F;
542#if GTP_WITH_PEN
543 id = coor_data[pos];
544 if (id == 128) {
Shantanu Jain7dc2b472013-09-27 11:55:09 +0530545 dev_dbg(&ts->client->dev,
546 "Pen touch DOWN(Slot)!");
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800547 input_x = coor_data[pos + 1]
548 | (coor_data[pos + 2] << 8);
549 input_y = coor_data[pos + 3]
550 | (coor_data[pos + 4] << 8);
551 input_w = coor_data[pos + 5]
552 | (coor_data[pos + 6] << 8);
Andrew Huangab45aab2013-06-27 10:06:56 +0000553
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800554 input_report_key(ts->input_dev,
555 BTN_TOOL_PEN, 1);
556 input_mt_slot(ts->input_dev, 5);
557 input_report_abs(ts->input_dev,
558 ABS_MT_TRACKING_ID, 5);
559 input_report_abs(ts->input_dev,
560 ABS_MT_POSITION_X, input_x);
561 input_report_abs(ts->input_dev,
562 ABS_MT_POSITION_Y, input_y);
563 input_report_abs(ts->input_dev,
564 ABS_MT_TOUCH_MAJOR, input_w);
Shantanu Jain7dc2b472013-09-27 11:55:09 +0530565 dev_dbg(&ts->client->dev,
566 "Pen/Stylus: (%d, %d)[%d]",
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800567 input_x, input_y, input_w);
568 pre_pen = 1;
569 pre_touch = 0;
570 }
571#endif
572
573 touch_index |= (0x01<<id);
574 }
575
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800576 for (i = 0; i < GTP_MAX_TOUCH; i++) {
577#if GTP_WITH_PEN
578 if (pre_pen == 1)
579 break;
580#endif
581 if (touch_index & (0x01<<i)) {
582 input_x = coor_data[pos + 1] |
583 coor_data[pos + 2] << 8;
584 input_y = coor_data[pos + 3] |
585 coor_data[pos + 4] << 8;
586 input_w = coor_data[pos + 5] |
587 coor_data[pos + 6] << 8;
588
589 gtp_touch_down(ts, id,
590 input_x, input_y, input_w);
591 pre_touch |= 0x01 << i;
592
593 pos += 8;
594 id = coor_data[pos] & 0x0F;
595 touch_index |= (0x01<<id);
596 } else {
597 gtp_touch_up(ts, i);
598 pre_touch &= ~(0x01 << i);
599 }
600 }
601 }
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800602 input_sync(ts->input_dev);
Andrew Huangab45aab2013-06-27 10:06:56 +0000603
604exit_work_func:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800605 if (!ts->gtp_rawdiff_mode) {
606 ret = gtp_i2c_write(ts->client, end_cmd, 3);
607 if (ret < 0)
608 dev_warn(&ts->client->dev, "I2C write end_cmd error!\n");
609
610 }
611 if (ts->use_irq)
612 gtp_irq_enable(ts);
613
614 return;
Andrew Huangab45aab2013-06-27 10:06:56 +0000615}
616
617/*******************************************************
618Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800619 Timer interrupt service routine for polling mode.
Andrew Huangab45aab2013-06-27 10:06:56 +0000620Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800621 timer: timer struct pointer
Andrew Huangab45aab2013-06-27 10:06:56 +0000622Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800623 Timer work mode.
624 HRTIMER_NORESTART: no restart mode
Andrew Huangab45aab2013-06-27 10:06:56 +0000625*********************************************************/
626static enum hrtimer_restart goodix_ts_timer_handler(struct hrtimer *timer)
627{
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800628 struct goodix_ts_data
629 *ts = container_of(timer, struct goodix_ts_data, timer);
Andrew Huangab45aab2013-06-27 10:06:56 +0000630
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800631 queue_work(ts->goodix_wq, &ts->work);
632 hrtimer_start(&ts->timer, ktime_set(0, (GTP_POLL_TIME + 6) * 1000000),
633 HRTIMER_MODE_REL);
634 return HRTIMER_NORESTART;
Andrew Huangab45aab2013-06-27 10:06:56 +0000635}
636
637/*******************************************************
638Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800639 External interrupt service routine for interrupt mode.
Andrew Huangab45aab2013-06-27 10:06:56 +0000640Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800641 irq: interrupt number.
642 dev_id: private data pointer
Andrew Huangab45aab2013-06-27 10:06:56 +0000643Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800644 Handle Result.
645 IRQ_HANDLED: interrupt handled successfully
Andrew Huangab45aab2013-06-27 10:06:56 +0000646*********************************************************/
647static irqreturn_t goodix_ts_irq_handler(int irq, void *dev_id)
648{
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800649 struct goodix_ts_data *ts = dev_id;
Andrew Huangab45aab2013-06-27 10:06:56 +0000650
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800651 gtp_irq_disable(ts);
652
653 queue_work(ts->goodix_wq, &ts->work);
654
655 return IRQ_HANDLED;
Andrew Huangab45aab2013-06-27 10:06:56 +0000656}
657/*******************************************************
658Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800659 Synchronization.
Andrew Huangab45aab2013-06-27 10:06:56 +0000660Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800661 ms: synchronization time in millisecond.
Andrew Huangab45aab2013-06-27 10:06:56 +0000662Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800663 None.
Andrew Huangab45aab2013-06-27 10:06:56 +0000664*******************************************************/
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800665void gtp_int_sync(struct goodix_ts_data *ts, int ms)
Andrew Huangab45aab2013-06-27 10:06:56 +0000666{
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800667 gpio_direction_output(ts->pdata->irq_gpio, 0);
668 msleep(ms);
669 gpio_direction_input(ts->pdata->irq_gpio);
Andrew Huangab45aab2013-06-27 10:06:56 +0000670}
671
672/*******************************************************
673Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800674 Reset chip.
Andrew Huangab45aab2013-06-27 10:06:56 +0000675Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800676 ms: reset time in millisecond, must >10ms
Andrew Huangab45aab2013-06-27 10:06:56 +0000677Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800678 None.
Andrew Huangab45aab2013-06-27 10:06:56 +0000679*******************************************************/
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800680static void gtp_reset_guitar(struct goodix_ts_data *ts, int ms)
Andrew Huangab45aab2013-06-27 10:06:56 +0000681{
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800682 /* This reset sequence will selcet I2C slave address */
683 gpio_direction_output(ts->pdata->reset_gpio, 0);
684 msleep(ms);
Andrew Huangab45aab2013-06-27 10:06:56 +0000685
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800686 if (ts->client->addr == GTP_I2C_ADDRESS_HIGH)
687 gpio_direction_output(ts->pdata->irq_gpio, 1);
688 else
689 gpio_direction_output(ts->pdata->irq_gpio, 0);
Andrew Huangab45aab2013-06-27 10:06:56 +0000690
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800691 usleep(RESET_DELAY_T3_US);
692 gpio_direction_output(ts->pdata->reset_gpio, 1);
693 msleep(RESET_DELAY_T4);
Andrew Huangab45aab2013-06-27 10:06:56 +0000694
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800695 gpio_direction_input(ts->pdata->reset_gpio);
696
697 gtp_int_sync(ts, 50);
698
Andrew Huangab45aab2013-06-27 10:06:56 +0000699#if GTP_ESD_PROTECT
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800700 gtp_init_ext_watchdog(ts->client);
Andrew Huangab45aab2013-06-27 10:06:56 +0000701#endif
702}
703
Bingzhe Cai7d49bb32013-07-22 18:23:20 +0800704#if defined(CONFIG_HAS_EARLYSUSPEND) || defined(CONFIG_FB)
Andrew Huangab45aab2013-06-27 10:06:56 +0000705#if GTP_SLIDE_WAKEUP
706/*******************************************************
707Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800708 Enter doze mode for sliding wakeup.
Andrew Huangab45aab2013-06-27 10:06:56 +0000709Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800710 ts: goodix tp private data
Andrew Huangab45aab2013-06-27 10:06:56 +0000711Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800712 1: succeed, otherwise failed
Andrew Huangab45aab2013-06-27 10:06:56 +0000713*******************************************************/
714static s8 gtp_enter_doze(struct goodix_ts_data *ts)
715{
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800716 int ret = -1;
717 s8 retry = 0;
718 u8 i2c_control_buf[3] = {
719 (u8)(GTP_REG_SLEEP >> 8),
720 (u8)GTP_REG_SLEEP, 8};
Andrew Huangab45aab2013-06-27 10:06:56 +0000721
Andrew Huangab45aab2013-06-27 10:06:56 +0000722#if GTP_DBL_CLK_WAKEUP
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800723 i2c_control_buf[2] = 0x09;
Andrew Huangab45aab2013-06-27 10:06:56 +0000724#endif
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800725 gtp_irq_disable(ts);
Andrew Huangab45aab2013-06-27 10:06:56 +0000726
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800727 while (retry++ < 5) {
728 i2c_control_buf[0] = 0x80;
729 i2c_control_buf[1] = 0x46;
730 ret = gtp_i2c_write(ts->client, i2c_control_buf, 3);
731 if (ret < 0) {
Shantanu Jain7dc2b472013-09-27 11:55:09 +0530732 dev_err(&ts->client->dev,
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800733 "failed to set doze flag into 0x8046, %d",
734 retry);
735 continue;
736 }
737 i2c_control_buf[0] = 0x80;
738 i2c_control_buf[1] = 0x40;
739 ret = gtp_i2c_write(ts->client, i2c_control_buf, 3);
740 if (ret > 0) {
741 doze_status = DOZE_ENABLED;
742 dev_dbg(&ts->client->dev,
743 "GTP has been working in doze mode!");
744 gtp_irq_enable(ts);
745 return ret;
746 }
747 msleep(20);
748 }
749 dev_err(&ts->client->dev, "GTP send doze cmd failed.\n");
750 gtp_irq_enable(ts);
751 return ret;
Andrew Huangab45aab2013-06-27 10:06:56 +0000752}
Andrew Huangab45aab2013-06-27 10:06:56 +0000753#else
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800754/*******************************************************
755Function:
756 Enter sleep mode.
757Input:
758 ts: private data.
759Output:
760 Executive outcomes.
761 1: succeed, otherwise failed.
762*******************************************************/
763static s8 gtp_enter_sleep(struct goodix_ts_data *ts)
764{
765 int ret = -1;
766 s8 retry = 0;
767 u8 i2c_control_buf[3] = {
768 (u8)(GTP_REG_SLEEP >> 8),
769 (u8)GTP_REG_SLEEP, 5};
Andrew Huangab45aab2013-06-27 10:06:56 +0000770
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800771 ret = gpio_direction_output(ts->pdata->irq_gpio, 0);
772 usleep(5000);
773 while (retry++ < 5) {
774 ret = gtp_i2c_write(ts->client, i2c_control_buf, 3);
775 if (ret > 0) {
776 dev_dbg(&ts->client->dev,
777 "GTP enter sleep!");
778 return ret;
779 }
780 msleep(20);
781 }
782 dev_err(&ts->client->dev, "GTP send sleep cmd failed.\n");
783 return ret;
Andrew Huangab45aab2013-06-27 10:06:56 +0000784}
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800785#endif
Andrew Huangab45aab2013-06-27 10:06:56 +0000786
787/*******************************************************
788Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800789 Wakeup from sleep.
Andrew Huangab45aab2013-06-27 10:06:56 +0000790Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800791 ts: private data.
Andrew Huangab45aab2013-06-27 10:06:56 +0000792Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800793 Executive outcomes.
794 >0: succeed, otherwise: failed.
Andrew Huangab45aab2013-06-27 10:06:56 +0000795*******************************************************/
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800796static s8 gtp_wakeup_sleep(struct goodix_ts_data *ts)
Andrew Huangab45aab2013-06-27 10:06:56 +0000797{
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800798 u8 retry = 0;
799 s8 ret = -1;
800
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800801#if GTP_POWER_CTRL_SLEEP
Bingzhe Cai7d49bb32013-07-22 18:23:20 +0800802 gtp_reset_guitar(ts, 20);
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800803
Bingzhe Cai7d49bb32013-07-22 18:23:20 +0800804 ret = gtp_send_cfg(ts);
805 if (ret > 0) {
806 dev_dbg(&ts->client->dev,
807 "Wakeup sleep send config success.");
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800808 return 1;
809 }
810#else
811 while (retry++ < 10) {
812#if GTP_SLIDE_WAKEUP
813 /* wakeup not by slide */
814 if (DOZE_WAKEUP != doze_status)
815 gtp_reset_guitar(ts, 10);
816 else
817 /* wakeup by slide */
818 doze_status = DOZE_DISABLED;
819#else
820 if (chip_gt9xxs == 1) {
821 gtp_reset_guitar(ts, 10);
822 } else {
823 ret = gpio_direction_output(ts->pdata->irq_gpio, 1);
824 usleep(5000);
825 }
826#endif
827 ret = gtp_i2c_test(ts->client);
828 if (ret > 0) {
829 dev_dbg(&ts->client->dev, "GTP wakeup sleep.");
830#if (!GTP_SLIDE_WAKEUP)
831 if (chip_gt9xxs == 0) {
832 gtp_int_sync(ts, 25);
833 msleep(20);
834#if GTP_ESD_PROTECT
835 gtp_init_ext_watchdog(ts->client);
836#endif
837 }
838#endif
839 return ret;
840 }
841 gtp_reset_guitar(ts, 20);
842 }
843#endif
844
845 dev_err(&ts->client->dev, "GTP wakeup sleep failed.\n");
846 return ret;
847}
Bingzhe Cai7d49bb32013-07-22 18:23:20 +0800848#endif /* !CONFIG_HAS_EARLYSUSPEND && !CONFIG_FB*/
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800849
850/*******************************************************
851Function:
852 Initialize gtp.
853Input:
854 ts: goodix private data
855Output:
856 Executive outcomes.
857 > =0: succeed, otherwise: failed
858*******************************************************/
859static int gtp_init_panel(struct goodix_ts_data *ts)
860{
861 struct i2c_client *client = ts->client;
862 unsigned char *config_data;
863 int ret = -EIO;
Andrew Huangab45aab2013-06-27 10:06:56 +0000864
865#if GTP_DRIVER_SEND_CFG
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800866 int i;
867 u8 check_sum = 0;
868 u8 opr_buf[16];
869 u8 sensor_id = 0;
Andrew Huangab45aab2013-06-27 10:06:56 +0000870
Bingzhe Cai15d14462013-09-11 18:52:17 +0800871 for (i = 0; i < GOODIX_MAX_CFG_GROUP; i++)
872 dev_dbg(&client->dev, "Config Groups(%d) Lengths: %d",
873 i, ts->pdata->config_data_len[i]);
Andrew Huangab45aab2013-06-27 10:06:56 +0000874
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800875 ret = gtp_i2c_read_dbl_check(ts->client, 0x41E4, opr_buf, 1);
876 if (SUCCESS == ret) {
877 if (opr_buf[0] != 0xBE) {
878 ts->fw_error = 1;
879 dev_err(&client->dev,
880 "Firmware error, no config sent!");
881 return -EINVAL;
882 }
883 }
Bingzhe Cai15d14462013-09-11 18:52:17 +0800884
885 for (i = 1; i < GOODIX_MAX_CFG_GROUP; i++) {
886 if (ts->pdata->config_data_len[i])
887 break;
888 }
889 if (i == GOODIX_MAX_CFG_GROUP) {
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800890 sensor_id = 0;
891 } else {
892 ret = gtp_i2c_read_dbl_check(ts->client, GTP_REG_SENSOR_ID,
893 &sensor_id, 1);
894 if (SUCCESS == ret) {
Bingzhe Cai15d14462013-09-11 18:52:17 +0800895 if (sensor_id >= GOODIX_MAX_CFG_GROUP) {
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800896 dev_err(&client->dev,
897 "Invalid sensor_id(0x%02X), No Config Sent!",
898 sensor_id);
899 return -EINVAL;
900 }
901 } else {
902 dev_err(&client->dev,
903 "Failed to get sensor_id, No config sent!");
904 return -EINVAL;
905 }
906 }
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800907
Bingzhe Cai15d14462013-09-11 18:52:17 +0800908 dev_info(&client->dev, "Sensor ID selected: %d", sensor_id);
909
910 if (ts->pdata->config_data_len[sensor_id] < GTP_CONFIG_MIN_LENGTH ||
911 !ts->pdata->config_data[sensor_id]) {
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800912 dev_err(&client->dev,
Bingzhe Cai15d14462013-09-11 18:52:17 +0800913 "Sensor_ID(%d) matches with NULL or invalid config group!\n",
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800914 sensor_id);
915 return -EINVAL;
916 }
Bingzhe Cai15d14462013-09-11 18:52:17 +0800917
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800918 ret = gtp_i2c_read_dbl_check(ts->client, GTP_REG_CONFIG_DATA,
919 &opr_buf[0], 1);
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800920 if (ret == SUCCESS) {
921 if (opr_buf[0] < 90) {
922 /* backup group config version */
Bingzhe Cai15d14462013-09-11 18:52:17 +0800923 grp_cfg_version =
924 ts->pdata->config_data[sensor_id][GTP_ADDR_LENGTH];
925 ts->pdata->config_data[sensor_id][GTP_ADDR_LENGTH] =
926 0x00;
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800927 ts->fixed_cfg = 0;
928 } else {
929 /* treated as fixed config, not send config */
930 dev_warn(&client->dev,
931 "Ic fixed config with config version(%d, 0x%02X)",
932 opr_buf[0], opr_buf[0]);
933 ts->fixed_cfg = 1;
934 }
935 } else {
936 dev_err(&client->dev,
937 "Failed to get ic config version!No config sent!");
938 return -EINVAL;
939 }
940
Bingzhe Cai15d14462013-09-11 18:52:17 +0800941 config_data = ts->pdata->config_data[sensor_id];
942 ts->config_data = ts->pdata->config_data[sensor_id];
943 ts->gtp_cfg_len = ts->pdata->config_data_len[sensor_id];
Andrew Huangab45aab2013-06-27 10:06:56 +0000944
945#if GTP_CUSTOM_CFG
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800946 config_data[RESOLUTION_LOC] =
947 (unsigned char)(GTP_MAX_WIDTH && 0xFF);
948 config_data[RESOLUTION_LOC + 1] =
949 (unsigned char)(GTP_MAX_WIDTH >> 8);
950 config_data[RESOLUTION_LOC + 2] =
951 (unsigned char)(GTP_MAX_HEIGHT && 0xFF);
952 config_data[RESOLUTION_LOC + 3] =
953 (unsigned char)(GTP_MAX_HEIGHT >> 8);
Andrew Huangab45aab2013-06-27 10:06:56 +0000954
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800955 if (GTP_INT_TRIGGER == 0)
956 config_data[TRIGGER_LOC] &= 0xfe;
957 else if (GTP_INT_TRIGGER == 1)
958 config_data[TRIGGER_LOC] |= 0x01;
959#endif /* !GTP_CUSTOM_CFG */
Andrew Huangab45aab2013-06-27 10:06:56 +0000960
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800961 check_sum = 0;
962 for (i = GTP_ADDR_LENGTH; i < ts->gtp_cfg_len; i++)
963 check_sum += config_data[i];
964
965 config_data[ts->gtp_cfg_len] = (~check_sum) + 1;
966
967#else /* DRIVER NOT SEND CONFIG */
968 ts->gtp_cfg_len = GTP_CONFIG_MAX_LENGTH;
969 ret = gtp_i2c_read(ts->client, config_data,
970 ts->gtp_cfg_len + GTP_ADDR_LENGTH);
971 if (ret < 0) {
972 dev_err(&client->dev,
973 "Read Config Failed, Using DEFAULT Resolution & INT Trigger!\n");
974 ts->abs_x_max = GTP_MAX_WIDTH;
975 ts->abs_y_max = GTP_MAX_HEIGHT;
976 ts->int_trigger_type = GTP_INT_TRIGGER;
977 }
978#endif /* !DRIVER NOT SEND CONFIG */
979
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800980 if ((ts->abs_x_max == 0) && (ts->abs_y_max == 0)) {
981 ts->abs_x_max = (config_data[RESOLUTION_LOC + 1] << 8)
982 + config_data[RESOLUTION_LOC];
983 ts->abs_y_max = (config_data[RESOLUTION_LOC + 3] << 8)
984 + config_data[RESOLUTION_LOC + 2];
985 ts->int_trigger_type = (config_data[TRIGGER_LOC]) & 0x03;
986 }
987 ret = gtp_send_cfg(ts);
988 if (ret < 0)
989 dev_err(&client->dev, "%s: Send config error.\n", __func__);
990
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800991 msleep(20);
992 return ret;
Andrew Huangab45aab2013-06-27 10:06:56 +0000993}
994
995/*******************************************************
996Function:
Shantanu Jain3f80bef2013-09-18 12:05:02 +0530997 Read firmware version
Andrew Huangab45aab2013-06-27 10:06:56 +0000998Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +0800999 client: i2c device
1000 version: buffer to keep ic firmware version
Andrew Huangab45aab2013-06-27 10:06:56 +00001001Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001002 read operation return.
Shantanu Jain3f80bef2013-09-18 12:05:02 +05301003 0: succeed, otherwise: failed
Andrew Huangab45aab2013-06-27 10:06:56 +00001004*******************************************************/
Shantanu Jain3f80bef2013-09-18 12:05:02 +05301005static int gtp_read_fw_version(struct i2c_client *client, u16 *version)
Andrew Huangab45aab2013-06-27 10:06:56 +00001006{
Shantanu Jain3f80bef2013-09-18 12:05:02 +05301007 int ret = 0;
1008 u8 buf[GTP_FW_VERSION_BUFFER_MAXSIZE] = {
1009 GTP_REG_FW_VERSION >> 8, GTP_REG_FW_VERSION & 0xff };
Andrew Huangab45aab2013-06-27 10:06:56 +00001010
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001011 ret = gtp_i2c_read(client, buf, sizeof(buf));
1012 if (ret < 0) {
1013 dev_err(&client->dev, "GTP read version failed.\n");
Shantanu Jain3f80bef2013-09-18 12:05:02 +05301014 return -EIO;
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001015 }
Andrew Huangab45aab2013-06-27 10:06:56 +00001016
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001017 if (version)
Shantanu Jain3f80bef2013-09-18 12:05:02 +05301018 *version = (buf[3] << 8) | buf[2];
1019
1020 return ret;
1021}
1022/*******************************************************
1023Function:
1024 Read and check chip id.
1025Input:
1026 client: i2c device
1027Output:
1028 read operation return.
1029 0: succeed, otherwise: failed
1030*******************************************************/
1031static int gtp_check_product_id(struct i2c_client *client)
1032{
1033 int ret = 0;
1034 char product_id[GTP_PRODUCT_ID_MAXSIZE];
1035 struct goodix_ts_data *ts = i2c_get_clientdata(client);
1036 /* 04 bytes are used for the Product-id in the register space.*/
1037 u8 buf[GTP_PRODUCT_ID_BUFFER_MAXSIZE] = {
1038 GTP_REG_PRODUCT_ID >> 8, GTP_REG_PRODUCT_ID & 0xff };
1039
1040 ret = gtp_i2c_read(client, buf, sizeof(buf));
1041 if (ret < 0) {
1042 dev_err(&client->dev, "GTP read version failed.\n");
1043 return -EIO;
1044 }
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001045
1046 if (buf[5] == 0x00) {
Shantanu Jain3f80bef2013-09-18 12:05:02 +05301047 /* copy (GTP_PRODUCT_ID_MAXSIZE - 1) from buffer. Ex: 915 */
1048 strlcpy(product_id, &buf[2], GTP_PRODUCT_ID_MAXSIZE - 1);
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001049 } else {
1050 if (buf[5] == 'S' || buf[5] == 's')
1051 chip_gt9xxs = 1;
Shantanu Jain3f80bef2013-09-18 12:05:02 +05301052 /* copy GTP_PRODUCT_ID_MAXSIZE from buffer. Ex: 915s */
1053 strlcpy(product_id, &buf[2], GTP_PRODUCT_ID_MAXSIZE);
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001054 }
Shantanu Jain3f80bef2013-09-18 12:05:02 +05301055
1056 dev_info(&client->dev, "Goodix Product ID = %s\n", product_id);
1057
1058 if (!IS_ERR(ts->pdata->product_id))
1059 ret = strcmp(product_id, ts->pdata->product_id);
1060
1061 if (ret != 0)
1062 return -EINVAL;
1063
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001064 return ret;
Andrew Huangab45aab2013-06-27 10:06:56 +00001065}
1066
1067/*******************************************************
1068Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001069 I2c test Function.
Andrew Huangab45aab2013-06-27 10:06:56 +00001070Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001071 client:i2c client.
Andrew Huangab45aab2013-06-27 10:06:56 +00001072Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001073 Executive outcomes.
1074 2: succeed, otherwise failed.
Andrew Huangab45aab2013-06-27 10:06:56 +00001075*******************************************************/
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001076static int gtp_i2c_test(struct i2c_client *client)
Andrew Huangab45aab2013-06-27 10:06:56 +00001077{
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001078 u8 buf[3] = { GTP_REG_CONFIG_DATA >> 8, GTP_REG_CONFIG_DATA & 0xff };
1079 int retry = 5;
1080 int ret = -EIO;
1081
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001082 while (retry--) {
1083 ret = gtp_i2c_read(client, buf, 3);
1084 if (ret > 0)
1085 return ret;
1086 dev_err(&client->dev, "GTP i2c test failed time %d.\n", retry);
1087 msleep(20);
1088 }
1089 return ret;
Andrew Huangab45aab2013-06-27 10:06:56 +00001090}
1091
1092/*******************************************************
1093Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001094 Request gpio(INT & RST) ports.
Andrew Huangab45aab2013-06-27 10:06:56 +00001095Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001096 ts: private data.
Andrew Huangab45aab2013-06-27 10:06:56 +00001097Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001098 Executive outcomes.
1099 = 0: succeed, != 0: failed
Andrew Huangab45aab2013-06-27 10:06:56 +00001100*******************************************************/
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001101static int gtp_request_io_port(struct goodix_ts_data *ts)
Andrew Huangab45aab2013-06-27 10:06:56 +00001102{
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001103 struct i2c_client *client = ts->client;
1104 struct goodix_ts_platform_data *pdata = ts->pdata;
1105 int ret;
1106 if (gpio_is_valid(pdata->irq_gpio)) {
1107 ret = gpio_request(pdata->irq_gpio, "goodix_ts_irq_gpio");
1108 if (ret) {
1109 dev_err(&client->dev, "irq gpio request failed\n");
1110 goto pwr_off;
1111 }
1112 ret = gpio_direction_input(pdata->irq_gpio);
1113 if (ret) {
1114 dev_err(&client->dev,
1115 "set_direction for irq gpio failed\n");
1116 goto free_irq_gpio;
1117 }
1118 } else {
1119 dev_err(&client->dev, "irq gpio is invalid!\n");
1120 ret = -EINVAL;
1121 goto free_irq_gpio;
1122 }
Andrew Huangab45aab2013-06-27 10:06:56 +00001123
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001124 if (gpio_is_valid(pdata->reset_gpio)) {
1125 ret = gpio_request(pdata->reset_gpio, "goodix_ts__reset_gpio");
1126 if (ret) {
1127 dev_err(&client->dev, "reset gpio request failed\n");
1128 goto free_irq_gpio;
1129 }
Andrew Huangab45aab2013-06-27 10:06:56 +00001130
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001131 ret = gpio_direction_output(pdata->reset_gpio, 0);
1132 if (ret) {
1133 dev_err(&client->dev,
1134 "set_direction for reset gpio failed\n");
1135 goto free_reset_gpio;
1136 }
1137 } else {
1138 dev_err(&client->dev, "reset gpio is invalid!\n");
1139 ret = -EINVAL;
1140 goto free_reset_gpio;
1141 }
1142 gpio_direction_input(pdata->reset_gpio);
Andrew Huangab45aab2013-06-27 10:06:56 +00001143
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001144 return ret;
Andrew Huangab45aab2013-06-27 10:06:56 +00001145
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001146free_reset_gpio:
1147 if (gpio_is_valid(pdata->reset_gpio))
1148 gpio_free(pdata->reset_gpio);
1149free_irq_gpio:
1150 if (gpio_is_valid(pdata->irq_gpio))
1151 gpio_free(pdata->irq_gpio);
1152pwr_off:
1153 return ret;
Andrew Huangab45aab2013-06-27 10:06:56 +00001154}
1155
1156/*******************************************************
1157Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001158 Request interrupt.
Andrew Huangab45aab2013-06-27 10:06:56 +00001159Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001160 ts: private data.
Andrew Huangab45aab2013-06-27 10:06:56 +00001161Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001162 Executive outcomes.
1163 0: succeed, -1: failed.
Andrew Huangab45aab2013-06-27 10:06:56 +00001164*******************************************************/
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001165static int gtp_request_irq(struct goodix_ts_data *ts)
Andrew Huangab45aab2013-06-27 10:06:56 +00001166{
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001167 int ret;
1168 const u8 irq_table[] = GTP_IRQ_TAB;
Andrew Huangab45aab2013-06-27 10:06:56 +00001169
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001170 ret = request_irq(ts->client->irq, goodix_ts_irq_handler,
1171 irq_table[ts->int_trigger_type],
1172 ts->client->name, ts);
1173 if (ret) {
1174 dev_err(&ts->client->dev, "Request IRQ failed!ERRNO:%d.\n",
1175 ret);
1176 gpio_direction_input(ts->pdata->irq_gpio);
Andrew Huangab45aab2013-06-27 10:06:56 +00001177
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001178 hrtimer_init(&ts->timer, CLOCK_MONOTONIC,
1179 HRTIMER_MODE_REL);
1180 ts->timer.function = goodix_ts_timer_handler;
1181 hrtimer_start(&ts->timer, ktime_set(1, 0),
1182 HRTIMER_MODE_REL);
1183 ts->use_irq = false;
1184 return ret;
1185 } else {
1186 gtp_irq_disable(ts);
1187 ts->use_irq = true;
1188 return 0;
1189 }
Andrew Huangab45aab2013-06-27 10:06:56 +00001190}
1191
1192/*******************************************************
1193Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001194 Request input device Function.
Andrew Huangab45aab2013-06-27 10:06:56 +00001195Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001196 ts:private data.
Andrew Huangab45aab2013-06-27 10:06:56 +00001197Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001198 Executive outcomes.
1199 0: succeed, otherwise: failed.
Andrew Huangab45aab2013-06-27 10:06:56 +00001200*******************************************************/
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001201static int gtp_request_input_dev(struct goodix_ts_data *ts)
Andrew Huangab45aab2013-06-27 10:06:56 +00001202{
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001203 int ret;
1204 char phys[PHY_BUF_SIZE];
Andrew Huangab45aab2013-06-27 10:06:56 +00001205#if GTP_HAVE_TOUCH_KEY
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001206 int index = 0;
Andrew Huangab45aab2013-06-27 10:06:56 +00001207#endif
Andrew Huangab45aab2013-06-27 10:06:56 +00001208
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001209 ts->input_dev = input_allocate_device();
1210 if (ts->input_dev == NULL) {
1211 dev_err(&ts->client->dev,
1212 "Failed to allocate input device.\n");
1213 return -ENOMEM;
1214 }
1215
1216 ts->input_dev->evbit[0] =
1217 BIT_MASK(EV_SYN) | BIT_MASK(EV_KEY) | BIT_MASK(EV_ABS) ;
Bingzhe Cai6beb0112013-08-27 00:27:26 +08001218 set_bit(BTN_TOOL_FINGER, ts->input_dev->keybit);
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001219 __set_bit(INPUT_PROP_DIRECT, ts->input_dev->propbit);
1220 input_mt_init_slots(ts->input_dev, 10);/* in case of "out of memory" */
Andrew Huangab45aab2013-06-27 10:06:56 +00001221
1222#if GTP_HAVE_TOUCH_KEY
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001223 for (index = 0; index < GTP_MAX_KEY_NUM; index++) {
1224 input_set_capability(ts->input_dev,
1225 EV_KEY, touch_key_array[index]);
1226 }
Andrew Huangab45aab2013-06-27 10:06:56 +00001227#endif
1228
1229#if GTP_SLIDE_WAKEUP
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001230 input_set_capability(ts->input_dev, EV_KEY, KEY_POWER);
1231#endif
Andrew Huangab45aab2013-06-27 10:06:56 +00001232
1233#if GTP_WITH_PEN
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001234 /* pen support */
1235 __set_bit(BTN_TOOL_PEN, ts->input_dev->keybit);
1236 __set_bit(INPUT_PROP_DIRECT, ts->input_dev->propbit);
1237 __set_bit(INPUT_PROP_POINTER, ts->input_dev->propbit);
Andrew Huangab45aab2013-06-27 10:06:56 +00001238#endif
1239
1240#if GTP_CHANGE_X2Y
Shantanu Jain7dc2b472013-09-27 11:55:09 +05301241 swap(ts->abs_x_max, ts->abs_y_max);
Andrew Huangab45aab2013-06-27 10:06:56 +00001242#endif
1243
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001244 input_set_abs_params(ts->input_dev, ABS_MT_POSITION_X,
1245 0, ts->abs_x_max, 0, 0);
1246 input_set_abs_params(ts->input_dev, ABS_MT_POSITION_Y,
1247 0, ts->abs_y_max, 0, 0);
1248 input_set_abs_params(ts->input_dev, ABS_MT_WIDTH_MAJOR,
1249 0, 255, 0, 0);
1250 input_set_abs_params(ts->input_dev, ABS_MT_TOUCH_MAJOR,
1251 0, 255, 0, 0);
1252 input_set_abs_params(ts->input_dev, ABS_MT_TRACKING_ID,
1253 0, 255, 0, 0);
Andrew Huangab45aab2013-06-27 10:06:56 +00001254
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001255 snprintf(phys, PHY_BUF_SIZE, "input/ts");
1256 ts->input_dev->name = GOODIX_DEV_NAME;
1257 ts->input_dev->phys = phys;
1258 ts->input_dev->id.bustype = BUS_I2C;
1259 ts->input_dev->id.vendor = 0xDEAD;
1260 ts->input_dev->id.product = 0xBEEF;
1261 ts->input_dev->id.version = 10427;
Andrew Huangab45aab2013-06-27 10:06:56 +00001262
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001263 ret = input_register_device(ts->input_dev);
1264 if (ret) {
1265 dev_err(&ts->client->dev,
1266 "Register %s input device failed.\n",
1267 ts->input_dev->name);
1268 goto exit_free_inputdev;
1269 }
1270
1271 return 0;
1272
1273exit_free_inputdev:
1274 input_free_device(ts->input_dev);
1275 ts->input_dev = NULL;
1276 return ret;
Andrew Huangab45aab2013-06-27 10:06:56 +00001277}
1278
Bingzhe Caic68e6112013-07-22 00:12:48 +08001279static int reg_set_optimum_mode_check(struct regulator *reg, int load_uA)
1280{
1281 return (regulator_count_voltages(reg) > 0) ?
1282 regulator_set_optimum_mode(reg, load_uA) : 0;
1283}
1284
1285/**
1286 * goodix_power_on - Turn device power ON
1287 * @ts: driver private data
1288 *
1289 * Returns zero on success, else an error.
1290 */
1291static int goodix_power_on(struct goodix_ts_data *ts)
1292{
1293 int ret;
1294
1295 if (!IS_ERR(ts->avdd)) {
1296 ret = reg_set_optimum_mode_check(ts->avdd,
1297 GOODIX_VDD_LOAD_MAX_UA);
1298 if (ret < 0) {
1299 dev_err(&ts->client->dev,
1300 "Regulator avdd set_opt failed rc=%d\n", ret);
1301 goto err_set_opt_avdd;
1302 }
1303 ret = regulator_enable(ts->avdd);
1304 if (ret) {
1305 dev_err(&ts->client->dev,
1306 "Regulator avdd enable failed ret=%d\n", ret);
1307 goto err_enable_avdd;
1308 }
1309 }
1310
1311 if (!IS_ERR(ts->vdd)) {
1312 ret = regulator_set_voltage(ts->vdd, GOODIX_VTG_MIN_UV,
1313 GOODIX_VTG_MAX_UV);
1314 if (ret) {
1315 dev_err(&ts->client->dev,
1316 "Regulator set_vtg failed vdd ret=%d\n", ret);
1317 goto err_set_vtg_vdd;
1318 }
1319 ret = reg_set_optimum_mode_check(ts->vdd,
1320 GOODIX_VDD_LOAD_MAX_UA);
1321 if (ret < 0) {
1322 dev_err(&ts->client->dev,
1323 "Regulator vdd set_opt failed rc=%d\n", ret);
1324 goto err_set_opt_vdd;
1325 }
1326 ret = regulator_enable(ts->vdd);
1327 if (ret) {
1328 dev_err(&ts->client->dev,
1329 "Regulator vdd enable failed ret=%d\n", ret);
1330 goto err_enable_vdd;
1331 }
1332 }
1333
1334 if (!IS_ERR(ts->vcc_i2c)) {
1335 ret = regulator_set_voltage(ts->vcc_i2c, GOODIX_I2C_VTG_MIN_UV,
1336 GOODIX_I2C_VTG_MAX_UV);
1337 if (ret) {
1338 dev_err(&ts->client->dev,
1339 "Regulator set_vtg failed vcc_i2c ret=%d\n",
1340 ret);
1341 goto err_set_vtg_vcc_i2c;
1342 }
1343 ret = reg_set_optimum_mode_check(ts->vcc_i2c,
1344 GOODIX_VIO_LOAD_MAX_UA);
1345 if (ret < 0) {
1346 dev_err(&ts->client->dev,
1347 "Regulator vcc_i2c set_opt failed rc=%d\n",
1348 ret);
1349 goto err_set_opt_vcc_i2c;
1350 }
1351 ret = regulator_enable(ts->vcc_i2c);
1352 if (ret) {
1353 dev_err(&ts->client->dev,
1354 "Regulator vcc_i2c enable failed ret=%d\n",
1355 ret);
1356 regulator_disable(ts->vdd);
1357 goto err_enable_vcc_i2c;
1358 }
1359 }
1360
1361 return 0;
1362
1363err_enable_vcc_i2c:
1364err_set_opt_vcc_i2c:
1365 if (!IS_ERR(ts->vcc_i2c))
1366 regulator_set_voltage(ts->vcc_i2c, 0, GOODIX_I2C_VTG_MAX_UV);
1367err_set_vtg_vcc_i2c:
1368 if (!IS_ERR(ts->vdd))
1369 regulator_disable(ts->vdd);
1370err_enable_vdd:
1371err_set_opt_vdd:
1372 if (!IS_ERR(ts->vdd))
1373 regulator_set_voltage(ts->vdd, 0, GOODIX_VTG_MAX_UV);
1374err_set_vtg_vdd:
1375 if (!IS_ERR(ts->avdd))
1376 regulator_disable(ts->avdd);
1377err_enable_avdd:
1378err_set_opt_avdd:
1379 return ret;
1380}
1381
1382/**
1383 * goodix_power_off - Turn device power OFF
1384 * @ts: driver private data
1385 *
1386 * Returns zero on success, else an error.
1387 */
1388static int goodix_power_off(struct goodix_ts_data *ts)
1389{
1390 int ret;
1391
1392 if (!IS_ERR(ts->vcc_i2c)) {
1393 ret = regulator_set_voltage(ts->vcc_i2c, 0,
1394 GOODIX_I2C_VTG_MAX_UV);
1395 if (ret < 0)
1396 dev_err(&ts->client->dev,
1397 "Regulator vcc_i2c set_vtg failed ret=%d\n",
1398 ret);
1399 ret = regulator_disable(ts->vcc_i2c);
1400 if (ret)
1401 dev_err(&ts->client->dev,
1402 "Regulator vcc_i2c disable failed ret=%d\n",
1403 ret);
1404 }
1405
1406 if (!IS_ERR(ts->vdd)) {
1407 ret = regulator_set_voltage(ts->vdd, 0, GOODIX_VTG_MAX_UV);
1408 if (ret < 0)
1409 dev_err(&ts->client->dev,
1410 "Regulator vdd set_vtg failed ret=%d\n", ret);
1411 ret = regulator_disable(ts->vdd);
1412 if (ret)
1413 dev_err(&ts->client->dev,
1414 "Regulator vdd disable failed ret=%d\n", ret);
1415 }
1416
1417 if (!IS_ERR(ts->avdd)) {
1418 ret = regulator_disable(ts->avdd);
1419 if (ret)
1420 dev_err(&ts->client->dev,
1421 "Regulator avdd disable failed ret=%d\n", ret);
1422 }
1423
1424 return 0;
1425}
1426
1427/**
1428 * goodix_power_init - Initialize device power
1429 * @ts: driver private data
1430 *
1431 * Returns zero on success, else an error.
1432 */
1433static int goodix_power_init(struct goodix_ts_data *ts)
1434{
1435 int ret;
1436
1437 ts->avdd = regulator_get(&ts->client->dev, "avdd");
1438 if (IS_ERR(ts->avdd)) {
1439 ret = PTR_ERR(ts->avdd);
1440 dev_info(&ts->client->dev,
1441 "Regulator get failed avdd ret=%d\n", ret);
1442 }
1443
1444 ts->vdd = regulator_get(&ts->client->dev, "vdd");
1445 if (IS_ERR(ts->vdd)) {
1446 ret = PTR_ERR(ts->vdd);
1447 dev_info(&ts->client->dev,
1448 "Regulator get failed vdd ret=%d\n", ret);
1449 }
1450
1451 ts->vcc_i2c = regulator_get(&ts->client->dev, "vcc-i2c");
1452 if (IS_ERR(ts->vcc_i2c)) {
1453 ret = PTR_ERR(ts->vcc_i2c);
1454 dev_info(&ts->client->dev,
1455 "Regulator get failed vcc_i2c ret=%d\n", ret);
1456 }
1457
1458 return 0;
1459}
1460
1461/**
1462 * goodix_power_deinit - Deinitialize device power
1463 * @ts: driver private data
1464 *
1465 * Returns zero on success, else an error.
1466 */
1467static int goodix_power_deinit(struct goodix_ts_data *ts)
1468{
1469 regulator_put(ts->vdd);
1470 regulator_put(ts->vcc_i2c);
1471 regulator_put(ts->avdd);
1472
1473 return 0;
1474}
1475
Bingzhe Caid0361172013-07-18 19:22:11 +08001476static int goodix_ts_get_dt_coords(struct device *dev, char *name,
1477 struct goodix_ts_platform_data *pdata)
1478{
1479 struct property *prop;
1480 struct device_node *np = dev->of_node;
1481 int rc;
1482 u32 coords[GOODIX_COORDS_ARR_SIZE];
1483
1484 prop = of_find_property(np, name, NULL);
1485 if (!prop)
1486 return -EINVAL;
1487 if (!prop->value)
1488 return -ENODATA;
1489
1490 rc = of_property_read_u32_array(np, name, coords,
1491 GOODIX_COORDS_ARR_SIZE);
1492 if (rc && (rc != -EINVAL)) {
1493 dev_err(dev, "Unable to read %s\n", name);
1494 return rc;
1495 }
1496
1497 if (!strcmp(name, "goodix,panel-coords")) {
1498 pdata->panel_minx = coords[0];
1499 pdata->panel_miny = coords[1];
1500 pdata->panel_maxx = coords[2];
1501 pdata->panel_maxy = coords[3];
1502 } else if (!strcmp(name, "goodix,display-coords")) {
1503 pdata->x_min = coords[0];
1504 pdata->y_min = coords[1];
1505 pdata->x_max = coords[2];
1506 pdata->y_max = coords[3];
1507 } else {
1508 dev_err(dev, "unsupported property %s\n", name);
1509 return -EINVAL;
1510 }
1511
1512 return 0;
1513}
1514
1515static int goodix_parse_dt(struct device *dev,
1516 struct goodix_ts_platform_data *pdata)
1517{
1518 int rc;
1519 struct device_node *np = dev->of_node;
1520 struct property *prop;
1521 u32 temp_val, num_buttons;
1522 u32 button_map[MAX_BUTTONS];
Bingzhe Cai15d14462013-09-11 18:52:17 +08001523 char prop_name[PROP_NAME_SIZE];
1524 int i, read_cfg_num;
Bingzhe Caid0361172013-07-18 19:22:11 +08001525
1526 rc = goodix_ts_get_dt_coords(dev, "goodix,panel-coords", pdata);
1527 if (rc && (rc != -EINVAL))
1528 return rc;
1529
1530 rc = goodix_ts_get_dt_coords(dev, "goodix,display-coords", pdata);
1531 if (rc)
1532 return rc;
1533
1534 pdata->i2c_pull_up = of_property_read_bool(np,
1535 "goodix,i2c-pull-up");
1536
1537 pdata->no_force_update = of_property_read_bool(np,
1538 "goodix,no-force-update");
1539 /* reset, irq gpio info */
1540 pdata->reset_gpio = of_get_named_gpio_flags(np, "reset-gpios",
1541 0, &pdata->reset_gpio_flags);
1542 if (pdata->reset_gpio < 0)
1543 return pdata->reset_gpio;
1544
1545 pdata->irq_gpio = of_get_named_gpio_flags(np, "interrupt-gpios",
1546 0, &pdata->irq_gpio_flags);
1547 if (pdata->irq_gpio < 0)
1548 return pdata->irq_gpio;
1549
Shantanu Jain3f80bef2013-09-18 12:05:02 +05301550 rc = of_property_read_string(np, "goodix,product-id",
1551 &pdata->product_id);
1552 if (rc < 0 || strlen(pdata->product_id) > GTP_PRODUCT_ID_MAXSIZE)
Bingzhe Caid0361172013-07-18 19:22:11 +08001553 return rc;
1554
1555 prop = of_find_property(np, "goodix,button-map", NULL);
1556 if (prop) {
1557 num_buttons = prop->length / sizeof(temp_val);
1558 if (num_buttons > MAX_BUTTONS)
1559 return -EINVAL;
1560
1561 rc = of_property_read_u32_array(np,
1562 "goodix,button-map", button_map,
1563 num_buttons);
1564 if (rc) {
1565 dev_err(dev, "Unable to read key codes\n");
1566 return rc;
1567 }
1568 }
1569
Bingzhe Cai15d14462013-09-11 18:52:17 +08001570 read_cfg_num = 0;
1571 for (i = 0; i < GOODIX_MAX_CFG_GROUP; i++) {
1572 snprintf(prop_name, sizeof(prop_name), "goodix,cfg-data%d", i);
1573 prop = of_find_property(np, prop_name,
1574 &pdata->config_data_len[i]);
1575 if (!prop || !prop->value) {
1576 pdata->config_data_len[i] = 0;
1577 pdata->config_data[i] = NULL;
1578 continue;
1579 }
1580 pdata->config_data[i] = devm_kzalloc(dev,
1581 GTP_CONFIG_MAX_LENGTH + GTP_ADDR_LENGTH,
1582 GFP_KERNEL);
1583 if (!pdata->config_data[i]) {
1584 dev_err(dev,
1585 "Not enough memory for panel config data %d\n",
1586 i);
Bingzhe Caid0361172013-07-18 19:22:11 +08001587 return -ENOMEM;
1588 }
Bingzhe Cai15d14462013-09-11 18:52:17 +08001589 pdata->config_data[i][0] = GTP_REG_CONFIG_DATA >> 8;
1590 pdata->config_data[i][1] = GTP_REG_CONFIG_DATA & 0xff;
1591 memcpy(&pdata->config_data[i][GTP_ADDR_LENGTH],
1592 prop->value, pdata->config_data_len[i]);
1593 read_cfg_num++;
Bingzhe Caid0361172013-07-18 19:22:11 +08001594 }
Bingzhe Cai15d14462013-09-11 18:52:17 +08001595 dev_dbg(dev, "%d config data read from device tree.\n", read_cfg_num);
Bingzhe Caid0361172013-07-18 19:22:11 +08001596
1597 return 0;
1598}
Bingzhe Caic68e6112013-07-22 00:12:48 +08001599
Andrew Huangab45aab2013-06-27 10:06:56 +00001600/*******************************************************
1601Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001602 I2c probe.
Andrew Huangab45aab2013-06-27 10:06:56 +00001603Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001604 client: i2c device struct.
1605 id: device id.
Andrew Huangab45aab2013-06-27 10:06:56 +00001606Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001607 Executive outcomes.
1608 0: succeed.
Andrew Huangab45aab2013-06-27 10:06:56 +00001609*******************************************************/
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001610
1611static int goodix_ts_probe(struct i2c_client *client,
1612 const struct i2c_device_id *id)
Andrew Huangab45aab2013-06-27 10:06:56 +00001613{
Bingzhe Caid0361172013-07-18 19:22:11 +08001614 struct goodix_ts_platform_data *pdata;
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001615 struct goodix_ts_data *ts;
1616 u16 version_info;
1617 int ret;
Andrew Huangab45aab2013-06-27 10:06:56 +00001618
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001619 dev_dbg(&client->dev, "GTP I2C Address: 0x%02x\n", client->addr);
Bingzhe Caid0361172013-07-18 19:22:11 +08001620 if (client->dev.of_node) {
1621 pdata = devm_kzalloc(&client->dev,
1622 sizeof(struct goodix_ts_platform_data), GFP_KERNEL);
1623 if (!pdata) {
1624 dev_err(&client->dev,
1625 "GTP Failed to allocate memory for pdata\n");
1626 return -ENOMEM;
1627 }
1628
1629 ret = goodix_parse_dt(&client->dev, pdata);
1630 if (ret)
1631 return ret;
1632 } else {
1633 pdata = client->dev.platform_data;
1634 }
1635
1636 if (!pdata) {
1637 dev_err(&client->dev, "GTP invalid pdata\n");
1638 return -EINVAL;
1639 }
Andrew Huangab45aab2013-06-27 10:06:56 +00001640
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001641#if GTP_ESD_PROTECT
1642 i2c_connect_client = client;
1643#endif
Bingzhe Caid0361172013-07-18 19:22:11 +08001644
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001645 if (!i2c_check_functionality(client->adapter, I2C_FUNC_I2C)) {
1646 dev_err(&client->dev, "GTP I2C not supported\n");
1647 return -ENODEV;
1648 }
Andrew Huangab45aab2013-06-27 10:06:56 +00001649
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001650 ts = kzalloc(sizeof(*ts), GFP_KERNEL);
1651 if (!ts) {
1652 dev_err(&client->dev, "GTP not enough memory for ts\n");
1653 return -ENOMEM;
1654 }
Andrew Huangab45aab2013-06-27 10:06:56 +00001655
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001656 memset(ts, 0, sizeof(*ts));
1657 ts->client = client;
Bingzhe Caid0361172013-07-18 19:22:11 +08001658 ts->pdata = pdata;
1659 /* For 2.6.39 & later use spin_lock_init(&ts->irq_lock)
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001660 * For 2.6.39 & before, use ts->irq_lock = SPIN_LOCK_UNLOCKED
1661 */
1662 spin_lock_init(&ts->irq_lock);
1663 i2c_set_clientdata(client, ts);
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001664 ts->gtp_rawdiff_mode = 0;
1665
Bingzhe Caic68e6112013-07-22 00:12:48 +08001666 ret = goodix_power_init(ts);
1667 if (ret) {
1668 dev_err(&client->dev, "GTP power init failed\n");
1669 goto exit_free_client_data;
1670 }
1671
1672 ret = goodix_power_on(ts);
1673 if (ret) {
1674 dev_err(&client->dev, "GTP power on failed\n");
1675 goto exit_deinit_power;
1676 }
1677
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001678 ret = gtp_request_io_port(ts);
1679 if (ret) {
1680 dev_err(&client->dev, "GTP request IO port failed.\n");
1681 goto exit_power_off;
1682 }
1683
1684 gtp_reset_guitar(ts, 20);
1685
1686 ret = gtp_i2c_test(client);
1687 if (ret != 2) {
1688 dev_err(&client->dev, "I2C communication ERROR!\n");
1689 goto exit_free_io_port;
1690 }
Andrew Huangab45aab2013-06-27 10:06:56 +00001691
1692#if GTP_AUTO_UPDATE
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001693 ret = gup_init_update_proc(ts);
1694 if (ret < 0) {
1695 dev_err(&client->dev,
1696 "GTP Create firmware update thread error.\n");
1697 goto exit_free_io_port;
1698 }
Andrew Huangab45aab2013-06-27 10:06:56 +00001699#endif
Andrew Huangab45aab2013-06-27 10:06:56 +00001700
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001701 ret = gtp_init_panel(ts);
1702 if (ret < 0) {
1703 dev_err(&client->dev, "GTP init panel failed.\n");
1704 ts->abs_x_max = GTP_MAX_WIDTH;
1705 ts->abs_y_max = GTP_MAX_HEIGHT;
1706 ts->int_trigger_type = GTP_INT_TRIGGER;
1707 }
Andrew Huangab45aab2013-06-27 10:06:56 +00001708
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001709 ret = gtp_request_input_dev(ts);
1710 if (ret) {
1711 dev_err(&client->dev, "GTP request input dev failed.\n");
1712 goto exit_free_inputdev;
1713 }
1714
Bingzhe Cai7d49bb32013-07-22 18:23:20 +08001715#if defined(CONFIG_FB)
1716 ts->fb_notif.notifier_call = fb_notifier_callback;
1717 ret = fb_register_client(&ts->fb_notif);
1718 if (ret)
1719 dev_err(&ts->client->dev,
1720 "Unable to register fb_notifier: %d\n",
1721 ret);
1722#elif defined(CONFIG_HAS_EARLYSUSPEND)
1723 ts->early_suspend.level = EARLY_SUSPEND_LEVEL_BLANK_SCREEN + 1;
1724 ts->early_suspend.suspend = goodix_ts_early_suspend;
1725 ts->early_suspend.resume = goodix_ts_late_resume;
1726 register_early_suspend(&ts->early_suspend);
1727#endif
1728
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001729 ts->goodix_wq = create_singlethread_workqueue("goodix_wq");
1730 INIT_WORK(&ts->work, goodix_ts_work_func);
1731
1732 ret = gtp_request_irq(ts);
1733 if (ret < 0)
1734 dev_info(&client->dev, "GTP works in polling mode.\n");
1735 else
1736 dev_info(&client->dev, "GTP works in interrupt mode.\n");
1737
Shantanu Jain3f80bef2013-09-18 12:05:02 +05301738 ret = gtp_read_fw_version(client, &version_info);
1739 if (ret != 0)
1740 dev_err(&client->dev, "GTP firmware version read failed.\n");
1741
1742 ret = gtp_check_product_id(client);
1743 if (ret != 0) {
1744 dev_err(&client->dev, "GTP Product id doesn't match.\n");
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001745 goto exit_free_irq;
1746 }
1747 if (ts->use_irq)
1748 gtp_irq_enable(ts);
1749
Andrew Huangab45aab2013-06-27 10:06:56 +00001750#if GTP_CREATE_WR_NODE
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001751 init_wr_node(client);
Andrew Huangab45aab2013-06-27 10:06:56 +00001752#endif
Andrew Huangab45aab2013-06-27 10:06:56 +00001753
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001754#if GTP_ESD_PROTECT
1755 gtp_esd_switch(client, SWITCH_ON);
1756#endif
1757 init_done = true;
1758 return 0;
1759exit_free_irq:
Bingzhe Cai7d49bb32013-07-22 18:23:20 +08001760#if defined(CONFIG_FB)
1761 if (fb_unregister_client(&ts->fb_notif))
1762 dev_err(&client->dev,
1763 "Error occurred while unregistering fb_notifier.\n");
1764#elif defined(CONFIG_HAS_EARLYSUSPEND)
1765 unregister_early_suspend(&ts->early_suspend);
1766#endif
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001767 if (ts->use_irq)
1768 free_irq(client->irq, ts);
1769 else
1770 hrtimer_cancel(&ts->timer);
1771 cancel_work_sync(&ts->work);
1772 flush_workqueue(ts->goodix_wq);
1773 destroy_workqueue(ts->goodix_wq);
1774
1775 input_unregister_device(ts->input_dev);
1776 if (ts->input_dev) {
1777 input_free_device(ts->input_dev);
1778 ts->input_dev = NULL;
1779 }
1780exit_free_inputdev:
1781 kfree(ts->config_data);
1782exit_free_io_port:
Bingzhe Caid0361172013-07-18 19:22:11 +08001783 if (gpio_is_valid(pdata->reset_gpio))
1784 gpio_free(pdata->reset_gpio);
1785 if (gpio_is_valid(pdata->irq_gpio))
1786 gpio_free(pdata->irq_gpio);
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001787exit_power_off:
Bingzhe Caic68e6112013-07-22 00:12:48 +08001788 goodix_power_off(ts);
1789exit_deinit_power:
1790 goodix_power_deinit(ts);
1791exit_free_client_data:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001792 i2c_set_clientdata(client, NULL);
1793 kfree(ts);
1794 return ret;
1795}
Andrew Huangab45aab2013-06-27 10:06:56 +00001796
1797/*******************************************************
1798Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001799 Goodix touchscreen driver release function.
Andrew Huangab45aab2013-06-27 10:06:56 +00001800Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001801 client: i2c device struct.
Andrew Huangab45aab2013-06-27 10:06:56 +00001802Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001803 Executive outcomes. 0---succeed.
Andrew Huangab45aab2013-06-27 10:06:56 +00001804*******************************************************/
1805static int goodix_ts_remove(struct i2c_client *client)
1806{
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001807 struct goodix_ts_data *ts = i2c_get_clientdata(client);
1808
Bingzhe Cai7d49bb32013-07-22 18:23:20 +08001809#if defined(CONFIG_FB)
1810 if (fb_unregister_client(&ts->fb_notif))
1811 dev_err(&client->dev,
1812 "Error occurred while unregistering fb_notifier.\n");
1813#elif defined(CONFIG_HAS_EARLYSUSPEND)
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001814 unregister_early_suspend(&ts->early_suspend);
Andrew Huangab45aab2013-06-27 10:06:56 +00001815#endif
1816
1817#if GTP_CREATE_WR_NODE
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001818 uninit_wr_node();
Andrew Huangab45aab2013-06-27 10:06:56 +00001819#endif
1820
1821#if GTP_ESD_PROTECT
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001822 cancel_work_sync(gtp_esd_check_workqueue);
1823 flush_workqueue(gtp_esd_check_workqueue);
1824 destroy_workqueue(gtp_esd_check_workqueue);
Andrew Huangab45aab2013-06-27 10:06:56 +00001825#endif
1826
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001827 if (ts) {
1828 if (ts->use_irq)
1829 free_irq(client->irq, ts);
1830 else
1831 hrtimer_cancel(&ts->timer);
Andrew Huangab45aab2013-06-27 10:06:56 +00001832
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001833 cancel_work_sync(&ts->work);
1834 flush_workqueue(ts->goodix_wq);
1835 destroy_workqueue(ts->goodix_wq);
1836
1837 input_unregister_device(ts->input_dev);
1838 if (ts->input_dev) {
1839 input_free_device(ts->input_dev);
1840 ts->input_dev = NULL;
1841 }
1842 kfree(ts->config_data);
1843
1844 if (gpio_is_valid(ts->pdata->reset_gpio))
1845 gpio_free(ts->pdata->reset_gpio);
1846 if (gpio_is_valid(ts->pdata->irq_gpio))
1847 gpio_free(ts->pdata->irq_gpio);
1848
Bingzhe Caic68e6112013-07-22 00:12:48 +08001849 goodix_power_off(ts);
1850 goodix_power_deinit(ts);
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001851 i2c_set_clientdata(client, NULL);
1852 kfree(ts);
1853 }
1854
1855 return 0;
Andrew Huangab45aab2013-06-27 10:06:56 +00001856}
1857
Bingzhe Cai7d49bb32013-07-22 18:23:20 +08001858#if defined(CONFIG_HAS_EARLYSUSPEND) || defined(CONFIG_FB)
Andrew Huangab45aab2013-06-27 10:06:56 +00001859/*******************************************************
1860Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001861 Early suspend function.
Andrew Huangab45aab2013-06-27 10:06:56 +00001862Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001863 h: early_suspend struct.
Andrew Huangab45aab2013-06-27 10:06:56 +00001864Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001865 None.
Andrew Huangab45aab2013-06-27 10:06:56 +00001866*******************************************************/
Bingzhe Cai7d49bb32013-07-22 18:23:20 +08001867static void goodix_ts_suspend(struct goodix_ts_data *ts)
Andrew Huangab45aab2013-06-27 10:06:56 +00001868{
Shantanu Jain33b1aee2013-08-05 16:52:47 +05301869 int ret = -1, i;
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001870
Andrew Huangab45aab2013-06-27 10:06:56 +00001871#if GTP_ESD_PROTECT
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001872 ts->gtp_is_suspend = 1;
1873 gtp_esd_switch(ts->client, SWITCH_OFF);
Andrew Huangab45aab2013-06-27 10:06:56 +00001874#endif
1875
1876#if GTP_SLIDE_WAKEUP
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001877 ret = gtp_enter_doze(ts);
Andrew Huangab45aab2013-06-27 10:06:56 +00001878#else
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001879 if (ts->use_irq)
1880 gtp_irq_disable(ts);
1881 else
1882 hrtimer_cancel(&ts->timer);
Shantanu Jain33b1aee2013-08-05 16:52:47 +05301883
1884 for (i = 0; i < GTP_MAX_TOUCH; i++)
1885 gtp_touch_up(ts, i);
1886
Shantanu Jain33b1aee2013-08-05 16:52:47 +05301887 input_sync(ts->input_dev);
1888
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001889 ret = gtp_enter_sleep(ts);
1890#endif
1891 if (ret < 0)
1892 dev_err(&ts->client->dev, "GTP early suspend failed.\n");
1893 /* to avoid waking up while not sleeping,
1894 * delay 48 + 10ms to ensure reliability
1895 */
1896 msleep(58);
Andrew Huangab45aab2013-06-27 10:06:56 +00001897}
1898
1899/*******************************************************
1900Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001901 Late resume function.
Andrew Huangab45aab2013-06-27 10:06:56 +00001902Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001903 h: early_suspend struct.
Andrew Huangab45aab2013-06-27 10:06:56 +00001904Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001905 None.
Andrew Huangab45aab2013-06-27 10:06:56 +00001906*******************************************************/
Bingzhe Cai7d49bb32013-07-22 18:23:20 +08001907static void goodix_ts_resume(struct goodix_ts_data *ts)
Andrew Huangab45aab2013-06-27 10:06:56 +00001908{
Bingzhe Cai7d49bb32013-07-22 18:23:20 +08001909 int ret = -1;
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001910
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001911 ret = gtp_wakeup_sleep(ts);
Andrew Huangab45aab2013-06-27 10:06:56 +00001912
1913#if GTP_SLIDE_WAKEUP
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001914 doze_status = DOZE_DISABLED;
Andrew Huangab45aab2013-06-27 10:06:56 +00001915#endif
1916
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001917 if (ret < 0)
Bingzhe Cai7d49bb32013-07-22 18:23:20 +08001918 dev_err(&ts->client->dev, "GTP resume failed.\n");
Andrew Huangab45aab2013-06-27 10:06:56 +00001919
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001920 if (ts->use_irq)
1921 gtp_irq_enable(ts);
1922 else
1923 hrtimer_start(&ts->timer,
1924 ktime_set(1, 0), HRTIMER_MODE_REL);
Andrew Huangab45aab2013-06-27 10:06:56 +00001925
1926#if GTP_ESD_PROTECT
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001927 ts->gtp_is_suspend = 0;
1928 gtp_esd_switch(ts->client, SWITCH_ON);
Andrew Huangab45aab2013-06-27 10:06:56 +00001929#endif
1930}
Bingzhe Cai7d49bb32013-07-22 18:23:20 +08001931
1932#if defined(CONFIG_FB)
1933static int fb_notifier_callback(struct notifier_block *self,
1934 unsigned long event, void *data)
1935{
1936 struct fb_event *evdata = data;
1937 int *blank;
1938 struct goodix_ts_data *ts =
1939 container_of(self, struct goodix_ts_data, fb_notif);
1940
1941 if (evdata && evdata->data && event == FB_EVENT_BLANK &&
1942 ts && ts->client) {
1943 blank = evdata->data;
1944 if (*blank == FB_BLANK_UNBLANK)
1945 goodix_ts_resume(ts);
1946 else if (*blank == FB_BLANK_POWERDOWN)
1947 goodix_ts_suspend(ts);
1948 }
1949
1950 return 0;
1951}
1952#elif defined(CONFIG_HAS_EARLYSUSPEND)
1953/*******************************************************
1954Function:
1955 Early suspend function.
1956Input:
1957 h: early_suspend struct.
1958Output:
1959 None.
1960*******************************************************/
1961static void goodix_ts_early_suspend(struct early_suspend *h)
1962{
1963 struct goodix_ts_data *ts;
1964
1965 ts = container_of(h, struct goodix_ts_data, early_suspend);
1966 goodix_ts_suspend(ts);
1967 return;
1968}
1969
1970/*******************************************************
1971Function:
1972 Late resume function.
1973Input:
1974 h: early_suspend struct.
1975Output:
1976 None.
1977*******************************************************/
1978static void goodix_ts_late_resume(struct early_suspend *h)
1979{
1980 struct goodix_ts_data *ts;
1981
1982 ts = container_of(h, struct goodix_ts_data, early_suspend);
1983 goodix_ts_late_resume(ts);
1984 return;
1985}
Andrew Huangab45aab2013-06-27 10:06:56 +00001986#endif
Bingzhe Cai7d49bb32013-07-22 18:23:20 +08001987#endif /* !CONFIG_HAS_EARLYSUSPEND && !CONFIG_FB*/
Andrew Huangab45aab2013-06-27 10:06:56 +00001988
1989#if GTP_ESD_PROTECT
1990/*******************************************************
1991Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001992 switch on & off esd delayed work
Andrew Huangab45aab2013-06-27 10:06:56 +00001993Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001994 client: i2c device
1995 on: SWITCH_ON / SWITCH_OFF
Andrew Huangab45aab2013-06-27 10:06:56 +00001996Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001997 void
Andrew Huangab45aab2013-06-27 10:06:56 +00001998*********************************************************/
Bingzhe Cai83caeea2013-07-16 17:57:37 +08001999void gtp_esd_switch(struct i2c_client *client, int on)
Andrew Huangab45aab2013-06-27 10:06:56 +00002000{
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002001 struct goodix_ts_data *ts;
2002
2003 ts = i2c_get_clientdata(client);
2004 if (SWITCH_ON == on) {
2005 /* switch on esd */
2006 if (!ts->esd_running) {
2007 ts->esd_running = 1;
2008 dev_dbg(&client->dev, "Esd started\n");
2009 queue_delayed_work(gtp_esd_check_workqueue,
2010 &gtp_esd_check_work, GTP_ESD_CHECK_CIRCLE);
2011 }
2012 } else {
2013 /* switch off esd */
2014 if (ts->esd_running) {
2015 ts->esd_running = 0;
2016 dev_dbg(&client->dev, "Esd cancelled\n");
2017 cancel_delayed_work_sync(&gtp_esd_check_work);
2018 }
2019 }
Andrew Huangab45aab2013-06-27 10:06:56 +00002020}
2021
2022/*******************************************************
2023Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002024 Initialize external watchdog for esd protect
Andrew Huangab45aab2013-06-27 10:06:56 +00002025Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002026 client: i2c device.
Andrew Huangab45aab2013-06-27 10:06:56 +00002027Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002028 result of i2c write operation.
2029 1: succeed, otherwise: failed
Andrew Huangab45aab2013-06-27 10:06:56 +00002030*********************************************************/
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002031static int gtp_init_ext_watchdog(struct i2c_client *client)
Andrew Huangab45aab2013-06-27 10:06:56 +00002032{
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002033 /* in case of recursively reset by calling gtp_i2c_write*/
2034 struct i2c_msg msg;
2035 u8 opr_buffer[4] = {0x80, 0x40, 0xAA, 0xAA};
2036 int ret;
2037 int retries = 0;
Andrew Huangab45aab2013-06-27 10:06:56 +00002038
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002039 msg.flags = !I2C_M_RD;
2040 msg.addr = client->addr;
2041 msg.len = 4;
2042 msg.buf = opr_buffer;
Andrew Huangab45aab2013-06-27 10:06:56 +00002043
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002044 while (retries < 5) {
2045 ret = i2c_transfer(client->adapter, &msg, 1);
2046 if (ret == 1)
2047 return 1;
2048 retries++;
2049 }
2050 if (retries >= 5)
2051 dev_err(&client->dev, "init external watchdog failed!");
2052 return 0;
Andrew Huangab45aab2013-06-27 10:06:56 +00002053}
2054
2055/*******************************************************
2056Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002057 Esd protect function.
2058 Added external watchdog by meta, 2013/03/07
Andrew Huangab45aab2013-06-27 10:06:56 +00002059Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002060 work: delayed work
Andrew Huangab45aab2013-06-27 10:06:56 +00002061Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002062 None.
Andrew Huangab45aab2013-06-27 10:06:56 +00002063*******************************************************/
2064static void gtp_esd_check_func(struct work_struct *work)
2065{
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002066 s32 i;
2067 s32 ret = -1;
2068 struct goodix_ts_data *ts = NULL;
2069 u8 test[4] = {0x80, 0x40};
Andrew Huangab45aab2013-06-27 10:06:56 +00002070
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002071 ts = i2c_get_clientdata(i2c_connect_client);
2072
2073 if (ts->gtp_is_suspend) {
2074 dev_dbg(&ts->client->dev, "Esd terminated!\n");
2075 ts->esd_running = 0;
2076 return;
2077 }
2078#ifdef CONFIG_GT9XX_TOUCHPANEL_UPDATE
2079 if (ts->enter_update)
2080 return;
2081#endif
2082
2083 for (i = 0; i < 3; i++) {
2084 ret = gtp_i2c_read(ts->client, test, 4);
2085
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002086 if ((ret < 0)) {
2087 /* IC works abnormally..*/
2088 continue;
2089 } else {
2090 if ((test[2] == 0xAA) || (test[3] != 0xAA)) {
2091 /* IC works abnormally..*/
2092 i = 3;
2093 break;
2094 } else {
2095 /* IC works normally, Write 0x8040 0xAA*/
2096 test[2] = 0xAA;
2097 gtp_i2c_write(ts->client, test, 3);
2098 break;
2099 }
2100 }
2101 }
2102 if (i >= 3) {
2103 dev_err(&ts->client->dev,
2104 "IC Working ABNORMALLY, Resetting Guitar...\n");
2105 gtp_reset_guitar(ts, 50);
2106 }
2107
2108 if (!ts->gtp_is_suspend)
2109 queue_delayed_work(gtp_esd_check_workqueue,
2110 &gtp_esd_check_work, GTP_ESD_CHECK_CIRCLE);
2111 else {
2112 dev_dbg(&ts->client->dev, "Esd terminated!\n");
2113 ts->esd_running = 0;
2114 }
2115
2116 return;
Andrew Huangab45aab2013-06-27 10:06:56 +00002117}
2118#endif
2119
2120static const struct i2c_device_id goodix_ts_id[] = {
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002121 { GTP_I2C_NAME, 0 },
2122 { }
Andrew Huangab45aab2013-06-27 10:06:56 +00002123};
2124
Bingzhe Caid0361172013-07-18 19:22:11 +08002125static struct of_device_id goodix_match_table[] = {
2126 { .compatible = "goodix,gt9xx", },
2127 { },
2128};
2129
Andrew Huangab45aab2013-06-27 10:06:56 +00002130static struct i2c_driver goodix_ts_driver = {
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002131 .probe = goodix_ts_probe,
2132 .remove = goodix_ts_remove,
2133#ifdef CONFIG_HAS_EARLYSUSPEND
2134 .suspend = goodix_ts_early_suspend,
2135 .resume = goodix_ts_late_resume,
Andrew Huangab45aab2013-06-27 10:06:56 +00002136#endif
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002137 .id_table = goodix_ts_id,
2138 .driver = {
2139 .name = GTP_I2C_NAME,
2140 .owner = THIS_MODULE,
Bingzhe Caid0361172013-07-18 19:22:11 +08002141 .of_match_table = goodix_match_table,
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002142 },
Andrew Huangab45aab2013-06-27 10:06:56 +00002143};
2144
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002145/*******************************************************
Andrew Huangab45aab2013-06-27 10:06:56 +00002146Function:
2147 Driver Install function.
2148Input:
2149 None.
2150Output:
2151 Executive Outcomes. 0---succeed.
2152********************************************************/
2153static int __devinit goodix_ts_init(void)
2154{
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002155 int ret;
Andrew Huangab45aab2013-06-27 10:06:56 +00002156
Andrew Huangab45aab2013-06-27 10:06:56 +00002157#if GTP_ESD_PROTECT
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002158 INIT_DELAYED_WORK(&gtp_esd_check_work, gtp_esd_check_func);
2159 gtp_esd_check_workqueue = create_workqueue("gtp_esd_check");
Andrew Huangab45aab2013-06-27 10:06:56 +00002160#endif
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002161 ret = i2c_add_driver(&goodix_ts_driver);
2162 return ret;
Andrew Huangab45aab2013-06-27 10:06:56 +00002163}
2164
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002165/*******************************************************
Andrew Huangab45aab2013-06-27 10:06:56 +00002166Function:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002167 Driver uninstall function.
Andrew Huangab45aab2013-06-27 10:06:56 +00002168Input:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002169 None.
Andrew Huangab45aab2013-06-27 10:06:56 +00002170Output:
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002171 Executive Outcomes. 0---succeed.
Andrew Huangab45aab2013-06-27 10:06:56 +00002172********************************************************/
2173static void __exit goodix_ts_exit(void)
2174{
Bingzhe Cai83caeea2013-07-16 17:57:37 +08002175 i2c_del_driver(&goodix_ts_driver);
Andrew Huangab45aab2013-06-27 10:06:56 +00002176}
2177
2178late_initcall(goodix_ts_init);
2179module_exit(goodix_ts_exit);
2180
2181MODULE_DESCRIPTION("GTP Series Driver");
2182MODULE_LICENSE("GPL");