STM32标准库+HAL库 输入捕获测量PWM的脉冲频率+占空比_stm32 pwm 捕获 hal(2)

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

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

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

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

在将计数器counter period的值,也就是自动重装载值ARR拉满时(0~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_CH4
    
    GPIO_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)  //第一次下降沿
**自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。**

**深知大多数嵌入式工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!**

**因此收集整理了一份《2024年嵌入式&物联网开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。**

![img](https://img-blog.csdnimg.cn/img_convert/c76c1903c0139f0fb83015b4075cbbd7.png)

![img](https://img-blog.csdnimg.cn/img_convert/51f914fad4d4903257c2e6166217df55.jpeg)

![img](https://img-blog.csdnimg.cn/img_convert/43f7d7ae743618d2a1ccdbfe98b08e85.png)

 **既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上嵌入式&物联网开发知识点,真正体系化!**

![img](https://img-blog.csdnimg.cn/img_convert/d459d3791308928992e0e6e0472ef363.png)

![img](https://img-blog.csdnimg.cn/img_convert/769a5c6d4839e912075ba1924eef332d.png)

 

**由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新**

**如果你觉得这些内容对你有帮助,可以+V:Vip1104z获取!!! (备注:嵌入式)**

<img src="https://img-community.csdnimg.cn/images/73bb5de17851459088c6af944156ee24.jpg" alt="img" style="zoom: 67%;" />



# 最后

**资料整理不易,觉得有帮助的朋友可以帮忙点赞分享支持一下小编~**

**你的支持,我的动力;祝各位前程似锦,offer不断,步步高升!!!**

后续会持续更新**

**如果你觉得这些内容对你有帮助,可以+V:Vip1104z获取!!! (备注:嵌入式)**

<img src="https://img-community.csdnimg.cn/images/73bb5de17851459088c6af944156ee24.jpg" alt="img" style="zoom: 67%;" />



# 最后

**资料整理不易,觉得有帮助的朋友可以帮忙点赞分享支持一下小编~**

**你的支持,我的动力;祝各位前程似锦,offer不断,步步高升!!!**

**[更多资料点击此处获qu!!](https://bbs.csdn.net/topics/618376385)**
  • 5
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
你好!关于STM32 HAL库PWM输入捕获功能,我可以给你一些基本的介绍。 在STM32中,PWM输入捕获是一种用于测量外部信号的功能。通过配置PWM输入捕获,你可以测量一个PWM信号的频率占空比或者脉冲宽度。 首先,你需要选择一个可用的定时器和通道来进行PWM输入捕获。然后,使用HAL库的函数来配置定时器和通道,并启用输入捕获模式。 以下是一个简单的示例代码来实现PWM输入捕获: ```c #include "stm32f4xx_hal.h" TIM_HandleTypeDef htim3; void HAL_TIM_IC_CaptureCallback(TIM_HandleTypeDef *htim) { if(htim->Instance == TIM3) { uint32_t captureValue = HAL_TIM_ReadCapturedValue(htim, TIM_CHANNEL_1); // 读取捕获的值 // 进行处理或者显示 } } int main(void) { HAL_Init(); SystemClock_Config(); __HAL_RCC_TIM3_CLK_ENABLE(); htim3.Instance = TIM3; htim3.Init.Prescaler = 0; htim3.Init.CounterMode = TIM_COUNTERMODE_UP; htim3.Init.Period = 0xFFFF; htim3.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1; HAL_TIM_IC_Init(&htim3); TIM_IC_InitTypeDef sConfigIC; sConfigIC.ICPolarity = TIM_ICPOLARITY_RISING; sConfigIC.ICSelection = TIM_ICSELECTION_DIRECTTI; sConfigIC.ICPrescaler = TIM_ICPSC_DIV1; sConfigIC.ICFilter = 0; HAL_TIM_IC_ConfigChannel(&htim3, &sConfigIC, TIM_CHANNEL_1); HAL_TIM_IC_Start_IT(&htim3, TIM_CHANNEL_1); while (1) { // Your code here } } ``` 在上述代码中,我们首先初始化定时器TIM3,并配置通道1为输入捕获模式。然后,在主循环中你可以添加自己的代码逻辑。 需要注意的是,在使用PWM输入捕获之前,你需要根据你的需求进行适当的配置和调整代码。 希望能对你有所帮助!如果有任何进一步的问题,请随时提问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值