GD32F470之Timer定时功能开发

前言

关于Timer部分,之前一值使用STM32型MCU,定时器时钟大多数都是基于APB1或APB2上分频获得,再设置预分频值和装载值,即可计算出定时时间。但GD32F470型定时器使用定时时间计算与之有很大区别。

时钟源

使用定时器,首先需要从时钟树上获取以下几个信息:

1、外部晶振:本文举例选用25MHz;

2、系统时钟:通过system_gd32f4xx.c文件中开启对应晶振和系统时钟的宏定义。支持8M、16M和25M三种晶振,系统时钟分别为120MHz、168MHz、200MHz以及240MHz。

本文举例使用25M晶振,系统时钟为200MHz。对应宏定义如下:

#define __SYSTEM_CLOCK_200M_PLL_25M_HXTAL       (uint32_t)(200000000)

此时系统时钟CK_SYS  :200MHz

AHB:CK_SYS       CFG0->APHPSC:0X00  不分频       200MHz

APB1:CK_AHB/4    CFG0->APB1PSC:0X05  4分频      50MHz

APB2:CK_AHB/2    CFG0->APB2PSC:0X04  2分频      100MHz

3、时钟树中定时器挂在哪个总线上

选用Timer1,定时器1挂接在APB1总线上 

4、时钟源选择CFG1->TIMERSEL

CFG1->TIMERSEL默认为0时:

CFG0->APB1PSC:0X05时,时钟源CK_TIMER1 = 2*CK_APB1 = 2*CH_AHB/4 = 100MHz

CFG1->TIMERSEL选择为1时:

CFG0->APB1PSC:0X05时,时钟源CK_TIMER1 = CH_AHB = 200MHz

5、定时时间计算

prescaler:设置用来作为TIMx时钟频率除数的预分频值;

period:设置在下一个更新事件装入活动的自动重装载寄存器周期的值;

定时时间: (prescaler+1)/ CK_TIMER1* (period+1)

例如:prescaler为9(计数从0开始,到prescaler值结束,因而共(prescaler+1)个时钟),period为1999(计数从0开始,到period值结束,因而共(period+1)个时钟)时,CFG1->TIMERSEL为0时,定时时间2ms; CFG1->TIMERSEL为1时,定时时间1ms

 定时器参数配置

定时器实现函数如下:

void DRV_TIM_Config(unsigned int arr, unsigned int psc)
{
    timer_parameter_struct initpara;
    rcu_periph_clock_enable(RCU_TIMER1);//开启时钟
    rcu_timer_clock_prescaler_config(RCU_TIMER_PSC_MUL4);//对应时钟源选择CFG1->TIMERSEL选择为1
    initpara.prescaler         = psc;//设置用来作为TIMx时钟频率除数的预分频值
    initpara.period            = arr;//设置在下一个更新事件装入活动的自动重装载寄存器周期的值
 //   initpara->alignedmode       = TIMER_COUNTER_EDGE;
    initpara.counterdirection  = TIMER_COUNTER_UP;
    initpara.clockdivision     = TIMER_CKDIV_DIV1;

    timer_init(TIMER1, &initpara);
    //清状态,设置中断
    timer_flag_clear(TIMER1, TIMER_FLAG_UP);
    timer_interrupt_enable(TIMER1, TIMER_INT_UP);
    //定时器中断
    /* enable and set timer interrupt priority */
    nvic_irq_enable(TIMER1_IRQn, 1U, 1U);
    timer_enable(TIMER1);
}

main函数中调用以下语句:

DRV_TIM_Config(99,1999);//1ms定时  200MHZ

//定时器中断

void TIMER1_IRQHandler(void)

{

    static unsigned int tcntt = 0;

    if(SET == timer_interrupt_flag_get(TIMER1, TIMER_INT_FLAG_UP))

    {

        //添加定时处理代码

        tcntt++;

        timer_interrupt_flag_clear(TIMER1, TIMER_INT_FLAG_UP);

    }

}

总结:

GD32F470定时器使用时,需要根据定时器时钟挂接总线时钟,定时器时钟源CFG1->TIMERSEL以及预分频值和重装载值来计算定时时间。

  • 2
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
首先,你需要将Timer1的CH0配置为PWM输出模式,并设置PWM的频率和占空比。然后,你需要使用DMA将PWM数据传输到Timer1的CCR寄存器中。 下面是一个简单的示例代码来实现这个功能: ```c // 定义PWM输出的频率和占空比 #define PWM_FREQ 10000 #define PWM_DUTY_CYCLE 50 // 初始化Timer1 CH0为PWM输出模式 void TIM1_CH0_PWM_Init(void) { // 使能Timer1时钟 rcu_periph_clock_enable(RCU_TIMER1); // 配置Timer1 CH0为PWM输出模式 timer_oc_parameter_struct timer_ocinitpara; timer_ocinitpara.oc_mode = TIMER_OC_MODE_PWM0; timer_ocinitpara.output_state = TIMER_CCX_ENABLE; timer_ocinitpara.output_nstate = TIMER_CCXN_DISABLE; timer_ocinitpara.oc_polarity = TIMER_OC_POLARITY_HIGH; timer_ocinitpara.oc_npolarity = TIMER_OCCP_NO_CHANGE; timer_ocinitpara.oc_idle_state = TIMER_OC_IDLE_STATE_LOW; timer_ocinitpara.oc_nidle_state = TIMER_OCNIDLE_STATE_NO_CHANGE; timer_channel_output_config(TIMER1, TIMER_CH_0, &timer_ocinitpara); // 配置Timer1的预分频器和重载值,计算PWM周期 timer_parameter_struct timer_initpara; timer_initpara.prescaler = SystemCoreClock / PWM_FREQ / 1000000 - 1; timer_initpara.alignedmode = TIMER_COUNTER_EDGE_ALIGNED_PWM_MODE; timer_initpara.counterdirection = TIMER_COUNTER_UP; timer_initpara.period = (1000000 / PWM_FREQ) - 1; timer_initpara.clockdivision = TIMER_CKDIV_DIV1; timer_init(TIMER1, &timer_initpara); // 配置PWM占空比 uint16_t duty_cycle = ((100 - PWM_DUTY_CYCLE) * (timer_initpara.period + 1)) / 100; timer_channel_output_pulse_value_config(TIMER1, TIMER_CH_0, duty_cycle); // 使能Timer1 timer_enable(TIMER1); } // 定义要输出的PWM数据 #define PWM_DATA_SIZE 100 uint16_t pwm_data[PWM_DATA_SIZE]; // 初始化DMA void DMA_Init(void) { // 使能DMA时钟 rcu_periph_clock_enable(RCU_DMA); // 配置DMA通道0 dma_parameter_struct dma_initpara; dma_struct_para_init(&dma_initpara); dma_deinit(DMA0, DMA_CH0); dma_initpara.direction = DMA_MEMORY_TO_PERIPHERAL; dma_initpara.memory_addr = (uint32_t)pwm_data; dma_initpara.memory_inc = DMA_MEMORY_INCREASE_ENABLE; dma_initpara.periph_addr = (uint32_t)(&TIMER1->CCR0); dma_initpara.periph_inc = DMA_PERIPH_INCREASE_DISABLE; dma_initpara.memory_width = DMA_MEMORY_WIDTH_16BIT; dma_initpara.periph_width = DMA_PERIPHERAL_WIDTH_16BIT; dma_initpara.number = PWM_DATA_SIZE; dma_initpara.priority = DMA_PRIORITY_ULTRA_HIGH; dma_initpara.circular_mode = DMA_CIRCULAR_MODE_ENABLE; dma_init(DMA0, DMA_CH0, &dma_initpara); // 使能DMA通道0 dma_channel_enable(DMA0, DMA_CH0); } int main(void) { // 初始化PWM输出 TIM1_CH0_PWM_Init(); // 初始化DMA DMA_Init(); // 填充PWM数据 for (int i = 0; i < PWM_DATA_SIZE; i++) { pwm_data[i] = (i & 1) ? 0 : ((100 - PWM_DUTY_CYCLE) * (TIMER1->ARR + 1)) / 100; } while (1) { // 无限循环等待 } } ``` 这个示例代码中,首先通过`TIM1_CH0_PWM_Init()`函数初始化Timer1 CH0为PWM输出模式,并设置PWM的频率和占空比。然后,通过`DMA_Init()`函数初始化DMA通道0,将PWM数据传输到Timer1的CCR寄存器中。最后,通过填充`pwm_data`数组来生成PWM波形。在主函数中,使用一个无限循环来等待。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值