完整教程下载地址:http://www.armbbs.cn/forum.php?mod=viewthread&tid=86980
本章教程为大家讲解定时器应用之超时模式的停机唤醒,实际项目中有一定的使用价值,可以方便的配置系统在停机模式运行一段时间,时间到了可以自动唤醒。
38.1 初学者重要提示
38.2 低功耗定时器超时唤醒驱动设计
38.3 低功耗定时器板级支持包(bsp_lptim_pwm.c)
38.4 低功耗定时器驱动移植和使用
38.5 实验例程设计框架
38.6 实验例程说明(MDK)
38.7 实验例程说明(IAR)
38.8 总结
低功耗定时器超时唤醒驱动设计中有几个要注意的事项,下面逐一为大家做个说明。
由前面的第36章节,我们知道LPTIM1的时钟可以由LSE,LSI,APB或者外部输入时钟提供。使用LSE,LSI或者外部输入的好处是停机状态下,LPTIM1也可以正常工作。
System Clock source = PLL (HSE)
SYSCLK(Hz) = 400000000 (CPU Clock)
HCLK(Hz) = 200000000 (AXI and AHBs Clock)
AHB Prescaler = 2
D1 APB3 Prescaler = 2 (APB3 Clock 100MHz)
D2 APB1 Prescaler = 2 (APB1 Clock 100MHz)
D2 APB2 Prescaler = 2 (APB2 Clock 100MHz)
D3 APB4 Prescaler = 2 (APB4 Clock 100MHz)
因为APB1 prescaler != 1, 所以 APB1上的TIMxCLK = APB1 x 2 = 200MHz; 不含这个总线下的LPTIM1
因为APB2 prescaler != 1, 所以 APB2上的TIMxCLK = APB2 x 2 = 200MHz;
APB4上面的TIMxCLK没有分频,所以就是100MHz;
APB1 定时器有 TIM2, TIM3 ,TIM4, TIM5, TIM6, TIM7, TIM12, TIM13, TIM14,LPTIM1
APB2 定时器有 TIM1, TIM8 , TIM15, TIM16,TIM17
APB4 定时器有 LPTIM2,LPTIM3,LPTIM4,LPTIM5
下面为大家讲解下使用LSE,LSI或者APB时钟的配置方法。
选择LSE的配置如下:
#define LPTIM_CLOCK_SOURCE_LSE /* LSE 时钟32768Hz */
RCC_PeriphCLKInitTypeDef RCC_PeriphCLKInitStruct = {0};
RCC_OscInitTypeDef RCC_OscInitStruct = {0};
RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_LSE;
RCC_OscInitStruct.LSEState = RCC_LSE_ON;
RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE;
if (HAL_RCC_OscConfig(&RCC_OscInitStruct)!= HAL_OK)
{
Error_Handler(__FILE__, __LINE__);
}
RCC_PeriphCLKInitStruct.PeriphClockSelection = RCC_PERIPHCLK_LPTIM1;
RCC_PeriphCLKInitStruct.Lptim1ClockSelection = RCC_LPTIM1CLKSOURCE_LSE;
HAL_RCCEx_PeriphCLKConfig(&RCC_PeriphCLKInitStruct);
特别注意程序中置红的地方,这几个地方很容易配置错。配置后LPTIM1就会将LSE作为系统时钟。
选择LSI的配置如下:
//#define LPTIM_CLOCK_SOURCE_LSI /* LSI 时钟约32KHz */
RCC_PeriphCLKInitTypeDef RCC_PeriphCLKInitStruct = {0};
RCC_OscInitTypeDef RCC_OscInitStruct = {0};
RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_LSI;
RCC_OscInitStruct.LSIState = RCC_LSI_ON;
RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE;
if (HAL_RCC_OscConfig(&RCC_OscInitStruct)!= HAL_OK)
{
Error_Handler(__FILE__, __LINE__);
}
RCC_PeriphCLKInitStruct.PeriphClockSelection = RCC_PERIPHCLK_LPTIM1;
RCC_PeriphCLKInitStruct.Lptim1ClockSelection = RCC_LPTIM1CLKSOURCE_LSI;
HAL_RCCEx_PeriphCLKConfig(&RCC_PeriphCLKInitStruct);
使用LSI作为LPTIM1的系统是要注意两点:
1、LSI的实现有一定的误差,具体范围在数据手册有给出,由于不支持温补,温度对其也是有影响的。
2、特别注意程序中置红的地方,这几个地方很容易跟LSE搞混淆(复制粘贴的时候容易搞错)。
选择APB时钟的配置如下:
RCC_PeriphCLKInitTypeDef RCC_PeriphCLKInitStruct = {0};
RCC_PeriphCLKInitStruct.PeriphClockSelection = RCC_PERIPHCLK_LPTIM1;
RCC_PeriphCLKInitStruct.Lptim1ClockSelection = RCC_LPTIM1CLKSOURCE_D2PCLK1;
HAL_RCCEx_PeriphCLKConfig(&RCC_PeriphCLKInitStruct);
使用APB作为LPTIM系统时钟注意以下两点:
1、 LPTIM1 – LPTIM5的最高主频都是100MHz。
2、 注意参数RCC_LPTIM1CLKSOURCE_D2PCLK1。
LPTIM1使用的RCC_LPTIM1CLKSOURCE_D2PCLK1。
LPTIM2使用的RCC_LPTIM2CLKSOURCE_D3PCLK1。
LPTIM3-LPTIM5使用的RCC_LPTIM345CLKSOURCE_D3PCLK1。
下面使用LSE做低功耗定时器的系统时钟,做了8分频,并开启LPTIM1的超时中断。
1. /* 选择LPTIM的时钟源 */
2. #define LPTIM_CLOCK_SOURCE_LSE /* LSE 时钟32768Hz */
3. //#define LPTIM_CLOCK_SOURCE_LSI /* LSI 时钟32768Hz */
4. //#define LPTIM_CLOCK_SOURCE_PCLK /* PCLK 时钟100MHz */
5.
6. LPTIM_HandleTypeDef LptimHandle = {0};
7.
8. /*
9. ******************************************************************************************************
10. * 函 数 名: bsp_InitLPTIM
11. * 功能说明: 初始化LPTIM
12. * 形 参: 无
13. * 返 回 值: 无
14. ******************************************************************************************************
15. */
16. void bsp_InitLPTIM(void)
17. {
18. RCC_PeriphCLKInitTypeDef RCC_PeriphCLKInitStruct = {0};
19.
20.
21. /* ## - 1 - 使能LPTIM时钟和GPIO时钟 ####################################### */
22. __HAL_RCC_LPTIM1_CLK_ENABLE();
23.
24. /* ## - 2 - 配置LPTIM时钟,可以选择LSE,LSI或者PCLK ######################## */
25. #if defined (LPTIM_CLOCK_SOURCE_LSE)
26. {
27. RCC_OscInitTypeDef RCC_OscInitStruct = {0};
28.
29. RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_LSE;
30. RCC_OscInitStruct.LSEState = RCC_LSE_ON;
31. RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE;
32.
33. if (HAL_RCC_OscConfig(&RCC_OscInitStruct)!= HAL_OK)
34. {
35. Error_Handler(__FILE__, __LINE__);
36. }
37.
38. RCC_PeriphCLKInitStruct.PeriphClockSelection = RCC_PERIPHCLK_LPTIM1;
39. RCC_PeriphCLKInitStruct.Lptim1ClockSelection = RCC_LPTIM1CLKSOURCE_LSE;
40. HAL_RCCEx_PeriphCLKConfig(&RCC_PeriphCLKInitStruct);
41. }
42. #elif defined (LPTIM_CLOCK_SOURCE_LSI)
43. {
44. RCC_OscInitTypeDef RCC_OscInitStruct = {0};
45.
46. RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_LSI;
47. RCC_OscInitStruct.LSIState = RCC_LSI_ON;
48. RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE;
49.
50. if (HAL_RCC_OscConfig(&RCC_OscInitStruct)!= HAL_OK)
51. {
52. Error_Handler(__FILE__, __LINE__);
53. }
54.
55. RCC_PeriphCLKInitStruct.PeriphClockSelection = RCC_PERIPHCLK_LPTIM1;
56. RCC_PeriphCLKInitStruct.Lptim1ClockSelection = RCC_LPTIM1CLKSOURCE_LSI;
57. HAL_RCCEx_PeriphCLKConfig(&RCC_PeriphCLKInitStruct);
58. }
59. #elif defined (LPTIM_CLOCK_SOURCE_PCLK)
60. RCC_PeriphCLKInitStruct.PeriphClockSelection = RCC_PERIPHCLK_LPTIM1;
61. RCC_PeriphCLKInitStruct.Lptim1ClockSelection = RCC_LPTIM1CLKSOURCE_LSE;
62. HAL_RCCEx_PeriphCLKConfig(&RCC_PeriphCLKInitStruct);
63. #else
64. #error Please select the LPTIM Clock source inside the bsp_lptim_pwm.c file
65. #endif
66.
67. /* ## - 3 - 配置LPTIM ######################################################## */
68. LptimHandle.Instance = LPTIM1;
69. /* 对应寄存器CKSEL,选择内部时钟源 */
70. LptimHandle.Init.Clock.Source = LPTIM_CLOCKSOURCE_APBCLOCK_LPOSC;
71. /* 设置LPTIM时钟分频 */
72. LptimHandle.Init.Clock.Prescaler = LPTIM_PRESCALER_DIV8;
73. /* LPTIM计数器对内部时钟源计数 */
74. LptimHandle.Init.CounterSource = LPTIM_COUNTERSOURCE_INTERNAL;
75. /* 软件触发 */
76. LptimHandle.Init.Trigger.Source = LPTIM_TRIGSOURCE_SOFTWARE;
77. /* 超时模式用不到这个配置 */
78. LptimHandle.Init.OutputPolarity = LPTIM_OUTPUTPOLARITY_HIGH;
79. /* 比较寄存器和ARR自动重载寄存器选择更改后立即更新 */
80. LptimHandle.Init.UpdateMode = LPTIM_UPDATE_IMMEDIATE;
81. /* 外部输入1,本配置未使用 */
82. LptimHandle.Init.Input1Source = LPTIM_INPUT1SOURCE_GPIO;
83. /* 外部输入2,本配置未使用 */
84. LptimHandle.Init.Input2Source = LPTIM_INPUT2SOURCE_GPIO;
85.
86. if (HAL_LPTIM_Init(&LptimHandle) != HAL_OK)
87. {
88. Error_Handler(__FILE__, __LINE__);
89. }
90.
91. /* ## - 4 - 配置LPTIM ######################################################## */
92. /* 配置中断优先级并使能中断 */
93. HAL_NVIC_SetPriority(LPTIM1_IRQn, 1, 0);
94. HAL_NVIC_EnableIRQ(LPTIM1_IRQn);
95. }
这里把几个关键的地方阐释下:
启动低功耗定时器:
1. /*
2. ******************************************************************************************************
3. * 函 数 名: bsp_StartLPTIM
4. * 功能说明: 启动LPTIM
5. * 形 参: 无
6. * 返 回 值: 无
7. ******************************************************************************************************
8. */
9. void bsp_StartLPTIM(void)
10. {
11. /*
12. ARR是自动重装寄存器,对应函数HAL_LPTIM_TimeOut_Start_IT的第2个参数
13. Compare是比较寄存器,对应函数HAL_LPTIM_TimeOut_Start_IT的第3个参数
14.
15. ---------------------
16. LSE = 32768Hz
17. 分频设置为LPTIM_PRESCALER_DIV8,即8分频(函数bsp_InitLPTIM里面做的初始化配置)
18. ARR自动重载寄存器 = 32768
19. 实际测试发现溢出中断与ARR寄存器无关,全部由第3个参数,Compare寄存器决定
20.
21. LPTIM的计数器计数1次的时间是 1 / (32768 / 8) = 8 /32768。
22. 第三个参数配置的是32767,那么计数到32767就是 (32767 + 1)*(8 /32768) = 8秒,计算的时候要加1。
23. */
24. if (HAL_LPTIM_TimeOut_Start_IT(&LptimHandle, 0, 32767) != HAL_OK)
25. {
26. Error_Handler(__FILE__, __LINE__);
27. }
28. }
这里把几个关键的地方阐释下:
低功耗定时器中断的实现如下:
1. /*
2. ******************************************************************************************************
3. * 函 数 名: LPTIM1_IRQHandler
4. * 功能说明: LPTIM1中断服务程序
5. * 形 参: 无
6. * 返 回 值: 无
7. ******************************************************************************************************
8. */
9. void LPTIM1_IRQHandler(void)
10. {
11. if((LPTIM1->ISR & LPTIM_FLAG_CMPM) != RESET)
12. {
13. /* 清除比较匹配中断 */
14. LPTIM1->ICR = LPTIM_FLAG_CMPM;
15.
16. /* 关闭溢出中断 */
17. HAL_LPTIM_TimeOut_Stop_IT(&LptimHandle);
18.
19. bsp_LedToggle(4);
20. }
21. }
这里把几个关键的地方阐释下:
这里再强调下低功耗定时器唤醒的三个注意事项。
低功耗定时器驱动文件bsp_lptim_pwm.c供用户调用的两个函数:
下面将这两个函数的使用为大家做个说明。
函数原型:
void bsp_InitLPTIM(void)
函数描述:
此函数使用LSE做低功耗定时器的系统时钟,做了8分频,并开启LPTIM1的超时中断。
注意事项:
使用举例:
初始化函数在bsp.c文件的bsp_Init函数里面调用。
函数原型:
void bsp_StartLPTIM(void)
函数描述:
此函数通过调用函数HAL_LPTIM_TimeOut_Start_IT来启动低功耗定时器的超时模式,并开启了相应中断。
注意事项:
使用举例:
调用此函数前优先调用初始化函数bsp_InitLPTIM即可。
低功耗定时器的移植比较简单: