EDA365电子论坛网

标题: μC/OS-II的中断按键怎么消抖? [打印本页]

作者: BeardRen    时间: 2020-5-9 10:18
标题: μC/OS-II的中断按键怎么消抖?
本帖最后由 BeardRen 于 2020-5-9 10:20 编辑
. x* C9 q; e7 b6 S% i0 C2 A; a
随着性能的提高和价格的降低,越来越多的嵌入式应用采用了ARM处理器。在强大功能及丰富外设的支持下,嵌入式实时操作系统凭借较高的开发效率、可维护性和可靠性成为开发设计的理想选择。
8 B/ l' Z6 g8 v/ p& e引  言
8 |8 ?) M4 |: L+ J' u: \( |( W9 }- y# _( |8 I
随着性能的提高和价格的降低,越来越多的嵌入式应用采用了ARM处理器。在强大功能及丰富外设的支持下,嵌入式实时操作系统凭借较高的开发效率、可维护性和可靠性成为开发设计的理想选择。9 p3 v0 n2 G7 N1 ^

* f) q3 O. K0 G: D4 BμC/OS-II是一个完整的、可移植、可裁减的占先式实时多任务内核。它是用ANSI C语言编写的,包含一小部分汇编语言代码,可以供不同架构的微处理器使用。μC/OS由美国人Jean J.Labrosse于1992年完成,1998年发展到μC/OS-II,目前的版本为μC/OS-II v2.83。2000年,μC/OS得到美国航空管理局(FAA)的认证,可以用于飞行器中。作为一个典型的嵌入式操作系统,μC/OS-II的应用非常广泛,如照相机、医疗器械、音响设备、发动机控制、高速公路电话系统、自动提款机等等。
: z. S0 A1 I! _- s9 {% W* E5 C$ {+ K! P. U1 ?; [0 X
本文主要讨论了μC/OS-II环境下中断按键消抖处理及LCD多级菜单显示的实现问题,并通过一款产品的实例设计阐述了中断按键的处理流程以及多级菜单显示的程序框架。% G6 O3 d/ P) [8 g+ U+ H9 B6 @
1 n' n3 W6 M; ~/ l3 E( t& C/ `
1  显示控制系统硬件设计8 x9 O8 _( Y# m" a/ ^9 ~

8 o7 T$ b! A5 Z  O本设计采用基于Cortex-M3架构的ARM处理器LM3S1960和液晶显示器HS240128A作为显示控制系统的核心,按键采用中断的方式。显示控制系统电路原理图如图1所示。
( }4 _' @2 O1 F( O& k- x3 N$ x' V9 z9 ?
- p) b8 v/ ]! z+ @& N6 H+ V
: _: A, v; f3 _7 h
LM3S1960是Luminary公司推出的高性价比微处理器。它具有256 KBFlash,64 KB RAM,4个32位定时器,6个运动控制PWM,3个UART,2个I2C,2个SSI以及60个用户可用I/O。LM3S1960最高运行频率为50 MHz,其所有GPIO输入可触发中断,支持IRDA的UART,非常适合嵌入式控制系统。$ z+ ]0 l- g$ Q5 b* A# \- p( t

3 m6 X( R" k. t' n$ ~4 S) Q7 P7 Y! rHS240128A是240×128图形点阵液晶显示模块,采用T6963C作为内置控制器,内置字符发生器和32 KB显示缓冲区,具有接口简单、控制指令集功能齐全的特点。2中断按键消抖处理" S) T9 [8 J+ }

& I3 M% D# a9 ?( F* ]: a* m# ?在按键数目较多的情况下,一般采用扫描查询的方式。本设计中按键全部采用中断的方式,主要是考虑到按键数目不多,且处理器的每一个引脚都具有中断功能,实现简单,响应速度快。
* k) t* j; U6 G! v- m; A0 r3 y4 G! a, o$ v2 S! Z" d
在按键的过程中容易产生抖动,没有按键按下有时也可能会有干扰脉冲,如果不加以处理,容易引起误操作。所以,消除抖动是按键处理的必要过程。在一般的处理器中,消抖处理多采用延时判断的方法,这种方法不适合在操作系统中实现。因为在中断中加入大的时延,会大大降低系统的实时性和响应速度,所以本设计采用图2所示的处理方法。# B: u7 u8 E) N2 s7 h% G
0 y, Q: U7 D7 A, J7 v& H. r
9 Y. d6 I/ a: l0 z4 Z% Q
9 L+ o3 v' _# m0 d
图2(a)为按键中断处理程序流程。为了提高中断的实时性,在中断处理过程中尽可能进行少的操作。本设计中只进行了保存键值和发送按键中断信号量的操作,大大提高了中断响应的实时性。
0 W" J6 `3 h8 V% @% L. m5 O! c& k# O) l2 {: g: ]
图2(b)为按键处理任务流程。在完成初始化后,任务开始等待中断发过来需要判断的原始按键值的信号量OSSemPend(OSKeyRawSem,0,&err)。其中的参数OSKeyRawSem是原始按键中断信号量。如果没有信号量发过来,那么任务会在这里被挂起。接收到后首先关闭按键中断,进行系统延时OSTimeDlyHMSM(0,0,0,JitterTime),JitterTime是延时时间。在这个延时的过程中,会进行任务的调度,本任务也会被挂起,直到延时结束,重新被加入等待任务列表。在延时结束任务重新获得CPU后,进行按键的再次判断,判断是否真的发生了按键中断。如果判断正确,则需要等待按键的松开。在判断是否松开的过程中,同样加入了系统延时判断OSTimeDlyHMSM(0,0,0,CheekTime),Cheek27ime是每次检测延时的时间。按键松开后,任务发送按键信号量给其他任务使用OSSemPost(OSKeySem),其中OSKeySem为经过确认的按键中断。如果判断错误,则清除键值。最后,打开按键中断,重新等待新的按键中断。) t4 V1 T+ ~) i+ w9 t4 F5 [
3 LCD多级菜单显示
; P. Y$ b. O. ~# ~, f6 f
) P. f& C) x* k对于多级菜单的编写,控制关系比较复杂,特别是对于不同的界面,人机交互的内容一般都不相同。所以在操作系统实现时,一般把每一级菜单的实现都作为一个任务分别设计。这样思路清晰,不容易出现问题,而且方便删减和增加菜单项,也便于程序的维护和更新。
8 d8 j( p6 t3 e% [- V
" g# k, E  j1 x! u8 [主菜单任务循环体内程序如下:% G( d( V$ y) `" {" G
2 e, z# \$ Z: l/ E" {

# q# w" h7 r, ]" |) u8 Q2 f2 }
) H, I: q( R. }2 w程序说明如下:
8 ]6 P" l8 y# ]" q! |1 S
. O: i9 X0 X. s8 R+ d①在任务开始时判断系统模式SystemMode是否为主菜单模式SysMode_MainMenu。如果不是,那么就需要把主菜单显示任务挂起,直到从其他任务返回。  _0 j; w2 Y- L9 J
: t1 W& g& {* T
②等待按键信号量OSKeySem,根据用户的按键执行相应的操作,RfreshTime是菜单的刷新时间。信号量等待函数在RfreshTime时间内等待,如果等待时间到了按键还没有操作,那么它继续执行下面的程序,err变量会返回错误的数值。
0 O& x7 H+ D- M" m1 c4 Q7 X4 O, Q+ ?9 L7 W# m) }9 X
③如果②中的err返回没有错误,说明在刷新时间内发生了按键中断,那么进入按键的操作处理程序。
# g8 Z& ]( t8 p6 v( J
# a7 r# q6 w0 d0 t  X: O8 P6 Y④根据键值变量KeyNumber进行相应的操作,例如确定按键的处理、返回按键的处理等。  O* j- a8 m1 g

$ P& ]# n: s6 h7 a⑤假如是确定键KEY_OK按下,那么首先把系统子菜单的编号SubMenuNumber赋给SystemMode变量,并进行相应的初始化操作。& E0 T, O' H$ n8 c3 w3 V, V4 i  j& ]
. d* @6 Z* J# J; W
⑥例如是XXXX子菜单,那么在初始化完成后发送信号量OSSemPost(OSDisXXXXSem)。其中,OSDisXXXXSem是子菜单XXXX的显示信号量。2 g! ]3 a8 P5 A* k( T0 Q
4 k' t, |' x7 T4 ~
⑦目的是调用主菜单显示函数。之所以再次进行判断,是因为在⑥中发送信号量要进行系统调度。如果子系统任务优先级比主菜单高,主菜单就会被挂起而去执行子菜单,在子菜单执行间隙会重新调度回来进行主菜单显示,那么就造成显示错误。6 [0 J! E0 e  `2 ~) X

# ]8 @2 ~( B) M子菜单显示的任务流程和主菜单类似。在按返回键时发送启动主菜单的信号量OSSemPost(OSDisMainMenuSem),系统就会重新返回到主菜单。
/ W! c& ]4 {4 d. K4 }9 m7 A+ g8 v" h2 R# K9 j# D
结  语+ q' \& t9 Z6 t  o" b3 p6 y- t

6 f4 z; {8 K. m本文对μC/OS-II操作系统下的中断按键消抖处理及LCD多级菜单显示方法进行了研究,并提出了一种简单、可靠性高、维护方便的实现方案。该方法已经应用于某产品的开发设计中,效果较好,运行稳定可靠。- y! o0 _  U2 Z7 P

22.jpg (15.7 KB, 下载次数: 3)

22.jpg

作者: Racheler    时间: 2020-5-9 13:34
假如是确定键KEY_OK按下,那么首先把系统子菜单的编号SubMenuNumber赋给SystemMode变量,并进行相应的初始化操作。




欢迎光临 EDA365电子论坛网 (https://bbs.eda365.com/) Powered by Discuz! X3.2