GD32系列的Systick定时器轮询模式配置

4 篇文章 1 订阅
1 篇文章 0 订阅

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档


前言

`在使用GD32F303单片机的时候不可避免要使用到延时功能,如果要求延时不精确的话可以使用for循环语句来达到延时的目的,另外还可以使用定时器来达到定时的目的,但大多数我们会使用系统内部的Systick定时器来延时。


`

一、Systick定时器

我们先看下时钟树
在这里插入图片描述
蓝色的线表示Systick的时钟线,是经过HCLK 8分频后得到,然后我们需要知道Systick定时器是一个24Bit递减计数器,当定时器计数到0会产生一个中断请求,控制和状态寄存器也会在相应位置1,具体可参考cortex-M4的编程手册。如果想让计数器继续工作,就必须往重装载寄存器写入值。

二、Systick的程序说明

1.使用中断延时

代码如下:

#include "gd32f30x.h"
#include "systick.h"

volatile static uint32_t delay;

/*!
    \brief      configure systick
    \param[in]  none
    \param[out] none
    \retval     none
*/
void systick_config(void)
{
    /* setup systick timer for 1000Hz interrupts */
    if (SysTick_Config(SystemCoreClock / 1000U)){
        /* capture error */
        while (1){
        }
    }
    /* configure the systick handler priority */
    NVIC_SetPriority(SysTick_IRQn, 0x00U);
}

/*!
    \brief      delay a time in milliseconds
    \param[in]  count: count in milliseconds
    \param[out] none
    \retval     none
*/
void delay_1ms(uint32_t count)
{
    delay = count;

    while(0U != delay){
    }
}

/*!
    \brief      delay decrement
    \param[in]  none
    \param[out] none
    \retval     none
*/
void delay_decrement(void)
{
    if (0U != delay){
        delay--;
    }
}


//SysTick_Config函数
__STATIC_INLINE uint32_t SysTick_Config(uint32_t ticks)
{
  if ((ticks - 1) > SysTick_LOAD_RELOAD_Msk)  return (1);      /* Reload value impossible */

  SysTick->LOAD  = ticks - 1;                                  /* set reload register */
  NVIC_SetPriority (SysTick_IRQn, (1<<__NVIC_PRIO_BITS) - 1);  /* set Priority for Systick Interrupt */
  SysTick->VAL   = 0;                                          /* Load the SysTick Counter Value */
  SysTick->CTRL  = SysTick_CTRL_CLKSOURCE_Msk |
                   SysTick_CTRL_TICKINT_Msk   |
                   SysTick_CTRL_ENABLE_Msk;                    /* Enable SysTick IRQ and SysTick Timer */
  return (0);                                                  /* Function successful */
}

#endif

SystemCoreClock / 1000U就是要写入SysTick->LOAD这里。如果我们的系统时钟是8Mhz,那周期就是1/8Mhz,重装载的值为8M/1000=8Khz,那就可以算出延时时间为8Khz/8Mhz,也就是1/1000,1ms。打开中断使能和定时器,计数器就开始计时,设置中断优先级。

void SysTick_Handler(void)
{
    delay_decrement();
}

中断服务函数就运行delay_decrement();就是每次进入中断就进行 delay-- 操作,所以delay_1ms(uint32_t count);这个函数只需要给delay一个值就可以知道进入了几次中断,每次中断配置的是1ms,直到delay为0才跳出函数,就可以达到延时多少Ms的功能。
这种方法使得单片机不断进入中断处理,如果1us的话单片机几乎无法执行其它指令,所以一般不用中断延时的办法。

2.轮询方式延时

代码如下:

#include "gd32f30x.h"
#include "systick.h"

//volatile static uint32_t delay;

volatile static float count_us = 0;
volatile static float count_ms = 0;

/*!
    \brief      configure systick
    \param[in]  none
    \param[out] none
    \retval     none
*/
void systick_config(void)
{
    /* systick clock source is from HCLK/8 内部时钟的120M/8分频 */
    systick_clksource_set(SYSTICK_CLKSOURCE_HCLK_DIV8);
    count_us = (float)SystemCoreClock/8000000;//延时1us需要的时钟数量 120M/8M = 15 
    count_ms = (float)count_us * 1000;//延时1ms需要的时钟数量 120M/8M *1000= 15000 
}


/*!
    \brief      delay a time in milliseconds
    \param[in]  count: count in milliseconds
    \param[out] none
    \retval     none
*/
void delay_us(uint32_t count)
{
    uint32_t ctl;
    
    /* reload the count value */
    SysTick->LOAD = (uint32_t)(count * count_us);
    /* clear the current count value */
    SysTick->VAL = 0x0000U;
    /* enable the systick timer */
    SysTick->CTRL = SysTick_CTRL_ENABLE_Msk;
    /* wait for the COUNTFLAG flag set */
    do{
        ctl = SysTick->CTRL;
    }while((ctl&SysTick_CTRL_ENABLE_Msk)&&!(ctl & SysTick_CTRL_COUNTFLAG_Msk));
    /* disable the systick timer */
    SysTick->CTRL &= ~SysTick_CTRL_ENABLE_Msk;
    /* clear the current count value */
    SysTick->VAL = 0x0000U;
}

/*!
    \brief      delay decrement
    \param[in]  none
    \param[out] none
    \retval     none
*/
/*!
    \brief      delay a time in milliseconds in polling mode
    \param[in]  count: count in milliseconds
    \param[out] none
    \retval     none
*/
void delay_ms(uint32_t count)
{
    uint32_t ctl;
    
    /* reload the count value */
    SysTick->LOAD = (uint32_t)(count * count_ms);
    /* clear the current count value */
    SysTick->VAL = 0x0000U;
    /* enable the systick timer */
    SysTick->CTRL = SysTick_CTRL_ENABLE_Msk;
    /* wait for the COUNTFLAG flag set */
    do{
        ctl = SysTick->CTRL;
    }while((ctl&SysTick_CTRL_ENABLE_Msk)&&!(ctl & SysTick_CTRL_COUNTFLAG_Msk));
    /* disable the systick timer */
    SysTick->CTRL &= ~SysTick_CTRL_ENABLE_Msk;
    /* clear the current count value */
    SysTick->VAL = 0x0000U;
}

这里我的 SysTick的时钟源选择HCLK的8分频,HCLK时钟为120Mhz,systick_clksource_set(SYSTICK_CLKSOURCE_HCLK_DIV8);我注释掉这段代码,或者改成SYSTICK_CLKSOURCE_HCLK,不分频效果尽然是一样的,可能内部默认的已经就是8分频的无法更改。

  • 6
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

亚特兰蒂斯MQ

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值