GD32基于Systick实现us级和ms级的精准延时方案

/*!
    \file  systick.c
    \brief the systick configuration file
*/

/*
    Copyright (C) 2017 GigaDevice

    2014-12-26, V1.0.0, platform GD32F1x0(x=3,5)
    2016-01-15, V2.0.0, platform GD32F1x0(x=3,5,7,9)
    2016-04-30, V3.0.0, firmware update for GD32F1x0(x=3,5,7,9)
    2017-06-19, V3.1.0, firmware update for GD32F1x0(x=3,5,7,9)
*/

#include "gd32f1x0.h"
#include "systick.h"

volatile static float count_1us = 0;
volatile static float count_1ms = 0;

/*!
    \brief      configure systick
    \param[in]  none
    \param[out] none
    \retval     none
*/
void systick_config(void)
{
    /* systick clock source is from HCLK/8 */
    systick_clksource_set(SYSTICK_CLKSOURCE_HCLK_DIV8);
    count_1us = (float)SystemCoreClock/8000000;
    count_1ms = (float)count_1us * 1000;
}

/*!
    \brief      delay a time in microseconds in polling mode
    \param[in]  count: count in microseconds
    \param[out] none
    \retval     none
*/
void delay_1us(uint32_t count)
{
    uint32_t ctl;
    
    /* reload the count value */
    SysTick->LOAD = (uint32_t)(count * count_1us);
    /* 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 a time in milliseconds in polling mode
    \param[in]  count: count in milliseconds
    \param[out] none
    \retval     none
*/
void delay_1ms(uint32_t count)
{
    uint32_t ctl;
    
    /* reload the count value */
    SysTick->LOAD = (uint32_t)(count * count_1ms);
    /* 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;
}

### GD32F305 单片机滴答定时器延时配置方法 对于GD32F305单片机中的滴答定时器延时功能,通常会使用SysTick定时器来实现精确的微秒或毫秒延迟。以下是详细的配置过程: #### SysTick 定时器初始化函数 为了设置一个基于SysTick延时函数,可以定义如下初始化函数: ```c void systick_delay_init(uint32_t ticks) { if (ticks > 0) { SysTick->LOAD = ticks - 1; /* set reload register */ NVIC_SetPriority(SysTick_IRQn, (1 << __NVIC_PRIO_BITS) - 1);/* set Priority for Systick Interrupt */ SysTick->VAL = 0UL; /* 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 */ } } ``` 此部分代码用于设定加载值启用中断以及计数器[^4]。 #### 实现延时函数 接着创建具体的延时函数`delay_ms()``delay_us()`, 这两个函数分别用来提供毫秒微妙别的等待时间: ```c static volatile uint32_t msTicks; void SysTick_Handler(void){ msTicks++; } // Milliseconds delay function using systick timer. void delay_ms(uint32_t dlyTimeInMs) { uint32_t curTicks; msTicks = 0; curTicks = msTicks; while ((msTicks - curTicks) < dlyTimeInMs); } // Microseconds delay function using busy-wait loop. void delay_us(uint32_t us) { uint32_t i; for(i=us*3;i>0;i--); } ``` 上述实现了通过轮询方式完成指定数量的时间单位流逝后的返回操作;其中`delay_us`采用的是简单的循环消耗CPU周期的方式模拟微秒延时,在实际应用中可能需要根据具体处理器频率调整乘法因子以获得更精准的结果[^4]。 #### 初始化调用 最后在主程序入口处调用`systick_delay_init`来进行全局性的延时服务启动: ```c int main(void) { rcu_periph_clock_enable(RCU_GPIOA); // Initialize system tick with a period of 1 millisecond per tick systick_delay_init(SystemCoreClock / 1000U); while (1) { // Your application code here... // Example usage of delays: delay_ms(500); // Delay half second // ... more codes ... } } ``` 这段代码展示了如何利用系统核心时钟计算每秒钟内的节拍次数,以此为基础构建起整个系统的软实时调度框架[^4]。
评论 9
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值