Linux下的触摸屏驱动

发布者:心怀感恩最新更新时间:2023-01-11 来源: elecfans关键字:Linux  触摸屏驱动 手机看文章 扫描二维码
随时随地手机看文章

一.触摸屏理论概述

对于触摸屏驱动,我们主要需要掌握触摸屏驱动代码和应用层测试代码。下面讲的是基于Mini2440的触摸屏驱动,现在的驱动我们都将设备和驱动分离,挂在平台设备总线上,让设备和驱动去匹配。而我们在linu2.6.32.2内核版本中的触摸屏驱动仍然没有将设备和驱动分离,这样就不存在匹配问题,这种现象其实我们并不陌生,在我们学习驱动的前期,都会研究简单字符驱动代表LED驱动,那个驱动就是把设备和驱动写在了一起。总结下,驱动和设备可以分离也可以不分离,建议分离,而本触摸屏驱动没有分离设备和驱动,有兴趣可以将设备和驱动进行分离。


先说明下触摸屏的工作原理,当有人在触摸屏上按下触笔时,触摸屏的四个引脚会产生不同的电压值,这样触摸屏控制器就能检测到这种变化,从而产生INT_TC中断,表示触笔按下。然后在得到CPU指示的情况下,触摸屏控制器可以根据四个引脚上产生的不同电压值进行AD转换,从而计算出X和Y坐标的数值,并在将这两个值保持到其内部寄存器后,发出INT_ADC中断,表示坐标转换已完成,从而软件就可以读取按下触笔的位置。


二.触摸屏驱动分析

本驱动分析很有特点,我对触摸屏驱动的分析是按照整个触摸事件的执行顺序进行代码分析的,有的函数由于每次被执行完成的任务不同,所以需要多次分析。同时,我把整个触摸事件的来龙去脉也都说的很清楚了。


驱动分析/driver/input/touchscreen/s3c2410_ts.c

staTIc int __init s3c2410ts_init(void)

{

struct input_dev *input_dev;

adc_clock = clk_get(NULL, "adc");  //获取时钟

if (!adc_clock) {

printk(KERN_ERR "failed to get adc clock source ");

return -ENOENT;

}

clk_enable(adc_clock);   //使能时钟

base_addr=ioremap(S3C2410_PA_ADC,0x20);  //物理地址转为虚拟地址

if (base_addr == NULL) {

printk(KERN_ERR "Failed to remap register block ");

return -ENOMEM;

}

s3c2410_ts_connect();  //触摸屏端口配置

//使能预分频,分频系数为0xff

iowrite32(S3C2410_ADCCON_PRSCEN| S3C2410_ADCCON_PRSCVL(0xFF),base_addr+S3C2410_ADCCON);

iowrite32(0xffff,  base_addr+S3C2410_ADCDLY); //延时

//检查光标按下中断信号,等待中断

iowrite32(WAIT4INT(0), base_addr+S3C2410_ADCTSC);

input_dev = input_allocate_device();  //分配input设备

if (!input_dev) {

printk(KERN_ERR "Unable to allocate the input device !! ");

return -ENOMEM;

}

dev = input_dev;

//支持按键事件、坐标事件

dev->evbit[0] = BIT(EV_SYN) | BIT(EV_KEY) | BIT(EV_ABS);

dev->keybit[BITS_TO_LONGS(BTN_TOUCH)] = BIT(BTN_TOUCH);

//对于X轴范围是0-ox3ff,数据误差是0,中心平滑位置是0

input_set_abs_params(dev, ABS_X, 0, 0x3FF, 0, 0);

input_set_abs_params(dev, ABS_Y, 0, 0x3FF, 0, 0);

input_set_abs_params(dev, ABS_PRESSURE, 0, 1, 0, 0);

dev->name = s3c2410ts_name;

dev->id.bustype = BUS_RS232;

dev->id.vendor = 0xDEAD;

dev->id.product = 0xBEEF;

dev->id.version = S3C2410TSVERSION;

//申请AD转换中断

if(request_irq(IRQ_ADC,stylus_acTIon,IRQF_SHARED|IRQF_SAMPLE_RANDOM,"s3c2410_acTIon", dev)) {

printk(KERN_ERR "s3c2410_ts.c: Could not allocate ts IRQ_ADC ! ");

iounmap(base_addr);

return -EIO;

}

//申请触摸中断

if (request_irq(IRQ_TC, stylus_updown, IRQF_SAMPLE_RANDOM,

"s3c2410_acTIon", dev)) {

printk(KERN_ERR "s3c2410_ts.c: Could not allocate ts IRQ_TC ! ");

iounmap(base_addr);

return -EIO;

}

printk(KERN_INFO "%s successfully loaded ", s3c2410ts_name);

input_register_device(dev);

return 0;

}

下面是这个模块加载函数中调用的一个配置端口函数

static inline void s3c2410_ts_connect(void)

{

//将触摸屏用到的四个端口配置成触摸屏模式

s3c2410_gpio_cfgpin(S3C2410_GPG(12), S3C2410_GPG12_XMON);

s3c2410_gpio_cfgpin(S3C2410_GPG(13), S3C2410_GPG13_nXPON);

s3c2410_gpio_cfgpin(S3C2410_GPG(14), S3C2410_GPG14_YMON);

s3c2410_gpio_cfgpin(S3C2410_GPG(15), S3C2410_GPG15_nYPON);

}

我们来分析两种情况,第一种情况,如果没有按下触摸屏

驱动中定义了一个定时器

static struct timer_list touch_timer =

TIMER_INITIALIZER(touch_timer_fire, 0, 0);

因为这个定时器的期限时间设置为0,这表示当驱动加载后就会执行一次定时函数touch_timer_fire

static void touch_timer_fire(unsigned long data)

{

unsigned long data0;

unsigned long data1;

int updown;

data0 = ioread32(base_addr+S3C2410_ADCDAT0);  //读取X坐标

data1 = ioread32(base_addr+S3C2410_ADCDAT1);  //读取Y坐标

updown = (!(data0 & S3C2410_ADCDAT0_UPDOWN)) && (!(data1 & S3C2410_ADCDAT0_UPDOWN)); //触摸屏是否被按下,如果按下updowm=1

if (updown) {

if (count != 0) {

long tmp;                                                                                        

tmp = xp;

xp = yp;

yp = tmp;                                                                                        

xp >>= 2;

yp >>= 2;

input_report_abs(dev, ABS_X, xp);

input_report_abs(dev, ABS_Y, yp);

input_report_key(dev, BTN_TOUCH, 1);

input_report_abs(dev, ABS_PRESSURE, 1);

input_sync(dev);

}

xp = 0;    

yp = 0;

count = 0;

iowrite32(S3C2410_ADCTSC_PULL_UP_DISABLE | AUTOPST, base_addr+S3C2410_ADCTSC);

iowrite32(ioread32(base_addr+S3C2410_ADCCON) | S3C2410_ADCCON_ENABLE_START, base_addr+S3C2410_ADCCON);

} else { //没有被按下

count = 0;        //初始化count为0,表示当前AD转换没发生

input_report_key(dev, BTN_TOUCH, 0);  //向input子系统报告未按下

input_report_abs(dev, ABS_PRESSURE, 0);

input_sync(dev);

iowrite32(WAIT4INT(0), base_addr+S3C2410_ADCTSC); //等待按键中断

if (OwnADC) { //OwnADC是获取一把锁标示,在此为0

OwnADC = 0;

up(&ADC_LOCK);

}

}

}

第二种情况,如果触摸屏被按下,首先触发触摸中断,执行stylus_updown函数

static irqreturn_t stylus_updown(int irq, void *dev_id)

{

unsigned long data0;

unsigned long data1;

int updown;

if (down_trylock(&ADC_LOCK) == 0) {  //获取一把锁

OwnADC = 1;       //表示获得锁

data0 = ioread32(base_addr+S3C2410_ADCDAT0); //读取X轴数据

data1 = ioread32(base_addr+S3C2410_ADCDAT1); //读取Y轴数据

updown = (!(data0 & S3C2410_ADCDAT0_UPDOWN)) && (!(data1 & S3C2410_ADCDAT0_UPDOWN)); //触摸屏是否被按下,按下updowm=1

if (updown) {

touch_timer_fire(0);  // 如果触摸屏被按下,执行touch_timer_fire

} else {  //去抖动操作,释放锁

OwnADC = 0;

up(&ADC_LOCK);

}

}

return IRQ_HANDLED;

}

下面我们第二次分析touch_timer_fire,而这次主要是因为触摸中断中调用了这个函数,假设当前触摸屏被按下后,坐标值还没进行AD转换

static void touch_timer_fire(unsigned long data)

{

unsigned long data0;

unsigned long data1;

int updown;

data0 = ioread32(base_addr+S3C2410_ADCDAT0);

data1 = ioread32(base_addr+S3C2410_ADCDAT1);

updown = (!(data0 & S3C2410_ADCDAT0_UPDOWN)) && (!(data1 & S3C2410_ADCDAT0_UPDOWN));

if (updown) {  //触摸屏被按下

if (count != 0) { //count是全局变量,统计AD转换次数,目前未AD转换

long tmp;                                                                                        

tmp = xp;

xp = yp;

yp = tmp;                                                                                         

xp >>= 2;

yp >>= 2;

input_report_abs(dev, ABS_X, xp);

input_report_abs(dev, ABS_Y, yp);

input_report_key(dev, BTN_TOUCH, 1);

input_report_abs(dev, ABS_PRESSURE, 1);

input_sync(dev);

}

xp = 0;      //虽然触摸屏被按下,但是未完成AD转换

yp = 0;

count = 0;

//自动连续测量X和Y坐标

iowrite32(S3C2410_ADCTSC_PULL_UP_DISABLE | AUTOPST, base_addr+S3C2410_ADCTSC);

//AD转换开始且该位在开始后清零

iowrite32(ioread32(base_addr+S3C2410_ADCCON) | S3C2410_ADCCON_ENABLE_START, base_addr+S3C2410_ADCCON);

} else {

count = 0;

input_report_key(dev, BTN_TOUCH, 0);

input_report_abs(dev, ABS_PRESSURE, 0);

input_sync(dev);

iowrite32(WAIT4INT(0), base_addr+S3C2410_ADCTSC);

if (OwnADC) {

OwnADC = 0;

up(&ADC_LOCK);

}

}

}

现在我们知道,如果触摸屏被按下,但是AD还没转换完毕,那么我们会开启AD转换,自动测量X和Y坐标,这样就会触发AD转换中断,执行AD转换的中断处理程序。其实当我们的触摸屏被按下,当X和Y轴获取电压值,然后就会进行AD转换,执行AD转换的中断处理程序。好了,我们该看看AD转换的中断代码了。

static irqreturn_t stylus_action(int irq, void *dev_id)

{

unsigned long data0;

unsigned long data1;

if (OwnADC) { //只有触摸屏被按下,相应了触摸中断该标志才为1

data0 = ioread32(base_addr+S3C2410_ADCDAT0); //读取X坐标

data1 = ioread32(base_addr+S3C2410_ADCDAT1);  //读取Y坐标

xp += data0 & S3C2410_ADCDAT0_XPDATA_MASK;//叠加X坐标

yp += data1 & S3C2410_ADCDAT1_YPDATA_MASK;//叠加Y坐标

count++;  //统计AD转换次数

if (count < (1<<2)) { //如果AD转换次数不足4次

//自动连续测量X和Y坐标

iowrite32(S3C2410_ADCTSC_PULL_UP_DISABLE | AUTOPST, base_addr+S3C2410_ADCTSC);

//AD转换开始且该位在开始后清零

iowrite32(ioread32(base_addr+S3C2410_ADCCON) |S3C2410_ADCCON_ENABLE_START,base_addr+S3C2410_ADCCON);

} else {

mod_timer(&touch_timer, jiffies+1); //四次AD转换后,修改定时时间

iowrite32(WAIT4INT(1), base_addr+S3C2410_ADCTSC);//等待释放

}

}

return IRQ_HANDLED;

}

在这个AD转换的中断程序中,有一个全局变量count令人费解,count是什么作用呢?我们做AD转换时,其实是对一个点进行了四次采样,然后把四次采样结果进行叠加然后取平均。这样提高AD转换的精确度。在上面这个AD转换的中断处理程序中,我们知道,当count不足4次时,会继续进行自动连续测量X和Y坐标并开启AD转换,只有当AD转换次数达到四次后,就会修改定时器的时间,使得下一个节拍到来时执行定时器处理程序,并且等待按键被释放。当AD转换完毕,我们在下一个节拍到来后,会执行一次定时器程序touch_timer_fire

static void touch_timer_fire(unsigned long data)

{

unsigned long data0;

unsigned long data1;

int updown;

data0 = ioread32(base_addr+S3C2410_ADCDAT0);

data1 = ioread32(base_addr+S3C2410_ADCDAT1);

updown = (!(data0 & S3C2410_ADCDAT0_UPDOWN)) && (!(data1 & S3C2410_ADCDAT0_UPDOWN));

if (updown) {  //触摸屏被按下

if (count != 0) { //count是全局变量,统计AD转换次数,目前已经4次

long tmp;                                                                                        

[1] [2]
关键字:Linux  触摸屏驱动 引用地址:Linux下的触摸屏驱动

上一篇:基于GPS和CDMA的物流运输车辆监控系统
下一篇:基于3G手机的视频监控系统设计

推荐阅读最新更新时间:2024-11-12 18:23

Linux-2.6.12移植到斯道s3c2410
硬件设备: 网卡: dm9000 NandFlash: K9F1208U0C 64M 1, 下载并解压缩内核,修改顶层Makefile的编译器选项: ARCH ?= arm CROSS_COMPILE ?= arm-softfloat-linux-gnu- 2, 修改linux-2.6.12/arch/arm/mach-s3c2410/devs.c, 使其支持NandFlash和DM9000网卡: 添加头文件: #include linux/dm9000.h //该头文件将从linux-2.6.14的内核中copy到include/linux中去 #include linux/mtd/partit
[单片机]
【补充】s3c2440启动过程详细分析
2440启动过程算是一个难点,不太容易理解,而对于2440启动过程的理解,影响了后面裸机代码执行流程的分析,从而看出2440启动过程的重要性。 2440启动方式和启动方式选择 在S3C2440的datasheet《S3C2440A_UserManual_Rev13.pdf》中搜索map,可以在第5章(P195)中搜索到下图。 从此图中,可以得知 OM = 01,10,Not using NAND flash for boot ROM OM = 00, Using NAND flash for boot ROM 而OM 又是什么呢? 从S3C2440的datasheet《S3C2440A_UserMan
[单片机]
【补充】s3c<font color='red'>2440</font>启动过程详细分析
ARM嵌入式Linux设备树简介及应用示例
1). 简介 设备树(Device Tree)是一种用来描述系统硬件的数据结构,一些硬件设备设计机制就是可被系统发现的(如PCI Express或者USB总线),而有一些则不是(尤其是内存映射外设)。对于后一种情况,不同于X86架构系统采用BIOS和操作系统沟通硬件拓扑信息,ARM Linux通常情况是将硬件设备描述硬编码到系统内核(Linux Kernel)中,但由于ARM嵌入式设备的多样和离散性,即便如此也不能保证覆盖到所有设备,而且长久以来给ARM Linux内核代码维护造成了很大负担;基于这种情况,设备树的概念就被提出,将ARM SOC和板卡硬件平台描述信息从内核独立出来成为设备树文件,通过bootloader传递给内核来识
[单片机]
ARM嵌入式<font color='red'>Linux</font>设备树简介及应用示例
mini2440 dm9000 网卡驱动详解 (三)
*dm9000_get_drvinfo() 该函数去的设备的基本信息(设备名,版本,总线名)传给ethtool_drvinfo结构体变量。代码清单如下: static void dm9000_get_drvinfo(struct net_device *dev, struct ethtool_drvinfo *info) { board_info_t *dm = to_dm9000_board(dev); /*to_dm9000_board实际上就是调用了netdev_priv(dev)*/ strcpy(info- driver, CARDNAME);
[单片机]
MINI2440和GQ2440烧录问题
声明:以下是自己使用GQ2440和MINI2440的学习过程,仅供参考。 刚学习嵌入式Linux时候,可能会接触到很多的知识点,内核,根文件,引导什么的,对于新手来说是相当的苦恼,而且很多问题其实只是某些配置,安装环节的失误导致后续工作根本进行不下去,我一开始使用了国嵌的教程来进行学习的,一开始就因为各种驱动安装失败的问题困扰了很久。 在解决了些硬件上的问题后,一头扎进了supervivi,和uboot的世界(实际刚学的时候建议不要去碰太多这些东西,真的头痛,其实就是类似与windows中开启的BIOS差不多的,友情说明下国嵌里自带的supervivi是不能用的。),很多的教程都是介绍JTAG进行BIOS的烧录,但是一般大部
[单片机]
01-S3C2440学习入门概念+环境搭建
一、心得: 这两年学过很多东西,有点杂,总感觉不够踏实,于是准备写些东西出来记录一下,希望以后复习方便,同时也给大家学习带来一点帮助。从头把JZ2440系统学习一下,希望后面学习心理会更踏实。以下只是自己学习思路,仅供大家参考。 (1)嵌入式后为什么是LINUX: 之前一直把嵌入式和Linux弄混,到底嵌入式与Linux有什么关系呢。大部分嵌入式设备都是运行Linux系统,因此嵌入式后常加着Linux。Linux是嵌入式软件的核心部分,所以做嵌入式软件Linux一定是要慢慢做到很熟悉的。 (2)嵌入式Linux: 以应用为中心,以计算机技术为基础,软件硬件可裁剪的专用计算机系统。具有很强的定制性特点。 (3
[单片机]
01-S3C<font color='red'>2440</font>学习入门概念+环境搭建
linux之i2c子系统架构---总线驱动
编写i2c设备驱动(从设备)一般有两种方式: 1.用户自己编写独立的从设备驱动,应用程序直接使用即可。 2.linux内核内部已经实现了一个通用的设备驱动,利用通用设备驱动编写一个应用程序(用户态驱动),在应用程序中用到大量设备驱动提供的接口,通过应用程序来控制从设备。 总线驱动 4.1 概述 I2C总线驱动是I2C适配器的软件实现,提供I2C适配器与从设备间完成数据通信的能力,比如起始,停止,应答信号和master_xfer的实现函数。 I2C总线驱动由i2c_adapter和i2c_algorithm来描述 4.2 S3c2440I2C控制器的硬件描述 S3c2440处理器内部集成了一个I2C控制器,通过四个寄
[单片机]
<font color='red'>linux</font>之i2c子系统架构---总线<font color='red'>驱动</font>
基于Linux的嵌入式实时操作系统的研究
1.引言   从上世纪八十年代开始,开始出现各种各样的商用嵌入式操作系统,这些操作系统大部分都是为专用或通用系统而开发,如VxWorks、Windows CE、pSOS、Palm OS、OS-9、LynxOS、QNX、LYNX 等,它们的优点是为用户提供良好的开发环境,提高了应用系统的开发效率,运行效率高、实时性好,缺点是价格昂贵且源代码封闭。这就不仅影响了开发者的积极性,而且使得整个产品的成本急剧上升。[1]   结合国内实情,嵌入式系统需要的是一套高度简练、界面友好、质量可靠、应用广泛、易开发、多任务、价格低廉的实时操作系统。   在嵌入式产品的开发中,有必要寻找一种廉价的嵌入式实时操作系统,以降低产品的开发成本和系统复杂度
[模拟电子]
基于<font color='red'>Linux</font>的嵌入式实时操作系统的研究
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
随便看看

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved