国民技术
直播中

hehung

9年用户 659经验值
擅长:嵌入式技术
私信 关注

【国民技术N32项目移植】4. 硬件定时器实现2ms定时

过往分享

【国民技术N32项目移植】1. 新建工程+LED与按键跳坑总结

【国民技术N32项目移植】2. 硬件IIC驱动OLED

【国民技术N32项目移植】3. 硬件IIC驱动RT-Thread OS SSD1306软件包

前言

项目中需要使用到定时器,定时时间为2ms,本文将实现2ms的硬件定时器的操作逻辑。

本文基于RT-Thread OS来操作硬件定时器,因为N32G45XVL-STB的底层驱动官方已经实现了,所以直接使用RT-Thread OS提供的硬件定时器的API是较为简单的操作方法。

本文基于RT-Thread studio开发。

使能硬件定时器1

如下图,使能硬件定时器1。

1.jpg

关于硬件定时器的操作逻辑可以参考RT-Thread官方教程,因为使用RT-Thread,用户无需关注底层驱动的实现,只需要按照RT-Thread OS提供的API操作接口。

https://www.rt-thread.org/document/site/#/rt-thread-version/rt-thread-standard/programming-manual/device/hwtimer/hwtimer

软件实现

  1. 定时器初始化,初始化定时器为每2ms超时一次,注册回调函数Timer1_TimeoutCallback,超时之后会自动调用回调函数Timer1_TimeoutCallback。
#include <rtthread.h>
#include <rtdevice.h>

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


static rt_err_t Timer1_TimeoutCallback(rt_device_t dev, rt_size_t size);


/* 定时器初始化函数 */
void Timer_Init(void)
{
    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);
    }

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

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

    /* 设置计数频率(若未设置该项,默认为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);
    }

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

/* 定时器超时回调函数 */
static rt_err_t Timer1_TimeoutCallback(rt_device_t dev, rt_size_t size)
{
    static uint64_t i;

    i++;
    if (i > 500)
    {
        i = 0;
        rt_kprintf("1s timer \n");
    }

    return 0;
}

上述代码实现的软件逻辑是,没2ms进入一次回调函数,在回调函数中累加500次进入则打印字符串"1s timer"到串口,可以使用串口调试助手监控打印信息。

实现效果

如下图,为串口调试助手监控的打印信息,粗略估算2ms定时是准确的。

2.jpg

更多回帖

发帖
×
20
完善资料,
赚取积分