5.1 按键实验
本节课将直接以实验的方式讲解如何使用HAL按键驱动API,本节课的实验内容是按下按键后闪烁LED。
按键API文件
HAL提供了完善按键驱动API,其定义在在hal_key.h和hal_key.c中,如下图所示。
按键的定义与映射
(1)打开头文件hal_led.h,可以找到如下代码:
(2)其中的SW_1到SW_5是方向按键,用得很少,故不展开讲解。SW_6和SW_7是常规按键。
(3)与LED类似,可以在头文件hal_board_cfg.h中配置按键与CC2530引脚的对接关系。以下是SW_6的默认是配置,代码如下:
/* S1 按键 */#define PUSH1_BV BV(1)#define PUSH1_SBIT P0_1//按键按下时的电平#if defined (HAL_BOARD_CC2530EB_REV17)#define PUSH1_POLARITY ACTIVE_HIGH#elif defined (HAL_BOARD_CC2530EB_REV13)#define PUSH1_POLARITY ACTIVE_LOW#else#error Unknown Board Indentifier#endif
(4)通常,这个按键已经能满足大多数设备需求了。配套的ZigBee开发板中的按键连接的引脚正是P0_1。然而,我们按键的电路是按下时为低电平,未按下时为高电平,所以还需要修改一下配置,修改后的配置代码如下:
/* S1 按键 */#define PUSH1_BV BV(1)#define PUSH1_SBIT P0_1//表示低电平时驱动按键#define PUSH1_POLARITY ACTIVE_LOW
提示:
在前面的章节中已经讲解过按键的电路,有需要的读者可以翻阅一下。
处理按键事件
(1)由于按键是受OSAL的调度的,所以在Z-Stack中使用按键之前,必须要现在OSAL中注册。可以在应用层中注册,应用层的初始化函数zclSampleSw_Init默认已经完成了此注册工作,如图所示:
(2)注册后,如果按键被按下,那么就会产生一个应用层的系统事件KEY_CHANGE。打开zcl_samplesw.c文件中的zclSampleSw_event_loop函数,可以找到KEY_CHANGE事件的处理函数zclSampleSw_HandleKeys,如图所示。
(3)事件处理函数zclSampleSw_HandleKeys的代码定义如下:
(4)zclSampleSw_HandleKeys函数默认的工作内容是在LCD显示屏中显示按键按下的信息。开发者可在此处添加按键事件处理代码,例如前面章节讲解过的闪烁LED,参考代码如下:
启用按键的宏定义
(1)与LED类似,在使用按键前,需要启用按键对应的宏定义HAL_KEY。在下图中的Defined sybols中输入以下代码。
HAL_KEY=TRUEISR_KEYINTERRUPT
(2)添加后,如图所示。
(3)定义宏ISR_KEYINTERRUPT是因为系统是使用中断的方式检测按键行为的,读者暂时大概了解一下即可。
调试仿真
编译工程后把程序烧录到开发板中并且运行后,当按键被按下时可以看到LED闪烁。
5.2 HAL 按键框架详解(选修)
概述
上节课讲解HAL 按键的API的使用,本节课将深入分析一下其中原理。如果读者暂时不需要学习其中的原理,可以跳过本节课。
hal_key.c文件
HAL的按键框架的主要代码在hal_key.c文件中,打开该文件可以找到以下这几个重要的函数:
HalKeyInitHalKeyConfigHalKeyPollHAL_ISR_FUNCTION( halKeyPort0Isr, P0INT_VECTOR )
初始化函数HalKeyInit
初始化函数HalKeyInit的代码如下:
1.void HalKeyInit( void )2.{3. /* Initialize previous key to 0 */4. halKeySavedKeys = 0;5.6. HAL_KEY_SW_6_SEL &= ~(HAL_KEY_SW_6_BIT); // I/O为普通GPIO7.8.#if ! defined ENABLE_LED4_DISABLE_S19. HAL_KEY_SW_6_DIR &= ~(HAL_KEY_SW_6_BIT); // I/O配置为输入10.#endif11.12. HAL_KEY_JOY_MOVE_SEL &= ~(HAL_KEY_JOY_MOVE_BIT);13. HAL_KEY_JOY_MOVE_DIR &= ~(HAL_KEY_JOY_MOVE_BIT);14.15. /* Initialize callback function */16. pHalKeyProcessFunction = NULL;17.18. /* Start with key is not configured */19. HalKeyConfigured = FALSE;20.}
初始化函数主要工作内容是把GPIO配置为输入功能。HAL_KEY_SW_6_SEL等定义其实是按键GPIO配置相关的寄存器,这些定义可以在hal_key.c中找到,如果我们需要添加其他按键,也遵循这个方法即可。
配置HAL_KEY_6相关寄存器的定义如下:
在初始化函数HalKeyInit中有个地方需要说明,这个函数中有一段代码(用了快配置HAL_KEY_SW_6这个按键的引脚为输入功能):
1.#if ! defined ENABLE_LED4_DISABLE_S12. HAL_KEY_SW_6_DIR &= ~(HAL_KEY_SW_6_BIT); // I/O配置为输入3.#endif
ZStack的本意是S1按键引脚P0_1可以用来驱动LED,就是和按键复用了,是否用作LED取决于宏ENABLE_LED4_DISABLE_S1是否定义,但是协议栈没有开源的那部分代码中定义了这个宏,也算是ZStack的一个Bug,所以这段代码我们需做修改,把预编译去掉(注释掉),否则按键引脚不会被配置为输入:
1.// #if ! defined ENABLE_LED4_DISABLE_S12. HAL_KEY_SW_6_DIR &= ~(HAL_KEY_SW_6_BIT); // I/O配置为输入3.// #endif
HalKeyConfig
这个函数主要用来配置和中断相关的内容:
1.void HalKeyConfig (bool interruptEnable, halKeyCBack_t cback)2.{3. Hal_KeyIntEnable = interruptEnable; // 标志:是否使能中断4.5. pHalKeyProcessFunction = cback; // 回调函数,按键按下时调用6.7. /* Determine if interrupt is enable or not */8. if (Hal_KeyIntEnable) // 如果使能了中断,必须配置中断相关内容9. {10. PICTL &= ~(HAL_KEY_SW_6_EDGEBIT); /* Clear the edge bit */11. /* For falling edge, the bit must be set. */12. #if (HAL_KEY_SW_6_EDGE == HAL_KEY_FALLING_EDGE)13. PICTL |= HAL_KEY_SW_6_EDGEBIT;14. #endif15.16. /* Interrupt configuration:17. * - Enable interrupt generation at the port18. * - Enable CPU interrupt19. * - Clear any pending interrupt20. */21. HAL_KEY_SW_6_ICTL |= HAL_KEY_SW_6_ICTLBIT;22. HAL_KEY_SW_6_IEN |= HAL_KEY_SW_6_IENBIT;23. HAL_KEY_SW_6_PXIFG = ~(HAL_KEY_SW_6_BIT);24.25. ......... // 屏蔽无关紧要的代码26.27. if (HalKeyConfigured == TRUE)28. {29. osal_stop_timerEx(Hal_TaskID, HAL_KEY_EVENT); // 不需要轮询30. }31. }32. else // 如果没有使用中断,必须把中断相关内容关闭掉33. {34. HAL_KEY_SW_6_ICTL &= ~(HAL_KEY_SW_6_ICTLBIT);35. HAL_KEY_SW_6_IEN &= ~(HAL_KEY_SW_6_IENBIT);36. osal_set_event(Hal_TaskID, HAL_KEY_EVENT); // 按键轮询事件37. }38.39. HalKeyConfigured = TRUE;40.}
HalKeyPoll
1.void HalKeyPoll (void)2.{3. uint8 keys = 0;4. if ((HAL_KEY_JOY_MOVE_PORT & HAL_KEY_JOY_MOVE_BIT))5. {6. keys = halGetJoyKeyInput();7. }8. /*9. * .........10. */11. if (!Hal_KeyIntEnable)12. {13. if (keys == halKeySavedKeys)14. {15. return;16. }17. halKeySavedKeys = keys;18. }19. else20. {21. /* Key interrupt handled here */22. }23. if (HAL_PUSH_BUTTON1()) // 检测HAL_KEY_SW_6是否按下,添加其他24. { // 按键时,同样也需要在这里做检测!25. keys |= HAL_KEY_SW_6;26. }27.28. if (pHalKeyProcessFunction29.#ifdef HAL_LEGACY_KEYS30. && keys31.#endif32. )33. {34. (pHalKeyProcessFunction) (keys, HAL_KEY_STATE_NORMAL);35. }36.}
这个函数是用来查询按键是否按键,如果是中断方式,检测到中断时去抖然后进入这个函数;如果不是中断方式那么采用周期性轮询查看按键是否按下。
函数HAL_ISR_FUNCTION( halKeyPort0Isr, P0INT_VECTOR ):
这是个中断处理函数,中断向量是P0INT_VECTOR也就是P0口的中断,按键连接的是P0_1,按下产生中断就会来到这个函数:
1.HAL_ISR_FUNCTION( halKeyPort0Isr, P0INT_VECTOR )2.{3. HAL_ENTER_ISR();4. if (HAL_KEY_SW_6_PXIFG & HAL_KEY_SW_6_BIT) // P0_1中断5. {6. halProcessKeyInterrupt(); // 调用这个函数来处理中断7. // 最终会调用 HalKeyPoll8. }9. /*10. Clear the CPU interrupt flag for Port_011. PxIFG has to be cleared before PxIF12. */13. HAL_KEY_SW_6_PXIFG = 0; // 清除中断标志14. HAL_KEY_CPU_PORT_0_IF = 0;15. CLEAR_SLEEP_MODE();16. HAL_EXIT_ISR();17.}
最终中断会在函数halProcessKeyInterrupt()中被处理:
1.void halProcessKeyInterrupt (void)2.{3. bool valid=FALSE;4. // 清除HAL_KEY_SW_6引脚对应的中断标志位5. if (HAL_KEY_SW_6_PXIFG & HAL_KEY_SW_6_BIT)6. {7. HAL_KEY_SW_6_PXIFG = ~(HAL_KEY_SW_6_BIT);8. valid = TRUE;9. }10.11. if (valid)12. { // 启动一个事件,25ms后到期(去抖),事件会在hal_drivers.c13. // 中被处理(Hal_ProcessEvent),最终会调用HalKeyPollosal_start_timerEx (Hal_TaskID, HAL_KEY_EVENT,HAL_KEY_DEBOUNCE_VALUE);14. }15.}
问题:上面讲解的函数最终会在哪里被调用呢?
我们提供一种方法可以用来在IAR中查看所有与关键字相关的内容,进而可以找到函数被调用的地方:
这个搜索功能是全局搜索,也就是说会IAR搜索工程中的所有文件,然后匹配我们要搜索的关键字,再将结果显示出来;比如我们想搜索与关键字HalKeyInit相关的所有内容,可以在弹出的对话框中输入这个关键字,然后点击”Find”即可,然后我们根据搜索出来的结果,可以通过鼠标双击对应的结果来查看相关内容:
