关于STM32单片机延时微妙(delay_us)函数-hal库

news2025/1/18 9:44:38

关于STM32单片机延时微妙(delay_us)函数-hal库

  • Chapter1 关于STM32单片机延时微妙(delay_us)函数-hal库
    • delay_us 函数
    • delay_ms 函数
    • 验证
    • 补充


Chapter1 关于STM32单片机延时微妙(delay_us)函数-hal库

原文链接:https://blog.csdn.net/qq_22146161/article/details/106527001

叙述
近段时间调试一个STM32带多个DS18B20传感器,发现这个传感器对时序要求特别高,而使用的固件是HAL库版的,里面没有专门微米级别的延时函数,通不过不断尝试,调通了延时函数,使DS18B20成功跑起来了。写此篇文章来记录本次调试。

实现延时三种代码
现在能在网上查到STM32延时方式基本就是三种,循环延时,滴答定时器延时 和 滴答定时器中断延时。
对于这三种在这里就不再一一赘述了,可以看看这篇文章参考;
参考链接:https://blog.csdn.net/weibo1230123/article/details/81136564:
这里主要是说说第三种,在HAL库调试的过程以及情况。

滴答定时器延时方式
在HAL库下添加延时函数,参考正点原子HAL库,但是自己使用的是单片机是L073,共分为3步。

第一步
先确认自己时钟频率,说白了,就是要知道自己单片机主频,好计算出1us。
自己外部晶振为8M,查单片机对映的是时钟框图如下,可以看出时钟从外部HSE_OSC来的为8M,经过以一个选择器后,进入PLL锁相环,之后去了AHB PRESC,最后经过一个/8的分频后就是滴答定时器的频率。好的,以上是分析时钟框图看出来的东西,整个路径就是这样的一个情况。

但是,HAL 库的延时函数有一个局限性,在中断服务函数中使用 HAL_Delay 会引起混乱
(虽然一般禁止在中断中使用延时函数),因为它是通过中断方式实现,而 Systick 的中断优先级
是最低的,所以在中断中运行 HAL_Delay 会导致延时出现严重误差。所以一般情况下,推荐大
家使用正点原子提供的延时函数库。
HAL 库的 ms 级别的延时函数__weak void HAL_Delay(uint32_t Delay);它是弱定义函数,
所以用户可以自己重新定义该函数。例如:我们在 deley.c 文件可以这样重新定义该函数:

/**
 * @brief HAL 库延时函数重定义
 * @param Delay 要延时的毫秒数
 * @retval None
 */
void HAL_Delay(uint32_t Delay)
{
 delay_ms(Delay);
}

在这里插入图片描述
SYSTEM 文件夹里面的代码由正点原子提供,是 STM32F4xx 系列的底层核心驱动函数,
可以用在 STM32F4xx 系列的各个型号上面,方便大家快速构建自己的工程。本章,我们将向大
家介绍这些代码的由来及其功能,也希望大家可以灵活使用 SYSTEM 文件夹提供的函数,来快
速构建工程,并实际应用到自己的项目中去。
SYSTEM 文件夹下包含了 delay、sys、usart 等三个文件夹。分别包含了 delay.c、sys.c、usart.c
及其头文件。这 3 个 c 文件提供了系统时钟设置、延时和串口 1 调试功能,任何一款 STM32F4
都具备这几个基本外设,所以可以快速地将这些设置应用到任意一款 STM32F4 产品上,通过
这些驱动文件实现快速移植和辅助开发的效果。

本章将分为如下几个小节:
deley 文件夹代码介绍
sys 文件夹代码介绍
usart 文件夹代码介绍

deley 文件夹代码介绍
delay 文件夹内包含了 delay.c 和 delay.h 两个文件,这两个文件用来实现系统的延时功能,
其中包含 7 个函数:

void delay_osschedlock(void);
void delay_osschedunlock(void);
void delay_ostimedly(uint32_t ticks);
void SysTick_Handler(void);
void delay_init(uint16_t sysclk);
void delay_us(uint32_t nus);
void delay_ms(uint16_t nms);

前面 4 个函数,仅在支持操作系统(OS)的时候,需要用到,而后面 3 个函数,则不论是
否支持 OS 都需要用到。

delay_us 函数

该函数用来延时指定的 us,其参数 nus 为要延时的微秒数。具体的函数如下:

/**
* @brief 延时 nus
* * @note 无论是否使用 OS, 都是用时钟摘取法来做 us 延时
* @param nus: 要延时的 us 数
* @note nus 取值范围: 0 ~ (2^32 / fac_us) (fac_us 一般等于系统主频, 自行套入计算)
* @retval 无
*/
void delay_us(uint32_t nus)
{
 uint32_t ticks;
 uint32_t told, tnow, tcnt = 0;
 uint32_t reload = SysTick->LOAD; /* LOAD 的值 */
 ticks = nus * g_fac_us; /* 需要的节拍数 */
 
#if SYS_SUPPORT_OS /* 如果需要支持 OS */
 delay_osschedlock(); /* 锁定 OS 的任务调度器 */
#endif
 told = SysTick->VAL; /* 刚进入时的计数器值 */
 while (1)
 {
 tnow = SysTick->VAL;
 if (tnow != told)
 {
 if (tnow < told)
 {
 tcnt += told - tnow; 
/* 这里注意一下 SYSTICK 是一个递减的计数器就可以了 */
 }
 else
 {
 tcnt += reload - tnow + told;
 }
 told = tnow;
 if (tcnt >= ticks)
 {
 break; /* 时间超过/等于要延迟的时间,则退出 */
 }
 }
 }
#if SYS_SUPPORT_OS /* 如果需要支持 OS */
 delay_osschedunlock(); /* 恢复 OS 的任务调度器 */
#endif 
}

这里就正是利用了我们前面提到的时钟摘取法,ticks 是延时 nus 需要等待的 SysTick 计数
次数(也就是延时时间),told 用于记录最近一次的 SysTick->VAL 值,然后 tnow 则是当前的
SysTick->VAL 值,通过他们的对比累加,实现 SysTick 计数次数的统计,统计值存放在 tcnt 里
面,然后通过对比 tcnt 和 ticks,来判断延时是否到达,从而达到不修改 SysTick 实现 nus 的延
时,从而可以和 OS 共用一个 SysTick。
上面的 delay_osschedlock 和 delay_osschedunlock 是 OS 提供的两个函数,用于调度上锁和
解锁,这里为了防止 OS 在 delay_us 的时候打断延时,可能导致的延时不准,所以我们利用这
两个函数来实现免打断,从而保证延时精度!在 168MHz 主频下,此时的 delay_us 函数,可以
实现最长 2^32/g_fac_us 的延时,大概是 25.56 秒。

delay_ms 函数

该函数是用来延时指定的 ms 的,其参数 nms 为要延时的毫秒数。具体的函数如下:

/**
* @brief 延时 nms
* @param nms: 要延时的 ms 数 (0< nms <= (2^32 / fac_us / 1000))
(fac_us 一般等于系统主频, 自行套入计算)
* @retval 无
*/
void delay_ms(uint16_t nms)
{
 
#if SYS_SUPPORT_OS /* 如果需要支持 OS, 则根据情况调用 os 延时以释放 CPU */
if (delay_osrunning && delay_osintnesting == 0) 
/* 如果 OS 已经在跑了,并且不是在中断里面(中断里面不能任务调度) */
 {
 if (nms >= g_fac_ms) /* 延时的时间大于 OS 的最少时间周期 */
 {
 delay_ostimedly(nms / g_fac_ms); /* OS 延时 */
 }
 nms %= g_fac_ms; 
/* OS 已经无法提供这么小的延时了,采用普通方式延时 */
 }
#endif
 delay_us((uint32_t)(nms * 1000)); /* 普通方式延时 */
}

该函数中,delay_osrunning 是 OS 正在运行的标志,delay_osintnesting 则是 OS 中断嵌套次
数,必须 delay_osrunning 为真,且 delay_osintnesting 为 0 的时候,才可以调用 OS 自带的延时
函数进行延时(可以进行任务调度),delay_ostimedly 函数就是利用 OS 自带的延时函数,实现
任 务 级延 时 的, 其 参数 代 表延 时 的时 钟 节拍 数 (假 设 delay_ostickspersec=200 ,那么
delay_ostimedly(1),就代表延时 5ms)。
当 OS 还未运行的时候,我们的 delay_ms 就是直接由 delay_us 实现的,OS 下的 delay_us
可以实现很长的延时(达到 65 秒)而不溢出!,所以放心的使用 delay_us 来实现 delay_ms,不
过由于 delay_us 的时候,任务调度被上锁了,所以还是建议不要用 delay_us 来延时很长的时间,
否则影响整个系统的性能。
当 OS 运行的时候,我们的 delay_ms 函数将先判断延时时长是否大于等于 1 个 OS 时钟节
拍(g_fac_ms),当大于这个值的时候,我们就通过调用 OS 的延时函数来实现(此时任务可以
调度),不足 1 个时钟节拍的时候,直接调用 delay_us 函数实现(此时任务无法调度)。

验证

写完之后应该验证一下,否则自己写的对错都不知道,验证代码的话,就驱动一个IO一秒改变一下,如果下图,这个应该不难。
在这里插入图片描述
上示波器验证看看IO是不是1秒翻转一下。如下图
在这里插入图片描述
可以看到图中a 和 b之间 差不多为972.0ms,差不多是1秒,验证自己的代码。

补充

开始比较真就想要个1us的延时,查了一些资料,在这种频率为几兆到几百兆的单片机,很难做到这么精确的延时。
就是说一个100M以内单片机很难做到1us的精准延时。
以上是关于这次的延时的一个小总结。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1577292.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

窗口函数的排名

在排序时考虑到每行的前后关系&#xff0c;并根据这些关系进行排序。这使得在排序结果集时可以得到更精确、准确的排序。 row_number()例如90、 80、80、70排名后为1、2、3、4rank()例如90、 80、80、70排名后为1、2、2、4dense_rank()例如90、 80、80、70排名后为1、2、2、3 …

大模型 概念科普|提示词是什么

一、基本概念 5W2H方法是一种有效的分析和问题解决框架&#xff0c;它包括了“什么&#xff08;What&#xff09;”、“为什么&#xff08;Why&#xff09;”、“谁&#xff08;Who&#xff09;”、“何时&#xff08;When&#xff09;”、“在哪里&#xff08;Where&#xff…

用API技术为数据安全“上保险”——双重保障

&#x1f50d;API在数据安全领域的核心地位 随着数字化进程的狂飙突进&#xff0c;应用程序接口&#xff08;API&#xff09;已化身为企业内部、不同平台间以及用户交互的关键纽带。它们不仅是数据流动与共享的驱动引擎&#xff0c;更是守护数据安全的重要防线。其中&#xf…

高精度原边控制离线式PWM功率开关芯片D3820的特征和详细的工作原理介绍

D3820是一款高精度原边控制离线式PWM功率开关。本文主要介绍D3820的特征和详细的工作原理&#xff0c;对反激式隔离AC-DC开关电源提供较为详细的测试过程。 特 点 1、全电压范围CC/CV精度保持在5%以内 2、用原边控制&#xff0c;无需TL431和光耦 3、欠压锁定&#xff08…

实践笔记-linux内核版本升级(centos7)

linux内核版本升级 1.查看当前内核版本信息2.采用yum方式进行版本升级2.1导入仓库源2.2选择 ML 或 LT 版本安装2.3设置内核启动 3.删除旧版本内核 1.查看当前内核版本信息 #查看操作系统版本 cat /etc/redhat-release #查看系统内核 uname -r2.采用yum方式进行版本升级 2.1导…

Openstack(T)部署ceilometer-gnocchi-panko-aodh

前言&#xff1a;简单概述4个服务做什么&#xff0c;如何协同起来 ceilometer负责采集数据&#xff0c;采集的数据分为计量数据和事件数据&#xff0c;计量数据交给gnocchi处理(本文gnocchi将计量数据存储在ceph存储池中)&#xff0c;事件数据存储在panko中&#xff0c;aodh负…

RH850从0搭建Autosar开发环境【3X】- Davinci Configurator之RTE模块配置详解(上)

RTE模块配置详解 - 上 一、RTE模块配置实操1.1 打开RTE模块1.2 RTE模块错误消除在这里插入图片描述 这里我们利用工具的自动处理功能。二、Configurator工具Validation总结本节我们就手把手详解RTE配置实现,其实也没有什么过多的操作。。。这个模块更多是工具自动处理的。 一、…

C++从入门到精通——类的6个默认成员函数之构造函数

构造函数 前言一、构造函数的概念二、构造函数特性 前言 类的6个默认成员函数&#xff1a;如果一个类中什么成员都没有&#xff0c;简称为空类。 空类中真的什么都没有吗&#xff1f;并不是&#xff0c;任何类在什么都不写时&#xff0c;编译器会自动生成以下6个默认成员函数…

数据库连接查询

一、联合查询 1.使用 union 连接两个 select 语句进行联合查询 select 列 1&#xff0c;列 2... from 表名 where 条件 union select 列 1&#xff0c;列 2... from 表名 where 条 件; select vend_id,prod_id,prod_name,prod_price from products where prod_price < 5…

【Leetcode】拓扑排序—课程表系列

有向无环图DAG图示&#xff1a; 拓扑排序结果&#xff1a;{2,3,5,1,7,4,6} {3,2,1,5,7,6,4} 不唯一 拓扑排序结果满足&#xff1a;对于图中的每条有向边(u,v)&#xff0c;u要排序在v之前&#xff1b; 应用&#xff1a;判断有向图中是否有环&#xff0c;可以生成拓…

FreeRTOS学习 -- 再识

工作中一直使用FreeRTOS进行着开发&#xff0c;但是没有进行过系统的总结过。现在将快速使用几天时间将FreeRTOS相关知识点加以总结。 官网&#xff1a; https://www.freertos.org/zh-cn-cmn-s/ 参看资料&#xff1a; 正点原子 STM32F1 FreeRTOS开发手册_V1.2.pdf The FreeRTOS…

揭秘!兆欧表测量接地电阻的步骤是什么?

兆欧表&#xff0c;又被称为绝缘电阻测试仪或摇表&#xff0c;是一种可携式仪器&#xff0c;用于测量电气设备、电缆、电机绕组和其他导体之间&#xff0c;以及导体与地之间的绝缘电阻。该仪表能够提供较高的直流电压&#xff08;通常为500V、1000V、2500V甚至更高&#xff09;…

第45期 | GPTSecurity周报

GPTSecurity是一个涵盖了前沿学术研究和实践经验分享的社区&#xff0c;集成了生成预训练Transformer&#xff08;GPT&#xff09;、人工智能生成内容&#xff08;AIGC&#xff09;以及大语言模型&#xff08;LLM&#xff09;等安全领域应用的知识。在这里&#xff0c;您可以找…

LeetCode 使数组连续的最少操作数

地址&#xff1a;. - 力扣&#xff08;LeetCode&#xff09; 难度&#xff1a;困难 题目描述&#xff1a;给你一个整数数组 nums 。每一次操作中&#xff0c;你可以将 nums 中 任意 一个元素替换成 **任意 **整数。 如果 nums 满足以下条件&#xff0c;那么它是 连续的 &#x…

竞赛 协同过滤电影推荐系统

文章目录 1 简介1 设计概要2 课题背景和目的3 协同过滤算法原理3.1 基于用户的协同过滤推荐算法实现原理3.1.1 步骤13.1.2 步骤23.1.3 步骤33.1.4 步骤4 4 系统实现4.1 开发环境4.2 系统功能描述4.3 系统数据流程4.3.1 用户端数据流程4.3.2 管理员端数据流程 4.4 系统功能设计 …

YOLOv8的多分类模型如何计算准确率(Accuracy)、精确率(Precision)、召回率(recall)和F1-Score模型评估参数

《博主简介》 小伙伴们好&#xff0c;我是阿旭。专注于人工智能、AIGC、python、计算机视觉相关分享研究。 ✌更多学习资源&#xff0c;可关注公-仲-hao:【阿旭算法与机器学习】&#xff0c;共同学习交流~ &#x1f44d;感谢小伙伴们点赞、关注&#xff01; 《------往期经典推…

算法刷题Day27 | 39. 组合总和、40.组合总和II、131.分割回文串

目录 0 引言1 组合总和1.1 我的解题 2 组合总和II2.1 解题 3 分割回文串3.1 切割3.2 总结&#xff1a;分割和组合的区别 &#x1f64b;‍♂️ 作者&#xff1a;海码007&#x1f4dc; 专栏&#xff1a;算法专栏&#x1f4a5; 标题&#xff1a;算法刷题Day27 | 39. 组合总和、40.…

【Entity Framework】EF配置文件设置详解

【Entity Framework】EF配置文件设置详解 文章目录 【Entity Framework】EF配置文件设置详解一、概述二、实体框架配置部分三、连接字符串四、EF数据库提供程序五、EF侦听器六、将数据库操作记录到文件中七、Code First默认连接工厂八、数据库初始值设定项 一、概述 EF实体框架…

Taro打包生成不同目录

使用taro init创建taro项目时&#xff0c;taro默认打包目录是&#xff1a; /config/index.js outputRoot:dist默认的目录&#xff0c;编译不同平台代码时就会覆盖掉&#xff0c;为了达到多端同步调试的目的&#xff0c;这时需要修改默认生成目录了&#xff0c;通过查看官方文…