Linux驱动开发(16):输入子系统–电容触摸驱动实验

有关电容触摸的基础知识内容可以参考野火STM32相关教程,这里只介绍电容触摸驱动的相关内容。

本章配套源码、设备树以及更新固件位于“~/embed_linux_driver_tutorial_imx6_code/linux_driver/touch_scream_GTxxx”目录下。

触摸面板通过双面胶粘在显示屏上,他们在硬件上没有关联,通常情况下我们会设置触摸面板的“分辨率”与显示屏的分辨率一致。触摸芯片自动完成触摸信息的采集和处理,我们要做的就是配置触摸芯片、读取触点坐标。

常见的触摸接口原理图如下所示:

02|

触摸芯片有四个引脚连接到了单片机,其中scl、sda是IIC通信引脚。RSTN是触摸的复位引脚,配置触摸芯片时也会用到。INT是中断引脚,默认高电平。发生触摸后该引脚会发送一个低电平的脉冲信号。

从硬件连接我们可以大致得知电容触摸驱动要用到IIC设备驱动、中断驱动。由于电容触摸驱动要向应用层提交触摸消息所以还会用到输入子系统

IIC设备驱动、中断驱动和输入子系统在之前章节已经介绍,我们现在可以自己从零写一个触摸驱动,但是我们不这样做,因为大多情况下触摸芯片厂商已经编写好了触摸驱动,我们只需要稍加修改就可以使用。

我们使用的触摸芯片型号如下,4.3寸使用的GT5688,5寸使用的是GT917S或GT9157,7寸屏使用的是GT911。它们都属于GOODIX公司生产的触摸芯片,厂商已经写好了触摸驱动并且添加默认添加到了内核中。源码位于“内核根目录/drivers/input/touchscreen/goodix.c”这 个驱动程序并不能完全适配我们使用的这些触摸型号,所以我们需要稍加修改才能使用。下面我们结合源码简单介绍驱动实现原理以及如何根据实际需要修改驱动。

1. 添加设备树节点

设备树节点有两个任务,第一,添加触摸使用的引脚,第二,添加IIC设备节点。介绍如下。

根据之前讲解,触摸芯片共占用4个IO口。这四个IO如下所示。

引脚

功能

SNVS_TAMPER9引脚

复用为GPIO5_IO09用作触摸芯片的irq引脚,接收触摸中断

LCD_RESET引脚

复用为GPIO3_IO04用作触摸芯片的复位引脚

UART4_TX_DATA引脚

复用为I2C1_SCL,用作IIC1的SCL引脚

UART4_RX_DATA引脚

复用为I2C1_SDA,用作IIC1的 SDA引脚

知道了各个引脚的复用功能,在设备树中把它们追加到iomuxc就很容易了,需要注意的是SNVS_TAMPER9引脚被复用为GPIO5_IO09,需要追加到iomuxc_snvs节点,以“SNVS”开头的引脚与普通引脚使用的驱动不同,特别注意要分开。源码如下:

在设备树中添加引脚信息:

&iomuxc {/*-----------其他内容省略-----------*/pinctrl_i2c1: i2c1grp {fsl,pins = <MX6UL_PAD_UART4_TX_DATA__I2C1_SCL 0x4001b8b0MX6UL_PAD_UART4_RX_DATA__I2C1_SDA 0x4001b8b0>;};pinctrl_tsc_reset: tscresetgrp {fsl,pins = </* used for tsc reset */MX6UL_PAD_LCD_RESET__GPIO3_IO04         0x10b0>;};
};&iomuxc_snvs {pinctrl_tsc_irq: tsc_irq {fsl,pins = <MX6ULL_PAD_SNVS_TAMPER9__GPIO5_IO09        0x4001b8b0>;};
};

触摸驱动作为一个IIC设备挂载在IIC1总线上,它和我们之前讲解的iic接口的OLED驱动一样,需要在IIC1设备节点下追加相应的子节点,不同的是这里用到的两个GPIO和一个中断,设备节点如下所示。

在iic1节点追加:

&i2c1 {/*--------------第一部分--------------*/clock-frequency = <100000>;pinctrl-names = "default";pinctrl-0 = <&pinctrl_i2c1>;#address-cells = <1>;#size-cells = <0>;status = "okay";gtxx_tsc@5d {compatible = "fire,gt9xx_test";/*--------------第二部分--------------*/pinctrl-0 = <&pinctrl_tsc_reset>;pinctrl-1 = <&pinctrl_tsc_irq>;reg = <0x5d>;       -----------------①status = "okay";/*--------------第三部分--------------*//*gpio*/reset-gpios = <&gpio3 4 GPIO_ACTIVE_LOW>;irq-gpios = <&gpio5 9 GPIO_ACTIVE_HIGH>;/*--------------第四部分--------------*//*中断*/interrupt-parent = <&gpio5>;interrupts = <9 IRQ_TYPE_EDGE_FALLING>;irq-flags = <2>;                /*1:rising 2: falling*/};
};

结合以上内容简单介绍如下:

第一部分,定义IIC1的一些基本属性,主要包括 IIC1使用的引脚以及IIC1的scl的时钟频率,时钟频率的范围要参考触摸芯片手册,不能超出芯片最大支持的频率。我们使用的GTxxx触摸芯片最大支持400KHz,这里将频率设置为100KHz,满足要求。

第二部分,这里是添加的触摸驱动要使用的中断引脚以及复位引脚。标号①处是触摸芯片在IIC1总线上的地址

第三部分,添加使用的GPIO,这里使用GPIO子系统将触摸芯片使用的irq引脚、rest引脚复用为GPIO.

第四部分,添加中断相关内容,这里将触发方式设置为上升和下降沿触发,具体内容可参考中断章节,这里不再赘述。

2022年8月后购买的4.3寸以及5寸的屏幕触摸芯片都为gt1151,其i2c地址都是0x14,如果是这个日期之后购买的屏幕,需要修改通信地址为14,如下所示。如果是7寸屏幕则无需修改。

确认触摸芯片并修改地址:

     gtxx_tsc@14 {compatible = "fire,gt9xx_test";/*--------------第二部分--------------*/pinctrl-0 = <&pinctrl_tsc_reset>;pinctrl-1 = <&pinctrl_tsc_irq>;reg = <0x14>;       -----------------①status = "okay";/*--------------第三部分--------------*/

2. goodix官方触摸驱动讲解

由于goodix官方触摸驱动稍复杂,这里只讲解实现方法,以及如何简单修改驱动以适配多种触摸芯片,阅读本小节时推荐打开“内核根目录/drivers/input/touchscreen/goodix.c”源码。

2.1. 修改设备树匹配信息

和其他驱动类似,打开官方驱动后首先要找到“设备树匹配”相关内容。如下所示。

在iic1节点追加:

static const struct of_device_id goodix_of_match[] = {{ .compatible = "goodix,gt1151" },{ .compatible = "goodix,gt911" },{ .compatible = "goodix,gt9110" },{ .compatible = "goodix,gt912" },{ .compatible = "goodix,gt927" },{ .compatible = "goodix,gt9271" },{ .compatible = "goodix,gt928" },{ .compatible = "goodix,gt967" },---------------①{ }
};
MODULE_DEVICE_TABLE(of, goodix_of_match);
#endifstatic struct i2c_driver goodix_ts_driver = {   -----②.probe = goodix_ts_probe,.remove = goodix_ts_remove,// .id_table = goodix_ts_id, --------------------③.driver = {.name = "Goodix-TS",-------------------------④.acpi_match_table = ACPI_PTR(goodix_acpi_match),.of_match_table = of_match_ptr(goodix_of_match),.pm = &goodix_pm_ops,},
};

结合以上代码介绍如下。标号①,这里就是用于和设备树节点匹配的匹配值,我们将前面编写的设备树节点添加进去即可。标号②,这个就是i2c设备驱动结构体,它代表了一个I2C设备。标号③处是传统的匹配方式,我们不用可以屏蔽掉。标号④处是驱动的名字,如果内核也开启了GOODIX触摸驱动(默认是开启了) 这里需要修改驱动名字,例如我们例程中将其修改为“Goodix-TS-CHANGE”.

2.2. prob函数实现

.prob函数完成初始化工作,代码如下:

.prob函数:

static int goodix_ts_probe(struct i2c_client *client,const struct i2c_device_id *id)
{struct goodix_ts_data *ts;int error;/*------------------第一部分---------------*/dev_dbg(&client->dev, "I2C Address: 0x%02x\n", client->addr); ------①if (!i2c_check_functionality(client->adapter, I2C_FUNC_I2C)) {  -----②dev_err(&client->dev, "I2C check functionality failed.\n");return -ENXIO;}/*------------------第二部分---------------*/ts = devm_kzalloc(&client->dev, sizeof(*ts), GFP_KERNEL);if (!ts)return -ENOMEM;ts->client = client;i2c_set_clientdata(client, ts);init_completion(&ts->firmware_loading_complete);/*------------------第三部分---------------*/error = goodix_get_gpio_config(ts);  -------------③if (error)return error;if (ts->gpiod_int && ts->gpiod_rst) {/* reset the controller */error = goodix_reset(ts);--------------------④if (error) {dev_err(&client->dev, "Controller reset failed.\n");return error;}}error = goodix_i2c_test(client);----------------⑤if (error) {dev_err(&client->dev, "I2C communication failure: %d\n", error);return error;}error = goodix_read_version(ts);----------------⑥if (error) {dev_err(&client->dev, "Read version failed.\n");return error;}ts->chip = goodix_get_chip_data(ts->id);--------⑦dev_err(&client->dev, " goodix_get_chip_data \n");/*------------------第四部分---------------*/if (ts->gpiod_int && ts->gpiod_rst) {/* update device config */ts->cfg_name = devm_kasprintf(&client->dev, GFP_KERNEL,"goodix_%d_cfg.bin", ts->id); --------⑧if (!ts->cfg_name)return -ENOMEM;error = request_firmware_nowait(THIS_MODULE, true, ts->cfg_name,&client->dev, GFP_KERNEL, ts,goodix_config_cb);--------------------⑨if (error) {dev_err(&client->dev,"Failed to invoke firmware loader: %d\n",error);return error;}return 0;}else{error = goodix_configure_dev(ts);if (error)return error;}return 0;
}

prob函数较长,但是理解起来很简单,也没有什么需要修改的地方,结合源码简单介绍如下:

第一部分,进入.prob函数后做了一些简单的检查,比如,标号①处打印触摸设备的i2c地址。这个地址是在触摸的设备节点中设置的地址。标号②处检查是否支持IIC功能。

第二部分,为goodix_ts_data类型的结构体变量ts申请空间并初始化,在驱动中使用goodix_ts_data结构体保存触摸驱动信息。结构体原型如下。

.goodix_ts_data结构体:

struct goodix_ts_data {struct i2c_client *client;  //i2c 从设备结构体struct input_dev *input_dev; //输入设备结构体const struct goodix_chip_data *chip;  //goodix相关内容struct touchscreen_properties prop;   //未知内容unsigned int max_touch_num;     //做大支持的触摸点unsigned int int_trigger_type;  //触摸类型struct gpio_desc *gpiod_int;    //触摸中断引脚struct gpio_desc *gpiod_rst;    //触摸芯片复位引脚u16 id;                                         //触摸芯片u16 version;                    //版本const char *cfg_name;           //名字struct completion firmware_loading_complete;  //固件加载完成标志unsigned long irq_flags;        //中断标记
};

后面的初始化将会使用这个结构体。结构体成员含义见注释,这里不再赘述。接着回到.prob函数。

第三部分,完成一些基本的初始化。这部分内容调用一些以“goodix_"开头的函数,这些函数是goodix官方实现的一些函数,定义在该文件内,从函数名我们可以大致知道函数的功能,简单说明如下,标号③,获取触摸芯片rst和int使用的GPIO。标号④,复位触摸芯片。标号⑤,测试IIC,尝试与触摸芯片通信 。标号⑥,读取触摸芯片版本,后边会根据触摸芯片版本来初始化触摸芯片。标号⑦,根据标号⑥获取的触摸芯片型号指定触摸的一些配置参数。函数原型如下所示。

 goodix_get_chip_data函数:

    /*------------------第一部分------------------*/
static const struct goodix_chip_data *goodix_get_chip_data(u16 id)
{switch (id) {case 1151:return &gt1x_chip_data;case 911:case 9271:case 9110:case 927:case 928:return &gt911_chip_data;case 912:case 9157:         // fire 新增return &gt9x_chip_data;case 917:         // fire 新增return &gt917_chip_data;case 5688:         // fire 新增return &gt5688_chip_data;case 967:return &gt967_chip_data;default:return &gt9x_chip_data;}
}/*------------------第二部分------------------*//*结构体原型*/
struct goodix_chip_data {u16 config_addr;int config_len;int (*check_config)(struct goodix_ts_data *, const struct firmware *);
};static const struct goodix_chip_data gt9x_chip_data = {.config_addr            = GOODIX_GT9X_REG_CONFIG_DATA,------①.config_len             = GOODIX_CONFIG_MAX_LENGTH,-------------②.check_config           = goodix_check_cfg_8,  -------------③
};/*fire 新增 ----*/
static const struct goodix_chip_data gt917_chip_data = {.config_addr            = GOODIX_GT917_REG_CONFIG_DATA,.config_len             = GOODIX_CONFIG_917_LENGTH,.check_config           = goodix_check_cfg_16,
};/*fire 新增 ----*/
static const struct goodix_chip_data gt5688_chip_data = {.config_addr            = GOODIX_GT917_REG_CONFIG_DATA,.config_len             = GOODIX_CONFIG_5688_LENGTH,.check_config           = goodix_check_cfg_16,
};/*------------------第三部分------------------*/
#define GOODIX_CONFIG_917_LENGTH    242    //fire 新增
#define GOODIX_CONFIG_5688_LENGTH   242    //fire 新增#define GOODIX_GT5688_REG_CONFIG_DATA       0x8050  //fire 新增
#define GOODIX_GT917_REG_CONFIG_DATA        0x8050  //fire 新增

结合源码介绍如下。

第一部分,goodix_get_chip_data函数实现,它很简单,仅仅根据芯片ID返回不同的goodix_chip_data结构体地址。goodix_chip_data结构体原型以及初始化实例如第二部分所示,该结构体共有三个参数,第一个用于指定触摸芯片的配置寄存器地址,这个地址是触摸芯片的内部地 址,不同触摸芯片有所不同,查找触摸手册即可,如第三部分所示。第二个参数用于指定配置信息的最大长度,不同触摸芯片配置信息长度是不同的,根据手册设置即可,这里也通过宏定义指出,如第三部分所示。第三个参数是一个函数指针,用于指定“校验”配置信息的函数。再想触摸芯片写入配置信息之前要校验配置信息。根据触摸芯 片的不同分为8位校验和16位校验。稍后会详细讲解校验函数。接着回到.prob函数。

第四部分,这部分内容是驱动程序的重点。从以上三部分可知,到目前为止我们初始化了触摸芯片使用的引脚并且能够与触摸芯片通信了。这部分的内容完成后续的中断的申请、输入设备的注册、触摸配置信息的读取与更新、触摸事件的上报工作。不过不必担心这部分呢内容几乎不需要我们去修改。

标号⑧处使用devm_kasprintf函数根据触摸芯片的ID合成触摸配置文件的文件名(以下简称为固件)。例如我们使用的GT911,它的ID为911,则触摸更新固件名为“goodix_911_cfg.bin”。

有关更新固件这里简单说明如下。通常情况下我们从供应商那里买到的触摸板已经正确配置了固件,如果买野火的屏幕(带触摸)默认也是配置好了触摸固件,无需进行修改。如果是公司用户批量生产通常情况下也可以和触摸屏供应商沟通让供应商按照你的要求提前烧写好固件。

标号⑨处使用request_firmware_nowait函数从用户空间获取固件。它的最后一个参数是一个函数指针,用于指定获取成功后的回调函数,回调函数原型如下所示:

firmware回调函数goodix_config_cb:

static void goodix_config_cb(const struct firmware *cfg, void *ctx)
{struct goodix_ts_data *ts = ctx;int error;if (cfg) {/* send device configuration to the firmware *//*------------------第一部分------------------*/error = goodix_send_cfg(ts, cfg);if (error)goto err_release_cfg;}/*------------------第二部分------------------*/goodix_configure_dev(ts);err_release_cfg:release_firmware(cfg);complete_all(&ts->firmware_loading_complete);
}

从以上代码可以看出,正常情况下该函数只会执行两个以“goodix_”开头的函数,这两个函数完成了后续的初始化。goodix_send_cfg函数完成触摸芯片更新固件的读取、校验、写入工作。goodix_configure_dev函数完成中断的申请、注册,输入设备的注册、设置上报事件等等工作。最终的触摸事件上报由中断服务函数完成。由于这部分内容较长,我们将这两个函数独立出来讲解,如下所示。

2.3. goodix_send_cfg函数实现

函数原型如下所示:

goodix_send_cfg函数实现:

static int  goodix_send_cfg(struct goodix_ts_data *ts,const struct firmware *cfg)
{int error;
/*------------------第一部分------------------*/error = goodix_check_cfg(ts, cfg);if (error)return error;/*------------------第二部分------------------*/error = goodix_i2c_write(ts->client, ts->chip->config_addr, cfg->data,cfg->size);if (error) {dev_err(&ts->client->dev, "Failed to write config data: %d",error);return error;}dev_dbg(&ts->client->dev, "Config sent successfully.");
/*------------------第三部分------------------*//* Let the firmware reconfigure itself, so sleep for 10ms */usleep_range(10000, 11000);return 0;
}

以上函数功能是校验从应用空间读取得到的触摸更新固件,如果校验通过则调用第二部分的代码将固件写入触摸芯片,我们重点看第一部分的校验函数。函数实现如下所示。

固件校验函数:

static int goodix_check_cfg(struct goodix_ts_data *ts,const struct firmware *cfg)
{/*------------------第一部分------------------*/if (cfg->size > GOODIX_CONFIG_MAX_LENGTH) {dev_err(&ts->client->dev,"The length of the config fw is not correct");return -EINVAL;}/*------------------第二部分------------------*/return ts->chip->check_config(ts, cfg);
}

第一部分,校验固件长度是否大于最大支持的固件长度。官方驱动中这里设置为240由于GT917S和GT5688的固件会超过这个最大值,这里要按照GT917S和GT5688的最大值来计算。最终结果是我们要将“GOODIX_CONFIG_MAX_LENGTH”宏定义的值重新定义为242。

第二部分,调用校验函数。在讲解.prob函数的第三部分,我们根据触摸ID指定了校验函数和地址信息。以GT911为例,如下所示。

gt911的goodix_chip_data结构体:

static const struct goodix_chip_data gt911_chip_data = {.config_addr            = GOODIX_GT9X_REG_CONFIG_DATA,.config_len             = GOODIX_CONFIG_911_LENGTH,.check_config           = goodix_check_cfg_8,
};

可以看到“check_config”是一个函数指针,它指向了“goodix_check_cfg_8”函数,下面将会调用goodix_check_cfg_8函数完成GT911固件的 校验工作,函数实现如下所示。

固件校验函数:

static int goodix_check_cfg_8(struct goodix_ts_data *ts,const struct firmware *cfg)
{int i, raw_cfg_len = cfg->size - 2;u8 check_sum = 0;/*---------------第一部分---------------*/for (i = 0; i < raw_cfg_len; i++)check_sum += cfg->data[i];check_sum = (~check_sum) + 1;/*---------------第二部分---------------*/if (check_sum != cfg->data[raw_cfg_len]) {dev_err(&ts->client->dev,"The checksum of the config fw is not correct");return -EINVAL;}/*---------------第三部分---------------*/if (cfg->data[raw_cfg_len + 1] != 1) {dev_err(&ts->client->dev,"Config fw must have Config_Fresh register set");return -EINVAL;}return 0;
}

校验过程比较简单,与stm32稍有差别。在stm32中我们是计算出配置信息的校验和然后追加到配置信息,然后在最后面添加更新标志。这里读取出来的固件已经加上了校验和并且在固件的最后添加了更新标志,所以这里只需要重新计算校验和并比较是否一致即可代码的第三部分是检测是否有更新标志。

2.4. goodix_configure_dev函数实现

在prob函数的最后会调用两个函数一个是我们上面讲解的固件跟新函数10.2.3 goodix_send_cfg,另外一个是我们这小节要讲解的goodix_configure_dev函数。

总的来说,以上代码是一个IIC设备驱动,它实现了通过IIC1与触摸芯片的通信,但我们最终目标是检测到“按下”或“抬起”事件后通过输入子系统上报给应用层。goodix_configure_dev函数就是用作完成这些后续工作,实现代码如下所示。

完成设备初始化函数:

static int goodix_configure_dev(struct goodix_ts_data *ts)
{int error;/*---------------第一部分---------------*/ts->int_trigger_type = GOODIX_INT_TRIGGER;ts->max_touch_num = GOODIX_MAX_CONTACTS;ts->input_dev = devm_input_allocate_device(&ts->client->dev);if (!ts->input_dev) {dev_err(&ts->client->dev, "Failed to allocate input device.");return -ENOMEM;}ts->input_dev->name = "Goodix Capacitive TouchScreen";ts->input_dev->phys = "input/ts";ts->input_dev->id.bustype = BUS_I2C;ts->input_dev->id.vendor = 0x0416;ts->input_dev->id.product = ts->id;ts->input_dev->id.version = ts->version;/*---------------第二部分---------------*//* Capacitive Windows/Home button on some devices */input_set_capability(ts->input_dev, EV_KEY, KEY_LEFTMETA);input_set_capability(ts->input_dev, EV_ABS, ABS_MT_POSITION_X);input_set_capability(ts->input_dev, EV_ABS, ABS_MT_POSITION_Y);input_set_abs_params(ts->input_dev, ABS_MT_WIDTH_MAJOR, 0, 255, 0, 0);input_set_abs_params(ts->input_dev, ABS_MT_TOUCH_MAJOR, 0, 255, 0, 0);/* Read configuration and apply touchscreen parameters */goodix_read_config(ts);/* Try overriding touchscreen parameters via device properties */touchscreen_parse_properties(ts->input_dev, true, &ts->prop);/*---------------第三部分---------------*/if (!ts->prop.max_x || !ts->prop.max_y || !ts->max_touch_num) {dev_err(&ts->client->dev, "Invalid config, using defaults\n");ts->prop.max_x = GOODIX_MAX_WIDTH - 1;ts->prop.max_y = GOODIX_MAX_HEIGHT - 1;ts->max_touch_num = GOODIX_MAX_CONTACTS;input_abs_set_max(ts->input_dev,ABS_MT_POSITION_X, ts->prop.max_x);input_abs_set_max(ts->input_dev,ABS_MT_POSITION_Y, ts->prop.max_y);}if (dmi_check_system(rotated_screen)) {ts->prop.invert_x = true;ts->prop.invert_y = true;dev_dbg(&ts->client->dev,"Applying '180 degrees rotated screen' quirk\n");}/*---------------第四部分---------------*/error = input_mt_init_slots(ts->input_dev, ts->max_touch_num,INPUT_MT_DIRECT | INPUT_MT_DROP_UNUSED);if (error) {dev_err(&ts->client->dev,"Failed to initialize MT slots: %d", error);return error;}error = input_register_device(ts->input_dev);if (error) {dev_err(&ts->client->dev,"Failed to register input device: %d", error);return error;}/*---------------第五部分---------------*/ts->irq_flags = goodix_irq_flags[ts->int_trigger_type] | IRQF_ONESHOT;error = goodix_request_irq(ts);if (error) {dev_err(&ts->client->dev, "request IRQ failed: %d\n", error);return error;}return 0;
}

函数内容较多,但是这部分内容不需要我们修改,结合源码介绍如下:

第一部分,根据现有参数初始化输入设备结构体input_dev,一个输入设备结构体代表了一输入设备,再注册它之前需要初始化它的一些参数。

第二部分,同样是初始化输入设备结构体,这部分内容用于设置输入设备能够上报的事件类型以及上报事件。从这里部分代码可以看到上报事件类型有EV_KEY按键事件、绝对坐标事件EV_ABS,绝对坐标事件的键值又分为X坐标值和Y坐标值。

函数goodix_read_config 用于从触摸芯片中读取触摸配置信息并用这些配置信息初始化输入设备。

第三部分,检查主要参数是否出错,如果出错则只用默认的参数配置输入设备。

第四部分,输入设备结构体初始化完成后调用input_register_device函数注册输入设备,注册成功后我们就可以向应用层上报输入触摸事件了。

第五部分,调用goodix_request_irq函数完成中断的申请,函数实现如下所示。

中断申请函数:

static int goodix_request_irq(struct goodix_ts_data *ts)
{return devm_request_threaded_irq(&ts->client->dev, ts->client->irq,NULL, goodix_ts_irq_handler,ts->irq_flags, ts->client->name, ts);
}

申请函数使用了devm_request_threaded_irq函数,在驱动中我们会经常看到“devm_"开头的函数,这些函数大多用于注册、申请工作,使用这一类函数注册、申请的内容无需我们手动注销,驱动退出之前系统会自动完成注销。这里我们重点关注中断的处理函数goodix_ts_irq_handle r,触摸中断发生有将会在中断服务函数中上报触摸事件。

中断服务函数如下所示:

中断服务函数:

static irqreturn_t goodix_ts_irq_handler(int irq, void *dev_id)
{struct goodix_ts_data *ts = dev_id;goodix_process_events(ts); -------------------①if (goodix_i2c_write_u8(ts->client, GOODIX_READ_COOR_ADDR, 0) < 0)dev_err(&ts->client->dev, "I2C write end_cmd error\n");return IRQ_HANDLED;
}

标号②处的函数用于处理触摸事件,具体的处理过程这里不再介绍,读者可参考输入子系统章节自行阅读。至此,我么可以知道,当触摸中断发生后将会在中断服务函数中上报输入事件,而应用程序只需要从“/dev/input”目录下对应的节点读取状态即可。

3. 驱动测试

实际使用时我们将LCD和触摸写进一个设备树插件,修改后的设备树 插件保存在“~/embed_linux_driver_tutorial_imx6_code/linux/touch_scream_GTxxx”。4.3寸显示屏使用“imx-fire-lcd43-overlay.dts”设备树插件,5寸和7寸 使用“imx-fire-lcd5-overlay.dts”设备树插件。如何放置设备树插件已经多次介绍,这里不再赘述。

由于系统中默认已经添加了触摸驱动,所以测试之前要将其屏蔽掉。在开发板中打开“/boot/uEnv.txt”, 把LCD和触摸屏的设备树插件 屏蔽掉,添加本实验的设备树插件(这里使用的是5.0寸显示屏)。

03|

屏蔽之后重启开发板,由于内核驱动没有对应的设备树,开发板启动后无法触摸。下一步就要将我们修改的驱动程序通过“insmod”命令加载到系统。

编译本章配套驱动程序,将生成的.ko文件拷贝到开发板。由于系统默认关闭了较低等级的内核打印,为便于观察驱动加载情况,我们需要手动开启所有等级的内核打印,命令如下“echo 8 1 4 7 > /proc/sys/kernel/printk”。注意,数字之间 要有空格,并且系统重启之后要重新执行该命令。执行成功后直接使用insmod 命令加载驱动,如下所示。

03|

驱动成功加载之后输出以上内容,正常情况下这时触摸已经能够正常运行。下面简单说明一下内核打印内容。

  • 标号①是读取得到的原始ID,在驱动中它是一个字符串。

  • 标号②中ID是将原始ID(字符串)转化为数字(字母被忽略),version是版本。

  • 标号③输出的是更新固件的名字,它的格式为“goodix_xxx_cfg.bin”中间是触摸的ID。

  • 标号④,输出获取更新固件失败了,这是因为我们没有添加更新固件。我们之前说过,触摸屏出厂时已经配置好了,所以通常情况下我们无需再次更新。

  • 标号⑤,输出触摸对应的input编号。

更新触摸固件的方法很简单,介绍如下。首先将“~/embed_linux_driver_tutorial_imx6_code/linux/touch_scream_GTxxx”下的固件拷贝到开发板的“/lib/firmware”目录下,然后重启开发板,按照 前面的步骤重新加载驱动即可。添加更新固件后再次加载驱动输入结果如下所示。

04|

从上图可以看出,添加固件后驱动成功获取更新固件,根据输出内容可知,更新固件长度为242,使 用的是16位校验(不同型号触摸更新固件长度和校验位数不同)。

通过如下命令,我们可以查看到系统中的输入设备,如无意外会有event0、event1等设备(这里本实验对应的触摸屏设备是event1)。

ls /dev/input/

下面我们使用hexdump简单检测一下触摸功能。

hexdump /dev/input/event1

执行命令后,在屏幕上轻触,会打印出大量格式化的数据,这些就是触摸数据了。

04|

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

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

相关文章

QML自定义滑动条Slider的样式

代码展示 import QtQuick 2.9 import QtQuick.Window 2.2 import QtQuick.Controls 2.1Window {visible: truewidth: 640height: 480title: qsTr("Hello World")Slider {id: controlvalue: 0.5background: Rectangle {x: control.leftPaddingy: control.topPadding …

【项目开发】C#环境配置及VScode运行C#教程(学生管理系统)

原创文章,禁止转载。 文章目录 下载.NETVScode配置运行程序下载.NET 官网链接: https://dotnet.microsoft.com/en-us/download选择任意版本下载: 下载完成后,双击运行exe文件,等待安装完成。 在控制台输入: dotnet --version若出现版本信息,说明安装成功: VScode配…

卡码网 ACM答题编程模板

背景&#xff1a; input() 在 ACM 编程中的底层调用原理 1. input() 的核心原理 在 Python 中&#xff0c;input() 的底层实现依赖于标准输入流 sys.stdin。每次调用 input() 时&#xff0c;Python 会从 sys.stdin 中读取一行字符串&#xff0c;直到遇到换行符 \n 或文件结束…

Linux驱动开发(18):linux驱动并发与竞态

并发是指多个执行单元同时、并行执行&#xff0c;而并发的执行单元对共享资源(硬件资源和软件上的全局变量、静态变量等)的访问 则很容易导致竞态。对于多核系统&#xff0c;很容易理解&#xff0c;由于多个CPU同时执行&#xff0c;多个CPU同时读、写共享资源时很容易造成竞态。…

k8s基础(3)—Kubernetes-Deployment

一、 Deployment概述 ‌ Kubernetes Deployment‌是Kubernetes中的一个核心概念&#xff0c;它是一种高级别的控制器&#xff0c;用于管理Pod和ReplicaSet&#xff0c;确保应用程序的高可用性和稳定性。Deployment通过声明式配置来创建和更新Pod和ReplicaSet&#xff0c;从而…

windows11(或centos7)安装nvidia显卡驱动、CUDA、cuDNN

本文是我瞎搞时写的问题汇总及参考文献&#xff0c;记录了一些问题解决的进度及对问题的思考。 最近一次更新时间&#xff1a;2025年1月4日 一、安装或更新nvidia显卡驱动 首先&#xff0c;需要确保你的设备安装了最新的显卡驱动。 &#xff08;1&#xff09;centos7安装显…

【光纤通信】光纤结构

光纤结构主要由折射率较高的纤芯&#xff08;core&#xff09;部分和折射率较低的包层&#xff08;cladding&#xff09;部分以及涂覆层&#xff08;buffer coating&#xff09;组成。 光纤的分类方法有很多种&#xff0c;如按纤芯折射率分布、按纤芯结构、按二次涂覆层结构、按…

急需升级,D-Link 路由器漏洞被僵尸网络广泛用于 DDoS 攻击

僵尸网络活动增加 &#xff1a;新的“FICORA”和“CAPSAICIN”僵尸网络&#xff08;Mirai 和 Kaiten 的变体&#xff09;的活动激增。 被利用的漏洞 &#xff1a;攻击者利用已知的 D-Link 路由器漏洞&#xff08;例如 CVE-2015-2051、CVE-2024-33112&#xff09;来执行恶意命…

Eplan 项目结构(高层代号、安装地点、位置代号)

Eplan中的项目结构分为3个层次&#xff1a; &#xff08;1&#xff09;功能面结构。指明这个系统的功能&#xff0c;有什么用途。在EPlan中&#xff0c;指的就是"高层代号&#xff08;&#xff09;"。 一般指的是线体。 &#xff08;2&#xff09;位置面结构。指明该…

网络安全之高防IP的实时监控精准防护

高防IP是一种网络安全设备&#xff0c;用于保护网络服务不受到各类攻击的影响&#xff0c;确保业务的持续稳定运行。它通过监控、识别和封锁恶意攻击流量&#xff0c;提供高级别的防护&#xff0c;降低业务被攻击的风险&#xff0c;并提升网络的稳定性和可靠性。 一、实时监控的…

RabbitMQ实现生产者消费者

一.启动MQ 注意管理员身份进入cmd才行,我这里是在本地安装的MQ,推荐使用虚拟机安装 二.思路 官方解释RabbitMQ结构: 自我理解RabbitMQ结构: 其实RabbitMQ的服务器就像邮局一样,我们的生产者和消费者对于这个服务器来说都是消费者,因为服务器都可以向两者发送消息 环境准备 …

【计算机视觉技术 - 人脸生成】2.GAN网络的构建和训练

GAN 是一种常用的优秀的图像生成模型。我们使用了支持条件生成的 cGAN。下面介绍简单 cGAN 模型的构建以及训练过程。 2.1 在 model 文件夹中新建 nets.py 文件 import torch import torch.nn as nn# 生成器类 class Generator(nn.Module):def __init__(self, nz100, nc3, n…

matlab中高精度计算函数vpa与非厄米矩阵本征值的求解

clear;clc;close all tic %并行设置% delete(gcp(nocreate));%关闭之前的并行 cparcluster(local); c.NumWorkers50;%手动设置线程数(否则默认最大线程为12) parpool(c, c.NumWorkers); %并行设置%w1; u2.5;N30;valstozeros(2*N2,100); v10linspace(-3,3,100).;parfor jj1:leng…

旧服务改造及微服务架构演进

旧服务改造及微服务架构演进 微服务架构演进1.微服务架构2.微服务架构的特点3.单体架构与微服务架构之间的对比4.微服务架构演进历程 旧服务改造1. 微服务拆分的一些通用原则2.微服务拆分策略&#xff08;1&#xff09;功能维度拆分策略&#xff08;2&#xff09;非功能维度拆分…

springmvc--请求参数的绑定

目录 一、创建项目&#xff0c;pom文件 二、web.xml 三、spring-mvc.xml 四、index.jsp 五、实体类 Address类 User类 六、UserController类 七、请求参数解决中文乱码 八、配置tomcat,然后启动tomcat 1. 2. 3. 4. 九、接收Map类型 1.直接接收Map类型 &#x…

Navicat 17 for Mac 数据库管理软件

Mac分享吧 文章目录 效果一、准备工作二、开始安装1. 双击运行软件&#xff0c;将其从左侧拖入右侧文件夹中&#xff0c;等待安装完毕。2. 应用程序/启动台显示Navicat图标&#xff0c;表示安装成功。 二、运行测试运行后提示&#xff1a;“Navicat Premium.pp”已损坏&#x…

Lombok @Data无法 import 导入的问题解决办法

问题原因&#xff1a;Maven Pom中依赖的Lombok版本与安装在eclipse根目录下的Lombok版本不一致 解决办法&#xff1a;找到Maven Pom依赖版本的Lombok Jar包&#xff0c;执行命令&#xff1a;java -jar lombok-1.18.36.jar&#xff0c;运行如下图&#xff0c;然后安装即可。这样…

对计网大题的一些指正(中间介绍一下CDM的原理和应用)

目录 前言&#xff1a; &#xff08;1&#xff09;五层原理体系结构每层功能&#xff1a; 下面是文档的答案&#xff1a; 我在之前的博客里面有介绍过五层原理体系结构&#xff0c; 按理来说&#xff0c;第五层应该是应用层才对&#xff0c;而会话层的功能应该被放到应用层…

RISC-V学习笔记

1.RISC ISA1个基本整数指令集多个可选的扩展指令集&#xff0c;如RV32I表示支持32位整数指令集。I表示基本指令集&#xff0c;M表示整数乘法与除法指令集&#xff0c;A表示存储器原子指令集&#xff0c;F表示单精度浮点指令集&#xff0c;D表示双精度浮点指令集等&#xff0c;C…

SpringBoot入门之创建一个Hello World项目

文章目录 一、使用传统的方式1、创建一个SpringBoot项目2、配置pom.xml文件3、下载Maven依赖4、创建一个Controller类&#xff1a;com.devops.controller.HelloController5、创建一个引导类&#xff1a;com.devops.HelloApplication6、启动项目8、访问80809、完整项目结构 二、…