GPT对话代码库——STM32G431微秒(us)级delay函数

目录

1,问:

1,答:

方法一:使用定时器(Timer)

方法二:使用SysTick定时器

方法三:使用内联汇编

选择合适的方法

2,问:

2,答:

SysTick配置和延时函数

代码详细解释

3,问:

3,答:

具体解释

示例代码说明

4,问:

4,答:

 配置SysTick定时器并使能中断

代码详细解释


提问模型:GPT-4o-Free

提问时间:2024.06.24

1,问:

我现在要使用STM32G431作为主控,实现温湿度检测功能,使用DHT11检测温湿度,现在碰到一个问题,DHT11的驱动需要us级的延时,而hal库自带的是ms级延时,需要自己写us级延时函数,delay函数涉及到单片机本身的系统时钟,而STM32的F1和G4主频不同,我该如何解决这个问题

1,答:

要在STM32G431上实现us级延时,可以通过以下几种方法:

方法一:使用定时器(Timer)

这是最准确和常用的方法。可以使用STM32G431上的定时器来产生精确的us级延时。下面是使用TIM2定时器来实现us级延时的步骤:

  1. 配置定时器时钟

    • 启动TIM2时钟
    • 配置定时器预分频器(Prescaler),使定时器频率达到1MHz(即每计数一次为1us)
  2. 编写延时函数

下面是一个示例代码,用于配置TIM2并实现us级延时:

#include "stm32g4xx_hal.h"// 定时器句柄
TIM_HandleTypeDef htim2;// 初始化TIM2
void TIM2_Init(void) {__HAL_RCC_TIM2_CLK_ENABLE();  // 启动TIM2时钟htim2.Instance = TIM2;htim2.Init.Prescaler = (uint32_t)(SystemCoreClock / 1000000) - 1;  // 预分频器设置htim2.Init.CounterMode = TIM_COUNTERMODE_UP;htim2.Init.Period = 0xFFFFFFFF;  // 最大周期htim2.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1;htim2.Init.AutoReloadPreload = TIM_AUTORELOAD_PRELOAD_DISABLE;HAL_TIM_Base_Init(&htim2);HAL_TIM_Base_Start(&htim2);
}// 微秒延时函数
void delay_us(uint32_t us) {uint32_t start = __HAL_TIM_GET_COUNTER(&htim2);while ((__HAL_TIM_GET_COUNTER(&htim2) - start) < us);
}

方法二:使用SysTick定时器

SysTick定时器是一个内置于Cortex-M内核中的定时器,可以用于产生us级延时。配置和使用如下:

  1. 配置SysTick

    • 设置SysTick定时器的时钟源和重装载值
  2. 编写延时函数

示例如下:

#include "stm32g4xx_hal.h"// 初始化SysTick
void SysTick_Init(void) {// SysTick时钟源设置为主频HAL_SYSTICK_Config(HAL_RCC_GetHCLKFreq() / 1000000);  // 配置SysTick以1us为单位
}// 微秒延时函数
void delay_us(uint32_t us) {uint32_t start = SysTick->VAL;uint32_t ticks = us * (HAL_RCC_GetHCLKFreq() / 1000000);while ((start - SysTick->VAL) < ticks) {// 等待计数完成}
}

方法三:使用内联汇编

如果想要更加直接和简单的方法,可以使用内联汇编来实现一个简单的us级延时函数:

void delay_us(uint32_t us) {for (uint32_t i = 0; i < us * (SystemCoreClock / 1000000) / 5; i++) {__asm("NOP");  // 空操作指令,消耗一个时钟周期}
}

选择合适的方法

方法一最为精确,适合对延时精度要求较高的场景。方法二较为简洁,但依赖于系统时钟的准确性。方法三是最简单的实现,但精度稍逊。

在实际应用中,推荐使用方法一,通过定时器来实现us级延时,因为这样可以确保延时的准确性和稳定性。配置完定时器后,使用简单的延时函数即可满足DHT11的时序要求。

2,问:

现在要使用STM32G431作为主控,实现温湿度检测功能,使用DHT11检测温湿度,现在已经将STM32G431的主频配置成80MHZ,通过SysTick定时器产生us级延时,按我的要求编写代码,并在代码段中每句代码后面附上详细解释

2,答:

好的,下面是一个基于SysTick定时器的us级延时函数的代码示例,以及每句代码的详细解释。

SysTick配置和延时函数

#include "stm32g4xx_hal.h"  // 包含STM32 HAL库的头文件// 初始化SysTick定时器
void SysTick_Init(void) {// 配置SysTick定时器,使其产生1us的时间间隔HAL_SYSTICK_Config(HAL_RCC_GetHCLKFreq() / 1000000);  // HAL_RCC_GetHCLKFreq()获取HCLK时钟频率,除以1000000得到每微秒的时钟计数// 设置SysTick定时器的时钟源为HCLKHAL_SYSTICK_CLKSourceConfig(SYSTICK_CLKSOURCE_HCLK);  // 配置SysTick的时钟源为HCLK,以确保时钟精度
}// 微秒延时函数
void delay_us(uint32_t us) {uint32_t start = SysTick->VAL;  // 获取当前SysTick计数器的值uint32_t ticks = us * (HAL_RCC_GetHCLKFreq() / 1000000);  // 计算需要等待的时钟周期数,us为微秒数,乘以每微秒的时钟计数while ((start - SysTick->VAL) < ticks) {  // 等待计数完成// 循环等待直到计数器的值减去开始的值达到需要的时钟周期数}
}int main(void) {HAL_Init();  // 初始化HAL库SystemClock_Config();  // 配置系统时钟,确保主频为80MHzSysTick_Init();  // 初始化SysTick定时器while (1) {delay_us(1000);  // 延时1000us,即1ms// 可以在这里添加DHT11数据读取的代码}
}

代码详细解释

  1. #include "stm32g4xx_hal.h":

    • 包含STM32 HAL库的头文件,提供HAL库函数的声明。
  2. void SysTick_Init(void):

    • 定义SysTick初始化函数。
  3. HAL_SYSTICK_Config(HAL_RCC_GetHCLKFreq() / 1000000):

    • 配置SysTick定时器,使其每1us产生一次中断。
    • HAL_RCC_GetHCLKFreq() 获取当前HCLK的频率,除以1000000后得到1us所需要的时钟周期数。
  4. HAL_SYSTICK_CLKSourceConfig(SYSTICK_CLKSOURCE_HCLK):

    • 配置SysTick定时器的时钟源为HCLK,以确保计时的准确性。
  5. void delay_us(uint32_t us):

    • 定义微秒延时函数。
  6. uint32_t start = SysTick->VAL:

    • 获取当前SysTick计数器的值,作为起始计数值。
  7. uint32_t ticks = us * (HAL_RCC_GetHCLKFreq() / 1000000):

    • 计算需要延时的总时钟周期数。
    • us 是需要延时的微秒数,乘以每微秒的时钟计数(HAL_RCC_GetHCLKFreq() / 1000000)得到总的时钟周期数。
  8. while ((start - SysTick->VAL) < ticks):

    • 使用一个while循环来等待,直到计数器的值减去起始值达到所需的时钟周期数。
  9. HAL_Init():

    • 初始化HAL库,配置时钟、外设和GPIO等。
  10. SystemClock_Config():

    • 配置系统时钟,确保STM32G431的主频为80MHz。这个函数的具体实现取决于您的时钟配置。
  11. SysTick_Init():

    • 调用SysTick初始化函数,配置SysTick定时器。
  12. delay_us(1000):

    • 调用微秒延时函数,延时1000us,即1ms。

您可以在while(1)循环中添加DHT11的数据读取代码,以实现温湿度检测功能。这样,SysTick定时器就能提供精确的us级延时,满足DHT11的时序要求。

3,问:

为什么用uint32_t start = SysTick->VAL;  这句就可以 获取当前SysTick计数器的值

3,答:

在STM32系列微控制器中,SysTick定时器是Cortex-M内核的一部分。SysTick->VAL寄存器是SysTick定时器的当前计数值寄存器,它保存了当前SysTick定时器的计数值。当我们读取这个寄存器时,我们可以获取当前SysTick计数器的值,这是实现精确延时的关键步骤。

具体解释

SysTick定时器是一个24位的递减计数器。它的工作原理如下:

  1. SysTick定时器的计数

    • SysTick定时器从一个预设值开始递减计数,直到计数器值为0,然后重新加载预设值并继续递减。
    • SysTick->VAL寄存器保存了当前的计数值。通过读取这个寄存器,我们可以知道SysTick定时器当前计数到了哪个值。
  2. 实现延时

    • 在调用延时函数时,我们首先获取当前的计数器值(即SysTick->VAL)。
    • 然后计算需要等待的时钟周期数。
    • 在一个循环中不断读取SysTick->VAL,直到计数器的值减去开始的值达到所需的时钟周期数。

这样,SysTick->VAL寄存器的读取操作能够提供当前SysTick计数器的值,使我们能够通过检查这个值来实现精确的延时。

示例代码说明

void delay_us(uint32_t us) {uint32_t start = SysTick->VAL;  // 获取当前SysTick计数器的值uint32_t ticks = us * (HAL_RCC_GetHCLKFreq() / 1000000);  // 计算需要等待的时钟周期数,us为微秒数,乘以每微秒的时钟计数while ((start - SysTick->VAL) < ticks) {  // 等待计数完成// 循环等待直到计数器的值减去开始的值达到需要的时钟周期数}
}

在这个代码片段中:

  1. uint32_t start = SysTick->VAL;:

    • 通过读取SysTick->VAL,我们获取当前的SysTick计数器值并将其存储在变量start中。
  2. uint32_t ticks = us * (HAL_RCC_GetHCLKFreq() / 1000000);:

    • 计算需要等待的时钟周期数。这里us是我们希望延时的微秒数,乘以每微秒的时钟计数得到需要等待的总时钟周期数。
  3. while ((start - SysTick->VAL) < ticks) {}:

    • 使用一个循环来等待,直到计数器的值减去起始值达到所需的时钟周期数。
    • 这个等待过程通过不断读取SysTick->VAL来实现。

由于SysTick定时器是直接由硬件支持的,它的计数器值更新非常迅速且准确,因此可以通过这种方法实现精确的us级延时。

4,问:

配个systick定时器中断

4,答:

要使用SysTick定时器中断来实现us级延时并处理中断,需要配置SysTick定时器的中断,并编写相应的中断服务函数(ISR)。以下是详细的步骤和代码示例:

 配置SysTick定时器并使能中断

#include "stm32g4xx_hal.h"  // 包含STM32 HAL库的头文件void SysTick_Init(void) {// 配置SysTick定时器,使其产生1us的时间间隔HAL_SYSTICK_Config(HAL_RCC_GetHCLKFreq() / 1000000);  // 配置SysTick以1us为单位// 设置SysTick定时器的时钟源为HCLKHAL_SYSTICK_CLKSourceConfig(SYSTICK_CLKSOURCE_HCLK);  // 配置SysTick的时钟源为HCLK// 配置SysTick中断优先级HAL_NVIC_SetPriority(SysTick_IRQn, 0, 0);  // 设置SysTick中断的优先级// 使能SysTick中断HAL_NVIC_EnableIRQ(SysTick_IRQn);  // 使能SysTick中断请求
}// SysTick中断服务函数
void SysTick_Handler(void) {HAL_IncTick();  // 调用HAL库的时间基准递增函数HAL_SYSTICK_IRQHandler();  // 调用HAL库的SysTick中断处理函数
}volatile uint32_t usTicks;  // 用于计数us的全局变量// 微秒延时函数
void delay_us(uint32_t us) {usTicks = us;  // 设置要延时的us数while(usTicks);  // 等待usTicks递减到0
}// SysTick中断处理函数,每次中断发生时调用
void HAL_SYSTICK_Callback(void) {if (usTicks) {  // 如果usTicks不为0usTicks--;  // 递减usTicks}
}int main(void) {HAL_Init();  // 初始化HAL库SystemClock_Config();  // 配置系统时钟,确保主频为80MHzSysTick_Init();  // 初始化SysTick定时器while (1) {delay_us(1000);  // 延时1000us,即1ms// 可以在这里添加DHT11数据读取的代码}
}

代码详细解释

SysTick 初始化

void SysTick_Init(void) {HAL_SYSTICK_Config(HAL_RCC_GetHCLKFreq() / 1000000);  // 配置SysTick以1us为单位HAL_SYSTICK_CLKSourceConfig(SYSTICK_CLKSOURCE_HCLK);  // 配置SysTick的时钟源为HCLKHAL_NVIC_SetPriority(SysTick_IRQn, 0, 0);  // 设置SysTick中断的优先级HAL_NVIC_EnableIRQ(SysTick_IRQn);  // 使能SysTick中断请求
}
  • HAL_SYSTICK_Config 配置SysTick定时器,以便每1us产生一次中断。
  • HAL_SYSTICK_CLKSourceConfig 设置SysTick的时钟源为HCLK。
  • HAL_NVIC_SetPriority 设置SysTick中断的优先级。
  • HAL_NVIC_EnableIRQ 使能SysTick中断请求。

SysTick中断服务函数

void SysTick_Handler(void) {HAL_IncTick();  // 调用HAL库的时间基准递增函数HAL_SYSTICK_IRQHandler();  // 调用HAL库的SysTick中断处理函数
}
  • SysTick_Handler 是SysTick的中断服务函数,每当SysTick中断发生时被调用。
  • HAL_IncTick 和 HAL_SYSTICK_IRQHandler 是HAL库中用于处理SysTick中断的函数。

微秒延时函数

volatile uint32_t usTicks;  // 用于计数us的全局变量void delay_us(uint32_t us) {usTicks = us;  // 设置要延时的us数while(usTicks);  // 等待usTicks递减到0
}
  • usTicks 是一个全局变量,用于在中断中递减。
  • delay_us 函数设置延时的us数,并等待usTicks递减到0。

SysTick中断回调函数

void HAL_SYSTICK_Callback(void) {if (usTicks) {  // 如果usTicks不为0usTicks--;  // 递减usTicks}
}
  • HAL_SYSTICK_Callback 是HAL库中提供的SysTick中断回调函数,每当SysTick中断发生时被调用。
  • 在这个函数中递减usTicks,以实现精确的微秒延时。

主函数

int main(void) {HAL_Init();  // 初始化HAL库SystemClock_Config();  // 配置系统时钟,确保主频为80MHzSysTick_Init();  // 初始化SysTick定时器while (1) {delay_us(1000);  // 延时1000us,即1ms// 可以在这里添加DHT11数据读取的代码}
}
  • 初始化HAL库和系统时钟,确保主频为80MHz。
  • 初始化SysTick定时器并使能中断。
  • 在主循环中调用delay_us函数进行延时。

通过上述配置和代码,您可以使用SysTick定时器产生精确的us级延时,同时利用中断机制处理延时。这样可以确保延时的精度和可靠性。

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

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

相关文章

如何集成CppCheck到visual studio中

1.CPPCheck安装 在Cppcheck官方网站下载最新版本1.70&#xff0c;官网链接&#xff1a;http://cppcheck.sourceforge.net/ 安装Cppcheck 2.集成步骤 打开VS&#xff0c;菜单栏工具->外部工具->添加&#xff0c;按照下图设置&#xff0c;记得勾选“使用输出窗口” 2.…

深入解析 IPython 命名空间与作用域机制

IPython 是一个强大的交互式 Python 解释器&#xff0c;它提供了许多增强的功能来改善用户的编程体验。在 IPython 中&#xff0c;命名空间&#xff08;namespace&#xff09;和作用域&#xff08;scope&#xff09;的概念对于理解变量的生命周期和访问方式至关重要。本文将详细…

word2016中新建页面显示出来的页面没有页眉页脚,只显示正文部分。解决办法

问题描述&#xff1a;word2016中新建页面显示出来的页面没有页眉页脚&#xff0c;只显示正文部分。设置了页边距也不管用。 如图1 图1 解决&#xff1a; 点击“视图”——“多页”——“单页”&#xff0c;即可。如图2操作 图2 结果展示&#xff1a;如图3 图3

【Unity】数据持久化--JSON

1、JSON基础语法 1.1 注释内容 单行注释 // 多行注释 /* 内容 */ //注释内容 /* 多行注释 123 e1 ds */ /* 1.2 符号含义 大括号 {} 对象 中括号 [] 数组 冒号 : 键值对对应关系 逗号 , 数据分割 双引号 "" 键名/字符串 1.3 键值对表示 “ "键…

AI 大模型之美 | 更新完结

AI 大模型&#xff1a;技术的壮丽与美感 在当今的人工智能领域&#xff0c;大模型如同一座座巨大的桥梁&#xff0c;将计算机科学、语言学、认知科学等多个领域连接在一起。它们不仅仅是技术的象征&#xff0c;更是人类智慧与创新的结晶。本文将探讨AI大模型的壮丽与美感&…

基于Java的订餐小程序【附源码】

一、本选题的依据&#xff08;阐述所选课题的研究背景、研究目的和意义、分析国内外研究现状及趋势&#xff09; 研究背景&#xff1a; 随着移动互联网的普及和智能手机的发展&#xff0c;人们的生活方式正在发生深刻的变化。特别是在餐饮行业&#xff0c;传统的堂食模式已不能…

‘pip‘ 不是内部或外部命令,也不是可运行的程序 或批处理文件。

&#x1f4da;博客主页&#xff1a;knighthood2001 ✨公众号&#xff1a;认知up吧 &#xff08;目前正在带领大家一起提升认知&#xff0c;感兴趣可以来围观一下&#xff09; &#x1f383;知识星球&#xff1a;【认知up吧|成长|副业】介绍 ❤️如遇文章付费&#xff0c;可先看…

【深度学习】快速入门KerasNLP:微调BERT模型完成电影评论情感分类任务

简介&#xff1a;本文将介绍 KerasNLP 的安装及使用&#xff0c;以及如何使用它在情感分析任务中微调 BERT 的预训练模型。 1. KerasNLP库 KerasNLP 是一个自然语言处理库&#xff0c;兼容 TensorFlow、JAX 和 PyTorch 等多种深度学习框架。基于 Keras 3 构建&#xff0c;这些…

核密度估计kde的本质

核密度估计的本质就是插值&#xff0c;不是拟合&#xff0c;只是不要求必须过已知点。 核为box窗函数 核为高斯函数

python利用cartopy绘制带有经纬度的地图

参考&#xff1a; https://makersportal.com/blog/2020/4/24/geographic-visualizations-in-python-with-cartopy https://scitools.org.uk/cartopy/docs/latest/ https://stackoverflow.com/questions/69465435/cartopy-show-tick-marks-of-axes 具体实现方式&#xff1a; …

201.回溯算法:全排列(力扣)

class Solution { public:vector<int> res; // 用于存储当前排列组合vector<vector<int>> result; // 用于存储所有的排列组合void backtracing(vector<int>& nums, vector<bool>& used) {// 如果当前排列组合的长度等于 nums 的长度&am…

【Android】软键盘空白问题

问题描述 A界面弹出软键盘&#xff0c;跳到B界面&#xff0c;然后返回A界面时软键盘出现空白 解决方案&#xff1a; A界面的onResume方法、跳B界面方法调用前&#xff0c;加一个清除输入框焦点的方法 if (editText!null){editText.clearFocus();}清单文件里A界面添加属性&…

Mybatis 到 MyBatisPlus

Mybatis 到 MyBatisPlus Mybatis MyBatis&#xff08;官网&#xff1a;https://mybatis.org/mybatis-3/zh/index.html &#xff09;是一款优秀的 持久层 &#xff08;ORM&#xff09;框架&#xff0c;用于简化JDBC的开发。是 Apache的一个开源项目iBatis&#xff0c;2010年这…

[亲测可用] 一行代码分页---springBoot PageHelper 不生效解决方案!!

今天做mybatis查询的时候 不管是用框架查询sql还是手动写sql&#xff0c;查询分页都不生效&#xff0c;很简单&#xff0c;你配置不对&#xff0c;或者缺少了配置。我下面是直接配置 不用写代码配置。框架查询sql还是手动写sql都支持 这是我查询的sql (注意&#xff01;&#…

【图像处理实战】去除光照不均(Python)

这篇文章主要是对参考文章里面实现一种小拓展&#xff1a; 可处理彩色图片&#xff08;通过对 HSV 的 V 通道进行处理&#xff09;本来想将嵌套循环改成矩阵运算的&#xff0c;但是太麻烦了&#xff0c;而且代码也不好理解&#xff0c;所以放弃了。 代码 import cv2 import …

虚拟化 之八 详解构造带有 jailhouse 的 openEuler 发行版(ARM 飞腾派)

基本环境 嵌入式平台下,由于资源的限制,通常不具备通用性的 Linux 发行版,各大主流厂商都会提供自己的 Linux 发行版。这个发行版通常是基于某个 Linux 发行版构建系统来构建的,而不是全部手动构建,目前主流的 Linux 发行版构建系统是 Linux 基金会开发的 Yocto 构建系统。…

【大数据技术原理与应用(概念、存储、处理、分析与应用)】第3章-分布式文件系统HDFS习题与知识回顾

文章目录 单选题多选题知识回顾什么是HDFS?分布式文件系统结构HDFS的设计目标与局限性块的概念名称节点(NameNode)数据节点(DataNode)第二名称节点HDFS体系结构的局限性HDFS存储原理冗余存储数据存取策略数据读取策略HDFS数据读写过程读数据的过程写数据的过程单选题 1、分…

用一个暑假|用AlGC-stable diffusion 辅助服装设计及展示,让你在同龄人中脱颖而出!

大家好&#xff0c;我是设计师阿威 Stable Diffusion是一款开源AI绘画工具&#xff0c; 用户输入语言指令&#xff0c;即可自动生成各种风格的绘画图片 Stable Diffusion功能强大&#xff0c;生态完整、使用方便。支持大部分视觉模型上传&#xff0c;且可自己定制模型&#x…

C++ boost::property_tree::read_json读取虚拟墙文件

输入是一个json文件如下 VirtualWall.json {"LAYERSUM":1,"LAYER":[{"LAYERNUM":0,"LINESUM":23,"LINE":[{"BEGIN":{"X":-3028.6428,"Y":-10671.736},"END":{"X":-3167…

什么是大模型?一文读懂大模型的基本概念

大模型是指具有大规模参数和复杂计算结构的机器学习模型。本文从大模型的基本概念出发&#xff0c;对大模型领域容易混淆的相关概念进行区分&#xff0c;并就大模型的发展历程、特点和分类、泛化与微调进行了详细解读&#xff0c;供大家在了解大模型基本知识的过程中起到一定参…