学习笔记(2)RT-Thread使用硬件定时器(HWTIMER)

文章介绍了如何在STM32ZET6微控制器上使用RT-Thread操作系统配置和使用硬件定时器。通过开启RT-ThreadSettings中的硬件定时器,编辑board.h和stm32f1xx_hal_conf_bak.h文件,配置CubeMX以及处理编译冲突。在main.c中编写回调函数和示例代码,实现定时器超时回调功能,每隔5秒打印消息。注意回调函数中不能使用延迟函数。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

本篇文章使用的是芯片是STM32ZET6。

参考的RT-Thread开发文档:HWTIMER设备 (rt-thread.org)

首先我们可以打开board.h文件,查看配置硬件定时器需要哪些步骤。

 和使用其他外设一样,使用硬件定时器主要分为4步。

1:在RT-Thread Settings中开启硬件定时器:

 2:打开board.h文件,并且将

#define BSP_USING_TIM
#define BSP_USING_TIM2 取消掉注释。(由于我使用的为定时器2,所以将其修改为TIM2)

 3:打开CubeMx,进行配置定时器的配置。

 生成文件后将stm32f1xx_hal_msp.c中的HAL_TIM_Base_MspInit函数复制到board.c中.

 拷贝过去以后,发现编译报错说函数重复定义,只需要双击报错语句,就会自动跳转到报错的地方,将重复函数删除即可。

 4:打开driver目录下的stm32f1xx_hal_conf_bak.h文件,将HAL_TIM_MODULE_ENABLED取消掉注释。

这样我们就配置好了定时器2。 

假如说程序报错说我们没有TIM_CONFIG,那我们可以打开tim_config.h文件,在TIM2_CONFIG前面加入TIM1_CONFIG.(rt-thread好像没有TIM1_CONFIG所以需要自己加),里面的参数和TIM2_CONFIG相同,需要将TIM2改成TIM1,但是会有其他的一些问题,所以我们用硬件定时器的话我们就用有的TIM2、3、4、5)就行了

 随后我们在main.c中进行函数的编写。这里我参考了官方文档。程序如下:

/*
 * 程序清单:这是一个 hwtimer 设备使用例程
 * 例程导出了 hwtimer_sample 命令到控制终端
 * 命令调用格式:hwtimer_sample
 * 程序功能:硬件定时器超时回调函数周期性的打印当前tick值,2次tick值之差换算为时间等同于定时时间值。
*/

#include <rtthread.h>
#include <rtdevice.h>

#define HWTIMER_DEV_NAME   "timer2"     /* 定时器名称 */

/* 定时器超时回调函数 */
static rt_err_t timeout_cb(rt_device_t dev, rt_size_t size)
{
    rt_kprintf("this is hwtimer timeout callback fucntion!\n");
    rt_kprintf("tick is :%d !\n", rt_tick_get());

    return 0;
}

static int hwtimer_sample(int argc, char *argv[])
{
    rt_err_t ret = RT_EOK;
    rt_hwtimerval_t timeout_s;      /* 定时器超时值 */
    rt_device_t hw_dev = RT_NULL;   /* 定时器设备句柄 */
    rt_hwtimer_mode_t mode;         /* 定时器模式 */
    rt_uint32_t freq = 10000;               /* 计数频率 */

    /* 查找定时器设备 */
    hw_dev = rt_device_find(HWTIMER_DEV_NAME);
    if (hw_dev == RT_NULL)
    {
        rt_kprintf("hwtimer sample run failed! can't find %s device!\n", HWTIMER_DEV_NAME);
        return RT_ERROR;
    }

    /* 以读写方式打开设备 */
    ret = rt_device_open(hw_dev, RT_DEVICE_OFLAG_RDWR);
    if (ret != RT_EOK)
    {
        rt_kprintf("open %s device failed!\n", HWTIMER_DEV_NAME);
        return ret;
    }

    /* 设置超时回调函数 */
    rt_device_set_rx_indicate(hw_dev, timeout_cb);

    /* 设置计数频率(若未设置该项,默认为1Mhz 或 支持的最小计数频率) */
    rt_device_control(hw_dev, HWTIMER_CTRL_FREQ_SET, &freq);
    /* 设置模式为周期性定时器(若未设置,默认是HWTIMER_MODE_ONESHOT)*/
    mode = HWTIMER_MODE_PERIOD;
    ret = rt_device_control(hw_dev, HWTIMER_CTRL_MODE_SET, &mode);
    if (ret != RT_EOK)
    {
        rt_kprintf("set mode failed! ret is :%d\n", ret);
        return ret;
    }

    /* 设置定时器超时值为5s并启动定时器 */
    timeout_s.sec = 5;      /* 秒 */
    timeout_s.usec = 0;     /* 微秒 */
    if (rt_device_write(hw_dev, 0, &timeout_s, sizeof(timeout_s)) != sizeof(timeout_s))
    {
        rt_kprintf("set timeout value failed\n");
        return RT_ERROR;
    }

    /* 延时3500ms */
    rt_thread_mdelay(3500);

    /* 读取定时器当前值 */
    rt_device_read(hw_dev, 0, &timeout_s, sizeof(timeout_s));
    rt_kprintf("Read: Sec = %d, Usec = %d\n", timeout_s.sec, timeout_s.usec);

    return ret;
}
/* 导出到 msh 命令列表中 */
MSH_CMD_EXPORT(hwtimer_sample, hwtimer sample);

int main(void)
{
    rt_kprintf("hello");
}

这里需要注意的是,我们必须要定义一个main函数,才能在调试串口中使用hwtimer_sample这个指令,如果没有main函数的话,我们在串口调试中是运行不了hwtimer_sample。同时我们需要根据自己的需求来更改程序中的HWTIMER_DEV_NAME即定时器名称、timeout_cb即回调函数、freq计数频率、mode模式和定时时间   

timeout_s.sec = 5;      /* 秒 */
timeout_s.usec = 0;     /* 微秒 */。

随后我们将程序下载到开发板上,我们就可以通过串口查看到,每隔5s就会打印一条消息。

 证明我们使用硬件定时器成功。需要注意的是,我们在超时回调函数timeout_cb中,不能用rt_thread_mdelay这样的延时函数。比如我们想在超时回调函数中让指示灯进行闪烁这是做不到的。

### nRF52833 协议栈中的定时器使用 对于nRF52833设备而言,在其协议栈中,定时器模块扮演着至关重要的角色。该器件支持多个硬件定时器,这些定时器可以通过软件配置用于各种应用场景。 #### 定时器初始化与配置 为了启动一个定时器,开发者需先通过调用`app_timer_init()`函数完成全局定时器服务的初始化工作[^1]。此过程确保了后续创建的具体计时实例能够正常运作于系统之上。当准备就绪之后,则可以利用如下所示的方式定义并激活单个定时器: ```c static void timer_handler(void * p_context) { // Timer event handler code here. } void create_and_start_timer(uint32_t timeout_in_ms, bool is_repeating) { ret_code_t err_code; static app_timer_id_t my_timer_id; err_code = app_timer_create(&my_timer_id, APP_TIMER_MODE_REPEATED, timer_handler); APP_ERROR_CHECK(err_code); err_code = app_timer_start(my_timer_id, APP_TIMER_TICKS(timeout_in_ms), NULL); APP_ERROR_CHECK(err_code); } ``` 上述代码片段展示了如何基于应用层API建立重复触发模式下的定时任务;其中`APP_TIMER_MODE_SINGLE_SHOT`可用于一次性延迟操作而无需循环执行。 #### 获取当前时间戳 有时应用程序可能需要记录特定事件发生的确切时刻或是计算两个不同瞬间之间的时间差值。此时可借助`sdk_config.h`文件内预设宏定义来获取高精度计数值作为参考依据之一: ```c #include "nrf.h" uint32_t get_current_tick_count() { return NRF_RTT->COUNTER; /* For RTT */ // 或者针对其他类型的计数器源... } ``` 值得注意的是,这里给出的例子仅适用于某些特殊场景,并不是所有情况下最推荐的做法。通常建议优先考虑采用SDK所提供的标准化接口来进行此类操作以保持兼容性和稳定性。 #### 停止与删除定时器 一旦不再需要某个已注册的任务计划表项,应当及时释放其所占用资源以免造成不必要的浪费。这可通过下面的方法轻松达成目的: ```c ret_code_t stop_and_delete_timer(app_timer_id_t timer_id) { ret_code_t err_code; err_code = app_timer_stop(timer_id); if (err_code != NRFX_SUCCESS){ return err_code; } err_code = app_timer_delete(timer_id); return err_code; } ``` 以上就是有关nRF52833平台下定时器组件的基础介绍及其典型运用案例解析。希望可以帮助到正在探索这一领域的朋友!
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值