蓝桥杯STM32G431——普通输入捕获模式测量PWM频率

输入捕获

输入捕获可以对输入的信号的上升沿,下降沿或者双边沿进行捕获,常用的有测量输入信号的脉宽和测量 PWM 输入信号的频率和占空比这两种

输入捕获分为普通输入捕获模式和PWM输入模式
在这里插入图片描述
输入捕获的两大核心功能:
1.捕获定时器的数值。
2.产生中断,类似于外中断,比如,上升沿产生中断。

在进入下面的学习之前首先要掌握输入通道与捕获通道的区别
输入通道:需要被测量的信号从定时器的外部引脚 TIMx_CH1/2/3/4 进入,通常叫 TI1/2/3/4,在后面的捕获讲解中对于要被测量的信号我们都以 TIx 为标准的叫法
捕获通道:捕获通道就是输入捕获整体结构图中的 IC1/2/3/4,每个捕获通道都有相对应的捕获寄存器 CCR1/2/3/4,当发生捕获的时候,计数器 CNT 的值就会被锁存到捕获寄存器

输入通道是用来输入信号的,捕获通道是用来捕获输入信号的通道,一个输入通道的信号可以同时输入给两个捕获通道。

例如输入通道 TI1 的信号经过滤波边沿检测器之后的 TI1FP1触发 和 TI1FP2 触发可以分别进入到捕获通道IC1 和 IC2,其实这就是 PWM 输入模式(用于测量输入PWM占空比),只有一路输入信号(TI1)却占用了两个捕获通道(IC1 和 IC2)。当只需要测量输入PWM的频率时候,用一个捕获通道即可(用于测量输入PWM的频率),输入通道和捕获通道的映射关系具体由寄存器 CCMRx 的位 CCxS[1:0]配置

测量PWM频率:普通输入捕获模式

当捕获到信号的跳变沿的时候,把计数器 CNT 的值锁存到捕获寄存器 CCR 中,这就是普通输入捕获模式
在这里插入图片描述
在测量频率的过程中其步骤为:
1、当捕获通道 TIx 上出现上升沿时,发生第一次捕获,计数器 CNT 的值会被锁存到捕获寄存器 CCR 中
2、进入捕获中断,在中断服务程序中记录一次捕获,并把捕获寄存器中的值读取到 count1 中,count1的值一般为0
3、当出现第二次上升沿时,发生第二次捕获,计数器 CNT 的值会再次被锁存到捕获寄存器 CCR 中
4、再次进入捕获中断,在捕获中断中,把捕获寄存器的值读取到 count2 中,并清除捕获记录标志。利用 count2和count1 的差值我们就可以算出信号的周期T=(count2-count1)/1MHz,F频率即为1/T=1M/(count2-0) Hz。

PWM信号发生原理图

在这里插入图片描述
由上图可知两路PWM信号分别由PA15和PB4输入,其复用引脚图如下所示
参考STM32G431RB数据手册第61页
在这里插入图片描述
在这里插入图片描述

PB4引脚输入一路PWM的CubeMX配置(PA15配置同理仅将引脚改变)

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
打开中断,配置中断服务优先级
在这里插入图片描述
在这里插入图片描述

编写测量输入捕获两路PWM频率值的代码

示波器测得pwm的频率范围

PWM1 output = 720hz-22.4khz
PWM2 output = 700hz-24.0khz

pwm_tim23.c文件

#include "pwm_tim23.h"

TIM_HandleTypeDef htim2;
TIM_HandleTypeDef htim3;	//句柄结构体

void PWM_TIM2_Init(void)	//定时器2初始化
{
  TIM_ClockConfigTypeDef sClockSourceConfig = {0};
  TIM_SlaveConfigTypeDef sSlaveConfig = {0};
  TIM_MasterConfigTypeDef sMasterConfig = {0};
  TIM_IC_InitTypeDef sConfigIC = {0};

  htim2.Instance = TIM2;
  htim2.Init.Prescaler = 79;
  htim2.Init.CounterMode = TIM_COUNTERMODE_UP;
  htim2.Init.Period = 65535;
  htim2.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1;
  htim2.Init.AutoReloadPreload = TIM_AUTORELOAD_PRELOAD_DISABLE;
  if (HAL_TIM_Base_Init(&htim2) != HAL_OK)
  {
    Error_Handler();
  }
  sClockSourceConfig.ClockSource = TIM_CLOCKSOURCE_INTERNAL;
  if (HAL_TIM_ConfigClockSource(&htim2, &sClockSourceConfig) != HAL_OK)
  {
    Error_Handler();
  }
  if (HAL_TIM_IC_Init(&htim2) != HAL_OK)
  {
    Error_Handler();
  }
  sSlaveConfig.SlaveMode = TIM_SLAVEMODE_RESET;
  sSlaveConfig.InputTrigger = TIM_TS_TI1FP1;
  sSlaveConfig.TriggerPolarity = TIM_INPUTCHANNELPOLARITY_RISING;
  sSlaveConfig.TriggerFilter = 0;
  if (HAL_TIM_SlaveConfigSynchro(&htim2, &sSlaveConfig) != HAL_OK)
  {
    Error_Handler();
  }
  sMasterConfig.MasterOutputTrigger = TIM_TRGO_RESET;
  sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE;
  if (HAL_TIMEx_MasterConfigSynchronization(&htim2, &sMasterConfig) != HAL_OK)
  {
    Error_Handler();
  }
  sConfigIC.ICPolarity = TIM_INPUTCHANNELPOLARITY_RISING;
  sConfigIC.ICSelection = TIM_ICSELECTION_DIRECTTI;
  sConfigIC.ICPrescaler = TIM_ICPSC_DIV1;
  sConfigIC.ICFilter = 0;
  if (HAL_TIM_IC_ConfigChannel(&htim2, &sConfigIC, TIM_CHANNEL_1) != HAL_OK)
  {
    Error_Handler();
  }
}

/* TIM3 init function */
void PWM_TIM3_Init(void) //定时器3初始化
{

  TIM_ClockConfigTypeDef sClockSourceConfig = {0};
  TIM_SlaveConfigTypeDef sSlaveConfig = {0};
  TIM_MasterConfigTypeDef sMasterConfig = {0};
  TIM_IC_InitTypeDef sConfigIC = {0};

  htim3.Instance = TIM3;
  htim3.Init.Prescaler = 79;
  htim3.Init.CounterMode = TIM_COUNTERMODE_UP;	
  htim3.Init.Period = 65535;
  htim3.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1;
  htim3.Init.AutoReloadPreload = TIM_AUTORELOAD_PRELOAD_DISABLE;
  if (HAL_TIM_Base_Init(&htim3) != HAL_OK)
  {
    Error_Handler();
  }
  sClockSourceConfig.ClockSource = TIM_CLOCKSOURCE_INTERNAL;
  if (HAL_TIM_ConfigClockSource(&htim3, &sClockSourceConfig) != HAL_OK)
  {
    Error_Handler();
  }
  if (HAL_TIM_IC_Init(&htim3) != HAL_OK)
  {
    Error_Handler();
  }
  sSlaveConfig.SlaveMode = TIM_SLAVEMODE_RESET;
  sSlaveConfig.InputTrigger = TIM_TS_TI1FP1;
  sSlaveConfig.TriggerPolarity = TIM_INPUTCHANNELPOLARITY_RISING;
  sSlaveConfig.TriggerFilter = 0;
  if (HAL_TIM_SlaveConfigSynchro(&htim3, &sSlaveConfig) != HAL_OK)
  {
    Error_Handler();
  }
  sMasterConfig.MasterOutputTrigger = TIM_TRGO_RESET;
  sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE;
  if (HAL_TIMEx_MasterConfigSynchronization(&htim3, &sMasterConfig) != HAL_OK)
  {
    Error_Handler();
  }
  sConfigIC.ICPolarity = TIM_INPUTCHANNELPOLARITY_RISING;
  sConfigIC.ICSelection = TIM_ICSELECTION_DIRECTTI;
  sConfigIC.ICPrescaler = TIM_ICPSC_DIV1;
  sConfigIC.ICFilter = 0;
  if (HAL_TIM_IC_ConfigChannel(&htim3, &sConfigIC, TIM_CHANNEL_1) != HAL_OK)
  {
    Error_Handler();
  }

}

void HAL_TIM_Base_MspInit(TIM_HandleTypeDef* tim_baseHandle)	//IO口配置函数(包含time6)
{

	GPIO_InitTypeDef GPIO_InitStruct = {0};
  if(tim_baseHandle->Instance==TIM2)
  {
    /* TIM2 clock enable */
    __HAL_RCC_TIM2_CLK_ENABLE();

    __HAL_RCC_GPIOA_CLK_ENABLE();
    /**TIM2 GPIO Configuration
    PA15     ------> TIM2_CH1
    */
    GPIO_InitStruct.Pin = GPIO_PIN_15;
    GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
    GPIO_InitStruct.Pull = GPIO_NOPULL;
    GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
    GPIO_InitStruct.Alternate = GPIO_AF1_TIM2;
    HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);

    /* TIM2 interrupt Init */
    HAL_NVIC_SetPriority(TIM2_IRQn, 3, 0);
    HAL_NVIC_EnableIRQ(TIM2_IRQn);
  }
  else if(tim_baseHandle->Instance==TIM3)
  {
    /* TIM3 clock enable */
    __HAL_RCC_TIM3_CLK_ENABLE();

    __HAL_RCC_GPIOB_CLK_ENABLE();
    /**TIM3 GPIO Configuration
    PB4     ------> TIM3_CH1
    */
    GPIO_InitStruct.Pin = GPIO_PIN_4;
    GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
    GPIO_InitStruct.Pull = GPIO_NOPULL;
    GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
    GPIO_InitStruct.Alternate = GPIO_AF2_TIM3;
    HAL_GPIO_Init(GPIOB, &GPIO_InitStruct);

    /* TIM3 interrupt Init */
    HAL_NVIC_SetPriority(TIM3_IRQn, 3, 0);
    HAL_NVIC_EnableIRQ(TIM3_IRQn);
  }
  else if(tim_baseHandle->Instance==TIM6)
  {
    /* TIM6 clock enable */
    __HAL_RCC_TIM6_CLK_ENABLE();

    /* TIM6 interrupt Init */
    HAL_NVIC_SetPriority(TIM6_DAC_IRQn, 2, 0);
    HAL_NVIC_EnableIRQ(TIM6_DAC_IRQn);
  }
}

pwm_tim23.h文件

#ifndef __PWM_TIM23_H
#define __PWM_TIM23_H
#include "main.h"
extern TIM_HandleTypeDef htim3;
extern TIM_HandleTypeDef htim2;

void PWM_TIM3_Init(void);
void PWM_TIM2_Init(void);
#endif

main.c文件

#include "main.h"
#include "stdio.h"
#include "string.h"
#include "lcd.h"
#include "basic_tim6.h"
#include "pwm_tim23.h"

__IO uint32_t uwTick_LCD_State_Pointer;
unsigned char Lcd_Disp_String[21];
uint8_t i;	//用于基础定时器6中的变量
uint16_t pwm1_value;
uint16_t pwm2_value;	//用于pwm回调函数中的变量

void SystemClock_Config(void);
void LCD_Proc(void);
int main(void)
{

  HAL_Init();
  /* Configure the system clock */
  SystemClock_Config();

	//**LCD初始化
	LCD_Init();
	LCD_Clear(White);
	LCD_SetBackColor(White);
	LCD_SetTextColor(Blue);
	
	//基础定时器6的初始化 若没有或不需要基础定时器6可将此部分去除
	BASIC_TIM6_Init();
	HAL_TIM_Base_Start_IT(&htim6);

	//PWM初始化
	PWM_TIM2_Init(); 	//定时器2初始化
	PWM_TIM3_Init();	//定时器3初始化
	HAL_TIM_Base_Start_IT(&htim2);	//启动定时器2
	HAL_TIM_Base_Start_IT(&htim3);	//启动定时器3
	HAL_TIM_IC_Start_IT(&htim2 , TIM_CHANNEL_1);	//启动定时器2开启通道输入捕获并开启中断
	HAL_TIM_IC_Start_IT(&htim3 , TIM_CHANNEL_1);	//启动定时器3开启通道输入捕获并开启中断
	
  while (1)
  {
		LCD_Proc();
  }
}
void LCD_Proc(void)
{

	
	if(uwTick-uwTick_LCD_State_Pointer<300) return;
	uwTick_LCD_State_Pointer=uwTick;	
	
	memset(Lcd_Disp_String,0,sizeof(Lcd_Disp_String));
	sprintf((char*)Lcd_Disp_String, "Timer6_Num : %03d" ,i);
	LCD_DisplayStringLine(Line0, Lcd_Disp_String);
	
	memset(Lcd_Disp_String,0,sizeof(Lcd_Disp_String));
	sprintf((char*)Lcd_Disp_String, "  pwm1_freq: %06d  ",(unsigned int)1000000/pwm1_value); //频率值为1M/pwm_value 1M=1000000
	LCD_DisplayStringLine(Line3, Lcd_Disp_String);
	
	memset(Lcd_Disp_String,0,sizeof(Lcd_Disp_String));
	sprintf((char*)Lcd_Disp_String, "  pwm2_freq: %06d  ",(unsigned int)1000000/pwm2_value);
	LCD_DisplayStringLine(Line4, Lcd_Disp_String);
}

void HAL_TIM_IC_CaptureCallback(TIM_HandleTypeDef *htim) //输入捕获中断回调函数
{
	if(htim->Instance == TIM2)	//加判断语句用于判别是time2还是time3
  {
		if(htim->Channel == HAL_TIM_ACTIVE_CHANNEL_1)
		{
			pwm2_value = HAL_TIM_ReadCapturedValue(htim, TIM_CHANNEL_1)+1;
		}
	}
	
	if(htim->Instance == TIM3)
  {
		if(htim->Channel == HAL_TIM_ACTIVE_CHANNEL_1)
		{
			pwm1_value = HAL_TIM_ReadCapturedValue(htim, TIM_CHANNEL_1)+1;
		}
	}
	
}

void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)	//基本定时器6中断回调函数
{
	if(htim->Instance == TIM6)
	{
		i++;
		HAL_TIM_Base_Start_IT(&htim6);
	}
}

总结

配置步骤
1、初始化需要用到的定时器
2、启动定时器 HAL_TIM_Base_Start_IT()
3、启动定时器开启对应通道的输入捕获模式并开启中断 HAL_TIM_IC_Start_IT()
4、编写中断回调函数 void HAL_TIM_IC_CaptureCallback(TIM_HandleTypeDef *htim)
5、计算频率值f=(80M/80)/count(捕获寄存器CCR中的值),后在LCD上显示
测量到的最低频率效果图为
在这里插入图片描述

  • 11
    点赞
  • 52
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 8
    评论
### 回答1: 蓝桥杯是全国性的计算机大赛,旨在提高大学生的计算机技术水平和创新能力。嵌入式STM32G431是一款嵌入式芯片,具有高性能和低功耗的特点。第十二届第一场省赛停车计费系统是蓝桥杯的一道工程题目,要求参赛选手设计一个停车场计费系统。 停车计费系统是指通过嵌入式技术实现对车辆进入和离开停车场的自动检、计时和计费的系统。对于这个题目,参赛选手可以根据题目要求,使用STM32G431芯片设计一个具有以下功能的系统: 1. 车辆进入检:使用车辆传感器检车辆的进入,触发计时。 2. 车辆离开检:使用车辆传感器检车辆的离开,停止计时。 3. 计时功能:使用内部时钟模块获取进入和离开的时间,并计算停车时间。 4. 计费功能:根据停车时间,按照一定的计费规则进行计费。 5. 显示功能:使用液晶显示屏显示当前的停车信息,如车牌号、停车时间、费用等。 6. 数据存储功能:使用闪存等储存介质将停车数据进行存储,以便后续的数据分析和查询。 设计一个停车计费系统涉及到硬件设计和软件编程两个方面。硬件方面,参赛选手需要选择合适的传感器、显示屏、存储介质等,以及设计电路和接口进行连接。软件方面,需要使用C语言或者汇编语言编写程序,对芯片进行编程,实现各项功能。 通过此次比赛,选手们可以锻炼嵌入式系统的设计能力和编程能力,了解实际应用中嵌入式系统的工作原理和应用场景。同时,也能提升对STM32G431芯片的理解和运用能力。这对于培养嵌入式技术人才,推动物联网技术的发展都具有积极意义。 ### 回答2: 蓝桥杯是面向计算机爱好者的智力竞赛,而嵌入式STM32G431是一款嵌入式系统开发板。第十二届第一场省赛的停车计费系统真题工程,则是要求参赛选手设计并实现一个能够进行停车计费的系统。 停车计费系统是一种用于自动计费和管理停车场的系统。这个系统可以通过识别车辆的进出以及停车的时间,自动计算并收费。在这个工程中,选手需要使用嵌入式STM32G431开发板以及相关的硬件和软件,来设计一个能够实现停车计费功能的系统。 首先,选手需要使用传感器或摄像头来实现车辆进出的检。当车辆进入或离开停车场时,传感器会发出信号并通过STM32G431进行检。接着,选手需要编程实现识别车辆的算法,以便能够识别不同的车辆。当车辆停放时,系统会通过计时器记录停车的时间。 然后,选手需要编写计费算法,根据停车的时间来计算费用。这个算法可以根据停车场的规则来确定费用的计算方式,例如按小时计费或按照不同的时间段采用不同的费率。 最后,选手还需要设计一个界面,使得系统能够与用户进行交互。用户可以通过该界面查询停车费用,并进行付款。同时,选手还需要保证系统的稳定性和安全性,确保数据的准确性和保密性。 总而言之,蓝桥杯嵌入式STM32G431第十二届第一场省赛停车计费系统真题工程要求选手使用硬件和软件开发能力,设计并实现一个完整的停车计费系统。这需要选手具备嵌入式系统开发、传感器技术、算法设计和界面设计等多方面的知识和技能。 ### 回答3: 蓝桥杯嵌入式stm32g431——第十二届第一场省赛停车计费系统真题工程是一个基于STM32G431开发板的停车计费系统设计题目。该系统的主要功能是实现停车场的车辆进入、出场的计费管理。 首先,该系统需要实现一个车辆进入检的功能。当车辆进入停车场时,系统会通过传感器检到车辆的到达,并记录下进入的时间。 然后,系统需要实现一个车辆出场的功能。当车辆准备离开停车场时,系统会通过传感器检到车辆的离开,并记录下离开的时间。 接下来,系统需要计算车辆停留的时间。通过进入时间和离开时间的差值计算得出车辆停留的时长。 最后,系统需要计算停车费用。根据停车场的计费规则,根据车辆停留的时长计算出停车费用,并将费用显示在屏幕上。 在实现上述功能的过程中,需要运用STM32G431开发板的各种功能和模块,包括GPIO口、定时器、中断、串口通信等。通过编写相应的代码,实现按钮的控制、传感器检、时间的计算以及屏幕的显示等功能。 该项目的设计需要考虑多种情况,比如车辆重复进入出场、车辆进入出场顺序错乱等,要充分考虑这些异常情况并进行相应的处理。在代码编写过程中,还需注意代码的优化和资源的合理利用,以提高系统的性能和效率。 通过完成这个项目的设计与实现,可以加深对STM32G431嵌入式系统的了解和应用,并提升嵌入式系统开发的能力。同时,也能锻炼自己的逻辑思维、问题解决能力和团队合作精神。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

lzya.

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值