历史上的今天
返回首页

历史上的今天

今天是:2025年11月16日(星期日)

正在发生

2022年11月16日 | Linux驱动:s3c2410/s3c2440 ts驱动分析 之二

2022-11-16 来源:zhihu

一,前言

前面结合“平台总线-设备-驱动”模型分析了ts(触摸)驱动的加载过程,现在进入驱动程序分析下其具体的实现。涉及到输入子系统详解、input核心层程序分析、evdev事件处理层程序分析、ts设备驱动层程序分析以及s3c2440的ADC转换和触摸控制器的操作。

二,涉及的寄存器

三,调用probe函数

根据上一篇的分析,驱动层通过platform_driver_register注册后,会调用到该驱动层的probe函数。


四,s3c2410ts_probe函数分析

4.1 硬件寄存器设置

4.1.1 获取设备参数

struct s3c2410_ts_mach_info *info;


    info = ( struct s3c2410_ts_mach_info *)pdev->dev.platform_data;


    /*

    info

        .delay = 10000,   // ADC conversion start delay value

        .presc = 49,      // ADC clk

        .oversampling_shift = 2, // 采样精度

    */


    if (!info)

    {

        printk(KERN_ERR "Hm... too bad : no platform data for tsn");

        return -EINVAL;

    }

4.1.2 使能ADC

adc_clock = clk_get(NULL, "adc");

if (!adc_clock) {

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

    return -ENOENT;

}

clk_enable(adc_clock);

4.1.3 获取ADC & TOUCH SCREEN 寄存器

static inline void s3c2410_ts_connect(void)

{

    s3c2410_gpio_cfgpin(S3C2410_GPG12, S3C2410_GPG12_XMON);

    s3c2410_gpio_cfgpin(S3C2410_GPG13, S3C2410_GPG13_nXPON);

    s3c2410_gpio_cfgpin(S3C2410_GPG14, S3C2410_GPG14_YMON);

    s3c2410_gpio_cfgpin(S3C2410_GPG15, S3C2410_GPG15_nYPON);

}


{

    .....

    base_addr=ioremap(S3C2410_PA_ADC,0x20);  //#define S3C2410_PA_ADC      (0x58000000)

    if (base_addr == NULL) {

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

        return -ENOMEM;

    }


    /* Configure GPIOs */

    s3c2410_ts_connect(); // 设置GPIO功能

    .....

}

4.1.4 设置ADC转换频率

if ((info->presc&0xff) > 0)

        iowrite32(S3C2410_ADCCON_PRSCEN | S3C2410_ADCCON_PRSCVL(info->presc&0xFF),

                 base_addr+S3C2410_ADCCON);    

// 设置ADC clk 

// A/D converter freq = PCLK/(info->presc+1)  

// 一次adc转换所需时间Conversion time = 1/(A/D converter freq / 5cycles)

4.1.5 设置ADC转换开始的延时时间

因为ADC中断产生后的一段时间内电压还未稳定,如果立即进行ADC转换,那么转换值会有误差,这时就可以通过设置延时时间,待电压稳定后再执行转换以确保数值的准确性。


// 设置ADC conversion start delay value

if ((info->delay&0xffff) > 0)

        iowrite32(info->delay & 0xffff,  base_addr+S3C2410_ADCDLY);

4.1.6 进入等待触摸按下模式

iowrite32(WAIT4INT(0), base_addr+S3C2410_ADCTSC);  // 等待触摸按下中断模式

自此,ADC & TOUCH SCREEN 寄存器初始设置完成。

4.2 注册ADC和TC中断

注册TC中断,监测触摸屏的按下和抬起;注册ADC中断,进行ADC转换。


/* Get irqs */

if (request_irq(IRQ_ADC, stylus_action, IRQF_SAMPLE_RANDOM | SA_SHIRQ,

                "s3c2410_action", ts.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", ts.dev)) {

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

    iounmap(base_addr);

    return -EIO;

}

4.2.1 TC中断处理函数 stylus_updown

static irqreturn_t stylus_updown(int irq, void *dev_id)

{

    unsigned long data0;

    unsigned long data1;

    int updown;


    // 读取寄存器DATA0和DATA1获取x,y轴的ADC转换值

    data0 = ioread32(base_addr+S3C2410_ADCDAT0);

    data1 = ioread32(base_addr+S3C2410_ADCDAT1);


    // ADCDAT0 bit[15] 0 为按下,1 为松开 即updown 为true 则按下,为false 则松开

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


    /* TODO we should never get an interrupt with updown set while

     * the timer is running, but maybe we ought to verify that the

     * timer isn't running anyways. */


    if (updown)

    {

        // 如果为按下状态

        touch_timer_fire(0);

    }


    return IRQ_HANDLED;

}

4.2.2 ADC中断处理函数 stylus_action

static irqreturn_t stylus_action(int irq, void *dev_id)

{

    unsigned long data0;

    unsigned long data1;


//  if (bADCForTS) {


        data0 = ioread32(base_addr+S3C2410_ADCDAT0);

        data1 = ioread32(base_addr+S3C2410_ADCDAT1);


        ts.xp += data0 & S3C2410_ADCDAT0_XPDATA_MASK;  // 为四次adc转换值的累加

        ts.yp += data1 & S3C2410_ADCDAT1_YPDATA_MASK;  // 为四次adc转换值的累加

        ts.count++;


//      bADCForTS = 0;

//      up(&gADClock);

        //  ts.count < 4 ,即四次adc转换值为一次按下的结果                             

        if (ts.count < (1<//              if (!down_trylock(&gADClock)) {

//                  bADCForTS = 1;

                    iowrite32(S3C2410_ADCTSC_PULL_UP_DISABLE | AUTOPST, base_addr+S3C2410_ADCTSC); // 进入 等待adc 转换模式 

                    iowrite32(ioread32(base_addr+S3C2410_ADCCON) | S3C2410_ADCCON_ENABLE_START, base_addr+S3C2410_ADCCON); // 开启adc转换,转换在info->delay后进行,转换完成后会产生一个ADC中断

//          }

        } else {


            mod_timer(&touch_timer, jiffies+1); // 启动一个定时器,一个jiffies(系统滴答时间) 后进入定时器处理函数 touch_timer_fire

            iowrite32(WAIT4INT(1), base_addr+S3C2410_ADCTSC); // 同时等待 触摸抬起中断

        }

//  }

    return IRQ_HANDLED;

}

4.2.3 定时器处理函数 touch_timer_fire

static struct timer_list touch_timer =

        TIMER_INITIALIZER(touch_timer_fire, 0, 0);


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));

    // updown 为true 则按下,为false 则松开


    if (updown) {

        // 处于按下状态的处理


        // 首次按下 产生了四次ADC转换后,才去处理x,y值

        if (ts.count != 0) {

            long tmp;  


            // 横纵坐标转换

            tmp = ts.xp;

            ts.xp = ts.yp;

            ts.yp = tmp;


            ts.xp >>= ts.shift; // 四次adc转换值的平均值为一次按下的结果

            ts.yp >>= ts.shift; // 四次adc转换值的平均值为一次按下的结果


#ifdef CONFIG_TOUCHSCREEN_S3C2410_DEBUG

            {

                struct timeval tv;

                do_gettimeofday(&tv);

                printk(DEBUG_LVL "T: %06d, X: %03ld, Y: %03ldn", (int)tv.tv_usec, ts.xp, ts.yp);

            }

#endif


            input_report_abs(ts.dev, ABS_X, ts.xp);   // 上报x的坐标

            input_report_abs(ts.dev, ABS_Y, ts.yp);   // 上报y的坐标


            input_report_key(ts.dev, BTN_TOUCH, 1);     // 上报BIN_TOUCH 按下

            input_report_abs(ts.dev, ABS_PRESSURE, 1);  // 上报ABS_PRESSURE 按下

            input_sync(ts.dev);                         // 上报事件完成

        }


        ts.xp = 0;

        ts.yp = 0;

        ts.count = 0;


//      if (!down_trylock(&gADClock)) {

//          bADCForTS = 1;

        iowrite32(S3C2410_ADCTSC_PULL_UP_DISABLE | AUTOPST, base_addr+S3C2410_ADCTSC);  // 进入 等待adc 转换模式 

        iowrite32(ioread32(base_addr+S3C2410_ADCCON) | S3C2410_ADCCON_ENABLE_START, base_addr+S3C2410_ADCCON); // 开启adc转换,完成后会产生一个ADC中断

//      }

    } else {

        // 松开的处理


        ts.count = 0;


        input_report_key(ts.dev, BTN_TOUCH, 0);   // 上报BIN_TOUCH 松开

        input_report_abs(ts.dev, ABS_PRESSURE, 0);// 上报ABS_PRESSURE 松开

        input_sync(ts.dev);                       // 上报事件完成


        iowrite32(WAIT4INT(0), base_addr+S3C2410_ADCTSC);  // 进入等待触摸按下中断

    }

}

4.3 注册到输入子系统

4.3.1 申请input_dev

input_dev = input_allocate_device();


if (!input_dev) {

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

    return -ENOMEM;

}

4.3.2 设置input_dev

ts.dev = input_dev;

// 设置产生的事件类型,同步类事件、按键类事件和绝对位移事件

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

// 设置某类事件类型中的具体事件

// 按键类事件中的触摸事件

ts.dev->keybit[LONG(BTN_TOUCH)] = BIT(BTN_TOUCH);

// 绝对位移类事件中的ABS_X、ABS_Y、压力值的值范围,0x3ff是该ADC转换器是10bit精度的,最大为0x3fff

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

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

input_set_abs_params(ts.dev, ABS_PRESSURE, 0, 1, 0, 0); // 只设置有按下和抬起(0,1),没有具体的压力值

推荐阅读

史海拾趣

DIPTRONICS MANUFACTURING INC公司的发展小趣事

在稳定国内市场的基础上,圜达实业积极拓展海外市场。凭借优良的品质和快速的交期,公司的产品逐渐打入国际市场,并赢得了众多客户的信赖。如今,圜达实业的产品已营销至全球100余个国家,遍及五大洲。

CYANLITE公司的发展小趣事

CYANLITE公司一直秉持着绿色环保的理念,致力于推广节能减排的照明产品。他们不仅在产品设计上注重节能和环保,还积极参与各种环保公益活动,向公众普及LED照明的优势。这种积极的环保态度使得CYANLITE公司在市场上树立了良好的形象,也吸引了越来越多的消费者选择他们的产品。同时,公司还积极与政府部门合作,推动LED照明在公共领域的普及和应用。

国芯佳品公司的发展小趣事

CYANLITE公司成立于XXXX年,由一群富有创新精神的电子工程师创立。他们发现当时市场上对于高效能LED照明产品的需求日益增长,于是决定利用自己的技术背景,专注于研发和生产高品质的LED照明产品。起初,公司规模较小,资金有限,但他们凭借对技术的热情和坚持不懈的努力,成功研发出了第一款具有竞争力的LED灯具,并逐渐在市场上获得了认可。随着产品的热销,CYANLITE公司逐渐扩大了生产规模,并增加了研发投入,以保持其在行业中的领先地位。

亿宝科技(CNIBAO)公司的发展小趣事

亿宝科技在成立初期,就明确了以技术创新为核心的发展战略。公司创始人带领研发团队,深入研究市场需求,不断推出具有竞争力的电子产品。在一次偶然的机会中,他们成功研发出一款具有高性能、低功耗特点的芯片,这一创新技术迅速获得了市场的认可。此后,亿宝科技不断加大研发投入,形成了一套完整的自主研发体系,为公司后续的发展奠定了坚实的基础。

富士康(FOXCONN)公司的发展小趣事

1996年,FMS在台湾正式成立,标志着美丽微半导体股份有限公司的诞生。公司由杨照霞女士创办,初期便专注于二极管及场效应管(MOS)等分立式器件的研发与制造。这一决策基于对市场需求的敏锐洞察,以及对半导体技术未来发展的坚定信心。在杨照霞女士的带领下,FMS迅速建立起一支专业的研发团队,并着手构建半导体上下游资源整合体系,为公司的长远发展奠定了坚实基础。

启臣微(Chip)公司的发展小趣事

随着公司业务的不断发展,启臣微开始积极拓展海外市场。公司在西安设立研发中心,在深圳和昆山设立销售中心,并与多家国际知名企业建立了合作关系。这些举措不仅提升了公司的国际影响力,也为公司带来了更多的发展机遇。

问答坊 | AI 解惑

晶体注意事项

晶振注意事项 1)电压 2)频率 3)稳定度 4)温漂 5)输出类型 6)尺寸…

查看全部问答>

TFT LCD移植问题

最近在做LCD的驱动,遇到了些问题,请大家帮忙分析,解决,先谢谢~ 问题是这样的: 我把原来在2440上用的TFT-LCD(3.5\",320×240)移植到2443上,原来2443用的是4.3\"的屏(480×272) 换过后,图像基本能出来,可是就是屏幕最下面不正常, ...…

查看全部问答>

s3c6410 linux2.6.29写寄存器的问题

今天写了个在linux 2.6.29下测试 IIS0的测试 driver, 碰到一个奇怪的问题,模块代码如下: static int __devinit iis0_test_init(void) {     unsigned char *virt;         virt = ioremap_nocache(0x7f00200 ...…

查看全部问答>

软件开发应届生找不到工作

我是上海一大学软件工程专业的09届应届生,本科,不是很会说话。 在学校里学的是单片机开发之类的(偏软件 C语言),接触过ARM+Wince之类的嵌入式软件的开发。(其实学校里嵌入式软件方面的课基本就没有,都是我凭着兴趣自学的)单片机的水平达到自 ...…

查看全部问答>

LED进军传统照明 向“四面”发起强攻

LED进军传统照明 向“四面”发起强攻   针对市场调查有70%的商家表示,看好LED 未来的市场发展,而在具体的行动上他们则仍然保持着冷静的态度,会根据LED 在市场上的销售表现再来决定调整产品结构,看是否现阶段就加 ...…

查看全部问答>

爆lunchpad设计上的一个问题。

不知道用lunchpad的朋友有没有发觉,接上lunchpad的32k晶振很容易受干扰。程序上用到了32k晶振,把lunchpad放桌面上,程序就跑不了,拿起lunchpad就能跑。用手摸引通过排针引出来的引脚,XIN、XOUT,板子也不能工作。所以我觉得XIN和XOUT不应该引出 ...…

查看全部问答>

基于LABVIEW的STM32F0多路数据采集系统(源码+视频)

             基于LABVIEW的STM32F0多路数据采集系统(源码+视频) 原始方案计划   一、应用背景        在一般的工业控制应用项目中,都需要涉及到数据采集部分,即利用 ...…

查看全部问答>