STM32标准库+HAL库 | 输入捕获测量PWM的脉冲频率+占空比

提醒:本文的代码Demo中使用的是,单通道捕捉采集PWM输入信号的频率+占空比。

在上一篇博客中已经讲解了过PWM输出配置,本文主要讲解TIM输入捕获配置。STM32标准库+HAL库 | 高精度动态调节PWM输出频率+占空比_hal库改变pwm频率-CSDN博客

目录

1、TIM输入捕获基础

2、HAL库版TIM输入捕获

3、标准库版TIM输入捕获


1、TIM输入捕获基础

        在众多的仪器仪表类产品如示波器、逻辑分析仪,医疗设备、智能手表及工业控制的电机设备开发中,经常需要测量PWM输入波形的频率及占空比或TIM定时器脉冲计数值等数据。因此掌握TIM输入捕获是在各个行业从事嵌入式开发的一项非常基本的技能。当然,在一些大学生的省级、国家级的电子综合设计竞赛中,也经常考察对TIM、PWM的配置使用,由此可见掌握TIM-PWM的重要性。

        随着ST公司推出了越来越多的新款芯片并没有配套的标准库,并且目前在各大平台没有前辈对TIM输入捕获将两个库同时进行讲解。因此本文以STM32为例,对STM32的标准库和HAL库输入捕获功能进行讲解。标准库使用STM32F407进行输入捕获测试,HAL库使用STM32G4进行输入捕获测试。STM32G4官方没有推出对应的标准库。如果是其它系列芯片,其配置思路和本文所说的大致相同。

        输入捕获可以对输入的信号的上升沿,下降沿或者双边沿进行捕获,常用的有测量输入信号的脉宽和测量PWM输入信号的频率和占空比这两种。输入捕获的原理大概就是,当捕获到信号的跳变沿的时候,把计数器 CNT 的值锁存到捕获寄存器 CCR 中,把前后两次捕获到的 CCR 寄存器中的值相减,就可以算出脉宽或者频率。如果捕获的脉宽的时间长度超过捕获定时器的周期,就会发生溢出,就需要做额外的处理。因此配置TIM定时器输入捕获时,尽可能将TIM定时器的周期拉大。

        如下图所示是STM32的官方参考手册,可以得知STM32的TIM定时器有两种模式都可以用于输入捕获,即TIM输入捕获模式PWM输入模式。  其中PWM输入模式是TIM输入捕获的一个特例,配置方法是基本相同的,但是多了几步其它的操作。 且 PWM输入模式会采用两路通道进行采集信号,但是配置时,仅需配置一路主通道,TIM会自动设置另一路通道, 这两路通道是极性相反的。

        输入捕获一般以中断形式启动,在中断中对上升沿和下降沿进行处理,读取TIM定时器的脉冲捕捉通道的计数值,将一个周期的数据记录并通过公式计算处理后,即可得出输入信号的频率、占空比信息。如下所示为一个周期的PWM脉冲,如果需要测量其PWM频率+占空比,则需读取其一个周期内的信号起始上升沿下降沿最后一个上升沿信号

周期:T = TH + TL

频率:F = 1 / T

占空比:D = TH  / (TH + TL)

TIM输入捕获的流程配置思路:

  1. 配置总线时钟
  2. 配置GPIO端口引脚
  3. 配置TIM定时器时基
  4. 配置NVIC中断
  5. 配置IC输入捕获模式

        如果只采集PMW脉冲的频率,那么只需要使用单通道采集上升沿(或下降沿)信号,就可以求得频率值。(两次高电平或两次低电平直接的时间值,就可以计算出PWM的频率)。如果不仅需要采集PWM频率,还要采集PWM的占空比,那么需要同时采集上升沿和下降沿信号。

        采集PWM脉冲频率及占空比数据可以使用单通道采集,也可以使用双通道采集。本文讲解单通道采集PWM频率+占空比,即采集一个周期内的 2次上升沿+1次下降沿或2次下降沿+1次上升沿。

在设置计数器counter period,即自动重装载值拉满时(0xFFFF),修改配置预分频PSC的值可以调整定时器的定时时间t。

2、HAL库版TIM输入捕获

        配置PA7为脉冲捕捉引脚,TIM3_CH2

        在STM32CubeMX中配置步骤如下:

生成了代码工程代码后,在其回调函数中,进行读取PWM输入捕获计数值,以进行频率+占空比计算操作。

HAL库TIM中断回调函数:

volatile float TIM3CH2_Freq = 0.0;
volatile float TIM3CH2_Duty = 0.0;
volatile int capture_end_flag = 0;volatile uint32_t high_val = 0;
volatile uint32_t low_val = 0;//TIM单通道采集PWM频率+占空比
void HAL_TIM_IC_CaptureCallback(TIM_HandleTypeDef *htim)
{    static uint8_t capture_cnt = 1;    //电平捕捉计数if(htim->Instance == TIM3)        //判断是否由定时器3产生{if(htim->Channel == HAL_TIM_ACTIVE_CHANNEL_2)    //TIM3 通道2{    if(capture_end_flag == 0){if(capture_cnt == 1)        //第一个上升沿{capture_cnt = 2;__HAL_TIM_SET_CAPTUREPOLARITY(htim, TIM_CHANNEL_2, TIM_INPUTCHANNELPOLARITY_FALLING); //设置成下降沿触发__HAL_TIM_SetCounter(htim, 0);    //清空定时器计数值high_val = HAL_TIM_ReadCapturedValue(htim, TIM_CHANNEL_2);    //由第一个上升沿设为起始位置}else if(capture_cnt == 2)    //第一个下降沿{capture_cnt = 3;low_val = HAL_TIM_ReadCapturedValue(htim, TIM_CHANNEL_2);    //低电平起始位置__HAL_TIM_SET_CAPTUREPOLARITY(htim, TIM_CHANNEL_2,TIM_INPUTCHANNELPOLARITY_RISING); //设置成上升沿触发}else if(capture_cnt == 3)    //第二个上升沿{capture_cnt = 1;high_val = HAL_TIM_ReadCapturedValue(htim, TIM_CHANNEL_2);//计算频率TIM3CH2_Freq = (float)80000000 / 80 / (high_val+1);//计算占空比TIM3CH2_Duty = (float)low_val / (high_val+1);capture_end_flag = 1;}}}}
}

HAL库主函数:

#include "main.h"
#include "stdio.h"
#include "adc.h"
#include "tim.h"
#include "usart.h"
#include "gpio.h"void SystemClock_Config(void);void main(void)
{HAL_Init();SystemClock_Config();      MX_USART1_UART_Init();MX_TIM3_Init();HAL_TIM_IC_Start_IT(&htim3, TIM_CHANNEL_2);while (1){HAL_Delay(3000); if(capture_end_flag == 1){//计算频率//TIM3CH2_Freq = (float)80000000 / 80 / (high_val+1);//计算占空比//TIM3CH2_Duty = (float)(low_val+1) / (high_val+1);//printf("high_val:%d\r\n",  high_val);//printf("low_val:%d\r\n",  low_val);		printf("捕获PWM频率:%.2f\r\n", TIM3CH2_Freq);printf("捕获PWM占空比:%.2f\r\n", TIM3CH2_Duty);capture_end_flag = 0;}}
}

输出2000Hz,占空比为45%的PWM信号

HAL版的TIM输入捕获到的信号数据

3、标准库版TIM输入捕获

        配置PD15引脚,TIM4_CH4为输入捕获模式

标准库TIM配置及中断服务函数配置:

//TIM输入捕获配置
void Capture_Wave_Init(void)
{GPIO_InitTypeDef     GPIO_InitStructure;NVIC_InitTypeDef     NVIC_InitStructure;TIM_ICInitTypeDef      TIM_ICInitStructure;    TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM4, ENABLE);RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOD, ENABLE);    //PD15 TIM4_CH4GPIO_InitStructure.GPIO_Pin =  GPIO_Pin_15;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_DOWN;GPIO_Init(GPIOD, &GPIO_InitStructure);GPIO_PinAFConfig(GPIOD, GPIO_PinSource15, GPIO_AF_TIM4);TIM_TimeBaseInitStructure.TIM_Period = 65535;TIM_TimeBaseInitStructure.TIM_Prescaler = 84-1;TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;TIM_TimeBaseInitStructure.TIM_RepetitionCounter = 0;TIM_TimeBaseInit(TIM4, &TIM_TimeBaseInitStructure);NVIC_InitStructure.NVIC_IRQChannel = TIM4_IRQn;NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;NVIC_Init(&NVIC_InitStructure);TIM_ICInitStructure.TIM_Channel = TIM_Channel_4;TIM_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Rising;TIM_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI;TIM_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1;TIM_ICInitStructure.TIM_ICFilter = 0xF;TIM_ICInit(TIM4, &TIM_ICInitStructure);TIM_Cmd(TIM4, ENABLE);TIM_ITConfig(TIM4, TIM_IT_CC4, ENABLE);
}uint32_t Cap_Freq = 0;    
uint8_t  Cap_Duty = 0;
uint8_t capture_cnt = 1;
uint8_t capture_end_flag = 0;
volatile uint32_t high_val = 0;
volatile uint32_t low_val = 0;
//TIM输入捕获中断函数
void TIM4_IRQHandler(void)
{if(TIM_GetITStatus(TIM4, TIM_IT_CC4) == SET) {TIM_ClearITPendingBit(TIM4, TIM_IT_CC4);if(capture_cnt == 1)        //第一次上升沿{TIM_SetCounter(TIM4, 0);high_val = TIM_GetCapture4(TIM4);TIM_OC4PolarityConfig(TIM4, TIM_ICPolarity_Falling);    //下降沿捕获    capture_cnt = 2;}else if(capture_cnt == 2)  //第一次下降沿{low_val = TIM_GetCapture4(TIM4);TIM_OC4PolarityConfig(TIM4, TIM_ICPolarity_Rising);    //上升沿捕获capture_cnt = 3;}else if(capture_cnt == 3)  //第二次上升沿{high_val = TIM_GetCapture4(TIM4);Cap_Freq = 84000000 / 84 / (high_val+1);    //如果是高级定时器(TIM1、TIM8))需要修改为Cap_Freq = 168000000 / 168 / (high_val+1);Cap_Duty = (low_val+1) * 100 / (high_val+1);capture_cnt = 1;capture_end_flag = 1;}}
}

标准库主函数:

#include "stm32f4xx.h"
#include <stdio.h>
#include "usart.h"
#include "delay.h"
#include "pwm.h"int main(void)
{NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);USART1_Init();printf("\r\nStarting...\r\n");Capture_Wave_Init();int cnt = 0;while(1){delay_ms(1000);printf("--------%d--------\r\n", cnt++);if(capture_end_flag == 1){printf("Cap_Freq:%d\r\n", Cap_Freq);printf("Cap_Duty:%d\r\n", Cap_Duty);printf("high_val:%d\r\n", high_val);printf("low_val:%d\r\n",  low_val);}		}return 0;
}

输出100Hz,占空比为99%的PWM信号

标准库版的TIM输入捕获到的信号数据

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

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

相关文章

Mac安装配置maven

Mac安装配置maven 官网下载地址&#xff1a;https://maven.apache.org/download.cgi 下载好以后解压配置 maven 环境变量 打开终端&#xff0c;输入命令打开配置文件./bash_profile open ~/.bash_profile输入i进入编辑模式,进行maven配置; MAVEN_HOME为maven的本地路径 ex…

机器学习3-简单线性回归

需求&#xff1a; 现在要根据学生的学习时间来预测学习成绩&#xff0c;给出现有数据&#xff0c;用来训练模型并预测新数据。 分析&#xff1a; 使用线性回归模型。 代码&#xff1a; import pandas as pd import matplotlib.pyplot as plt from sklearn.model_selection i…

蓝桥杯备战——8.DS1302时钟芯片

1.分析原理图 由上图可以看到&#xff0c;芯片的时钟引脚SCK接到了P17,数据输出输入引脚IO接到P23,复位引脚RST接到P13。 2.查阅DS1302芯片手册 具体细节还需自行翻阅手册&#xff0c;我只截出重点部分 总结&#xff1a;数据在上升沿写出&#xff0c;下降沿读入&#xff0c;…

PDF控件Spire.PDF for .NET【安全】演示:使用 C# 检测签名的 PDF 是否被修改

对 PDF 文档进行数字签名后&#xff0c;PDF 将被锁定以防止更改或允许检测更改。在本文中&#xff0c;我们将介绍如何使用 Spire.PDF 检测签名的 PDF 是否被修改。 Spire.PDF for .NET 是一款独立 PDF 控件&#xff0c;用于 .NET 程序中创建、编辑和操作 PDF 文档。使用 Spire…

鸿蒙首批原生应用!无感验证已完美适配鸿蒙系统

顶象无感验证已成功适配鸿蒙系统&#xff0c;成为首批鸿蒙原生应用&#xff0c;助力鸿蒙生态的快速发展。 作为全场景分布式操作系统&#xff0c;鸿蒙系统旨在打破不同设备之间的界限&#xff0c;实现极速发现、极速连接、硬件互助、资源共享。迄今生态设备数已突破8亿台&…

结构体与共用体基础

结构体基础用法与共用体简述 1.结构体的定义2.结构体声明及使用3.结构体成员初始化4.结构体占用空间探究4.1 结构体成员所在地址4.2 按地址值访问结构体内容4.3 内存对齐 5.共用体6.总结 1.结构体的定义 之前的课程中&#xff0c;我们介绍了很多数据类型&#xff0c;如整形、浮…

测试用例级别该如何定义 ? 在工作中该如何应用它 ? 把握好这5个场景即可。

1.级别的作用 在编写测试用例的过程中&#xff0c;用例的级别经常是一个不可缺少的字段 &#xff0c;本篇幅就来聊下这个字段 &#xff0c;首先从它的作用是什么呢 &#xff1f;我觉得主要有两点 &#xff0c;分别是 &#xff1a; 用于测试用例不同套件的选取 &#xff0c;即用…

MMCLMC公差计算.exe

一、概要 软件及完整代码请戳这里&#xff1a;MMC&LMC公差计算软件及代码 图1 软件操作界面 本软件功能主要是根据实际应用选择MMR或者LMR原则&#xff0c;输入基本尺寸、形位公差尺寸和实际测量尺寸&#xff0c;即可计算出对应的公差值。以孔的MMR为例见如图2、3&#xf…

Java - JDBC

Java - JDBC 文章目录 Java - JDBC引言JDBC1 什么是JDBC2 MySQL数据库驱动3 JDBC开发步骤4 具体介绍 引言 思考: 当下我们如何操作数据库&#xff1f; 使用客户端工具访问数据库&#xff0c;手工建立连接&#xff0c;输入用户名和密码登录。编写SQL语句&#xff0c;点击执行…

【论文阅读|半监督小苹果检测方法S3AD】

论文题目 &#xff1a; : Semi-supervised Small Apple Detection in Orchard Environments 项目链接&#xff1a;https://www.inf.uni-hamburg.de/en/inst/ab/cv/people/wilms/mad.html 摘要&#xff08;Abstract&#xff09; 农作物检测是自动估产或水果采摘等精准农业应用不…

GitCode|部分项目开源代码

1.EasyKeyboard 基于MFC的简单软键盘&#xff0c;使用vs2017开发 PangCoder / EasyKeyboard GitCode基于Windows平台的软键盘&#xff0c;使用VS2017开发&#xff0c;使用MFC框架https://gitcode.net/qq_36251561/easykeyboard 2.EncoderSimulator 基于WPF应用的编码器模拟工…

Vue2 通过.sync修饰符实现数据双向绑定

App.vue <template><div class"app"><buttonv-on:clickisShowtrue>退出按钮</button><BaseDialog:visible.syncisShow></BaseDialog></div> </template><script> import BaseDialog from "./components…