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.
kernel_samsung_sm7125/drivers/input/w9019/wacom_i2c.c

3083 lines
85 KiB

/*
* 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");