blob: d8881a4f32c0494a95a5cb142058ef544a5766e9 [file] [log] [blame]
Steve Mucklef132c6c2012-06-06 18:30:57 -07001/* Source for:
2 * Cypress TrueTouch(TM) Standard Product I2C touchscreen driver.
3 * drivers/input/touchscreen/cyttsp-i2c.c
4 *
5 * Copyright (C) 2009, 2010 Cypress Semiconductor, Inc.
Amy Maloche304d9f52012-06-14 14:22:50 -07006 * Copyright (c) 2012, Code Aurora Forum. All rights reserved.
Steve Mucklef132c6c2012-06-06 18:30:57 -07007 *
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License
10 * version 2, and only version 2, as published by the
11 * Free Software Foundation.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License along
19 * with this program; if not, write to the Free Software Foundation, Inc.,
20 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
21 *
22 * Cypress reserves the right to make changes without further notice
23 * to the materials described herein. Cypress does not assume any
24 * liability arising out of the application described herein.
25 *
26 * Contact Cypress Semiconductor at www.cypress.com
27 *
28 */
29
30#include <linux/delay.h>
31#include <linux/init.h>
32#include <linux/module.h>
33#include <linux/i2c.h>
34#include <linux/input.h>
35#include <linux/slab.h>
36#include <linux/gpio.h>
37#include <linux/irq.h>
38#include <linux/interrupt.h>
39#include <linux/timer.h>
40#include <linux/byteorder/generic.h>
41#include <linux/bitops.h>
42#include <linux/pm_runtime.h>
43#include <linux/firmware.h>
44#include <linux/mutex.h>
Amy Maloche304d9f52012-06-14 14:22:50 -070045#include <linux/completion.h>
Steve Mucklef132c6c2012-06-06 18:30:57 -070046#include <linux/regulator/consumer.h>
47#ifdef CONFIG_HAS_EARLYSUSPEND
48#include <linux/earlysuspend.h>
49#endif /* CONFIG_HAS_EARLYSUSPEND */
50
51#define CY_DECLARE_GLOBALS
52
53#include <linux/cyttsp-qc.h>
54
55uint32_t cyttsp_tsdebug1 = 0xff;
56module_param_named(tsdebug1, cyttsp_tsdebug1, uint, 0664);
57
58#define FW_FNAME_LEN 40
59#define TTSP_BUFF_SIZE 50
60
Amy Maloche304d9f52012-06-14 14:22:50 -070061enum cyttsp_powerstate {
62 CY_IDLE = 0, /* IC cannot be reached */
63 CY_READY, /* pre-operational; ready to go to ACTIVE */
64 CY_ACTIVE, /* app is running, IC is scanning */
65 CY_LOW_PWR, /* not currently used */
66 CY_SLEEP, /* app is running, IC is idle */
67 CY_BL, /* bootloader is running */
68 CY_LDR, /* loader is running */
69 CY_SYSINFO, /* switching to sysinfo mode */
70 CY_INVALID, /* always last in the list */
71};
72static char *cyttsp_powerstate_string[] = {
73 /* Order must match enum cyttsp_powerstate above */
74 "IDLE",
75 "READY",
76 "ACTIVE",
77 "LOW_PWR",
78 "SLEEP",
79 "BOOTLOADER",
80 "LOADER",
81 "SYSINFO",
82 "INVALID",
83};
84
Steve Mucklef132c6c2012-06-06 18:30:57 -070085/* CY TTSP I2C Driver private data */
86struct cyttsp {
87 struct i2c_client *client;
88 struct input_dev *input;
89 struct timer_list timer;
90 struct mutex mutex;
91 char phys[32];
92 struct cyttsp_platform_data *platform_data;
93 u8 num_prv_st_tch;
Amy Maloche304d9f52012-06-14 14:22:50 -070094 u8 power_settings[3];
Steve Mucklef132c6c2012-06-06 18:30:57 -070095 u16 fw_start_addr;
Amy Maloche304d9f52012-06-14 14:22:50 -070096 enum cyttsp_powerstate power_state;
Steve Mucklef132c6c2012-06-06 18:30:57 -070097 u16 act_trk[CY_NUM_TRK_ID];
98 u16 prv_st_tch[CY_NUM_ST_TCH_ID];
99 u16 prv_mt_tch[CY_NUM_MT_TCH_ID];
100 u16 prv_mt_pos[CY_NUM_TRK_ID][2];
101 atomic_t irq_enabled;
Amy Maloche304d9f52012-06-14 14:22:50 -0700102 struct completion si_int_running;
Steve Mucklef132c6c2012-06-06 18:30:57 -0700103 bool cyttsp_update_fw;
104 bool cyttsp_fwloader_mode;
105 bool is_suspended;
106 struct regulator **vdd;
107 char fw_fname[FW_FNAME_LEN];
108#ifdef CONFIG_HAS_EARLYSUSPEND
109 struct early_suspend early_suspend;
110#endif /* CONFIG_HAS_EARLYSUSPEND */
111};
112static u8 irq_cnt; /* comparison counter with register valuw */
113static u32 irq_cnt_total; /* total interrupts */
114static u32 irq_err_cnt; /* count number of touch interrupts with err */
115#define CY_IRQ_CNT_MASK 0x000000FF /* mapped for sizeof count in reg */
116#define CY_IRQ_CNT_REG 0x00 /* tt_undef[0]=reg 0x1B - Gen3 only */
117
118#ifdef CONFIG_HAS_EARLYSUSPEND
119static void cyttsp_early_suspend(struct early_suspend *handler);
120static void cyttsp_late_resume(struct early_suspend *handler);
121#endif /* CONFIG_HAS_EARLYSUSPEND */
122
123
124/* ****************************************************************************
125 * Prototypes for static functions
126 * ************************************************************************** */
127static irqreturn_t cyttsp_irq(int irq, void *handle);
128static int cyttsp_inlist(u16 prev_track[],
129 u8 cur_trk_id, u8 *prev_loc, u8 num_touches);
130static int cyttsp_next_avail_inlist(u16 cur_trk[],
131 u8 *new_loc, u8 num_touches);
132static int cyttsp_putbl(struct cyttsp *ts, int show,
133 int show_status, int show_version, int show_cid);
134static int __devinit cyttsp_probe(struct i2c_client *client,
135 const struct i2c_device_id *id);
136static int __devexit cyttsp_remove(struct i2c_client *client);
137static int cyttsp_resume(struct device *dev);
138static int cyttsp_suspend(struct device *dev);
139
140/* Static variables */
141static struct cyttsp_gen3_xydata_t g_xy_data;
142static struct cyttsp_bootloader_data_t g_bl_data;
143static struct cyttsp_sysinfo_data_t g_sysinfo_data;
144static const struct i2c_device_id cyttsp_id[] = {
145 { CY_I2C_NAME, 0 }, { }
146};
147static u8 bl_cmd[] = {
148 CY_BL_FILE0, CY_BL_CMD, CY_BL_EXIT,
149 CY_BL_KEY0, CY_BL_KEY1, CY_BL_KEY2,
150 CY_BL_KEY3, CY_BL_KEY4, CY_BL_KEY5,
151 CY_BL_KEY6, CY_BL_KEY7};
152
153MODULE_DEVICE_TABLE(i2c, cyttsp_id);
154
155#ifdef CONFIG_PM
156static const struct dev_pm_ops cyttsp_pm_ops = {
157#ifndef CONFIG_HAS_EARLYSUSPEND
158 .suspend = cyttsp_suspend,
159 .resume = cyttsp_resume,
160#endif
161};
162#endif
163
164static struct i2c_driver cyttsp_driver = {
165 .driver = {
166 .name = CY_I2C_NAME,
167 .owner = THIS_MODULE,
168#ifdef CONFIG_PM
169 .pm = &cyttsp_pm_ops,
170#endif
171 },
172 .probe = cyttsp_probe,
173 .remove = __devexit_p(cyttsp_remove),
174 .id_table = cyttsp_id,
175};
176
177MODULE_LICENSE("GPL");
178MODULE_DESCRIPTION("Cypress TrueTouch(R) Standard touchscreen driver");
179MODULE_AUTHOR("Cypress");
180
181static ssize_t cyttsp_irq_status(struct device *dev,
182 struct device_attribute *attr, char *buf)
183{
184 struct i2c_client *client = container_of(dev, struct i2c_client, dev);
185 struct cyttsp *ts = i2c_get_clientdata(client);
186 return snprintf(buf, TTSP_BUFF_SIZE, "%u\n",
187 atomic_read(&ts->irq_enabled));
188}
189
Amy Maloche304d9f52012-06-14 14:22:50 -0700190static int cyttsp_hndshk(struct cyttsp *ts, u8 hst_mode)
191{
192 int retval = 0;
193 u8 mode = 0;
194
195 mode = hst_mode & CY_HNDSHK_BIT ?
196 hst_mode & ~CY_HNDSHK_BIT :
197 hst_mode | CY_HNDSHK_BIT;
198
199 retval = i2c_smbus_write_i2c_block_data(ts->client,
200 CY_REG_BASE, sizeof(mode), &mode);
201
202 if (retval < 0) {
203 pr_err("%s: bus write fail on handshake r=%d\n",
204 __func__, retval);
205 }
206
207 return retval;
208}
209
210static void cyttsp_change_state(struct cyttsp *ts,
211 enum cyttsp_powerstate new_state)
212{
213 ts->power_state = new_state;
214 pr_info("%s: %s\n", __func__,
215 (ts->power_state < CY_INVALID) ?
216 cyttsp_powerstate_string[ts->power_state] :
217 "INVALID");
218}
219
220static int cyttsp_wait_ready(struct cyttsp *ts, struct completion *complete,
221 u8 *cmd, size_t cmd_size, unsigned long timeout_ms)
222{
223 unsigned long timeout = 0;
224 unsigned long uretval = 0;
225 int retval = 0;
226
227 timeout = msecs_to_jiffies(timeout_ms);
228 INIT_COMPLETION(*complete);
229 if ((cmd != NULL) && (cmd_size != 0)) {
230 retval = i2c_smbus_write_i2c_block_data(ts->client,
231 CY_REG_BASE, cmd_size, cmd);
232 if (retval < 0) {
233 pr_err("%s: bus write fail switch mode r=%d\n",
234 __func__, retval);
235 cyttsp_change_state(ts, CY_IDLE);
236 goto _cyttsp_wait_ready_exit;
237 }
238 }
239 uretval = wait_for_completion_interruptible_timeout(complete, timeout);
240 if (uretval == 0) {
241 pr_err("%s: Switch Mode Timeout waiting " \
242 "for ready interrupt - try reading regs\n", __func__);
243 /* continue anyway */
244 retval = 0;
245 }
246
247_cyttsp_wait_ready_exit:
248 return retval;
249}
250
Steve Mucklef132c6c2012-06-06 18:30:57 -0700251static ssize_t cyttsp_irq_enable(struct device *dev,
252 struct device_attribute *attr,
253 const char *buf, size_t size)
254{
255 struct i2c_client *client = container_of(dev, struct i2c_client, dev);
256 struct cyttsp *ts = i2c_get_clientdata(client);
257 int err = 0;
258 unsigned long value;
259
260 if (size > 2)
261 return -EINVAL;
262
263 err = strict_strtoul(buf, 10, &value);
264 if (err != 0)
265 return err;
266
267 switch (value) {
268 case 0:
269 if (atomic_cmpxchg(&ts->irq_enabled, 1, 0)) {
270 pr_info("touch irq disabled!\n");
271 disable_irq_nosync(ts->client->irq);
272 }
273 err = size;
274 break;
275 case 1:
276 if (!atomic_cmpxchg(&ts->irq_enabled, 0, 1)) {
277 pr_info("touch irq enabled!\n");
278 enable_irq(ts->client->irq);
279 }
280 err = size;
281 break;
282 default:
283 pr_info("cyttsp_irq_enable failed -> irq_enabled = %d\n",
284 atomic_read(&ts->irq_enabled));
285 err = -EINVAL;
286 break;
287 }
288
289 return err;
290}
291
292static DEVICE_ATTR(irq_enable, 0664, cyttsp_irq_status, cyttsp_irq_enable);
293
294static ssize_t cyttsp_fw_show(struct device *dev,
295 struct device_attribute *attr, char *buf)
296{
297 return snprintf(buf, TTSP_BUFF_SIZE, "%d.%d.%d\n", g_bl_data.appid_lo,
298 g_bl_data.appver_hi, g_bl_data.appver_lo);
299}
300
301static DEVICE_ATTR(cyttsp_fw_ver, 0664, cyttsp_fw_show, NULL);
302
303/* firmware flashing block */
304#define BLK_SIZE 16
305#define DATA_REC_LEN 64
306#define BLK_SEED 0xff
307#define RECAL_REG 0x1b
308
309enum bl_commands {
310 BL_CMD_WRBLK = 0x39,
311 BL_CMD_INIT = 0x38,
312 BL_CMD_TERMINATE = 0x3b,
313};
314/* TODO: Add key as part of platform data */
315#define KEY_CS (0 + 1 + 2 + 3 + 4 + 5 + 6 + 7)
316#define KEY {0, 1, 2, 3, 4, 5, 6, 7}
317
318static const char _key[] = KEY;
319#define KEY_LEN sizeof(_key)
320
321static int rec_cnt;
322struct fw_record {
323 u8 seed;
324 u8 cmd;
325 u8 key[KEY_LEN];
326 u8 blk_hi;
327 u8 blk_lo;
328 u8 data[DATA_REC_LEN];
329 u8 data_cs;
330 u8 rec_cs;
331};
332#define fw_rec_size (sizeof(struct fw_record))
333
334struct cmd_record {
335 u8 reg;
336 u8 seed;
337 u8 cmd;
338 u8 key[KEY_LEN];
339};
340#define cmd_rec_size (sizeof(struct cmd_record))
341
342static struct fw_record data_record = {
343 .seed = BLK_SEED,
344 .cmd = BL_CMD_WRBLK,
345 .key = KEY,
346};
347
348static const struct cmd_record terminate_rec = {
349 .reg = 0,
350 .seed = BLK_SEED,
351 .cmd = BL_CMD_TERMINATE,
352 .key = KEY,
353};
354static const struct cmd_record initiate_rec = {
355 .reg = 0,
356 .seed = BLK_SEED,
357 .cmd = BL_CMD_INIT,
358 .key = KEY,
359};
360
361#define BL_REC1_ADDR 0x0780
362#define BL_REC2_ADDR 0x07c0
363#define BL_CHECKSUM_MASK 0x01
364
365#define ID_INFO_REC ":40078000"
366#define ID_INFO_OFFSET_IN_REC 77
367
368#define REC_START_CHR ':'
369#define REC_LEN_OFFSET 1
370#define REC_ADDR_HI_OFFSET 3
371#define REC_ADDR_LO_OFFSET 5
372#define REC_TYPE_OFFSET 7
373#define REC_DATA_OFFSET 9
374#define REC_LINE_SIZE 141
375
376#define NUM_CHAR_IN_HEX 2
377#define ID_INFO_REC_LEN 9
378
379static int cyttsp_soft_reset(struct cyttsp *ts)
380{
381 int retval = 0, tries = 0;
382 u8 host_reg = CY_SOFT_RESET_MODE;
383
384 do {
385 retval = i2c_smbus_write_i2c_block_data(ts->client,
386 CY_REG_BASE, sizeof(host_reg), &host_reg);
387 if (retval < 0)
388 msleep(20);
389 } while (tries++ < 10 && (retval < 0));
390
391 if (retval < 0) {
392 pr_err("%s: failed\n", __func__);
393 return retval;
394 }
395
396 tries = 0;
397 do {
398 msleep(20);
399 cyttsp_putbl(ts, 1, true, true, false);
400 } while (g_bl_data.bl_status != 0x10 &&
401 g_bl_data.bl_status != 0x11 &&
402 tries++ < 100);
403
404 if (g_bl_data.bl_status != 0x11 && g_bl_data.bl_status != 0x10)
405 return -EINVAL;
406
407 return 0;
408}
409
410static void cyttsp_exit_bl_mode(struct cyttsp *ts)
411{
412 int retval, tries = 0;
413
414 do {
415 retval = i2c_smbus_write_i2c_block_data(ts->client,
416 CY_REG_BASE, sizeof(bl_cmd), bl_cmd);
417 if (retval < 0)
418 msleep(20);
419 } while (tries++ < 10 && (retval < 0));
420}
421
Amy Maloche304d9f52012-06-14 14:22:50 -0700422static int cyttsp_set_sysinfo_mode(struct cyttsp *ts, u8 sleep)
Steve Mucklef132c6c2012-06-06 18:30:57 -0700423{
Amy Maloche304d9f52012-06-14 14:22:50 -0700424 int retval;
425 u8 mode = CY_SYSINFO_MODE + sleep;
Steve Mucklef132c6c2012-06-06 18:30:57 -0700426
Amy Maloche304d9f52012-06-14 14:22:50 -0700427 cyttsp_change_state(ts, CY_SYSINFO);
Steve Mucklef132c6c2012-06-06 18:30:57 -0700428
Amy Maloche304d9f52012-06-14 14:22:50 -0700429 retval = cyttsp_wait_ready(ts, &ts->si_int_running,
430 &mode, sizeof(mode), CY_HALF_SEC_TMO_MS);
431
432 if (retval < 0) {
433 pr_err("%s: fail wait ready r=%d\n", __func__, retval);
434 goto cyttsp_set_sysinfo_mode_exit;
435 }
436
437 if (GET_HSTMODE(g_sysinfo_data.hst_mode) !=
438 GET_HSTMODE(CY_SYSINFO_MODE)) {
439 pr_err("%s: Fail enter Sysinfo mode hst_mode=0x%02X\n",
440 __func__, g_sysinfo_data.hst_mode);
441 retval = -EIO;
442 } else {
443 cyttsp_debug("%s: Enter Sysinfo mode hst_mode=0x%02X\n",
444 __func__, g_sysinfo_data.hst_mode);
445 }
446
447cyttsp_set_sysinfo_mode_exit:
448 return retval;
Steve Mucklef132c6c2012-06-06 18:30:57 -0700449}
450
Amy Maloche304d9f52012-06-14 14:22:50 -0700451static void cyttsp_set_opmode(struct cyttsp *ts, u8 sleep)
Steve Mucklef132c6c2012-06-06 18:30:57 -0700452{
453 int retval, tries = 0;
Amy Maloche304d9f52012-06-14 14:22:50 -0700454 u8 host_reg = CY_OP_MODE + sleep;
Steve Mucklef132c6c2012-06-06 18:30:57 -0700455
Amy Maloche304d9f52012-06-14 14:22:50 -0700456 cyttsp_change_state(ts, CY_ACTIVE);
Steve Mucklef132c6c2012-06-06 18:30:57 -0700457 do {
458 retval = i2c_smbus_write_i2c_block_data(ts->client,
459 CY_REG_BASE, sizeof(host_reg), &host_reg);
460 if (retval < 0)
461 msleep(20);
462 } while (tries++ < 10 && (retval < 0));
463}
464
Amy Maloche304d9f52012-06-14 14:22:50 -0700465static int cyttsp_set_lp_mode(struct cyttsp *ts)
466{
467 int retval = 0, tries = 0;
468
469 retval = cyttsp_set_sysinfo_mode(ts, CY_LOW_PWR_MODE);
470 if (retval < 0) {
471 pr_err("%s: failed to enter sysinfo mode, retval =%x\n",
472 __func__, retval);
473 goto exit_low_power_mode;
474 }
475 do {
476 retval = i2c_smbus_write_i2c_block_data(
477 ts->client,
478 CY_REG_ACT_INTRVL,
479 sizeof(ts->power_settings), ts->power_settings);
480 if (retval < 0)
481 msleep(20);
482 } while ((retval < 0) && (tries++ < 5));
483 if (retval < 0)
484 pr_err("%s: failed to write power_settings, retval =%x\n",
485 __func__, retval);
486 msleep(CY_DLY_SYSINFO);
487exit_low_power_mode:
488 cyttsp_set_opmode(ts, CY_LOW_PWR_MODE);
489 return retval;
490
491}
492
Steve Mucklef132c6c2012-06-06 18:30:57 -0700493static int str2uc(char *str, u8 *val)
494{
495 char substr[3];
496 unsigned long ulval;
497 int rc;
498
499 if (!str)
500 return -EINVAL;
501
502 if (strnlen(str, NUM_CHAR_IN_HEX) < 2)
503 return -EINVAL;
504
505 substr[0] = str[0];
506 substr[1] = str[1];
507 substr[2] = '\0';
508
509 rc = strict_strtoul(substr, 16, &ulval);
510 if (rc != 0)
511 return rc;
512
513 *val = (u8) ulval;
514
515 return 0;
516}
517
518static int flash_block(struct cyttsp *ts, u8 *blk, int len)
519{
520 int retval, i, tries = 0;
521 char buf[(2 * (BLK_SIZE + 1)) + 1];
522 char *p = buf;
523
524 for (i = 0; i < len; i++, p += 2)
525 snprintf(p, TTSP_BUFF_SIZE, "%02x", blk[i]);
526 pr_debug("%s: size %d, pos %ld payload %s\n",
527 __func__, len, (long)0, buf);
528
529 do {
530 retval = i2c_smbus_write_i2c_block_data(ts->client,
531 CY_REG_BASE, len, blk);
532 if (retval < 0)
533 msleep(20);
534 } while (tries++ < 20 && (retval < 0));
535
536 if (retval < 0) {
537 pr_err("%s: failed\n", __func__);
538 return retval;
539 }
540
541 return 0;
542}
543
544static int flash_command(struct cyttsp *ts, const struct cmd_record *record)
545{
546 return flash_block(ts, (u8 *)record, cmd_rec_size);
547}
548
549static void init_data_record(struct fw_record *rec, unsigned short addr)
550{
551 addr >>= 6;
552 rec->blk_hi = (addr >> 8) & 0xff;
553 rec->blk_lo = addr & 0xff;
554 rec->rec_cs = rec->blk_hi + rec->blk_lo +
555 (unsigned char)(BLK_SEED + BL_CMD_WRBLK + KEY_CS);
556 rec->data_cs = 0;
557}
558
559static int check_record(struct cyttsp *ts, u8 *rec)
560{
561 int rc;
562 u16 addr;
563 u8 r_len, type, hi_off, lo_off;
564
565 rc = str2uc(rec + REC_LEN_OFFSET, &r_len);
566 if (rc < 0)
567 return rc;
568
569 rc = str2uc(rec + REC_TYPE_OFFSET, &type);
570 if (rc < 0)
571 return rc;
572
573 if (*rec != REC_START_CHR || r_len != DATA_REC_LEN || type != 0)
574 return -EINVAL;
575
576 rc = str2uc(rec + REC_ADDR_HI_OFFSET, &hi_off);
577 if (rc < 0)
578 return rc;
579
580 rc = str2uc(rec + REC_ADDR_LO_OFFSET, &lo_off);
581 if (rc < 0)
582 return rc;
583
584 addr = (hi_off << 8) | lo_off;
585
586 if (addr >= ts->fw_start_addr || addr == BL_REC1_ADDR
587 || addr == BL_REC2_ADDR)
588 return 0;
589
590 return -EINVAL;
591}
592
593static struct fw_record *prepare_record(u8 *rec)
594{
595 int i, rc;
596 u16 addr;
597 u8 hi_off, lo_off;
598 u8 *p;
599
600 rc = str2uc(rec + REC_ADDR_HI_OFFSET, &hi_off);
601 if (rc < 0)
602 return ERR_PTR((long) rc);
603
604 rc = str2uc(rec + REC_ADDR_LO_OFFSET, &lo_off);
605 if (rc < 0)
606 return ERR_PTR((long) rc);
607
608 addr = (hi_off << 8) | lo_off;
609
610 init_data_record(&data_record, addr);
611 p = rec + REC_DATA_OFFSET;
612 for (i = 0; i < DATA_REC_LEN; i++) {
613 rc = str2uc(p, &data_record.data[i]);
614 if (rc < 0)
615 return ERR_PTR((long) rc);
616 data_record.data_cs += data_record.data[i];
617 data_record.rec_cs += data_record.data[i];
618 p += 2;
619 }
620 data_record.rec_cs += data_record.data_cs;
621
622 return &data_record;
623}
624
625static int flash_record(struct cyttsp *ts, const struct fw_record *record)
626{
627 int len = fw_rec_size;
628 int blk_len, rc;
629 u8 *rec = (u8 *)record;
630 u8 data[BLK_SIZE + 1];
631 u8 blk_offset;
632
633 for (blk_offset = 0; len; len -= blk_len) {
634 data[0] = blk_offset;
635 blk_len = len > BLK_SIZE ? BLK_SIZE : len;
636 memcpy(data + 1, rec, blk_len);
637 rec += blk_len;
638 rc = flash_block(ts, data, blk_len + 1);
639 if (rc < 0)
640 return rc;
641 blk_offset += blk_len;
642 }
643 return 0;
644}
645
646static int flash_data_rec(struct cyttsp *ts, u8 *buf)
647{
648 struct fw_record *rec;
649 int rc, tries;
650
651 if (!buf)
652 return -EINVAL;
653
654 rc = check_record(ts, buf);
655
656 if (rc < 0) {
657 pr_debug("%s: record ignored %s", __func__, buf);
658 return 0;
659 }
660
661 rec = prepare_record(buf);
662 if (IS_ERR_OR_NULL(rec))
663 return PTR_ERR(rec);
664
665 rc = flash_record(ts, rec);
666 if (rc < 0)
667 return rc;
668
669 tries = 0;
670 do {
671 if (rec_cnt%2)
672 msleep(20);
673 cyttsp_putbl(ts, 4, true, false, false);
674 } while (g_bl_data.bl_status != 0x10 &&
675 g_bl_data.bl_status != 0x11 &&
676 tries++ < 100);
677 rec_cnt++;
678 return rc;
679}
680
681static int cyttspfw_flash_firmware(struct cyttsp *ts, const u8 *data,
682 int data_len)
683{
684 u8 *buf;
685 int i, j;
686 int rc, tries = 0;
687
688 /* initiate bootload: this will erase all the existing data */
689 rc = flash_command(ts, &initiate_rec);
690 if (rc < 0)
691 return rc;
692
693 do {
694 msleep(100);
695 cyttsp_putbl(ts, 4, true, false, false);
696 } while (g_bl_data.bl_status != 0x10 &&
697 g_bl_data.bl_status != 0x11 &&
698 tries++ < 100);
699
700 buf = kzalloc(REC_LINE_SIZE + 1, GFP_KERNEL);
701 if (!buf) {
702 pr_err("%s: no memory\n", __func__);
703 return -ENOMEM;
704 }
705
706 rec_cnt = 0;
707 /* flash data records */
708 for (i = 0, j = 0; i < data_len; i++, j++) {
709 if ((data[i] == REC_START_CHR) && j) {
710 buf[j] = 0;
711 rc = flash_data_rec(ts, buf);
712 if (rc < 0)
713 return rc;
714 j = 0;
715 }
716 buf[j] = data[i];
717 }
718
719 /* flash last data record */
720 if (j) {
721 buf[j] = 0;
722 rc = flash_data_rec(ts, buf);
723 if (rc < 0)
724 return rc;
725 }
726
727 kfree(buf);
728
729 /* termiate bootload */
730 tries = 0;
731 rc = flash_command(ts, &terminate_rec);
732 do {
733 msleep(100);
734 cyttsp_putbl(ts, 4, true, false, false);
735 } while (g_bl_data.bl_status != 0x10 &&
736 g_bl_data.bl_status != 0x11 &&
737 tries++ < 100);
738
739 return rc;
740}
741
742static int get_hex_fw_ver(u8 *p, u8 *ttspver_hi, u8 *ttspver_lo,
743 u8 *appid_hi, u8 *appid_lo, u8 *appver_hi,
744 u8 *appver_lo, u8 *cid_0, u8 *cid_1, u8 *cid_2)
745{
746 int rc;
747
748 p = p + ID_INFO_OFFSET_IN_REC;
749 rc = str2uc(p, ttspver_hi);
750 if (rc < 0)
751 return rc;
752 p += 2;
753 rc = str2uc(p, ttspver_lo);
754 if (rc < 0)
755 return rc;
756 p += 2;
757 rc = str2uc(p, appid_hi);
758 if (rc < 0)
759 return rc;
760 p += 2;
761 rc = str2uc(p, appid_lo);
762 if (rc < 0)
763 return rc;
764 p += 2;
765 rc = str2uc(p, appver_hi);
766 if (rc < 0)
767 return rc;
768 p += 2;
769 rc = str2uc(p, appver_lo);
770 if (rc < 0)
771 return rc;
772 p += 2;
773 rc = str2uc(p, cid_0);
774 if (rc < 0)
775 return rc;
776 p += 2;
777 rc = str2uc(p, cid_1);
778 if (rc < 0)
779 return rc;
780 p += 2;
781 rc = str2uc(p, cid_2);
782 if (rc < 0)
783 return rc;
784
785 return 0;
786}
787
788static void cyttspfw_flash_start(struct cyttsp *ts, const u8 *data,
789 int data_len, u8 *buf, bool force)
790{
791 int rc;
792 u8 ttspver_hi = 0, ttspver_lo = 0, fw_upgrade = 0;
793 u8 appid_hi = 0, appid_lo = 0;
794 u8 appver_hi = 0, appver_lo = 0;
795 u8 cid_0 = 0, cid_1 = 0, cid_2 = 0;
796 char *p = buf;
797
798 /* get hex firmware version */
799 rc = get_hex_fw_ver(p, &ttspver_hi, &ttspver_lo,
800 &appid_hi, &appid_lo, &appver_hi,
801 &appver_lo, &cid_0, &cid_1, &cid_2);
802
803 if (rc < 0) {
804 pr_err("%s: unable to get hex firmware version\n", __func__);
805 return;
806 }
807
808 /* disable interrupts before flashing */
809 if (ts->client->irq == 0)
810 del_timer(&ts->timer);
811 else
812 disable_irq(ts->client->irq);
813
814 /* enter bootloader idle mode */
815 rc = cyttsp_soft_reset(ts);
816
817 if (rc < 0) {
818 pr_err("%s: try entering into idle mode"
819 " second time\n", __func__);
820 msleep(1000);
821 rc = cyttsp_soft_reset(ts);
822 }
823
824 if (rc < 0) {
825 pr_err("%s: try again later\n", __func__);
826 return;
827 }
828
829
830 pr_info("Current firmware: %d.%d.%d", g_bl_data.appid_lo,
831 g_bl_data.appver_hi, g_bl_data.appver_lo);
832 pr_info("New firmware: %d.%d.%d", appid_lo, appver_hi, appver_lo);
833
834 if (force)
835 fw_upgrade = 1;
836 else if (!(g_bl_data.bl_status & BL_CHECKSUM_MASK) &&
837 (appid_lo == ts->platform_data->correct_fw_ver))
838 fw_upgrade = 1;
839 else if ((appid_hi == g_bl_data.appid_hi) &&
840 (appid_lo == g_bl_data.appid_lo))
841 if (appver_hi > g_bl_data.appver_hi)
842 fw_upgrade = 1;
843 else if ((appver_hi == g_bl_data.appver_hi) &&
844 (appver_lo > g_bl_data.appver_lo))
845 fw_upgrade = 1;
846 else {
847 fw_upgrade = 0;
848 pr_info("%s: Firmware version "
849 "lesser/equal to existing firmware, "
850 "upgrade not needed\n", __func__);
851 }
852 else if (appid_lo == ts->platform_data->correct_fw_ver)
853 fw_upgrade = 1;
854 else {
855 fw_upgrade = 0;
856 pr_info("%s: Firmware versions do not match, "
857 "cannot upgrade\n", __func__);
858 }
859
860 if (fw_upgrade) {
861 pr_info("%s: Starting firmware upgrade\n", __func__);
862 rc = cyttspfw_flash_firmware(ts, data, data_len);
863 if (rc < 0)
864 pr_err("%s: firmware upgrade failed\n", __func__);
865 else
866 pr_info("%s: firmware upgrade success\n", __func__);
867 }
868
Steve Mucklef132c6c2012-06-06 18:30:57 -0700869 /* enable interrupts */
870 if (ts->client->irq == 0)
871 mod_timer(&ts->timer, jiffies + TOUCHSCREEN_TIMEOUT);
872 else
873 enable_irq(ts->client->irq);
Amy Maloche304d9f52012-06-14 14:22:50 -0700874
875 /* enter bootloader idle mode */
876 cyttsp_soft_reset(ts);
877 /* exit bootloader mode */
878 cyttsp_exit_bl_mode(ts);
879 msleep(100);
880 /* set low power mode and enter application mode*/
881 cyttsp_set_lp_mode(ts);
Steve Mucklef132c6c2012-06-06 18:30:57 -0700882}
883
884static void cyttspfw_upgrade_start(struct cyttsp *ts, const u8 *data,
885 int data_len, bool force)
886{
887 int i, j;
888 u8 *buf;
889
890 buf = kzalloc(REC_LINE_SIZE + 1, GFP_KERNEL);
891 if (!buf) {
892 pr_err("%s: no memory\n", __func__);
893 return;
894 }
895
896 for (i = 0, j = 0; i < data_len; i++, j++) {
897 if ((data[i] == REC_START_CHR) && j) {
898 buf[j] = 0;
899 j = 0;
900 if (!strncmp(buf, ID_INFO_REC,
901 strnlen(ID_INFO_REC, ID_INFO_REC_LEN))) {
902 cyttspfw_flash_start(ts, data, data_len,
903 buf, force);
904 break;
905 }
906 }
907 buf[j] = data[i];
908 }
909
910 /* check in the last record of firmware */
911 if (j) {
912 buf[j] = 0;
913 if (!strncmp(buf, ID_INFO_REC,
914 strnlen(ID_INFO_REC, ID_INFO_REC_LEN))) {
915 cyttspfw_flash_start(ts, data, data_len,
916 buf, force);
917 }
918 }
919
920 kfree(buf);
921}
922
923static void cyttspfw_upgrade(struct device *dev, bool force)
924{
925 struct cyttsp *ts = dev_get_drvdata(dev);
926 const struct firmware *cyttsp_fw;
927 int retval = 0;
928
929 if (ts->is_suspended == true) {
930 pr_err("%s: in suspend state, resume it\n", __func__);
931 retval = cyttsp_resume(dev);
932 if (retval < 0) {
933 pr_err("%s: unable to resume\n", __func__);
934 return;
935 }
936 }
937
938 retval = request_firmware(&cyttsp_fw, ts->fw_fname, dev);
939 if (retval < 0) {
940 pr_err("%s: %s request failed(%d)\n", __func__,
941 ts->fw_fname, retval);
942 } else {
943 /* check and start upgrade */
944 cyttspfw_upgrade_start(ts, cyttsp_fw->data,
945 cyttsp_fw->size, force);
946 release_firmware(cyttsp_fw);
947 }
948}
949
950static ssize_t cyttsp_update_fw_show(struct device *dev,
951 struct device_attribute *attr, char *buf)
952{
953 struct cyttsp *ts = dev_get_drvdata(dev);
954 return snprintf(buf, 2, "%d\n", ts->cyttsp_fwloader_mode);
955}
956
957static ssize_t cyttsp_force_update_fw_store(struct device *dev,
958 struct device_attribute *attr,
959 const char *buf, size_t size)
960{
961 struct cyttsp *ts = dev_get_drvdata(dev);
962 unsigned long val;
963 int rc;
964
965 if (size > 2)
966 return -EINVAL;
967
968 rc = strict_strtoul(buf, 10, &val);
969 if (rc != 0)
970 return rc;
971
972 mutex_lock(&ts->mutex);
973 if (!ts->cyttsp_fwloader_mode && val) {
974 ts->cyttsp_fwloader_mode = 1;
975 cyttspfw_upgrade(dev, true);
976 ts->cyttsp_fwloader_mode = 0;
977 }
978 mutex_unlock(&ts->mutex);
979 return size;
980}
981
982static DEVICE_ATTR(cyttsp_force_update_fw, 0664, cyttsp_update_fw_show,
983 cyttsp_force_update_fw_store);
984
985static ssize_t cyttsp_update_fw_store(struct device *dev,
986 struct device_attribute *attr,
987 const char *buf, size_t size)
988{
989 struct cyttsp *ts = dev_get_drvdata(dev);
990 unsigned long val;
991 int rc;
992
993 if (size > 2)
994 return -EINVAL;
995
996 rc = strict_strtoul(buf, 10, &val);
997 if (rc != 0)
998 return rc;
999
1000 mutex_lock(&ts->mutex);
1001 if (!ts->cyttsp_fwloader_mode && val) {
1002 ts->cyttsp_fwloader_mode = 1;
1003 cyttspfw_upgrade(dev, false);
1004 ts->cyttsp_fwloader_mode = 0;
1005 }
1006 mutex_unlock(&ts->mutex);
1007
1008 return size;
1009}
1010
1011static DEVICE_ATTR(cyttsp_update_fw, 0664, cyttsp_update_fw_show,
1012 cyttsp_update_fw_store);
1013
1014static ssize_t cyttsp_fw_name_show(struct device *dev,
1015 struct device_attribute *attr, char *buf)
1016{
1017 struct cyttsp *ts = dev_get_drvdata(dev);
1018 return snprintf(buf, FW_FNAME_LEN - 1, "%s\n", ts->fw_fname);
1019}
1020
1021static ssize_t cyttsp_fw_name_store(struct device *dev,
1022 struct device_attribute *attr,
1023 const char *buf, size_t size)
1024{
1025 struct cyttsp *ts = dev_get_drvdata(dev);
1026
1027 if (size > FW_FNAME_LEN - 1)
1028 return -EINVAL;
1029
1030 strlcpy(ts->fw_fname, buf, size);
1031 if (ts->fw_fname[size-1] == '\n')
1032 ts->fw_fname[size-1] = 0;
1033
1034 return size;
1035}
1036
1037static DEVICE_ATTR(cyttsp_fw_name, 0664, cyttsp_fw_name_show,
1038 cyttsp_fw_name_store);
1039
1040static void cyttsp_xy_handler(struct cyttsp *ts)
1041{
1042 u8 id, tilt, rev_x, rev_y;
1043 u8 i, loc;
1044 u8 prv_tch; /* number of previous touches */
1045 u8 cur_tch; /* number of current touches */
1046 u16 tmp_trk[CY_NUM_MT_TCH_ID];
1047 u16 snd_trk[CY_NUM_MT_TCH_ID];
1048 u16 cur_trk[CY_NUM_TRK_ID];
1049 u16 cur_st_tch[CY_NUM_ST_TCH_ID];
1050 u16 cur_mt_tch[CY_NUM_MT_TCH_ID];
1051 /* if NOT CY_USE_TRACKING_ID then
1052 * only uses CY_NUM_MT_TCH_ID positions */
1053 u16 cur_mt_pos[CY_NUM_TRK_ID][2];
1054 /* if NOT CY_USE_TRACKING_ID then
1055 * only uses CY_NUM_MT_TCH_ID positions */
1056 u8 cur_mt_z[CY_NUM_TRK_ID];
1057 u8 curr_tool_width;
1058 u16 st_x1, st_y1;
1059 u8 st_z1;
1060 u16 st_x2, st_y2;
1061 u8 st_z2;
1062 s32 retval;
1063 int val;
1064
1065 cyttsp_xdebug("TTSP handler start 1:\n");
1066
1067 /* get event data from CYTTSP device */
1068 i = CY_NUM_RETRY;
1069 do {
1070 retval = i2c_smbus_read_i2c_block_data(ts->client,
1071 CY_REG_BASE,
1072 sizeof(struct cyttsp_gen3_xydata_t), (u8 *)&g_xy_data);
1073 } while ((retval < CY_OK) && --i);
1074
1075 if (retval < CY_OK) {
1076 /* return immediately on
1077 * failure to read device on the i2c bus */
1078 goto exit_xy_handler;
1079 }
1080
1081 cyttsp_xdebug("TTSP handler start 2:\n");
1082
1083 /* compare own irq counter with the device irq counter */
1084 if (ts->client->irq) {
Steve Mucklef132c6c2012-06-06 18:30:57 -07001085 u8 cur_cnt;
1086 if (ts->platform_data->use_hndshk) {
Amy Maloche304d9f52012-06-14 14:22:50 -07001087 retval = cyttsp_hndshk(ts, g_xy_data.hst_mode);
1088 if (retval < 0) {
1089 pr_err("%s: Fail write handshake r=%d\n",
1090 __func__, retval);
1091 retval = 0;
1092 }
Steve Mucklef132c6c2012-06-06 18:30:57 -07001093 }
1094 cur_cnt = g_xy_data.tt_undef[CY_IRQ_CNT_REG];
1095 irq_cnt_total++;
1096 irq_cnt++;
1097 if (irq_cnt != cur_cnt) {
1098 irq_err_cnt++;
1099 cyttsp_debug("i_c_ER: dv=%d fw=%d hm=%02X t=%lu te=%lu\n", \
1100 irq_cnt, \
1101 cur_cnt, g_xy_data.hst_mode, \
1102 (unsigned long)irq_cnt_total, \
1103 (unsigned long)irq_err_cnt);
1104 } else {
1105 cyttsp_debug("i_c_ok: dv=%d fw=%d hm=%02X t=%lu te=%lu\n", \
1106 irq_cnt, \
1107 cur_cnt, g_xy_data.hst_mode, \
1108 (unsigned long)irq_cnt_total, \
1109 (unsigned long)irq_err_cnt);
1110 }
1111 irq_cnt = cur_cnt;
1112 }
1113
1114 /* Get the current num touches and return if there are no touches */
1115 if ((GET_BOOTLOADERMODE(g_xy_data.tt_mode) == 1) ||
1116 (GET_HSTMODE(g_xy_data.hst_mode) != CY_OK)) {
1117 u8 host_reg, tries;
1118 /* the TTSP device has suffered spurious reset or mode switch */
1119 cyttsp_debug( \
1120 "Spurious err opmode (tt_mode=%02X hst_mode=%02X)\n", \
1121 g_xy_data.tt_mode, g_xy_data.hst_mode);
1122 cyttsp_debug("Reset TTSP Device; Terminating active tracks\n");
1123 /* terminate all active tracks */
1124 cur_tch = CY_NTCH;
1125 /* reset TTSP part and take it back out of Bootloader mode */
1126 /* reset TTSP Device back to bootloader mode */
1127 host_reg = CY_SOFT_RESET_MODE;
1128 retval = i2c_smbus_write_i2c_block_data(ts->client, CY_REG_BASE,
1129 sizeof(host_reg), &host_reg);
1130 /* wait for TTSP Device to complete reset back to bootloader */
1131 tries = 0;
1132 do {
1133 usleep_range(1000, 1000);
1134 cyttsp_putbl(ts, 1, false, false, false);
1135 } while (g_bl_data.bl_status != 0x10 &&
1136 g_bl_data.bl_status != 0x11 &&
1137 tries++ < 100);
1138 retval = cyttsp_putbl(ts, 1, true, true, true);
1139 /* switch back to operational mode */
1140 /* take TTSP device out of bootloader mode;
1141 * switch back to TrueTouch operational mode */
1142 if (!(retval < CY_OK)) {
1143 int tries;
1144 retval = i2c_smbus_write_i2c_block_data(ts->client,
1145 CY_REG_BASE,
1146 sizeof(bl_cmd), bl_cmd);
1147 /* wait for TTSP Device to complete
1148 * switch to Operational mode */
1149 tries = 0;
1150 do {
1151 msleep(100);
1152 cyttsp_putbl(ts, 2, false, false, false);
1153 } while (GET_BOOTLOADERMODE(g_bl_data.bl_status) &&
1154 tries++ < 100);
1155 cyttsp_putbl(ts, 2, true, false, false);
1156 }
Amy Maloche304d9f52012-06-14 14:22:50 -07001157 cyttsp_set_lp_mode(ts);
Steve Mucklef132c6c2012-06-06 18:30:57 -07001158 goto exit_xy_handler;
1159 } else {
1160 cur_tch = GET_NUM_TOUCHES(g_xy_data.tt_stat);
1161 if (IS_LARGE_AREA(g_xy_data.tt_stat)) {
1162 /* terminate all active tracks */
1163 cur_tch = CY_NTCH;
1164 cyttsp_debug("Large obj detect (tt_stat=0x%02X). Terminate act trks\n", \
1165 g_xy_data.tt_stat);
1166 } else if (cur_tch > CY_NUM_MT_TCH_ID) {
1167 /* if the number of fingers on the touch surface
1168 * is more than the maximum then
1169 * there will be no new track information
1170 * even for the original touches.
1171 * Therefore, terminate all active tracks.
1172 */
1173 cur_tch = CY_NTCH;
1174 cyttsp_debug("Num touch err (tt_stat=0x%02X). Terminate act trks\n", \
1175 g_xy_data.tt_stat);
1176 }
1177 }
1178
1179 /* set tool size */
1180 curr_tool_width = CY_SMALL_TOOL_WIDTH;
1181
1182 /* translate Gen2 interface data into comparable Gen3 data */
1183 if (ts->platform_data->gen == CY_GEN2) {
1184 struct cyttsp_gen2_xydata_t *pxy_gen2_data;
1185 pxy_gen2_data = (struct cyttsp_gen2_xydata_t *)(&g_xy_data);
1186
1187 /* use test data? */
1188 cyttsp_testdat(&g_xy_data, &tt_gen2_testray, \
1189 sizeof(struct cyttsp_gen3_xydata_t));
1190
1191 if (ts->platform_data->disable_ghost_det &&
1192 (cur_tch == CY_GEN2_GHOST))
1193 cur_tch = CY_GEN2_2TOUCH;
1194
1195 if (pxy_gen2_data->evnt_idx == CY_GEN2_NOTOUCH) {
1196 cur_tch = 0;
1197 } else if (cur_tch == CY_GEN2_GHOST) {
1198 cur_tch = 0;
1199 } else if (cur_tch == CY_GEN2_2TOUCH) {
1200 /* stuff artificial track ID1 and ID2 */
1201 g_xy_data.touch12_id = 0x12;
1202 g_xy_data.z1 = CY_MAXZ;
1203 g_xy_data.z2 = CY_MAXZ;
1204 cur_tch--; /* 2 touches */
1205 } else if (cur_tch == CY_GEN2_1TOUCH) {
1206 /* stuff artificial track ID1 and ID2 */
1207 g_xy_data.touch12_id = 0x12;
1208 g_xy_data.z1 = CY_MAXZ;
1209 g_xy_data.z2 = CY_NTCH;
1210 if (pxy_gen2_data->evnt_idx == CY_GEN2_TOUCH2) {
1211 /* push touch 2 data into touch1
1212 * (first finger up; second finger down) */
1213 /* stuff artificial track ID1 for touch2 info */
1214 g_xy_data.touch12_id = 0x20;
1215 /* stuff touch 1 with touch 2 coordinate data */
1216 g_xy_data.x1 = g_xy_data.x2;
1217 g_xy_data.y1 = g_xy_data.y2;
1218 }
1219 } else {
1220 cur_tch = 0;
1221 }
1222 } else {
1223 /* use test data? */
1224 cyttsp_testdat(&g_xy_data, &tt_gen3_testray, \
1225 sizeof(struct cyttsp_gen3_xydata_t));
1226 }
1227
1228
1229
1230 /* clear current active track ID array and count previous touches */
1231 for (id = 0, prv_tch = CY_NTCH;
1232 id < CY_NUM_TRK_ID; id++) {
1233 cur_trk[id] = CY_NTCH;
1234 prv_tch += ts->act_trk[id];
1235 }
1236
1237 /* send no events if no previous touches and no new touches */
1238 if ((prv_tch == CY_NTCH) &&
1239 ((cur_tch == CY_NTCH) ||
1240 (cur_tch > CY_NUM_MT_TCH_ID))) {
1241 goto exit_xy_handler;
1242 }
1243
1244 cyttsp_debug("prev=%d curr=%d\n", prv_tch, cur_tch);
1245
1246 for (id = 0; id < CY_NUM_ST_TCH_ID; id++) {
1247 /* clear current single touches array */
1248 cur_st_tch[id] = CY_IGNR_TCH;
1249 }
1250
1251 /* clear single touch positions */
1252 st_x1 = CY_NTCH;
1253 st_y1 = CY_NTCH;
1254 st_z1 = CY_NTCH;
1255 st_x2 = CY_NTCH;
1256 st_y2 = CY_NTCH;
1257 st_z2 = CY_NTCH;
1258
1259 for (id = 0; id < CY_NUM_MT_TCH_ID; id++) {
1260 /* clear current multi-touches array and
1261 * multi-touch positions/z */
1262 cur_mt_tch[id] = CY_IGNR_TCH;
1263 }
1264
1265 if (ts->platform_data->use_trk_id) {
1266 for (id = 0; id < CY_NUM_MT_TCH_ID; id++) {
1267 cur_mt_pos[id][CY_XPOS] = 0;
1268 cur_mt_pos[id][CY_YPOS] = 0;
1269 cur_mt_z[id] = 0;
1270 }
1271 } else {
1272 for (id = 0; id < CY_NUM_TRK_ID; id++) {
1273 cur_mt_pos[id][CY_XPOS] = 0;
1274 cur_mt_pos[id][CY_YPOS] = 0;
1275 cur_mt_z[id] = 0;
1276 }
1277 }
1278
1279 /* Determine if display is tilted */
1280 if (FLIP_DATA(ts->platform_data->flags))
1281 tilt = true;
1282 else
1283 tilt = false;
1284
1285 /* Check for switch in origin */
1286 if (REVERSE_X(ts->platform_data->flags))
1287 rev_x = true;
1288 else
1289 rev_x = false;
1290
1291 if (REVERSE_Y(ts->platform_data->flags))
1292 rev_y = true;
1293 else
1294 rev_y = false;
1295
1296 if (cur_tch) {
1297 struct cyttsp_gen2_xydata_t *pxy_gen2_data;
1298 struct cyttsp_gen3_xydata_t *pxy_gen3_data;
1299 switch (ts->platform_data->gen) {
1300 case CY_GEN2: {
1301 pxy_gen2_data =
1302 (struct cyttsp_gen2_xydata_t *)(&g_xy_data);
1303 cyttsp_xdebug("TTSP Gen2 report:\n");
1304 cyttsp_xdebug("%02X %02X %02X\n", \
1305 pxy_gen2_data->hst_mode, \
1306 pxy_gen2_data->tt_mode, \
1307 pxy_gen2_data->tt_stat);
1308 cyttsp_xdebug("%04X %04X %02X %02X\n", \
1309 pxy_gen2_data->x1, \
1310 pxy_gen2_data->y1, \
1311 pxy_gen2_data->z1, \
1312 pxy_gen2_data->evnt_idx);
1313 cyttsp_xdebug("%04X %04X %02X\n", \
1314 pxy_gen2_data->x2, \
1315 pxy_gen2_data->y2, \
1316 pxy_gen2_data->tt_undef1);
1317 cyttsp_xdebug("%02X %02X %02X\n", \
1318 pxy_gen2_data->gest_cnt, \
1319 pxy_gen2_data->gest_id, \
1320 pxy_gen2_data->gest_set);
1321 break;
1322 }
1323 case CY_GEN3:
1324 default: {
1325 pxy_gen3_data =
1326 (struct cyttsp_gen3_xydata_t *)(&g_xy_data);
1327 cyttsp_xdebug("TTSP Gen3 report:\n");
1328 cyttsp_xdebug("%02X %02X %02X\n", \
1329 pxy_gen3_data->hst_mode,
1330 pxy_gen3_data->tt_mode,
1331 pxy_gen3_data->tt_stat);
1332 cyttsp_xdebug("%04X %04X %02X %02X", \
1333 pxy_gen3_data->x1,
1334 pxy_gen3_data->y1,
1335 pxy_gen3_data->z1, \
1336 pxy_gen3_data->touch12_id);
1337 cyttsp_xdebug("%04X %04X %02X\n", \
1338 pxy_gen3_data->x2, \
1339 pxy_gen3_data->y2, \
1340 pxy_gen3_data->z2);
1341 cyttsp_xdebug("%02X %02X %02X\n", \
1342 pxy_gen3_data->gest_cnt, \
1343 pxy_gen3_data->gest_id, \
1344 pxy_gen3_data->gest_set);
1345 cyttsp_xdebug("%04X %04X %02X %02X\n", \
1346 pxy_gen3_data->x3, \
1347 pxy_gen3_data->y3, \
1348 pxy_gen3_data->z3, \
1349 pxy_gen3_data->touch34_id);
1350 cyttsp_xdebug("%04X %04X %02X\n", \
1351 pxy_gen3_data->x4, \
1352 pxy_gen3_data->y4, \
1353 pxy_gen3_data->z4);
1354 break;
1355 }
1356 }
1357 }
1358
1359 /* process the touches */
1360 switch (cur_tch) {
1361 case 4: {
1362 g_xy_data.x4 = be16_to_cpu(g_xy_data.x4);
1363 g_xy_data.y4 = be16_to_cpu(g_xy_data.y4);
1364 if (tilt)
1365 FLIP_XY(g_xy_data.x4, g_xy_data.y4);
1366
1367 if (rev_x) {
1368 val = INVERT_X(g_xy_data.x4,
1369 ts->platform_data->panel_maxx);
1370 if (val >= 0)
1371 g_xy_data.x4 = val;
1372 else
1373 pr_debug("X value is negative. Please configure"
1374 " maxx in platform data structure\n");
1375 }
1376 if (rev_y) {
1377 val = INVERT_X(g_xy_data.y4,
1378 ts->platform_data->panel_maxy);
1379 if (val >= 0)
1380 g_xy_data.y4 = val;
1381 else
1382 pr_debug("Y value is negative. Please configure"
1383 " maxy in platform data structure\n");
1384
1385 }
1386 id = GET_TOUCH4_ID(g_xy_data.touch34_id);
1387 if (ts->platform_data->use_trk_id) {
1388 cur_mt_pos[CY_MT_TCH4_IDX][CY_XPOS] =
1389 g_xy_data.x4;
1390 cur_mt_pos[CY_MT_TCH4_IDX][CY_YPOS] =
1391 g_xy_data.y4;
1392 cur_mt_z[CY_MT_TCH4_IDX] = g_xy_data.z4;
1393 } else {
1394 cur_mt_pos[id][CY_XPOS] = g_xy_data.x4;
1395 cur_mt_pos[id][CY_YPOS] = g_xy_data.y4;
1396 cur_mt_z[id] = g_xy_data.z4;
1397 }
1398 cur_mt_tch[CY_MT_TCH4_IDX] = id;
1399 cur_trk[id] = CY_TCH;
1400 if (ts->prv_st_tch[CY_ST_FNGR1_IDX] <
1401 CY_NUM_TRK_ID) {
1402 if (ts->prv_st_tch[CY_ST_FNGR1_IDX] == id) {
1403 st_x1 = g_xy_data.x4;
1404 st_y1 = g_xy_data.y4;
1405 st_z1 = g_xy_data.z4;
1406 cur_st_tch[CY_ST_FNGR1_IDX] = id;
1407 } else if (ts->prv_st_tch[CY_ST_FNGR2_IDX] == id) {
1408 st_x2 = g_xy_data.x4;
1409 st_y2 = g_xy_data.y4;
1410 st_z2 = g_xy_data.z4;
1411 cur_st_tch[CY_ST_FNGR2_IDX] = id;
1412 }
1413 }
1414 cyttsp_xdebug("4th XYZ:% 3d,% 3d,% 3d ID:% 2d\n\n", \
1415 g_xy_data.x4, g_xy_data.y4, g_xy_data.z4, \
1416 (g_xy_data.touch34_id & 0x0F));
1417 /* do not break */
1418 }
1419 case 3: {
1420 g_xy_data.x3 = be16_to_cpu(g_xy_data.x3);
1421 g_xy_data.y3 = be16_to_cpu(g_xy_data.y3);
1422 if (tilt)
1423 FLIP_XY(g_xy_data.x3, g_xy_data.y3);
1424
1425 if (rev_x) {
1426 val = INVERT_X(g_xy_data.x3,
1427 ts->platform_data->panel_maxx);
1428 if (val >= 0)
1429 g_xy_data.x3 = val;
1430 else
1431 pr_debug("X value is negative. Please configure"
1432 " maxx in platform data structure\n");
1433
1434 }
1435 if (rev_y) {
1436 val = INVERT_X(g_xy_data.y3,
1437 ts->platform_data->panel_maxy);
1438 if (val >= 0)
1439 g_xy_data.y3 = val;
1440 else
1441 pr_debug("Y value is negative. Please configure"
1442 " maxy in platform data structure\n");
1443
1444 }
1445 id = GET_TOUCH3_ID(g_xy_data.touch34_id);
1446 if (ts->platform_data->use_trk_id) {
1447 cur_mt_pos[CY_MT_TCH3_IDX][CY_XPOS] =
1448 g_xy_data.x3;
1449 cur_mt_pos[CY_MT_TCH3_IDX][CY_YPOS] =
1450 g_xy_data.y3;
1451 cur_mt_z[CY_MT_TCH3_IDX] = g_xy_data.z3;
1452 } else {
1453 cur_mt_pos[id][CY_XPOS] = g_xy_data.x3;
1454 cur_mt_pos[id][CY_YPOS] = g_xy_data.y3;
1455 cur_mt_z[id] = g_xy_data.z3;
1456 }
1457 cur_mt_tch[CY_MT_TCH3_IDX] = id;
1458 cur_trk[id] = CY_TCH;
1459 if (ts->prv_st_tch[CY_ST_FNGR1_IDX] <
1460 CY_NUM_TRK_ID) {
1461 if (ts->prv_st_tch[CY_ST_FNGR1_IDX] == id) {
1462 st_x1 = g_xy_data.x3;
1463 st_y1 = g_xy_data.y3;
1464 st_z1 = g_xy_data.z3;
1465 cur_st_tch[CY_ST_FNGR1_IDX] = id;
1466 } else if (ts->prv_st_tch[CY_ST_FNGR2_IDX] == id) {
1467 st_x2 = g_xy_data.x3;
1468 st_y2 = g_xy_data.y3;
1469 st_z2 = g_xy_data.z3;
1470 cur_st_tch[CY_ST_FNGR2_IDX] = id;
1471 }
1472 }
1473 cyttsp_xdebug("3rd XYZ:% 3d,% 3d,% 3d ID:% 2d\n", \
1474 g_xy_data.x3, g_xy_data.y3, g_xy_data.z3, \
1475 ((g_xy_data.touch34_id >> 4) & 0x0F));
1476 /* do not break */
1477 }
1478 case 2: {
1479 g_xy_data.x2 = be16_to_cpu(g_xy_data.x2);
1480 g_xy_data.y2 = be16_to_cpu(g_xy_data.y2);
1481 if (tilt)
1482 FLIP_XY(g_xy_data.x2, g_xy_data.y2);
1483
1484 if (rev_x) {
1485 val = INVERT_X(g_xy_data.x2,
1486 ts->platform_data->panel_maxx);
1487 if (val >= 0)
1488 g_xy_data.x2 = val;
1489 else
1490 pr_debug("X value is negative. Please configure"
1491 " maxx in platform data structure\n");
1492 }
1493 if (rev_y) {
1494 val = INVERT_X(g_xy_data.y2,
1495 ts->platform_data->panel_maxy);
1496 if (val >= 0)
1497 g_xy_data.y2 = val;
1498 else
1499 pr_debug("Y value is negative. Please configure"
1500 " maxy in platform data structure\n");
1501 }
1502 id = GET_TOUCH2_ID(g_xy_data.touch12_id);
1503 if (ts->platform_data->use_trk_id) {
1504 cur_mt_pos[CY_MT_TCH2_IDX][CY_XPOS] =
1505 g_xy_data.x2;
1506 cur_mt_pos[CY_MT_TCH2_IDX][CY_YPOS] =
1507 g_xy_data.y2;
1508 cur_mt_z[CY_MT_TCH2_IDX] = g_xy_data.z2;
1509 } else {
1510 cur_mt_pos[id][CY_XPOS] = g_xy_data.x2;
1511 cur_mt_pos[id][CY_YPOS] = g_xy_data.y2;
1512 cur_mt_z[id] = g_xy_data.z2;
1513 }
1514 cur_mt_tch[CY_MT_TCH2_IDX] = id;
1515 cur_trk[id] = CY_TCH;
1516 if (ts->prv_st_tch[CY_ST_FNGR1_IDX] <
1517 CY_NUM_TRK_ID) {
1518 if (ts->prv_st_tch[CY_ST_FNGR1_IDX] == id) {
1519 st_x1 = g_xy_data.x2;
1520 st_y1 = g_xy_data.y2;
1521 st_z1 = g_xy_data.z2;
1522 cur_st_tch[CY_ST_FNGR1_IDX] = id;
1523 } else if (ts->prv_st_tch[CY_ST_FNGR2_IDX] == id) {
1524 st_x2 = g_xy_data.x2;
1525 st_y2 = g_xy_data.y2;
1526 st_z2 = g_xy_data.z2;
1527 cur_st_tch[CY_ST_FNGR2_IDX] = id;
1528 }
1529 }
1530 cyttsp_xdebug("2nd XYZ:% 3d,% 3d,% 3d ID:% 2d\n", \
1531 g_xy_data.x2, g_xy_data.y2, g_xy_data.z2, \
1532 (g_xy_data.touch12_id & 0x0F));
1533 /* do not break */
1534 }
1535 case 1: {
1536 g_xy_data.x1 = be16_to_cpu(g_xy_data.x1);
1537 g_xy_data.y1 = be16_to_cpu(g_xy_data.y1);
1538 if (tilt)
1539 FLIP_XY(g_xy_data.x1, g_xy_data.y1);
1540
1541 if (rev_x) {
1542 val = INVERT_X(g_xy_data.x1,
1543 ts->platform_data->panel_maxx);
1544 if (val >= 0)
1545 g_xy_data.x1 = val;
1546 else
1547 pr_debug("X value is negative. Please configure"
1548 " maxx in platform data structure\n");
1549 }
1550 if (rev_y) {
1551 val = INVERT_X(g_xy_data.y1,
1552 ts->platform_data->panel_maxy);
1553 if (val >= 0)
1554 g_xy_data.y1 = val;
1555 else
1556 pr_debug("Y value is negative. Please configure"
1557 " maxy in platform data structure");
1558 }
1559 id = GET_TOUCH1_ID(g_xy_data.touch12_id);
1560 if (ts->platform_data->use_trk_id) {
1561 cur_mt_pos[CY_MT_TCH1_IDX][CY_XPOS] =
1562 g_xy_data.x1;
1563 cur_mt_pos[CY_MT_TCH1_IDX][CY_YPOS] =
1564 g_xy_data.y1;
1565 cur_mt_z[CY_MT_TCH1_IDX] = g_xy_data.z1;
1566 } else {
1567 cur_mt_pos[id][CY_XPOS] = g_xy_data.x1;
1568 cur_mt_pos[id][CY_YPOS] = g_xy_data.y1;
1569 cur_mt_z[id] = g_xy_data.z1;
1570 }
1571 cur_mt_tch[CY_MT_TCH1_IDX] = id;
1572 cur_trk[id] = CY_TCH;
1573 if (ts->prv_st_tch[CY_ST_FNGR1_IDX] <
1574 CY_NUM_TRK_ID) {
1575 if (ts->prv_st_tch[CY_ST_FNGR1_IDX] == id) {
1576 st_x1 = g_xy_data.x1;
1577 st_y1 = g_xy_data.y1;
1578 st_z1 = g_xy_data.z1;
1579 cur_st_tch[CY_ST_FNGR1_IDX] = id;
1580 } else if (ts->prv_st_tch[CY_ST_FNGR2_IDX] == id) {
1581 st_x2 = g_xy_data.x1;
1582 st_y2 = g_xy_data.y1;
1583 st_z2 = g_xy_data.z1;
1584 cur_st_tch[CY_ST_FNGR2_IDX] = id;
1585 }
1586 }
1587 cyttsp_xdebug("1st XYZ:% 3d,% 3d,% 3d ID:% 2d\n", \
1588 g_xy_data.x1, g_xy_data.y1, g_xy_data.z1, \
1589 ((g_xy_data.touch12_id >> 4) & 0x0F));
1590 break;
1591 }
1592 case 0:
1593 default:{
1594 break;
1595 }
1596 }
1597
1598 /* handle Single Touch signals */
1599 if (ts->platform_data->use_st) {
1600 cyttsp_xdebug("ST STEP 0 - ST1 ID=%d ST2 ID=%d\n", \
1601 cur_st_tch[CY_ST_FNGR1_IDX], \
1602 cur_st_tch[CY_ST_FNGR2_IDX]);
1603 if (cur_st_tch[CY_ST_FNGR1_IDX] > CY_NUM_TRK_ID) {
1604 /* reassign finger 1 and 2 positions to new tracks */
1605 if (cur_tch > 0) {
1606 /* reassign st finger1 */
1607 if (ts->platform_data->use_trk_id) {
1608 id = CY_MT_TCH1_IDX;
1609 cur_st_tch[CY_ST_FNGR1_IDX] = cur_mt_tch[id];
1610 } else {
1611 id = GET_TOUCH1_ID(g_xy_data.touch12_id);
1612 cur_st_tch[CY_ST_FNGR1_IDX] = id;
1613 }
1614 st_x1 = cur_mt_pos[id][CY_XPOS];
1615 st_y1 = cur_mt_pos[id][CY_YPOS];
1616 st_z1 = cur_mt_z[id];
1617 cyttsp_xdebug("ST STEP 1 - ST1 ID=%3d\n", \
1618 cur_st_tch[CY_ST_FNGR1_IDX]);
1619 if ((cur_tch > 1) &&
1620 (cur_st_tch[CY_ST_FNGR2_IDX] >
1621 CY_NUM_TRK_ID)) {
1622 /* reassign st finger2 */
1623 if (cur_tch > 1) {
1624 if (ts->platform_data->use_trk_id) {
1625 id = CY_MT_TCH2_IDX;
1626 cur_st_tch[CY_ST_FNGR2_IDX] = cur_mt_tch[id];
1627 } else {
1628 id = GET_TOUCH2_ID(g_xy_data.touch12_id);
1629 cur_st_tch[CY_ST_FNGR2_IDX] = id;
1630 }
1631 st_x2 = cur_mt_pos[id][CY_XPOS];
1632 st_y2 = cur_mt_pos[id][CY_YPOS];
1633 st_z2 = cur_mt_z[id];
1634 cyttsp_xdebug("ST STEP 2 - ST2 ID=%3d\n", \
1635 cur_st_tch[CY_ST_FNGR2_IDX]);
1636 }
1637 }
1638 }
1639 } else if (cur_st_tch[CY_ST_FNGR2_IDX] > CY_NUM_TRK_ID) {
1640 if (cur_tch > 1) {
1641 /* reassign st finger2 */
1642 if (ts->platform_data->use_trk_id) {
1643 /* reassign st finger2 */
1644 id = CY_MT_TCH2_IDX;
1645 cur_st_tch[CY_ST_FNGR2_IDX] =
1646 cur_mt_tch[id];
1647 } else {
1648 /* reassign st finger2 */
1649 id = GET_TOUCH2_ID(g_xy_data.touch12_id);
1650 cur_st_tch[CY_ST_FNGR2_IDX] = id;
1651 }
1652 st_x2 = cur_mt_pos[id][CY_XPOS];
1653 st_y2 = cur_mt_pos[id][CY_YPOS];
1654 st_z2 = cur_mt_z[id];
1655 cyttsp_xdebug("ST STEP 3 - ST2 ID=%3d\n", \
1656 cur_st_tch[CY_ST_FNGR2_IDX]);
1657 }
1658 }
1659 /* if the 1st touch is missing and there is a 2nd touch,
1660 * then set the 1st touch to 2nd touch and terminate 2nd touch
1661 */
1662 if ((cur_st_tch[CY_ST_FNGR1_IDX] > CY_NUM_TRK_ID) &&
1663 (cur_st_tch[CY_ST_FNGR2_IDX] < CY_NUM_TRK_ID)) {
1664 st_x1 = st_x2;
1665 st_y1 = st_y2;
1666 st_z1 = st_z2;
1667 cur_st_tch[CY_ST_FNGR1_IDX] =
1668 cur_st_tch[CY_ST_FNGR2_IDX];
1669 cur_st_tch[CY_ST_FNGR2_IDX] =
1670 CY_IGNR_TCH;
1671 }
1672 /* if the 2nd touch ends up equal to the 1st touch,
1673 * then just report a single touch */
1674 if (cur_st_tch[CY_ST_FNGR1_IDX] ==
1675 cur_st_tch[CY_ST_FNGR2_IDX]) {
1676 cur_st_tch[CY_ST_FNGR2_IDX] =
1677 CY_IGNR_TCH;
1678 }
1679 /* set Single Touch current event signals */
1680 if (cur_st_tch[CY_ST_FNGR1_IDX] < CY_NUM_TRK_ID) {
1681 input_report_abs(ts->input,
1682 ABS_X, st_x1);
1683 input_report_abs(ts->input,
1684 ABS_Y, st_y1);
1685 input_report_abs(ts->input,
1686 ABS_PRESSURE, st_z1);
1687 input_report_key(ts->input,
1688 BTN_TOUCH,
1689 CY_TCH);
1690 input_report_abs(ts->input,
1691 ABS_TOOL_WIDTH,
1692 curr_tool_width);
1693 cyttsp_debug("ST->F1:%3d X:%3d Y:%3d Z:%3d\n", \
1694 cur_st_tch[CY_ST_FNGR1_IDX], \
1695 st_x1, st_y1, st_z1);
1696 if (cur_st_tch[CY_ST_FNGR2_IDX] < CY_NUM_TRK_ID) {
1697 input_report_key(ts->input, BTN_2, CY_TCH);
1698 input_report_abs(ts->input, ABS_HAT0X, st_x2);
1699 input_report_abs(ts->input, ABS_HAT0Y, st_y2);
1700 cyttsp_debug("ST->F2:%3d X:%3d Y:%3d Z:%3d\n", \
1701 cur_st_tch[CY_ST_FNGR2_IDX],
1702 st_x2, st_y2, st_z2);
1703 } else {
1704 input_report_key(ts->input,
1705 BTN_2,
1706 CY_NTCH);
1707 }
1708 } else {
1709 input_report_abs(ts->input, ABS_PRESSURE, CY_NTCH);
1710 input_report_key(ts->input, BTN_TOUCH, CY_NTCH);
1711 input_report_key(ts->input, BTN_2, CY_NTCH);
1712 }
1713 /* update platform data for the current single touch info */
1714 ts->prv_st_tch[CY_ST_FNGR1_IDX] = cur_st_tch[CY_ST_FNGR1_IDX];
1715 ts->prv_st_tch[CY_ST_FNGR2_IDX] = cur_st_tch[CY_ST_FNGR2_IDX];
1716
1717 }
1718
1719 /* handle Multi-touch signals */
1720 if (ts->platform_data->use_mt) {
1721 if (ts->platform_data->use_trk_id) {
1722 /* terminate any previous touch where the track
1723 * is missing from the current event */
1724 for (id = 0; id < CY_NUM_TRK_ID; id++) {
1725 if ((ts->act_trk[id] != CY_NTCH) &&
1726 (cur_trk[id] == CY_NTCH)) {
1727 input_report_abs(ts->input,
1728 ABS_MT_TRACKING_ID,
1729 id);
1730 input_report_abs(ts->input,
1731 ABS_MT_TOUCH_MAJOR,
1732 CY_NTCH);
1733 input_report_abs(ts->input,
1734 ABS_MT_WIDTH_MAJOR,
1735 curr_tool_width);
1736 input_report_abs(ts->input,
1737 ABS_MT_POSITION_X,
1738 ts->prv_mt_pos[id][CY_XPOS]);
1739 input_report_abs(ts->input,
1740 ABS_MT_POSITION_Y,
1741 ts->prv_mt_pos[id][CY_YPOS]);
1742 CY_MT_SYNC(ts->input);
1743 ts->act_trk[id] = CY_NTCH;
1744 ts->prv_mt_pos[id][CY_XPOS] = 0;
1745 ts->prv_mt_pos[id][CY_YPOS] = 0;
1746 }
1747 }
1748 /* set Multi-Touch current event signals */
1749 for (id = 0; id < CY_NUM_MT_TCH_ID; id++) {
1750 if (cur_mt_tch[id] < CY_NUM_TRK_ID) {
1751 input_report_abs(ts->input,
1752 ABS_MT_TRACKING_ID,
1753 cur_mt_tch[id]);
1754 input_report_abs(ts->input,
1755 ABS_MT_TOUCH_MAJOR,
1756 cur_mt_z[id]);
1757 input_report_abs(ts->input,
1758 ABS_MT_WIDTH_MAJOR,
1759 curr_tool_width);
1760 input_report_abs(ts->input,
1761 ABS_MT_POSITION_X,
1762 cur_mt_pos[id][CY_XPOS]);
1763 input_report_abs(ts->input,
1764 ABS_MT_POSITION_Y,
1765 cur_mt_pos[id][CY_YPOS]);
1766 CY_MT_SYNC(ts->input);
1767 ts->act_trk[id] = CY_TCH;
1768 ts->prv_mt_pos[id][CY_XPOS] =
1769 cur_mt_pos[id][CY_XPOS];
1770 ts->prv_mt_pos[id][CY_YPOS] =
1771 cur_mt_pos[id][CY_YPOS];
1772 }
1773 }
1774 } else {
1775 /* set temporary track array elements to voids */
1776 for (id = 0; id < CY_NUM_MT_TCH_ID; id++) {
1777 tmp_trk[id] = CY_IGNR_TCH;
1778 snd_trk[id] = CY_IGNR_TCH;
1779 }
1780
1781 /* get what is currently active */
1782 for (i = 0, id = 0;
1783 id < CY_NUM_TRK_ID && i < CY_NUM_MT_TCH_ID;
1784 id++) {
1785 if (cur_trk[id] == CY_TCH) {
1786 /* only incr counter if track found */
1787 tmp_trk[i] = id;
1788 i++;
1789 }
1790 }
1791 cyttsp_xdebug("T1: t0=%d, t1=%d, t2=%d, t3=%d\n", \
1792 tmp_trk[0], tmp_trk[1], tmp_trk[2], \
1793 tmp_trk[3]);
1794 cyttsp_xdebug("T1: p0=%d, p1=%d, p2=%d, p3=%d\n", \
1795 ts->prv_mt_tch[0], ts->prv_mt_tch[1], \
1796 ts->prv_mt_tch[2], ts->prv_mt_tch[3]);
1797
1798 /* pack in still active previous touches */
1799 for (id = 0, prv_tch = 0;
1800 id < CY_NUM_MT_TCH_ID; id++) {
1801 if (tmp_trk[id] < CY_NUM_TRK_ID) {
1802 if (cyttsp_inlist(ts->prv_mt_tch,
1803 tmp_trk[id], &loc,
1804 CY_NUM_MT_TCH_ID)) {
1805 loc &= CY_NUM_MT_TCH_ID - 1;
1806 snd_trk[loc] = tmp_trk[id];
1807 prv_tch++;
1808 cyttsp_xdebug("inlist s[%d]=%d t[%d]=%d l=%d p=%d\n", \
1809 loc, snd_trk[loc], \
1810 id, tmp_trk[id], \
1811 loc, prv_tch);
1812 } else {
1813 cyttsp_xdebug("not inlist s[%d]=%d t[%d]=%d l=%d \n", \
1814 id, snd_trk[id], \
1815 id, tmp_trk[id], \
1816 loc);
1817 }
1818 }
1819 }
1820 cyttsp_xdebug("S1: s0=%d, s1=%d, s2=%d, s3=%d p=%d\n", \
1821 snd_trk[0], snd_trk[1], snd_trk[2], \
1822 snd_trk[3], prv_tch);
1823
1824 /* pack in new touches */
1825 for (id = 0; id < CY_NUM_MT_TCH_ID; id++) {
1826 if (tmp_trk[id] < CY_NUM_TRK_ID) {
1827 if (!cyttsp_inlist(snd_trk, tmp_trk[id], &loc, CY_NUM_MT_TCH_ID)) {
1828 cyttsp_xdebug("not inlist t[%d]=%d l=%d\n", \
1829 id, tmp_trk[id], loc);
1830 if (cyttsp_next_avail_inlist(snd_trk, &loc, CY_NUM_MT_TCH_ID)) {
1831 loc &= CY_NUM_MT_TCH_ID - 1;
1832 snd_trk[loc] = tmp_trk[id];
1833 cyttsp_xdebug("put inlist s[%d]=%d t[%d]=%d\n",
1834 loc, snd_trk[loc], id, tmp_trk[id]);
1835 }
1836 } else {
1837 cyttsp_xdebug("is in list s[%d]=%d t[%d]=%d loc=%d\n", \
1838 id, snd_trk[id], id, tmp_trk[id], loc);
1839 }
1840 }
1841 }
1842 cyttsp_xdebug("S2: s0=%d, s1=%d, s2=%d, s3=%d\n", \
1843 snd_trk[0], snd_trk[1],
1844 snd_trk[2], snd_trk[3]);
1845
1846 /* sync motion event signals for each current touch */
1847 for (id = 0; id < CY_NUM_MT_TCH_ID; id++) {
1848 /* z will either be 0 (NOTOUCH) or
1849 * some pressure (TOUCH) */
1850 cyttsp_xdebug("MT0 prev[%d]=%d temp[%d]=%d send[%d]=%d\n", \
1851 id, ts->prv_mt_tch[id], \
1852 id, tmp_trk[id], \
1853 id, snd_trk[id]);
1854 if (snd_trk[id] < CY_NUM_TRK_ID) {
1855 input_report_abs(ts->input,
1856 ABS_MT_TOUCH_MAJOR,
1857 cur_mt_z[snd_trk[id]]);
1858 input_report_abs(ts->input,
1859 ABS_MT_WIDTH_MAJOR,
1860 curr_tool_width);
1861 input_report_abs(ts->input,
1862 ABS_MT_POSITION_X,
1863 cur_mt_pos[snd_trk[id]][CY_XPOS]);
1864 input_report_abs(ts->input,
1865 ABS_MT_POSITION_Y,
1866 cur_mt_pos[snd_trk[id]][CY_YPOS]);
1867 CY_MT_SYNC(ts->input);
1868 cyttsp_debug("MT1->TID:%2d X:%3d Y:%3d Z:%3d touch-sent\n", \
1869 snd_trk[id], \
1870 cur_mt_pos[snd_trk[id]][CY_XPOS], \
1871 cur_mt_pos[snd_trk[id]][CY_YPOS], \
1872 cur_mt_z[snd_trk[id]]);
1873 } else if (ts->prv_mt_tch[id] < CY_NUM_TRK_ID) {
1874 /* void out this touch */
1875 input_report_abs(ts->input,
1876 ABS_MT_TOUCH_MAJOR,
1877 CY_NTCH);
1878 input_report_abs(ts->input,
1879 ABS_MT_WIDTH_MAJOR,
1880 curr_tool_width);
1881 input_report_abs(ts->input,
1882 ABS_MT_POSITION_X,
1883 ts->prv_mt_pos[ts->prv_mt_tch[id]][CY_XPOS]);
1884 input_report_abs(ts->input,
1885 ABS_MT_POSITION_Y,
1886 ts->prv_mt_pos[ts->prv_mt_tch[id]][CY_YPOS]);
1887 CY_MT_SYNC(ts->input);
1888 cyttsp_debug("MT2->TID:%2d X:%3d Y:%3d Z:%3d lift off-sent\n", \
1889 ts->prv_mt_tch[id], \
1890 ts->prv_mt_pos[ts->prv_mt_tch[id]][CY_XPOS], \
1891 ts->prv_mt_pos[ts->prv_mt_tch[id]][CY_YPOS], \
1892 CY_NTCH);
1893 } else {
1894 /* do not stuff any signals for this
1895 * previously and currently
1896 * void touches */
1897 cyttsp_xdebug("MT3->send[%d]=%d - No touch - NOT sent\n", \
1898 id, snd_trk[id]);
1899 }
1900 }
1901
1902 /* save current posted tracks to
1903 * previous track memory */
1904 for (id = 0; id < CY_NUM_MT_TCH_ID; id++) {
1905 ts->prv_mt_tch[id] = snd_trk[id];
1906 if (snd_trk[id] < CY_NUM_TRK_ID) {
1907 ts->prv_mt_pos[snd_trk[id]][CY_XPOS] =
1908 cur_mt_pos[snd_trk[id]][CY_XPOS];
1909 ts->prv_mt_pos[snd_trk[id]][CY_YPOS] =
1910 cur_mt_pos[snd_trk[id]][CY_YPOS];
1911 cyttsp_xdebug("MT4->TID:%2d X:%3d Y:%3d Z:%3d save for previous\n", \
1912 snd_trk[id], \
1913 ts->prv_mt_pos[snd_trk[id]][CY_XPOS], \
1914 ts->prv_mt_pos[snd_trk[id]][CY_YPOS], \
1915 CY_NTCH);
1916 }
1917 }
1918 for (id = 0; id < CY_NUM_TRK_ID; id++)
1919 ts->act_trk[id] = CY_NTCH;
1920 for (id = 0; id < CY_NUM_MT_TCH_ID; id++) {
1921 if (snd_trk[id] < CY_NUM_TRK_ID)
1922 ts->act_trk[snd_trk[id]] = CY_TCH;
1923 }
1924 }
1925 }
1926
1927 /* handle gestures */
1928 if (ts->platform_data->use_gestures) {
1929 if (g_xy_data.gest_id) {
1930 input_report_key(ts->input,
1931 BTN_3, CY_TCH);
1932 input_report_abs(ts->input,
1933 ABS_HAT1X, g_xy_data.gest_id);
1934 input_report_abs(ts->input,
1935 ABS_HAT2Y, g_xy_data.gest_cnt);
1936 }
1937 }
1938
1939 /* signal the view motion event */
1940 input_sync(ts->input);
1941
1942 for (id = 0; id < CY_NUM_TRK_ID; id++) {
1943 /* update platform data for the current MT information */
1944 ts->act_trk[id] = cur_trk[id];
1945 }
1946
1947exit_xy_handler:
1948 /* restart event timer */
1949 if (ts->client->irq == 0)
1950 mod_timer(&ts->timer, jiffies + TOUCHSCREEN_TIMEOUT);
1951 return;
1952}
1953
1954static int cyttsp_inlist(u16 prev_track[], u8 cur_trk_id,
1955 u8 *prev_loc, u8 num_touches)
1956{
1957 u8 id = 0;
1958
1959 *prev_loc = CY_IGNR_TCH;
1960
1961 cyttsp_xdebug("IN p[%d]=%d c=%d n=%d loc=%d\n", \
1962 id, prev_track[id], cur_trk_id, \
1963 num_touches, *prev_loc);
1964 for (id = 0, *prev_loc = CY_IGNR_TCH;
1965 (id < num_touches); id++) {
1966 cyttsp_xdebug("p[%d]=%d c=%d n=%d loc=%d\n", \
1967 id, prev_track[id], cur_trk_id, \
1968 num_touches, *prev_loc);
1969 if (prev_track[id] == cur_trk_id) {
1970 *prev_loc = id;
1971 break;
1972 }
1973 }
1974 cyttsp_xdebug("OUT p[%d]=%d c=%d n=%d loc=%d\n", \
1975 id, prev_track[id], cur_trk_id, num_touches, *prev_loc);
1976
1977 return ((*prev_loc < CY_NUM_TRK_ID) ? true : false);
1978}
1979
1980static int cyttsp_next_avail_inlist(u16 cur_trk[],
1981 u8 *new_loc, u8 num_touches)
1982{
1983 u8 id;
1984
1985 for (id = 0, *new_loc = CY_IGNR_TCH;
1986 (id < num_touches); id++) {
1987 if (cur_trk[id] > CY_NUM_TRK_ID) {
1988 *new_loc = id;
1989 break;
1990 }
1991 }
1992
1993 return ((*new_loc < CY_NUM_TRK_ID) ? true : false);
1994}
1995
1996/* Timer function used as dummy interrupt driver */
1997static void cyttsp_timer(unsigned long handle)
1998{
1999 struct cyttsp *ts = (struct cyttsp *) handle;
2000
2001 cyttsp_xdebug("TTSP Device timer event\n");
2002
2003 /* schedule motion signal handling */
2004 cyttsp_xy_handler(ts);
2005
2006 return;
2007}
2008
2009
2010
2011/* ************************************************************************
2012 * ISR function. This function is general, initialized in drivers init
2013 * function
2014 * ************************************************************************ */
2015static irqreturn_t cyttsp_irq(int irq, void *handle)
2016{
2017 struct cyttsp *ts = (struct cyttsp *) handle;
Amy Maloche304d9f52012-06-14 14:22:50 -07002018 int retval = 0;
Steve Mucklef132c6c2012-06-06 18:30:57 -07002019
2020 cyttsp_xdebug("%s: Got IRQ\n", CY_I2C_NAME);
Amy Maloche304d9f52012-06-14 14:22:50 -07002021 switch (ts->power_state) {
2022 case CY_SYSINFO:
2023 retval = i2c_smbus_read_i2c_block_data(ts->client,
2024 CY_REG_BASE,
2025 sizeof(struct cyttsp_sysinfo_data_t),
2026 (u8 *)&g_sysinfo_data);
2027 if (retval < 0) {
2028 pr_err("%s: Fail read status and version regs r=%d\n",
2029 __func__, retval);
2030 goto cyttsp_irq_sysinfo_exit;
2031 }
2032 if (ts->platform_data->use_hndshk) {
2033 retval = cyttsp_hndshk(ts, g_sysinfo_data.hst_mode);
2034 if (retval < 0) {
2035 pr_err("%s: Fail write handshake r=%d\n",
2036 __func__, retval);
2037 retval = 0;
2038 }
2039 }
2040 udelay(100); /* irq pulse: sysinfo mode switch=50us */
2041 complete(&ts->si_int_running);
2042cyttsp_irq_sysinfo_exit:
2043 break;
2044 case CY_ACTIVE:
2045 cyttsp_xy_handler(ts);
2046 break;
2047 default:
2048 pr_err("%s: Unexpected power state with interrupt ps=%d\n",
2049 __func__, ts->power_state);
2050 break;
2051 }
Steve Mucklef132c6c2012-06-06 18:30:57 -07002052
2053 return IRQ_HANDLED;
2054}
2055
2056/* ************************************************************************
2057 * Probe initialization functions
2058 * ************************************************************************ */
2059static int cyttsp_putbl(struct cyttsp *ts, int show,
2060 int show_status, int show_version, int show_cid)
2061{
2062 int retval = CY_OK;
2063
2064 int num_bytes = (show_status * 3) + (show_version * 6) + (show_cid * 3);
2065
2066 if (show_cid)
2067 num_bytes = sizeof(struct cyttsp_bootloader_data_t);
2068 else if (show_version)
2069 num_bytes = sizeof(struct cyttsp_bootloader_data_t) - 3;
2070 else
2071 num_bytes = sizeof(struct cyttsp_bootloader_data_t) - 9;
2072
2073 if (show) {
2074 retval = i2c_smbus_read_i2c_block_data(ts->client,
2075 CY_REG_BASE, num_bytes, (u8 *)&g_bl_data);
2076 if (show_status) {
2077 cyttsp_debug("BL%d: f=%02X s=%02X err=%02X bl=%02X%02X bld=%02X%02X\n", \
2078 show, \
2079 g_bl_data.bl_file, \
2080 g_bl_data.bl_status, \
2081 g_bl_data.bl_error, \
2082 g_bl_data.blver_hi, g_bl_data.blver_lo, \
2083 g_bl_data.bld_blver_hi, g_bl_data.bld_blver_lo);
2084 }
2085 if (show_version) {
2086 cyttsp_debug("BL%d: ttspver=0x%02X%02X appid=0x%02X%02X appver=0x%02X%02X\n", \
2087 show, \
2088 g_bl_data.ttspver_hi, g_bl_data.ttspver_lo, \
2089 g_bl_data.appid_hi, g_bl_data.appid_lo, \
2090 g_bl_data.appver_hi, g_bl_data.appver_lo);
2091 }
2092 if (show_cid) {
2093 cyttsp_debug("BL%d: cid=0x%02X%02X%02X\n", \
2094 show, \
2095 g_bl_data.cid_0, \
2096 g_bl_data.cid_1, \
2097 g_bl_data.cid_2);
2098 }
2099 }
2100
2101 return retval;
2102}
2103
2104#ifdef CY_INCLUDE_LOAD_FILE
2105#define CY_MAX_I2C_LEN 256
2106#define CY_MAX_TRY 10
2107#define CY_BL_PAGE_SIZE 16
2108#define CY_BL_NUM_PAGES 5
2109static int cyttsp_i2c_wr_blk_chunks(struct cyttsp *ts, u8 command,
2110 u8 length, const u8 *values)
2111{
2112 int retval = CY_OK;
2113 int block = 1;
2114
2115 u8 dataray[CY_MAX_I2C_LEN];
2116
2117 /* first page already includes the bl page offset */
2118 retval = i2c_smbus_write_i2c_block_data(ts->client, CY_REG_BASE,
2119 CY_BL_PAGE_SIZE+1, values);
2120 values += CY_BL_PAGE_SIZE+1;
2121 length -= CY_BL_PAGE_SIZE+1;
2122
2123 /* rem blocks require bl page offset stuffing */
2124 while (length &&
2125 (block < CY_BL_NUM_PAGES) &&
2126 !(retval < CY_OK)) {
2127 udelay(43*2); /* TRM * 2 */
2128 dataray[0] = CY_BL_PAGE_SIZE*block;
2129 memcpy(&dataray[1], values,
2130 length >= CY_BL_PAGE_SIZE ?
2131 CY_BL_PAGE_SIZE : length);
2132 retval = i2c_smbus_write_i2c_block_data(ts->client,
2133 CY_REG_BASE,
2134 length >= CY_BL_PAGE_SIZE ?
2135 CY_BL_PAGE_SIZE + 1 : length+1, dataray);
2136 values += CY_BL_PAGE_SIZE;
2137 length = length >= CY_BL_PAGE_SIZE ?
2138 length - CY_BL_PAGE_SIZE : 0;
2139 block++;
2140 }
2141
2142 return retval;
2143}
2144
2145static int cyttsp_bootload_app(struct cyttsp *ts)
2146{
2147 int retval = CY_OK;
2148 int i, tries;
2149 u8 host_reg;
2150
2151 cyttsp_debug("load new firmware \n");
2152 /* reset TTSP Device back to bootloader mode */
2153 host_reg = CY_SOFT_RESET_MODE;
2154 retval = i2c_smbus_write_i2c_block_data(ts->client, CY_REG_BASE,
2155 sizeof(host_reg), &host_reg);
2156 /* wait for TTSP Device to complete reset back to bootloader */
2157 tries = 0;
2158 do {
2159 usleep_range(1000, 1000);
2160 cyttsp_putbl(ts, 3, false, false, false);
2161 } while (g_bl_data.bl_status != 0x10 &&
2162 g_bl_data.bl_status != 0x11 &&
2163 tries++ < 100);
2164 cyttsp_debug("load file - tver=0x%02X%02X a_id=0x%02X%02X aver=0x%02X%02X\n", \
2165 cyttsp_fw_tts_verh, cyttsp_fw_tts_verl, \
2166 cyttsp_fw_app_idh, cyttsp_fw_app_idl, \
2167 cyttsp_fw_app_verh, cyttsp_fw_app_verl);
2168
2169 /* download new TTSP Application to the Bootloader */
2170 if (!(retval < CY_OK)) {
2171 i = 0;
2172 /* send bootload initiation command */
2173 if (cyttsp_fw[i].Command == CY_BL_INIT_LOAD) {
2174 g_bl_data.bl_file = 0;
2175 g_bl_data.bl_status = 0;
2176 g_bl_data.bl_error = 0;
2177 retval = i2c_smbus_write_i2c_block_data(ts->client,
2178 CY_REG_BASE,
2179 cyttsp_fw[i].Length, cyttsp_fw[i].Block);
2180 /* delay to allow bl to get ready for block writes */
2181 i++;
2182 tries = 0;
2183 do {
2184 msleep(100);
2185 cyttsp_putbl(ts, 4, false, false, false);
2186 } while (g_bl_data.bl_status != 0x10 &&
2187 g_bl_data.bl_status != 0x11 &&
2188 tries++ < 100);
2189 cyttsp_debug("wait init f=%02X, s=%02X, e=%02X t=%d\n", \
2190 g_bl_data.bl_file, g_bl_data.bl_status, \
2191 g_bl_data.bl_error, tries);
2192 /* send bootload firmware load blocks */
2193 if (!(retval < CY_OK)) {
2194 while (cyttsp_fw[i].Command == CY_BL_WRITE_BLK) {
2195 retval = cyttsp_i2c_wr_blk_chunks(ts,
2196 CY_REG_BASE,
2197 cyttsp_fw[i].Length,
2198 cyttsp_fw[i].Block);
2199 cyttsp_xdebug("BL DNLD Rec=% 3d Len=% 3d Addr=%04X\n", \
2200 cyttsp_fw[i].Record, \
2201 cyttsp_fw[i].Length, \
2202 cyttsp_fw[i].Address);
2203 i++;
2204 if (retval < CY_OK) {
2205 cyttsp_debug("BL fail Rec=%3d retval=%d\n", \
2206 cyttsp_fw[i-1].Record, \
2207 retval);
2208 break;
2209 } else {
2210 tries = 0;
2211 cyttsp_putbl(ts, 5, false, false, false);
2212 while (!((g_bl_data.bl_status == 0x10) &&
2213 (g_bl_data.bl_error == 0x20)) &&
2214 !((g_bl_data.bl_status == 0x11) &&
2215 (g_bl_data.bl_error == 0x20)) &&
2216 (tries++ < 100)) {
2217 usleep_range(1000, 1000);
2218 cyttsp_putbl(ts, 5, false, false, false);
2219 }
2220 }
2221 }
2222
2223 if (!(retval < CY_OK)) {
2224 while (i < cyttsp_fw_records) {
2225 retval = i2c_smbus_write_i2c_block_data(ts->client, CY_REG_BASE,
2226 cyttsp_fw[i].Length,
2227 cyttsp_fw[i].Block);
2228 i++;
2229 tries = 0;
2230 do {
2231 msleep(100);
2232 cyttsp_putbl(ts, 6, true, false, false);
2233 } while (g_bl_data.bl_status != 0x10 &&
2234 g_bl_data.bl_status != 0x11 &&
2235 tries++ < 100);
2236 cyttsp_debug("wait term f=%02X, s=%02X, e=%02X t=%d\n", \
2237 g_bl_data.bl_file, \
2238 g_bl_data.bl_status, \
2239 g_bl_data.bl_error, \
2240 tries);
2241 if (retval < CY_OK)
2242 break;
2243 }
2244 }
2245 }
2246 }
2247 }
2248
2249 /* reset TTSP Device back to bootloader mode */
2250 host_reg = CY_SOFT_RESET_MODE;
2251 retval = i2c_smbus_write_i2c_block_data(ts->client, CY_REG_BASE,
2252 sizeof(host_reg), &host_reg);
2253 /* wait for TTSP Device to complete reset back to bootloader */
2254 tries = 0;
2255 do {
2256 usleep_range(1000, 1000);
2257 cyttsp_putbl(ts, 3, false, false, false);
2258 } while (g_bl_data.bl_status != 0x10 &&
2259 g_bl_data.bl_status != 0x11 &&
2260 tries++ < 100);
2261
2262 /* set arg2 to non-0 to activate */
2263 retval = cyttsp_putbl(ts, 8, true, true, true);
2264
2265 return retval;
2266}
2267#else
2268static int cyttsp_bootload_app(struct cyttsp *ts)
2269{
2270 cyttsp_debug("no-load new firmware \n");
2271 return CY_OK;
2272}
2273#endif /* CY_INCLUDE_LOAD_FILE */
2274
2275
2276static int cyttsp_power_on(struct cyttsp *ts)
2277{
2278 int retval = CY_OK;
2279 u8 host_reg;
2280 int tries;
2281
2282 cyttsp_debug("Power up \n");
2283
2284 /* check if the TTSP device has a bootloader installed */
2285 host_reg = CY_SOFT_RESET_MODE;
2286 retval = i2c_smbus_write_i2c_block_data(ts->client, CY_REG_BASE,
2287 sizeof(host_reg), &host_reg);
2288 tries = 0;
2289 do {
2290 usleep_range(1000, 1000);
2291
2292 /* set arg2 to non-0 to activate */
2293 retval = cyttsp_putbl(ts, 1, true, true, true);
2294 cyttsp_info("BL%d: f=%02X s=%02X err=%02X bl=%02X%02X bld=%02X%02X R=%d\n", \
2295 101, \
2296 g_bl_data.bl_file, g_bl_data.bl_status, \
2297 g_bl_data.bl_error, \
2298 g_bl_data.blver_hi, g_bl_data.blver_lo, \
2299 g_bl_data.bld_blver_hi, g_bl_data.bld_blver_lo,
2300 retval);
2301 cyttsp_info("BL%d: tver=%02X%02X a_id=%02X%02X aver=%02X%02X\n", \
2302 102, \
2303 g_bl_data.ttspver_hi, g_bl_data.ttspver_lo, \
2304 g_bl_data.appid_hi, g_bl_data.appid_lo, \
2305 g_bl_data.appver_hi, g_bl_data.appver_lo);
2306 cyttsp_info("BL%d: c_id=%02X%02X%02X\n", \
2307 103, \
2308 g_bl_data.cid_0, g_bl_data.cid_1, g_bl_data.cid_2);
2309 } while (!(retval < CY_OK) &&
2310 !GET_BOOTLOADERMODE(g_bl_data.bl_status) &&
2311 !(g_bl_data.bl_file == CY_OP_MODE + CY_LOW_PWR_MODE) &&
2312 tries++ < 100);
2313
2314 /* is bootloader missing? */
2315 if (!(retval < CY_OK)) {
2316 cyttsp_xdebug("Ret=%d Check if bootloader is missing...\n", \
2317 retval);
2318 if (!GET_BOOTLOADERMODE(g_bl_data.bl_status)) {
2319 /* skip all bl and sys info and go to op mode */
2320 if (!(retval < CY_OK)) {
2321 cyttsp_xdebug("Bl is missing (ret=%d)\n", \
2322 retval);
2323 host_reg = CY_OP_MODE/* + CY_LOW_PWR_MODE*/;
2324 retval = i2c_smbus_write_i2c_block_data(ts->client, CY_REG_BASE,
2325 sizeof(host_reg), &host_reg);
2326 /* wait for TTSP Device to complete switch to
2327 * Operational mode */
2328 msleep(1000);
2329 goto bypass;
2330 }
2331 }
2332 }
2333
2334
2335 /* take TTSP out of bootloader mode; go to TrueTouch operational mode */
2336 if (!(retval < CY_OK)) {
2337 cyttsp_xdebug1("exit bootloader; go operational\n");
2338 tries = 0;
2339 do {
2340 msleep(100);
2341 retval = i2c_smbus_write_i2c_block_data(ts->client,
2342 CY_REG_BASE, sizeof(bl_cmd), bl_cmd);
2343 if (retval == CY_OK)
2344 break;
2345 } while (tries++ < 5);
2346
2347 if (retval == CY_OK) {
2348 tries = 0;
2349 do {
2350 msleep(100);
2351 cyttsp_putbl(ts, 4, true, false, false);
2352 cyttsp_info("BL%d: f=%02X s=%02X err=%02X" \
2353 "bl=%02X%02X bld=%02X%02X\n", 104, \
2354 g_bl_data.bl_file, \
2355 g_bl_data.bl_status, \
2356 g_bl_data.bl_error, \
2357 g_bl_data.blver_hi, \
2358 g_bl_data.blver_lo, \
2359 g_bl_data.bld_blver_hi, \
2360 g_bl_data.bld_blver_lo);
2361 } while (GET_BOOTLOADERMODE(g_bl_data.bl_status) &&
2362 tries++ < 5);
2363 }
2364 }
2365
2366 if (!(retval < CY_OK) &&
2367 cyttsp_app_load()) {
2368 if (CY_DIFF(g_bl_data.ttspver_hi, cyttsp_tts_verh()) ||
2369 CY_DIFF(g_bl_data.ttspver_lo, cyttsp_tts_verl()) ||
2370 CY_DIFF(g_bl_data.appid_hi, cyttsp_app_idh()) ||
2371 CY_DIFF(g_bl_data.appid_lo, cyttsp_app_idl()) ||
2372 CY_DIFF(g_bl_data.appver_hi, cyttsp_app_verh()) ||
2373 CY_DIFF(g_bl_data.appver_lo, cyttsp_app_verl()) ||
2374 CY_DIFF(g_bl_data.cid_0, cyttsp_cid_0()) ||
2375 CY_DIFF(g_bl_data.cid_1, cyttsp_cid_1()) ||
2376 CY_DIFF(g_bl_data.cid_2, cyttsp_cid_2()) ||
2377 cyttsp_force_fw_load()) {
2378 cyttsp_debug("blttsp=0x%02X%02X flttsp=0x%02X%02X force=%d\n", \
2379 g_bl_data.ttspver_hi, g_bl_data.ttspver_lo, \
2380 cyttsp_tts_verh(), cyttsp_tts_verl(), \
2381 cyttsp_force_fw_load());
2382 cyttsp_debug("blappid=0x%02X%02X flappid=0x%02X%02X\n", \
2383 g_bl_data.appid_hi, g_bl_data.appid_lo, \
2384 cyttsp_app_idh(), cyttsp_app_idl());
2385 cyttsp_debug("blappver=0x%02X%02X flappver=0x%02X%02X\n", \
2386 g_bl_data.appver_hi, g_bl_data.appver_lo, \
2387 cyttsp_app_verh(), cyttsp_app_verl());
2388 cyttsp_debug("blcid=0x%02X%02X%02X flcid=0x%02X%02X%02X\n", \
2389 g_bl_data.cid_0, \
2390 g_bl_data.cid_1, \
2391 g_bl_data.cid_2, \
2392 cyttsp_cid_0(), \
2393 cyttsp_cid_1(), \
2394 cyttsp_cid_2());
2395 /* enter bootloader to load new app into TTSP Device */
2396 retval = cyttsp_bootload_app(ts);
2397 /* take TTSP device out of bootloader mode;
2398 * switch back to TrueTouch operational mode */
2399 if (!(retval < CY_OK)) {
2400 retval = i2c_smbus_write_i2c_block_data(ts->client,
2401 CY_REG_BASE,
2402 sizeof(bl_cmd), bl_cmd);
2403 /* wait for TTSP Device to complete
2404 * switch to Operational mode */
2405 tries = 0;
2406 do {
2407 msleep(100);
2408 cyttsp_putbl(ts, 9, false, false, false);
2409 } while (GET_BOOTLOADERMODE(g_bl_data.bl_status) &&
2410 tries++ < 100);
2411 cyttsp_putbl(ts, 9, true, false, false);
2412 }
2413 }
2414 }
2415
2416bypass:
Steve Mucklef132c6c2012-06-06 18:30:57 -07002417 if (!(retval < CY_OK))
2418 ts->platform_data->power_state = CY_ACTIVE_STATE;
2419 else
2420 ts->platform_data->power_state = CY_IDLE_STATE;
2421
2422 cyttsp_debug("Retval=%d Power state is %s\n", \
2423 retval, \
2424 ts->platform_data->power_state == CY_ACTIVE_STATE ? \
2425 "ACTIVE" : "IDLE");
2426
2427 return retval;
2428}
2429
2430static int cyttsp_power_device(struct cyttsp *ts, bool on)
2431{
2432 int rc = 0, i;
2433 const struct cyttsp_regulator *reg_info =
2434 ts->platform_data->regulator_info;
2435 u8 num_reg = ts->platform_data->num_regulators;
2436
2437 if (!reg_info) {
2438 pr_err("regulator pdata not specified\n");
2439 return -EINVAL;
2440 }
2441
2442 if (on == false) /* Turn off the regulators */
2443 goto ts_reg_disable;
2444
2445 ts->vdd = kzalloc(num_reg * sizeof(struct regulator *), GFP_KERNEL);
2446 if (!ts->vdd) {
2447 pr_err("unable to allocate memory\n");
2448 return -ENOMEM;
2449 }
2450
2451 for (i = 0; i < num_reg; i++) {
2452 ts->vdd[i] = regulator_get(&ts->client->dev, reg_info[i].name);
2453 if (IS_ERR(ts->vdd[i])) {
2454 rc = PTR_ERR(ts->vdd[i]);
2455 pr_err("%s:regulator get failed rc=%d\n",
2456 __func__, rc);
2457 goto error_vdd;
2458 }
2459
2460 if (regulator_count_voltages(ts->vdd[i]) > 0) {
2461 rc = regulator_set_voltage(ts->vdd[i],
2462 reg_info[i].min_uV, reg_info[i].max_uV);
2463 if (rc) {
2464 pr_err("%s: regulator_set_voltage"
2465 "failed rc =%d\n", __func__, rc);
2466 regulator_put(ts->vdd[i]);
2467 goto error_vdd;
2468 }
2469
2470 rc = regulator_set_optimum_mode(ts->vdd[i],
2471 reg_info[i].hpm_load_uA);
2472 if (rc < 0) {
2473 pr_err("%s: regulator_set_optimum_mode failed "
2474 "rc=%d\n", __func__, rc);
2475
2476 regulator_set_voltage(ts->vdd[i], 0,
2477 reg_info[i].max_uV);
2478 regulator_put(ts->vdd[i]);
2479 goto error_vdd;
2480 }
2481 }
2482
2483 rc = regulator_enable(ts->vdd[i]);
2484 if (rc) {
2485 pr_err("%s: regulator_enable failed rc =%d\n",
2486 __func__, rc);
2487 if (regulator_count_voltages(ts->vdd[i]) > 0) {
2488 regulator_set_optimum_mode(ts->vdd[i], 0);
2489 regulator_set_voltage(ts->vdd[i], 0,
2490 reg_info[i].max_uV);
2491 }
2492 regulator_put(ts->vdd[i]);
2493 goto error_vdd;
2494 }
2495 }
2496
2497 return rc;
2498
2499ts_reg_disable:
2500 i = ts->platform_data->num_regulators;
2501error_vdd:
2502 while (--i >= 0) {
2503 if (regulator_count_voltages(ts->vdd[i]) > 0) {
2504 regulator_set_voltage(ts->vdd[i], 0,
2505 reg_info[i].max_uV);
2506 regulator_set_optimum_mode(ts->vdd[i], 0);
2507 }
2508 regulator_disable(ts->vdd[i]);
2509 regulator_put(ts->vdd[i]);
2510 }
2511 kfree(ts->vdd);
2512 return rc;
2513}
2514
Amy Maloche304d9f52012-06-14 14:22:50 -07002515static void sysinfo_debug_msg(struct cyttsp *ts)
2516{
2517 cyttsp_debug("SI2: hst_mode=0x%02X mfg_cmd=0x%02X " \
2518 "mfg_stat=0x%02X\n", \
2519 g_sysinfo_data.hst_mode, \
2520 g_sysinfo_data.mfg_cmd, \
2521 g_sysinfo_data.mfg_stat);
2522 cyttsp_debug("SI2: bl_ver=0x%02X%02X\n", \
2523 g_sysinfo_data.bl_verh, \
2524 g_sysinfo_data.bl_verl);
2525 cyttsp_debug("SI2: sysinfo act_int=0x%02X " \
2526 "tch_tmout=0x%02X lp_int=0x%02X\n", \
2527 g_sysinfo_data.act_intrvl, \
2528 g_sysinfo_data.tch_tmout, \
2529 g_sysinfo_data.lp_intrvl);
2530 cyttsp_info("SI%d: tver=%02X%02X a_id=%02X%02X " \
2531 "aver=%02X%02X\n", \
2532 102, \
2533 g_sysinfo_data.tts_verh, \
2534 g_sysinfo_data.tts_verl, \
2535 g_sysinfo_data.app_idh, \
2536 g_sysinfo_data.app_idl, \
2537 g_sysinfo_data.app_verh, \
2538 g_sysinfo_data.app_verl);
2539 cyttsp_info("SI%d: c_id=%02X%02X%02X\n", \
2540 103, \
2541 g_sysinfo_data.cid[0], \
2542 g_sysinfo_data.cid[1], \
2543 g_sysinfo_data.cid[2]);
2544 cyttsp_debug("SI2: platinfo " \
2545 "act_intrvl=0x%02X tch_tmout=0x%02X " \
2546 "lp_intrvl=0x%02X\n", \
2547 ts->platform_data->act_intrvl, \
2548 ts->platform_data->tch_tmout, \
2549 ts->platform_data->lp_intrvl);
2550}
2551
2552static int set_bypass_modes(struct cyttsp *ts)
2553{
2554 int retval = 0, tries = 0;
2555
2556 /* switch to System Information mode to read versions
2557 * and set interval registers */
2558 retval = cyttsp_set_sysinfo_mode(ts, CY_LOW_PWR_MODE);
2559 if (!(retval < CY_OK)) {
2560 retval = i2c_smbus_read_i2c_block_data(ts->client,
2561 CY_REG_BASE,
2562 sizeof(struct cyttsp_sysinfo_data_t),
2563 (u8 *)&g_sysinfo_data);
2564 sysinfo_debug_msg(ts);
2565 /* set power settings registers */
2566 do {
2567 retval = i2c_smbus_write_i2c_block_data(ts->client,
2568 CY_REG_ACT_INTRVL, sizeof(ts->power_settings),
2569 ts->power_settings);
2570 if (retval < 0)
2571 msleep(20);
2572 } while ((retval < 0) && (tries++ < 5));
2573 if (retval < 0)
2574 pr_err("%s: failed to write power_settings, " \
2575 "retval =%x\n", __func__, retval);
2576 }
2577 /* switch back to Operational mode */
2578 cyttsp_debug("switch back to operational mode\n");
2579 if (!(retval < CY_OK)) {
2580 cyttsp_set_opmode(ts, CY_LOW_PWR_MODE);
2581 /* wait for TTSP Device to complete
2582 * switch to Operational mode */
2583 msleep(100);
2584 }
2585 /* init gesture setup;
2586 * this is required even if not using gestures
2587 * in order to set the active distance */
2588 if (!(retval < CY_OK)) {
2589 u8 gesture_setup;
2590 cyttsp_debug("init gesture setup\n");
2591 gesture_setup = ts->platform_data->gest_set;
2592 retval = i2c_smbus_write_i2c_block_data(ts->client,
2593 CY_REG_GEST_SET,
2594 sizeof(gesture_setup), &gesture_setup);
2595 msleep(CY_DLY_DFLT);
2596 }
2597 return retval;
2598}
2599
Steve Mucklef132c6c2012-06-06 18:30:57 -07002600/* cyttsp_initialize: Driver Initialization. This function takes
2601 * care of the following tasks:
2602 * 1. Create and register an input device with input layer
2603 * 2. Take CYTTSP device out of bootloader mode; go operational
2604 * 3. Start any timers/Work queues. */
2605static int cyttsp_initialize(struct i2c_client *client, struct cyttsp *ts)
2606{
2607 struct input_dev *input_device;
2608 int error = 0;
2609 int retval = CY_OK;
2610 u8 id;
2611
2612 /* Create the input device and register it. */
2613 input_device = input_allocate_device();
2614 if (!input_device) {
2615 error = -ENOMEM;
2616 cyttsp_xdebug1("err input allocate device\n");
2617 goto error_free_device;
2618 }
2619
2620 if (!client) {
2621 error = ~ENODEV;
2622 cyttsp_xdebug1("err client is Null\n");
2623 goto error_free_device;
2624 }
2625
2626 if (!ts) {
2627 error = ~ENODEV;
2628 cyttsp_xdebug1("err context is Null\n");
2629 goto error_free_device;
2630 }
2631
2632 ts->input = input_device;
2633 input_device->name = CY_I2C_NAME;
2634 input_device->phys = ts->phys;
2635 input_device->dev.parent = &client->dev;
2636
Amy Maloche304d9f52012-06-14 14:22:50 -07002637 ts->power_state = CY_ACTIVE;
2638
2639 if (ts->platform_data->act_intrvl)
2640 ts->power_settings[0] = ts->platform_data->act_intrvl;
2641 else
2642 ts->power_settings[0] = CY_ACT_INTRVL_DFLT;
2643 if (ts->platform_data->tch_tmout)
2644 ts->power_settings[1] = ts->platform_data->tch_tmout;
2645 else
2646 ts->power_settings[1] = CY_TCH_TMOUT_DFLT;
2647 if (ts->platform_data->lp_intrvl)
2648 ts->power_settings[2] = ts->platform_data->lp_intrvl;
2649 else
2650 ts->power_settings[2] = CY_LP_INTRVL_DFLT;
2651
Steve Mucklef132c6c2012-06-06 18:30:57 -07002652 /* init the touch structures */
2653 ts->num_prv_st_tch = CY_NTCH;
2654 for (id = 0; id < CY_NUM_TRK_ID; id++) {
2655 ts->act_trk[id] = CY_NTCH;
2656 ts->prv_mt_pos[id][CY_XPOS] = 0;
2657 ts->prv_mt_pos[id][CY_YPOS] = 0;
2658 }
2659
2660 for (id = 0; id < CY_NUM_MT_TCH_ID; id++)
2661 ts->prv_mt_tch[id] = CY_IGNR_TCH;
2662
2663 for (id = 0; id < CY_NUM_ST_TCH_ID; id++)
2664 ts->prv_st_tch[id] = CY_IGNR_TCH;
2665
2666 set_bit(EV_SYN, input_device->evbit);
2667 set_bit(EV_KEY, input_device->evbit);
2668 set_bit(EV_ABS, input_device->evbit);
2669 set_bit(BTN_TOUCH, input_device->keybit);
2670 set_bit(BTN_2, input_device->keybit);
Amy Malocheda85bc62012-06-19 14:11:33 -07002671 set_bit(INPUT_PROP_DIRECT, input_device->propbit);
2672
Steve Mucklef132c6c2012-06-06 18:30:57 -07002673 if (ts->platform_data->use_gestures)
2674 set_bit(BTN_3, input_device->keybit);
2675
2676 input_set_abs_params(input_device, ABS_X, ts->platform_data->disp_minx,
2677 ts->platform_data->disp_maxx, 0, 0);
2678 input_set_abs_params(input_device, ABS_Y, ts->platform_data->disp_miny,
2679 ts->platform_data->disp_maxy, 0, 0);
2680 input_set_abs_params(input_device,
2681 ABS_TOOL_WIDTH, 0, CY_LARGE_TOOL_WIDTH, 0 , 0);
2682 input_set_abs_params(input_device,
2683 ABS_PRESSURE, 0, CY_MAXZ, 0, 0);
2684 input_set_abs_params(input_device,
2685 ABS_HAT0X, 0, ts->platform_data->panel_maxx, 0, 0);
2686 input_set_abs_params(input_device,
2687 ABS_HAT0Y, 0, ts->platform_data->panel_maxy, 0, 0);
2688 if (ts->platform_data->use_gestures) {
2689 input_set_abs_params(input_device,
2690 ABS_HAT1X, 0, CY_MAXZ, 0, 0);
2691 input_set_abs_params(input_device,
2692 ABS_HAT1Y, 0, CY_MAXZ, 0, 0);
2693 }
2694 if (ts->platform_data->use_mt) {
2695 input_set_abs_params(input_device, ABS_MT_POSITION_X,
2696 ts->platform_data->disp_minx,
2697 ts->platform_data->disp_maxx, 0, 0);
2698 input_set_abs_params(input_device, ABS_MT_POSITION_Y,
2699 ts->platform_data->disp_miny,
2700 ts->platform_data->disp_maxy, 0, 0);
2701 input_set_abs_params(input_device,
2702 ABS_MT_TOUCH_MAJOR, 0, CY_MAXZ, 0, 0);
2703 input_set_abs_params(input_device,
2704 ABS_MT_WIDTH_MAJOR, 0, CY_LARGE_TOOL_WIDTH, 0, 0);
2705 if (ts->platform_data->use_trk_id) {
2706 input_set_abs_params(input_device,
2707 ABS_MT_TRACKING_ID, 0, CY_NUM_TRK_ID, 0, 0);
2708 }
2709 }
2710
2711 /* set dummy key to make driver work with virtual keys */
2712 input_set_capability(input_device, EV_KEY, KEY_PROG1);
2713
2714 cyttsp_info("%s: Register input device\n", CY_I2C_NAME);
2715 error = input_register_device(input_device);
2716 if (error) {
2717 cyttsp_alert("%s: Failed to register input device\n", \
2718 CY_I2C_NAME);
2719 retval = error;
2720 goto error_free_device;
2721 }
2722
2723 if (gpio_is_valid(ts->platform_data->resout_gpio)) {
2724 /* configure touchscreen reset out gpio */
2725 retval = gpio_request(ts->platform_data->resout_gpio,
2726 "cyttsp_resout_gpio");
2727 if (retval) {
2728 pr_err("%s: unable to request reset gpio %d\n",
2729 __func__, ts->platform_data->resout_gpio);
2730 goto error_free_device;
2731 }
2732
2733 retval = gpio_direction_output(
2734 ts->platform_data->resout_gpio, 1);
2735 if (retval) {
2736 pr_err("%s: unable to set direction for gpio %d\n",
2737 __func__, ts->platform_data->resout_gpio);
2738 goto error_resout_gpio_dir;
2739 }
2740 }
2741
2742 if (gpio_is_valid(ts->platform_data->sleep_gpio)) {
2743 /* configure touchscreen reset out gpio */
2744 retval = gpio_request(ts->platform_data->sleep_gpio,
2745 "cy8c_sleep_gpio");
2746 if (retval) {
2747 pr_err("%s: unable to request sleep gpio %d\n",
2748 __func__, ts->platform_data->sleep_gpio);
2749 goto error_sleep_gpio_req;
2750 }
2751
2752 retval = gpio_direction_output(
2753 ts->platform_data->sleep_gpio, 0);
2754 if (retval) {
2755 pr_err("%s: unable to set direction for gpio %d\n",
2756 __func__, ts->platform_data->resout_gpio);
2757 goto error_sleep_gpio_dir;
2758 }
2759 }
2760
2761 if (gpio_is_valid(ts->platform_data->irq_gpio)) {
2762 /* configure touchscreen irq gpio */
2763 retval = gpio_request(ts->platform_data->irq_gpio,
2764 "ts_irq_gpio");
2765 if (retval) {
2766 pr_err("%s: unable to request gpio [%d]\n", __func__,
2767 ts->platform_data->irq_gpio);
2768 goto error_irq_gpio_req;
2769 }
2770 retval = gpio_direction_input(ts->platform_data->irq_gpio);
2771 if (retval) {
2772 pr_err("%s: unable to set_direction for gpio [%d]\n",
2773 __func__, ts->platform_data->irq_gpio);
2774 goto error_irq_gpio_dir;
2775 }
2776 }
2777
2778 if (ts->platform_data->regulator_info) {
2779 retval = cyttsp_power_device(ts, true);
2780 if (retval) {
2781 pr_err("%s: Unable to power device %d\n",
2782 __func__, retval);
2783 goto error_irq_gpio_dir;
2784 }
2785 }
2786
2787 /* Power on the chip and make sure that I/Os are set as specified
2788 * in the platform */
2789 if (ts->platform_data->init) {
2790 retval = ts->platform_data->init(client);
2791 if (retval) {
2792 pr_err("%s: ts init failed\n", __func__);
2793 goto error_power_device;
2794 }
2795 }
2796
2797 msleep(100);
2798
2799 /* check this device active by reading first byte/register */
2800 retval = i2c_smbus_read_byte_data(ts->client, 0x01);
2801 if (retval < 0) {
2802 pr_err("%s: i2c sanity check failed\n", __func__);
2803 goto error_power_device;
2804 }
2805
2806 retval = cyttsp_power_on(ts);
2807 if (retval < 0) {
2808 pr_err("%s: cyttsp_power_on failed\n", __func__);
2809 goto error_power_device;
2810 }
2811
2812 /* Timer or Interrupt setup */
2813 if (ts->client->irq == 0) {
2814 cyttsp_info("Setting up timer\n");
2815 setup_timer(&ts->timer, cyttsp_timer, (unsigned long) ts);
2816 mod_timer(&ts->timer, jiffies + TOUCHSCREEN_TIMEOUT);
2817 } else {
2818 cyttsp_info("Setting up interrupt\n");
2819 error = request_threaded_irq(client->irq, NULL, cyttsp_irq,
2820 IRQF_TRIGGER_FALLING | IRQF_ONESHOT,
2821 client->dev.driver->name, ts);
2822 if (error) {
2823 cyttsp_alert("error: could not request irq\n");
2824 retval = error;
2825 goto error_power_device;
2826 }
2827 }
2828
2829 irq_cnt = 0;
2830 irq_cnt_total = 0;
2831 irq_err_cnt = 0;
2832
2833 atomic_set(&ts->irq_enabled, 1);
2834 retval = device_create_file(&ts->client->dev, &dev_attr_irq_enable);
2835 if (retval < CY_OK) {
2836 cyttsp_alert("File device creation failed: %d\n", retval);
2837 retval = -ENODEV;
2838 goto error_free_irq;
2839 }
2840
2841 retval = device_create_file(&client->dev, &dev_attr_cyttsp_fw_ver);
2842 if (retval) {
2843 cyttsp_alert("sysfs entry for firmware version failed\n");
2844 goto error_rm_dev_file_irq_en;
2845 }
2846
2847 ts->cyttsp_fwloader_mode = 0;
2848 retval = device_create_file(&client->dev, &dev_attr_cyttsp_update_fw);
2849 if (retval) {
2850 cyttsp_alert("sysfs entry for firmware update failed\n");
2851 goto error_rm_dev_file_fw_ver;
2852 }
2853
2854 retval = device_create_file(&client->dev,
2855 &dev_attr_cyttsp_force_update_fw);
2856 if (retval) {
2857 cyttsp_alert("sysfs entry for force firmware update failed\n");
2858 goto error_rm_dev_file_update_fw;
2859 }
2860 if (ts->platform_data->correct_fw_ver) {
2861 if (g_bl_data.appid_lo != ts->platform_data->correct_fw_ver)
2862 pr_warn("%s: Invalid firmware version detected;"
2863 " Please update.\n", __func__);
2864 }
2865
2866 retval = device_create_file(&client->dev,
2867 &dev_attr_cyttsp_fw_name);
2868 if (retval) {
2869 cyttsp_alert("sysfs entry for file name selection failed\n");
2870 goto error_rm_dev_file_fupdate_fw;
2871 }
2872
Amy Maloche304d9f52012-06-14 14:22:50 -07002873 set_bypass_modes(ts);
Steve Mucklef132c6c2012-06-06 18:30:57 -07002874 cyttsp_info("%s: Successful registration\n", CY_I2C_NAME);
2875
2876 goto success;
2877
2878error_rm_dev_file_fupdate_fw:
2879 device_remove_file(&client->dev, &dev_attr_cyttsp_force_update_fw);
2880error_rm_dev_file_update_fw:
2881 device_remove_file(&client->dev, &dev_attr_cyttsp_update_fw);
2882error_rm_dev_file_fw_ver:
2883 device_remove_file(&client->dev, &dev_attr_cyttsp_fw_ver);
2884error_rm_dev_file_irq_en:
2885 device_remove_file(&client->dev, &dev_attr_irq_enable);
2886error_free_irq:
2887 if (ts->client->irq)
2888 free_irq(client->irq, ts);
2889error_power_device:
2890 if (ts->platform_data->regulator_info)
2891 cyttsp_power_device(ts, false);
2892error_irq_gpio_dir:
2893 if (gpio_is_valid(ts->platform_data->irq_gpio))
2894 gpio_free(ts->platform_data->irq_gpio);
2895error_irq_gpio_req:
2896 if (gpio_is_valid(ts->platform_data->sleep_gpio))
2897 gpio_direction_output(ts->platform_data->sleep_gpio, 1);
2898error_sleep_gpio_dir:
2899 if (gpio_is_valid(ts->platform_data->sleep_gpio))
2900 gpio_free(ts->platform_data->sleep_gpio);
2901error_sleep_gpio_req:
2902 if (gpio_is_valid(ts->platform_data->resout_gpio))
2903 gpio_direction_output(ts->platform_data->resout_gpio, 0);
2904error_resout_gpio_dir:
2905 if (gpio_is_valid(ts->platform_data->resout_gpio))
2906 gpio_free(ts->platform_data->resout_gpio);
2907error_free_device:
2908 if (input_device)
2909 input_free_device(input_device);
2910
2911success:
2912 return retval;
2913}
2914
2915/* I2C driver probe function */
2916static int __devinit cyttsp_probe(struct i2c_client *client,
2917 const struct i2c_device_id *id)
2918{
2919 struct cyttsp *ts;
2920 int error;
2921 int retval = CY_OK;
2922
2923 cyttsp_info("Start Probe 1.2\n");
2924
2925 /* allocate and clear memory */
2926 ts = kzalloc(sizeof(struct cyttsp), GFP_KERNEL);
2927 if (ts == NULL) {
2928 cyttsp_xdebug1("err kzalloc for cyttsp\n");
2929 return -ENOMEM;
2930 }
2931
2932 /* Enable runtime PM ops, start in ACTIVE mode */
2933 error = pm_runtime_set_active(&client->dev);
2934 if (error < 0)
2935 dev_dbg(&client->dev, "unable to set runtime pm state\n");
2936 pm_runtime_enable(&client->dev);
2937
2938 if (!(retval < CY_OK)) {
2939 /* register driver_data */
2940 ts->client = client;
2941 ts->platform_data = client->dev.platform_data;
2942
2943 if (ts->platform_data->fw_fname)
2944 strlcpy(ts->fw_fname, ts->platform_data->fw_fname,
2945 FW_FNAME_LEN - 1);
2946 else
2947 strlcpy(ts->fw_fname, "cyttsp.hex", FW_FNAME_LEN - 1);
2948
2949 if (ts->platform_data->gen == CY_GEN3) {
2950 ts->fw_start_addr = 0x0b00;
2951 } else if (ts->platform_data->gen == CY_GEN2) {
2952 ts->fw_start_addr = 0x0880;
2953 } else {
2954 pr_err("%s: unsupported cypress chip\n", __func__);
2955 kfree(ts);
2956 return -EINVAL;
2957 }
2958
2959 i2c_set_clientdata(client, ts);
2960
Amy Maloche304d9f52012-06-14 14:22:50 -07002961 init_completion(&ts->si_int_running);
2962
Steve Mucklef132c6c2012-06-06 18:30:57 -07002963 error = cyttsp_initialize(client, ts);
2964 if (error) {
2965 cyttsp_xdebug1("err cyttsp_initialize\n");
2966 /* deallocate memory */
2967 kfree(ts);
2968/*
2969 i2c_del_driver(&cyttsp_driver);
2970*/
2971 return -ENODEV;
2972 }
2973 }
2974
2975#ifdef CONFIG_HAS_EARLYSUSPEND
2976 if (!(retval < CY_OK)) {
2977 ts->early_suspend.level = EARLY_SUSPEND_LEVEL_BLANK_SCREEN + 1;
2978 ts->early_suspend.suspend = cyttsp_early_suspend;
2979 ts->early_suspend.resume = cyttsp_late_resume;
2980 register_early_suspend(&ts->early_suspend);
2981 }
2982#endif /* CONFIG_HAS_EARLYSUSPEND */
2983 device_init_wakeup(&client->dev, ts->platform_data->wakeup);
2984 mutex_init(&ts->mutex);
Amy Maloche304d9f52012-06-14 14:22:50 -07002985 retval = cyttsp_set_lp_mode(ts);
Steve Mucklef132c6c2012-06-06 18:30:57 -07002986
2987 cyttsp_info("Start Probe %s\n", \
2988 (retval < CY_OK) ? "FAIL" : "PASS");
2989
2990 return retval;
2991}
2992
2993#ifdef CONFIG_PM
2994static int cyttsp_regulator_lpm(struct cyttsp *ts, bool on)
2995{
2996 int rc = 0, i;
2997 const struct cyttsp_regulator *reg_info =
2998 ts->platform_data->regulator_info;
2999 u8 num_reg = ts->platform_data->num_regulators;
3000
3001 if (on == false)
3002 goto regulator_hpm;
3003
3004 for (i = 0; i < num_reg; i++) {
3005 if (regulator_count_voltages(ts->vdd[i]) < 0)
3006 continue;
3007 rc = regulator_set_optimum_mode(ts->vdd[i],
3008 reg_info[i].lpm_load_uA);
3009 if (rc < 0) {
3010 pr_err("%s: regulator_set_optimum failed rc = %d\n",
3011 __func__, rc);
3012 goto fail_regulator_lpm;
3013 }
3014
3015 }
3016
3017 return 0;
3018
3019regulator_hpm:
3020 for (i = 0; i < num_reg; i++) {
3021 if (regulator_count_voltages(ts->vdd[i]) < 0)
3022 continue;
3023 rc = regulator_set_optimum_mode(ts->vdd[i],
3024 reg_info[i].hpm_load_uA);
3025 if (rc < 0) {
3026 pr_err("%s: regulator_set_optimum failed"
3027 "rc = %d\n", __func__, rc);
3028 goto fail_regulator_hpm;
3029 }
3030 }
3031
3032 return 0;
3033
3034fail_regulator_lpm:
3035 while (i--) {
3036 if (regulator_count_voltages(ts->vdd[i]) < 0)
3037 continue;
3038 regulator_set_optimum_mode(ts->vdd[i],
3039 reg_info[i].hpm_load_uA);
3040 }
3041
3042 return rc;
3043
3044fail_regulator_hpm:
3045 while (i--) {
3046 if (regulator_count_voltages(ts->vdd[i]) < 0)
3047 continue;
3048 regulator_set_optimum_mode(ts->vdd[i],
3049 reg_info[i].lpm_load_uA);
3050 }
3051
3052 return rc;
3053}
3054
3055/* Function to manage power-on resume */
3056static int cyttsp_resume(struct device *dev)
3057{
3058 struct cyttsp *ts = dev_get_drvdata(dev);
3059 int retval = CY_OK;
3060
3061 cyttsp_debug("Wake Up\n");
3062
3063 if (ts->is_suspended == false) {
3064 pr_err("%s: in wakeup state\n", __func__);
3065 return 0;
3066 }
3067
3068 if (device_may_wakeup(dev)) {
3069 if (ts->client->irq)
3070 disable_irq_wake(ts->client->irq);
3071 return 0;
3072 }
3073
3074 /* re-enable the interrupt prior to wake device */
3075 if (ts->client->irq)
3076 enable_irq(ts->client->irq);
3077
3078 if (ts->platform_data->use_sleep &&
3079 (ts->platform_data->power_state != CY_ACTIVE_STATE)) {
3080 if (ts->platform_data->resume)
3081 retval = ts->platform_data->resume(ts->client);
3082 else
3083 retval = cyttsp_regulator_lpm(ts, false);
3084 /* take TTSP device out of bootloader mode;
3085 * switch back to TrueTouch operational mode */
3086 if (!(retval < CY_OK)) {
3087 int tries = 0;
3088 do {
3089 msleep(100);
3090 retval = i2c_smbus_write_i2c_block_data(
3091 ts->client, CY_REG_BASE,
3092 sizeof(bl_cmd), bl_cmd);
3093 if (retval == CY_OK)
3094 break;
3095 } while (tries++ < 2);
3096 /* wait for TTSP Device to complete
3097 * switch to Operational mode */
3098 tries = 0;
3099 do {
3100 msleep(100);
3101 cyttsp_putbl(ts, 16, false, false, false);
3102 } while (GET_BOOTLOADERMODE(g_bl_data.bl_status) &&
3103 tries++ < 2);
3104 cyttsp_putbl(ts, 16, true, false, false);
3105 }
3106 }
3107
3108 if (!(retval < CY_OK) &&
3109 (GET_HSTMODE(g_bl_data.bl_file) == CY_OK)) {
3110 ts->platform_data->power_state = CY_ACTIVE_STATE;
3111
3112 /* re-enable the timer after resuming */
3113 if (ts->client->irq == 0)
3114 mod_timer(&ts->timer, jiffies + TOUCHSCREEN_TIMEOUT);
3115 } else
3116 retval = -ENODEV;
3117
3118 ts->is_suspended = false;
3119 cyttsp_debug("Wake Up %s\n", \
3120 (retval < CY_OK) ? "FAIL" : "PASS");
3121
Amy Maloche304d9f52012-06-14 14:22:50 -07003122 return cyttsp_set_lp_mode(ts);
Steve Mucklef132c6c2012-06-06 18:30:57 -07003123}
3124
3125/* Function to manage low power suspend */
3126static int cyttsp_suspend(struct device *dev)
3127{
3128 struct cyttsp *ts = dev_get_drvdata(dev);
3129 u8 sleep_mode = CY_OK;
3130 int retval = CY_OK;
3131
3132 cyttsp_debug("Enter Sleep\n");
3133
3134 if (ts->is_suspended == true) {
3135 pr_err("%s: in sleep state\n", __func__);
3136 return 0;
3137 }
3138
3139 mutex_lock(&ts->mutex);
3140 if (ts->cyttsp_fwloader_mode) {
3141 pr_err("%s:firmware upgrade mode:"
3142 "suspend not allowed\n", __func__);
3143 mutex_unlock(&ts->mutex);
3144 return -EBUSY;
3145 }
3146 mutex_unlock(&ts->mutex);
3147
3148 if (device_may_wakeup(dev)) {
3149 if (ts->client->irq)
3150 enable_irq_wake(ts->client->irq);
3151 return 0;
3152 }
3153
3154 if (ts->client->irq == 0)
3155 del_timer(&ts->timer);
3156 else
3157 disable_irq(ts->client->irq);
3158
3159 if (!(retval < CY_OK)) {
3160 if (ts->platform_data->use_sleep &&
3161 (ts->platform_data->power_state == CY_ACTIVE_STATE)) {
3162 if (ts->platform_data->suspend) {
3163 retval =
3164 ts->platform_data->suspend(ts->client);
3165 } else {
3166 retval = cyttsp_regulator_lpm(ts, true);
3167 }
3168 if (ts->platform_data->use_sleep & CY_USE_DEEP_SLEEP_SEL)
3169 sleep_mode = CY_DEEP_SLEEP_MODE;
3170 else
3171 sleep_mode = CY_LOW_PWR_MODE;
3172
3173 if (!(retval < CY_OK)) {
3174 retval =
3175 i2c_smbus_write_i2c_block_data(ts->client,
3176 CY_REG_BASE,
3177 sizeof(sleep_mode), &sleep_mode);
3178 }
3179 }
3180 }
3181
3182 if (!(retval < CY_OK)) {
3183 if (sleep_mode == CY_DEEP_SLEEP_MODE)
3184 ts->platform_data->power_state = CY_SLEEP_STATE;
3185 else if (sleep_mode == CY_LOW_PWR_MODE)
3186 ts->platform_data->power_state = CY_LOW_PWR_STATE;
3187 }
3188
3189 ts->is_suspended = true;
3190 cyttsp_debug("Sleep Power state is %s\n", \
3191 (ts->platform_data->power_state == CY_ACTIVE_STATE) ? \
3192 "ACTIVE" : \
3193 ((ts->platform_data->power_state == CY_SLEEP_STATE) ? \
3194 "SLEEP" : "LOW POWER"));
3195
3196 return retval;
3197}
3198#endif
3199
3200/* registered in driver struct */
3201static int __devexit cyttsp_remove(struct i2c_client *client)
3202{
3203 /* clientdata registered on probe */
3204 struct cyttsp *ts = i2c_get_clientdata(client);
3205 int err;
3206
3207 cyttsp_alert("Unregister\n");
3208
3209 pm_runtime_set_suspended(&client->dev);
3210 pm_runtime_disable(&client->dev);
3211
3212 device_init_wakeup(&client->dev, 0);
3213 device_remove_file(&ts->client->dev, &dev_attr_irq_enable);
3214 device_remove_file(&client->dev, &dev_attr_cyttsp_fw_ver);
3215 device_remove_file(&client->dev, &dev_attr_cyttsp_update_fw);
3216 device_remove_file(&client->dev, &dev_attr_cyttsp_force_update_fw);
3217 device_remove_file(&client->dev, &dev_attr_cyttsp_fw_name);
3218
3219 /* free up timer or irq */
3220 if (ts->client->irq == 0) {
3221 err = del_timer(&ts->timer);
3222 if (err < CY_OK)
3223 cyttsp_alert("error: failed to delete timer\n");
3224 } else
3225 free_irq(client->irq, ts);
3226
3227 if (ts->platform_data->regulator_info)
3228 cyttsp_power_device(ts, false);
3229
3230#ifdef CONFIG_HAS_EARLYSUSPEND
3231 unregister_early_suspend(&ts->early_suspend);
3232#endif /* CONFIG_HAS_EARLYSUSPEND */
3233
3234 mutex_destroy(&ts->mutex);
3235
3236 if (gpio_is_valid(ts->platform_data->sleep_gpio)) {
3237 gpio_direction_output(ts->platform_data->sleep_gpio, 1);
3238 gpio_free(ts->platform_data->sleep_gpio);
3239 }
3240
3241 if (gpio_is_valid(ts->platform_data->resout_gpio)) {
3242 gpio_direction_output(ts->platform_data->resout_gpio, 0);
3243 gpio_free(ts->platform_data->resout_gpio);
3244 }
3245
3246 if (gpio_is_valid(ts->platform_data->irq_gpio))
3247 gpio_free(ts->platform_data->irq_gpio);
3248
3249 /* housekeeping */
3250 kfree(ts);
3251
3252 cyttsp_alert("Leaving\n");
3253
3254 return 0;
3255}
3256
3257#ifdef CONFIG_HAS_EARLYSUSPEND
3258static void cyttsp_early_suspend(struct early_suspend *handler)
3259{
3260 struct cyttsp *ts;
3261
3262 ts = container_of(handler, struct cyttsp, early_suspend);
3263 cyttsp_suspend(&ts->client->dev);
3264}
3265
3266static void cyttsp_late_resume(struct early_suspend *handler)
3267{
3268 struct cyttsp *ts;
3269
3270 ts = container_of(handler, struct cyttsp, early_suspend);
3271 cyttsp_resume(&ts->client->dev);
3272}
3273#endif /* CONFIG_HAS_EARLYSUSPEND */
3274
3275static int cyttsp_init(void)
3276{
3277 int ret;
3278
3279 cyttsp_info("Cypress TrueTouch(R) Standard Product\n");
3280 cyttsp_info("I2C Touchscreen Driver (Built %s @ %s)\n", \
3281 __DATE__, __TIME__);
3282
3283 ret = i2c_add_driver(&cyttsp_driver);
3284
3285 return ret;
3286}
3287
3288static void cyttsp_exit(void)
3289{
3290 return i2c_del_driver(&cyttsp_driver);
3291}
3292
3293module_init(cyttsp_init);
3294module_exit(cyttsp_exit);
3295MODULE_FIRMWARE("cyttsp.fw");
3296