mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	 f051ae4f6c
			
		
	
	
		f051ae4f6c
		
	
	
	
	
		
			
			gcc -Warray-bounds warns about a serious bug in
cyapa_pip_retrieve_data_structure:
drivers/input/mouse/cyapa_gen6.c: In function 'cyapa_pip_retrieve_data_structure.constprop':
include/linux/unaligned/access_ok.h:40:17: warning: array subscript -1 is outside array bounds of 'struct retrieve_data_struct_cmd[1]' [-Warray-bounds]
   40 |  *((__le16 *)p) = cpu_to_le16(val);
drivers/input/mouse/cyapa_gen6.c:569:13: note: while referencing 'cmd'
  569 |  } __packed cmd;
      |             ^~~
Apparently the '-2' was added to the pointer instead of the value,
writing garbage into the stack next to this variable.
Fixes: c2c06c41f7 ("Input: cyapa - add gen6 device module support")
Signed-off-by: Arnd Bergmann <arnd@arndb.de>
Link: https://lore.kernel.org/r/20201026161332.3708389-1-arnd@kernel.org
Cc: stable@vger.kernel.org
Signed-off-by: Dmitry Torokhov <dmitry.torokhov@gmail.com>
		
	
			
		
			
				
	
	
		
			747 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			747 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Cypress APA trackpad with I2C interface
 | |
|  *
 | |
|  * Author: Dudley Du <dudl@cypress.com>
 | |
|  *
 | |
|  * Copyright (C) 2015 Cypress Semiconductor, Inc.
 | |
|  *
 | |
|  * This file is subject to the terms and conditions of the GNU General Public
 | |
|  * License.  See the file COPYING in the main directory of this archive for
 | |
|  * more details.
 | |
|  */
 | |
| 
 | |
| #include <linux/delay.h>
 | |
| #include <linux/i2c.h>
 | |
| #include <linux/input.h>
 | |
| #include <linux/input/mt.h>
 | |
| #include <linux/mutex.h>
 | |
| #include <linux/completion.h>
 | |
| #include <linux/slab.h>
 | |
| #include <asm/unaligned.h>
 | |
| #include <linux/crc-itu-t.h>
 | |
| #include "cyapa.h"
 | |
| 
 | |
| 
 | |
| #define GEN6_ENABLE_CMD_IRQ	0x41
 | |
| #define GEN6_DISABLE_CMD_IRQ	0x42
 | |
| #define GEN6_ENABLE_DEV_IRQ	0x43
 | |
| #define GEN6_DISABLE_DEV_IRQ	0x44
 | |
| 
 | |
| #define GEN6_POWER_MODE_ACTIVE		0x01
 | |
| #define GEN6_POWER_MODE_LP_MODE1	0x02
 | |
| #define GEN6_POWER_MODE_LP_MODE2	0x03
 | |
| #define GEN6_POWER_MODE_BTN_ONLY	0x04
 | |
| 
 | |
| #define GEN6_SET_POWER_MODE_INTERVAL	0x47
 | |
| #define GEN6_GET_POWER_MODE_INTERVAL	0x48
 | |
| 
 | |
| #define GEN6_MAX_RX_NUM 14
 | |
| #define GEN6_RETRIEVE_DATA_ID_RX_ATTENURATOR_IDAC	0x00
 | |
| #define GEN6_RETRIEVE_DATA_ID_ATTENURATOR_TRIM		0x12
 | |
| 
 | |
| 
 | |
| struct pip_app_cmd_head {
 | |
| 	__le16 addr;
 | |
| 	__le16 length;
 | |
| 	u8 report_id;
 | |
| 	u8 resv;  /* Reserved, must be 0 */
 | |
| 	u8 cmd_code;  /* bit7: resv, set to 0; bit6~0: command code.*/
 | |
| } __packed;
 | |
| 
 | |
| struct pip_app_resp_head {
 | |
| 	__le16 length;
 | |
| 	u8 report_id;
 | |
| 	u8 resv;  /* Reserved, must be 0 */
 | |
| 	u8 cmd_code;  /* bit7: TGL; bit6~0: command code.*/
 | |
| 	/*
 | |
| 	 * The value of data_status can be the first byte of data or
 | |
| 	 * the command status or the unsupported command code depending on the
 | |
| 	 * requested command code.
 | |
| 	*/
 | |
| 	u8 data_status;
 | |
| } __packed;
 | |
| 
 | |
| struct pip_fixed_info {
 | |
| 	u8 silicon_id_high;
 | |
| 	u8 silicon_id_low;
 | |
| 	u8 family_id;
 | |
| };
 | |
| 
 | |
| static u8 pip_get_bl_info[] = {
 | |
| 	0x04, 0x00, 0x0B, 0x00, 0x40, 0x00, 0x01, 0x38,
 | |
| 	0x00, 0x00, 0x70, 0x9E, 0x17
 | |
| };
 | |
| 
 | |
| static bool cyapa_sort_pip_hid_descriptor_data(struct cyapa *cyapa,
 | |
| 		u8 *buf, int len)
 | |
| {
 | |
| 	if (len != PIP_HID_DESCRIPTOR_SIZE)
 | |
| 		return false;
 | |
| 
 | |
| 	if (buf[PIP_RESP_REPORT_ID_OFFSET] == PIP_HID_APP_REPORT_ID ||
 | |
| 		buf[PIP_RESP_REPORT_ID_OFFSET] == PIP_HID_BL_REPORT_ID)
 | |
| 		return true;
 | |
| 
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| static int cyapa_get_pip_fixed_info(struct cyapa *cyapa,
 | |
| 		struct pip_fixed_info *pip_info, bool is_bootloader)
 | |
| {
 | |
| 	u8 resp_data[PIP_READ_SYS_INFO_RESP_LENGTH];
 | |
| 	int resp_len;
 | |
| 	u16 product_family;
 | |
| 	int error;
 | |
| 
 | |
| 	if (is_bootloader) {
 | |
| 		/* Read Bootloader Information to determine Gen5 or Gen6. */
 | |
| 		resp_len = sizeof(resp_data);
 | |
| 		error = cyapa_i2c_pip_cmd_irq_sync(cyapa,
 | |
| 				pip_get_bl_info, sizeof(pip_get_bl_info),
 | |
| 				resp_data, &resp_len,
 | |
| 				2000, cyapa_sort_tsg_pip_bl_resp_data,
 | |
| 				false);
 | |
| 		if (error || resp_len < PIP_BL_GET_INFO_RESP_LENGTH)
 | |
| 			return error ? error : -EIO;
 | |
| 
 | |
| 		pip_info->family_id = resp_data[8];
 | |
| 		pip_info->silicon_id_low = resp_data[10];
 | |
| 		pip_info->silicon_id_high = resp_data[11];
 | |
| 
 | |
| 		return 0;
 | |
| 	}
 | |
| 
 | |
| 	/* Get App System Information to determine Gen5 or Gen6. */
 | |
| 	resp_len = sizeof(resp_data);
 | |
| 	error = cyapa_i2c_pip_cmd_irq_sync(cyapa,
 | |
| 			pip_read_sys_info, PIP_READ_SYS_INFO_CMD_LENGTH,
 | |
| 			resp_data, &resp_len,
 | |
| 			2000, cyapa_pip_sort_system_info_data, false);
 | |
| 	if (error || resp_len < PIP_READ_SYS_INFO_RESP_LENGTH)
 | |
| 		return error ? error : -EIO;
 | |
| 
 | |
| 	product_family = get_unaligned_le16(&resp_data[7]);
 | |
| 	if ((product_family & PIP_PRODUCT_FAMILY_MASK) !=
 | |
| 		PIP_PRODUCT_FAMILY_TRACKPAD)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	pip_info->family_id = resp_data[19];
 | |
| 	pip_info->silicon_id_low = resp_data[21];
 | |
| 	pip_info->silicon_id_high = resp_data[22];
 | |
| 
 | |
| 	return 0;
 | |
| 
 | |
| }
 | |
| 
 | |
| int cyapa_pip_state_parse(struct cyapa *cyapa, u8 *reg_data, int len)
 | |
| {
 | |
| 	u8 cmd[] = { 0x01, 0x00};
 | |
| 	struct pip_fixed_info pip_info;
 | |
| 	u8 resp_data[PIP_HID_DESCRIPTOR_SIZE];
 | |
| 	int resp_len;
 | |
| 	bool is_bootloader;
 | |
| 	int error;
 | |
| 
 | |
| 	cyapa->state = CYAPA_STATE_NO_DEVICE;
 | |
| 
 | |
| 	/* Try to wake from it deep sleep state if it is. */
 | |
| 	cyapa_pip_deep_sleep(cyapa, PIP_DEEP_SLEEP_STATE_ON);
 | |
| 
 | |
| 	/* Empty the buffer queue to get fresh data with later commands. */
 | |
| 	cyapa_empty_pip_output_data(cyapa, NULL, NULL, NULL);
 | |
| 
 | |
| 	/*
 | |
| 	 * Read description info from trackpad device to determine running in
 | |
| 	 * APP mode or Bootloader mode.
 | |
| 	 */
 | |
| 	resp_len = PIP_HID_DESCRIPTOR_SIZE;
 | |
| 	error = cyapa_i2c_pip_cmd_irq_sync(cyapa,
 | |
| 			cmd, sizeof(cmd),
 | |
| 			resp_data, &resp_len,
 | |
| 			300,
 | |
| 			cyapa_sort_pip_hid_descriptor_data,
 | |
| 			false);
 | |
| 	if (error)
 | |
| 		return error;
 | |
| 
 | |
| 	if (resp_data[PIP_RESP_REPORT_ID_OFFSET] == PIP_HID_BL_REPORT_ID)
 | |
| 		is_bootloader = true;
 | |
| 	else if (resp_data[PIP_RESP_REPORT_ID_OFFSET] == PIP_HID_APP_REPORT_ID)
 | |
| 		is_bootloader = false;
 | |
| 	else
 | |
| 		return -EAGAIN;
 | |
| 
 | |
| 	/* Get PIP fixed information to determine Gen5 or Gen6. */
 | |
| 	memset(&pip_info, 0, sizeof(struct pip_fixed_info));
 | |
| 	error = cyapa_get_pip_fixed_info(cyapa, &pip_info, is_bootloader);
 | |
| 	if (error)
 | |
| 		return error;
 | |
| 
 | |
| 	if (pip_info.family_id == 0x9B && pip_info.silicon_id_high == 0x0B) {
 | |
| 		cyapa->gen = CYAPA_GEN6;
 | |
| 		cyapa->state = is_bootloader ? CYAPA_STATE_GEN6_BL
 | |
| 					     : CYAPA_STATE_GEN6_APP;
 | |
| 	} else if (pip_info.family_id == 0x91 &&
 | |
| 		   pip_info.silicon_id_high == 0x02) {
 | |
| 		cyapa->gen = CYAPA_GEN5;
 | |
| 		cyapa->state = is_bootloader ? CYAPA_STATE_GEN5_BL
 | |
| 					     : CYAPA_STATE_GEN5_APP;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int cyapa_gen6_read_sys_info(struct cyapa *cyapa)
 | |
| {
 | |
| 	u8 resp_data[PIP_READ_SYS_INFO_RESP_LENGTH];
 | |
| 	int resp_len;
 | |
| 	u16 product_family;
 | |
| 	u8 rotat_align;
 | |
| 	int error;
 | |
| 
 | |
| 	/* Get App System Information to determine Gen5 or Gen6. */
 | |
| 	resp_len = sizeof(resp_data);
 | |
| 	error = cyapa_i2c_pip_cmd_irq_sync(cyapa,
 | |
| 			pip_read_sys_info, PIP_READ_SYS_INFO_CMD_LENGTH,
 | |
| 			resp_data, &resp_len,
 | |
| 			2000, cyapa_pip_sort_system_info_data, false);
 | |
| 	if (error || resp_len < sizeof(resp_data))
 | |
| 		return error ? error : -EIO;
 | |
| 
 | |
| 	product_family = get_unaligned_le16(&resp_data[7]);
 | |
| 	if ((product_family & PIP_PRODUCT_FAMILY_MASK) !=
 | |
| 		PIP_PRODUCT_FAMILY_TRACKPAD)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	cyapa->platform_ver = (resp_data[67] >> PIP_BL_PLATFORM_VER_SHIFT) &
 | |
| 			      PIP_BL_PLATFORM_VER_MASK;
 | |
| 	cyapa->fw_maj_ver = resp_data[9];
 | |
| 	cyapa->fw_min_ver = resp_data[10];
 | |
| 
 | |
| 	cyapa->electrodes_x = resp_data[33];
 | |
| 	cyapa->electrodes_y = resp_data[34];
 | |
| 
 | |
| 	cyapa->physical_size_x =  get_unaligned_le16(&resp_data[35]) / 100;
 | |
| 	cyapa->physical_size_y = get_unaligned_le16(&resp_data[37]) / 100;
 | |
| 
 | |
| 	cyapa->max_abs_x = get_unaligned_le16(&resp_data[39]);
 | |
| 	cyapa->max_abs_y = get_unaligned_le16(&resp_data[41]);
 | |
| 
 | |
| 	cyapa->max_z = get_unaligned_le16(&resp_data[43]);
 | |
| 
 | |
| 	cyapa->x_origin = resp_data[45] & 0x01;
 | |
| 	cyapa->y_origin = resp_data[46] & 0x01;
 | |
| 
 | |
| 	cyapa->btn_capability = (resp_data[70] << 3) & CAPABILITY_BTN_MASK;
 | |
| 
 | |
| 	memcpy(&cyapa->product_id[0], &resp_data[51], 5);
 | |
| 	cyapa->product_id[5] = '-';
 | |
| 	memcpy(&cyapa->product_id[6], &resp_data[56], 6);
 | |
| 	cyapa->product_id[12] = '-';
 | |
| 	memcpy(&cyapa->product_id[13], &resp_data[62], 2);
 | |
| 	cyapa->product_id[15] = '\0';
 | |
| 
 | |
| 	/* Get the number of Rx electrodes. */
 | |
| 	rotat_align = resp_data[68];
 | |
| 	cyapa->electrodes_rx =
 | |
| 		rotat_align ? cyapa->electrodes_y : cyapa->electrodes_x;
 | |
| 	cyapa->aligned_electrodes_rx = (cyapa->electrodes_rx + 3) & ~3u;
 | |
| 
 | |
| 	if (!cyapa->electrodes_x || !cyapa->electrodes_y ||
 | |
| 		!cyapa->physical_size_x || !cyapa->physical_size_y ||
 | |
| 		!cyapa->max_abs_x || !cyapa->max_abs_y || !cyapa->max_z)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int cyapa_gen6_bl_read_app_info(struct cyapa *cyapa)
 | |
| {
 | |
| 	u8 resp_data[PIP_BL_APP_INFO_RESP_LENGTH];
 | |
| 	int resp_len;
 | |
| 	int error;
 | |
| 
 | |
| 	resp_len = sizeof(resp_data);
 | |
| 	error = cyapa_i2c_pip_cmd_irq_sync(cyapa,
 | |
| 			pip_bl_read_app_info, PIP_BL_READ_APP_INFO_CMD_LENGTH,
 | |
| 			resp_data, &resp_len,
 | |
| 			500, cyapa_sort_tsg_pip_bl_resp_data, false);
 | |
| 	if (error || resp_len < PIP_BL_APP_INFO_RESP_LENGTH ||
 | |
| 		!PIP_CMD_COMPLETE_SUCCESS(resp_data))
 | |
| 		return error ? error : -EIO;
 | |
| 
 | |
| 	cyapa->fw_maj_ver = resp_data[8];
 | |
| 	cyapa->fw_min_ver = resp_data[9];
 | |
| 
 | |
| 	cyapa->platform_ver = (resp_data[12] >> PIP_BL_PLATFORM_VER_SHIFT) &
 | |
| 			      PIP_BL_PLATFORM_VER_MASK;
 | |
| 
 | |
| 	memcpy(&cyapa->product_id[0], &resp_data[13], 5);
 | |
| 	cyapa->product_id[5] = '-';
 | |
| 	memcpy(&cyapa->product_id[6], &resp_data[18], 6);
 | |
| 	cyapa->product_id[12] = '-';
 | |
| 	memcpy(&cyapa->product_id[13], &resp_data[24], 2);
 | |
| 	cyapa->product_id[15] = '\0';
 | |
| 
 | |
| 	return 0;
 | |
| 
 | |
| }
 | |
| 
 | |
| static int cyapa_gen6_config_dev_irq(struct cyapa *cyapa, u8 cmd_code)
 | |
| {
 | |
| 	u8 cmd[] = { 0x04, 0x00, 0x05, 0x00, 0x2f, 0x00, cmd_code };
 | |
| 	u8 resp_data[6];
 | |
| 	int resp_len;
 | |
| 	int error;
 | |
| 
 | |
| 	resp_len = sizeof(resp_data);
 | |
| 	error = cyapa_i2c_pip_cmd_irq_sync(cyapa, cmd, sizeof(cmd),
 | |
| 			resp_data, &resp_len,
 | |
| 			500, cyapa_sort_tsg_pip_app_resp_data, false);
 | |
| 	if (error || !VALID_CMD_RESP_HEADER(resp_data, cmd_code) ||
 | |
| 			!PIP_CMD_COMPLETE_SUCCESS(resp_data)
 | |
| 			)
 | |
| 		return error < 0 ? error : -EINVAL;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int cyapa_gen6_set_proximity(struct cyapa *cyapa, bool enable)
 | |
| {
 | |
| 	int error;
 | |
| 
 | |
| 	cyapa_gen6_config_dev_irq(cyapa, GEN6_DISABLE_CMD_IRQ);
 | |
| 	error = cyapa_pip_set_proximity(cyapa, enable);
 | |
| 	cyapa_gen6_config_dev_irq(cyapa, GEN6_ENABLE_CMD_IRQ);
 | |
| 
 | |
| 	return error;
 | |
| }
 | |
| 
 | |
| static int cyapa_gen6_change_power_state(struct cyapa *cyapa, u8 power_mode)
 | |
| {
 | |
| 	u8 cmd[] = { 0x04, 0x00, 0x06, 0x00, 0x2f, 0x00, 0x46, power_mode };
 | |
| 	u8 resp_data[6];
 | |
| 	int resp_len;
 | |
| 	int error;
 | |
| 
 | |
| 	resp_len = sizeof(resp_data);
 | |
| 	error = cyapa_i2c_pip_cmd_irq_sync(cyapa, cmd, sizeof(cmd),
 | |
| 			resp_data, &resp_len,
 | |
| 			500, cyapa_sort_tsg_pip_app_resp_data, false);
 | |
| 	if (error || !VALID_CMD_RESP_HEADER(resp_data, 0x46))
 | |
| 		return error < 0 ? error : -EINVAL;
 | |
| 
 | |
| 	/* New power state applied in device not match the set power state. */
 | |
| 	if (resp_data[5] != power_mode)
 | |
| 		return -EAGAIN;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int cyapa_gen6_set_interval_setting(struct cyapa *cyapa,
 | |
| 		struct gen6_interval_setting *interval_setting)
 | |
| {
 | |
| 	struct gen6_set_interval_cmd {
 | |
| 		__le16 addr;
 | |
| 		__le16 length;
 | |
| 		u8 report_id;
 | |
| 		u8 rsvd;  /* Reserved, must be 0 */
 | |
| 		u8 cmd_code;
 | |
| 		__le16 active_interval;
 | |
| 		__le16 lp1_interval;
 | |
| 		__le16 lp2_interval;
 | |
| 	} __packed set_interval_cmd;
 | |
| 	u8 resp_data[11];
 | |
| 	int resp_len;
 | |
| 	int error;
 | |
| 
 | |
| 	memset(&set_interval_cmd, 0, sizeof(set_interval_cmd));
 | |
| 	put_unaligned_le16(PIP_OUTPUT_REPORT_ADDR, &set_interval_cmd.addr);
 | |
| 	put_unaligned_le16(sizeof(set_interval_cmd) - 2,
 | |
| 			   &set_interval_cmd.length);
 | |
| 	set_interval_cmd.report_id = PIP_APP_CMD_REPORT_ID;
 | |
| 	set_interval_cmd.cmd_code = GEN6_SET_POWER_MODE_INTERVAL;
 | |
| 	put_unaligned_le16(interval_setting->active_interval,
 | |
| 			   &set_interval_cmd.active_interval);
 | |
| 	put_unaligned_le16(interval_setting->lp1_interval,
 | |
| 			   &set_interval_cmd.lp1_interval);
 | |
| 	put_unaligned_le16(interval_setting->lp2_interval,
 | |
| 			   &set_interval_cmd.lp2_interval);
 | |
| 
 | |
| 	resp_len = sizeof(resp_data);
 | |
| 	error = cyapa_i2c_pip_cmd_irq_sync(cyapa,
 | |
| 			(u8 *)&set_interval_cmd, sizeof(set_interval_cmd),
 | |
| 			resp_data, &resp_len,
 | |
| 			500, cyapa_sort_tsg_pip_app_resp_data, false);
 | |
| 	if (error ||
 | |
| 		!VALID_CMD_RESP_HEADER(resp_data, GEN6_SET_POWER_MODE_INTERVAL))
 | |
| 		return error < 0 ? error : -EINVAL;
 | |
| 
 | |
| 	/* Get the real set intervals from response. */
 | |
| 	interval_setting->active_interval = get_unaligned_le16(&resp_data[5]);
 | |
| 	interval_setting->lp1_interval = get_unaligned_le16(&resp_data[7]);
 | |
| 	interval_setting->lp2_interval = get_unaligned_le16(&resp_data[9]);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int cyapa_gen6_get_interval_setting(struct cyapa *cyapa,
 | |
| 		struct gen6_interval_setting *interval_setting)
 | |
| {
 | |
| 	u8 cmd[] = { 0x04, 0x00, 0x05, 0x00, 0x2f, 0x00,
 | |
| 		     GEN6_GET_POWER_MODE_INTERVAL };
 | |
| 	u8 resp_data[11];
 | |
| 	int resp_len;
 | |
| 	int error;
 | |
| 
 | |
| 	resp_len = sizeof(resp_data);
 | |
| 	error = cyapa_i2c_pip_cmd_irq_sync(cyapa, cmd, sizeof(cmd),
 | |
| 			resp_data, &resp_len,
 | |
| 			500, cyapa_sort_tsg_pip_app_resp_data, false);
 | |
| 	if (error ||
 | |
| 		!VALID_CMD_RESP_HEADER(resp_data, GEN6_GET_POWER_MODE_INTERVAL))
 | |
| 		return error < 0 ? error : -EINVAL;
 | |
| 
 | |
| 	interval_setting->active_interval = get_unaligned_le16(&resp_data[5]);
 | |
| 	interval_setting->lp1_interval = get_unaligned_le16(&resp_data[7]);
 | |
| 	interval_setting->lp2_interval = get_unaligned_le16(&resp_data[9]);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int cyapa_gen6_deep_sleep(struct cyapa *cyapa, u8 state)
 | |
| {
 | |
| 	u8 ping[] = { 0x04, 0x00, 0x05, 0x00, 0x2f, 0x00, 0x00 };
 | |
| 
 | |
| 	if (state == PIP_DEEP_SLEEP_STATE_ON)
 | |
| 		/*
 | |
| 		 * Send ping command to notify device prepare for wake up
 | |
| 		 * when it's in deep sleep mode. At this time, device will
 | |
| 		 * response nothing except an I2C NAK.
 | |
| 		 */
 | |
| 		cyapa_i2c_pip_write(cyapa, ping, sizeof(ping));
 | |
| 
 | |
| 	return cyapa_pip_deep_sleep(cyapa, state);
 | |
| }
 | |
| 
 | |
| static int cyapa_gen6_set_power_mode(struct cyapa *cyapa,
 | |
| 		u8 power_mode, u16 sleep_time, enum cyapa_pm_stage pm_stage)
 | |
| {
 | |
| 	struct device *dev = &cyapa->client->dev;
 | |
| 	struct gen6_interval_setting *interval_setting =
 | |
| 			&cyapa->gen6_interval_setting;
 | |
| 	u8 lp_mode;
 | |
| 	int error;
 | |
| 
 | |
| 	if (cyapa->state != CYAPA_STATE_GEN6_APP)
 | |
| 		return 0;
 | |
| 
 | |
| 	if (PIP_DEV_GET_PWR_STATE(cyapa) == UNINIT_PWR_MODE) {
 | |
| 		/*
 | |
| 		 * Assume TP in deep sleep mode when driver is loaded,
 | |
| 		 * avoid driver unload and reload command IO issue caused by TP
 | |
| 		 * has been set into deep sleep mode when unloading.
 | |
| 		 */
 | |
| 		PIP_DEV_SET_PWR_STATE(cyapa, PWR_MODE_OFF);
 | |
| 	}
 | |
| 
 | |
| 	if (PIP_DEV_UNINIT_SLEEP_TIME(cyapa) &&
 | |
| 		PIP_DEV_GET_PWR_STATE(cyapa) != PWR_MODE_OFF)
 | |
| 		PIP_DEV_SET_SLEEP_TIME(cyapa, UNINIT_SLEEP_TIME);
 | |
| 
 | |
| 	if (PIP_DEV_GET_PWR_STATE(cyapa) == power_mode) {
 | |
| 		if (power_mode == PWR_MODE_OFF ||
 | |
| 			power_mode == PWR_MODE_FULL_ACTIVE ||
 | |
| 			power_mode == PWR_MODE_BTN_ONLY ||
 | |
| 			PIP_DEV_GET_SLEEP_TIME(cyapa) == sleep_time) {
 | |
| 			/* Has in correct power mode state, early return. */
 | |
| 			return 0;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (power_mode == PWR_MODE_OFF) {
 | |
| 		cyapa_gen6_config_dev_irq(cyapa, GEN6_DISABLE_CMD_IRQ);
 | |
| 
 | |
| 		error = cyapa_gen6_deep_sleep(cyapa, PIP_DEEP_SLEEP_STATE_OFF);
 | |
| 		if (error) {
 | |
| 			dev_err(dev, "enter deep sleep fail: %d\n", error);
 | |
| 			return error;
 | |
| 		}
 | |
| 
 | |
| 		PIP_DEV_SET_PWR_STATE(cyapa, PWR_MODE_OFF);
 | |
| 		return 0;
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	 * When trackpad in power off mode, it cannot change to other power
 | |
| 	 * state directly, must be wake up from sleep firstly, then
 | |
| 	 * continue to do next power sate change.
 | |
| 	 */
 | |
| 	if (PIP_DEV_GET_PWR_STATE(cyapa) == PWR_MODE_OFF) {
 | |
| 		error = cyapa_gen6_deep_sleep(cyapa, PIP_DEEP_SLEEP_STATE_ON);
 | |
| 		if (error) {
 | |
| 			dev_err(dev, "deep sleep wake fail: %d\n", error);
 | |
| 			return error;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	 * Disable device assert interrupts for command response to avoid
 | |
| 	 * disturbing system suspending or hibernating process.
 | |
| 	 */
 | |
| 	cyapa_gen6_config_dev_irq(cyapa, GEN6_DISABLE_CMD_IRQ);
 | |
| 
 | |
| 	if (power_mode == PWR_MODE_FULL_ACTIVE) {
 | |
| 		error = cyapa_gen6_change_power_state(cyapa,
 | |
| 				GEN6_POWER_MODE_ACTIVE);
 | |
| 		if (error) {
 | |
| 			dev_err(dev, "change to active fail: %d\n", error);
 | |
| 			goto out;
 | |
| 		}
 | |
| 
 | |
| 		PIP_DEV_SET_PWR_STATE(cyapa, PWR_MODE_FULL_ACTIVE);
 | |
| 
 | |
| 		/* Sync the interval setting from device. */
 | |
| 		cyapa_gen6_get_interval_setting(cyapa, interval_setting);
 | |
| 
 | |
| 	} else if (power_mode == PWR_MODE_BTN_ONLY) {
 | |
| 		error = cyapa_gen6_change_power_state(cyapa,
 | |
| 				GEN6_POWER_MODE_BTN_ONLY);
 | |
| 		if (error) {
 | |
| 			dev_err(dev, "fail to button only mode: %d\n", error);
 | |
| 			goto out;
 | |
| 		}
 | |
| 
 | |
| 		PIP_DEV_SET_PWR_STATE(cyapa, PWR_MODE_BTN_ONLY);
 | |
| 	} else {
 | |
| 		/*
 | |
| 		 * Gen6 internally supports to 2 low power scan interval time,
 | |
| 		 * so can help to switch power mode quickly.
 | |
| 		 * such as runtime suspend and system suspend.
 | |
| 		 */
 | |
| 		if (interval_setting->lp1_interval == sleep_time) {
 | |
| 			lp_mode = GEN6_POWER_MODE_LP_MODE1;
 | |
| 		} else if (interval_setting->lp2_interval == sleep_time) {
 | |
| 			lp_mode = GEN6_POWER_MODE_LP_MODE2;
 | |
| 		} else {
 | |
| 			if (interval_setting->lp1_interval == 0) {
 | |
| 				interval_setting->lp1_interval = sleep_time;
 | |
| 				lp_mode = GEN6_POWER_MODE_LP_MODE1;
 | |
| 			} else {
 | |
| 				interval_setting->lp2_interval = sleep_time;
 | |
| 				lp_mode = GEN6_POWER_MODE_LP_MODE2;
 | |
| 			}
 | |
| 			cyapa_gen6_set_interval_setting(cyapa,
 | |
| 							interval_setting);
 | |
| 		}
 | |
| 
 | |
| 		error = cyapa_gen6_change_power_state(cyapa, lp_mode);
 | |
| 		if (error) {
 | |
| 			dev_err(dev, "set power state to 0x%02x failed: %d\n",
 | |
| 				lp_mode, error);
 | |
| 			goto out;
 | |
| 		}
 | |
| 
 | |
| 		PIP_DEV_SET_SLEEP_TIME(cyapa, sleep_time);
 | |
| 		PIP_DEV_SET_PWR_STATE(cyapa,
 | |
| 			cyapa_sleep_time_to_pwr_cmd(sleep_time));
 | |
| 	}
 | |
| 
 | |
| out:
 | |
| 	cyapa_gen6_config_dev_irq(cyapa, GEN6_ENABLE_CMD_IRQ);
 | |
| 	return error;
 | |
| }
 | |
| 
 | |
| static int cyapa_gen6_initialize(struct cyapa *cyapa)
 | |
| {
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int cyapa_pip_retrieve_data_structure(struct cyapa *cyapa,
 | |
| 		u16 read_offset, u16 read_len, u8 data_id,
 | |
| 		u8 *data, int *data_buf_lens)
 | |
| {
 | |
| 	struct retrieve_data_struct_cmd {
 | |
| 		struct pip_app_cmd_head head;
 | |
| 		__le16 read_offset;
 | |
| 		__le16 read_length;
 | |
| 		u8 data_id;
 | |
| 	} __packed cmd;
 | |
| 	u8 resp_data[GEN6_MAX_RX_NUM + 10];
 | |
| 	int resp_len;
 | |
| 	int error;
 | |
| 
 | |
| 	memset(&cmd, 0, sizeof(cmd));
 | |
| 	put_unaligned_le16(PIP_OUTPUT_REPORT_ADDR, &cmd.head.addr);
 | |
| 	put_unaligned_le16(sizeof(cmd) - 2, &cmd.head.length);
 | |
| 	cmd.head.report_id = PIP_APP_CMD_REPORT_ID;
 | |
| 	cmd.head.cmd_code = PIP_RETRIEVE_DATA_STRUCTURE;
 | |
| 	put_unaligned_le16(read_offset, &cmd.read_offset);
 | |
| 	put_unaligned_le16(read_len, &cmd.read_length);
 | |
| 	cmd.data_id = data_id;
 | |
| 
 | |
| 	resp_len = sizeof(resp_data);
 | |
| 	error = cyapa_i2c_pip_cmd_irq_sync(cyapa,
 | |
| 				(u8 *)&cmd, sizeof(cmd),
 | |
| 				resp_data, &resp_len,
 | |
| 				500, cyapa_sort_tsg_pip_app_resp_data,
 | |
| 				true);
 | |
| 	if (error || !PIP_CMD_COMPLETE_SUCCESS(resp_data) ||
 | |
| 		resp_data[6] != data_id ||
 | |
| 		!VALID_CMD_RESP_HEADER(resp_data, PIP_RETRIEVE_DATA_STRUCTURE))
 | |
| 		return (error < 0) ? error : -EAGAIN;
 | |
| 
 | |
| 	read_len = get_unaligned_le16(&resp_data[7]);
 | |
| 	if (*data_buf_lens < read_len) {
 | |
| 		*data_buf_lens = read_len;
 | |
| 		return -ENOBUFS;
 | |
| 	}
 | |
| 
 | |
| 	memcpy(data, &resp_data[10], read_len);
 | |
| 	*data_buf_lens = read_len;
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static ssize_t cyapa_gen6_show_baseline(struct device *dev,
 | |
| 		struct device_attribute *attr, char *buf)
 | |
| {
 | |
| 	struct cyapa *cyapa = dev_get_drvdata(dev);
 | |
| 	u8 data[GEN6_MAX_RX_NUM];
 | |
| 	int data_len;
 | |
| 	int size = 0;
 | |
| 	int i;
 | |
| 	int error;
 | |
| 	int resume_error;
 | |
| 
 | |
| 	if (!cyapa_is_pip_app_mode(cyapa))
 | |
| 		return -EBUSY;
 | |
| 
 | |
| 	/* 1. Suspend Scanning*/
 | |
| 	error = cyapa_pip_suspend_scanning(cyapa);
 | |
| 	if (error)
 | |
| 		return error;
 | |
| 
 | |
| 	/* 2. IDAC and RX Attenuator Calibration Data (Center Frequency). */
 | |
| 	data_len = sizeof(data);
 | |
| 	error = cyapa_pip_retrieve_data_structure(cyapa, 0, data_len,
 | |
| 			GEN6_RETRIEVE_DATA_ID_RX_ATTENURATOR_IDAC,
 | |
| 			data, &data_len);
 | |
| 	if (error)
 | |
| 		goto resume_scanning;
 | |
| 
 | |
| 	size = scnprintf(buf, PAGE_SIZE, "%d %d %d %d %d %d ",
 | |
| 			data[0],  /* RX Attenuator Mutual */
 | |
| 			data[1],  /* IDAC Mutual */
 | |
| 			data[2],  /* RX Attenuator Self RX */
 | |
| 			data[3],  /* IDAC Self RX */
 | |
| 			data[4],  /* RX Attenuator Self TX */
 | |
| 			data[5]	  /* IDAC Self TX */
 | |
| 			);
 | |
| 
 | |
| 	/* 3. Read Attenuator Trim. */
 | |
| 	data_len = sizeof(data);
 | |
| 	error = cyapa_pip_retrieve_data_structure(cyapa, 0, data_len,
 | |
| 			GEN6_RETRIEVE_DATA_ID_ATTENURATOR_TRIM,
 | |
| 			data, &data_len);
 | |
| 	if (error)
 | |
| 		goto resume_scanning;
 | |
| 
 | |
| 	/* set attenuator trim values. */
 | |
| 	for (i = 0; i < data_len; i++)
 | |
| 		size += scnprintf(buf + size, PAGE_SIZE - size,	"%d ", data[i]);
 | |
| 	size += scnprintf(buf + size, PAGE_SIZE - size, "\n");
 | |
| 
 | |
| resume_scanning:
 | |
| 	/* 4. Resume Scanning*/
 | |
| 	resume_error = cyapa_pip_resume_scanning(cyapa);
 | |
| 	if (resume_error || error) {
 | |
| 		memset(buf, 0, PAGE_SIZE);
 | |
| 		return resume_error ? resume_error : error;
 | |
| 	}
 | |
| 
 | |
| 	return size;
 | |
| }
 | |
| 
 | |
| static int cyapa_gen6_operational_check(struct cyapa *cyapa)
 | |
| {
 | |
| 	struct device *dev = &cyapa->client->dev;
 | |
| 	int error;
 | |
| 
 | |
| 	if (cyapa->gen != CYAPA_GEN6)
 | |
| 		return -ENODEV;
 | |
| 
 | |
| 	switch (cyapa->state) {
 | |
| 	case CYAPA_STATE_GEN6_BL:
 | |
| 		error = cyapa_pip_bl_exit(cyapa);
 | |
| 		if (error) {
 | |
| 			/* Try to update trackpad product information. */
 | |
| 			cyapa_gen6_bl_read_app_info(cyapa);
 | |
| 			goto out;
 | |
| 		}
 | |
| 
 | |
| 		cyapa->state = CYAPA_STATE_GEN6_APP;
 | |
| 		fallthrough;
 | |
| 
 | |
| 	case CYAPA_STATE_GEN6_APP:
 | |
| 		/*
 | |
| 		 * If trackpad device in deep sleep mode,
 | |
| 		 * the app command will fail.
 | |
| 		 * So always try to reset trackpad device to full active when
 | |
| 		 * the device state is required.
 | |
| 		 */
 | |
| 		error = cyapa_gen6_set_power_mode(cyapa,
 | |
| 				PWR_MODE_FULL_ACTIVE, 0, CYAPA_PM_ACTIVE);
 | |
| 		if (error)
 | |
| 			dev_warn(dev, "%s: failed to set power active mode.\n",
 | |
| 				__func__);
 | |
| 
 | |
| 		/* By default, the trackpad proximity function is enabled. */
 | |
| 		error = cyapa_pip_set_proximity(cyapa, true);
 | |
| 		if (error)
 | |
| 			dev_warn(dev, "%s: failed to enable proximity.\n",
 | |
| 				__func__);
 | |
| 
 | |
| 		/* Get trackpad product information. */
 | |
| 		error = cyapa_gen6_read_sys_info(cyapa);
 | |
| 		if (error)
 | |
| 			goto out;
 | |
| 		/* Only support product ID starting with CYTRA */
 | |
| 		if (memcmp(cyapa->product_id, product_id,
 | |
| 				strlen(product_id)) != 0) {
 | |
| 			dev_err(dev, "%s: unknown product ID (%s)\n",
 | |
| 				__func__, cyapa->product_id);
 | |
| 			error = -EINVAL;
 | |
| 		}
 | |
| 		break;
 | |
| 	default:
 | |
| 		error = -EINVAL;
 | |
| 	}
 | |
| 
 | |
| out:
 | |
| 	return error;
 | |
| }
 | |
| 
 | |
| const struct cyapa_dev_ops cyapa_gen6_ops = {
 | |
| 	.check_fw = cyapa_pip_check_fw,
 | |
| 	.bl_enter = cyapa_pip_bl_enter,
 | |
| 	.bl_initiate = cyapa_pip_bl_initiate,
 | |
| 	.update_fw = cyapa_pip_do_fw_update,
 | |
| 	.bl_activate = cyapa_pip_bl_activate,
 | |
| 	.bl_deactivate = cyapa_pip_bl_deactivate,
 | |
| 
 | |
| 	.show_baseline = cyapa_gen6_show_baseline,
 | |
| 	.calibrate_store = cyapa_pip_do_calibrate,
 | |
| 
 | |
| 	.initialize = cyapa_gen6_initialize,
 | |
| 
 | |
| 	.state_parse = cyapa_pip_state_parse,
 | |
| 	.operational_check = cyapa_gen6_operational_check,
 | |
| 
 | |
| 	.irq_handler = cyapa_pip_irq_handler,
 | |
| 	.irq_cmd_handler = cyapa_pip_irq_cmd_handler,
 | |
| 	.sort_empty_output_data = cyapa_empty_pip_output_data,
 | |
| 	.set_power_mode = cyapa_gen6_set_power_mode,
 | |
| 
 | |
| 	.set_proximity = cyapa_gen6_set_proximity,
 | |
| };
 |