历史上的今天
返回首页

历史上的今天

今天是:2025年02月19日(星期三)

正在发生

2018年02月19日 | 嵌入式Linux系统的键盘驱动实现

2018-02-19 来源:eefocus

1引言

Linux由于其具有内核强大且稳定,易于扩展和裁减,丰富的硬件支持等诸多优点,在嵌入式系统中得到了广泛的应用。很多嵌入式Linux系统,特别是一些具有与用户强交互的嵌入式系统,往往需要配备一个特殊键盘,此时开发者需要根据实际情况,为自己的特殊键盘编写驱动程序。

2Linux键盘驱动简介

Linux中的大多数驱动程序都采用了层次型的体系结构,键盘驱动程序也不例外。在Linux中,键盘驱动被划分成两层来实现。其中,上层是一个通用的键盘抽象层,完成键盘驱动中不依赖于底层具体硬件的一些功能,并且负责为底层提供服务;下层则是硬件处理层,与具体硬件密切相关,主要负责对硬件进行直接操作。键盘驱动程序的上层公共部分都在driver/keyboard。c中。该文件中最重要的就是内核用EXPORT_SYMBOL这个宏导出的handle_scancode函数。handle_scancode完成的功能是:首先将扫描码转换成键码,接着根据shift,alt等扩展键的按下情况将键码转换成目标码,一般情况下是ASCII码,最后将该ASCII码放到终端设备的缓冲区中,并且调度一个tasklet负责将其在显示器上回显出来。可以看出,这个函数完成的是键盘驱动程序中最核心的一些工作,而这些核心的逻辑功能是不依赖于底层硬件的,所以可以将其独立出来,并且导出给底层的硬件处理函数调用。在这个文件中还定义了其它几个回调函数,它们由键盘驱动程序中的上层公共部分调用,并由底层硬件处理函数实现。比如kbd_init_hw,kbd_translate,kbd_unexpected_up等等。其中kbd_translate由handle_scancode调用,负责将扫描码转换成键码;键盘驱动程序的底层硬件处理部分则根据不同的硬件有不同的实现。例如PC平台上标准键盘的底层硬件处理函数都集中在driver/Pc_keyb。c中。这个文件包括了键盘中断处理函数keyboard_interrupt,扫描码到键码转换函数pckbd_translate等其他一些与底层硬件密切相关的函数。

在这种体系结构下,要添加一块特殊键盘到系统中就显得格外清晰。开发者只需为其编写驱动程序中的底层硬件处理函数,就可以将该键盘驱动起来。一般说来,底层硬件处理函数中最重要的工作就是在键盘中断处理中获取被按下键的扫描码,并且以它为参数调用handle_scancode,该扫描码可以自己定义,但它必须唯一地标识出被按下键在键盘上的位置。此外,开发者还需要提供对应的从自定义扫描码到键码的转换函数kbd_translate。具体的键码转换,将目标码放到终端的输入缓冲区,以及回显等工作都由handle_scancode负责完成。在此我们也可以看出,内核导出函数handle_scancode在整个键盘驱动程序中,起着将上层通用抽象层和底层硬件处理层粘和起来的关键作用。

3应用实例

下面我们将以一个具体的应用实例来说明在嵌入式Linux系统中给一个特殊键盘编写驱动程序的具体过程。

3。1硬件模块描述

本系统的构建选用了三星公司的S3C2410开发板作为硬件平台。特殊键盘的硬件模块主要由两个SN74hc164芯片和一个4行16列的矩阵扫描电路构成。SN74hc164是一个8位的串形输入并形输出移位寄存器,它的内部由8个D触发器串联而成。其工作原理简单说来是这样的,SN74hc164芯片在时钟CLK脉冲的上升沿将A,B引脚上的串形输入在8个时钟脉冲以后并行输出到输出引脚QA到QH。其真值表见图1所示。

两个SN74hc164芯片先串联后,将它们的CLK引脚和CLR引脚分别接到S3C2410开发板的GPB2和GPB4端口上,并且将第一个SN74hc164芯片的A,B引脚接到开发板的GPB1端口上,这三个GPIO端口配置成输出端口。这样我们就借助于两个SN74hc164寄存器,实现了只占用3个GPIO端口,给矩阵扫描电路的16列提供输入,从而既节约了成本,又避免了GPIO资源的浪费。但这同时也给键盘驱动程序的实现带来了一定的麻烦,驱动程序首先要将SN74hc164驱动起来,然后才能对矩阵电路的16列进行控制。该矩阵电路的4个行引脚分别被接到S3C2410的GPG6,GPG7,GPG8,GPG9端口上,并且这四个端口被配置成中断源。无键按下时直接读为高电位,使用时通过SN74hc164芯片先将键盘的16列置低电位,任何一个键被按下,相应的行GPG端口就会有从高到低的电压跳变,从而触发一次中断。

3。2软件模块描述

初始化部分。这部分包括硬件层和软件层上的初始化。在本例中,需要先对矩阵电路和SN74hc164芯片所使用到的GPIO端口作配置,以使CPU可以对它们进行控制和访问。为了要将某个GPIO端口配置成输入输出或者是中断源,需要在对应的GPIO控制寄存器中设置正确的值,具体的值可以通过查阅S3C2410开发板手册来获得。比如,为了将GPB1设置成SN74hc164的输入端,需要将GPBCON这个控制字中2,3两位设置成二进制的01,为了将GPG6设置成电压低跳变中断源,需要将GPGCON中12,13两位设置成二进制的10。在完成了硬件初始化操作以后,就是软件层上的初始化了。首先将键盘中断处理函数注册到系统,然后设置好一个定时器结构,以便在中断发生时将其挂到内核的定时器队列中去,该定时器将触发对键盘的扫描操作。最后通过SN74hc164将矩阵电路的16列置零。

中断处理部分。如前所述,这部分软件应该完成的工作就是扫描特殊键盘,确定哪个键被按下,并且拿到稳定的扫描码,然后调用内核导出函数handle_scancode。在这个应用中,该特殊键盘的布局与PC标准键盘的布局比较相似,所以我们直接将PC键盘上对应键的系统扫描码作为我们特殊键盘上各个键的扫描码,同时我们将PC键盘驱动程序中扫描码到键码的转换函数pckbd_translate作为我们的kbd_translate函数。

确定哪一个键被按下的算法如下。在中断到来时,我们已经可以根据中断号确定被按下的键在哪一行,我们还需要确定被按下的键在哪一列。为此,我们先给串联的两个SN74hc164芯片送一个CLR信号,清零,然后送16个1,使得特殊键盘的列均为高电位,此时我们在键盘的行端口读到的都是高电位。在16个时钟脉冲下,给SN74hc164芯片送入1个0和15个1,使得0在每一列上都唯一出现一次,于此同时在键盘行端口进行扫描。当被按下键所在列置0时,其所在行就会读到一个低电位。使用这种“走0法”,我们就可以确定出键盘上哪个键被按下了。但是这种简单的扫描算法还不够,因为在这种类型的矩阵扫描键盘中,键的每次按下和抬起都会有10~20ms(这段时间的长短由硬件特性决定)的毛刺抖动存在,如图2所示,所以为了获取稳定的按键信息,必须要想办法去掉这种抖动,才能避免将用户的一次按键误当作几次按键来处理。去毛刺的一种常见的方法是在有键盘中断到达时,并不立即去扫描键盘,而是先等待一段时间,等跳过毛刺抖动以后再去扫描键盘,其伪代码如下所示:

等待一段时间,跳过抖动;

扫描键盘;

if键盘上没有键被按下

结束返回;

if键盘上有键被按下

再次等待一段时间然后检查同样的键是否依然处于被按下状态;

if同样的键任然是按下

将读到的扫描码返回;

else

直接返回;

这种解决方案固然可行,但是它使用了忙等的方法去毛刺,在忙等期间,系统做不了任何有用的工作。这对于计算资源本身就很有限的嵌入式Linux系统来说,是一种奢侈的浪费。本应用中,我们设计了一种适合嵌入式系统的去毛刺解决方案,使用效果良好。

由于Linux内核提供了定时器队列,所以我们可以使用这种机制来避免忙等,提高系统的性能。当键盘上有键被按下时,键盘中断处理程序首先关闭中断源,进入轮询模式,将一个timerlist对象挂入定时器队列以后就结束了。挂入内核的定时器按时地被触发,它所触发的函数完成以下一些工作:先对整个键盘上所有的键进行一次扫描,并且将扫描得到的结果保存到一个静态2维数组变量snap_shot_matrix[16][4]中。该变量描述的是在本次键盘扫描的这个时刻,键盘上所有键的按下情况。如果某个键没有被按下,即处于松开状态,那么将snap_shot_matrix中对应的值置为0,如果某个键处于按下状态,那么将snap_shot_matrix中对应的值作自增1操作,若该值在自增1以后大于某个预先指定的数,我们就可以认为这是一个稳定值,并且将另一个大小为16*4的2维数组变量current_matrix对应坐标中的值置1,否则置0。这个变量描述的就是当前键盘上按键情况的稳定值了。也就是说我们首先把在本次扫描中得到的采样数据作处理以后保存到snap_shot_matrix中,然后依据该变量中的值,过滤得到current_matrix,通过这样一个过程来做去毛刺处理。在得到了本次扫描的稳定值current_matrix以后,我们将其与上次得到的稳定值previous_matrix作比较,从而确定与上次扫描时相比,此刻键盘上的按键情况是否发生了变化,以及此刻键盘上是否有键按下。如果发现键盘上没有任何键被按下,则打开键盘中断,再次切回到中断模式。如果键盘上有键被按下,并且是不同于上次扫描到的被按下键,我们立刻调用按键处理函数process_key,它会调用键盘驱动中的上层函数handle_scancode。如果键盘上按下的键就是上次按下的那个键,我们将递增一个计数器,当这个计数器达到某个指定值以后,我们就启动所谓的Autorepeat功能,即用户一直按着某个键,驱动程序自动重复产生键盘输入。该计数器在被按下键发生变化时置0。但是只要键盘上仍然有键处于被按下状态,我们就将当前读到的键盘稳定值current_matrix拷贝到previous_matrix中去,并且再次将前面描述的定时器对象挂到内核定时器队列中,过一段时间以后再次扫描整个键盘,直至键盘上没有键被按下。

4结束语

随着信息社会以及计算机软硬件技术的进步,嵌入式信息产品的设计和应用得到了迅速的发展,需要为自己的嵌入式Linux系统添加特殊键盘驱动的需求也越来越普遍。本文在介绍了Linux中键盘驱动程序的整体框架以后,以S3C2410开发板上的一个特殊键盘为例子,重点描述了在嵌入式Linux环境下,为特殊键盘编写驱动程序时需要完成的工作,为类似的开发提供了一种思路和参考。


推荐阅读

史海拾趣

Carlisle Interconnect Components公司的发展小趣事

在电子行业快速发展的背景下,Carlisle Interconnect Components公司始终将技术研发和持续创新作为公司发展的核心动力。公司投入大量资源用于研发新型连接器技术,不断推出具有更高性能、更小型化、更智能化的产品。这些创新产品的问世,不仅为公司带来了更多的市场机遇,更推动了整个电子连接器行业的进步。

ACP Optoelectronic Technology Co Ltd公司的发展小趣事

ACP Optoelectronic Technology Co Ltd始终坚持创新驱动的发展理念。公司不断投入研发资金,引进先进设备和技术人才,推动产品升级换代。随着市场的不断变化和客户需求的不断提升,ACP Optoelectronic Technology Co Ltd的产品线不断丰富和完善,产品质量和性能也得到了显著提升。

Base Two (2) Systems公司的发展小趣事

Base Two (2) Systems深知技术创新的重要性,因此不断加大对研发的投入。公司引进了一批高端研发人才和先进设备,为研发团队提供了强有力的支持。在研发团队的努力下,公司不断推出新产品,并对现有产品进行升级改进,以满足市场的不断变化和客户的多样化需求。

捷茂微(GATEMODE)公司的发展小趣事

随着公司业务的不断发展壮大,Base Two (2) Systems开始实施国际化战略。公司积极拓展海外市场,与多个国家和地区的合作伙伴建立了合作关系。同时,公司还在海外设立了研发中心和生产基地,以便更好地服务全球客户。这一国际化战略的实施,使得公司的业务范围更加广泛,也为公司的未来发展打开了新的篇章。

这五个故事展示了Base Two (2) Systems在电子行业中的发展历程和取得的成就。通过技术创新、市场扩张、研发投入、品质管理和国际化战略的实施,公司逐渐发展成为电子行业中的佼佼者,为行业的发展做出了积极的贡献。

EPT公司的发展小趣事

随着品牌影响力的不断提升,EPT开始积极拓展OEM/ODM业务。公司凭借先进的生产技术和严格的质量管理体系,成功与多家国内外知名企业建立了合作关系。通过为客户提供定制化的产品和服务,EPT不仅实现了业务的快速增长,还为其积累了丰富的行业经验。

Falco Electronics公司的发展小趣事

Falco Electronics非常重视企业文化建设。公司倡导“以人为本”的管理理念,注重员工的成长和发展。公司为员工提供了良好的工作环境和福利待遇,还定期开展各种培训活动,帮助员工提升技能水平。这种良好的企业文化吸引了大量优秀人才的加入,也为公司的持续发展提供了有力保障。

问答坊 | AI 解惑

写给DSP初学者:轻松入门,快速精通

随着3G技术的发展,要求处理器的速度越来越高,体积越来越小,DSP的发展正好能满足这一发展的要求。因为,传统的其它处理器都有不同的缺陷:MCU的速度较慢;CPU体积较大,功耗较高;嵌入CPU的成本较高。 DSP的发展,使得在许多速度要求较高,算法较 ...…

查看全部问答>

VHDL语言学习

VHDL语言学习,望对大家有用…

查看全部问答>

电动车用电机控制器原理

车用电机控制器近年来的发展速度之快,使人难以想象,操作上越来越“傻瓜”化,而显示则越来越复杂化。比如,车速的控制已经发展到“巡航锁定”;驱动方面,有的同时具有电动性能和助力功能,如果转换到助力状态,借助链条张力测力器,或中轴扭力传 ...…

查看全部问答>

MSP430register_guide常用模块原理

MSP430register_guide常用模块原理.…

查看全部问答>

【藏书阁】51 常用子程序

自动往返小汽车 中断应用 在线下载器源程序 源程序 应用类 音乐 移位除 延时 显示子程序 通讯 数字频率计 数字多用表 实用子程序 软件工程 频率计 排序类 排序 滤波 抗干扰 交通灯 键盘与显示类 键盘扫描程序 汇编与C的衔接 ...…

查看全部问答>

走嵌入式开发,应该学习哪方面的知识

本人对嵌入式不是很了解,但是身边有些朋友想做嵌入式方面的开发,于是来问问,如果学习嵌入式开发,具体应该学习哪方面的知识?…

查看全部问答>

VS编译环境的构建

我按照网上的资料按顺序 安装了   1、安装ActiveSync同步工具,我安装的是ActiveSync 4.5     2、安装VS2005(中文)     3、安装vs2005 SP1(431M),(中文补丁)     4、安装。NET CF 2.0SP2   ...…

查看全部问答>

怎样在winCE里用socket通信给FTP上传文件?

如题:小弟想在winCE里用socket通信给远程主机服务器FTP上传文件(现在是上传的bmp文件),现在连接远程FTP正常,上传的文件却大小都为零,打不开,为什么?是不是要设置为二进制传输模式呢???主要代码如下:         in ...…

查看全部问答>

关于一些wifi网络测试软件问题

一、如何测试wifi的信号强度? 二、如何测试wifi的丢包情况? 有没有这样现成的软件,从未做过应用,请各位老兄提示一下,最好有现成的了。…

查看全部问答>

在wince上做phone功能,难度大不大?有无现成方案?

如题!现成方案不是指smartphone 我是新手,刚接触wince,不是很熟 如果要做phone功能,需要做些什么工作,难度大不大? 有哪些现成方案可以选择或者参考的…

查看全部问答>