You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
3083 lines
85 KiB
3083 lines
85 KiB
2 years ago
|
/*
|
||
|
* wacom_i2c.c - Wacom Digitizer Controller (I2C bus)
|
||
|
*
|
||
|
*
|
||
|
* This program is free software; you can redistribute it and/or modify
|
||
|
* it under the terms of the GNU General Public License as published by
|
||
|
* the Free Software Foundation; either version 2 of the License, or
|
||
|
* (at your option) any later version.
|
||
|
*
|
||
|
* This program is distributed in the hope that it will be useful,
|
||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
* GNU General Public License for more details.
|
||
|
*
|
||
|
* You should have received a copy of the GNU General Public License
|
||
|
* along with this program; if not, write to the Free Software
|
||
|
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||
|
*/
|
||
|
|
||
|
#include "wacom_dev.h"
|
||
|
|
||
|
struct wacom_i2c *g_wac_i2c_w9019;
|
||
|
|
||
|
void w9019_forced_release(struct wacom_i2c *wac_i2c)
|
||
|
{
|
||
|
#if WACOM_PRODUCT_SHIP
|
||
|
if (wac_i2c->pen_pressed) {
|
||
|
input_info(true, &wac_i2c->client->dev, "%s : [R] dd:%d,%d mc:%d & [HO] dd:%d,%d\n",
|
||
|
__func__, wac_i2c->x - wac_i2c->p_x, wac_i2c->y - wac_i2c->p_y,
|
||
|
wac_i2c->mcount, wac_i2c->x - wac_i2c->hi_x, wac_i2c->y - wac_i2c->hi_y);
|
||
|
#if IS_ENABLED(CONFIG_INPUT_SEC_NOTIFIER)
|
||
|
sec_input_notify(&wac_i2c->nb, NOTIFIER_WACOM_PEN_HOVER_OUT, NULL);
|
||
|
#endif
|
||
|
} else if (wac_i2c->pen_prox) {
|
||
|
input_info(true, &wac_i2c->client->dev, "%s : [HO] dd:%d,%d mc:%d\n",
|
||
|
__func__, wac_i2c->x - wac_i2c->hi_x, wac_i2c->y - wac_i2c->hi_y, wac_i2c->mcount);
|
||
|
#if IS_ENABLED(CONFIG_INPUT_SEC_NOTIFIER)
|
||
|
sec_input_notify(&wac_i2c->nb, NOTIFIER_WACOM_PEN_HOVER_OUT, NULL);
|
||
|
#endif
|
||
|
} else {
|
||
|
input_info(true, &wac_i2c->client->dev, "%s : pen_prox(%d), pen_pressed(%d)\n",
|
||
|
__func__, wac_i2c->pen_prox, wac_i2c->pen_pressed);
|
||
|
}
|
||
|
#else
|
||
|
if (wac_i2c->pen_pressed) {
|
||
|
input_info(true, &wac_i2c->client->dev, "%s : [R] lx:%d ly:%d dd:%d,%d mc:%d & [HO] dd:%d,%d\n",
|
||
|
__func__, wac_i2c->x, wac_i2c->y,
|
||
|
wac_i2c->x - wac_i2c->p_x, wac_i2c->y - wac_i2c->p_y,
|
||
|
wac_i2c->mcount, wac_i2c->x - wac_i2c->hi_x, wac_i2c->y - wac_i2c->hi_y);
|
||
|
#if IS_ENABLED(CONFIG_INPUT_SEC_NOTIFIER)
|
||
|
sec_input_notify(&wac_i2c->nb, NOTIFIER_WACOM_PEN_HOVER_OUT, NULL);
|
||
|
#endif
|
||
|
} else if (wac_i2c->pen_prox) {
|
||
|
input_info(true, &wac_i2c->client->dev, "%s : [HO] lx:%d ly:%d dd:%d,%d mc:%d\n",
|
||
|
__func__, wac_i2c->x, wac_i2c->y,
|
||
|
wac_i2c->x - wac_i2c->hi_x, wac_i2c->y - wac_i2c->hi_y, wac_i2c->mcount);
|
||
|
#if IS_ENABLED(CONFIG_INPUT_SEC_NOTIFIER)
|
||
|
sec_input_notify(&wac_i2c->nb, NOTIFIER_WACOM_PEN_HOVER_OUT, NULL);
|
||
|
#endif
|
||
|
} else {
|
||
|
input_info(true, &wac_i2c->client->dev, "%s : pen_prox(%d), pen_pressed(%d)\n",
|
||
|
__func__, wac_i2c->pen_prox, wac_i2c->pen_pressed);
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
input_report_abs(wac_i2c->input_dev, ABS_X, 0);
|
||
|
input_report_abs(wac_i2c->input_dev, ABS_Y, 0);
|
||
|
input_report_abs(wac_i2c->input_dev, ABS_PRESSURE, 0);
|
||
|
input_report_abs(wac_i2c->input_dev, ABS_DISTANCE, 0);
|
||
|
|
||
|
input_report_key(wac_i2c->input_dev, BTN_STYLUS, 0);
|
||
|
input_report_key(wac_i2c->input_dev, BTN_TOUCH, 0);
|
||
|
input_report_key(wac_i2c->input_dev, BTN_TOOL_RUBBER, 0);
|
||
|
input_report_key(wac_i2c->input_dev, BTN_TOOL_PEN, 0);
|
||
|
|
||
|
input_sync(wac_i2c->input_dev);
|
||
|
|
||
|
wac_i2c->hi_x = 0;
|
||
|
wac_i2c->hi_y = 0;
|
||
|
wac_i2c->p_x = 0;
|
||
|
wac_i2c->p_y = 0;
|
||
|
|
||
|
wac_i2c->pen_prox = 0;
|
||
|
wac_i2c->pen_pressed = 0;
|
||
|
wac_i2c->side_pressed = 0;
|
||
|
wac_i2c->mcount = 0;
|
||
|
}
|
||
|
|
||
|
int w9019_i2c_send_sel(struct wacom_i2c *wac_i2c, const char *buf, int count, bool mode)
|
||
|
{
|
||
|
struct i2c_client *client = mode ? wac_i2c->client_boot : wac_i2c->client;
|
||
|
int retry = WACOM_I2C_RETRY;
|
||
|
int ret;
|
||
|
u8 *buff;
|
||
|
int i;
|
||
|
|
||
|
/* in LPM, waiting blsp block resume */
|
||
|
if (wac_i2c->pm_suspend) {
|
||
|
__pm_wakeup_event(wac_i2c->wacom_ws, jiffies_to_msecs(500));
|
||
|
ret = wait_for_completion_interruptible_timeout(&wac_i2c->resume_done, msecs_to_jiffies(500));
|
||
|
if (ret <= 0) {
|
||
|
input_err(true, &wac_i2c->client->dev,
|
||
|
"%s: LPM: pm resume is not handled [timeout]\n", __func__);
|
||
|
return -ENOMEM;
|
||
|
} else {
|
||
|
input_info(true, &wac_i2c->client->dev,
|
||
|
"%s: run LPM interrupt handler, %d\n", __func__, jiffies_to_msecs(ret));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
buff = kzalloc(count, GFP_KERNEL);
|
||
|
if (!buff)
|
||
|
return -ENOMEM;
|
||
|
|
||
|
mutex_lock(&wac_i2c->i2c_mutex);
|
||
|
|
||
|
memcpy(buff, buf, count);
|
||
|
|
||
|
reinit_completion(&wac_i2c->i2c_done);
|
||
|
do {
|
||
|
if (!wac_i2c->power_enable) {
|
||
|
input_err(true, &client->dev, "%s: Power status off\n", __func__);
|
||
|
ret = -EIO;
|
||
|
|
||
|
goto out;
|
||
|
}
|
||
|
|
||
|
ret = i2c_master_send(client, buff, count);
|
||
|
if (ret == count)
|
||
|
break;
|
||
|
|
||
|
if (retry < WACOM_I2C_RETRY) {
|
||
|
input_err(true, &client->dev, "%s: I2C retry(%d) mode(%d)\n",
|
||
|
__func__, WACOM_I2C_RETRY - retry, mode);
|
||
|
wac_i2c->i2c_fail_count++;
|
||
|
}
|
||
|
} while (--retry);
|
||
|
|
||
|
out:
|
||
|
complete_all(&wac_i2c->i2c_done);
|
||
|
mutex_unlock(&wac_i2c->i2c_mutex);
|
||
|
|
||
|
if (wac_i2c->debug_flag & WACOM_DEBUG_PRINT_I2C_WRITE_CMD) {
|
||
|
pr_info("sec_input : i2c_cmd: W: ");
|
||
|
for (i = 0; i < count; i++)
|
||
|
pr_cont("%02X ", buf[i]);
|
||
|
pr_cont("\n");
|
||
|
}
|
||
|
|
||
|
kfree(buff);
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
int w9019_i2c_recv_sel(struct wacom_i2c *wac_i2c, char *buf, int count, bool mode)
|
||
|
{
|
||
|
struct i2c_client *client = mode ? wac_i2c->client_boot : wac_i2c->client;
|
||
|
int retry = WACOM_I2C_RETRY;
|
||
|
int ret;
|
||
|
u8 *buff;
|
||
|
int i;
|
||
|
|
||
|
/* in LPM, waiting blsp block resume */
|
||
|
if (wac_i2c->pm_suspend) {
|
||
|
__pm_wakeup_event(wac_i2c->wacom_ws, jiffies_to_msecs(500));
|
||
|
ret = wait_for_completion_interruptible_timeout(&wac_i2c->resume_done, msecs_to_jiffies(500));
|
||
|
if (ret <= 0) {
|
||
|
input_err(true, &wac_i2c->client->dev,
|
||
|
"%s: LPM: pm resume is not handled [timeout]\n", __func__);
|
||
|
return -ENOMEM;
|
||
|
} else {
|
||
|
input_info(true, &wac_i2c->client->dev,
|
||
|
"%s: run LPM interrupt handler, %d\n", __func__, jiffies_to_msecs(ret));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
buff = kzalloc(count, GFP_KERNEL);
|
||
|
if (!buff)
|
||
|
return -ENOMEM;
|
||
|
|
||
|
mutex_lock(&wac_i2c->i2c_mutex);
|
||
|
|
||
|
reinit_completion(&wac_i2c->i2c_done);
|
||
|
do {
|
||
|
if (!wac_i2c->power_enable) {
|
||
|
input_err(true, &client->dev, "%s: Power status off\n", __func__);
|
||
|
ret = -EIO;
|
||
|
|
||
|
goto out;
|
||
|
}
|
||
|
|
||
|
ret = i2c_master_recv(client, buff, count);
|
||
|
if (ret == count)
|
||
|
break;
|
||
|
|
||
|
if (retry < WACOM_I2C_RETRY) {
|
||
|
input_err(true, &client->dev, "%s: I2C retry(%d) mode(%d)\n",
|
||
|
__func__, WACOM_I2C_RETRY - retry, mode);
|
||
|
wac_i2c->i2c_fail_count++;
|
||
|
}
|
||
|
} while (--retry);
|
||
|
|
||
|
memcpy(buf, buff, count);
|
||
|
|
||
|
if (wac_i2c->debug_flag & WACOM_DEBUG_PRINT_I2C_READ_CMD) {
|
||
|
pr_info("sec_input : i2c_cmd: R: ");
|
||
|
for (i = 0; i < count; i++)
|
||
|
pr_cont("%02X ", buf[i]);
|
||
|
pr_cont("\n");
|
||
|
}
|
||
|
|
||
|
out:
|
||
|
complete_all(&wac_i2c->i2c_done);
|
||
|
mutex_unlock(&wac_i2c->i2c_mutex);
|
||
|
|
||
|
kfree(buff);
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
int w9019_i2c_send_boot(struct wacom_i2c *wac_i2c, const char *buf, int count)
|
||
|
{
|
||
|
return w9019_i2c_send_sel(wac_i2c, buf, count, WACOM_I2C_MODE_BOOT);
|
||
|
}
|
||
|
|
||
|
int w9019_i2c_send(struct wacom_i2c *wac_i2c, const char *buf, int count)
|
||
|
{
|
||
|
return w9019_i2c_send_sel(wac_i2c, buf, count, WACOM_I2C_MODE_NORMAL);
|
||
|
}
|
||
|
|
||
|
int w9019_i2c_recv_boot(struct wacom_i2c *wac_i2c, char *buf, int count)
|
||
|
{
|
||
|
return w9019_i2c_recv_sel(wac_i2c, buf, count, WACOM_I2C_MODE_BOOT);
|
||
|
}
|
||
|
|
||
|
int w9019_i2c_recv(struct wacom_i2c *wac_i2c, char *buf, int count)
|
||
|
{
|
||
|
return w9019_i2c_recv_sel(wac_i2c, buf, count, WACOM_I2C_MODE_NORMAL);
|
||
|
}
|
||
|
|
||
|
int w9019_start_stop_cmd(struct wacom_i2c *wac_i2c, int mode)
|
||
|
{
|
||
|
int retry;
|
||
|
int ret = 0;
|
||
|
char buff;
|
||
|
|
||
|
input_info(true, &wac_i2c->client->dev, "%s: mode (%d)\n", __func__, mode);
|
||
|
|
||
|
reset_start:
|
||
|
if (mode == WACOM_STOP_CMD) {
|
||
|
buff = COM_SAMPLERATE_STOP;
|
||
|
} else if (mode == WACOM_START_CMD) {
|
||
|
buff = COM_SAMPLERATE_START;
|
||
|
} else if (mode == WACOM_STOP_AND_START_CMD) {
|
||
|
buff = COM_SAMPLERATE_STOP;
|
||
|
wac_i2c->samplerate_state = WACOM_STOP_CMD;
|
||
|
} else {
|
||
|
input_info(true, &wac_i2c->client->dev, "%s: abnormal mode (%d)\n", __func__, mode);
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
retry = WACOM_CMD_RETRY;
|
||
|
do {
|
||
|
ret = w9019_i2c_send(wac_i2c, &buff, 1);
|
||
|
msleep(50);
|
||
|
if (ret < 0)
|
||
|
input_err(true, &wac_i2c->client->dev,
|
||
|
"%s: failed to send 0x%02X(%d)\n", __func__, buff, retry);
|
||
|
else
|
||
|
break;
|
||
|
|
||
|
} while (--retry);
|
||
|
|
||
|
if (mode == WACOM_STOP_AND_START_CMD) {
|
||
|
mode = WACOM_START_CMD;
|
||
|
goto reset_start;
|
||
|
}
|
||
|
|
||
|
if (ret == 1)
|
||
|
wac_i2c->samplerate_state = mode;
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
int w9019_checksum(struct wacom_i2c *wac_i2c)
|
||
|
{
|
||
|
struct i2c_client *client = wac_i2c->client;
|
||
|
int ret = 0, retry = 10;
|
||
|
int i = 0;
|
||
|
u8 buf[5] = { 0, };
|
||
|
|
||
|
while (retry--) {
|
||
|
buf[0] = COM_CHECKSUM;
|
||
|
ret = w9019_i2c_send(wac_i2c, &buf[0], 1);
|
||
|
if (ret < 0) {
|
||
|
input_err(true, &client->dev, "i2c fail, retry, %d\n", __LINE__);
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
msleep(200);
|
||
|
|
||
|
ret = w9019_i2c_recv(wac_i2c, buf, 5);
|
||
|
if (ret < 0) {
|
||
|
input_err(true, &client->dev, "i2c fail, retry, %d\n", __LINE__);
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
if (buf[0] == 0x1F)
|
||
|
break;
|
||
|
|
||
|
input_info(true, &client->dev, "buf[0]: 0x%x, checksum retry\n", buf[0]);
|
||
|
}
|
||
|
|
||
|
if (ret >= 0) {
|
||
|
input_info(true, &client->dev, "received checksum %x, %x, %x, %x, %x\n",
|
||
|
buf[0], buf[1], buf[2], buf[3], buf[4]);
|
||
|
}
|
||
|
|
||
|
for (i = 0; i < 5; ++i) {
|
||
|
if (buf[i] != wac_i2c->fw_chksum[i]) {
|
||
|
input_info(true, &client->dev, "checksum fail %dth %x %x\n",
|
||
|
i, buf[i], wac_i2c->fw_chksum[i]);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
wac_i2c->checksum_result = (i == 5);
|
||
|
|
||
|
return wac_i2c->checksum_result;
|
||
|
}
|
||
|
|
||
|
int w9019_i2c_query(struct wacom_i2c *wac_i2c)
|
||
|
{
|
||
|
struct wacom_g5_platform_data *pdata = wac_i2c->pdata;
|
||
|
struct i2c_client *client = wac_i2c->client;
|
||
|
u8 data[COM_QUERY_BUFFER] = { 0, };
|
||
|
u8 *query = data + COM_QUERY_POS;
|
||
|
int read_size = COM_QUERY_BUFFER;
|
||
|
int ret;
|
||
|
int i;
|
||
|
|
||
|
for (i = 0; i < RETRY_COUNT; i++) {
|
||
|
ret = w9019_i2c_recv(wac_i2c, data, read_size);
|
||
|
if (ret < 0) {
|
||
|
input_err(true, &client->dev, "%s: failed to recv\n", __func__);
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
input_info(true, &client->dev, "%s: %dth ret of wacom query=%d\n", __func__, i, ret);
|
||
|
|
||
|
if (read_size != ret) {
|
||
|
input_err(true, &client->dev, "%s: read size error %d of %d\n", __func__, ret, read_size);
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
input_info(true, &client->dev,
|
||
|
"(retry:%d) %X, %X, %X, %X, %X, %X, %X, %X, %X, %X, %X, %X, %X, %X, %X\n",
|
||
|
i, query[0], query[1], query[2], query[3], query[4], query[5],
|
||
|
query[6], query[7], query[8], query[9], query[10], query[11],
|
||
|
query[12], query[13], query[14]);
|
||
|
|
||
|
if (query[EPEN_REG_HEADER] == 0x0F) {
|
||
|
wac_i2c->fw_ver_ic = (query[EPEN_REG_FWVER1] << 8) | query[EPEN_REG_FWVER2];
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
input_info(true, &client->dev,
|
||
|
"%X, %X, %X, %X, %X, %X, %X, %X, %X, %X, %X, %X, %X, %X, %X\n",
|
||
|
query[0], query[1], query[2], query[3], query[4], query[5],
|
||
|
query[6], query[7], query[8], query[9], query[10], query[11],
|
||
|
query[12], query[13], query[14]);
|
||
|
|
||
|
if (ret < 0) {
|
||
|
input_err(true, &client->dev, "%s: failed to read query\n", __func__);
|
||
|
wac_i2c->fw_ver_ic = 0;
|
||
|
wac_i2c->query_status = false;
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
wac_i2c->query_status = true;
|
||
|
|
||
|
if (pdata->use_dt_coord == false) {
|
||
|
pdata->max_x = ((u16)query[EPEN_REG_X1] << 8) + query[EPEN_REG_X2];
|
||
|
pdata->max_y = ((u16)query[EPEN_REG_Y1] << 8) + query[EPEN_REG_Y2];
|
||
|
}
|
||
|
pdata->max_pressure = ((u16)query[EPEN_REG_PRESSURE1] << 8) + query[EPEN_REG_PRESSURE2];
|
||
|
pdata->max_x_tilt = query[EPEN_REG_TILT_X];
|
||
|
pdata->max_y_tilt = query[EPEN_REG_TILT_Y];
|
||
|
pdata->max_height = query[EPEN_REG_HEIGHT];
|
||
|
pdata->ic_type = query[EPEN_REG_MPUVER];
|
||
|
pdata->bl_ver = query[EPEN_REG_BLVER];
|
||
|
|
||
|
input_info(true, &client->dev, "use_dt_coord: %d, max_x: %d max_y: %d, max_pressure: %d\n",
|
||
|
pdata->use_dt_coord, pdata->max_x, pdata->max_y, pdata->max_pressure);
|
||
|
input_info(true, &client->dev, "fw_ver_ic=0x%X\n", wac_i2c->fw_ver_ic);
|
||
|
|
||
|
input_info(true, &client->dev, "mpu: 0x%X bl: 0x%X, tiltX: %d, tiltY: %d, maxH: %d\n",
|
||
|
pdata->ic_type, pdata->bl_ver, pdata->max_x_tilt, pdata->max_y_tilt, pdata->max_height);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int w9019_i2c_set_sense_mode(struct wacom_i2c *wac_i2c)
|
||
|
{
|
||
|
struct i2c_client *client = wac_i2c->client;
|
||
|
int retval;
|
||
|
char data[4] = { 0, 0, 0, 0 };
|
||
|
|
||
|
input_info(true, &wac_i2c->client->dev, "%s cmd mod(%d)\n", __func__, wac_i2c->wcharging_mode);
|
||
|
|
||
|
if (wac_i2c->wcharging_mode == 1)
|
||
|
data[0] = COM_LOW_SENSE_MODE;
|
||
|
else if (wac_i2c->wcharging_mode == 3)
|
||
|
data[0] = COM_LOW_SENSE_MODE2;
|
||
|
else {
|
||
|
/* it must be 0 */
|
||
|
data[0] = COM_NORMAL_SENSE_MODE;
|
||
|
}
|
||
|
|
||
|
retval = w9019_i2c_send(wac_i2c, &data[0], 1);
|
||
|
if (retval < 0) {
|
||
|
input_err(true, &client->dev, "%s: failed to send wacom i2c mode, %d\n", __func__, retval);
|
||
|
return retval;
|
||
|
}
|
||
|
|
||
|
msleep(60);
|
||
|
|
||
|
retval = w9019_start_stop_cmd(wac_i2c, WACOM_STOP_CMD);
|
||
|
if (retval < 0) {
|
||
|
input_err(true, &client->dev, "%s: failed to set stop cmd, %d\n",
|
||
|
__func__, retval);
|
||
|
return retval;
|
||
|
}
|
||
|
|
||
|
#if 0
|
||
|
/* temp block not to receive gabage irq by cmd */
|
||
|
data[1] = COM_REQUEST_SENSE_MODE;
|
||
|
retval = w9019_i2c_send(wac_i2c, &data[1], 1);
|
||
|
if (retval < 0) {
|
||
|
input_err(true, &client->dev,
|
||
|
"%s: failed to read wacom i2c send2, %d\n", __func__,
|
||
|
retval);
|
||
|
return retval;
|
||
|
}
|
||
|
|
||
|
msleep(60);
|
||
|
|
||
|
retval = w9019_i2c_recv(wac_i2c, &data[2], 2);
|
||
|
if (retval != 2) {
|
||
|
input_err(true, &client->dev,
|
||
|
"%s: failed to read wacom i2c recv, %d\n", __func__,
|
||
|
retval);
|
||
|
return retval;
|
||
|
}
|
||
|
|
||
|
input_info(true, &client->dev, "%s: mode:%X, %X\n", __func__, data[2],
|
||
|
data[3]);
|
||
|
|
||
|
data[0] = COM_SAMPLERATE_STOP;
|
||
|
retval = w9019_i2c_send(wac_i2c, &data[0], 1);
|
||
|
if (retval < 0) {
|
||
|
input_err(true, &client->dev,
|
||
|
"%s: failed to read wacom i2c send3, %d\n", __func__,
|
||
|
retval);
|
||
|
return retval;
|
||
|
}
|
||
|
|
||
|
msleep(60);
|
||
|
#endif
|
||
|
retval = w9019_start_stop_cmd(wac_i2c, WACOM_START_CMD);
|
||
|
if (retval < 0) {
|
||
|
input_err(true, &client->dev, "%s: failed to set start cmd, %d\n",
|
||
|
__func__, retval);
|
||
|
return retval;
|
||
|
}
|
||
|
|
||
|
return retval; //data[3];
|
||
|
}
|
||
|
|
||
|
void w9019_select_survey_mode(struct wacom_i2c *wac_i2c, bool enable)
|
||
|
{
|
||
|
struct i2c_client *client = wac_i2c->client;
|
||
|
|
||
|
mutex_lock(&wac_i2c->mode_lock);
|
||
|
|
||
|
if (enable) {
|
||
|
if (wac_i2c->epen_blocked ||
|
||
|
(wac_i2c->battery_saving_mode && !(wac_i2c->function_result & EPEN_EVENT_PEN_OUT))) {
|
||
|
if (wac_i2c->pdata->support_cover_detection) {
|
||
|
input_info(true, &client->dev, "%s: %s cover detection only\n",
|
||
|
__func__, wac_i2c->epen_blocked ? "epen blocked" : "ps on & pen in");
|
||
|
w9019_i2c_set_survey_mode(wac_i2c, EPEN_SURVEY_MODE_COVER_DETECTION_ONLY);
|
||
|
} else if (wac_i2c->pdata->use_garage) {
|
||
|
input_info(true, &client->dev, "%s: %s & garage on. garage only mode\n",
|
||
|
__func__, wac_i2c->epen_blocked ? "epen blocked" : "ps on & pen in");
|
||
|
|
||
|
w9019_i2c_set_survey_mode(wac_i2c, EPEN_SURVEY_MODE_GARAGE_ONLY);
|
||
|
} else {
|
||
|
input_info(true, &client->dev, "%s: %s power off\n", __func__,
|
||
|
wac_i2c->epen_blocked ? "epen blocked" : "ps on & pen in");
|
||
|
|
||
|
w9019_enable_irq(wac_i2c, false);
|
||
|
w9019_power(wac_i2c, false);
|
||
|
|
||
|
wac_i2c->survey_mode = EPEN_SURVEY_MODE_NONE;
|
||
|
wac_i2c->function_result &= ~EPEN_EVENT_SURVEY;
|
||
|
}
|
||
|
} else if (wac_i2c->survey_mode) {
|
||
|
input_info(true, &client->dev, "%s: exit aop mode\n", __func__);
|
||
|
|
||
|
w9019_i2c_set_survey_mode(wac_i2c, EPEN_SURVEY_MODE_NONE);
|
||
|
} else {
|
||
|
input_info(true, &client->dev, "%s: power on\n", __func__);
|
||
|
|
||
|
w9019_power(wac_i2c, true);
|
||
|
msleep(100);
|
||
|
|
||
|
w9019_enable_irq(wac_i2c, true);
|
||
|
}
|
||
|
} else {
|
||
|
if (wac_i2c->epen_blocked || (wac_i2c->battery_saving_mode &&
|
||
|
!(wac_i2c->function_result & EPEN_EVENT_PEN_OUT))) {
|
||
|
if (wac_i2c->pdata->support_cover_detection) {
|
||
|
input_info(true, &client->dev, "%s: %s cover detection only\n",
|
||
|
__func__, wac_i2c->epen_blocked ? "epen blocked" : "ps on & pen in");
|
||
|
w9019_i2c_set_survey_mode(wac_i2c, EPEN_SURVEY_MODE_COVER_DETECTION_ONLY);
|
||
|
} else if (wac_i2c->pdata->use_garage) {
|
||
|
input_info(true, &client->dev, "%s: %s & garage on. garage only mode\n",
|
||
|
__func__, wac_i2c->epen_blocked ? "epen blocked" : "ps on & pen in");
|
||
|
|
||
|
w9019_i2c_set_survey_mode(wac_i2c, EPEN_SURVEY_MODE_GARAGE_ONLY);
|
||
|
} else {
|
||
|
input_info(true, &client->dev, "%s: %s power off\n",
|
||
|
__func__, wac_i2c->epen_blocked ? "epen blocked" : "ps on & pen in");
|
||
|
|
||
|
w9019_enable_irq(wac_i2c, false);
|
||
|
w9019_power(wac_i2c, false);
|
||
|
|
||
|
wac_i2c->survey_mode = EPEN_SURVEY_MODE_NONE;
|
||
|
wac_i2c->function_result &= ~EPEN_EVENT_SURVEY;
|
||
|
}
|
||
|
} else if (!(wac_i2c->function_set & EPEN_SETMODE_AOP)) {
|
||
|
if (wac_i2c->pdata->support_cover_detection) {
|
||
|
input_info(true, &client->dev, "%s: AOP off. cover detection only\n",
|
||
|
__func__);
|
||
|
w9019_i2c_set_survey_mode(wac_i2c, EPEN_SURVEY_MODE_COVER_DETECTION_ONLY);
|
||
|
} else if (wac_i2c->pdata->use_garage) {
|
||
|
input_info(true, &client->dev, "%s: aop off & garage on. garage only mode\n", __func__);
|
||
|
|
||
|
w9019_i2c_set_survey_mode(wac_i2c, EPEN_SURVEY_MODE_GARAGE_ONLY);
|
||
|
} else {
|
||
|
input_info(true, &client->dev, "%s: aop off & garage off. power off\n", __func__);
|
||
|
|
||
|
w9019_enable_irq(wac_i2c, false);
|
||
|
w9019_power(wac_i2c, false);
|
||
|
|
||
|
wac_i2c->survey_mode = EPEN_SURVEY_MODE_NONE;
|
||
|
wac_i2c->function_result &= ~EPEN_EVENT_SURVEY;
|
||
|
}
|
||
|
} else {
|
||
|
/* aop on => (aod : screen off memo = 1:1 or 1:0 or 0:1)
|
||
|
* double tab & hover + button event will be occurred,
|
||
|
* but some of them will be skipped at reporting by mode
|
||
|
*/
|
||
|
input_info(true, &client->dev, "%s: aop on. aop mode\n", __func__);
|
||
|
|
||
|
if (!wac_i2c->power_enable) {
|
||
|
input_info(true, &client->dev, "%s: power on\n", __func__);
|
||
|
|
||
|
w9019_power(wac_i2c, true);
|
||
|
msleep(100);
|
||
|
|
||
|
w9019_enable_irq(wac_i2c, true);
|
||
|
}
|
||
|
|
||
|
w9019_i2c_set_survey_mode(wac_i2c, EPEN_SURVEY_MODE_GARAGE_AOP);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (wac_i2c->power_enable) {
|
||
|
input_info(true, &client->dev, "%s: screen %s, survey mode:%d, result:%d\n",
|
||
|
__func__, enable ? "on" : "off", wac_i2c->survey_mode,
|
||
|
wac_i2c->function_result & EPEN_EVENT_SURVEY);
|
||
|
|
||
|
if ((wac_i2c->function_result & EPEN_EVENT_SURVEY) != wac_i2c->survey_mode) {
|
||
|
input_err(true, &client->dev, "%s: survey mode cmd failed\n", __func__);
|
||
|
|
||
|
w9019_i2c_set_survey_mode(wac_i2c, wac_i2c->survey_mode);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
mutex_unlock(&wac_i2c->mode_lock);
|
||
|
}
|
||
|
|
||
|
int w9019_i2c_set_survey_mode(struct wacom_i2c *wac_i2c, int mode)
|
||
|
{
|
||
|
struct i2c_client *client = wac_i2c->client;
|
||
|
int retval;
|
||
|
char data[4] = { 0, 0, 0, 0 };
|
||
|
|
||
|
switch (mode) {
|
||
|
case EPEN_SURVEY_MODE_NONE:
|
||
|
data[0] = COM_SURVEY_EXIT;
|
||
|
break;
|
||
|
case EPEN_SURVEY_MODE_GARAGE_ONLY:
|
||
|
if (!wac_i2c->pdata->use_garage) {
|
||
|
input_err(true, &client->dev, "%s: garage mode is not supported\n", __func__);
|
||
|
return -EPERM;
|
||
|
}
|
||
|
data[0] = COM_SURVEY_GARAGE_ONLY;
|
||
|
break;
|
||
|
case EPEN_SURVEY_MODE_GARAGE_AOP:
|
||
|
if ((wac_i2c->function_set & EPEN_SETMODE_AOP_OPTION_AOD_LCD_ON) == EPEN_SETMODE_AOP_OPTION_AOD_LCD_ON)
|
||
|
data[0] = COM_SURVEY_SYNC_SCAN;
|
||
|
else
|
||
|
data[0] = COM_SURVEY_ASYNC_SCAN;
|
||
|
break;
|
||
|
case EPEN_SURVEY_MODE_COVER_DETECTION_ONLY:
|
||
|
data[0] = COM_SURVEY_GARAGE_ONLY;
|
||
|
break;
|
||
|
default:
|
||
|
input_err(true, &client->dev, "%s: wrong param %d\n", __func__, mode);
|
||
|
return -EINVAL;
|
||
|
}
|
||
|
|
||
|
wac_i2c->survey_mode = mode;
|
||
|
input_info(true, &client->dev, "%s: ps %s & mode : %d cmd(0x%2X)\n", __func__,
|
||
|
wac_i2c->battery_saving_mode ? "on" : "off", mode, data[0]);
|
||
|
|
||
|
retval = w9019_i2c_send(wac_i2c, &data[0], 1);
|
||
|
if (retval < 0) {
|
||
|
input_err(true, &client->dev, "%s: failed to send data(%02x %d)\n", __func__, data[0], retval);
|
||
|
wac_i2c->reset_flag = true;
|
||
|
|
||
|
return retval;
|
||
|
}
|
||
|
|
||
|
wac_i2c->check_survey_mode = mode;
|
||
|
|
||
|
if (mode)
|
||
|
msleep(300);
|
||
|
|
||
|
wac_i2c->reset_flag = false;
|
||
|
wac_i2c->function_result &= ~EPEN_EVENT_SURVEY;
|
||
|
wac_i2c->function_result |= mode;
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
void w9019_forced_release_fullscan(struct wacom_i2c *wac_i2c)
|
||
|
{
|
||
|
input_info(true, &wac_i2c->client->dev, "%s: full scan OUT\n", __func__);
|
||
|
|
||
|
#if IS_ENABLED(CONFIG_INPUT_SEC_NOTIFIER)
|
||
|
wac_i2c->tsp_scan_mode = sec_input_notify(&wac_i2c->nb, NOTIFIER_TSP_BLOCKING_RELEASE, NULL);
|
||
|
wac_i2c->is_tsp_block = false;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
int w9019_power(struct wacom_i2c *wac_i2c, bool on)
|
||
|
{
|
||
|
struct i2c_client *client = wac_i2c->client;
|
||
|
int ret = 0;
|
||
|
static bool boot_on = true;
|
||
|
|
||
|
if (wac_i2c->power_enable == on) {
|
||
|
input_info(true, &client->dev, "pwr already %s\n", on ? "enabled" : "disabled");
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
if (on) {
|
||
|
if (!regulator_is_enabled(wac_i2c->pdata->avdd) || boot_on) {
|
||
|
ret = regulator_enable(wac_i2c->pdata->avdd);
|
||
|
if (ret) {
|
||
|
input_err(true, &client->dev, "%s: Failed to enable vdd: %d\n", __func__, ret);
|
||
|
regulator_disable(wac_i2c->pdata->avdd);
|
||
|
goto out;
|
||
|
}
|
||
|
} else {
|
||
|
input_err(true, &client->dev, "%s: avdd is already enabled\n", __func__);
|
||
|
}
|
||
|
} else {
|
||
|
if (regulator_is_enabled(wac_i2c->pdata->avdd)) {
|
||
|
ret = regulator_disable(wac_i2c->pdata->avdd);
|
||
|
if (ret) {
|
||
|
input_err(true, &client->dev, "%s: failed to disable avdd: %d\n", __func__, ret);
|
||
|
goto out;
|
||
|
}
|
||
|
} else {
|
||
|
input_err(true, &client->dev, "%s: avdd is already disabled\n", __func__);
|
||
|
}
|
||
|
}
|
||
|
wac_i2c->power_enable = on;
|
||
|
|
||
|
out:
|
||
|
input_info(true, &client->dev, "%s: %s: avdd:%s\n",
|
||
|
__func__, on ? "on" : "off", regulator_is_enabled(wac_i2c->pdata->avdd) ? "on" : "off");
|
||
|
boot_on = false;
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
void w9019_reset_hw(struct wacom_i2c *wac_i2c)
|
||
|
{
|
||
|
w9019_power(wac_i2c, false);
|
||
|
/* recommended delay in spec */
|
||
|
msleep(100);
|
||
|
w9019_power(wac_i2c, true);
|
||
|
|
||
|
msleep(200);
|
||
|
}
|
||
|
|
||
|
void w9019_compulsory_flash_mode(struct wacom_i2c *wac_i2c, bool enable)
|
||
|
{
|
||
|
gpio_direction_output(wac_i2c->pdata->fwe_gpio, enable);
|
||
|
input_info(true, &wac_i2c->client->dev, "%s : enable(%d) fwe(%d)\n",
|
||
|
__func__, enable, gpio_get_value(wac_i2c->pdata->fwe_gpio));
|
||
|
}
|
||
|
|
||
|
void w9019_enable_irq(struct wacom_i2c *wac_i2c, bool enable)
|
||
|
{
|
||
|
struct irq_desc *desc = irq_to_desc(wac_i2c->irq);
|
||
|
|
||
|
if (desc == NULL) {
|
||
|
input_err(true, &wac_i2c->client->dev, "%s : irq desc is NULL\n", __func__);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
mutex_lock(&wac_i2c->irq_lock);
|
||
|
|
||
|
if (enable) {
|
||
|
while (desc->depth > 0)
|
||
|
enable_irq(wac_i2c->irq);
|
||
|
} else {
|
||
|
disable_irq(wac_i2c->irq);
|
||
|
}
|
||
|
|
||
|
mutex_unlock(&wac_i2c->irq_lock);
|
||
|
}
|
||
|
|
||
|
static void w9019_enable_irq_wake(struct wacom_i2c *wac_i2c, bool enable)
|
||
|
{
|
||
|
|
||
|
if (enable) {
|
||
|
enable_irq_wake(wac_i2c->irq);
|
||
|
} else {
|
||
|
disable_irq_wake(wac_i2c->irq);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void w9019_wakeup_sequence(struct wacom_i2c *wac_i2c)
|
||
|
{
|
||
|
struct i2c_client *client = wac_i2c->client;
|
||
|
int retry = 1;
|
||
|
int ret;
|
||
|
|
||
|
mutex_lock(&wac_i2c->lock);
|
||
|
#if WACOM_SEC_FACTORY
|
||
|
if (wac_i2c->fac_garage_mode)
|
||
|
input_info(true, &client->dev, "%s: garage mode\n", __func__);
|
||
|
#endif
|
||
|
|
||
|
if (wac_i2c->wacom_fw_ws->active) {
|
||
|
input_info(true, &client->dev, "fw wake lock active. pass %s\n", __func__);
|
||
|
goto out_power_on;
|
||
|
}
|
||
|
|
||
|
if (wac_i2c->screen_on) {
|
||
|
input_info(true, &client->dev, "already enabled. pass %s\n", __func__);
|
||
|
goto out_power_on;
|
||
|
}
|
||
|
|
||
|
reset:
|
||
|
if (wac_i2c->reset_flag) {
|
||
|
input_info(true, &client->dev, "%s: IC reset start\n", __func__);
|
||
|
|
||
|
wac_i2c->abnormal_reset_count++;
|
||
|
wac_i2c->reset_flag = false;
|
||
|
wac_i2c->survey_mode = EPEN_SURVEY_MODE_NONE;
|
||
|
wac_i2c->function_result &= ~EPEN_EVENT_SURVEY;
|
||
|
|
||
|
w9019_enable_irq(wac_i2c, false);
|
||
|
|
||
|
w9019_reset_hw(wac_i2c);
|
||
|
|
||
|
input_info(true, &client->dev, "%s: IC reset end\n", __func__);
|
||
|
|
||
|
if (wac_i2c->pdata->support_cover_noti && wac_i2c->cover) {
|
||
|
w9019_swap_compensation(wac_i2c, wac_i2c->cover);
|
||
|
}
|
||
|
|
||
|
w9019_enable_irq(wac_i2c, true);
|
||
|
}
|
||
|
|
||
|
w9019_select_survey_mode(wac_i2c, true);
|
||
|
|
||
|
if (wac_i2c->reset_flag && retry--)
|
||
|
goto reset;
|
||
|
|
||
|
if (wac_i2c->wcharging_mode)
|
||
|
w9019_i2c_set_sense_mode(wac_i2c);
|
||
|
|
||
|
#if IS_ENABLED(CONFIG_INPUT_SEC_NOTIFIER)
|
||
|
if (wac_i2c->tsp_scan_mode < 0) {
|
||
|
wac_i2c->tsp_scan_mode = sec_input_notify(&wac_i2c->nb, NOTIFIER_TSP_BLOCKING_RELEASE, NULL);
|
||
|
wac_i2c->is_tsp_block = false;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
if (!gpio_get_value(wac_i2c->pdata->irq_gpio)) {
|
||
|
u8 data[COM_COORD_NUM + 1] = { 0, };
|
||
|
|
||
|
input_info(true, &client->dev, "%s: irq was enabled\n", __func__);
|
||
|
|
||
|
ret = w9019_i2c_recv(wac_i2c, data, COM_COORD_NUM + 1);
|
||
|
if (ret < 0) {
|
||
|
input_err(true, &client->dev, "%s: failed to receive\n", __func__);
|
||
|
}
|
||
|
|
||
|
input_info(true, &client->dev,
|
||
|
"%x %x %x %x %x, %x %x %x %x %x, %x %x %x\n",
|
||
|
data[0], data[1], data[2], data[3], data[4], data[5],
|
||
|
data[6], data[7], data[8], data[9], data[10],
|
||
|
data[11], data[12]);
|
||
|
|
||
|
/* protection codes for cover status - need to check*/
|
||
|
// if ((data[0] & 0x0F) == NOTI_PACKET && data[1] == COVER_DETECT_PACKET)
|
||
|
// wacom_i2c_cover_handler(wac_i2c, data);
|
||
|
}
|
||
|
|
||
|
if (!wac_i2c->samplerate_state) {
|
||
|
char cmd = COM_SAMPLERATE_START;
|
||
|
|
||
|
input_info(true, &client->dev, "%s: samplerate state is %d, need to recovery\n",
|
||
|
__func__, wac_i2c->samplerate_state);
|
||
|
|
||
|
ret = w9019_i2c_send(wac_i2c, &cmd, 1);
|
||
|
if (ret < 0)
|
||
|
input_err(true, &client->dev, "%s: failed to sned start cmd %d\n", __func__, ret);
|
||
|
else
|
||
|
wac_i2c->samplerate_state = true;
|
||
|
}
|
||
|
|
||
|
input_info(true, &client->dev,
|
||
|
"%s: i(%d) f_set(0x%x) f_ret(0x%x) samplerate(%d)\n",
|
||
|
__func__, gpio_get_value(wac_i2c->pdata->irq_gpio), wac_i2c->function_set,
|
||
|
wac_i2c->function_result, wac_i2c->samplerate_state);
|
||
|
|
||
|
cancel_delayed_work(&wac_i2c->work_print_info);
|
||
|
wac_i2c->print_info_cnt_open = 0;
|
||
|
wac_i2c->tsp_block_cnt = 0;
|
||
|
schedule_work(&wac_i2c->work_print_info.work);
|
||
|
|
||
|
if (device_may_wakeup(&client->dev))
|
||
|
w9019_enable_irq_wake(wac_i2c, false);
|
||
|
|
||
|
wac_i2c->screen_on = true;
|
||
|
|
||
|
out_power_on:
|
||
|
mutex_unlock(&wac_i2c->lock);
|
||
|
|
||
|
input_info(true, &client->dev, "%s: end\n", __func__);
|
||
|
}
|
||
|
|
||
|
void w9019_sleep_sequence(struct wacom_i2c *wac_i2c)
|
||
|
{
|
||
|
struct i2c_client *client = wac_i2c->client;
|
||
|
int retry = 1;
|
||
|
|
||
|
mutex_lock(&wac_i2c->lock);
|
||
|
#if WACOM_SEC_FACTORY
|
||
|
if (wac_i2c->fac_garage_mode)
|
||
|
input_info(true, &client->dev, "%s: garage mode\n", __func__);
|
||
|
|
||
|
#endif
|
||
|
|
||
|
if (wac_i2c->wacom_fw_ws->active) {
|
||
|
input_info(true, &client->dev, "fw wake lock active. pass %s\n", __func__);
|
||
|
goto out_power_off;
|
||
|
}
|
||
|
|
||
|
if (!wac_i2c->screen_on) {
|
||
|
input_info(true, &client->dev, "already disabled. pass %s\n", __func__);
|
||
|
goto out_power_off;
|
||
|
}
|
||
|
cancel_delayed_work_sync(&wac_i2c->work_print_info);
|
||
|
w9019_print_info(wac_i2c);
|
||
|
|
||
|
reset:
|
||
|
if (wac_i2c->reset_flag) {
|
||
|
input_info(true, &client->dev, "%s: IC reset start\n", __func__);
|
||
|
|
||
|
wac_i2c->abnormal_reset_count++;
|
||
|
wac_i2c->reset_flag = false;
|
||
|
wac_i2c->survey_mode = EPEN_SURVEY_MODE_NONE;
|
||
|
wac_i2c->function_result &= ~EPEN_EVENT_SURVEY;
|
||
|
|
||
|
w9019_enable_irq(wac_i2c, false);
|
||
|
|
||
|
w9019_reset_hw(wac_i2c);
|
||
|
|
||
|
input_info(true, &client->dev, "%s : IC reset end\n", __func__);
|
||
|
|
||
|
w9019_enable_irq(wac_i2c, true);
|
||
|
}
|
||
|
|
||
|
w9019_select_survey_mode(wac_i2c, false);
|
||
|
|
||
|
/* release pen, if it is pressed */
|
||
|
if (wac_i2c->pen_pressed || wac_i2c->side_pressed || wac_i2c->pen_prox)
|
||
|
w9019_forced_release(wac_i2c);
|
||
|
|
||
|
w9019_forced_release_fullscan(wac_i2c);
|
||
|
|
||
|
if (wac_i2c->reset_flag && retry--)
|
||
|
goto reset;
|
||
|
|
||
|
if (device_may_wakeup(&client->dev))
|
||
|
w9019_enable_irq_wake(wac_i2c, true);
|
||
|
|
||
|
wac_i2c->screen_on = false;
|
||
|
|
||
|
out_power_off:
|
||
|
mutex_unlock(&wac_i2c->lock);
|
||
|
|
||
|
input_info(true, &client->dev, "%s end\n", __func__);
|
||
|
}
|
||
|
|
||
|
#if 0
|
||
|
static void wac_i2c_table_swap_reply(struct wacom_i2c *wac_i2c, char *data)
|
||
|
{
|
||
|
u8 table_id;
|
||
|
|
||
|
table_id = data[3];
|
||
|
|
||
|
if (table_id == 1) {
|
||
|
if (wac_i2c->pdata->table_swap == TABLE_SWAP_DEX_STATION)
|
||
|
wac_i2c->dp_connect_state = true;
|
||
|
|
||
|
if (wac_i2c->pdata->table_swap == TABLE_SWAP_KBD_COVER)
|
||
|
wac_i2c->kbd_cur_conn_state = true;
|
||
|
} else if (!table_id) {
|
||
|
if (wac_i2c->pdata->table_swap == TABLE_SWAP_DEX_STATION)
|
||
|
wac_i2c->dp_connect_state = false;
|
||
|
|
||
|
if (wac_i2c->pdata->table_swap == TABLE_SWAP_KBD_COVER)
|
||
|
wac_i2c->kbd_cur_conn_state = false;
|
||
|
}
|
||
|
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
#define EPEN_LOCATION_DETECT_SIZE 6
|
||
|
void w9019_epen_location_detect(struct wacom_i2c *wac_i2c, char *loc, int x, int y)
|
||
|
{
|
||
|
int i;
|
||
|
int max_x = wac_i2c->pdata->max_x;
|
||
|
int max_y = wac_i2c->pdata->max_y;
|
||
|
|
||
|
if (wac_i2c->pdata->xy_switch) {
|
||
|
max_x = wac_i2c->pdata->max_y;
|
||
|
max_y = wac_i2c->pdata->max_x;
|
||
|
}
|
||
|
|
||
|
if (wac_i2c->pdata->area_indicator == 0){
|
||
|
/* approximately value */
|
||
|
wac_i2c->pdata->area_indicator = max_y * 4 / 100;
|
||
|
wac_i2c->pdata->area_navigation = max_y * 6 / 100;
|
||
|
wac_i2c->pdata->area_edge = max_y * 3 / 100;
|
||
|
|
||
|
input_raw_info(true, &wac_i2c->client->dev,
|
||
|
"MAX XY(%d/%d), area_edge %d, area_indicator %d, area_navigation %d\n",
|
||
|
max_x, max_y, wac_i2c->pdata->area_edge,
|
||
|
wac_i2c->pdata->area_indicator, wac_i2c->pdata->area_navigation);
|
||
|
}
|
||
|
|
||
|
for (i = 0; i < EPEN_LOCATION_DETECT_SIZE; ++i)
|
||
|
loc[i] = 0;
|
||
|
|
||
|
if (x < wac_i2c->pdata->area_edge)
|
||
|
strcat(loc, "E.");
|
||
|
else if (x < (max_x - wac_i2c->pdata->area_edge))
|
||
|
strcat(loc, "C.");
|
||
|
else
|
||
|
strcat(loc, "e.");
|
||
|
|
||
|
if (y < wac_i2c->pdata->area_indicator)
|
||
|
strcat(loc, "S");
|
||
|
else if (y < (max_y - wac_i2c->pdata->area_navigation))
|
||
|
strcat(loc, "C");
|
||
|
else
|
||
|
strcat(loc, "N");
|
||
|
}
|
||
|
|
||
|
void w9019_i2c_coord_modify(struct wacom_i2c *wac_i2c)
|
||
|
{
|
||
|
struct wacom_g5_platform_data *pdata = wac_i2c->pdata;
|
||
|
s16 tmp;
|
||
|
|
||
|
/* origin */
|
||
|
wac_i2c->x = wac_i2c->x - pdata->origin[0];
|
||
|
wac_i2c->y = wac_i2c->y - pdata->origin[1];
|
||
|
|
||
|
/* change axis from wacom to lcd */
|
||
|
if (pdata->x_invert) {
|
||
|
wac_i2c->x = pdata->max_x - wac_i2c->x;
|
||
|
wac_i2c->tilt_x = -wac_i2c->tilt_x;
|
||
|
}
|
||
|
|
||
|
if (pdata->y_invert) {
|
||
|
wac_i2c->y = pdata->max_y - wac_i2c->y;
|
||
|
wac_i2c->tilt_y = -wac_i2c->tilt_y;
|
||
|
}
|
||
|
|
||
|
if (pdata->xy_switch) {
|
||
|
tmp = wac_i2c->x;
|
||
|
wac_i2c->x = wac_i2c->y;
|
||
|
wac_i2c->y = tmp;
|
||
|
|
||
|
tmp = wac_i2c->tilt_x;
|
||
|
wac_i2c->tilt_x = wac_i2c->tilt_y;
|
||
|
wac_i2c->tilt_y = tmp;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void wacom_i2c_coord_handler(struct wacom_i2c *wac_i2c, char *data)
|
||
|
{
|
||
|
struct wacom_g5_platform_data *pdata = wac_i2c->pdata;
|
||
|
struct i2c_client *client = wac_i2c->client;
|
||
|
bool prox = false;
|
||
|
bool rdy = false;
|
||
|
bool stylus;
|
||
|
char location[EPEN_LOCATION_DETECT_SIZE] = { 0, };
|
||
|
|
||
|
if (wac_i2c->debug_flag & WACOM_DEBUG_PRINT_COORDEVENT) {
|
||
|
input_info(true, &wac_i2c->client->dev,
|
||
|
"%s : %x %x %x %x %x %x %x %x %x %x %x %x %x %x\n",
|
||
|
__func__, data[0], data[1], data[2], data[3], data[4], data[5],
|
||
|
data[6], data[7], data[8], data[9], data[10], data[11],
|
||
|
data[12], data[13]);
|
||
|
}
|
||
|
|
||
|
rdy = !!(data[0] & 0x80);
|
||
|
|
||
|
wac_i2c->x = ((u16)data[1] << 8) + (u16)data[2];
|
||
|
wac_i2c->y = ((u16)data[3] << 8) + (u16)data[4];
|
||
|
wac_i2c->pressure = ((u16)(data[5] & 0x0F) << 8) + (u16)data[6];
|
||
|
wac_i2c->height = (u8)data[7];
|
||
|
wac_i2c->tilt_x = (s8)data[8];
|
||
|
wac_i2c->tilt_y = (s8)data[9];
|
||
|
|
||
|
w9019_i2c_coord_modify(wac_i2c);
|
||
|
|
||
|
if (rdy) {
|
||
|
prox = !!(data[0] & 0x10);
|
||
|
stylus = !!(data[0] & 0x20);
|
||
|
|
||
|
/* validation check */
|
||
|
if (unlikely (pdata->xy_switch == 0 && (wac_i2c->x > pdata->max_x || wac_i2c->y > pdata->max_y) ||
|
||
|
pdata->xy_switch == 1 && (wac_i2c->x > pdata->max_y || wac_i2c->y > pdata->max_x))) {
|
||
|
#if WACOM_PRODUCT_SHIP
|
||
|
input_info(true, &client->dev, "%s : Abnormal raw data x & y\n", __func__);
|
||
|
#else
|
||
|
input_info(true, &client->dev, "%s : Abnormal raw data x=%d, y=%d\n", __func__, wac_i2c->x, wac_i2c->y);
|
||
|
#endif
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (data[0] & 0x40)
|
||
|
wac_i2c->tool = BTN_TOOL_RUBBER;
|
||
|
else
|
||
|
wac_i2c->tool = BTN_TOOL_PEN;
|
||
|
|
||
|
if (!wac_i2c->pen_prox) {
|
||
|
wac_i2c->pen_prox = true;
|
||
|
|
||
|
w9019_epen_location_detect(wac_i2c, location, wac_i2c->x, wac_i2c->y);
|
||
|
wac_i2c->hi_x = wac_i2c->x;
|
||
|
wac_i2c->hi_y = wac_i2c->y;
|
||
|
#if WACOM_PRODUCT_SHIP
|
||
|
input_info(true, &client->dev, "[HI] loc:%s (%s)\n",
|
||
|
location, wac_i2c->tool == BTN_TOOL_PEN ? "p" : "r");
|
||
|
#else
|
||
|
input_info(true, &client->dev, "[HI] x:%d y:%d loc:%s (%s) \n",
|
||
|
wac_i2c->x, wac_i2c->y, location, wac_i2c->tool == BTN_TOOL_PEN ? "pen" : "rubber");
|
||
|
#endif
|
||
|
#if IS_ENABLED(CONFIG_INPUT_SEC_NOTIFIER)
|
||
|
sec_input_notify(&wac_i2c->nb, NOTIFIER_WACOM_PEN_HOVER_IN, NULL);
|
||
|
#endif
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
/* report info */
|
||
|
input_report_abs(wac_i2c->input_dev, ABS_X, wac_i2c->x);
|
||
|
input_report_abs(wac_i2c->input_dev, ABS_Y, wac_i2c->y);
|
||
|
input_report_key(wac_i2c->input_dev, BTN_STYLUS, stylus);
|
||
|
input_report_key(wac_i2c->input_dev, BTN_TOUCH, prox);
|
||
|
|
||
|
input_report_abs(wac_i2c->input_dev, ABS_PRESSURE, wac_i2c->pressure);
|
||
|
input_report_abs(wac_i2c->input_dev, ABS_DISTANCE, wac_i2c->height);
|
||
|
input_report_abs(wac_i2c->input_dev, ABS_TILT_X, wac_i2c->tilt_x);
|
||
|
input_report_abs(wac_i2c->input_dev, ABS_TILT_Y, wac_i2c->tilt_y);
|
||
|
input_report_key(wac_i2c->input_dev, wac_i2c->tool, 1);
|
||
|
input_sync(wac_i2c->input_dev);
|
||
|
|
||
|
if (wac_i2c->debug_flag & WACOM_DEBUG_PRINT_ALLEVENT)
|
||
|
input_info(true, &client->dev, "[A] x:%d y:%d, p:%d, h:%d, tx:%d, ty:%d\n",
|
||
|
wac_i2c->x, wac_i2c->y, wac_i2c->pressure, wac_i2c->height,
|
||
|
wac_i2c->tilt_x, wac_i2c->tilt_y);
|
||
|
|
||
|
wac_i2c->mcount++;
|
||
|
|
||
|
/* log */
|
||
|
if (prox && !wac_i2c->pen_pressed) {
|
||
|
w9019_epen_location_detect(wac_i2c, location, wac_i2c->x, wac_i2c->y);
|
||
|
wac_i2c->p_x = wac_i2c->x;
|
||
|
wac_i2c->p_y = wac_i2c->y;
|
||
|
|
||
|
#if WACOM_PRODUCT_SHIP
|
||
|
input_info(true, &client->dev, "[P] p:%d loc:%s tool:%x mc:%d\n",
|
||
|
wac_i2c->pressure, location, wac_i2c->tool, wac_i2c->mcount);
|
||
|
#else
|
||
|
input_info(true, &client->dev,
|
||
|
"[P] x:%d y:%d p:%d loc:%s tool:%x mc:%d\n",
|
||
|
wac_i2c->x, wac_i2c->y, wac_i2c->pressure, location, wac_i2c->tool, wac_i2c->mcount);
|
||
|
#endif
|
||
|
} else if (!prox && wac_i2c->pen_pressed) {
|
||
|
w9019_epen_location_detect(wac_i2c, location, wac_i2c->x, wac_i2c->y);
|
||
|
#if WACOM_PRODUCT_SHIP
|
||
|
input_info(true, &client->dev, "[R] dd:%d,%d loc:%s tool:%x mc:%d\n",
|
||
|
wac_i2c->x - wac_i2c->p_x, wac_i2c->y - wac_i2c->p_y,
|
||
|
location, wac_i2c->tool, wac_i2c->mcount);
|
||
|
#else
|
||
|
input_info(true, &client->dev,
|
||
|
"[R] lx:%d ly:%d dd:%d,%d loc:%s tool:%x mc:%d\n",
|
||
|
wac_i2c->x, wac_i2c->y,
|
||
|
wac_i2c->x - wac_i2c->p_x, wac_i2c->y - wac_i2c->p_y,
|
||
|
location, wac_i2c->tool, wac_i2c->mcount);
|
||
|
#endif
|
||
|
wac_i2c->p_x = wac_i2c->p_y = 0;
|
||
|
}
|
||
|
wac_i2c->pen_pressed = prox;
|
||
|
|
||
|
/* check side */
|
||
|
if (stylus && !wac_i2c->side_pressed)
|
||
|
input_info(true, &client->dev, "side on\n");
|
||
|
else if (!stylus && wac_i2c->side_pressed)
|
||
|
input_info(true, &client->dev, "side off\n");
|
||
|
|
||
|
wac_i2c->side_pressed = stylus;
|
||
|
} else {
|
||
|
if (wac_i2c->pen_prox) {
|
||
|
input_report_abs(wac_i2c->input_dev, ABS_PRESSURE, 0);
|
||
|
input_report_key(wac_i2c->input_dev, BTN_STYLUS, 0);
|
||
|
input_report_key(wac_i2c->input_dev, BTN_TOUCH, 0);
|
||
|
/* prevent invalid operation of input booster */
|
||
|
input_sync(wac_i2c->input_dev);
|
||
|
|
||
|
input_report_abs(wac_i2c->input_dev, ABS_DISTANCE, 0);
|
||
|
input_report_key(wac_i2c->input_dev, BTN_TOOL_RUBBER, 0);
|
||
|
input_report_key(wac_i2c->input_dev, BTN_TOOL_PEN, 0);
|
||
|
input_sync(wac_i2c->input_dev);
|
||
|
|
||
|
w9019_epen_location_detect(wac_i2c, location, wac_i2c->x, wac_i2c->y);
|
||
|
|
||
|
if (wac_i2c->pen_pressed) {
|
||
|
#if WACOM_PRODUCT_SHIP
|
||
|
input_info(true, &client->dev,
|
||
|
"[R] dd:%d,%d loc:%s tool:%x mc:%d & [HO] dd:%d,%d\n",
|
||
|
wac_i2c->x - wac_i2c->p_x, wac_i2c->y - wac_i2c->p_y,
|
||
|
location, wac_i2c->tool, wac_i2c->mcount,
|
||
|
wac_i2c->x - wac_i2c->hi_x, wac_i2c->y - wac_i2c->hi_y);
|
||
|
#else
|
||
|
input_info(true, &client->dev,
|
||
|
"[R] lx:%d ly:%d dd:%d,%d loc:%s tool:%x mc:%d & [HO] dd:%d,%d\n",
|
||
|
wac_i2c->x, wac_i2c->y,
|
||
|
wac_i2c->x - wac_i2c->p_x, wac_i2c->y - wac_i2c->p_y,
|
||
|
location, wac_i2c->tool, wac_i2c->mcount,
|
||
|
wac_i2c->x - wac_i2c->hi_x, wac_i2c->y - wac_i2c->hi_y);
|
||
|
#endif
|
||
|
} else {
|
||
|
#if WACOM_PRODUCT_SHIP
|
||
|
input_info(true, &client->dev, "[HO] dd:%d,%d loc:%s mc:%d\n",
|
||
|
wac_i2c->x - wac_i2c->hi_x, wac_i2c->y - wac_i2c->hi_y,
|
||
|
location, wac_i2c->mcount);
|
||
|
|
||
|
#else
|
||
|
input_info(true, &client->dev, "[HO] lx:%d ly:%d dd:%d,%d loc:%s mc:%d\n",
|
||
|
wac_i2c->x, wac_i2c->y,
|
||
|
wac_i2c->x - wac_i2c->hi_x, wac_i2c->y - wac_i2c->hi_y,
|
||
|
location, wac_i2c->mcount);
|
||
|
#endif
|
||
|
}
|
||
|
#if IS_ENABLED(CONFIG_INPUT_SEC_NOTIFIER)
|
||
|
sec_input_notify(&wac_i2c->nb, NOTIFIER_WACOM_PEN_HOVER_OUT, NULL);
|
||
|
#endif
|
||
|
wac_i2c->p_x = wac_i2c->p_y = wac_i2c->hi_x = wac_i2c->hi_y = 0;
|
||
|
|
||
|
} else {
|
||
|
input_info(true, &client->dev,
|
||
|
"unexpected data : %x %x %x %x %x %x %x %x %x %x %x %x %x %x %x %x\n",
|
||
|
data[0], data[1], data[2], data[3], data[4], data[5],
|
||
|
data[6], data[7], data[8], data[9], data[10], data[11],
|
||
|
data[12], data[13], data[14], data[15]);
|
||
|
}
|
||
|
|
||
|
wac_i2c->pen_prox = 0;
|
||
|
wac_i2c->pen_pressed = 0;
|
||
|
wac_i2c->side_pressed = 0;
|
||
|
wac_i2c->mcount = 0;
|
||
|
}
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
static void wacom_i2c_aop_handler(struct wacom_i2c *wac_i2c, char *data)
|
||
|
{
|
||
|
struct wacom_g5_platform_data *pdata = wac_i2c->pdata;
|
||
|
//bool stylus, prox = false;
|
||
|
s16 x, y, tmp;
|
||
|
// u8 wacom_mode, wakeup_id;
|
||
|
s16 pressure, height, tilt_x, tilt_y;
|
||
|
|
||
|
if (wac_i2c->screen_on || !wac_i2c->survey_mode ||
|
||
|
!(wac_i2c->function_set & EPEN_SETMODE_AOP)) {
|
||
|
input_info(true, &wac_i2c->client->dev, "%s: unexpected status(%d %d %d)\n",
|
||
|
__func__, wac_i2c->screen_on, wac_i2c->survey_mode,
|
||
|
wac_i2c->function_set & EPEN_SETMODE_AOP);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (wac_i2c->function_set & EPEN_SETMODE_AOP) {
|
||
|
if (data[10] == AOP_BUTTON_HOVER) {
|
||
|
if (wac_i2c->function_set & EPEN_SETMODE_AOP_OPTION_SCREENOFFMEMO) {
|
||
|
input_info(true, &wac_i2c->client->dev, "Hover & Side Button detected\n");
|
||
|
|
||
|
input_report_key(wac_i2c->input_dev,
|
||
|
KEY_WAKEUP_UNLOCK, 1);
|
||
|
input_sync(wac_i2c->input_dev);
|
||
|
|
||
|
input_report_key(wac_i2c->input_dev,
|
||
|
KEY_WAKEUP_UNLOCK, 0);
|
||
|
input_sync(wac_i2c->input_dev);
|
||
|
|
||
|
x = ((u16) data[1] << 8) + (u16) data[2];
|
||
|
y = ((u16) data[3] << 8) + (u16) data[4];
|
||
|
|
||
|
/* origin */
|
||
|
x = x - pdata->origin[0];
|
||
|
y = y - pdata->origin[1];
|
||
|
/* change axis from wacom to lcd */
|
||
|
if (pdata->x_invert)
|
||
|
x = pdata->max_x - x;
|
||
|
|
||
|
if (pdata->y_invert)
|
||
|
y = pdata->max_y - y;
|
||
|
|
||
|
if (pdata->xy_switch) {
|
||
|
tmp = x;
|
||
|
x = y;
|
||
|
y = tmp;
|
||
|
}
|
||
|
|
||
|
wac_i2c->survey_pos.id = EPEN_POS_ID_SCREEN_OF_MEMO;
|
||
|
wac_i2c->survey_pos.x = x;
|
||
|
wac_i2c->survey_pos.y = y;
|
||
|
} else {
|
||
|
input_info(true, &wac_i2c->client->dev,
|
||
|
"AOP detected but skip report, screen_off_memo disabled\n");
|
||
|
}
|
||
|
} else if (data[10] == AOP_DOUBLE_TAB) {
|
||
|
if ((wac_i2c->function_set & EPEN_SETMODE_AOP_OPTION_AOD_LCD_ON) == EPEN_SETMODE_AOP_OPTION_AOD_LCD_ON) {
|
||
|
input_info(true, &wac_i2c->client->dev, "Double Tab detected in AOD\n");
|
||
|
|
||
|
x = ((u16) data[1] << 8) + (u16) data[2];
|
||
|
y = ((u16) data[3] << 8) + (u16) data[4];
|
||
|
pressure = ((u16) data[5] << 8) + (u16) data[6];
|
||
|
height = data[7];
|
||
|
tilt_x = (s8) data[9];
|
||
|
tilt_y = -(s8) data[8];
|
||
|
|
||
|
/* origin */
|
||
|
x = x - pdata->origin[0];
|
||
|
y = y - pdata->origin[1];
|
||
|
/* change axis from wacom to lcd */
|
||
|
if (pdata->x_invert)
|
||
|
x = pdata->max_x - x;
|
||
|
|
||
|
if (pdata->y_invert)
|
||
|
y = pdata->max_y - y;
|
||
|
|
||
|
if (pdata->xy_switch) {
|
||
|
tmp = x;
|
||
|
x = y;
|
||
|
y = tmp;
|
||
|
}
|
||
|
|
||
|
if (data[0] & 0x40)
|
||
|
wac_i2c->tool = BTN_TOOL_RUBBER;
|
||
|
else
|
||
|
wac_i2c->tool = BTN_TOOL_PEN;
|
||
|
|
||
|
/* make press / release event for AOP double tab gesture */
|
||
|
input_report_abs(wac_i2c->input_dev, ABS_X, x);
|
||
|
input_report_abs(wac_i2c->input_dev, ABS_Y, y);
|
||
|
input_report_key(wac_i2c->input_dev, wac_i2c->tool, 1);
|
||
|
input_sync(wac_i2c->input_dev);
|
||
|
|
||
|
input_report_abs(wac_i2c->input_dev, ABS_PRESSURE, pressure);
|
||
|
input_report_key(wac_i2c->input_dev, BTN_TOUCH, 1);
|
||
|
input_sync(wac_i2c->input_dev);
|
||
|
|
||
|
input_report_abs(wac_i2c->input_dev, ABS_PRESSURE, 0);
|
||
|
input_report_key(wac_i2c->input_dev, BTN_TOUCH, 0);
|
||
|
input_sync(wac_i2c->input_dev);
|
||
|
|
||
|
input_report_key(wac_i2c->input_dev, wac_i2c->tool, 0);
|
||
|
input_sync(wac_i2c->input_dev);
|
||
|
#if WACOM_PRODUCT_SHIP
|
||
|
input_info(true, &wac_i2c->client->dev, "[P/R] tool:%x\n",
|
||
|
wac_i2c->tool);
|
||
|
#else
|
||
|
input_info(true, &wac_i2c->client->dev,
|
||
|
"[P/R] x:%d y:%d tool:%x\n", x, y, wac_i2c->tool);
|
||
|
#endif
|
||
|
} else if (wac_i2c->function_set & EPEN_SETMODE_AOP_OPTION_AOT) {
|
||
|
input_info(true, &wac_i2c->client->dev, "Double Tab detected\n");
|
||
|
|
||
|
input_report_key(wac_i2c->input_dev, KEY_HOMEPAGE, 1);
|
||
|
input_sync(wac_i2c->input_dev);
|
||
|
input_report_key(wac_i2c->input_dev, KEY_HOMEPAGE, 0);
|
||
|
input_sync(wac_i2c->input_dev);
|
||
|
} else {
|
||
|
input_info(true, &wac_i2c->client->dev,
|
||
|
"AOP Double Tab detected but skip report, aod & aot disabled\n");
|
||
|
}
|
||
|
} else {
|
||
|
input_info(true, &wac_i2c->client->dev,
|
||
|
"unexpected AOP data : %x %x %x %x %x %x %x %x %x %x %x %x %x %x\n",
|
||
|
data[0], data[1], data[2], data[3], data[4], data[5],
|
||
|
data[6], data[7], data[8], data[9], data[10], data[11],
|
||
|
data[12], data[13]);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
static int wacom_event_handler(struct wacom_i2c *wac_i2c)
|
||
|
{
|
||
|
int ret;
|
||
|
bool debug = true;
|
||
|
char buff[COM_COORD_NUM + 1] = { 0, };
|
||
|
|
||
|
ret = w9019_i2c_recv(wac_i2c, buff, COM_COORD_NUM + 1);
|
||
|
if (ret < 0) {
|
||
|
input_err(true, &wac_i2c->client->dev, "%s: read failed\n", __func__);
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
if (!wac_i2c->screen_on && wac_i2c->survey_mode) {
|
||
|
input_info(true, &wac_i2c->client->dev,
|
||
|
"%s: lcd off & survey mode on\n", __func__);
|
||
|
wacom_i2c_aop_handler(wac_i2c, buff);
|
||
|
} else {
|
||
|
wacom_i2c_coord_handler(wac_i2c, buff);
|
||
|
debug = false;
|
||
|
}
|
||
|
|
||
|
if (debug) {
|
||
|
input_info(true, &wac_i2c->client->dev,
|
||
|
"%x %x %x %x %x %x %x %x %x %x %x %x %x %x\n",
|
||
|
buff[0], buff[1], buff[2], buff[3], buff[4], buff[5],
|
||
|
buff[6], buff[7], buff[8], buff[9], buff[10], buff[11],
|
||
|
buff[12], buff[13]);
|
||
|
}
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
static irqreturn_t wacom_interrupt(int irq, void *dev_id)
|
||
|
{
|
||
|
struct wacom_i2c *wac_i2c = dev_id;
|
||
|
int ret = 0;
|
||
|
|
||
|
/* in LPM, waiting blsp block resume */
|
||
|
if (wac_i2c->pm_suspend) {
|
||
|
__pm_wakeup_event(wac_i2c->wacom_ws, jiffies_to_msecs(500));
|
||
|
/* waiting for blsp block resuming, if not occurs i2c error */
|
||
|
ret = wait_for_completion_interruptible_timeout(&wac_i2c->resume_done, msecs_to_jiffies(500));
|
||
|
if (ret <= 0) {
|
||
|
input_err(true, &wac_i2c->client->dev, "LPM: pm resume is not handled [%d]\n", ret);
|
||
|
return IRQ_HANDLED;
|
||
|
} else {
|
||
|
input_info(true, &wac_i2c->client->dev,
|
||
|
"%s: run LPM interrupt handler, %d\n", __func__, jiffies_to_msecs(ret));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ret = wacom_event_handler(wac_i2c);
|
||
|
if (ret < 0) {
|
||
|
w9019_forced_release(wac_i2c);
|
||
|
w9019_forced_release_fullscan(wac_i2c);
|
||
|
wac_i2c->reset_flag = true;
|
||
|
}
|
||
|
|
||
|
return IRQ_HANDLED;
|
||
|
}
|
||
|
|
||
|
static void open_test_work(struct work_struct *work)
|
||
|
{
|
||
|
struct wacom_i2c *wac_i2c =
|
||
|
container_of(work, struct wacom_i2c, open_test_dwork.work);
|
||
|
char data;
|
||
|
int ret = 0;
|
||
|
|
||
|
#if !WACOM_SEC_FACTORY
|
||
|
input_info(true, &wac_i2c->client->dev, "%s : start!\n", __func__);
|
||
|
|
||
|
if (wac_i2c->pdata->support_garage_open_test) {
|
||
|
ret = w9019_open_test(wac_i2c, WACOM_GARAGE_TEST);
|
||
|
if (ret) {
|
||
|
input_err(true, &wac_i2c->client->dev, "grage test check failed %d\n", ret);
|
||
|
w9019_reset_hw(wac_i2c);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ret = w9019_open_test(wac_i2c, WACOM_DIGITIZER_TEST);
|
||
|
if (ret) {
|
||
|
input_err(true, &wac_i2c->client->dev, "open test check failed %d\n", ret);
|
||
|
w9019_reset_hw(wac_i2c);
|
||
|
}
|
||
|
|
||
|
#if IS_ENABLED(CONFIG_INPUT_SEC_NOTIFIER)
|
||
|
if (wac_i2c->is_tsp_block) {
|
||
|
wac_i2c->tsp_scan_mode = sec_input_notify(&wac_i2c->nb, NOTIFIER_TSP_BLOCKING_RELEASE, NULL);
|
||
|
wac_i2c->is_tsp_block = false;
|
||
|
input_err(true, &wac_i2c->client->dev, "%s : release tsp scan block\n", __func__);
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
input_info(true, &wac_i2c->client->dev, "%s : end!\n", __func__);
|
||
|
#else
|
||
|
input_info(true, &wac_i2c->client->dev, "open test skiped!\n");
|
||
|
#endif
|
||
|
|
||
|
/* occur cover status event*/
|
||
|
if (wac_i2c->pdata->support_cover_detection) {
|
||
|
data = COM_KBDCOVER_CHECK_STATUS;
|
||
|
ret = w9019_i2c_send(wac_i2c, &data, 1);
|
||
|
if (ret < 0) {
|
||
|
input_err(true, &wac_i2c->client->dev, "%s: failed to send cover status event %d\n", __func__, ret);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void probe_open_test(struct wacom_i2c *wac_i2c)
|
||
|
{
|
||
|
INIT_DELAYED_WORK(&wac_i2c->open_test_dwork, open_test_work);
|
||
|
|
||
|
/* update the current status */
|
||
|
schedule_delayed_work(&wac_i2c->open_test_dwork, msecs_to_jiffies(100));
|
||
|
}
|
||
|
|
||
|
static int wacom_i2c_input_open(struct input_dev *dev)
|
||
|
{
|
||
|
struct wacom_i2c *wac_i2c = input_get_drvdata(dev);
|
||
|
int ret = 0;
|
||
|
|
||
|
input_info(true, &wac_i2c->client->dev, "%s(%s)\n", __func__,
|
||
|
dev->name);
|
||
|
|
||
|
wac_i2c->pdata->enabled = true;
|
||
|
|
||
|
#if WACOM_SEC_FACTORY
|
||
|
if (wac_i2c->epen_blocked){
|
||
|
input_err(true, &wac_i2c->client->dev, "%s : FAC epen_blocked SKIP!!\n", __func__);
|
||
|
return ret;
|
||
|
}
|
||
|
#else
|
||
|
if (wac_i2c->epen_blocked){
|
||
|
input_err(true, &wac_i2c->client->dev, "%s : epen_blocked. disable mode \n", __func__);
|
||
|
w9019_disable_mode(wac_i2c, WACOM_DISABLE);
|
||
|
return ret;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
w9019_wakeup_sequence(wac_i2c);
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
static void wacom_i2c_input_close(struct input_dev *dev)
|
||
|
{
|
||
|
struct wacom_i2c *wac_i2c = input_get_drvdata(dev);
|
||
|
|
||
|
input_info(true, &wac_i2c->client->dev, "%s(%s)\n", __func__,
|
||
|
dev->name);
|
||
|
|
||
|
if (!wac_i2c->probe_done) {
|
||
|
input_err(true, &wac_i2c->client->dev, "%s : not probe done & skip!\n", __func__);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
wac_i2c->pdata->enabled = false;
|
||
|
|
||
|
#if WACOM_SEC_FACTORY
|
||
|
if (wac_i2c->epen_blocked){
|
||
|
input_err(true, &wac_i2c->client->dev, "%s : FAC epen_blocked SKIP!!\n", __func__);
|
||
|
return;
|
||
|
}
|
||
|
#else
|
||
|
if (wac_i2c->epen_blocked){
|
||
|
input_err(true, &wac_i2c->client->dev, "%s : epen_blocked. disable mode \n", __func__);
|
||
|
w9019_disable_mode(wac_i2c, WACOM_DISABLE);
|
||
|
return;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
w9019_sleep_sequence(wac_i2c);
|
||
|
}
|
||
|
|
||
|
static void wacom_i2c_set_input_values(struct wacom_i2c *wac_i2c,
|
||
|
struct input_dev *input_dev)
|
||
|
{
|
||
|
struct i2c_client *client = wac_i2c->client;
|
||
|
struct wacom_g5_platform_data *pdata = wac_i2c->pdata;
|
||
|
/* Set input values before registering input device */
|
||
|
|
||
|
input_dev->id.bustype = BUS_I2C;
|
||
|
input_dev->dev.parent = &client->dev;
|
||
|
|
||
|
input_dev->open = wacom_i2c_input_open;
|
||
|
input_dev->close = wacom_i2c_input_close;
|
||
|
|
||
|
input_set_abs_params(input_dev, ABS_PRESSURE, 0, pdata->max_pressure, 0, 0);
|
||
|
input_set_abs_params(input_dev, ABS_DISTANCE, 0, pdata->max_height, 0, 0);
|
||
|
input_set_abs_params(input_dev, ABS_TILT_X, -pdata->max_x_tilt, pdata->max_x_tilt, 0, 0);
|
||
|
input_set_abs_params(input_dev, ABS_TILT_Y, -pdata->max_y_tilt, pdata->max_y_tilt, 0, 0);
|
||
|
|
||
|
if (pdata->xy_switch) {
|
||
|
input_set_abs_params(input_dev, ABS_X, 0, pdata->max_y, 4, 0);
|
||
|
input_set_abs_params(input_dev, ABS_Y, 0, pdata->max_x, 4, 0);
|
||
|
} else {
|
||
|
input_set_abs_params(input_dev, ABS_X, 0, pdata->max_x, 4, 0);
|
||
|
input_set_abs_params(input_dev, ABS_Y, 0, pdata->max_y, 4, 0);
|
||
|
}
|
||
|
|
||
|
input_set_capability(input_dev, EV_KEY, BTN_TOOL_PEN);
|
||
|
input_set_capability(input_dev, EV_KEY, BTN_TOOL_RUBBER);
|
||
|
input_set_capability(input_dev, EV_KEY, BTN_TOUCH);
|
||
|
input_set_capability(input_dev, EV_KEY, BTN_STYLUS);
|
||
|
|
||
|
input_set_capability(input_dev, EV_SW, SW_PEN_INSERT);
|
||
|
|
||
|
/* AOP */
|
||
|
input_set_capability(input_dev, EV_KEY, KEY_WAKEUP_UNLOCK);
|
||
|
input_set_capability(input_dev, EV_KEY, KEY_HOMEPAGE);
|
||
|
|
||
|
/* flip cover */
|
||
|
input_set_capability(input_dev, EV_SW, SW_FLIP);
|
||
|
|
||
|
input_set_drvdata(input_dev, wac_i2c);
|
||
|
}
|
||
|
|
||
|
void w9019_print_info(struct wacom_i2c *wac_i2c)
|
||
|
{
|
||
|
if (!wac_i2c)
|
||
|
return;
|
||
|
|
||
|
if (!wac_i2c->client)
|
||
|
return;
|
||
|
|
||
|
wac_i2c->print_info_cnt_open++;
|
||
|
|
||
|
if (wac_i2c->print_info_cnt_open > 0xfff0)
|
||
|
wac_i2c->print_info_cnt_open = 0;
|
||
|
if (wac_i2c->scan_info_fail_cnt > 1000)
|
||
|
wac_i2c->scan_info_fail_cnt = 1000;
|
||
|
|
||
|
input_info(true, &wac_i2c->client->dev,
|
||
|
"%s: ps %s, pen %s, report_scan_seq %d, epen %s, count(%u,%u,%u), "
|
||
|
"mode(%d), block_cnt(%d), check(%d), test(%d,%d), ver[0x%x], cover(%d,%d) #%d\n",
|
||
|
__func__, wac_i2c->battery_saving_mode ? "on" : "off",
|
||
|
(wac_i2c->function_result & EPEN_EVENT_PEN_OUT) ? "out" : "in",
|
||
|
wac_i2c->report_scan_seq, wac_i2c->epen_blocked ? "blocked" : "unblocked",
|
||
|
wac_i2c->i2c_fail_count, wac_i2c->abnormal_reset_count, wac_i2c->scan_info_fail_cnt,
|
||
|
wac_i2c->check_survey_mode, wac_i2c->tsp_block_cnt, wac_i2c->check_elec,
|
||
|
wac_i2c->connection_check, wac_i2c->garage_connection_check,
|
||
|
wac_i2c->fw_ver_ic, wac_i2c->cover, wac_i2c->flip_state, wac_i2c->print_info_cnt_open);
|
||
|
}
|
||
|
|
||
|
static void wacom_print_info_work(struct work_struct *work)
|
||
|
{
|
||
|
struct wacom_i2c *wac_i2c = container_of(work, struct wacom_i2c, work_print_info.work);
|
||
|
|
||
|
w9019_print_info(wac_i2c);
|
||
|
schedule_delayed_work(&wac_i2c->work_print_info, msecs_to_jiffies(30000)); // 30s
|
||
|
}
|
||
|
|
||
|
int w9019_load_fw_built_in(struct wacom_i2c *wac_i2c, int fw_index)
|
||
|
{
|
||
|
int ret = 0;
|
||
|
const char *fw_load_path = NULL;
|
||
|
#if WACOM_SEC_FACTORY
|
||
|
int index = 0;
|
||
|
#endif
|
||
|
|
||
|
input_info(true, &wac_i2c->client->dev, "%s: load_fw_built_in (%d)\n", __func__, fw_index);
|
||
|
|
||
|
fw_load_path = wac_i2c->pdata->fw_path;
|
||
|
|
||
|
#if WACOM_SEC_FACTORY
|
||
|
if (fw_index != FW_BUILT_IN) {
|
||
|
if (fw_index == FW_FACTORY_GARAGE)
|
||
|
index = 0;
|
||
|
else if (fw_index == FW_FACTORY_UNIT)
|
||
|
index = 1;
|
||
|
|
||
|
ret = of_property_read_string_index(wac_i2c->client->dev.of_node,
|
||
|
"wacom,fw_fac_path", index, &wac_i2c->pdata->fw_fac_path);
|
||
|
if (ret) {
|
||
|
input_err(true, &wac_i2c->client->dev, "%s: failed to read fw_fac_path %d\n", __func__, ret);
|
||
|
wac_i2c->pdata->fw_fac_path = NULL;
|
||
|
}
|
||
|
|
||
|
fw_load_path = wac_i2c->pdata->fw_fac_path;
|
||
|
|
||
|
input_info(true, &wac_i2c->client->dev, "%s: load %s firmware\n",
|
||
|
__func__, fw_load_path);
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
if (fw_load_path == NULL) {
|
||
|
input_err(true, &wac_i2c->client->dev,
|
||
|
"Unable to open firmware. fw_path is NULL\n");
|
||
|
return -EINVAL;
|
||
|
}
|
||
|
|
||
|
ret = request_firmware(&wac_i2c->firm_data, fw_load_path, &wac_i2c->client->dev);
|
||
|
if (ret < 0) {
|
||
|
input_err(true, &wac_i2c->client->dev, "Unable to open firmware. ret %d\n", ret);
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
wac_i2c->fw_img = (struct fw_image *)wac_i2c->firm_data->data;
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
static int wacom_i2c_get_fw_size(struct wacom_i2c *wac_i2c)
|
||
|
{
|
||
|
u32 fw_size = 0;
|
||
|
|
||
|
if (wac_i2c->pdata->ic_type == MPU_W9020)
|
||
|
fw_size = 144 * 1024;
|
||
|
else if (wac_i2c->pdata->ic_type == MPU_W9021)
|
||
|
fw_size = 256 * 1024;
|
||
|
else if (wac_i2c->pdata->ic_type == MPU_WEZ01)
|
||
|
fw_size = 256 * 1024;
|
||
|
else
|
||
|
fw_size = 128 * 1024;
|
||
|
|
||
|
input_info(true, &wac_i2c->client->dev, "%s: type[%d] size[0x%X]\n",
|
||
|
__func__, wac_i2c->pdata->ic_type, fw_size);
|
||
|
|
||
|
return fw_size;
|
||
|
}
|
||
|
|
||
|
static int load_fw_sdcard(struct wacom_i2c *wac_i2c, u8 fw_index, const char *file_path)
|
||
|
{
|
||
|
struct i2c_client *client = wac_i2c->client;
|
||
|
long fsize;
|
||
|
int ret = 0;
|
||
|
unsigned int nSize;
|
||
|
unsigned long nSize2;
|
||
|
#ifdef SUPPORT_FW_SIGNED
|
||
|
long spu_ret;
|
||
|
#endif
|
||
|
|
||
|
nSize = wacom_i2c_get_fw_size(wac_i2c);
|
||
|
nSize2 = nSize + sizeof(struct fw_image);
|
||
|
|
||
|
ret = request_firmware(&wac_i2c->firm_data, file_path, &wac_i2c->client->dev);
|
||
|
if (ret) {
|
||
|
input_err(true, &client->dev, "firmware is not available %d.\n", ret);
|
||
|
ret = -ENOENT;
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
fsize = wac_i2c->firm_data->size;
|
||
|
input_info(true, &client->dev, "start, file path %s, size %ld Bytes\n", file_path, fsize);
|
||
|
|
||
|
#ifdef SUPPORT_FW_SIGNED
|
||
|
if (fw_index == FW_IN_SDCARD_SIGNED || fw_index == FW_SPU || fw_index == FW_VERIFICATION) {
|
||
|
/* name 5, digest 32, signature 512 */
|
||
|
fsize -= SPU_METADATA_SIZE(WACOM);
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
if ((fsize != nSize) && (fsize != nSize2)) {
|
||
|
input_err(true, &client->dev, "UMS firmware size is different\n");
|
||
|
ret = -EFBIG;
|
||
|
goto out;
|
||
|
}
|
||
|
|
||
|
#ifdef SUPPORT_FW_SIGNED
|
||
|
if (fw_index == FW_IN_SDCARD_SIGNED || fw_index == FW_SPU || fw_index == FW_VERIFICATION) {
|
||
|
/* name 5, digest 32, signature 512 */
|
||
|
|
||
|
spu_ret = spu_firmware_signature_verify("WACOM", wac_i2c->firm_data->data, wac_i2c->firm_data->size);
|
||
|
|
||
|
input_info(true, &client->dev, "%s: spu_ret : %ld, spu_fsize : %ld // fsize:%ld\n",
|
||
|
__func__, spu_ret, wac_i2c->firm_data->size, fsize);
|
||
|
|
||
|
if (spu_ret != fsize) {
|
||
|
input_err(true, &client->dev, "%s: signature verify failed, %ld\n", __func__, spu_ret);
|
||
|
ret = -ENOENT;
|
||
|
goto out;
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
wac_i2c->fw_img = (struct fw_image *)wac_i2c->firm_data->data;
|
||
|
return 0;
|
||
|
|
||
|
out:
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
int w9019_i2c_load_fw(struct wacom_i2c *wac_i2c, u8 fw_update_way)
|
||
|
{
|
||
|
int ret = 0;
|
||
|
struct fw_image *fw_img;
|
||
|
struct i2c_client *client = wac_i2c->client;
|
||
|
|
||
|
switch (fw_update_way) {
|
||
|
case FW_BUILT_IN:
|
||
|
#if WACOM_SEC_FACTORY
|
||
|
case FW_FACTORY_GARAGE:
|
||
|
case FW_FACTORY_UNIT:
|
||
|
#endif
|
||
|
ret = w9019_load_fw_built_in(wac_i2c, fw_update_way);
|
||
|
break;
|
||
|
case FW_IN_SDCARD:
|
||
|
ret = load_fw_sdcard(wac_i2c, fw_update_way, WACOM_PATH_EXTERNAL_FW);
|
||
|
break;
|
||
|
case FW_IN_SDCARD_SIGNED:
|
||
|
ret = load_fw_sdcard(wac_i2c, fw_update_way, WACOM_PATH_EXTERNAL_FW_SIGNED);
|
||
|
break;
|
||
|
case FW_SPU:
|
||
|
case FW_VERIFICATION:
|
||
|
ret = load_fw_sdcard(wac_i2c, fw_update_way, WACOM_PATH_SPU_FW_SIGNED);
|
||
|
break;
|
||
|
default:
|
||
|
input_info(true, &client->dev, "unknown path(%d)\n", fw_update_way);
|
||
|
goto err_load_fw;
|
||
|
}
|
||
|
|
||
|
if (ret < 0)
|
||
|
goto err_load_fw;
|
||
|
|
||
|
fw_img = wac_i2c->fw_img;
|
||
|
|
||
|
/* header check */
|
||
|
if (fw_img->hdr_ver == 1 && fw_img->hdr_len == sizeof(struct fw_image)) {
|
||
|
wac_i2c->fw_data = (u8 *) fw_img->data;
|
||
|
#if WACOM_SEC_FACTORY
|
||
|
if ((fw_update_way == FW_BUILT_IN) ||
|
||
|
(fw_update_way == FW_FACTORY_UNIT) || (fw_update_way == FW_FACTORY_GARAGE)) {
|
||
|
#else
|
||
|
if (fw_update_way == FW_BUILT_IN) {
|
||
|
#endif
|
||
|
wac_i2c->fw_ver_bin = fw_img->fw_ver1;
|
||
|
memcpy(wac_i2c->fw_chksum, fw_img->checksum, 5);
|
||
|
} else if (fw_update_way == FW_SPU || fw_update_way == FW_VERIFICATION) {
|
||
|
wac_i2c->fw_ver_spu = fw_img->fw_ver1;
|
||
|
memcpy(wac_i2c->fw_chksum, fw_img->checksum, 5);
|
||
|
}
|
||
|
} else {
|
||
|
input_err(true, &client->dev, "no hdr\n");
|
||
|
wac_i2c->fw_data = (u8 *) fw_img;
|
||
|
}
|
||
|
|
||
|
return ret;
|
||
|
|
||
|
err_load_fw:
|
||
|
wac_i2c->fw_data = NULL;
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
void w9019_i2c_unload_fw(struct wacom_i2c *wac_i2c)
|
||
|
{
|
||
|
switch (wac_i2c->fw_update_way) {
|
||
|
case FW_BUILT_IN:
|
||
|
#if WACOM_SEC_FACTORY
|
||
|
case FW_FACTORY_GARAGE:
|
||
|
case FW_FACTORY_UNIT:
|
||
|
#endif
|
||
|
release_firmware(wac_i2c->firm_data);
|
||
|
break;
|
||
|
case FW_IN_SDCARD:
|
||
|
case FW_IN_SDCARD_SIGNED:
|
||
|
case FW_SPU:
|
||
|
case FW_VERIFICATION:
|
||
|
release_firmware(wac_i2c->firm_data);
|
||
|
break;
|
||
|
default:
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
wac_i2c->fw_img = NULL;
|
||
|
wac_i2c->fw_update_way = FW_NONE;
|
||
|
wac_i2c->firm_data = NULL;
|
||
|
wac_i2c->fw_data = NULL;
|
||
|
}
|
||
|
|
||
|
int w9019_fw_update_on_hidden_menu(struct wacom_i2c *wac_i2c, u8 fw_update_way)
|
||
|
{
|
||
|
struct i2c_client *client = wac_i2c->client;
|
||
|
u32 fw_ver_ic = wac_i2c->fw_ver_ic;
|
||
|
int ret;
|
||
|
int retry = 3;
|
||
|
|
||
|
input_info(true, &client->dev, "%s: update:%d\n", __func__, fw_update_way);
|
||
|
|
||
|
if (wac_i2c->wacom_fw_ws->active) {
|
||
|
input_info(true, &client->dev, "update is already running. pass\n");
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
mutex_lock(&wac_i2c->update_lock);
|
||
|
w9019_enable_irq(wac_i2c, false);
|
||
|
|
||
|
/* release pen, if it is pressed */
|
||
|
if (wac_i2c->pen_pressed || wac_i2c->side_pressed || wac_i2c->pen_prox)
|
||
|
w9019_forced_release(wac_i2c);
|
||
|
|
||
|
if (wac_i2c->is_tsp_block)
|
||
|
w9019_forced_release_fullscan(wac_i2c);
|
||
|
|
||
|
ret = w9019_i2c_load_fw(wac_i2c, fw_update_way);
|
||
|
if (ret < 0) {
|
||
|
input_info(true, &client->dev, "failed to load fw data\n");
|
||
|
wac_i2c->update_status = FW_UPDATE_FAIL;
|
||
|
goto err_update_load_fw;
|
||
|
}
|
||
|
wac_i2c->fw_update_way = fw_update_way;
|
||
|
|
||
|
/* firmware info */
|
||
|
if (fw_update_way == FW_SPU || fw_update_way == FW_VERIFICATION)
|
||
|
input_info(true, &client->dev, "wacom ic fw ver : 0x%x, new fw ver : 0x%x\n",
|
||
|
wac_i2c->fw_ver_ic, wac_i2c->fw_ver_spu);
|
||
|
else
|
||
|
input_info(true, &client->dev, "wacom ic fw ver : 0x%x, new fw ver : 0x%x\n",
|
||
|
wac_i2c->fw_ver_ic, wac_i2c->fw_ver_bin);
|
||
|
|
||
|
// have to check it later
|
||
|
if (wac_i2c->fw_update_way == FW_BUILT_IN && wac_i2c->pdata->bringup == 1) {
|
||
|
input_info(true, &client->dev, "bringup #1. do not update\n");
|
||
|
wac_i2c->update_status = FW_UPDATE_FAIL;
|
||
|
goto out_update_fw;
|
||
|
}
|
||
|
|
||
|
/* If FFU firmware version is lower than IC's version, do not run update routine */
|
||
|
if (fw_update_way == FW_SPU && fw_ver_ic >= wac_i2c->fw_ver_spu) {
|
||
|
input_info(true, &client->dev, "FFU. update is skipped\n");
|
||
|
wac_i2c->update_status = FW_UPDATE_PASS;
|
||
|
|
||
|
w9019_i2c_unload_fw(wac_i2c);
|
||
|
w9019_enable_irq(wac_i2c, true);
|
||
|
mutex_unlock(&wac_i2c->update_lock);
|
||
|
return 0;
|
||
|
} else if (fw_update_way == FW_VERIFICATION) {
|
||
|
input_info(true, &client->dev, "SPU verified. update is skipped\n");
|
||
|
wac_i2c->update_status = FW_UPDATE_PASS;
|
||
|
|
||
|
w9019_i2c_unload_fw(wac_i2c);
|
||
|
w9019_enable_irq(wac_i2c, true);
|
||
|
mutex_unlock(&wac_i2c->update_lock);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
wac_i2c->update_status = FW_UPDATE_RUNNING;
|
||
|
|
||
|
while (retry--) {
|
||
|
ret = w9019_i2c_flash(wac_i2c);
|
||
|
if (ret) {
|
||
|
input_info(true, &client->dev, "failed to flash fw(%d) %d\n", ret, retry);
|
||
|
continue;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
retry = 3;
|
||
|
while (retry--) {
|
||
|
ret = w9019_i2c_query(wac_i2c);
|
||
|
if (ret < 0) {
|
||
|
input_info(true, &client->dev, "%s : failed to query to IC(%d) & reset, %d\n",
|
||
|
__func__, ret, retry);
|
||
|
w9019_compulsory_flash_mode(wac_i2c, false);
|
||
|
w9019_reset_hw(wac_i2c);
|
||
|
continue;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if (ret) {
|
||
|
wac_i2c->update_status = FW_UPDATE_FAIL;
|
||
|
wac_i2c->fw_ver_ic = 0;
|
||
|
goto out_update_fw;
|
||
|
}
|
||
|
|
||
|
wac_i2c->update_status = FW_UPDATE_PASS;
|
||
|
|
||
|
#if WACOM_SEC_FACTORY
|
||
|
ret = w9019_check_ub(wac_i2c->client);
|
||
|
if (ret < 0) {
|
||
|
w9019_i2c_set_survey_mode(wac_i2c, EPEN_SURVEY_MODE_GARAGE_AOP);
|
||
|
input_info(true, &client->dev, "Change mode for garage scan\n");
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
w9019_i2c_unload_fw(wac_i2c);
|
||
|
w9019_enable_irq(wac_i2c, true);
|
||
|
mutex_unlock(&wac_i2c->update_lock);
|
||
|
|
||
|
return 0;
|
||
|
|
||
|
out_update_fw:
|
||
|
w9019_i2c_unload_fw(wac_i2c);
|
||
|
err_update_load_fw:
|
||
|
w9019_enable_irq(wac_i2c, true);
|
||
|
mutex_unlock(&wac_i2c->update_lock);
|
||
|
|
||
|
#if WACOM_SEC_FACTORY
|
||
|
ret = w9019_check_ub(wac_i2c->client);
|
||
|
if (ret < 0) {
|
||
|
w9019_i2c_set_survey_mode(wac_i2c, EPEN_SURVEY_MODE_GARAGE_AOP);
|
||
|
input_info(true, &client->dev, "Change mode for garage scan\n");
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
int w9019_fw_update_on_probe(struct wacom_i2c *wac_i2c)
|
||
|
{
|
||
|
struct i2c_client *client = wac_i2c->client;
|
||
|
u32 fw_ver_ic = wac_i2c->fw_ver_ic;
|
||
|
int ret;
|
||
|
int retry = 3;
|
||
|
bool bforced = false;
|
||
|
|
||
|
input_info(true, &client->dev, "%s\n", __func__);
|
||
|
|
||
|
if (wac_i2c->pdata->bringup == 1) {
|
||
|
input_info(true, &client->dev, "bringup #1. do not update\n");
|
||
|
wac_i2c->fw_ver_bin = 0;
|
||
|
goto skip_update_fw;
|
||
|
}
|
||
|
|
||
|
ret = w9019_i2c_load_fw(wac_i2c, FW_BUILT_IN);
|
||
|
if (ret < 0) {
|
||
|
input_info(true, &client->dev, "failed to load fw data\n");
|
||
|
goto err_update_load_fw;
|
||
|
}
|
||
|
|
||
|
if (wac_i2c->query_status && (wac_i2c->pdata->ic_type != MPU_W9019)) {
|
||
|
input_err(true, &client->dev, "%s: MPU ver is not matched. unload driver. ic:%02X\n",
|
||
|
__func__, wac_i2c->pdata->ic_type);
|
||
|
goto err_update_fw;
|
||
|
}
|
||
|
|
||
|
if (wac_i2c->pdata->bringup == 2) {
|
||
|
input_info(true, &client->dev, "bringup #2. do not update\n");
|
||
|
goto out_update_fw;
|
||
|
}
|
||
|
|
||
|
wac_i2c->fw_update_way = FW_BUILT_IN;
|
||
|
|
||
|
/* firmware info */
|
||
|
input_info(true, &client->dev, "wacom ic fw ver : 0x%x, new fw ver : 0x%x\n",
|
||
|
wac_i2c->fw_ver_ic, wac_i2c->fw_ver_bin);
|
||
|
|
||
|
if (wac_i2c->pdata->bringup == 3) {
|
||
|
input_info(true, &client->dev, "bringup #3. force update\n");
|
||
|
bforced = true;
|
||
|
}
|
||
|
|
||
|
if (bforced) {
|
||
|
if (fw_ver_ic != wac_i2c->fw_ver_bin)
|
||
|
input_info(true, &client->dev, "not equal, update fw\n");
|
||
|
else
|
||
|
goto out_update_fw;
|
||
|
} else {
|
||
|
if (wac_i2c->fw_ver_ic != wac_i2c->fw_ver_bin) {
|
||
|
input_info(true, &client->dev, "panel firmware is not same with binary firmware, do update\n");
|
||
|
} else if (fw_ver_ic == wac_i2c->fw_ver_bin) {
|
||
|
ret = w9019_checksum(wac_i2c);
|
||
|
if (ret) {
|
||
|
input_info(true, &client->dev, "crc ok, do not update fw\n");
|
||
|
goto out_update_fw;
|
||
|
}
|
||
|
input_info(true, &client->dev, "crc err, do update\n");
|
||
|
|
||
|
} else if (fw_ver_ic > wac_i2c->fw_ver_bin) {
|
||
|
input_info(true, &client->dev, "ic version is high, do not update fw\n");
|
||
|
goto out_update_fw;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//fw_update:
|
||
|
while (retry--) {
|
||
|
ret = w9019_i2c_flash(wac_i2c);
|
||
|
if (ret) {
|
||
|
input_info(true, &client->dev, "failed to flash fw(%d) %d\n", ret, retry);
|
||
|
if (!wac_i2c->bl_mpu_match)
|
||
|
goto err_update_fw;
|
||
|
else
|
||
|
continue;
|
||
|
continue;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
retry = 3;
|
||
|
while (retry--) {
|
||
|
ret = w9019_i2c_query(wac_i2c);
|
||
|
if (ret) {
|
||
|
input_info(true, &client->dev, "%s : failed to query to IC(%d) & reset, %d\n",
|
||
|
__func__, ret, retry);
|
||
|
w9019_compulsory_flash_mode(wac_i2c, false);
|
||
|
w9019_reset_hw(wac_i2c);
|
||
|
continue;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
if (ret) {
|
||
|
wac_i2c->fw_ver_ic = 0;
|
||
|
goto err_update_fw;
|
||
|
}
|
||
|
|
||
|
out_update_fw:
|
||
|
w9019_i2c_unload_fw(wac_i2c);
|
||
|
skip_update_fw:
|
||
|
|
||
|
#if WACOM_SEC_FACTORY
|
||
|
ret = w9019_check_ub(wac_i2c->client);
|
||
|
if (ret < 0) {
|
||
|
w9019_i2c_set_survey_mode(wac_i2c, EPEN_SURVEY_MODE_GARAGE_AOP);
|
||
|
input_info(true, &client->dev, "Change mode for garage scan\n");
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
return 0;
|
||
|
|
||
|
err_update_fw:
|
||
|
w9019_i2c_unload_fw(wac_i2c);
|
||
|
err_update_load_fw:
|
||
|
|
||
|
#if WACOM_SEC_FACTORY
|
||
|
ret = w9019_check_ub(wac_i2c->client);
|
||
|
if (ret < 0) {
|
||
|
w9019_i2c_set_survey_mode(wac_i2c, EPEN_SURVEY_MODE_GARAGE_AOP);
|
||
|
input_info(true, &client->dev, "Change mode for garage scan\n");
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
/* epen_disable_mode
|
||
|
* 0 : wacom ic on
|
||
|
* 1 : wacom ic off
|
||
|
*/
|
||
|
void w9019_disable_mode(struct wacom_i2c *wac_i2c, wacom_disable_mode_t mode)
|
||
|
{
|
||
|
struct i2c_client *client = wac_i2c->client;
|
||
|
|
||
|
if (wac_i2c->epen_blocked == mode){
|
||
|
input_info(true, &client->dev, "%s: duplicate call %d!\n", __func__, mode);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (mode == WACOM_DISABLE) {
|
||
|
input_info(true, &client->dev, "%s: power off\n", __func__);
|
||
|
wac_i2c->epen_blocked = mode;
|
||
|
|
||
|
w9019_enable_irq(wac_i2c, false);
|
||
|
w9019_power(wac_i2c, false);
|
||
|
|
||
|
wac_i2c->survey_mode = EPEN_SURVEY_MODE_NONE;
|
||
|
wac_i2c->function_result &= ~EPEN_EVENT_SURVEY;
|
||
|
|
||
|
#if IS_ENABLED(CONFIG_INPUT_SEC_NOTIFIER)
|
||
|
wac_i2c->tsp_scan_mode = sec_input_notify(&wac_i2c->nb, NOTIFIER_TSP_BLOCKING_RELEASE, NULL);
|
||
|
wac_i2c->is_tsp_block = false;
|
||
|
#endif
|
||
|
w9019_forced_release(wac_i2c);
|
||
|
} else if (mode == WACOM_ENABLE) {
|
||
|
input_info(true, &client->dev, "%s: power on\n", __func__);
|
||
|
wac_i2c->epen_blocked = mode;
|
||
|
|
||
|
w9019_power(wac_i2c, true);
|
||
|
msleep(500);
|
||
|
|
||
|
w9019_enable_irq(wac_i2c, true);
|
||
|
}
|
||
|
input_info(true, &client->dev, "%s: done %d!\n", __func__, mode);
|
||
|
}
|
||
|
|
||
|
#if IS_ENABLED(CONFIG_MUIC_SUPPORT_KEYBOARDDOCK)
|
||
|
static void wac_i2c_kbd_work(struct work_struct *work)
|
||
|
{
|
||
|
struct wacom_i2c *wac_i2c = container_of(work, struct wacom_i2c, kbd_work);
|
||
|
char data;
|
||
|
int ret;
|
||
|
|
||
|
if (wac_i2c->kbd_conn_state == wac_i2c->kbd_cur_conn_state) {
|
||
|
input_info(true, &wac_i2c->client->dev, "%s: already %sconnected\n",
|
||
|
__func__, wac_i2c->kbd_conn_state ? "" : "dis");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
input_info(true, &wac_i2c->client->dev, "%s: %d\n", __func__, wac_i2c->kbd_conn_state);
|
||
|
|
||
|
if (!wac_i2c->power_enable) {
|
||
|
input_err(true, &wac_i2c->client->dev, "%s: powered off\n", __func__);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (wac_i2c->wacom_fw_ws->active) {
|
||
|
input_err(true, &wac_i2c->client->dev, "%s: fw update is running\n", __func__);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
ret = w9019_start_stop_cmd(wac_i2c, WACOM_STOP_CMD);
|
||
|
if (ret < 0) {
|
||
|
input_err(true, &wac_i2c->client->dev, "%s: failed to set stop cmd, %d\n", __func__, ret);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (wac_i2c->kbd_conn_state)
|
||
|
data = COM_SPECIAL_COMPENSATION;
|
||
|
else
|
||
|
data = COM_NORMAL_COMPENSATION;
|
||
|
|
||
|
ret = w9019_i2c_send(wac_i2c, &data, 1);
|
||
|
if (ret < 0) {
|
||
|
input_err(true, &wac_i2c->client->dev, "%s: failed to send table swap cmd %d\n", __func__, ret);
|
||
|
}
|
||
|
msleep(30);
|
||
|
|
||
|
ret = w9019_start_stop_cmd(wac_i2c, WACOM_STOP_AND_START_CMD);
|
||
|
if (ret < 0) {
|
||
|
input_err(true, &wac_i2c->client->dev, "%s: failed to set stop-start cmd, %d\n", __func__, ret);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
input_info(true, &wac_i2c->client->dev, "%s: %s\n", __func__, wac_i2c->kbd_conn_state ? "on" : "off");
|
||
|
}
|
||
|
|
||
|
static int wacom_i2c_keyboard_notification_cb(struct notifier_block *nb,
|
||
|
unsigned long action, void *data)
|
||
|
{
|
||
|
struct wacom_i2c *wac_i2c = container_of(nb, struct wacom_i2c, kbd_nb);
|
||
|
int state = !!action;
|
||
|
|
||
|
if (wac_i2c->kbd_conn_state == state)
|
||
|
goto out;
|
||
|
|
||
|
cancel_work_sync(&wac_i2c->kbd_work);
|
||
|
wac_i2c->kbd_conn_state = state;
|
||
|
input_info(true, &wac_i2c->client->dev, "%s: current %d change to %d\n",
|
||
|
__func__, wac_i2c->kbd_cur_conn_state, state);
|
||
|
schedule_work(&wac_i2c->kbd_work);
|
||
|
|
||
|
out:
|
||
|
return NOTIFY_DONE;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
void w9019_swap_compensation(struct wacom_i2c *wac_i2c, char cmd)
|
||
|
{
|
||
|
char data;
|
||
|
int ret;
|
||
|
|
||
|
input_info(true, &wac_i2c->client->dev, "%s: %d\n", __func__, cmd);
|
||
|
|
||
|
if (!wac_i2c->power_enable) {
|
||
|
input_err(true, &wac_i2c->client->dev,
|
||
|
"%s: powered off\n", __func__);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (wac_i2c->wacom_fw_ws->active) {
|
||
|
input_err(true, &wac_i2c->client->dev,
|
||
|
"%s: fw update is running\n", __func__);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
data = COM_SAMPLERATE_STOP;
|
||
|
ret = w9019_i2c_send_sel(wac_i2c, &data, 1, WACOM_I2C_MODE_NORMAL);
|
||
|
if (ret != 1) {
|
||
|
input_err(true, &wac_i2c->client->dev,
|
||
|
"%s: failed to send stop cmd %d\n",
|
||
|
__func__, ret);
|
||
|
wac_i2c->reset_flag = true;
|
||
|
return;
|
||
|
}
|
||
|
wac_i2c->samplerate_state = false;
|
||
|
msleep(50);
|
||
|
|
||
|
switch (cmd) {
|
||
|
case NOMAL_MODE:
|
||
|
data = COM_NORMAL_COMPENSATION;
|
||
|
break;
|
||
|
case BOOKCOVER_MODE:
|
||
|
data = COM_BOOKCOVER_COMPENSATION;
|
||
|
break;
|
||
|
case KBDCOVER_MODE:
|
||
|
data = COM_KBDCOVER_COMPENSATION;
|
||
|
break;
|
||
|
default:
|
||
|
input_err(true, &wac_i2c->client->dev,
|
||
|
"%s: get wrong cmd %d\n",
|
||
|
__func__, cmd);
|
||
|
}
|
||
|
|
||
|
ret = w9019_i2c_send_sel(wac_i2c, &data, 1, WACOM_I2C_MODE_NORMAL);
|
||
|
if (ret != 1) {
|
||
|
input_err(true, &wac_i2c->client->dev,
|
||
|
"%s: failed to send table swap cmd %d\n",
|
||
|
__func__, ret);
|
||
|
wac_i2c->reset_flag = true;
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
data = COM_SAMPLERATE_STOP;
|
||
|
ret = w9019_i2c_send_sel(wac_i2c, &data, 1, WACOM_I2C_MODE_NORMAL);
|
||
|
if (ret != 1) {
|
||
|
input_err(true, &wac_i2c->client->dev,
|
||
|
"%s: failed to send stop cmd %d\n",
|
||
|
__func__, ret);
|
||
|
wac_i2c->reset_flag = true;
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
msleep(50);
|
||
|
|
||
|
data = COM_SAMPLERATE_START;
|
||
|
ret = w9019_i2c_send_sel(wac_i2c, &data, 1, WACOM_I2C_MODE_NORMAL);
|
||
|
if (ret != 1) {
|
||
|
input_err(true, &wac_i2c->client->dev,
|
||
|
"%s: failed to send start cmd, %d\n",
|
||
|
__func__, ret);
|
||
|
wac_i2c->reset_flag = true;
|
||
|
return;
|
||
|
}
|
||
|
wac_i2c->samplerate_state = true;
|
||
|
|
||
|
input_info(true, &wac_i2c->client->dev, "%s:send cover cmd %x\n",
|
||
|
__func__, cmd);
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
#if IS_ENABLED(CONFIG_USB_TYPEC_MANAGER_NOTIFIER)
|
||
|
static void wacom_i2c_usb_typec_work(struct work_struct *work)
|
||
|
{
|
||
|
struct wacom_i2c *wac_i2c = container_of(work, struct wacom_i2c, typec_work);
|
||
|
char data[5] = { 0 };
|
||
|
int ret;
|
||
|
|
||
|
if (wac_i2c->dp_connect_state == wac_i2c->dp_connect_cmd)
|
||
|
return;
|
||
|
|
||
|
if (!wac_i2c->power_enable) {
|
||
|
input_err(true, &wac_i2c->client->dev, "%s: powered off now\n", __func__);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (wac_i2c->wacom_fw_ws->active) {
|
||
|
input_err(true, &wac_i2c->client->dev, "%s: fw update is running\n", __func__);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
ret = w9019_start_stop_cmd(wac_i2c, WACOM_STOP_CMD);
|
||
|
if (ret < 0) {
|
||
|
input_err(true, &wac_i2c->client->dev, "%s: failed to set stop cmd, %d\n", __func__, ret);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (wac_i2c->dp_connect_cmd)
|
||
|
data[0] = COM_SPECIAL_COMPENSATION;
|
||
|
else
|
||
|
data[0] = COM_NORMAL_COMPENSATION;
|
||
|
|
||
|
ret = w9019_i2c_send(wac_i2c, &data[0], 1);
|
||
|
if (ret < 0) {
|
||
|
input_err(true, &wac_i2c->client->dev, "%s: failed to send table swap cmd %d\n", __func__, ret);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
ret = w9019_start_stop_cmd(wac_i2c, WACOM_STOP_AND_START_CMD);
|
||
|
if (ret < 0) {
|
||
|
input_err(true, &wac_i2c->client->dev, "%s: failed to set stop-start cmd, %d\n", __func__, ret);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
input_info(true, &wac_i2c->client->dev, "%s: %s\n", __func__, wac_i2c->dp_connect_cmd ? "on" : "off");
|
||
|
}
|
||
|
|
||
|
static int wacom_i2c_usb_typec_notification_cb(struct notifier_block *nb,
|
||
|
unsigned long action, void *data)
|
||
|
{
|
||
|
struct wacom_i2c *wac_i2c = container_of(nb, struct wacom_i2c, typec_nb);
|
||
|
PD_NOTI_TYPEDEF usb_typec_info = *(PD_NOTI_TYPEDEF *)data;
|
||
|
|
||
|
if (usb_typec_info.src != PDIC_NOTIFY_DEV_CCIC ||
|
||
|
usb_typec_info.dest != PDIC_NOTIFY_DEV_DP ||
|
||
|
usb_typec_info.id != PDIC_NOTIFY_ID_DP_CONNECT)
|
||
|
goto out;
|
||
|
|
||
|
input_info(true, &wac_i2c->client->dev, "%s: %s (vid:0x%04X pid:0x%04X)\n",
|
||
|
__func__, usb_typec_info.sub1 ? "attached" : "detached",
|
||
|
usb_typec_info.sub2, usb_typec_info.sub3);
|
||
|
|
||
|
switch (usb_typec_info.sub1) {
|
||
|
case PDIC_NOTIFY_ATTACH:
|
||
|
if (usb_typec_info.sub2 != 0x04E8 || usb_typec_info.sub3 != 0xA020)
|
||
|
goto out;
|
||
|
break;
|
||
|
case PDIC_NOTIFY_DETACH:
|
||
|
break;
|
||
|
default:
|
||
|
input_err(true, &wac_i2c->client->dev, "%s: invalid value %d\n", __func__, usb_typec_info.sub1);
|
||
|
goto out;
|
||
|
}
|
||
|
|
||
|
cancel_work_sync(&wac_i2c->typec_work);
|
||
|
wac_i2c->dp_connect_cmd = usb_typec_info.sub1;
|
||
|
schedule_work(&wac_i2c->typec_work);
|
||
|
out:
|
||
|
return 0;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
static void wacom_i2c_nb_register_work(struct work_struct *work)
|
||
|
{
|
||
|
#if IS_ENABLED(CONFIG_USB_TYPEC_MANAGER_NOTIFIER) || IS_ENABLED(CONFIG_MUIC_SUPPORT_KEYBOARDDOCK)
|
||
|
struct wacom_i2c *wac_i2c = container_of(work, struct wacom_i2c,
|
||
|
nb_reg_work.work);
|
||
|
u32 table_swap = wac_i2c->pdata->table_swap;
|
||
|
int ret;
|
||
|
#endif
|
||
|
u32 count = 0;
|
||
|
|
||
|
#if IS_ENABLED(CONFIG_USB_TYPEC_MANAGER_NOTIFIER)
|
||
|
if (table_swap == TABLE_SWAP_DEX_STATION)
|
||
|
INIT_WORK(&wac_i2c->typec_work, wacom_i2c_usb_typec_work);
|
||
|
#endif
|
||
|
#if IS_ENABLED(CONFIG_MUIC_SUPPORT_KEYBOARDDOCK)
|
||
|
if (table_swap == TABLE_SWAP_KBD_COVER)
|
||
|
INIT_WORK(&wac_i2c->kbd_work, wac_i2c_kbd_work);
|
||
|
#endif
|
||
|
|
||
|
do {
|
||
|
#if IS_ENABLED(CONFIG_USB_TYPEC_MANAGER_NOTIFIER)
|
||
|
if (table_swap == TABLE_SWAP_DEX_STATION) {
|
||
|
static bool manager_flag = false;
|
||
|
|
||
|
if (!manager_flag) {
|
||
|
ret = manager_notifier_register(&wac_i2c->typec_nb,
|
||
|
wacom_i2c_usb_typec_notification_cb,
|
||
|
MANAGER_NOTIFY_PDIC_WACOM);
|
||
|
if (!ret) {
|
||
|
manager_flag = true;
|
||
|
input_info(true, &wac_i2c->client->dev,
|
||
|
"%s: typec notifier register success\n",
|
||
|
__func__);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
#if IS_ENABLED(CONFIG_MUIC_SUPPORT_KEYBOARDDOCK)
|
||
|
if (table_swap == TABLE_SWAP_KBD_COVER) {
|
||
|
static bool kbd_flag = false;
|
||
|
|
||
|
if (!kbd_flag) {
|
||
|
ret = keyboard_notifier_register(&wac_i2c->kbd_nb,
|
||
|
wacom_i2c_keyboard_notification_cb,
|
||
|
KEYBOARD_NOTIFY_DEV_WACOM);
|
||
|
if (!ret) {
|
||
|
kbd_flag = true;
|
||
|
input_info(true, &wac_i2c->client->dev,
|
||
|
"%s: kbd notifier register success\n",
|
||
|
__func__);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
count++;
|
||
|
msleep(30);
|
||
|
} while (count < 100);
|
||
|
}
|
||
|
|
||
|
#if IS_ENABLED(CONFIG_INPUT_SEC_NOTIFIER)
|
||
|
static int wacom_notifier_call(struct notifier_block *n, unsigned long data, void *v)
|
||
|
{
|
||
|
struct wacom_i2c *wac_i2c = container_of(n, struct wacom_i2c, nb);
|
||
|
|
||
|
switch (data) {
|
||
|
case NOTIFIER_SECURE_TOUCH_ENABLE:
|
||
|
w9019_disable_mode(wac_i2c, WACOM_DISABLE);
|
||
|
break;
|
||
|
case NOTIFIER_SECURE_TOUCH_DISABLE:
|
||
|
w9019_disable_mode(wac_i2c, WACOM_ENABLE);
|
||
|
break;
|
||
|
default:
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
static int wacom_request_gpio(struct i2c_client *client,
|
||
|
struct wacom_g5_platform_data *pdata)
|
||
|
{
|
||
|
int ret;
|
||
|
|
||
|
ret = devm_gpio_request(&client->dev, pdata->irq_gpio, "wacom_irq");
|
||
|
if (ret) {
|
||
|
input_err(true, &client->dev, "unable to request gpio for irq\n");
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
ret = devm_gpio_request(&client->dev, pdata->fwe_gpio, "wacom_fwe");
|
||
|
if (ret) {
|
||
|
input_err(true, &client->dev, "unable to request gpio for fwe\n");
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int w9019_check_ub(struct i2c_client *client)
|
||
|
{
|
||
|
int lcdtype = 0;
|
||
|
#if IS_ENABLED(CONFIG_EXYNOS_DPU30)
|
||
|
int connected;
|
||
|
#endif
|
||
|
|
||
|
#if IS_ENABLED(CONFIG_DISPLAY_SAMSUNG)
|
||
|
lcdtype = get_lcd_attached("GET");
|
||
|
if (lcdtype == 0xFFFFFF) {
|
||
|
input_info(true, &client->dev, "lcd is not attached\n");
|
||
|
return -ENODEV;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
#if IS_ENABLED(CONFIG_EXYNOS_DPU30)
|
||
|
connected = get_lcd_info("connected");
|
||
|
if (connected < 0) {
|
||
|
input_info(true, &client->dev, "Failed to get connection info\n");
|
||
|
return -ENODEV;
|
||
|
}
|
||
|
|
||
|
if (!connected) {
|
||
|
input_info(true, &client->dev, "lcd is not attached\n");
|
||
|
return -ENODEV;
|
||
|
}
|
||
|
|
||
|
lcdtype = get_lcd_info("id");
|
||
|
if (lcdtype < 0) {
|
||
|
input_info(true, &client->dev, "Failed to get id info\n");
|
||
|
return -ENODEV;
|
||
|
}
|
||
|
#endif
|
||
|
input_info(true, &client->dev, "%s: lcdtype 0x%08x\n", __func__, lcdtype);
|
||
|
|
||
|
return lcdtype;
|
||
|
}
|
||
|
|
||
|
#if IS_ENABLED(CONFIG_OF)
|
||
|
static struct wacom_g5_platform_data *wacom_parse_dt(struct i2c_client *client)
|
||
|
{
|
||
|
struct wacom_g5_platform_data *pdata;
|
||
|
struct device *dev = &client->dev;
|
||
|
struct device_node *np = dev->of_node;
|
||
|
u32 tmp[5] = { 0, };
|
||
|
int ret;
|
||
|
#if IS_ENABLED(CONFIG_DISPLAY_SAMSUNG)
|
||
|
int count = 0;
|
||
|
int ii;
|
||
|
int lcdtype = 0;
|
||
|
#endif
|
||
|
|
||
|
if (!np)
|
||
|
return ERR_PTR(-ENODEV);
|
||
|
|
||
|
pdata = devm_kzalloc(&client->dev, sizeof(*pdata), GFP_KERNEL);
|
||
|
if (!pdata)
|
||
|
return ERR_PTR(-ENOMEM);
|
||
|
|
||
|
pdata->irq_gpio = of_get_named_gpio(np, "wacom,irq-gpio", 0);
|
||
|
if (!gpio_is_valid(pdata->irq_gpio)) {
|
||
|
input_err(true, &client->dev, "failed to get irq-gpio\n");
|
||
|
return ERR_PTR(-EINVAL);
|
||
|
}
|
||
|
|
||
|
pdata->fwe_gpio = of_get_named_gpio(np, "wacom,fwe-gpio", 0);
|
||
|
if (!gpio_is_valid(pdata->fwe_gpio)) {
|
||
|
input_err(true, &client->dev, "failed to get fwe-gpio\n");
|
||
|
return ERR_PTR(-EINVAL);
|
||
|
}
|
||
|
|
||
|
/* get features */
|
||
|
ret = of_property_read_u32(np, "wacom,boot_addr", tmp);
|
||
|
if (ret < 0) {
|
||
|
input_err(true, &client->dev, "failed to read boot address %d\n", ret);
|
||
|
return ERR_PTR(-EINVAL);
|
||
|
}
|
||
|
pdata->boot_addr = tmp[0];
|
||
|
|
||
|
ret = of_property_read_u32_array(np, "wacom,origin", pdata->origin, 2);
|
||
|
if (ret < 0) {
|
||
|
input_err(true, dev, "failed to read origin %d\n", ret);
|
||
|
return ERR_PTR(-EINVAL);
|
||
|
}
|
||
|
|
||
|
pdata->use_dt_coord = of_property_read_bool(np, "wacom,use_dt_coord");
|
||
|
|
||
|
if (pdata->use_dt_coord) {
|
||
|
ret = of_property_read_u32_array(np, "wacom,max_coords", tmp, 2);
|
||
|
if (ret < 0) {
|
||
|
input_err(true, dev, "failed to read max coords %d\n", ret);
|
||
|
} else {
|
||
|
pdata->max_x = tmp[0];
|
||
|
pdata->max_y = tmp[1];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ret = of_property_read_u32(np, "wacom,max_pressure", tmp);
|
||
|
if (ret != -EINVAL) {
|
||
|
if (ret)
|
||
|
input_err(true, dev, "failed to read max pressure %d\n", ret);
|
||
|
else
|
||
|
pdata->max_pressure = tmp[0];
|
||
|
}
|
||
|
|
||
|
ret = of_property_read_u32_array(np, "wacom,max_tilt", tmp, 2);
|
||
|
if (ret != -EINVAL) {
|
||
|
if (ret) {
|
||
|
input_err(true, dev, "failed to read max x tilt %d\n", ret);
|
||
|
} else {
|
||
|
pdata->max_x_tilt = tmp[0];
|
||
|
pdata->max_y_tilt = tmp[1];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ret = of_property_read_u32(np, "wacom,max_height", tmp);
|
||
|
if (ret != -EINVAL) {
|
||
|
if (ret)
|
||
|
input_err(true, dev, "failed to read max height %d\n", ret);
|
||
|
else
|
||
|
pdata->max_height = tmp[0];
|
||
|
}
|
||
|
|
||
|
ret = of_property_read_u32_array(np, "wacom,invert", tmp, 3);
|
||
|
if (ret) {
|
||
|
input_err(true, &client->dev,
|
||
|
"failed to read inverts %d\n", ret);
|
||
|
return ERR_PTR(-EINVAL);
|
||
|
}
|
||
|
pdata->x_invert = tmp[0];
|
||
|
pdata->y_invert = tmp[1];
|
||
|
pdata->xy_switch = tmp[2];
|
||
|
|
||
|
#if IS_ENABLED(CONFIG_DISPLAY_SAMSUNG)
|
||
|
lcdtype = get_lcd_attached("GET");
|
||
|
if (lcdtype == 0xFFFFFF) {
|
||
|
input_err(true, &client->dev, "%s: lcd is not attached\n", __func__);
|
||
|
#if !WACOM_SEC_FACTORY
|
||
|
return ERR_PTR(-ENODEV);
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
input_info(true, &client->dev, "%s: lcdtype 0x%08X\n", __func__, lcdtype);
|
||
|
|
||
|
count = of_property_count_strings(np, "wacom,fw_path");
|
||
|
if (count <= 0) {
|
||
|
pdata->fw_path = NULL;
|
||
|
} else {
|
||
|
u8 lcd_id_num = of_property_count_u32_elems(np, "wacom,select_lcdid");
|
||
|
|
||
|
if ((lcd_id_num != count) || (lcd_id_num <= 0)) {
|
||
|
of_property_read_string_index(np, "wacom,fw_path", 0, &pdata->fw_path);
|
||
|
if (pdata->bringup == 4)
|
||
|
pdata->bringup = 3;
|
||
|
} else {
|
||
|
u32 lcd_id[10];
|
||
|
|
||
|
of_property_read_u32_array(np, "wacom,select_lcdid", lcd_id, lcd_id_num);
|
||
|
|
||
|
for (ii = 0; ii < lcd_id_num; ii++) {
|
||
|
if (lcd_id[ii] == lcdtype) {
|
||
|
of_property_read_string_index(np, "wacom,fw_path", ii, &pdata->fw_path);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if (!pdata->fw_path)
|
||
|
pdata->bringup = 1;
|
||
|
else if (strlen(pdata->fw_path) == 0)
|
||
|
pdata->bringup = 1;
|
||
|
}
|
||
|
}
|
||
|
#else
|
||
|
ret = of_property_read_string(np, "wacom,fw_path", &pdata->fw_path);
|
||
|
if (ret) {
|
||
|
input_err(true, &client->dev, "failed to read fw_path %d\n", ret);
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
ret = of_property_read_u32(np, "wacom,module_ver", &pdata->module_ver);
|
||
|
if (ret) {
|
||
|
input_err(true, &client->dev, "failed to read module_ver %d\n", ret);
|
||
|
/* default setting to open test */
|
||
|
pdata->module_ver = 1;
|
||
|
}
|
||
|
|
||
|
ret = of_property_read_u32(np, "wacom,support_garage_open_test", &pdata->support_garage_open_test);
|
||
|
if (ret) {
|
||
|
input_err(true, &client->dev, "failed to read support_garage_open_test %d\n", ret);
|
||
|
pdata->support_garage_open_test = 0;
|
||
|
}
|
||
|
|
||
|
ret = of_property_read_u32(np, "wacom,table_swap", &pdata->table_swap);
|
||
|
if (ret) {
|
||
|
input_err(true, &client->dev, "failed to read table_swap %d\n", ret);
|
||
|
pdata->table_swap = 0;
|
||
|
}
|
||
|
|
||
|
pdata->avdd = regulator_get(dev, "wacom_avdd");
|
||
|
if (IS_ERR_OR_NULL(pdata->avdd)) {
|
||
|
input_err(true, &client->dev, "%s: Failed to get wacom avdd regulator.\n",
|
||
|
__func__);
|
||
|
return ERR_PTR(-ENODEV);
|
||
|
}
|
||
|
|
||
|
pdata->regulator_boot_on = of_property_read_bool(np, "wacom,regulator_boot_on");
|
||
|
|
||
|
ret = of_property_read_u32(np, "wacom,bringup", &pdata->bringup);
|
||
|
if (ret) {
|
||
|
input_err(true, &client->dev, "failed to read bringup %d\n", ret);
|
||
|
/* default setting to open test */
|
||
|
pdata->bringup = 0;
|
||
|
}
|
||
|
|
||
|
pdata->support_cover_noti = of_property_read_bool(np, "wacom,support_cover_noti");
|
||
|
pdata->support_cover_detection = of_property_read_bool(np, "wacom,support_cover_detection");
|
||
|
|
||
|
input_info(true, &client->dev,
|
||
|
"boot_addr: 0x%X, origin: (%d,%d), max_coords: (%d,%d), "
|
||
|
"max_pressure: %d, max_height: %d, max_tilt: (%d,%d) "
|
||
|
"invert: (%d,%d,%d), fw_path: %s, "
|
||
|
"module_ver:%d, table_swap:%d%s%s, cover_noti:%d,"
|
||
|
"cover_detect:%d\n",
|
||
|
pdata->boot_addr, pdata->origin[0], pdata->origin[1],
|
||
|
pdata->max_x, pdata->max_y, pdata->max_pressure,
|
||
|
pdata->max_height, pdata->max_x_tilt, pdata->max_y_tilt,
|
||
|
pdata->x_invert, pdata->y_invert, pdata->xy_switch,
|
||
|
pdata->fw_path, pdata->module_ver, pdata->table_swap,
|
||
|
pdata->support_garage_open_test ? ", support garage open test" : "",
|
||
|
pdata->regulator_boot_on ? ", boot on" : "",
|
||
|
pdata->support_cover_noti, pdata->support_cover_detection);
|
||
|
|
||
|
return pdata;
|
||
|
}
|
||
|
#else
|
||
|
static struct wacom_g5_platform_data *wacom_parse_dt(struct i2c_client *client)
|
||
|
{
|
||
|
input_err(true, &client->dev, "no platform data specified\n");
|
||
|
return ERR_PTR(-EINVAL);
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
static int w9019_i2c_probe(struct i2c_client *client,
|
||
|
const struct i2c_device_id *id)
|
||
|
{
|
||
|
struct wacom_g5_platform_data *pdata = dev_get_platdata(&client->dev);
|
||
|
struct wacom_i2c *wac_i2c;
|
||
|
struct input_dev *input;
|
||
|
int ret = 0;
|
||
|
|
||
|
client->addr = 0x56;
|
||
|
|
||
|
pr_info("%s: %s: start!\n", SECLOG, __func__);
|
||
|
|
||
|
ret = i2c_check_functionality(client->adapter, I2C_FUNC_I2C);
|
||
|
if (!ret) {
|
||
|
input_err(true, &client->dev, "I2C functionality not supported\n");
|
||
|
return -EIO;
|
||
|
}
|
||
|
|
||
|
wac_i2c = devm_kzalloc(&client->dev, sizeof(*wac_i2c), GFP_KERNEL);
|
||
|
if (!wac_i2c)
|
||
|
return -ENOMEM;
|
||
|
|
||
|
if (!pdata) {
|
||
|
pdata = wacom_parse_dt(client);
|
||
|
if (IS_ERR(pdata)) {
|
||
|
input_err(true, &client->dev, "failed to parse dt\n");
|
||
|
return PTR_ERR(pdata);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ret = wacom_request_gpio(client, pdata);
|
||
|
if (ret) {
|
||
|
input_err(true, &client->dev, "failed to request gpio\n");
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
/* using managed input device */
|
||
|
input = devm_input_allocate_device(&client->dev);
|
||
|
if (!input) {
|
||
|
input_err(true, &client->dev, "failed to allocate input device\n");
|
||
|
return -ENOMEM;
|
||
|
}
|
||
|
|
||
|
/* using 2 slave address. one is normal mode, another is boot mode for
|
||
|
* fw update.
|
||
|
*/
|
||
|
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(5, 4, 0))
|
||
|
wac_i2c->client_boot = i2c_new_dummy_device(client->adapter, pdata->boot_addr);
|
||
|
#else
|
||
|
wac_i2c->client_boot = i2c_new_dummy(client->adapter, pdata->boot_addr);
|
||
|
#endif
|
||
|
|
||
|
if (IS_ERR_OR_NULL(wac_i2c->client_boot)) {
|
||
|
input_err(true, &client->dev, "failed to register sub client[0x%x]\n", pdata->boot_addr);
|
||
|
return -ENOMEM;
|
||
|
}
|
||
|
|
||
|
wac_i2c->client = client;
|
||
|
wac_i2c->pdata = pdata;
|
||
|
wac_i2c->input_dev = input;
|
||
|
wac_i2c->irq = gpio_to_irq(pdata->irq_gpio);
|
||
|
wac_i2c->fw_img = NULL;
|
||
|
wac_i2c->fw_update_way = FW_NONE;
|
||
|
#if IS_ENABLED(CONFIG_INPUT_SEC_NOTIFIER)
|
||
|
wac_i2c->tsp_scan_mode = DISABLE_TSP_SCAN_BLCOK;
|
||
|
#endif
|
||
|
wac_i2c->survey_mode = EPEN_SURVEY_MODE_NONE;
|
||
|
wac_i2c->function_result = EPEN_EVENT_PEN_OUT;
|
||
|
/* Consider about factory, it may be need to as default 1 */
|
||
|
#if WACOM_SEC_FACTORY
|
||
|
wac_i2c->battery_saving_mode = true;
|
||
|
#else
|
||
|
wac_i2c->battery_saving_mode = false;
|
||
|
#endif
|
||
|
wac_i2c->reset_flag = false;
|
||
|
wac_i2c->pm_suspend = false;
|
||
|
wac_i2c->samplerate_state = true;
|
||
|
wac_i2c->update_status = FW_UPDATE_PASS;
|
||
|
|
||
|
/*Set client data */
|
||
|
i2c_set_clientdata(client, wac_i2c);
|
||
|
i2c_set_clientdata(wac_i2c->client_boot, wac_i2c);
|
||
|
|
||
|
init_completion(&wac_i2c->i2c_done);
|
||
|
complete_all(&wac_i2c->i2c_done);
|
||
|
init_completion(&wac_i2c->resume_done);
|
||
|
complete_all(&wac_i2c->resume_done);
|
||
|
|
||
|
/* Power on */
|
||
|
if (gpio_get_value(pdata->fwe_gpio) == 1) {
|
||
|
input_info(true, &client->dev, "%s: fwe gpio is high, change low and reset device\n", __func__);
|
||
|
w9019_compulsory_flash_mode(wac_i2c, false);
|
||
|
w9019_reset_hw(wac_i2c);
|
||
|
msleep(200);
|
||
|
} else {
|
||
|
w9019_compulsory_flash_mode(wac_i2c, false);
|
||
|
w9019_power(wac_i2c, true);
|
||
|
if (!wac_i2c->pdata->regulator_boot_on)
|
||
|
msleep(200);
|
||
|
}
|
||
|
|
||
|
wac_i2c->screen_on = true;
|
||
|
|
||
|
input->name = "sec_e-pen";
|
||
|
|
||
|
w9019_i2c_query(wac_i2c);
|
||
|
|
||
|
/*Initializing for semaphor */
|
||
|
mutex_init(&wac_i2c->i2c_mutex);
|
||
|
mutex_init(&wac_i2c->lock);
|
||
|
mutex_init(&wac_i2c->update_lock);
|
||
|
mutex_init(&wac_i2c->irq_lock);
|
||
|
mutex_init(&wac_i2c->mode_lock);
|
||
|
mutex_init(&wac_i2c->ble_lock);
|
||
|
mutex_init(&wac_i2c->ble_charge_mode_lock);
|
||
|
|
||
|
wac_i2c->wacom_fw_ws = wakeup_source_register(&wac_i2c->client->dev, "wacom");
|
||
|
wac_i2c->wacom_ws = wakeup_source_register(&wac_i2c->client->dev, "wacom_wakelock");
|
||
|
|
||
|
INIT_DELAYED_WORK(&wac_i2c->work_print_info, wacom_print_info_work);
|
||
|
|
||
|
ret = w9019_fw_update_on_probe(wac_i2c);
|
||
|
if (ret)
|
||
|
goto err_register_input_dev;
|
||
|
|
||
|
wacom_i2c_set_input_values(wac_i2c, input);
|
||
|
|
||
|
ret = input_register_device(input);
|
||
|
if (ret) {
|
||
|
input_err(true, &client->dev, "failed to register input device\n");
|
||
|
/* managed input devices need not be explicitly unregistred or freed. */
|
||
|
goto err_register_input_dev;
|
||
|
|
||
|
}
|
||
|
|
||
|
/*Request IRQ */
|
||
|
ret = devm_request_threaded_irq(&client->dev, wac_i2c->irq, NULL, wacom_interrupt,
|
||
|
IRQF_ONESHOT | IRQF_TRIGGER_LOW, "sec_epen_irq", wac_i2c);
|
||
|
if (ret < 0) {
|
||
|
input_err(true, &client->dev, "failed to request irq(%d) - %d\n", wac_i2c->irq, ret);
|
||
|
goto err_request_irq;
|
||
|
}
|
||
|
input_info(true, &client->dev, "init irq %d\n", wac_i2c->irq);
|
||
|
|
||
|
ret = w9019_sec_init(wac_i2c);
|
||
|
if (ret)
|
||
|
goto err_sec_init;
|
||
|
|
||
|
device_init_wakeup(&client->dev, true);
|
||
|
|
||
|
if (wac_i2c->pdata->table_swap) {
|
||
|
INIT_DELAYED_WORK(&wac_i2c->nb_reg_work, wacom_i2c_nb_register_work);
|
||
|
schedule_delayed_work(&wac_i2c->nb_reg_work, msecs_to_jiffies(500));
|
||
|
}
|
||
|
|
||
|
#if IS_ENABLED(CONFIG_INPUT_SEC_NOTIFIER)
|
||
|
sec_input_register_notify(&wac_i2c->nb, wacom_notifier_call, 2);
|
||
|
#endif
|
||
|
|
||
|
input_info(true, &client->dev, "probe done\n");
|
||
|
input_log_fix();
|
||
|
|
||
|
wac_i2c->ble_hist = kzalloc(WACOM_BLE_HISTORY_SIZE, GFP_KERNEL);
|
||
|
if (!wac_i2c->ble_hist)
|
||
|
input_err(true, &client->dev, "failed to history mem\n");
|
||
|
|
||
|
wac_i2c->ble_hist1 = kzalloc(WACOM_BLE_HISTORY1_SIZE, GFP_KERNEL);
|
||
|
if (!wac_i2c->ble_hist1)
|
||
|
input_err(true, &client->dev, "failed to history1 mem\n");
|
||
|
|
||
|
probe_open_test(wac_i2c);
|
||
|
|
||
|
g_wac_i2c_w9019 = wac_i2c;
|
||
|
wac_i2c->probe_done = true;
|
||
|
pdata->enabled = true;
|
||
|
|
||
|
return 0;
|
||
|
|
||
|
err_sec_init:
|
||
|
cancel_delayed_work_sync(&wac_i2c->open_test_dwork);
|
||
|
err_request_irq:
|
||
|
err_register_input_dev:
|
||
|
wakeup_source_unregister(wac_i2c->wacom_fw_ws);
|
||
|
wakeup_source_unregister(wac_i2c->wacom_ws);
|
||
|
|
||
|
mutex_destroy(&wac_i2c->i2c_mutex);
|
||
|
mutex_destroy(&wac_i2c->irq_lock);
|
||
|
mutex_destroy(&wac_i2c->update_lock);
|
||
|
mutex_destroy(&wac_i2c->lock);
|
||
|
mutex_destroy(&wac_i2c->mode_lock);
|
||
|
mutex_destroy(&wac_i2c->ble_lock);
|
||
|
mutex_destroy(&wac_i2c->ble_charge_mode_lock);
|
||
|
|
||
|
w9019_power(wac_i2c, false);
|
||
|
i2c_unregister_device(wac_i2c->client_boot);
|
||
|
regulator_put(pdata->avdd);
|
||
|
|
||
|
if (wac_i2c->input_dev)
|
||
|
input_free_device(wac_i2c->input_dev);
|
||
|
|
||
|
input_err(true, &client->dev, "failed to probe\n");
|
||
|
input_log_fix();
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
#if IS_ENABLED(CONFIG_PM)
|
||
|
static int w9019_i2c_suspend(struct device *dev)
|
||
|
{
|
||
|
struct wacom_i2c *wac_i2c = dev_get_drvdata(dev);
|
||
|
int ret;
|
||
|
|
||
|
if (wac_i2c->i2c_done.done == 0) {
|
||
|
/* completion.done == 0 :: initialized
|
||
|
* completion.done > 0 :: completeted
|
||
|
*/
|
||
|
ret = wait_for_completion_interruptible_timeout(&wac_i2c->i2c_done, msecs_to_jiffies(500));
|
||
|
if (ret <= 0)
|
||
|
input_err(true, &wac_i2c->client->dev, "%s: completion expired, %d\n", __func__, ret);
|
||
|
}
|
||
|
|
||
|
wac_i2c->pm_suspend = true;
|
||
|
reinit_completion(&wac_i2c->resume_done);
|
||
|
|
||
|
#ifndef USE_OPEN_CLOSE
|
||
|
if (wac_i2c->input_dev->users)
|
||
|
w9019_sleep_sequence(wac_i2c);
|
||
|
#endif
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int w9019_i2c_resume(struct device *dev)
|
||
|
{
|
||
|
struct wacom_i2c *wac_i2c = dev_get_drvdata(dev);
|
||
|
|
||
|
wac_i2c->pm_suspend = false;
|
||
|
complete_all(&wac_i2c->resume_done);
|
||
|
#ifndef USE_OPEN_CLOSE
|
||
|
if (wac_i2c->input_dev->users)
|
||
|
w9019_wakeup_sequence(wac_i2c);
|
||
|
#endif
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static SIMPLE_DEV_PM_OPS(w9019_pm_ops, w9019_i2c_suspend, w9019_i2c_resume);
|
||
|
#endif
|
||
|
|
||
|
static void w9019_i2c_shutdown(struct i2c_client *client)
|
||
|
{
|
||
|
struct wacom_i2c *wac_i2c = i2c_get_clientdata(client);
|
||
|
|
||
|
if (!wac_i2c)
|
||
|
return;
|
||
|
|
||
|
g_wac_i2c_w9019 = NULL;
|
||
|
wac_i2c->probe_done = false;
|
||
|
|
||
|
input_info(true, &wac_i2c->client->dev, "%s called!\n", __func__);
|
||
|
|
||
|
#if IS_ENABLED(CONFIG_INPUT_SEC_NOTIFIER)
|
||
|
sec_input_unregister_notify(&wac_i2c->nb);
|
||
|
#endif
|
||
|
|
||
|
if (wac_i2c->pdata->table_swap) {
|
||
|
#if IS_ENABLED(CONFIG_MUIC_SUPPORT_KEYBOARDDOCK)
|
||
|
if (wac_i2c->pdata->table_swap == TABLE_SWAP_KBD_COVER)
|
||
|
keyboard_notifier_unregister(&wac_i2c->kbd_nb);
|
||
|
#endif
|
||
|
|
||
|
#if IS_ENABLED(CONFIG_USB_TYPEC_MANAGER_NOTIFIER)
|
||
|
if (wac_i2c->pdata->table_swap == TABLE_SWAP_DEX_STATION)
|
||
|
manager_notifier_unregister(&wac_i2c->typec_nb);
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
cancel_delayed_work_sync(&wac_i2c->open_test_dwork);
|
||
|
cancel_delayed_work_sync(&wac_i2c->work_print_info);
|
||
|
|
||
|
w9019_enable_irq(wac_i2c, false);
|
||
|
|
||
|
w9019_power(wac_i2c, false);
|
||
|
|
||
|
input_info(true, &wac_i2c->client->dev, "%s Done\n", __func__);
|
||
|
}
|
||
|
|
||
|
static int w9019_i2c_remove(struct i2c_client *client)
|
||
|
{
|
||
|
struct wacom_i2c *wac_i2c = i2c_get_clientdata(client);
|
||
|
|
||
|
g_wac_i2c_w9019 = NULL;
|
||
|
wac_i2c->probe_done = false;
|
||
|
|
||
|
input_info(true, &wac_i2c->client->dev, "%s called!\n", __func__);
|
||
|
|
||
|
if (wac_i2c->pdata->table_swap) {
|
||
|
#if IS_ENABLED(CONFIG_MUIC_SUPPORT_KEYBOARDDOCK)
|
||
|
if (wac_i2c->pdata->table_swap == TABLE_SWAP_KBD_COVER)
|
||
|
keyboard_notifier_unregister(&wac_i2c->kbd_nb);
|
||
|
#endif
|
||
|
|
||
|
#if IS_ENABLED(CONFIG_USB_TYPEC_MANAGER_NOTIFIER)
|
||
|
if (wac_i2c->pdata->table_swap == TABLE_SWAP_DEX_STATION)
|
||
|
manager_notifier_unregister(&wac_i2c->typec_nb);
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
cancel_delayed_work_sync(&wac_i2c->open_test_dwork);
|
||
|
cancel_delayed_work_sync(&wac_i2c->work_print_info);
|
||
|
|
||
|
device_init_wakeup(&client->dev, false);
|
||
|
|
||
|
w9019_enable_irq(wac_i2c, false);
|
||
|
|
||
|
w9019_power(wac_i2c, false);
|
||
|
|
||
|
wakeup_source_unregister(wac_i2c->wacom_fw_ws);
|
||
|
wakeup_source_unregister(wac_i2c->wacom_ws);
|
||
|
|
||
|
mutex_destroy(&wac_i2c->i2c_mutex);
|
||
|
mutex_destroy(&wac_i2c->irq_lock);
|
||
|
mutex_destroy(&wac_i2c->update_lock);
|
||
|
mutex_destroy(&wac_i2c->lock);
|
||
|
mutex_destroy(&wac_i2c->mode_lock);
|
||
|
mutex_destroy(&wac_i2c->ble_lock);
|
||
|
mutex_destroy(&wac_i2c->ble_charge_mode_lock);
|
||
|
|
||
|
w9019_sec_remove(wac_i2c);
|
||
|
|
||
|
i2c_unregister_device(wac_i2c->client_boot);
|
||
|
regulator_put(wac_i2c->pdata->avdd);
|
||
|
|
||
|
input_info(true, &wac_i2c->client->dev, "%s Done\n", __func__);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static const struct i2c_device_id w9019_i2c_id[] = {
|
||
|
{"wacom_w9019", 0},
|
||
|
{},
|
||
|
};
|
||
|
|
||
|
MODULE_DEVICE_TABLE(i2c, wacom_i2c_id);
|
||
|
|
||
|
#if IS_ENABLED(CONFIG_OF)
|
||
|
static const struct of_device_id w9019_dt_ids[] = {
|
||
|
{.compatible = "wacom,w9019"},
|
||
|
{}
|
||
|
};
|
||
|
#endif
|
||
|
|
||
|
static struct i2c_driver w9019_i2c_driver = {
|
||
|
.driver = {
|
||
|
.owner = THIS_MODULE,
|
||
|
.name = "wacom_w9019",
|
||
|
#if IS_ENABLED(CONFIG_PM)
|
||
|
.pm = &w9019_pm_ops,
|
||
|
#endif
|
||
|
#if IS_ENABLED(CONFIG_OF)
|
||
|
.of_match_table = of_match_ptr(w9019_dt_ids),
|
||
|
#endif
|
||
|
},
|
||
|
.probe = w9019_i2c_probe,
|
||
|
.remove = w9019_i2c_remove,
|
||
|
.shutdown = w9019_i2c_shutdown,
|
||
|
.id_table = w9019_i2c_id,
|
||
|
};
|
||
|
|
||
|
static int __init w9019_i2c_init(void)
|
||
|
{
|
||
|
int ret = 0;
|
||
|
|
||
|
pr_info("%s: %s start\n", SECLOG, __func__);
|
||
|
|
||
|
ret = i2c_add_driver(&w9019_i2c_driver);
|
||
|
if (ret)
|
||
|
pr_err("%s: %s: failed to add i2c driver\n", SECLOG, __func__);
|
||
|
|
||
|
pr_info("%s: %s end\n", SECLOG, __func__);
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
static void __exit w9019_i2c_exit(void)
|
||
|
{
|
||
|
i2c_del_driver(&w9019_i2c_driver);
|
||
|
}
|
||
|
|
||
|
module_init(w9019_i2c_init);
|
||
|
module_exit(w9019_i2c_exit);
|
||
|
|
||
|
MODULE_AUTHOR("Samsung");
|
||
|
MODULE_DESCRIPTION("Driver for Wacom Digitizer Controller");
|
||
|
MODULE_LICENSE("GPL");
|