1 引 言
许多嵌入式系统,尤其是一些人机交互(HMI)较频繁的嵌入式系统,键盘是一种应用最为广泛的输入设备。由于嵌入式设备的功能互异性,为其提供一种通用性键盘是不可行的,一般都需要根据嵌入式系统的实际功能来设计所需的特殊键盘,并实现相应的驱动程序。www.51kaifa.com
在嵌入式设备上扩展键盘的常用方式是通过使用CPU的GPIO端口扫描实现的,显然,这种方式会占用系统的GPIO资源,特别是在GPIO资源比较紧张而按键又较多的系统,这个问题就特别突出。当然,也可以通过外扩GPIO(如8255等)或外扩专用的键盘接口(如8279等)方式实现,但这种方式显然增加了系统的复杂度,在实际系统设计中颇感不便。
本文以在ARM9(AT91RM9200)嵌入式微处理器上实现一个POS机键盘(8×8)为例,呈现了一种在嵌入式设备上扩展多行列键盘的新设计思路,并在ARM-Linux系统实现了键盘的驱动程序。
2、接口电路的硬件设计
本文通过一个设计实例,说明如何使用一种比较简单的方式,来实现一个8×8的POS机矩阵键盘,POS机所采用的微处理器是AT91RM9200芯片。AT91RM9200是ATMEL公司生产的一款高性能的32位ARM9处理器,它是一款通用工业级ARM芯片,在工业控制、智能仪器仪表等领域内得到了大量的应用[3,4],其详细芯片特性可参见文献[2]。
在AT91RM9200上扩展键盘,一般都是通过其GPIO端口来实现。AT91RM9200虽提供了4×32个可编程的GPIO端口。但为减小芯片体积和功耗,其许多GPIO端口都是与系统的外围设备控制器端口或地址线、数据线进行复用的,所以实际可用于扩展的GPIO端口是很少的。而对于一个8×8键盘,若采用传统的GPIO端口扩展方式,则需要16个GPIO,这在一个比较复杂的POS系统中是很难满足的,因此需要采用其他方式来解决这个问题。
图1 键盘接口原理图
本文通过数据锁存的方式,充分利用32位处理器的数据宽度优势,使用数据线来替代键盘扩展所需的GPIO端口,从而减少对系统GPIO资源的占用。键盘接口的实现原理如图1所示。在图1所示的电路中,U1301(74LVCC4245)为三态缓冲器,U1302(74HC574)为锁存器,系统工作原理描述如下:
U1301的nOE端连接系统的译码输出nKey_CS,部件地址由系统译码电路决定,当向该地址写数据时,nKey_CS信号为低电平,数据可以通过U1301,同时,nKey_CS信号经两级反相器延时后作为锁存信号将数据锁存到U1302的输出端,作为键盘的行扫描信号,而键盘的列扫描信号则仍然使用系统的GPIO。依次向每行送出低电平信号,同时检测连接在GPIO的列信号,即可实现对键盘的扫描。在本系统中,只使用了系统32位数据的低8位作为行扫描信号,在实现8×8矩阵键盘扫描的情况下,仅需要占用8个GPIO口,如果采用同样的方式,分别使用16位数据或32位数据作为行扫描信号,则只需要占用4个或2个GPIO,显然,与传统的方式相比较,该方式可以大大节省系统的GPIO资源。
3、键盘的驱动模块设计
完成接口电路的设计之后,还需要编写相应的键盘驱动模块。本文采用AT91RM9200芯片中已经运行了ARM-Linux操作系统,因此给键盘的驱动程序开发提供了很大的方便。键盘的驱动模块可分为硬件初始化、文件操作函数的实现以及键盘扫描程序三个部分。
3.1 硬件初始化
键盘驱动程序的开发模式与Linux系统中一般字符设备的驱动开发步骤相似,关于Linux设备驱动开发的详细分析可参考文献[1]。首先需完成的是驱动程序模块的初始化函数和清除函数。在初始化函数中,除完成模块注册外,还应进行硬件初始化,下面是本文根据AT91RM9200芯片的GPIO控制器的特性[2],在模块的初始化函数中的硬件初始化的伪代码。
……
设置所使用GPIO端口为GPIO控制器控制
设置所使用GPIO端口的类型为输入
使能所使用GPIO端口的输入毛刺滤波功能
使能相应的GPIO控制器时钟
取指定地址的虚拟地址并向地址写入数据0x0
……
3.2 文件操作函数的实现
因为键盘在系统中一般只起输入作用,因此在键盘驱动程序的file_operations结构中,必须实现的文件操作函数只有文件读函数。
另外在实际应用中,为防止键盘按键的丢失,被按下键的扫描代码通常都放置在一个缓冲区内,直到应用程序准备处理一个按键为止。缓冲区大小一般视应用系统的需求而定,本例中缓冲区大小取为20个按键代码。而缓冲区的实现是以一个环形队列的形式实现。当按键按下后,扫描代码将被放置在队列的下一个空位置,若缓冲区已满,则下一按键将会被丢弃。应用程序则通过键盘的读函数read()从缓冲区的位置指针处起,读取所需个数的键码;完成读取操作后,还需将已被读取的键码从缓冲队列中删除,并更新缓冲区的位置指针。下面给出了本例中,实现的键盘读函数key_read()的伪代码:
ssize_t key_read(……)
{
定义并初始化变量;
if 缓冲区中可被读取的键码数大于0;
取得键码放置缓冲区的自旋锁;
计算此次读操作可读取的代码个数M(缓冲区中可读取的代码个数与程序要求个数之间的较小者);
从缓冲区位置指针开始,从缓冲区中拷贝M个的键码到用户空间缓冲区内;
更新缓冲区的位置指针和缓冲区中还剩余的键码个数;
释放自旋锁
返回此次读操作成功读取的代码个数。
Else
返回-1
}
3.3 键盘扫描程序
键盘的工作原理是通过键盘的行线和列线的状态来判断键盘中有无按键被按下。键盘扫描程序的功能就是用来判断处于按下状态的按键的具体位置及取得相应的键码值,因此扫描程序的设计是键盘驱动模块实现的核心。
键盘扫描程序的实现主要有两种,即轮询方式和中断方式[5]。在本例中,利用操作系统定时器队列与轮询扫描方式结合的方法对键盘的驱动程序进行了设计,主要是基于以下两个方面的原因。其一是AT91RM9200芯片的中断信号线是非常宝贵的硬件资源,每一组GPIO端口只配置了一根中断信号线,即32个GPIO端口共享一条信号线。这样若采用中断方式,则至少需要占用一条芯片中断信号线,对多行列的键盘,如果其所采用的GPIO端口不是来自于同一组时,就需要占用多条中断信号线。而且若其他设备使用的GPIO端口与键盘使用的GPIO口属于同一组,那么在两种设备的驱动程序设计中,必须进行中断共享,这样不仅使系统的软件设计更为复杂,且易产生中断丢失和中断竟态等问题,使设备性能受到影响。其二键盘是系统中属于一种相对低速的设备,采用轮询方式完全可以满足键盘的输入要求。
ARM-Linux操作系统提供了良好的定时器机制,因此通过简单定时器操作,就可以实现以固定间隔对键盘的状态进行扫描并对按键事件进程处理,固定间隔的大小可根据系统需求进行配置,定义器的详细操作可参见文献[1]。如前所述,键盘扫描程序的功能就是对键盘的状态进行判断和处理。若无按键按下,则扫描直接返回;若有按键按下,则对被按下键的位置进行判断,并将相应的键码值写入缓冲区中。因为本例中的键盘是为POS机配置,因此按键的准确性是至关重要,因此在扫描代码中对按键值进行了多次验证,下面是本例中使用的键盘扫描程序的伪代码:
int Scan_Keyboard()
{
定义并初始化变量;
取得键码放置缓冲区的自旋锁;
if 缓冲区中还有空;
① 依次判断各GPIO口的状态,若无低电平,则无键按下,直接退出if语句;否则,有键按下,且当前检验的GPIO口连接的行线即为按键所在的行;
② 给键盘列线连接的数据线依次送入高电平,再通过判断按键行线所在的GPIO端口的电平状态,得到按键所在的列;
延时一小段时间,以消除键盘抖动;
③ 再向给键盘列线连接的数据线全送低电平,使用代码段①再次判断是否有键按下,若有,则取得按键所在的行;
④ 同样使用代码段②重新判断按键所在的列;
⑤ 判断第一次得到的按键的行与列是否与第二次完全一样,若完全相同,则可进入下一步,否则退出if语句;
⑥ 重新向给键盘列线连接的数据线全送低电平,并判断按键是否弹起,若仍处于按下状态,则继续等待,否则根据行与列,转化为相应键值,并写入缓冲区;
if语句结束;
释放自旋锁;
函数返回 0;
}
完成驱动程序代码编写后,就可以将键盘的驱动程序加载到ARM-Linux内核中了,既可以采用静态加载方式,也可以采用动态方式进行加载。加载后,在应用程序中键盘的编程使用方式与其他字符设备一样。采用本文所述方式设计的键盘,目前已配置在笔者参与开发的POS机中交用户使用,据用户测试,键盘的输入准确率和反应时间都达到了设计要求。
4、结束语
本文以运行ARM-Linux的AT91RM9200系统为基础,提出了一种在ARM9上扩展特殊键盘的新设计方法,并对键盘扩展的硬件设计和驱动软件开发都作了详细说明。本设计方法利用数据锁存方式替代了常规的GPIO扩展,提高了系统硬件的资源利用率,这一思想也为在其他嵌入式设备扩展多行列键盘提供了一种新的设计思路。
评论
查看更多