【杂谈】针对Cortex M4内核使用Systick进行延时

目录

一、简介

1.寄存器组成(4个关键寄存器)

2.时钟源配置

3.小结

二、示例程序


在ST的数据手册中无法找到与内核相关的详细内容,因为M4内核使用的是ARMv7-M指令集,通过ARM官网可以找到并下载内核架构的数据手册

一、简介

SysTick是ARM Cortex-M4内核内置的系统定时器,不属于外设定时器 ,而是内核的一部分。它是一个24位递减计数器,主要用于:

  • 提供操作系统的时钟节拍(如FreeRTOS的时间片调度)
  • 实现精确的延时函数
  • 作为简单的时间基准

1.寄存器组成(4个关键寄存器)

  1. CTRL - 控制和状态寄存器
  2. LOAD - 重装载值寄存器
  3. VAL - 当前值寄存器
  4. CALIB - 校准值寄存器

2.时钟源配置

在STM32F4中,SysTick的时钟源有两种选择:

复制代码
// 查看 SystemCoreClock 的值(通常为168MHz)
SystemCoreClock = 168000000;  // HCLK频率

// 选择时钟源
SysTick->CTRL |= SysTick_CTRL_CLKSOURCE_Msk;    // 使用HCLK
// 或
SysTick->CTRL &= ~SysTick_CTRL_CLKSOURCE_Msk;   // 使用HCLK/8

3.小结

特性 说明
位数 24位递减计数器
时钟源 AHB时钟 或 AHB/8(可配置)
中断能力 计数器到0时产生SysTick中断
自动重载 支持自动重载功能
精度 高精度,基于系统时钟(最大只能计时1ms)

二、示例程序

复制代码
#include "stm32f4xx.h"

// 定义微秒和毫秒延时的基础计数值
// 注意:这些值需要根据你的实际SysTick时钟频率进行计算和调整
static uint32_t fac_us = 168 - 1;    // 假设SysTick时钟为168MHz时,1us的计数值
static uint32_t fac_ms = 168000 - 1; // 1ms的计数值,注意防止溢出


void Delay_Init(void) 
{
  // SysTick时钟源通常默认是HCLK,这里不更改配置,如果需要配置为HCLK/8,可以取消下一行的注释
  // SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8);
}


void Delay_us(uint32_t nus) 
{
    uint32_t temp;
    SysTick->LOAD = nus * fac_us;              // 设置重载值
    SysTick->VAL = 0x00;                       // 计数前先清空当前计数器
    SysTick->CTRL |= SysTick_CTRL_ENABLE_Msk;  // 使能SysTick,使用内核时钟,不开启中断 
                                               // SysTick_CTRL_ENABLE_Msk == 0x01

    do 
    {
        temp = SysTick->CTRL;
    } while ((temp & 0x01) && !(temp & (1 << 16))); // 等待时间到达,并检查计数器是否仍使能

    SysTick->CTRL = 0x00; // 关闭计数器
    SysTick->VAL = 0x00;  // 清空计数器
}


void Delay_ms(uint32_t nms) 
{
    uint32_t temp;
    // 注意:nms * fac_ms 的值可能超过24位,实际使用中需确保重载值不超过0xFFFFFF
    SysTick->LOAD = nms * fac_ms;              // 设置重载值
    SysTick->VAL = 0x00;                       // 清空当前计数器
    SysTick->CTRL |= SysTick_CTRL_ENABLE_Msk;  // 使能SysTick,使用内核时钟,不开启中断
                                               // SysTick_CTRL_ENABLE_Msk == 0x01

    do 
    {
        temp = SysTick->CTRL;
    } while ((temp & 0x01) && !(temp & (1 << 16))); // 等待时间到达

    SysTick->CTRL = 0x00; // 关闭计数器
    SysTick->VAL = 0x00;  // 清空计数器
}

其中

temp = Systick->CTRL,用temp检测Systick->CTRL寄存器

(temp & 0x01) 表示定时器仍在运行

!(temp & (1 << 16)) 表示COUNTFLAG位为0(时间未到)

相关推荐
测试专家2 小时前
USB 3.0,USB速率
单片机·嵌入式硬件
昵称只能一个月修改一次。。。2 小时前
嵌入式硬件编程
单片机·嵌入式硬件
FPGA-ADDA2 小时前
第四篇:嵌入式系统常用通信接口详解(I2C、SPI、UART、RS232/485、CAN、USB)
人工智能·单片机·嵌入式硬件·fpga开发·信息与通信
我不是程序猿儿3 小时前
【嵌入式】编码器分类
单片机·嵌入式硬件
给点sun,就shine3 小时前
电阻的作用
单片机·嵌入式硬件
CODE_RabbitV3 小时前
【3min 解决】keil5 编译stm32 出现一堆 core_cm3.c 报错问题
c语言·stm32·嵌入式硬件
FreakStudio3 小时前
MicroPython+PycoClaw,3 分钟搞定 ESP32 跑上 OpenClaw!
python·单片机·嵌入式·电子diy
【 STM32开发 】3 小时前
【STM32 + CubeMX 教程】RTC 实时时钟 之 闹钟唤醒 -- F407篇
stm32·嵌入式硬件·实时音视频
BT-BOX3 小时前
第6章《Stm32CubeMX+Proteus仿真入门》LCD1602显示
stm32·嵌入式硬件·proteus
’长谷深风‘4 小时前
51单片机入门(2)
单片机·嵌入式硬件·51单片机·定时器·中断