7. GD32E230 SysTick滴答定时器:从寄存器配置到1ms精准延时实战
7. GD32E230 SysTick滴答定时器从寄存器配置到1ms精准延时实战大家好我是老李一个在嵌入式行业摸爬滚打了十几年的工程师。最近有不少朋友从STM32转战到国产的GD32平台特别是GD32E230这款性价比很高的MCU经常问我“老李这SysTick定时器怎么用啊和STM32一样吗” 今天我就以GD32E230为例手把手带大家从最基础的寄存器开始一直讲到如何用官方库函数实现精准的毫秒、微秒延时最后再用这个延时让LED灯闪起来。整个过程我会把原理讲透把坑点说清保证你学完就能用上。1. SysTick是什么为什么它如此重要咱们先来认识一下今天的主角——SysTick定时器。你可以把它想象成单片机内部自带的一个“心脏起搏器”或者“节拍器”。它是什么SysTick是ARM Cortex-M系列内核包括咱们GD32E230用的Cortex-M23内部自带的一个24位向下计数器。简单说就是一个可以从某个数值比如1000开始每过一个时钟周期就减1减到0就自动重头再来的定时器。为什么它特别重要移植性极佳因为它是ARM内核自带的所以所有基于Cortex-M23的芯片不管哪个厂家都有这个定时器。你用GD32E230写的延时函数理论上换到其他M23内核的芯片上也能用不像每个芯片独有的通用定时器那样需要大改。操作系统的“心跳”如果你想玩RTOS实时操作系统比如FreeRTOS、uC/OS那SysTick就是它们的生命线。操作系统靠它来产生固定的时间片进行任务调度。没有SysTick操作系统就跑不起来。简单可靠的延时源对于不需要特别复杂定时功能的应用比如让LED闪烁、按键消抖用SysTick来实现延时非常方便不需要去配置那些寄存器多多的通用定时器。在GD32E230里SysTick的时钟源可以有两种选择通过配置来选择外部时钟HCLK/8这是默认的时钟源由系统主时钟HCLK经过8分频得到。如果你的系统主频是72MHz那这个时钟就是9MHz。内核时钟HCLK直接使用系统主时钟HCLK作为时钟源。速度更快精度更高。注意具体选择哪个时钟源需要根据你的系统时钟频率和所需的定时精度来决定。官方库函数默认配置以及我们常用的delay_1ms函数通常是基于外部时钟HCLK/8来计算的这点后面会详细说。2. 深入核心SysTick的寄存器要真正理解SysTick咱们得看看它背后是怎么工作的。SysTick主要由四个寄存器控制这些寄存器定义在ARM提供的核心头文件core_cm23.h里。理解它们就算不用库函数你也能直接操作寄存器来玩转SysTick。为了方便大家理解我把这四个关键寄存器整理成了下面这个表格寄存器名称地址偏移主要功能说明CTRL(控制和状态)0x00这是总开关。包含使能位、中断使能位、时钟源选择位以及一个非常重要的标志位——COUNTFLAG当计数器减到0时此位会被硬件自动置1。LOAD(重装载值)0x04你想让计数器从多少开始减就把这个值写进去。它是24位的所以最大值是0xFFFFFF约1677万。VAL(当前值)0x08这个寄存器显示计数器当前减到了多少。写任何值到它都会清空计数器清零同时也会清除CTRL寄存器中的COUNTFLAG标志。CALIB(校准值)0x0C这个寄存器提供了由芯片设计商预置的校准值用于在已知的时钟频率下产生固定的10ms中断。对于一般的延时我们不太直接用它。工作原理流程你向LOAD寄存器写入一个初始值比如SystemCoreClock/8000。使能CTRL寄存器计数器开始工作。每经过一个SysTick时钟周期VAL寄存器的值就自动减1。当VAL减到0时会发生两件事CTRL寄存器的COUNTFLAG标志位被硬件置1。VAL寄存器会自动从LOAD寄存器重新装载初始值然后继续递减自动重载。如果开启了中断在COUNTFLAG置1的同时就会触发SysTick中断。3. 化繁为简使用GD32官方库函数当然每次都去操作寄存器太麻烦了。GD32的固件库Firmware Library已经为我们封装好了易用的函数。我们重点掌握两个函数和一个配置。3.1 核心函数介绍systick_clksource_set(uint32_t systick_clksource)作用选择SysTick的时钟源。这是配置的第一步参数SYSTICK_CLKSOURCE_HCLK_DIV8选择外部时钟HCLK/8。这是默认且常用的选择。SYSTICK_CLKSOURCE_HCLK选择内核时钟HCLK。调用时机在初始化SysTick之前调用。SysTick_Config(uint32_t ticks)作用这是一个ARM CMSIS标准函数用于一次性配置SysTick。它会设置重装载值LOAD、清空当前值VAL、设置优先级并开启SysTick中断。参数ticks就是你要写入LOAD寄存器的值。关键来了这个值决定了中断的周期。中断周期计算中断周期 ticks / SysTick时钟频率。例如SysTick时钟选HCLK/8HCLK72MHz则SysTick时钟9MHz。如果想产生1ms中断ticks 9000。因为1ms 0.001s0.001s * 9,000,000 Hz 9000个 ticks。3.2 官方延时函数剖析GD32的官方示例代码里通常会在systick.c文件中提供一个systick_config()函数。这个函数是延时功能的基础。我们来看看它内部可能的样子基于常见实现void systick_config(void) { /* 1. 选择时钟源HCLK的8分频 */ systick_clksource_set(SYSTICK_CLKSOURCE_HCLK_DIV8); /* 2. 计算1ms需要的计数值并配置SysTick中断。 SystemCoreClock 是系统核心时钟频率HCLK比如72,000,000。 SystemCoreClock / 8000 就是用于产生1ms中断的ticks值。 (因为 72M / 8 9M, 9M / 1000 9000 所以 72M / 8000 9000) */ SysTick_Config(SystemCoreClock / 8000); }看到SystemCoreClock / 8000了吗这就是实现1ms延时的魔法数字的来源。它基于一个前提SysTick时钟源是HCLK/8。基于这个1ms的中断库函数提供了两个非常方便的阻塞延时函数void delay_1ms(uint32_t count)延时count毫秒。例如delay_1ms(500)延时半秒。void delay_1us(uint32_t count)延时count微秒。例如delay_1us(100)延时100微秒。重要提示这两个函数是“阻塞式”延时。意思是调用delay_1ms(1000)后单片机在这1秒钟里几乎什么都干不了就在那里空等计数完成。这在简单的裸机程序中没问题但在需要同时处理多个任务的系统中要谨慎使用否则会影响系统实时性。4. 实战演练用SysTick延时实现LED闪烁理论说得再多不如动手做一遍。接下来我们就在GD32E230开发板上用SysTick延时来实现LED的闪烁间隔1秒亮1秒灭1秒。假设你的工程已经正确配置了系统时钟SystemCoreClock 72MHz并且有一个LED连接在某个GPIO引脚上例如 PA1。步骤分解第一步初始化SysTick通常放在main函数开头int main(void) { // ... 其他初始化代码比如系统时钟配置 ... // 配置SysTick为delay_1ms函数提供基础 systick_config(); // 这个函数内部调用了 systick_clksource_set 和 SysTick_Config // ... 后续代码 ... }第二步配置LED对应的GPIO引脚这里以PA1为例配置为推挽输出模式。void led_init(void) { rcu_periph_clock_enable(RCU_GPIOA); // 使能GPIOA时钟 gpio_mode_set(GPIOA, GPIO_MODE_OUTPUT, GPIO_PUPD_NONE, GPIO_PIN_1); // 设置PA1为输出模式无上下拉 gpio_output_options_set(GPIOA, GPIO_OTYPE_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_1); // 推挽输出速度50MHz }第三步编写主循环使用延时控制LEDint main(void) { // 1. 系统时钟等初始化 // 2. 初始化SysTick延时 systick_config(); // 3. 初始化LED GPIO led_init(); while(1) { // 点亮LED (假设低电平点亮具体看你的电路) gpio_bit_reset(GPIOA, GPIO_PIN_1); // 延时1000毫秒即1秒 delay_1ms(1000); // 熄灭LED gpio_bit_set(GPIOA, GPIO_PIN_1); // 再延时1秒 delay_1ms(1000); } }第四步编译、下载、观察将代码编译后下载到GD32E230开发板你就能看到LED以1秒的间隔稳定地闪烁了。5. 常见问题与调试心得延时不准怎么办首先检查系统时钟SystemCoreClock这个全局变量值对不对它是在system_gd32e23x.c文件中根据你的时钟配置设置的。如果系统时钟不是72MHz但systick_config()里还是除以8000那延时肯定不准。你需要根据实际时钟重新计算。确认时钟源确保systick_config()里选择的时钟源HCLK或HCLK/8和你计算时用的是一致的。阻塞延时的弊端再次强调delay_1ms是阻塞的。在延时期间CPU被占用。对于更复杂的应用可以考虑使用SysTick中断在中断服务函数里维护一个非阻塞的计时变量。使用通用定时器用定时器的输出比较或PWM模式来实现精确定时不占用CPU。从STM32移植过来的注意点STM32的HAL库也有HAL_Delay()原理类似。但GD32的时钟树和库函数命名有所不同直接拷贝代码可能会编译不过。重点是理解原理然后根据GD32的库手册调整函数名和参数。好了关于GD32E230的SysTick定时器从寄存器到实战应用咱们就聊到这里。记住SysTick是你嵌入式开发路上一个非常得力的小助手无论是做简单的延时还是为未来的操作系统学习打下基础都值得你好好掌握。动手试试吧遇到问题随时来交流
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2409447.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!