mini2440驱动分析之触摸屏
mini2440触摸屏驱动对应的文件为mini2440_ts.c,他是作为输入设备注册到内核的,功能实现是通过输入子系统来完成的,现在分析触摸屏的实现。以后再分析输入子系统。
一. 分析一个驱动首先看它的模块初始化函数,下面是mini2440_ts.c的模块初始化函数:
static struct clk
*adc_clock; //这个时钟结构体代表时钟
- 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\n");
- 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\n");
- return -ENOMEM;
- }
-
-
- s3c2410_ts_connect();
-
- 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();
-
- if (!input_dev) {
- printk(KERN_ERR "Unable to allocate the input device !!\n");
- 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);
-
- 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;
-
-
-
-
-
- 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 !\n");
- 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 !\n");
- iounmap(base_addr);
- return -EIO;
- }
-
-
- printk(KERN_INFO "%s successfully loaded\n", s3c2410ts_name);
-
-
-
- input_register_device(dev);
-
- return 0;
- }
现在触摸屏已经注册到了输入子系统核心,触摸屏硬件已经处于等待中断的模式。当按下触摸屏时,会出发按下中断,进而调用stylus_updown函数进行中断处理,下面分析这个函数
二. 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);
- data1 = ioread32(base_addr+S3C2410_ADCDAT1);
-
-
-
-
-
- updown = (!(data0 & S3C2410_ADCDAT0_UPDOWN)) && (!(data1 & S3C2410_ADCDAT0_UPDOWN));
-
- if (updown) {
- touch_timer_fire(0);
- } else {
- OwnADC = 0;
- up(&ADC_LOCK);
- }
- }
-
- return IRQ_HANDLED;
- }
这个中断处理例程,通过读取ADCAT0, ADCDAT1的值,并且利用他们的第15位来判断是否真正按下,如果是调用touch_timer_fire启动AD转换,如果不是释放adc信号量,下面分析touch_timer_fire函数:
三. 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) {
- 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;
-
- 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);
- }
- }
- }
这个函数在触摸屏按下中断处理程序中是作为普通函数调用的,但是它还是内核定时器服务函数
- static struct timer_list touch_timer = TIMER_INITIALIZER(touch_timer_fire, 0, 0);
四. AD转换完成中断处理程序stylus_action
- static irqreturn_t stylus_action(int irq, void *dev_id)
- {
- unsigned long data0;
- unsigned long data1;
-
-
- if (OwnADC) {
- data0 = ioread32(base_addr+S3C2410_ADCDAT0);
- data1 = ioread32(base_addr+S3C2410_ADCDAT1);
-
-
- xp += data0 & S3C2410_ADCDAT0_XPDATA_MASK;
- yp += data1 & S3C2410_ADCDAT1_YPDATA_MASK;
- count++;
-
- if (count < (1<<2)) {
- 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 {
-
-
- mod_timer(&touch_timer, jiffies+1);
-
- iowrite32(WAIT4INT(1), base_addr+S3C2410_ADCTSC);
- }
- }
-
- return IRQ_HANDLED;
- }
五. 总结一下触摸屏驱动是如果工作的
1. 用户按下触摸屏进入中断处理程序stylus_updown,这个函数处理如下
(1)判断是否按下,如果是调用touch_timer_fire,如果不是那么说明现在松开了
(2)touch_timer_fire处理如下: 判断是否按下
<1>确实按下了,那先判断count是否为0,如果为0那么说明还没有进行AD转换,启动ad转换,转换完成进入中断处理程序stylus_action
<2>确实按下了,判断的count不是0,说明ad转换完成了,进行x/y坐标的事件报告,并报告触摸屏事件。这个才是一次正确的按下
<3>触摸屏这时松开了,那么只报告触摸屏事件,并设置触摸屏为等待按下中断的模式
2. stylus_action 函数处理过程,首先读取转换数据,然后判断count是否小于4
(1)count小于4 那么再次启动转换
(2)count不小于4,然后重新设置定时器,延时时间为一个时钟滴答,设置触摸屏为等待中断模式,等待松开中断
3 . 在定时器事件到达之前松开触摸屏,会进入stylus_updown,这个函数判断触摸屏松开了,释放信号量不会报告任何事件
如果定时器时间到执行touch_timer_fire,就会判断真正按下,而这时count不为0,报告触摸屏事件,报告x/y坐标
4. 这里的定时器是为了防止抖动而设置的,所以按下操作至少要保持一个时钟滴答加四次ad转换的时间。
5.
我认为还有一个问题就是在第三步执行touch_timer_fire的时候松开触摸屏,因为现在触摸屏处于检测松开中断的时候,就会进入
stylus_updown,这时判断触摸屏松开,释放了ADC信号量,然后从中断返回,继续执行touch_timer_fire,这种情况下会在没有
信号量的时候操作ad,这是潜在的并发条件。(后来证明这种猜测是错误的,因为代码释放信号亮的前提是重新获得信号量,而这里不会获得信号量,看来我低估
了写驱动人的水平!!!)
6. 为了直观的了解触摸屏的工作过程,我画了一个流程图来说明

图片说明:
1.
第一个红色的松开事件,是在AD转换了四次,定时器到时后,向输入子系统报告x/y坐标的时候发生的,因为触摸屏处于检测松开中断的状态,所以会进入按下
松开中断,从而判断松开,释放信号量,中断返回,继续中断前的程序运行,这里过后还操作了ad的寄存器,但是已经不拥有AD的信号量了,我认为这里引入了
潜在的并发。
2. 第二个红色的松开事件,是在AD转换了四次,但是定时器没有到时的时候发生的,同样进入按下松开中断,中断处理程序判断没有按下,释放信号量,中断返回。
3.
第三个红色的松开事件,是在按下触摸屏后马上放开时或者启动ad转换的时候发生的,前者概率很小,因为间隔时间太短。主要是后者,这也有两种情况,一种是
已经报告完x.y轴坐标的,这个类似情况2。一种是已经确认按下了,然后松开了,这种情况ad也正常转换四次,在设置触摸屏等待中断后(这样这个设置就没
有作用了,因为已经松开了),定时器中断到时调用touch_timer_fire,然后进入白色框框里面。这里可以看出定时器除了防止触摸屏抖动外,还
有一种功能就是,在第三种情况下,使触摸屏进入正常检测按下中断的状态。
六. 关于两个宏
1. WAIT4INT(x)
#define WAIT4INT(x) (((x)<<8) | \
S3C2410_ADCTSC_YM_SEN | S3C2410_ADCTSC_YP_SEN | S3C2410_ADCTSC_XP_SEN | \
S3C2410_ADCTSC_XY_PST(3))
这个宏是设置,ADCTSC寄存器的。设置为等待中断模式,这个寄存器的第八位,表示检测哪类中断,如果是0 检测按下中断,也就是WAIT4INT(0)检测按下中断。如果是1检测松开中断
2. AUTOPST
#define AUTOPST (S3C2410_ADCTSC_YM_SEN | S3C2410_ADCTSC_YP_SEN | S3C2410_ADCTSC_XP_SEN | \
S3C2410_ADCTSC_AUTO_PST | S3C2410_ADCTSC_XY_PST(0))
这个是设置ADCTSC寄存器的,设置AD转换方式为连续的x/y轴转换方式。