触摸屏控制原理,其实与ADC读取一个滑动变阻器中间触点电压的原理一样。只不过,读取触摸屏的XY方向上的电压需要两次,而且需要设置其工作模式以实现一个ADC读取两个通道的电压。

S3C2440ADC控制是很简单的,与普通单片机控制ADC的方法没有多大区别。大概的操作步骤都是如下所示。

(1)设置控制寄存器ADCCON

(2)启动ADC转换

(3)等待转换结束

(4)读取转换结果

但是,S3C2440的触摸屏控制有什么工作模式,如等待中断模式,分离的x/y轴坐标转换模式,自动(连续)x/y轴坐标转换模式,普通转换模式。通过官方芯片资料理解这些模式,似乎还是一头雾水。经过测试,我对这些模式有了初步的认识。

(一) ADC&TC模式分析

ADC控制模式

普通模式+无操作模式AUTO_PST = 0  XY_PST = 00

这种模式用在ADC的普通操作,而非触摸屏控制,例如我们想读取AIN2通道的电压,就是用这种模式。除了这种模式以外的模式都是触摸屏控制模式。

 

触摸屏控制模式

     分离的x/y轴坐标转换模式

普通模式+测量X轴坐标(AUTO_PST = 0  XY_PST = 01

这种模式只在X方向上测量电压。这种工作模式要求YM_SENYP_SENXM_SENXP_SEN分别为0110,这四个控制位只有这样才能采集X方向上的电压。ADCTSC = 0x69

 

普通模式+测量Y轴坐标(AUTO_PST = 0  XY_PST = 10

这种模式只在Y方向上测量电压。这种工作模式要求YM_SENYP_SENXM_SENXP_SEN分别为1001,这四个控制位只有这样才能采集Y方向上的电压。ADCTSC = 0x9A

自动(连续)

 

x/y轴坐标转换模式

自动x/y轴坐标转换模式+无操作模式(AUTO_PST = 1  XY_PST = 00

这种工作模式可以一次连续的采集两次电压,分别得到X方向上的和Y方向上的电压。这种模式似乎对YM_SENYP_SENXM_SENXP_SEN这四个信号没什么要求。ADCTSC = 0x0C

     等待中断模式

普通模式+等待中断模式(AUTO_PST = 0  XY_PST = 11

这种工作模式主要是用来检测触摸屏被按下或者被抬起。这种工作模式要求YM_SENYP_SENXM_SENXP_SEN分别为1101,并且PULL_UP0,即使能上拉。检测按下,ADCTSC = 0xD3;检测抬起 ADCTSC = 0x1D3

 

可以看到,其实工作模式的决定在于ADCTSC中的AUTO_PST(ADCTSC [2]) XY_PST(ADCTSC [1:0])三位。理论上,共可以确定8种工作模式,实际上只用到了5种工作模式。

 

(二) ADCTSCYM_SENYP_SENXM_SENXP_SEN四位的重要性

 

需要说明的是YM_SENYP_SENXM_SENXP_SEN这四个信号的重要性。起初,我在写程序过程中,设置完自动(连续)x/y轴坐标转换模式并且已经读取了XY方向上的值之后,直接通过读取ADCDAT0UPDOWN位来判断是否已经抬起。结果,我发现即使已经抬起,但是UPDOWN位还是0,而不会变到1。出错的程序在红色那一行。 

static void Isr_Tc(void)
{
    printf("Stylus Down: ");
    mode_auto_xy();     /* 进入自动(连续) X/Y轴坐标转换模式 */
    
    /* 设置位[0]为1,启动A/D转换
     * 注意:ADCDLY为50000,PCLK = 50MHz,
     *       要经过(1/50MHz)*50000=1ms之后才开始转换X坐标
     *       再经过1ms之后才开始转换Y坐标
     
*/
    ADCCON |= ADC_START;
    // 检测位[15],当它为1时表示转换结束
    while (!(ADCCON & ADC_ENDCVT));
    // 打印X、Y坐标值    
    printf("xdata = %4d, ydata = %4d\r\n", (int)(ADCDAT0 & 0x3ff), (int)(ADCDAT1 & 0x3ff));

    while(!(ADCDAT0 & 0x8000));
    printf("Stylus Up!!\n\r");
    wait_down_int();         /* 进入"等待中断模式",等待触摸屏被按下 */
    // 清INT_TC中断
    SUBSRCPND |= BIT_SUB_TC;
    SRCPND    |= BIT_ADC;
    INTPND    |= BIT_ADC;
}

 

经过分析,我得知当YM_SENYP_SENXM_SENXP_SEN不是1101时,触摸屏控制器就根本不能检测到触摸屏的抬起和按下(这是由触摸屏的工作原理决定的)。知道了错误原因,我在读取完XY方向上的值后,加入了“等待(抬起)中断模式”的代码,再来判断是否能检测到。结果,我发现当抬起时,UPDOWN位变为了1,证明了我的原因分析。修改的程序如下。

 

static void Isr_Tc(void)
{
    printf("Stylus Down: ");
    mode_auto_xy();     /* 进入自动(连续) X/Y轴坐标转换模式 */
    
    /* 设置位[0]为1,启动A/D转换
     * 注意:ADCDLY为50000,PCLK = 50MHz,
     *       要经过(1/50MHz)*50000=1ms之后才开始转换X坐标
     *       再经过1ms之后才开始转换Y坐标
     
*/
    ADCCON |= ADC_START;
    // 检测位[15],当它为1时表示转换结束
    while (!(ADCCON & ADC_ENDCVT));
    // 打印X、Y坐标值    
    printf("xdata = %4d, ydata = %4d\r\n", (int)(ADCDAT0 & 0x3ff), (int)(ADCDAT1 & 0x3ff));
    
    wait_up_int();
    while(!(ADCDAT0 & 0x8000));
    printf("Stylus Up!!\n\r");
    
    wait_down_int();         /* 进入"等待中断模式",等待触摸屏被按下 */
    // 清INT_TC中断
    SUBSRCPND |= BIT_SUB_TC;
    SRCPND    |= BIT_ADC;
    INTPND    |= BIT_ADC;
}

  

(三)触摸屏控制器ADCCON寄存器起到的作用

 

       无论是ADC操作,还是触摸屏操作都用到了ADCCON。它们都是用ADCCONENABLE_START来启动转换,ADC操作就是启动ADC对应通道的转换,触摸屏就是启动或者X方向通道,或者Y方向通道,或者X/Y两个通道的转换(启动一次转换读取两个通道的值分别保存在ADCDAT0ADCDAT1中)。

它们都能通过ADCCONECFLG来读取是否转换结束。

 

(四)ADC&TC中断信号INT_ADCINT_TC

 

          INT_ADC信号产生分四种情况如下:

      (1)普通ADC转换结束

      (2)触摸屏X方向转换结束
      (3)触摸屏Y方向转换结束

      (4)连续X/Y方向转换结束

        INT_TC信号产生分两种情况如下:

      (1)触摸屏被按下

      (2)触摸屏被抬起

  

总结如收功 练功不收功 到老一场空

posted on 2013-10-15 11:22  amanlikethis  阅读(669)  评论(0编辑  收藏  举报