概述
在使用 DHT11 的时候,时序通信需要微秒来操作,STM32CubeMX 自带一个系统时钟,但是实现的是毫秒级别的。因此就自己用通用计时器实现一个。
环境:
- 开发板:STM32F4探索者(正点原子)
1.配置定时器时钟
-
选择时钟源
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-KJD8ucuO-1606756942997)(I:\net\STM32CubeMX\文档\定时器相关\image\选择时钟源_us.png)]](https://i-blog.csdnimg.cn/blog_migrate/c1d3916292539e911ee2d5b5961ae366.png#pic_center)
这里选择的是内部时钟,来自 RCC 的TIMxCLK,在通用定时器框图中我们可以看到如下:
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-r83Q2abK-1606756943000)(I:\net\STM32CubeMX\文档\定时器相关\image\通用定时器部分系统框图_us.png)]](https://i-blog.csdnimg.cn/blog_migrate/9c831313350d7f84387fdc209bfe12d5.png#pic_center)
而我们可以在 STM32F4xx中文参考手册中找到,TIM2 在外设总线1(APB1上),因此其时钟为 84MHz,如下图所示:
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-o9sX1Wbv-1606756943001)(I:\net\STM32CubeMX\文档\定时器相关\image\TIM2时钟来源.png)]](https://i-blog.csdnimg.cn/blog_migrate/f5bf85375f9ccec9c56bb56ce0eb58a3.png#pic_center)
2.计数器时钟频率及计数模式
除了配置定时器的时钟,还需要配置计数器时钟频率,我们要实现微秒延时,因此计数器时钟频率应该是1MHz,
而要实现还需要以下3个参数:
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-VJGZUbDt-1606756943004)(I:\net\STM32CubeMX\文档\定时器相关\image\计数器3个参数配置.png)]](https://i-blog.csdnimg.cn/blog_migrate/e8391eb7ce2207f4167212ab6bf77308.png#pic_center)
根据STM32F4xx中文参考手册中的时钟频率计算,如下图所示:
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-faIwQNJk-1606756943006)(file:///I:/net/STM32CubeMX/%E6%96%87%E6%A1%A3/%E5%AE%9A%E6%97%B6%E5%99%A8%E7%9B%B8%E5%85%B3/image/%E8%AE%A1%E6%95%B0%E5%99%A8%E6%97%B6%E9%92%9F%E9%A2%91%E7%8E%87%E8%AE%A1%E7%AE%97.png?lastModify=1606749963)]](https://i-blog.csdnimg.cn/blog_migrate/2036795e4bc54619cc9afc57a410996c.png#pic_center)
其中fCK_PSC就是通用定时器框图中的CK_PSC, 即值为84MHz,而我们所要的计数器时钟频率1MHz
因此:
PSC[15:0] = (fCK_PSC/CK_CNT) - 1
= (84/1) - 1
= 83
因此预分频系数为 83
-
计数器模式
计数器这里采用向下计数模式,也就是 如设置计数值为 1000,那么每隔一个微秒,就减一,一直减到 0
-
自动重装载值
虽然我们并不使用自动重装载功能,但是,我们还是要对自动重装载寄存器进行赋值且不赋值为0即可,但是我测试时发送,如果为1,延时会出现偏差,因此这里赋值为 2,依据如下:
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-ZYCeSQm3-1606756943008)(I:\net\STM32CubeMX\文档\定时器相关\image\自动重装载值.png)]](https://i-blog.csdnimg.cn/blog_migrate/74dd0682260ffa817fe2a5dc938b39f3.png#pic_center)
3.打开定时器中断
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-bPmxGfdm-1606756943010)(I:\net\STM32CubeMX\文档\定时器相关\image\打开定时器2的中断.png)]](https://i-blog.csdnimg.cn/blog_migrate/9d7be65cf89f7375021d0a3d2b53ecf1.png#pic_center)
4.具体实现代码
volatile bool elapsed = false; //用于判断设置的计数值是否耗尽(向下计数模式),耗尽时,在中断中奖会设置为true
void setState(bool state)
{
elapsed = state;
}
bool getState()
{
return elapsed;
}
void usDelay(uint32_t time)
{
__HAL_TIM_SetCounter(&htim2,time); //设置计数值
setState(false);
HAL_TIM_Base_Start_IT(&htim2); //开启定时器
while(!getState()); //判断计数值是否耗尽
HAL_TIM_Base_Stop_IT(&htim2); //关闭定时器
}
计数值耗尽回掉函数
void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
{
if (htim->Instance == TIM2) {
setState(true);
}
}
5.代码测试
主函数的主循环中:
while (1)
{
/* USER CODE END WHILE */
/* USER CODE BEGIN 3 */
printf("hello usDelay");
for(int i = 0; i < 1000; i++)
{
usDelay(1000);
}
}
在串口调试助手中,可以看到如下,一秒打印一条语句![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-GQ4FK2gD-1606756943014)(I:\net\STM32CubeMX\文档\定时器相关\image\测试图片.png)]](https://i-blog.csdnimg.cn/blog_migrate/9100aa08902d3839dbc31ba9aae24776.png#pic_center)
项目代码已上传,欢迎下载!
后续也将分享如何使用该函数读取 DHT11 温湿度传感器的值,欢迎点赞,关注,赞赏我!
本文介绍了在使用DHT11时,因STM32CubeMX自带系统时钟为毫秒级,故用通用计时器实现微秒延时。详细说明了配置定时器时钟、计数器时钟频率及计数模式、打开定时器中断等步骤,给出具体实现代码和测试结果,还表示后续将分享读取DHT11温湿度传感器值的方法。
5457

被折叠的 条评论
为什么被折叠?



