如何快速调通触摸按键驱动

2023-10-25 09:59

本文主要是介绍如何快速调通触摸按键驱动,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

针对智能家居、小家电、门禁考勤等触摸按键领域,深圳华仕飞科技推出一系列定制化的软硬件产品,产品不仅仅包括触摸按键领域,也包括一些小家电控制、特别是带触摸按键功能和强电开关控制领域的定制化软硬件产品。最近,华仕飞正在免费派送一款针对门禁考勤系统的12个数字按键的触摸按键HTK100。现在,针对该产品及驱动软件调试方法进行一些简要的讲解。

华仕飞的触摸按键采用的是专业的触摸驱动方案,具有高效的触摸按键算法处理和抗干扰算法处理,是行业内具有良好性价比的产品。HTK100硬件接口可以同时兼容I2C、UART、IR三种通讯接口,支持中断输出,但是,在软件上暂时只能任选其中一种通讯协议,也就是说,只能烧录其中任意一种协议的软件。市面上免费派送的的HTK100支持的是I2C协议。由于I2C协议的很多优点,可以满足绝大部分用户的需求。当然,如果用户需要其它的协议,也可以通过访问官网,联系他们烧录特殊功能的软件,或定制化软件。

针对HTK100,官方网站现在已经推出了基于I2C和UART通讯协议的驱动,该驱动基于华仕飞自身所推出基于STM32的演示评估板,在官方网站和网上其他很多地方都有下载。其中,I2C采用的是GPIO口模拟I2C,程序简单易懂,可移植性强。当然,某些第三方论坛,也可以找到完全的华仕飞评估板的软件包,感兴趣的朋友可以上网找找。

现在,以HTK100的I2C协议为例,简述一下HTK100驱动调试方法。

(1)从官方免费申请或从网上买1~2块HTK100触摸按键板。HTK100采用6PIN1.26mm的接线座。早期的HTK100没有配送对应的线材,后期的才有配送。如果没有配送,用户需要自己在网上购买对应的线,或者自己手动焊接其它对应的线。

(2)接线:将HTK100接上用户自己的主板。主板需要至少有I2C总线接口。如果没有,可以通过GPIO口模拟I2C。为避免主MCU频繁读取HTK100所造成的资源浪费,应采用外部中断方式读取HTK100的按键,所以,INT接口也应接。另外,为防止出在极其特殊情况下HTK100工作不正常的情况,复位引脚RST最好也接上MCU以便可以根据需要对MCU进行复位。当然,在资源不够时,RST可以悬空,或者通过一个大的电阻(推荐100K以)进行上接。HTK100内部既有软件看门狗,也有硬件看门狗。当程序出现异常跑飞的情况时,内部是可以自动复位的。

硬件接口典型电路如下图所示:


(3)IO口配置:

对于INT脚,配置为上升沿触发模式或者上升沿和下降沿双触发模式即可。

对于RST脚,配置为输出模式即可,可以为OD模式,无须上拉或下拉电阻。如果上拉或下拉,应接一个比较大的电阻。

I2C口配置的总线时钟应为200KHz以下;如果采用的是,每字节数据采用的是MSB在前方式传输,即数据位传输顺序为从高位在前。

如果采用的是GPIO模拟I2C的方式,则SDA最好配置为OD模式,这样SDA总线就无段通过软件频繁进行输入输出切换。如果您对芯片的这一功能不太放心,或者您没有这方面的经验,也可以通过软件进行输入输出切换。

一种比较简单的通过软件自动切换的代码实现方式如下所示:

 

void __Set_SDA(void)

{

if(__I2C_SDA_IsInputMode())

{

           __SetI2C_SDA_OutputMode();

}

 

SET_SDA;

}

 

 

void __Clr_SDA(void)

{

if(__I2C_SDA_IsInputMode())

{

           __SetI2C_SDA_OutputMode();

}

 

CLR_SDA;

}

 

这样,通过以上代码,就无需在GPIO口模拟I2C的驱动程序中频繁对SDA的输入输出方式进行切换了。

 

(4)I2C驱动程序的编写。由于GPIO口模拟I2C对硬件的依赖性比较低,软件简单可控,所以笔者推荐在条件许可的情况下,尽量使用GPIO口模拟I2C。以下为GPIO口模拟I2C的全部驱动程序。

void __I2C_start(void)

{

       __Set_SDA();

       __Set_SCL();

       __delay_us(5);

       __Clr_SDA();

       __delay_us(5);

       __Clr_SCL();

       __delay_us(5);

}

 

void __I2C_stop(void)

{

       __Clr_SDA();

       __delay_us(5);

       __Set_SCL();

       __delay_us(5);

       __Set_SDA();

}

 

 

void __I2C_get_ack(void)

{

       __Clr_SCL();

       __delay_us(2);

       __Clr_SDA();

       __delay_us(5);

       __Set_SCL();

       __delay_us(5);

       __Clr_SCL();

}

 

void __I2C_get_nack(void)

{

       __Clr_SCL();

       __delay_us(2);

       __Set_SDA();

       __delay_us(5);

       __Set_SCL();

       __delay_us(5);

       __Clr_SCL();

       __delay_us(2);

}

 

 

void __I2C_multi_read(BYTE *pDat,BYTE num)

{

       BYTEi;

 

       //receive the n-1 bytes

       for(i=0;i<num-1;i++)

       {

              *pDat= __I2C_byte_get();

              //send ack

              __I2C_get_ack();

              pDat++;

       }

       //receive the last byte

       *pDat= __I2C_byte_get();

       //send no ack

       __I2C_get_nack();

}

 

BYTE __I2C_byte_read(void)

{

       BYTEdat = 0;

 

       dat= __I2C_byte_get();

      

       __I2C_get_nack();

       __delay_us(2);

      

       returndat;

}

 

 

BYTE __I2C_byte_get(void)

{

       BYTEdat=0;

       BYTEbitcnt;

 

       //receive data; gpio4 is input

       for(bitcnt=0;bitcnt<8; bitcnt++)

       {

              __Clr_SCL();

              __delay_us(5);

              __Set_SCL();

              dat= dat<<1;

              if(__Get_SDA_Data())

              {

                     dat= dat+1;

              }

              __delay_us(5);

       }

       __Clr_SCL();

       __delay_us(5);

       returndat;

}

 

 

BOOL __I2C_send_byte(BYTE dat)

{

       BYTEbitcnt;

       BOOLret;

       //send data; gpio4 is output

       for(bitcnt=0;bitcnt<8; bitcnt++)

       {

              if((dat<<bitcnt)&0x80)

//            if((dat>>bitcnt)&0x01)

              {

                     __Set_SDA();

              }

              else

              {

                     __Clr_SDA();

              }

              __delay_us(2);

              __Set_SCL();

              __delay_us(5);

              __Clr_SCL();

              __delay_us(2);

       }

       __Set_SDA();

       __delay_us(2);

       __Set_SCL();

       __delay_us(5);

       if(__Get_SDA_Data())

       {

              ret= 0;

       }

       else

       {

              ret= 1;

       }

       __Clr_SCL();

       __delay_us(5);

      

       returnret;

//     return__I2C_send_ack();

}

 

 

//等待从机应答信号

//返回值:1接收应答成功

// 0 接收应答失败

BOOL __I2C_send_ack(void)

{

       UINTcnt=0;

 

       __Set_SDA();  //主机释放数据总线,等待从机产生应答信号

       __delay_us(5);

       __Set_SCL();

       __delay_us(5);

       //等待从机对数据总线的操作。低电平代表应答

#if 1

       while(__Get_SDA_Data())

       {

              cnt++;

              __delay_us(1);

              //这个属于软件延时,不一定准确。

              if(cnt> 5) //如果时间超时,没有应答就停止。

              {

                     __I2C_stop();

                     returnFALSE;  //没有响应的话返回1.

              }

       }

#endif    

       __Clr_SCL();

       returnTRUE; //如果有响应的话就返回1.

}

 

 

(5)HTK100驱动程序

由于HTK100主要是读取触摸按键值,故其驱动比较简单

以下为来自官方HTK100的全部驱动。由于HTK100事实上并没有手势功能,所以针对手势读取的那部分驱动事实上是无效的。

/**************************************************************************

Parameters:           

                     [OUT]:pKeyVal:最佳键值

Return:          

                     TRUE:读取成功;

                     FALSE:读取失败

Description:   

                     通过I2C读取最优触摸按键键值,即当有多个按键同时按下时,取最优键的键值

**************************************************************************/

BOOL__HTK_I2C_Read_optimal_touch_keyval(BYTE *pKeyVal)

{

       volatileint i = 0;

//     DBG_printf("\n\nI2C_start1\n");

       __I2C_start();

       __delay_us(5);

//     DBG_printf("(WRITE)DEV ADDR\n");

       if(!__I2C_send_byte((__HTK_DEVICE_ADDR<<1)|0) )         // 发送器件地址+写命令

       {

              DBG_printf("FAIL\n");

              __I2C_stop();

              returnFALSE;

       }

       __delay_us(5);

//     DBG_printf("KEY_VALUE_CMD\n");

       if(!__I2C_send_byte(__HTK_TOUCH_KEY_VALUE_CMD))         //数据地址

       {

              DBG_printf("FAIL\n");

              __I2C_stop();

              returnFALSE;

       }

       __delay_us(5);

//     DBG_printf("I2C_start2\n");

       __I2C_start();

 

       __delay_us(5);

//     DBG_printf("[READ]DEV ADDR\n");

       if(!__I2C_send_byte((__HTK_DEVICE_ADDR<<1)|1) )         // 发送器件地址+读命令

       {

              DBG_printf("FAIL\n");

              __I2C_stop();

              returnFALSE;

       }

 

//     DBG_printf("READ DATA\n");

       __delay_us(100);

      

       *pKeyVal= __I2C_byte_read();

       if(*pKeyVal == 0)

       {

              i= 1;

       }

       __delay_us(5);

       __I2C_stop();

       returnTRUE;

 

}

 

 

/**************************************************************************

Parameters:           

                     [OUT]:pdwStatus:最佳键值

Return:          

                     TRUE:读取成功;

                     FALSE:读取失败

Description:   

                     通过I2C读取当前按键状态

                     *pdwStatus的BIT0~BIT31分别对应32个(如果按键数量不够,以实际按键数量为准)按键的状态

                     每个BIT的1表示按下,0:表示未按下

**************************************************************************/

BOOL__HTK_I2C_Read_touch_status(DWORD *pdwStatus)

{

       BYTEdat[4];

      

//     DBG_printf("\n\nI2C_start1\n");

       __I2C_start();

       __delay_us(5);

       if(!__I2C_send_byte((__HTK_DEVICE_ADDR<<1)|0) )         // 发送器件地址+读命令

       {

              DBG_printf("FAIL\n");

              __I2C_stop();

              returnFALSE;

       }

 

       __delay_us(5);

       if(!__I2C_send_byte(__HTK_TOUCH_KEY_STATUS_CMD))        //读命令

       {

              DBG_printf("FAIL\n");

              __I2C_stop();

              returnFALSE;

       }

//     DBG_printf("I2C_start2\n");

       __delay_us(5);

       __I2C_start();

 

       __delay_us(5);

//     DBG_printf("[READ]DEV ADDR\n");

       if(!__I2C_send_byte((__HTK_DEVICE_ADDR<<1)|1) )         // 发送器件地址+读命令

       {

              DBG_printf("FAIL\n");

              __I2C_stop();

              returnFALSE;

       }

 

//     DBG_printf("READ DATA\n");

       __delay_us(100);

      

       __I2C_multi_read(dat,4);

      

       __I2C_stop();

       //如系统为小端模式,调用如下代码

       *pdwStatus= (((DWORD)dat[0]<<24) | ((DWORD)dat[1]<<16) |((DWORD)dat[2]<<8) | ((DWORD)dat[3]));

 

       //如系统为大端模式,调用如下代码

//     *pdwStatus= (((DWORD)dat[3]<<24) | ((DWORD)dat[2]<<16) |((DWORD)dat[1]<<8) | ((DWORD)dat[0]))

       returnTRUE;

}

 

/**************************************************************************

Parameters:           

                     [OUT]:pGesture:手势

Return:          

                     TRUE:读取成功;

                     FALSE:读取失败

Description:   

                     通过I2C读取当前的手势值

**************************************************************************/

BOOL __HTK_I2C_Read_gesture(BYTE*pGesture)

{

//     DBG_printf("\n\nI2C_start1\n");

       __I2C_start();

       __delay_us(5);

//     DBG_printf("(WRITE)DEV ADDR\n");

       if(!__I2C_send_byte((__HTK_DEVICE_ADDR<<1)|0) )         // 发送器件地址+写命令

       {

              DBG_printf("FAIL\n");

              __I2C_stop();

              returnFALSE;

       }

       __delay_us(5);

//     DBG_printf("GESTURE_CMD\n");

       if(!__I2C_send_byte(__HTK_TOUCH_GESTURE_CMD))             //数据地址

       {

              DBG_printf("FAIL\n");

              __I2C_stop();

              returnFALSE;

       }

       __delay_us(5);

//     DBG_printf("I2C_start2\n");

       __I2C_start();

 

       __delay_us(5);

//     DBG_printf("[READ]DEV ADDR\n");

       if(!__I2C_send_byte((__HTK_DEVICE_ADDR<<1)|1) )         // 发送器件地址+读命令

       {

              DBG_printf("FAIL\n");

              __I2C_stop();

              returnFALSE;

       }

 

//     DBG_printf("READ DATA\n");

       __delay_us(100);

      

       *pGesture= __I2C_byte_read();

       __delay_us(5);

       __I2C_stop();

       returnTRUE;

 

}

 

(6)驱动的挂载及系统联调

对于一般的单片机系统而言,通过中断方式读取HTK100键值及其它状态值即可。当然,考虑到中断需要一定的时间,而中断服务程序需要需要快速响应且尽快退出,可以考虑在中断服务程序中置个标志位,然后在中循环中再读取相庆的键值及其它状态值即可。程序可以参考如下代码:

 

/**************************************************************************

Parameters:           

                     void

Return:          

                     void

Description:   

       上升沿中断处理程序

       即当中断服务程序收到一个触摸按键的中断时,在主循环中调用该函数以读取键值

       由于中断服务程序不适合处理时间较长的工作,所以该函数不建议在中断服务程序中运行,而在主循环中运行

**************************************************************************/

void __HTK_proc(void)

{

       BYTEKeyVal = 0;

       BYTEGesture = 0;

       DWORDKeyStatus = 0;

 

 

       if(__isGetHtkIntr)

       {

//            DBG_printf("__isGetHtkIntr\n");

 

              if(GET_HTK_INTR)    // 识别是否为上升沿中断

              {

                     DBG_printf("Rising\n");

 

#if defined(IIC_KEY_CFG)

 

                     //读取最佳按键

                     DBG_printf("Readkey value \n");

#if 1

 

                     if(__HTK_I2C_Read_optimal_touch_keyval(&KeyVal))

                     {

                            DBG_printf("touchkey value:%02X\n\n", KeyVal);

                            SetKeyValLed(KeyVal);

                     }

                     else

                     {

                            SetKeyValLed(0);

                            DBG_printf("Nokey\n");

                     }

#endif

#if 0

                     DBG_printf("Readkey status \n");

                     if(__HTK_I2C_Read_touch_status(&KeyStatus))

                     {

                            DBG_printf("touchkey status:0x%08X\n", KeyStatus);

                     }

                     else

                     {

                            DBG_printf("Nokey\n");

                     }

 

#endif

#if 0

                     DBG_printf("Readgesture \n");

                     if(__HTK_I2C_Read_gesture(&Gesture))

                     {

                            DBG_printf("touchkey gesture:%d\n", Gesture);

                     }

                     else

                     {

                            DBG_printf("Nogesture\n");

                     }

#endif

#elif defined(UART_KEY_CFG)

 

                     //读取最佳按键

                     DBG_printf("Readkey value \n");

 

 

                     if(__HTK_Uart_Read_optimal_touch_keyval(&KeyVal))

                     {

                            DBG_printf("touchkey value:%02X\n", KeyVal);

                            SetKeyValLed(KeyVal);

                     }

                     else

                     {

                            DBG_printf("Nokey\n");

                            SetKeyValLed(0);

                     }

 

#if 0

                     DBG_printf("Readkey status \n");

                     if(__HTK_Uart_Read_touch_status(&KeyStatus))

                     {

                            DBG_printf("touchkey status:0x%08X\n", KeyStatus);

                     }

                     else

                     {

                            DBG_printf("Nokey\n");

                     }

 

                     DBG_printf("Readgesture \n");

                     if(__HTK_Uart_Read_gesture(&Gesture))

                     {

                            DBG_printf("touchkey gesture:%d\n", Gesture);

                     }

                     else

                     {

                            DBG_printf("Nogesture\n");

                     }

#endif

#endif

              }

              else

              {

                     SetKeyValLed(0);

 

              }

              __isGetHtkIntr= FALSE;

       }

}

 

 

/**************************************************************************

Parameters:           

                     void

Return:          

                     void

Description:   

              上升沿中断服务程序

                     当上升沿中断来到来时,表示有触摸按键被按下

**************************************************************************/

void __HTK_Intr_isr(void)

{

       if(__isGetHtkIntr)

       {

              return;

       }

      

       __isGetHtkIntr= TRUE;

}

 

(7)有关复位:

一般情况下,只需要主控板在上电时,通过软件上拉时即可。如果为了增强软件的可靠性,以预防各种可能出现的异常情况,可以在程序的某些地方增加一部分代码,即通过调用_HTK_I2C_Read_optimal_touch_keyval()函数无法读取数据时,即可以调用复位函数启动HTK100的硬件复位功能。但该复位函数不应在INT中断服务程序中调用,因为如果HTK100没有正常工作,同时也不会出现中断信号。一种比较可靠的办法就是定时调用_HTK_I2C_Read_optimal_touch_keyval()检测HTK100是否正常工作。如果工作不正常,即启动复位程序。

复位程序如下所示:

/**************************************************************************

Parameters:           

                     void

Return:          

                     void

Description:   

                     触摸按键复位

**************************************************************************/

void__HTK_reset(void)

{

       SET_HTK_RST;

       __delay_us(10);

       CLR_HTK_RST;

       __delay_us(200);

       SET_HTK_RST;

}

 

祝您早日调通和使用华仕飞触摸按键板。


这篇关于如何快速调通触摸按键驱动的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/281802

相关文章

Rust中的Option枚举快速入门教程

《Rust中的Option枚举快速入门教程》Rust中的Option枚举用于表示可能不存在的值,提供了多种方法来处理这些值,避免了空指针异常,文章介绍了Option的定义、常见方法、使用场景以及注意事... 目录引言Option介绍Option的常见方法Option使用场景场景一:函数返回可能不存在的值场景

电脑桌面文件删除了怎么找回来?别急,快速恢复攻略在此

在日常使用电脑的过程中,我们经常会遇到这样的情况:一不小心,桌面上的某个重要文件被删除了。这时,大多数人可能会感到惊慌失措,不知所措。 其实,不必过于担心,因为有很多方法可以帮助我们找回被删除的桌面文件。下面,就让我们一起来了解一下这些恢复桌面文件的方法吧。 一、使用撤销操作 如果我们刚刚删除了桌面上的文件,并且还没有进行其他操作,那么可以尝试使用撤销操作来恢复文件。在键盘上同时按下“C

Linux_kernel驱动开发11

一、改回nfs方式挂载根文件系统         在产品将要上线之前,需要制作不同类型格式的根文件系统         在产品研发阶段,我们还是需要使用nfs的方式挂载根文件系统         优点:可以直接在上位机中修改文件系统内容,延长EMMC的寿命         【1】重启上位机nfs服务         sudo service nfs-kernel-server resta

hdu 4565 推倒公式+矩阵快速幂

题意 求下式的值: Sn=⌈ (a+b√)n⌉%m S_n = \lceil\ (a + \sqrt{b}) ^ n \rceil\% m 其中: 0<a,m<215 0< a, m < 2^{15} 0<b,n<231 0 < b, n < 2^{31} (a−1)2<b<a2 (a-1)^2< b < a^2 解析 令: An=(a+b√)n A_n = (a +

v0.dev快速开发

探索v0.dev:次世代开发者之利器 今之技艺日新月异,开发者之工具亦随之进步不辍。v0.dev者,新兴之开发者利器也,迅速引起众多开发者之瞩目。本文将引汝探究v0.dev之基本功能与优势,助汝速速上手,提升开发之效率。 何谓v0.dev? v0.dev者,现代化之开发者工具也,旨在简化并加速软件开发之过程。其集多种功能于一体,助开发者高效编写、测试及部署代码。无论汝为前端开发者、后端开发者

利用Django框架快速构建Web应用:从零到上线

随着互联网的发展,Web应用的需求日益增长,而Django作为一个高级的Python Web框架,以其强大的功能和灵活的架构,成为了众多开发者的选择。本文将指导你如何从零开始使用Django框架构建一个简单的Web应用,并将其部署到线上,让世界看到你的作品。 Django简介 Django是由Adrian Holovaty和Simon Willison于2005年开发的一个开源框架,旨在简

CentOs7上Mysql快速迁移脚本

因公司业务需要,对原来在/usr/local/mysql/data目录下的数据迁移到/data/local/mysql/mysqlData。 原因是系统盘太小,只有20G,几下就快满了。 参考过几篇文章,基于大神们的思路,我封装成了.sh脚本。 步骤如下: 1) 先修改好/etc/my.cnf,        ##[mysqld]       ##datadir=/data/loc

SAM2POINT:以zero-shot且快速的方式将任何 3D 视频分割为视频

摘要 我们介绍 SAM2POINT,这是一种采用 Segment Anything Model 2 (SAM 2) 进行零样本和快速 3D 分割的初步探索。 SAM2POINT 将任何 3D 数据解释为一系列多向视频,并利用 SAM 2 进行 3D 空间分割,无需进一步训练或 2D-3D 投影。 我们的框架支持各种提示类型,包括 3D 点、框和掩模,并且可以泛化到不同的场景,例如 3D 对象、室

UE5 半透明阴影 快速解决方案

Step 1: 打开该选项 Step 2: 将半透明材质给到模型后,设置光照的Shadow Resolution Scale,越大,阴影的效果越好

快速排序(java代码实现)

简介: 1.采用“分治”的思想,对于一组数据,选择一个基准元素,这里选择中间元素mid 2.通过第一轮扫描,比mid小的元素都在mid左边,比mid大的元素都在mid右边 3.然后使用递归排序这两部分,直到序列中所有数据均有序为止。 public class csdnTest {public static void main(String[] args){int[] arr = {3,