RK3288 android7.1 适配 ilitek i2c接口TP
一,Ilitek 触摸屏简介
-
Ilitek 提供多种型号的触控屏控制器,如 ILI6480、ILI9341 等,采用 I2C 接口。
这些控制器能够支持多点触控,并具有优秀的灵敏度和响应速度。 -
Ilitek 的触摸屏控制器监测屏幕上的触摸事件。
当触摸发生时,控制器将触摸坐标(通常是 x 和 y 坐标)通过 I2C 总线发送给主机。
二,DTS配置
&i2c1 {status = "okay";ilitek@41 {compatible = "tchip,ilitek";reg = <0x41>;ilitek,irq-gpio = <&gpio7 6 GPIO_ACTIVE_HIGH>;ilitek,reset-gpio = <&gpio7 15 GPIO_ACTIVE_HIGH>; //修改相对应的gpio参数pinctrl-names = "default";pinctrl-0 = <&ilitek_gpio_e>;status = "okay";};
};&pinctrl {touchscreen {ilitek_gpio_e: ilitek-gpio-e {rockchip,pins =<7 6 RK_FUNC_GPIO &pcfg_pull_up>,<7 15 RK_FUNC_GPIO &pcfg_pull_up>;};};
};
注:一定要对应的I2C bus节点(硬件原理图上会有标注)。
三,驱动文件配置
1. ilitek_lim.c
static int ilitek_i2c_probe(struct i2c_client *client,const struct i2c_device_id *id)
{int ret = 0;int revert_x=0;int revert_y=0;int revert_x_y=0;struct device_node *param_node;
#ifdef ILI_UPDATE_FW#ifdef UPDATE_THREADEstruct task_struct *thread_update = NULL;#endif
#endif#ifdef RECOGNITION_LONGTIME_BUTTON
struct task_struct *thread_longtime_button = NULL;
#endif#ifdef WRITE_2210_CFGstruct task_struct *thread_write_cfg = NULL;
#endif#if IC2120int i = 0;
#endif//int i = 0;//uint8_t buf[8] = {0};
#ifdef DEBUG_NETLINKstruct netlink_kernel_cfg cfg = {.groups = 0,.input = udp_receive,};#endiftp_log_info("Enter ilitek_i2c_probe +++++++++ client->addr = 0x%x++++++\n", client->addr);if(!i2c_check_functionality(client->adapter, I2C_FUNC_I2C)){tp_log_err("%s, I2C_FUNC_I2C not support\n", __func__);return -1;}// initialize global variable#ifdef TOOLmemset(&dev_ilitek, 0, sizeof(struct dev_data));#endifmemset(&i2c, 0, sizeof(struct i2c_data));// initialize mutex object
#if LINUX_VERSION_CODE < KERNEL_VERSION(2, 6, 37)init_MUTEX(&i2c.wr_sem);
#elsesema_init(&i2c.wr_sem,1);
#endifi2c.wr_sem.count = 1;mutex_init(&(i2c.irq_mutex));i2c.report_status = 1;i2c.client = client;#ifdef CONFIG_ENABLE_REGULATOR_POWER_ONret = ilitek_get_regulator();if (ret) {tp_log_err("%s, ilitek_get_regulator failed\n", __func__);return ret;}ret = ilitek_power_on(true);if (ret) {tp_log_err("%s, power on failed\n", __func__);return ret;}#endif//ret = ilitek_request_io_port(client);if (ret != 0){tp_log_err("%s, io error", __func__);return ret;}param_node = of_find_node_by_path("/smdt_param");if (!param_node){printk("gt9xx wxl error smdt_param not node!!!\n");}of_property_read_u32(param_node, "ctp_revert_x_flag", &revert_x);of_property_read_u32(param_node, "ctp_revert_y_flag", &revert_y);of_property_read_u32(param_node, "ctp_exchange_x_y_flag", &revert_x_y);if(revert_x == 1 )REVERT_X = 1;else REVERT_X = 0;if(revert_y == 1 )REVERT_Y = 1;else REVERT_Y = 0;if(revert_x_y == 1 )EXCHANG_XY = 1;else EXCHANG_XY = 0;i2c.client->irq = gpio_to_irq(i2c.irq_gpio);tp_log_info("ilitek i2c.irq_gpio = %d, i2c.client->irq = %d\n", i2c.irq_gpio, i2c.client->irq);i2c.valid_i2c_register = 1;#if 1ilitek_reset(i2c.reset_gpio);#endif#if !IC2120mdelay(200);#elsefor (i = 0; i < 30; i++ ) {ret = ilitek_poll_int();tp_log_info("ilitek int status = %d\n", ret);if (ret == 0) {break;}else {mdelay(5);}}if (i >= 30) {#ifdef ILI_UPDATE_FWtp_log_info("ilitek reset but int not pull low so driver_upgrade_flag = true\n");driver_upgrade_flag = true;#endif}else #endif{#if 0msleep(2000);for (i = 0; i < 127; i++) {//buf[0] = 0x10;client->addr = i;ret = ilitek_i2c_write_and_read(i2c.client, buf, 0, 10, buf, 4);tp_log_info("ilitek %s, write 0x10 read buf = %X, %X, %X\n", __func__, buf[0], buf[1], buf[2]);if (ret < 0) {mdelay(5);}else {tp_log_info("ilitek %s, client->addr = 0x%X, read buf = %X, %X, %X\n", __func__, client->addr, buf[0], buf[1], buf[2]);tp_log_info("ilitek %s, client->addr = 0x%X, read buf = %X, %X, %X\n", __func__, client->addr, buf[0], buf[1], buf[2]);tp_log_info("ilitek %s, client->addr = 0x%X, read buf = %X, %X, %X\n", __func__, client->addr, buf[0], buf[1], buf[2]);tp_log_info("ilitek %s, client->addr = 0x%X, read buf = %X, %X, %X\n", __func__, client->addr, buf[0], buf[1], buf[2]);tp_log_info("ilitek %s, client->addr = 0x%X, read buf = %X, %X, %X\n", __func__, client->addr, buf[0], buf[1], buf[2]);break;}}#endif// read touch parameterret = ilitek_i2c_read_tp_info();if(ret < 0){tp_log_err("ilitek read tp info fail free gpio\n");if (gpio_is_valid(i2c.irq_gpio)) {gpio_free(i2c.irq_gpio);}if (gpio_is_valid(i2c.reset_gpio)) {gpio_free(i2c.reset_gpio);}return ret;}}
#ifdef CLOCK_INTERRUPT
#ifdef REPORT_THREADi2c.thread = kthread_run(ilitek_i2c_touchevent_thread, NULL, "ilitek_i2c_thread");if(i2c.thread == (struct task_struct*)ERR_PTR){i2c.thread = NULL;tp_log_err("%s, kthread create, error\n", __func__);}
#endif
#endifilitek_handle_irqorpolling();i2c.input_dev = input_allocate_device();if(i2c.input_dev == NULL){tp_log_err("%s, allocate input device, error\n", __func__);return -1;}ilitek_set_input_param(i2c.input_dev, i2c.max_tp, i2c.max_x, i2c.max_y);ret = input_register_device(i2c.input_dev);if(ret){tp_log_err("%s, register input device, error\n", __func__);return ret;}i2c.valid_input_register = 1;tp_log_info("%s, register input device, success\n", __func__);
#ifdef ILI_UPDATE_FW#ifndef UPDATE_THREADE{update_wait_flag = 1;ret = ilitek_upgrade_firmware();if(ret == ILITEK_UPDATE_OK) {tp_log_info("update end\n");}else if(ret == ILITEK_I2C_TRANSFER_ERR) {tp_log_info("i2c communication error\n");}if(i2c.reset_request_success){ilitek_reset(i2c.reset_gpio);}// read touch parameterret=ilitek_i2c_read_tp_info();if(ret < 0){tp_log_err("ilitek_i2c_read_tp_info err\n");return ret;}update_wait_flag = 0;#if 1input_unregister_device(i2c.input_dev);#if 1i2c.input_dev = input_allocate_device();if(i2c.input_dev == NULL){tp_log_err("%s, allocate input device, error\n", __func__);return -1;}#endif// register input device//memset(i2c.input_dev, 0, sizeof(struct input_dev));// register input deviceilitek_set_input_param(i2c.input_dev, i2c.max_tp, i2c.max_x, i2c.max_y);ret = input_register_device(i2c.input_dev);if(ret){tp_log_err("%s, register input device, error\n", __func__);return ret;}tp_log_info("%s, register input device, success\n", __func__);#endif}#else{thread_update= kthread_run(ilitek_i2c_update_thread, NULL, "ilitek_i2c_updatethread");if(thread_update == (struct task_struct*)ERR_PTR){thread_update = NULL;tp_log_err("%s,thread_update kthread create, error\n", __func__);}}#endif
#endif#ifdef WRITE_2210_CFGthread_write_cfg = kthread_run(ilitek_i2c_write_cfg_thread, NULL, "ilitek_i2c_write_cfg_thread");if(thread_write_cfg == (struct task_struct*)ERR_PTR){thread_write_cfg = NULL;tp_log_err("%s,thread_write_cfg kthread create, error\n", __func__);}
#endif#ifdef RECOGNITION_LONGTIME_BUTTONthread_longtime_button = kthread_run(ilitek_i2c_check_longtime_button_thread, NULL, "ilitek_i2c_check_longtime_button_thread");if(thread_longtime_button == (struct task_struct*)ERR_PTR){thread_longtime_button = NULL;tp_log_err("%s,thread_longtime_button kthread create, error\n", __func__);}
#endif#ifdef GESTUREilitek_system_resume = 1;input_set_capability(i2c.input_dev, EV_KEY, KEY_POWER);input_set_capability(i2c.input_dev, EV_KEY, KEY_W);input_set_capability(i2c.input_dev, EV_KEY, KEY_LEFT);input_set_capability(i2c.input_dev, EV_KEY, KEY_RIGHT);input_set_capability(i2c.input_dev, EV_KEY, KEY_UP);input_set_capability(i2c.input_dev, EV_KEY, KEY_DOWN);input_set_capability(i2c.input_dev, EV_KEY, KEY_O);input_set_capability(i2c.input_dev, EV_KEY, KEY_C);input_set_capability(i2c.input_dev, EV_KEY, KEY_E);input_set_capability(i2c.input_dev, EV_KEY, KEY_M);
#endif#ifdef TOOLret = create_tool_node();
#endif#ifdef SENSOR_TEST
#ifdef SYS_ATTR_FILEilitek_sensor_test_init();
#endif
#endif#ifdef HALL_CHECK
#if HALL_CHECK == HALL_CHECK_OTHERilitek_hall_check_init();
#elif HALL_CHECK == HALL_CHECK_HWilitek_hall_check_hw_init();
#endif //end of HALL_CHECK
#endifReport_Flag=0;#ifdef PLAT_ROCKCHIPi2c.tp.tp_resume = ilitek_i2c_late_resume;i2c.tp.tp_suspend = ilitek_i2c_early_suspend;tp_register_fb(&i2c.tp);
#endif#ifdef ILITEK_ESD_CHECKINIT_DELAYED_WORK(&esd_work, ilitek_touch_esd_func);esd_wq = create_singlethread_workqueue("esd_wq"); if (!esd_wq) {return -ENOMEM;}queue_delayed_work(esd_wq, &esd_work, delay);
#endif #ifdef WRITE_2210_CFGret = sysfs_create_group(&client->dev.kobj, &ilitek_test_attribute_group);if (ret) {dev_err(&client->dev, "failed to create sysfs attributes: %d\n",ret);return ret;}
#endif
#ifdef DEBUG_NETLINK//netlink_sock = netlink_kernel_create(&init_net, 21, 0,udp_receive, NULL, THIS_MODULE);netlink_sock = netlink_kernel_create(&init_net, 21, &cfg);#endifdevice_init_wakeup(&client->dev, 1);return 0;
}
ilitek_i2c_probe 函数是 I2C 设备的初始化函数,在设备被识别和加载驱动时调用。它的主要职责包括:
- 初始化设备:根据 I2C 客户端信息和设备 ID 来配置和初始化设备。
- 分配所需资源:如内存、GPIO 等。
- 设置中断处理程序(如适用):为设备配置中断。
- 注册输入设备:根据设备类型设置输入设备的相关参数。
2. ilitek_ts.h
ilitek_ts.h用于定义驱动相关的数据结构、常量、函数原型以及其他必要的包含。该头文件的内容用于抽象和管理与 Ilitek 触摸屏控制器之间的接口。
3. ILI25110CN080O0003_CRC.ili
ILI25110CN080O0003_CRC.ili 可能是与 Ilitek 触摸屏控制器 ILI2511 系列相关的文件。
4. 驱动移植
a.将原厂提供的的驱动(ilitek)拷贝到下面的文件夹:
./kernel/drivers/input/touchscreen
b. 然后修改 touchscreen 目录下的 Makefile以及Kconfig。这个Makefile中定义的就是要编译哪个平台的触摸屏驱动的。
kernel/drivers/input/touchscreen/Makefile中添加:
obj-$(CONFIG_TOUCHSCREEN_ILITEK) += ilitek/--------------------------------------------------------
kernel/drivers/input/touchscreen/Kconfig中添加:
config TOUCHSCREEN_ILITEKtristate "ILITEK touchscreens matched with screen support for rockchip platform"depends on I2C && ARCH_ROCKCHIPhelpSay Y here if you have a touchscreen interface using the ILITEKtouchscreen chip on Rockchip platform, and your board-specificinitialization code includes that in its table of IIC devices.If unsure, say N.
c. 内核中加载驱动:
kernel/arch/arm/configs/rockchip_defconfig中添加:
CONFIG_TOUCHSCREEN_ILITEK=y
四,调试
- 确认 /sys/bus/i2c/devices 下有裝置节点 (-0041);
- 报点问题:
有触摸效果,只是坐标 mapping 问题。
X、Y 需要交換。
將 ILITEK_ROTATE_FLAG 设定值由 0 改为 1 或由 1 改为 0。
X、Y 值要做鏡像变化。
將 ILITEK_REVERT_X 或 ILITEK_REVERT_Y 的设定值由 0 改为 1 或由 1 改为 0。
若需要指定显示屏的解析度。
则开放 ILITEK_USE_LCM_RESOLUTION 宏,同时將TOUCH_SCREEN_X_MAX 和 TOUCH_SCREEN_Y_MAX 设为正确值。
相关文章:

RK3288 android7.1 适配 ilitek i2c接口TP
一,Ilitek 触摸屏简介 Ilitek 提供多种型号的触控屏控制器,如 ILI6480、ILI9341 等,采用 I2C 接口。 这些控制器能够支持多点触控,并具有优秀的灵敏度和响应速度。 Ilitek 的触摸屏控制器监测屏幕上的触摸事件。 当触摸发生时&am…...

C++ 越来越像函数式编程了!
C 越来越像函数式编程了 大家好,欢迎来到今天的博客话题。今天我们要聊的是 C 这门老牌的强类型语言是如何一步一步向函数式编程靠拢的。从最早的函数指针,到函数对象(Functor),再到 std::function 和 std::bind&…...

maven工程结构说明
1、maven工程文件目录 |-- pom.xml # Maven 项目管理文件 |-- src # 放项目源文件|-- main # 项目主要代码| |-- java # Java 源代码目录| | -- com/example/myapp…...

【GESP】C++一级真题练习(202312)luogu-B3921,小杨的考试
GESP一级真题练习。为2023年12月一级认证真题。逻辑计算问题。 题目题解详见:【GESP】C一级真题练习(202312)luogu-B3921,小杨的考试 | OneCoder 【GESP】C一级真题练习(202312)luogu-B3921,小杨的考试 | OneCoderGESP一级真题练习。为2023…...
游戏中Dubbo类的RPC设计时的注意要点
一.消费方 1.需要使用到动态代理,代理指定的接口,这样子接口被调用时,就可以拿到:"类名 方法名参数返回值" 这些类型。 2.既然是rpc,那么接口被调用时,肯定在动态代理中会进行网络消息的发送&a…...

ARXML汽车可扩展标记性语言规范讲解
ARXML: Automotive Extensible Markup Language (汽车可扩展标记语言) xmlns: Xml name space (xml 命名空间) xsd: Xml Schema Definition (xml 架构定义) 1、XML与HTML的区别,可扩展。 可扩展,主要是…...

Hadoop(HDFS)
Hadoop是一个开源的分布式系统架构,旨在解决海量数据的存储和计算问题,Hadoop的核心组件包括Hadoop分布式文件系统(HDFS)、MapReduce编程模型和YARN资源管理器,最近需求需要用到HDFS和YARN。 文章目录 HDFS优缺点HDFS的读写原理 常…...
机器学习系列----梯度下降算法
梯度下降算法(Gradient Descent)是机器学习和深度学习中最常用的优化算法之一。无论是在训练神经网络、线性回归模型,还是其他类型的机器学习模型时,梯度下降都是不可或缺的一部分。它的核心目标是最小化一个损失函数(…...
AI大模型:软件开发的未来之路
随着AI技术的快速发展,AI大模型正在对软件开发流程产生深远的影响。从代码自动生成到智能测试,AI大模型正在重塑软件开发的各个环节,为软件开发者、企业和整个产业链带来新的流程和模式变化。 首先,AI大模型的定义是指通过大规模…...

指标+AI+BI:构建数据分析新范式丨2024袋鼠云秋季发布会回顾
10月30日,袋鼠云成功举办了以“AI驱动,数智未来”为主题的2024年秋季发布会。大会深度探讨了如何凭借 AI 实现新的飞跃,重塑企业的经营管理方式,加速数智化进程。 作为大会的重要环节之一,袋鼠云数栈产品经理潮汐带来了…...

2024年第四届“网鼎杯”网络安全比赛---朱雀组Crypto- WriteUp
2024年第四届“网鼎杯”网络安全比赛---朱雀组Crypto-WriteUp Crypto:Crypto-2:Crypto-3: 前言:本次比赛已经结束,用于赛后复现,欢迎大家交流学习! Crypto: Crypto-2: …...

关于Markdown的一点疑问,为什么很多人说markdown比word好用?
markdown和word压根不是一类工具,不存在谁比谁好,只是应用场景不一样。 你写博客、写readme肯定得markdown,但写合同、写简历肯定word更合适。 markdown和word类似邮箱和微信的关系,这两者都可以通信,但微信因为功能…...
NoSQL大数据存储技术测试(1)绪论
写在前面:未完成测试的同学,请先完成测试,此博文供大家复习使用,(我的答案)均为正确答案,大家可以放心复习 单项选择题 第1题 以下不属于云计算部署模型的是( ) 公…...
Linux命令学习,git命令
Linux系统,Git是一个强大的版本管理系统,允许用户跟踪代码的更改、管理项目历史以及与他人协作。 Linux Git命令: 初始化仓库:当前目录创建一个Git仓库,生成.git隐藏目录存储版本历史和其他Git相关的元数据。 git init 克隆仓库…...

【AI大模型】Transformer中的编码器详解,小白必看!!
前言 Transformer中编码器的构造和运行位置如下图所示,其中编码器内部包含多层,对应下图encoder1…encoder N,每个层内部又包含多个子层:多头自注意力层、前馈神经网络层、归一化层,而最关键的是多头自注意力层。 自注…...
PostgreSQL 字段按逗号分隔成多条数据的技巧与实践 ️
全文目录: 开篇语前言 📚1. PostgreSQL 字段拆分的基本概念 🎯2. 使用 string_to_array 函数拆分字段 💬示例:使用 string_to_array 拆分字段结果: 3. 使用 unnest 和 string_to_array 结合拆分 ǵ…...
设计模式学习总结(一)
设计模式学习笔记 面向对象、设计原则、设计模式、编程规范、重构之间的关系 面向对象、设计原则、设计模式、编程规范、重构之间的关系 面向对象 现在,主流的编程范式或者是编程风格有三种:面向过程、面向对象和函数式编程。 需要掌握七大知识点&#…...

软考中级 软件设计师 上午考试内容笔记(个人向)Part.1
软考上午考试内容 1. 计算机系统 计算机硬件通过高/低电平来模拟1/0信息;【p进制】: K n K n − 1 . . . K 2 K 1 K 0 K − 1 K − 2... K − m K n r n . . . K 1 r 1 K 0 r 0 K − 1 r − 1 . . . K − m r − m K_nK_{n-1}...K_2K_1K_0K…...
PHP API的数据交互类型设计
PHP API的数据交互类型设计涉及多个方面,包括请求方法、数据格式、安全性考虑等。以下是对PHP API数据交互类型设计的详细探讨: 一、请求方法 在PHP API中,常见的请求方法包括GET、POST、PUT、DELETE等。这些方法在数据交互中各有其用途和特…...

【EFK】Linux集群部署Elasticsearch最新版本8.x
【EFK】Linux集群部署Elasticsearch最新版本8.x 摘要环境准备环境信息系统初始化启动先决条件 下载&安装修改elasticsearch.yml控制台启动Linux服务启动访问验证查看集群信息查看es健康状态查看集群节点查询集群状态 生成service token验证service tokenIK分词器下载 摘要 …...
Python爬虫实战:研究MechanicalSoup库相关技术
一、MechanicalSoup 库概述 1.1 库简介 MechanicalSoup 是一个 Python 库,专为自动化交互网站而设计。它结合了 requests 的 HTTP 请求能力和 BeautifulSoup 的 HTML 解析能力,提供了直观的 API,让我们可以像人类用户一样浏览网页、填写表单和提交请求。 1.2 主要功能特点…...

eNSP-Cloud(实现本地电脑与eNSP内设备之间通信)
说明: 想象一下,你正在用eNSP搭建一个虚拟的网络世界,里面有虚拟的路由器、交换机、电脑(PC)等等。这些设备都在你的电脑里面“运行”,它们之间可以互相通信,就像一个封闭的小王国。 但是&#…...

【Python】 -- 趣味代码 - 小恐龙游戏
文章目录 文章目录 00 小恐龙游戏程序设计框架代码结构和功能游戏流程总结01 小恐龙游戏程序设计02 百度网盘地址00 小恐龙游戏程序设计框架 这段代码是一个基于 Pygame 的简易跑酷游戏的完整实现,玩家控制一个角色(龙)躲避障碍物(仙人掌和乌鸦)。以下是代码的详细介绍:…...
应用升级/灾备测试时使用guarantee 闪回点迅速回退
1.场景 应用要升级,当升级失败时,数据库回退到升级前. 要测试系统,测试完成后,数据库要回退到测试前。 相对于RMAN恢复需要很长时间, 数据库闪回只需要几分钟。 2.技术实现 数据库设置 2个db_recovery参数 创建guarantee闪回点,不需要开启数据库闪回。…...
java_网络服务相关_gateway_nacos_feign区别联系
1. spring-cloud-starter-gateway 作用:作为微服务架构的网关,统一入口,处理所有外部请求。 核心能力: 路由转发(基于路径、服务名等)过滤器(鉴权、限流、日志、Header 处理)支持负…...
Java如何权衡是使用无序的数组还是有序的数组
在 Java 中,选择有序数组还是无序数组取决于具体场景的性能需求与操作特点。以下是关键权衡因素及决策指南: ⚖️ 核心权衡维度 维度有序数组无序数组查询性能二分查找 O(log n) ✅线性扫描 O(n) ❌插入/删除需移位维护顺序 O(n) ❌直接操作尾部 O(1) ✅内存开销与无序数组相…...
c++ 面试题(1)-----深度优先搜索(DFS)实现
操作系统:ubuntu22.04 IDE:Visual Studio Code 编程语言:C11 题目描述 地上有一个 m 行 n 列的方格,从坐标 [0,0] 起始。一个机器人可以从某一格移动到上下左右四个格子,但不能进入行坐标和列坐标的数位之和大于 k 的格子。 例…...

在WSL2的Ubuntu镜像中安装Docker
Docker官网链接: https://docs.docker.com/engine/install/ubuntu/ 1、运行以下命令卸载所有冲突的软件包: for pkg in docker.io docker-doc docker-compose docker-compose-v2 podman-docker containerd runc; do sudo apt-get remove $pkg; done2、设置Docker…...

什么是Ansible Jinja2
理解 Ansible Jinja2 模板 Ansible 是一款功能强大的开源自动化工具,可让您无缝地管理和配置系统。Ansible 的一大亮点是它使用 Jinja2 模板,允许您根据变量数据动态生成文件、配置设置和脚本。本文将向您介绍 Ansible 中的 Jinja2 模板,并通…...
return this;返回的是谁
一个审批系统的示例来演示责任链模式的实现。假设公司需要处理不同金额的采购申请,不同级别的经理有不同的审批权限: // 抽象处理者:审批者 abstract class Approver {protected Approver successor; // 下一个处理者// 设置下一个处理者pub…...