当前位置: 首页 > news >正文

【GD32】04 - Timer定时器

GD32中的定时器

GD32E230中有七个定时器,六种类型,其中通用的L4版本有两个,其他类型的各一个。

那我们就以通用L4这个类型来敲代码,其他流程是通用的。

通用L4

虽然每种类型的定时器都有自己的结构框图,但是其实大差不差,定时器来来回回不就那几个寄存器嘛,自动重装载寄存器,预分频器,要输出个PWM就再来个输出比较。

要注意的就是通用L4这边的时钟源只有一个。

因此用GD32的定时器的时候,要去手册翻翻对应的定时器里的介绍,因为GD32里定时器种类还不少,细节上还是有差异的。

那我们来看看给L4提供的时钟源是多少频率的,这关系到我们后面参数的设置。

从左边AHB过来是72MHz,给Timer0,14,15,16提供时钟之前经过一个分频器,这个分频器是如果APB2的分频是1分频,那么它也是1分频,如果不是,那么是2分频,而APB2就是1分频,因此它也是1分频,也就是说没有减少我们的时钟频率,因此给L4的时钟频率就是72MHz。

定时器溢出频率的公式跟STM32是一样的,就是时钟频率/((自动重装载寄存器的值+1)*(预分频器的值+1))

固件库函数

rcu_periph_clock_enable(RCU_TIMER15)

定时器相关函数很多,我们只挑这次用到的讲。

timer_init

初始化定时器,参数一选择是哪一个定时器。

参数二传入一个结构体变量的指针。

我们来看看这个结构体。

通过配置预分频值和周期来控制定时器溢出的频率,就用上面的公式,把这个周期看成是自动重装载计数器就行。

对齐方式一般选择边缘对齐TIMER_COUNTER_EDGE

计数方式选择向上计数TIMER_COUNTER_UP,L4里也只能向上计数。

时钟分频因子这个是输入捕获的时候才用的到的,我们可以不配。

重复计数器值是只溢出多少次之后再溢出才进一次中断的,不过只有高级定时器才有,我们也不配。

timer_enable

使能定时器,当我们配置完定时器包括中断之后就可以使能了。

timer_disable

失能定时器。

timer_prescaler_config

配置预分频器的模式,我们选择立即加载。

timer_interrupt_enable

使能中断,中断源我们就选第一个,更新中断,这是不管哪一种定时器都能用的,其他参数可以参考上面表格的说法。

timer_interrupt_flag_get

获取中断标志位,一样是在中断处理函数中判断是哪个中断源触发的中断。

timer_interrupt_flag_clear

清除中断标志位,除了在中断处理函数中使用,我们在一开始配置定时器的时候就需要用一下,否则会马上触发一次中断。

上面这些函数就足够我们进行最基础的一个计时了,但是还有个问题,就是中断,中断都需要配置NVIC。

NVIC使能的函数我们在上一篇外部中断里说过了,这边就直接把代码贴出来了。

nvic_irq_enable(TIMER15_IRQn,1);

最后一个就是中断处理函数了,我们一样是从启动文件的汇编文件里去找。

秒表示例代码

#include "gd32e23x.h"
#include "systick.h"
#include "OLED.h"uint16_t Z_Time_s = 0 ,Z_Time_10ms = 0;void TIMER15_IRQHandler(void){    if(timer_interrupt_flag_get(TIMER15,TIMER_INT_FLAG_UP) == SET){        timer_interrupt_flag_clear(TIMER15,TIMER_INT_FLAG_UP); //清除中断标志位   if(++Z_Time_10ms >= 100) {Z_Time_10ms = 0;++Z_Time_s;}}
} int main(void){systick_config();OLED_Init();rcu_periph_clock_enable(RCU_TIMER15);                   //开启定时器时钟  timer_parameter_struct timer_initpara;                  timer_initpara.prescaler = 200 -1;                      //预分频timer_initpara.alignedmode = TIMER_COUNTER_EDGE;        //边缘对齐timer_initpara.counterdirection = TIMER_COUNTER_UP;     //向上计数timer_initpara.period = 3600  - 1;                      //周期timer_init(TIMER15,&timer_initpara);                    //初始化定时器timer_prescaler_config(TIMER15, 1, TIMER_PSC_RELOAD_NOW);//设置预分频器为立即加载模式timer_interrupt_flag_clear(TIMER15, TIMER_INT_FLAG_UP); //清除中断标志位,否则一开始就会进一次中断nvic_irq_enable(TIMER15_IRQn, 1);                       //设置中断优先级timer_interrupt_enable(TIMER15, TIMER_INT_UP);          //使能更新中断 timer_enable(TIMER15);                                  //使能定时器OLED_ShowString(1,5,":");while(1){OLED_ShowNum(1,1,Z_Time_s,4);OLED_ShowNum(1,6,Z_Time_10ms,2);}
}

我这边做个秒表的效果,以精度在10ms,因此我们需要每秒定时器溢出中断100次。

因为时钟源是72MHz,因此需要让72MHz去除以(周期+1)再除以(预分频+1)最终等于100即可。

最后一点要注意的就是这俩寄存器是16位的,因此设置周期和预分频的值的时候要注意不要超过65535了。

大家可以配合上个文章的外部中断,在这个秒表的基础上加几个按钮,实现暂停继续,清零等功能。就当是课后作业了(bushi)

http://www.lryc.cn/news/359277.html

相关文章:

  • Golang | Leetcode Golang题解之第123题买卖股票的最佳时机III
  • Leetcode2028. 找出缺失的观测数据
  • 如何在CentOS中合理划分磁盘空间以优化系统性能
  • 算法(十一)贪婪算法
  • Rust之函数式语言特性:迭代器和闭包(一):概述
  • 配置资源管理
  • unity2020打包webGL时卡进程问题
  • 云原生架构相关技术_3.无服务器技术
  • Leetcode:Z 字形变换
  • Python 3 判断文件是否存在
  • (深度学习记录)第TR3周:Transformer 算法详解
  • 谷神前端组件增强:自定义列
  • 31-ESP32-S3-WIFI篇-02 Event Group (事件标记组)
  • 构建企业级AI私有知识库
  • C语言王国——杨氏矩阵
  • 陪玩小程序都需要怎么做?
  • postgressql——子事务可见性判断 性能问题(8)
  • 20240531在飞凌的OK3588-C开发板上跑原厂的Buildroot测试USB摄像头
  • 从0开始学统计-什么是回归?
  • Element-ui使用上传时弹框选择文件类型
  • 原生小程序一键获取手机号
  • ARM虚拟机安装OMV
  • 【协议开发系列】梳理关于TCP和UDP两种协议的区别和使用场景
  • vue blob实现自定义多sheet数据导出到excel文件
  • Python—面向对象小解(3)
  • Nginx超时时间
  • Imgs,GT,Edge,Gradient_all,Gradient_Foreground
  • 自学成才Flutter 弹性布局、线性布局
  • Part 3.1 深度优先搜索
  • 前端Vue小兔鲜儿电商项目实战Day03