基于“xxx” Androidx平台的驱动及系统开发 之 触摸板篇

目录

  • 一、基于全志 A133 Android10平台,适配1366x768 - ilitek2511触摸
    • 1、原理图分析
    • 2、驱动移植与适配
    • 3、补丁和资源文件
  • 二、基于瑞芯微 RK3566 Android11平台,适配GT9XX触摸
    • 1、原理图分析
    • 2、补丁及资源文件
  • 三、遇到的问题与解决
    • 1、基于amlogic Android9平台,使用(V-By-One接口LCD屏,USB接口触摸板)的一体式触摸屏
    • 2、基于RK3566 Android11平台,使用(LVDS接口LCD屏,I2C接口触摸板)的分离式触摸屏
    • 3、基于GT9XX触摸IC 触摸出现 -> "点偏位" 或 "划线异常不连续"
      • ◕对该触摸驱动如何通过“CTP_CFG_GROUPx”这个配置组表更新到触摸IC寄存器内部的
      • 有些人可能是通过外部去对触摸进行矫正的,那么就需要把自动写入对应的配置信息组表功能给关了
    • 4、在屏幕熄屏情况下做触摸唤醒屏幕功能,但是内核版本原因,无法使用原厂提供的手势识别功能

一、基于全志 A133 Android10平台,适配1366x768 - ilitek2511触摸

1、原理图分析

在这里插入图片描述  ilitek2511触摸使用的时I2C通信,因此重点关注SDA(数据)、SCK(时钟)、INT(中断)、RST(复位)这四个引脚即可。

2、驱动移植与适配

  首先我们需要弄清除,触摸驱动是干嘛的,其实它只不过是通过I2C通信方式获取触摸板上触发的每个坐标点信息,通过事件方式上报,像这种通用的事件上报方式(触摸屏),上层的HAL、Framework等,原生Android或者原厂都是已经给做好了,已经只需要移植驱动层即可,一般别的层无需做修改的。
  通过触摸IC厂商提供的驱动移植资料修改设备树,如下图原厂提供的设备树和我修改好的设备树。
在这里插入图片描述
  设备树配置好后,直接将相关的驱动文件拷贝到 a\longan\kernel\linux-4.9\drivers\input\touchscreen 目录下创建的ilitek文件夹下即可,如下图所示。
在这里插入图片描述  通过以上操作,基本是已经将触摸驱动移植好了。

3、补丁和资源文件

下载链接

二、基于瑞芯微 RK3566 Android11平台,适配GT9XX触摸

1、原理图分析

在这里插入图片描述

2、补丁及资源文件

  这里就不多分析了,直接给上补丁文件。
下载链接

三、遇到的问题与解决

1、基于amlogic Android9平台,使用(V-By-One接口LCD屏,USB接口触摸板)的一体式触摸屏

出现问题1:明明获取的USB设备,但是一直处于disconnected断开,一直打印不支持该USB设备,日志如下图所示
在这里插入图片描述
解决方法1:其实这个是控制板卡上的液晶屏的参数设置不对引起的,使得它们与触摸控制器不匹配,从而使触摸屏功能无法正常工作。因此只需要更具屏幕规格书修改屏幕参数即可。

2、基于RK3566 Android11平台,使用(LVDS接口LCD屏,I2C接口触摸板)的分离式触摸屏

出现问题1:因为触摸板和LCD屏是分离的,因此有可能出现整合者将触摸板和LCD屏摆放的起始方向不一致 或者 触摸板默认竖屏,而屏幕默认横屏, 这两种原因都会导致触摸和显示UI对不上。
解决方法1:在device/rockchip/3566x/目录下的build.prop文件中添加 persist.sys.touch.orientation=90, 如我设置90就是说将触摸板旋转方向为90度,这样就可以使UI和触摸板对上了。

3、基于GT9XX触摸IC 触摸出现 -> “点偏位” 或 “划线异常不连续”

出现的问题1:在原厂提供的文档可以可知,一般每一种触摸IC都有对应的“CTP_CFG_GROUP”这个触摸配置组的,它作用就是优化矫正触摸的。
解决方法1:在drivers\input\touchscreen\gt9xxnew\gt9xx.h文件中将原厂提供的.cfg峰位文件内从内容替换对应CTP_CFG_GROUPx中的内容,如果不确定CTP_CFG_GROUPx是哪个,那可以选择最无脑的方法,将CTP_CFG_GROUP[1~n]中的内容都给替换掉即可。

◕对该触摸驱动如何通过“CTP_CFG_GROUPx”这个配置组表更新到触摸IC寄存器内部的

  找到drivers\input\touchscreen\gt9xxnew\gt9xx.c这个文件,该文件是触摸主功能的驱动文件。
  在goodix_ts_probe函数下找到如下图的函数调用:
在这里插入图片描述
  通过该函数索引到函数实现的位置,这里我把源码贴出来,已注释方式解释了。

/*******************************************************
Function:Initialize gtp.
Input:ts: goodix private data
Output:Executive outcomes.0: succeed, otherwise: failed
*******************************************************/
static s32 gtp_init_panel(struct goodix_ts_data *ts)
{s32 ret = -1;#if GTP_DRIVER_SEND_CFGs32 i = 0;u8 check_sum = 0;u8 opr_buf[16] = {0};u8 sensor_id = 0;/***************************(1)*********************************//* 这一部分是获取gt9xx.h文件定义的CTP_CFG_GROUPx配置信息组的信息到cfg_info_groupx[]中 */u8 cfg_info_group1[] = CTP_CFG_GROUP1;u8 cfg_info_group2[] = CTP_CFG_GROUP2;u8 cfg_info_group3[] = CTP_CFG_GROUP3;u8 cfg_info_group4[] = CTP_CFG_GROUP4;u8 cfg_info_group5[] = CTP_CFG_GROUP5;u8 cfg_info_group6[] = CTP_CFG_GROUP6;u8 cfg_info_group7[] = CTP_CFG_GROUP7;u8 cfg_info_group8[] = CTP_CFG_GROUP8;/***************************(1)*********************************//***************************(2)*********************************//* 这个是创建send_cfg_buf数组,将上面活动的cfg_info_groupx信息保以元素存到该数组里 */u8 *send_cfg_buf[] = {cfg_info_group1, cfg_info_group2, cfg_info_group3,cfg_info_group4, cfg_info_group5, cfg_info_group6,cfg_info_group7, cfg_info_group8};/***************************(2)*********************************//***************************(3)*********************************//* CFG_GROUP_LEN这个方法在gt9xx.h文件中有定义,原型是#define CFG_GROUP_LEN(p_cfg_grp)  (sizeof(p_cfg_grp) / sizeof(p_cfg_grp[0])),它是计算数组的长度的 */u8 cfg_info_len[] = { CFG_GROUP_LEN(cfg_info_group1),CFG_GROUP_LEN(cfg_info_group2),CFG_GROUP_LEN(cfg_info_group3),CFG_GROUP_LEN(cfg_info_group4),CFG_GROUP_LEN(cfg_info_group5),CFG_GROUP_LEN(cfg_info_group6),CFG_GROUP_LEN(cfg_info_group7),CFG_GROUP_LEN(cfg_info_group8)};/***************************(3)*********************************/dprintk(DEBUG_INIT,"Config Groups Lengths: %d, %d, %d, %d, %d, %d, %d, %d\n",cfg_info_len[0], cfg_info_len[1], cfg_info_len[2], cfg_info_len[3],cfg_info_len[4], cfg_info_len[5], cfg_info_len[6], cfg_info_len[7]);#if GTP_COMPATIBLE_MODEif (CHIP_TYPE_GT9F == ts->chip_type){ts->fw_error = 0;}else
#endif{/***************************(4)*********************************//* 检测固件是否正常 */ret = gtp_i2c_read_dbl_check(ts->client, 0x41E4, opr_buf, 1);if (SUCCESS == ret){if (opr_buf[0] != 0xBE){ts->fw_error = 1;printk("Firmware error, no config sent!\n");return -1;}}/***************************(4)*********************************/}/***************************(5)*********************************/if ((!cfg_info_len[1]) && (!cfg_info_len[2]) &&(!cfg_info_len[3]) && (!cfg_info_len[4]) &&(!cfg_info_len[5]) && (!cfg_info_len[6]) &&(!cfg_info_len[7])){/* 如果所有配置信息组内容都为空,sensor_id值设置为0 */sensor_id = 0;}else{#if GTP_COMPATIBLE_MODEmsleep(50);#endif//       ret = gtp_i2c_read_dbl_check(ts->client, GTP_REG_SENSOR_ID, &sensor_id, 1);/* 下面是判断配置信息的名称是对应哪个触摸IC的名称,从而再设置其对应的sensor_id值 */dprintk(DEBUG_INIT,"CTP name : %s\n",config_info.name);if (!strcmp(config_info.name,"gt9271_mb976a9")){= 0;dprintk(DEBUG_INIT,"gt9xx:sensor_id = %d\n",sensor_id);} else if (!strcmp(config_info.name,"gt9110_wt097")){sensor_id = 1;dprintk(DEBUG_INIT,"gt9xx:sensor_id = %d\n",sensor_id);} else if (!strcmp(config_info.name,"gt9271_wt097")){sensor_id = 2;dprintk(DEBUG_INIT,"gt9xx:sensor_id = %d\n",sensor_id);} else if (!strcmp(config_info.name,"gt9110_g200")){sensor_id = 3;dprintk(DEBUG_INIT,"gt9xx:sensor_id = %d\n",sensor_id);} else if (!strcmp(config_info.name,"gt9271_noah")){sensor_id = 4;dprintk(DEBUG_INIT,"gt9xx:sensor_id = %d\n",sensor_id);} else if (!strcmp(config_info.name,"gt9271_p2")){sensor_id = 5;dprintk(DEBUG_INIT,"gt9xx:sensor_id = %d\n",sensor_id);} else if (!strcmp(config_info.name, "gt911_1060")) {sensor_id = 6;dprintk(DEBUG_INIT,"gt9xx:sensor_id = %d\n",sensor_id);} else if (!strcmp(config_info.name, "gt911_784")) {sensor_id = 7;dprintk(DEBUG_INIT,"gt9xx:sensor_id = %d\n",sensor_id);} else {sensor_id = 0;dprintk(DEBUG_INIT,"gt9xx:sensor_id = %d\n",sensor_id);}ret = SUCCESS;if (SUCCESS == ret){if (sensor_id >= 0x08){printk("Invalid sensor_id(0x%02X), No Config Sent!\n", sensor_id);ts->pnl_init_error = 1;return -1;}}else{printk("Failed to get sensor_id, No config sent!\n");ts->pnl_init_error = 1;return -1;}dprintk(DEBUG_INIT,"Sensor_ID: %d", sensor_id);}/***************************(5)*********************************//***************************(6)*********************************//* 上面设置的sensor_id值就是用来选择send_cfg_buf数组对应的触配置信息组元素的,这里获取对应配置信息组表的长度 */ts->gtp_cfg_len = cfg_info_len[sensor_id];/***************************(6)*********************************/dprintk(DEBUG_INIT,"CTP_CONFIG_GROUP%d used, config length: %d\n", sensor_id + 1, ts->gtp_cfg_len);/***************************(7)*********************************//* 用获取对应配置信息组表的长度与定义的配置信息组表最小长度作对比,确保配置信息组表的数据在长度上是符合的 */if (ts->gtp_cfg_len < GTP_CONFIG_MIN_LENGTH){printk("Config Group%d is INVALID CONFIG GROUP(Len: %d)! NO Config Sent! You need to check you header file CFG_GROUP section!\n", sensor_id+1, ts->gtp_cfg_len);ts->pnl_init_error = 1;return -1;}/***************************(7)*********************************/#if GTP_COMPATIBLE_MODEif (CHIP_TYPE_GT9F == ts->chip_type){ts->fixed_cfg = 0;}else
#endif{/***************************(8)*********************************//* 获取IC寄存器对应的配置信息版本 */ret = gtp_i2c_read_dbl_check(ts->client, GTP_REG_CONFIG_DATA, &opr_buf[0], 1);if (ret == SUCCESS){dprintk(DEBUG_INIT,"CFG_GROUP%d Config Version: %d, 0x%02X; IC Config Version: %d, 0x%02X\n", sensor_id+1,send_cfg_buf[sensor_id][0], send_cfg_buf[sensor_id][0], opr_buf[0], opr_buf[0]);if (opr_buf[0] < 155){grp_cfg_version = send_cfg_buf[sensor_id][0];       // backup group config versionsend_cfg_buf[sensor_id][0] = 0x00;ts->fixed_cfg = 0;	/* 如果是opr_buf[0] < 155这种条件下,那么这个配置信息组不设置为固定不变标志 */}else        // treated as fixed config, not send config{dprintk(DEBUG_INIT,"Ic fixed config with config version(%d, 0x%02X)\n", opr_buf[0], opr_buf[0]);ts->fixed_cfg = 1;	/* 配置信息组设置为固定不变标志 */gtp_get_info(ts);return 0;}}else{printk("Failed to get ic config version!No config sent!\n");return -1;}/***************************(8)*********************************/}/***************************(9)*********************************//* 将send_cfg_buf[sensor_id]这个对应的配置信息组表信息拷贝到 config[GTP_ADDR_LENGTH]这个数组指定的开始位置中 */memset(&config[GTP_ADDR_LENGTH], 0, GTP_CONFIG_MAX_LENGTH);memcpy(&config[GTP_ADDR_LENGTH], send_cfg_buf[sensor_id], ts->gtp_cfg_len);/***************************(9)*********************************/#if GTP_CUSTOM_CFGconfig[RESOLUTION_LOC]     = (u8)GTP_MAX_WIDTH;config[RESOLUTION_LOC + 1] = (u8)(GTP_MAX_WIDTH>>8);config[RESOLUTION_LOC + 2] = (u8)GTP_MAX_HEIGHT;config[RESOLUTION_LOC + 3] = (u8)(GTP_MAX_HEIGHT>>8);if (GTP_INT_TRIGGER == 0)  //RISING{config[TRIGGER_LOC] &= 0xfe;}else if (GTP_INT_TRIGGER == 1)  //FALLING{config[TRIGGER_LOC] |= 0x01;}
#endif  // GTP_CUSTOM_CFG/***************************(10)*********************************//* 计算校验和,并将校验和设置到config数组中 */check_sum = 0;for (i = GTP_ADDR_LENGTH; i < ts->gtp_cfg_len; i++){check_sum += config[i];}config[ts->gtp_cfg_len] = (~check_sum) + 1;/***************************(10)*********************************/#else // driver not send configts->gtp_cfg_len = GTP_CONFIG_MAX_LENGTH;ret = gtp_i2c_read(ts->client, config, ts->gtp_cfg_len + GTP_ADDR_LENGTH);if (ret < 0){GTP_ERROR("Read Config Failed, Using Default Resolution & INT Trigger!");ts->abs_x_max = GTP_MAX_WIDTH;ts->abs_y_max = GTP_MAX_HEIGHT;ts->int_trigger_type = GTP_INT_TRIGGER;}#endif // GTP_DRIVER_SEND_CFG/***************************(11)*********************************//* 修正abs_x_max、abs_y_max、int_trigger_type的值   */if ((ts->abs_x_max == 0) && (ts->abs_y_max == 0)){ts->abs_x_max = (config[RESOLUTION_LOC + 1] << 8) + config[RESOLUTION_LOC];ts->abs_y_max = (config[RESOLUTION_LOC + 3] << 8) + config[RESOLUTION_LOC + 2];ts->int_trigger_type = (config[TRIGGER_LOC]) & 0x03;}/***************************(11)*********************************/#if GTP_COMPATIBLE_MODEif (CHIP_TYPE_GT9F == ts->chip_type){u8 sensor_num = 0;u8 driver_num = 0;u8 have_key = 0;have_key = (config[GTP_REG_HAVE_KEY - GTP_REG_CONFIG_DATA + 2] & 0x01);if (1 == ts->is_950){driver_num = config[GTP_REG_MATRIX_DRVNUM - GTP_REG_CONFIG_DATA + 2];sensor_num = config[GTP_REG_MATRIX_SENNUM - GTP_REG_CONFIG_DATA + 2];if (have_key){driver_num--;}ts->bak_ref_len = (driver_num * (sensor_num - 1) + 2) * 2 * 6;}else{driver_num = (config[CFG_LOC_DRVA_NUM] & 0x1F) + (config[CFG_LOC_DRVB_NUM]&0x1F);if (have_key){driver_num--;}sensor_num = (config[CFG_LOC_SENS_NUM] & 0x0F) + ((config[CFG_LOC_SENS_NUM] >> 4) & 0x0F);ts->bak_ref_len = (driver_num * (sensor_num - 2) + 2) * 2;}GTP_INFO("Drv * Sen: %d * %d(key: %d), X_MAX: %d, Y_MAX: %d, TRIGGER: 0x%02x",driver_num, sensor_num, have_key, ts->abs_x_max,ts->abs_y_max,ts->int_trigger_type);return 0;}else
#endif{#if GTP_DRIVER_SEND_CFG/***************************(12)*********************************//* 通过调用gtp_send_cfg将已经整好的config配置信息组表数据写入到触摸IC对应的寄存器中 */ret = gtp_send_cfg(ts->client);if (ret < 0){GTP_ERROR("Send config error.");}// set config version to CTP_CFG_GROUP, for resume to send configconfig[GTP_ADDR_LENGTH] = grp_cfg_version;check_sum = 0;for (i = GTP_ADDR_LENGTH; i < ts->gtp_cfg_len; i++){check_sum += config[i];}config[ts->gtp_cfg_len] = (~check_sum) + 1;/***************************(12)*********************************/#endifGTP_INFO("X_MAX: %d, Y_MAX: %d, TRIGGER: 0x%02x", ts->abs_x_max,ts->abs_y_max,ts->int_trigger_type);}msleep(10);return 0;
}

  gtp_send_cfg函数原型,如下所示:

/*******************************************************
Function:Send config.
Input:client: i2c device.
Output:result of i2c write operation.1: succeed, otherwise: failed
*********************************************************/s32 gtp_send_cfg(struct i2c_client *client)
{s32 ret = 2;#if GTP_DRIVER_SEND_CFGs32 retry = 0;struct goodix_ts_data *ts = i2c_get_clientdata(client);if (ts->fixed_cfg){GTP_INFO("Ic fixed config, no config sent!");return 0;}else if (ts->pnl_init_error){GTP_INFO("Error occured in init_panel, no config sent");return 0;}GTP_INFO("Driver send config.");for (retry = 0; retry < 5; retry++){ret = gtp_i2c_write(client, config , GTP_CONFIG_MAX_LENGTH + GTP_ADDR_LENGTH);if (ret > 0){break;}}
#endifreturn ret;
}

有些人可能是通过外部去对触摸进行矫正的,那么就需要把自动写入对应的配置信息组表功能给关了

  在gt9xx.h文件中将#define GTP_DRIVER_SEND_CFG 1设置成0即可。

4、在屏幕熄屏情况下做触摸唤醒屏幕功能,但是内核版本原因,无法使用原厂提供的手势识别功能

出现问题1:由于我使用的内核版本与移植的触摸驱动使用的内核版本相差较大,导致无法使用原厂提供的手势识别功能。
须知:一般来说,触摸芯片厂商提供的触摸驱动内部有自带手势识别处理程序,也就是说一般是带有触摸唤醒的功能的。但是,一般默认是不使能的,也就是需要通过宏定义取打开的,我这里以Goodix Touchscreen驱动举例:

static void goodix_ts_work_func(struct work_struct *work)

  这个函数就是处理手势识别的处理工作函数,那么这个就需要先启用“GTP_GESTURE_WAKEUP”这个宏定义才能开启手势功能的程序,而手势功能的程序需要启用“CONFIG_HAS_EARLYSUSPEND”这个挂起功能宏定义,才能让手势功能正常使用,但遗憾的是开启“CONFIG_HAS_EARLYSUSPEND”这个挂机功能的宏定义,有几个使用的内核接口函数都不存在,也就说明该驱动程序的手势功能在我这个Linux内核版本不支持。
解决方法1:我的Linux内核版本不支持,因此只能自己手动获取屏幕熄屏和亮屏状态,然后在上报触摸事件是上报“KEY_POWER”事件来开启屏幕。所以我需要先获取屏幕是开启还是休眠状态,通过这个状态去控制是否上报“KEY_POWER”事件来开启屏幕,从而避免在开屏状态下上报了“KEY_POWER”事件导致关屏发生。
  这里我将我修改的补丁贴出,如下所示:

commit cdc3304684b7d6ef469a2e8ef401e5882ef07748
Author: dengjiawen <1411471554@qq.com>
Date:   Fri Oct 20 14:32:46 2023 +0800屏幕息屏或者睡眠模式下,可触摸唤醒屏幕(基于gt9xx触摸系列添加该功能)diff --git a/longan/kernel/linux-4.9/drivers/input/touchscreen/gt9xxnew/gt9xx.c b/longan/kernel/linux-4.9/drivers/input/touchscreen/gt9xxnew/gt9xx.c
index 77dc9bc566..af1bc4be38 100644
--- a/longan/kernel/linux-4.9/drivers/input/touchscreen/gt9xxnew/gt9xx.c
+++ b/longan/kernel/linux-4.9/drivers/input/touchscreen/gt9xxnew/gt9xx.c
@@ -50,7 +50,9 @@#include <linux/irq.h>#include "gt9xx.h"#include <linux/pm_runtime.h>
+#include "../../../video/fbdev/sunxi/disp2/disp/dev_disp.h"+extern int get_screen_status(void);     /* 声明引入获取屏幕状态函数 */#ifdef CONFIG_PM#include <linux/pm.h>#endif
@@ -664,6 +666,22 @@ static void gtp_touch_down(struct goodix_ts_data* ts,s32 id,s32 x,s32 y,s32 w)input_report_abs(ts->input_dev, ABS_MT_TOUCH_MAJOR, w);input_report_abs(ts->input_dev, ABS_MT_WIDTH_MAJOR, w);input_report_abs(ts->input_dev, ABS_MT_TRACKING_ID, id);
+
+/************************* 获取屏幕状态,上报KEY_POWER事件来打开屏幕 *********************************/
+    printk("get_screen_status = %d\n",get_screen_status());
+    if(get_screen_status()) {
+        /* 亮屏状态,不发出KEY_POWER事件唤醒屏幕 */
+        printk("screen is open\n");
+    } else {
+        /* 熄屏状态,发出KEY_POWER事件唤醒屏幕 */
+        input_event(ts->input_dev,EV_KEY,KEY_POWER,1);
+        input_sync(ts->input_dev);
+        input_event(ts->input_dev,EV_KEY,KEY_POWER,0);
+        input_sync(ts->input_dev);
+        printk("screen is close,in open the screen......\n");
+    }
+/*****************************************************************************************************/
+input_mt_sync(ts->input_dev);#endif@@ -2150,7 +2168,7 @@ static s8 gtp_request_input_dev(struct goodix_ts_data *ts)#if GTP_ICS_SLOT_REPORTinput_mt_init_slots(ts->input_dev, 16);     // in case of "out of memory"#else
-    ts->input_dev->keybit[BIT_WORD(BTN_TOUCH)] = BIT_MASK(BTN_TOUCH);
+    ts->input_dev->keybit[BIT_WORD(BTN_TOUCH)] = BIT_MASK(BTN_TOUCH) | BIT_MASK(EV_KEY);    /* 增加外部按键事件掩码,用于KEY_POWER事件 */#endif__set_bit(INPUT_PROP_DIRECT, ts->input_dev->propbit);@@ -2169,7 +2187,7 @@ if(gtp_gesture_wakeup)input_set_capability(ts->input_dev, EV_KEY, ctp_key_list[index]);}//#endif
-
+    input_set_capability(ts->input_dev, EV_KEY, KEY_POWER);     /* 将按键功能 KEY_POWER 添加到输入设备中的按键功能列表中。这意味着输入设备可以检测到并报告关于电源键的按下和释放事件 */if(1 == exchange_x_y_flag)swap(ts->abs_x_max, ts->abs_y_max);diff --git a/longan/kernel/linux-4.9/drivers/video/fbdev/sunxi/disp2/disp/dev_disp.c b/longan/kernel/linux-4.9/drivers/video/fbdev/sunxi/disp2/disp/dev_disp.c
index 16c45a6529..8c3006fd25 100644
--- a/longan/kernel/linux-4.9/drivers/video/fbdev/sunxi/disp2/disp/dev_disp.c
+++ b/longan/kernel/linux-4.9/drivers/video/fbdev/sunxi/disp2/disp/dev_disp.c
@@ -97,6 +97,8 @@ int composer_init(struct disp_drv_info *p_disp_drv);int hwc_dump(char *buf);#endif+static int screen_status = 0;	/* 定义一个静态的屏幕状态变量 */
+static void disp_shutdown(struct platform_device *pdev);static ssize_t disp_sys_show(struct device *dev,struct device_attribute *attr, char *buf)
@@ -3009,6 +3011,7 @@ static int disp_runtime_suspend(struct device *dev)if (ret < 0) {dev_err(dev, "can't request output direction lcd_pwr_gpio gpio \n");}
+		screen_status = 0;}pr_info("%s finish\n", __func__);@@ -3077,6 +3080,7 @@ static int disp_runtime_resume(struct device *dev)if (ret < 0) {dev_err(dev, "can't request output direction lcd_pwr_gpio gpio \n");}
+		screen_status = 1;}pr_info("%s finish\n", __func__);@@ -3111,6 +3115,14 @@ static int disp_runtime_idle(struct device *dev)}#endif+/* 实现获取屏幕状态的函数,并导出共享 */
+int get_screen_status(void)
+{
+	return screen_status;
+}
+EXPORT_SYMBOL(get_screen_status);
+/***********************************/
+int disp_suspend(struct device *dev){u32 screen_id = 0;

解决方法2:修改原厂驱动源码,使其手势功能程序兼容本地使用的内核版本。
  在gt9xx.c文件中找到static void goodix_ts_work_func(struct work_struct *work)这个函数,这个函数就是处理事件上报功能的,如下图是我截取的原厂驱动双击屏幕开屏的程序段。
在这里插入图片描述
  这图程序可以看出,它是通过读取记录触摸手势状态的寄存器值来判断触发双击开屏功能的,如果触发了那么就会上报"KEY_POWER"事件来达到开屏目的,而它的屏幕状态通过doze_status这个静态变量取设置与判断的,最后会清除记录触摸手势状态的寄存器的值。
  至于具体修改方法就自行取尝试啦。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/721524.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【c++】 string类的模拟实现

1.浅拷贝 浅拷贝&#xff1a;也称位拷贝&#xff0c;编译器只是将对象中的值拷贝过来。如果对象中管理资源&#xff0c;最后就会导致多个对象共享同一份资源&#xff0c;当一个对象销毁时就会将该资源释放掉&#xff0c;而此时另一些对象不知道该资源已经被释放&#xff0c;以…

【C++基础】STL容器面试题分享||上篇

&#x1f308;欢迎来到C基础专栏 &#x1f64b;&#x1f3fe;‍♀️作者介绍&#xff1a;前PLA队员 目前是一名普通本科大三的软件工程专业学生 &#x1f30f;IP坐标&#xff1a;湖北武汉 &#x1f349; 目前技术栈&#xff1a;C/C STL 1.请说说 STL 的基本组成部分2.详细的说&…

xss.haozi.me:0x03及04

这里有一个正则所以&#xff08;&#xff09;要用到实体编码 <a href"javascript:alert1">cc</a> 03 04都一样

密码安全:保护你的数据不被入侵的重要性

title: 密码安全&#xff1a;保护你的数据不被入侵的重要性 date: 2024/3/5 17:54:56 updated: 2024/3/5 17:54:56 tags: 密码安全个人隐私保护身份盗窃防护金融损失防范弱密码危害安全密码创建双因素认证 在数字时代&#xff0c;密码安全是保护个人和机构数据的关键。然而&am…

如何做代币分析:以 INJ 币为例

如何做代币分析&#xff1a;以 INJ 币为例 作者&#xff1a; lesleyfootprint.network 编译&#xff1a;cicifootprint.network 数据源&#xff1a;INJ 代币仪表板 &#xff08;仅包括以太坊数据&#xff09; 在加密货币和数字资产领域&#xff0c;代币分析起着至关重要的作…

Linux——自写一个简易的shell

目录 前言 一、打印提示信息 二、分割字符串 三、替换程序 前言 之前学习了很多进程相关的知识&#xff0c;包括环境变量、进程的创建与退出、进程等待、进程替换。现在可以用所学的作一个小总结&#xff0c;手撕一个shell解释器&#xff0c;大致的思路是先通过环境变量获…

(vue)适合后台管理系统开发的前端框架

(vue)适合后台管理系统开发的前端框架 1、D2admin 开源地址&#xff1a;https://github.com/d2-projects/d2-admin 文档地址&#xff1a;https://d2.pub/zh/doc/d2-admin/ 效果预览&#xff1a;https://d2.pub/d2-admin/preview/#/index 开源协议&#xff1a;MIT 2、vue-el…

服务器上部署WEb服务方法

部署Web服务在服务器上是一个比较复杂的过程。这不仅仅涉及到配置环境、选择软件和设置端口&#xff0c;更有众多其它因素需要考虑。以下是在服务器上部署WEb服务的步骤&#xff1a; 1. 选择服务器&#xff1a;根据项目规模和预期访问量&#xff0c;选择合适的服务器类型和配置…

Linux 进程间通信

目录 管道 匿名管道&#xff08;pipe&#xff09; 有名管道&#xff08;fifo&#xff09; 小结 共享内存 消息队列 信号量 System V IPC的结构设计 Posix与System V的关系 管道 匿名管道&#xff08;pipe&#xff09; 我们知道&#xff0c;在Linux中通过fork创建的子…

OpenDDS之QosXml库编译(Windows + VS2019)

目录 1、需求背景2、基础环境3、编译xercesc3.1、下载xercesc3.2、编译xercesc 4、编译ACE_XML_Utils4.1、生成XML_Utils解决方案4.2、编译XML_Utils 5、编译QOS_XML_XSC_Handlerd5.1、生成QOS_XML_XSC_Handlerd解决方案5.2、编译QOS_XML_XSC_Handlerd 6、测试例子6.1、生成dum…

光影交织:汽车穿越隧道的视觉盛宴

在繁忙的城市中&#xff0c;隧道成为了连接两端的重要通道。而对于汽车来说&#xff0c;穿越隧道不仅是一次简单的空间转移&#xff0c;更是一场融合了视觉、技术与安全的独特体验。 当汽车缓缓驶入隧道&#xff0c;外界的光线逐渐减弱&#xff0c;隧道内部的光线开始发挥作用。…

Ubuntu下anaconda迁移到另外的目录

文章目录 前言一、原因二、迁移1.复制到指定迁移目录2. 修改复制后的anaconda3 内容3. 修改对应搭建的每个环境的pip4.修改系统配置文件&#xff0c;使得设置生效 三、实际测试四、总结 前言 好记性不如烂笔头&#xff0c;简单的记录下在ubantu18.04下迁移anaconda的目录 一、…

2024年软考重大改革

中国计算机技术职业资格网 考试日期 考试级别 考试资格名称 5月25日至28日 高级 系统分析师 系统架构设计师 信息系统项目管理师 中级 软件设计师 网络工程师 软件评测师 电子商务设计师 嵌入式系统设计师 数据库系统工程师 信息系统管理工程师 初级 程序员 …

DiffusionMat:Alpha Matting as sequential refinement learning

1.introduction DiffusionMat的基本思想是未知区域的抠图可以逐步改进&#xff0c;并从每一次的迭代的反馈中受益&#xff0c;纠正和细化结果。 2.related works Segdiff、BitDiffusion、DiffusionDet、 3.Approach 通过一种新颖的校正策略将trimap引导转化为精确的alpha m…

2024-简单点-picamera2除了文档还有哪里可以学习实例?

picamera2学习例子 去github的picamera2库&#xff0c;找app和examples目录&#xff0c;然后学习

自动化测试基础——allure下载安装及配置及pytest + allure-pytest插件生成allure企业级测试报告及企业级定制

文章目录 前言一、allure下载二、allure安装三、allure目录介绍四、allure环境变量配置五、pytest allure-pytest插件生成allure企业级测试报告六、allure企业级报告的log定制七、allure企业级报告功能内容定制1.功能左边层级定制2.功能右边优先级定制3.功能右边测试用例描述定…

Qt绘制动态罗盘

介绍&#xff1a;罗盘指针以30角旋转巡逻&#xff0c;扫描航海范围内的点位&#xff0c;并绘制点云。字段信息在表格中显示&#xff0c;该数据都存储在数据库中。选择不同的范围&#xff0c;显示该范围内的点位。 #include "mainwindow.h" #include "ui_mainwi…

魔行观察-每日品牌监测-书亦烧仙草-开店趋势

今日监测对象&#xff1a;书亦烧仙草&#xff0c;监测时间段&#xff1a;2014年9月至2023年12月&#xff0c;发布时间&#xff1a;2024-03-05 数据获取地址&#xff1a;魔查查https://www.moxingdata.com/品牌基础信息 现有门店人均消费覆盖省份经营模式投资金额837918.431特…

10亿数据如何快速插入MySQL

最快的速度把10亿条数据导入到数据库,首先需要和面试官明确一下,10亿条数据什么形式存在哪里,每条数据多大,是否有序导入,是否不能重复,数据库是否是MySQL? 有如下约束 10亿条数据,每条数据 1 Kb 数据内容是非结构化的用户访问日志,需要解析后写入到数据库 数据存放在…

CMIP6数据处理方法与典型案例分析

气候变化对农业、生态系统、社会经济以及人类的生存与发展具有深远影响&#xff0c;是当前全球关注的核心议题之一。IPCC&#xff08;Intergovernmental Panel on Climate Change&#xff0c;政府间气候变化专门委员会&#xff09;的第六次评估报告明确&#xff1b;指出&#x…