RT-Thread 4.0.0

访问硬件定时器设备

应用程序通过 RT-Thread 提供的 I/O 设备管理接口来访问硬件定时器设备,相关接口如下所示:

函数 描述
rt_device_find() 查找定时器设备
rt_device_open() 以读写方式打开定时器设备
rt_device_set_rx_indicate() 设置超时回调函数
rt_device_control() 控制定时器设备,可以设置定时模式(单次/周期)/计数频率,或者停止定时器
rt_device_write() 设置定时器超时值,定时器随即启动
rt_device_read() 获取定时器当前值
rt_device_close() 关闭定时器设备

RT-Thread 提供的 I/O 设备硬件定时器,示例仅提供最通用简单的定时功能,其他定时器高级功能需自行在control中添加;

下面对基于CubeMX、Hal库的BSP的硬件定时器的使用做简单描述。

配置CubeMX

配置之后生成代码,

在 stm32f4xx_hal_conf.h 中 会实现 hal模块驱动

#define HAL_TIM_MODULE_ENABLED

修改工程目录下的 Kconfig

在 Kconfig 中添加对 TIM的支持

    menuconfig BSP_USING_TIM
bool "Enable Hardware TIM"
default n
select RT_USING_HWTIMER
if BSP_USING_TIM
config BSP_USING_TIM1
bool "Enable TIM1"
default n config BSP_USING_TIM2
bool "Enable TIM2"
default n config BSP_USING_TIM3
bool "Enable TIM3"
default n config BSP_USING_TIM4
bool "Enable TIM4"
default n config BSP_USING_TIM5
bool "Enable TIM5"
default n config BSP_USING_TIM6
bool "Enable TIM6"
default n

然后使用Env工具 menuconfig 中使能 TIM3、TIM4

然后 scons --target=mdk5

用keil打开工程后在 tim_config.h 中 添加 TIM3、TIM4的配置

#ifdef BSP_USING_TIM3
#ifndef TIM3_CONFIG
#define TIM3_CONFIG \
{ \
.tim_handle.Instance = TIM3, \
.tim_irqn = TIM3_IRQn, \
.name = "timer3", \
}
#endif /* TIM3_CONFIG */
#endif /* BSP_USING_TIM3 */ #ifdef BSP_USING_TIM4
#ifndef TIM4_CONFIG
#define TIM4_CONFIG \
{ \
.tim_handle.Instance = TIM4, \
.tim_irqn = TIM4_IRQn, \
.name = "timer4", \
}
#endif /* TIM4_CONFIG */
#endif /* BSP_USING_TIM4 */

然后就可以在应用中直接操作设备名为 "timer3" 和 "timer4" 的设备了。

设备驱动分析

定时器设备 I/O 实现

hwtimer.c  hwtimer.h

定时器底层驱动实现(操作Hal库)

drv_hwtimer.c  drv_hwtimer.h

下面主要追踪以下定时器设备的注册及其初始化

在 drv_hwtimer.c 中  定时器自动初始化,并注册设备

static int stm32_hwtimer_init(void)

INIT_BOARD_EXPORT(stm32_hwtimer_init);

其中 hwtimer_ops.rt_hwtimer_init 会 调用 timer_init

static void timer_init(struct rt_hwtimer_device *timer, rt_uint32_t state)
{
uint32_t prescaler_value = ;
TIM_HandleTypeDef *tim = RT_NULL;
struct stm32_hwtimer *tim_device = RT_NULL; RT_ASSERT(timer != RT_NULL);
if (state)
{
tim = (TIM_HandleTypeDef *)timer->parent.user_data;
tim_device = (struct stm32_hwtimer *)timer; /* time init */
#if defined(SOC_SERIES_STM32F4) || defined(SOC_SERIES_STM32F7)
if (tim->Instance == TIM9 || tim->Instance == TIM10 || tim->Instance == TIM11)
#elif defined(SOC_SERIES_STM32L4)
if (tim->Instance == TIM15 || tim->Instance == TIM16 || tim->Instance == TIM17)
#elif defined(SOC_SERIES_STM32F1) || defined(SOC_SERIES_STM32F0)
if ()
#endif
{
#ifndef SOC_SERIES_STM32F0
prescaler_value = (uint32_t)(HAL_RCC_GetPCLK2Freq() * / ) - ;
#endif
}
else
{
prescaler_value = (uint32_t)(HAL_RCC_GetPCLK1Freq() * / ) - ;
}
tim->Init.Period = - ;
tim->Init.Prescaler = prescaler_value;
tim->Init.ClockDivision = TIM_CLOCKDIVISION_DIV1;
if (timer->info->cntmode == HWTIMER_CNTMODE_UP)
{
tim->Init.CounterMode = TIM_COUNTERMODE_UP;
}
else
{
tim->Init.CounterMode = TIM_COUNTERMODE_DOWN;
}
tim->Init.RepetitionCounter = ;
#if defined(SOC_SERIES_STM32F1) || defined(SOC_SERIES_STM32L4) || defined(SOC_SERIES_STM32F0)
tim->Init.AutoReloadPreload = TIM_AUTORELOAD_PRELOAD_DISABLE;
#endif
if (HAL_TIM_Base_Init(tim) != HAL_OK)
{
LOG_E("%s init failed", tim_device->name);
return;
}
else
{
/* set the TIMx priority */
HAL_NVIC_SetPriority(tim_device->tim_irqn, , ); /* enable the TIMx global Interrupt */
HAL_NVIC_EnableIRQ(tim_device->tim_irqn); /* clear update flag */
__HAL_TIM_CLEAR_FLAG(tim, TIM_FLAG_UPDATE);
/* enable update request source */
__HAL_TIM_URS_ENABLE(tim); LOG_D("%s init success", tim_device->name);
}
}
}

我们发现之前设置的配置并没有使用,而是直接默认设置成频率10K,周期1s的定时器;

可根据情况后面用control修改,或注释掉 tim->Init. 的几条赋值语句,使用 user_data 传递过来的默认配置;

同时 stm32f4xx_hal_msp.c 中的TIM硬件初始化,也仅仅只是打开TIM外设时钟,所有锁CubeMX中只要使能对应TIM即可,无需配置;

void HAL_TIM_Base_MspInit(TIM_HandleTypeDef* htim_base)
{
if(htim_base->Instance==TIM3)
{
/* USER CODE BEGIN TIM3_MspInit 0 */ /* USER CODE END TIM3_MspInit 0 */
/* Peripheral clock enable */
__HAL_RCC_TIM3_CLK_ENABLE();
/* USER CODE BEGIN TIM3_MspInit 1 */ /* USER CODE END TIM3_MspInit 1 */
}
else if(htim_base->Instance==TIM4)
{
/* USER CODE BEGIN TIM4_MspInit 0 */ /* USER CODE END TIM4_MspInit 0 */
/* Peripheral clock enable */
__HAL_RCC_TIM4_CLK_ENABLE();
/* USER CODE BEGIN TIM4_MspInit 1 */ /* USER CODE END TIM4_MspInit 1 */
} }

最新文章

  1. SQL Server 2016 CTP2.2 安装手记
  2. 【5集iCore3_ADP演示视频】5-4 iCore3与应用开发平台的组装与拆卸
  3. Sunny-Code 最终版总结会议
  4. September 27th 2016 Week 40th Tuesday
  5. BZOJ 2433 智能车比赛(计算几何+最短路)
  6. Cassandra——类似levelDB的基于p2p架构的分布式NOSQL数据库
  7. URAL-1991 The battle near the swamp 水题
  8. Java异常的一个小知识
  9. 转载 JQuery.data()方法学习
  10. php sscanf() 函数使用
  11. (3)markdown软件的使用
  12. ACM HDU Bone Collector 01背包
  13. 2017-06-19 (cp mkdir rm 运行级别及修改)
  14. Apache Commons Configuration读取xml配置
  15. ajax 三种数据格式
  16. django请求和响应
  17. go框架gin的使用
  18. Luogu4630 APIO2018 Duathlon 圆方树、树形DP
  19. AJAX请求头Content-type
  20. 关于treeMap

热门文章

  1. Light oj-1100 - Again Array Queries,又是这个题,上次那个题用的线段树,这题差点就陷坑里了,简单的抽屉原理加暴力就可以了,真是坑~~
  2. 听dalao讲课 7.26
  3. How many ways?? 矩阵快速幂 邻接矩阵意义
  4. Codeforces Round #343 (Div. 2)【A,B水题】
  5. SpringBoot学习day01
  6. WCF - 自定义绑定
  7. how to get geometry type of layer using IMapServer3 and IMapLayerInfo? (C#)
  8. 【CV论文阅读】Image Captioning 总结
  9. Vue中对获取的数据进行重新排序
  10. Android应用程序安装过程浅析