STM32H743IIT6-TIM 单通道输入捕获测频率

一、CubeMX

NVIC
在这里插入图片描述
RCC
在这里插入图片描述
DEBUG
在这里插入图片描述
USART3
在这里插入图片描述
TIM2
在这里插入图片描述
时钟树
在这里插入图片描述

二、代码

usart.c 串口重定向

/* USER CODE BEGIN 0 */
#include <stdio.h>
/* USER CODE END 0 */
/* USER CODE BEGIN 1 */
int fputc(int ch, FILE *f)
{
  HAL_UART_Transmit(&huart3, (uint8_t *)&ch, 1, 0xffff);
  return ch;
}
 
/**
  * 函数功能: 重定向c库函数getchar,scanf到DEBUG_USARTx
  * 输入参数: 无
  * 返 回 值: 无
  * 说    明:无
  */
int fgetc(FILE *f)
{
  uint8_t ch = 0;
  HAL_UART_Receive(&huart3, &ch, 1, 0xffff);
  return ch;
}
/* USER CODE END 1 */

main.c

/* USER CODE BEGIN Includes */
#include <stdio.h>
/* USER CODE END Includes */

变量命名:此处出现过一个问题,某个地方出现过很大的值,而且是固定的。因为两个相减的数类型是无符号整数 uint32_t,相减出负数,因此把类型改成 int

/* USER CODE BEGIN PV */
volatile int IC_Value1 = 0; // 第一次输入捕获寄存器值
volatile int IC_Value2 = 0; // 第二次输入捕获寄存器值
volatile int IC_Value3 = 0; // 第二次输入捕获寄存器值
volatile int tick = 0; // 输入捕获寄存器的差值
volatile uint32_t tim2_up_cnt = 0; // 定时器溢出次数。如果信号周期较长,计数器可能在两次输入捕获之间发生溢出。
uint8_t flag_tim = 0;

volatile float positive_pulse_width = 0.0; // 正向脉宽
volatile float frequency = 0.0; // 频率
volatile float tim2_up_cnt1 = 0; // 到下降沿时计数器的值
/* USER CODE END PV */

自定义函数: 此处出现过的问题是
单纯捕获上升沿时,出现问题;加入标志位 flag_tim,区分两次上升沿捕获,效果不好,会跳变。设置标志位的三种状态,每一次改变捕获的极性,能正确采集。

HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)函数用于计数溢出中断的次数。

/* USER CODE BEGIN 0 */
void process_data(){
	frequency  = (float)100000000 / tick;
	positive_pulse_width = (IC_Value2 - IC_Value1 + tim2_up_cnt1 * 0XFFFF) / 10000000;
	
	printf("t1.txt=\"%.2fHz\"\xff\xff\xff", frequency);
	printf("t3.txt=\"%.2fms\"\xff\xff\xff", positive_pulse_width / 10);

}	

void HAL_TIM_IC_CaptureCallback(TIM_HandleTypeDef *htim) // TIM 捕捉回调函数
{
	if (htim->Instance == TIM2 && htim->Channel == HAL_TIM_ACTIVE_CHANNEL_1){ 
		if (flag_tim == 0) // 捕获上升沿
		{
			flag_tim = 1; // 进入捕获下降沿状态
			tim2_up_cnt = 0; // 清零捕获溢出值
			__HAL_TIM_SET_CAPTUREPOLARITY(&htim2, TIM_CHANNEL_1, TIM_ICPOLARITY_FALLING); // 下降沿捕获
//			__HAL_TIM_SET_COUNTER(&htim4, 0); // 定时器置 0
			IC_Value1 = HAL_TIM_ReadCapturedValue(&htim2, TIM_CHANNEL_1); // 获取当前捕获寄存器的值
		}
		else if (flag_tim == 1) // 捕获下降沿
		{
			flag_tim = 2;
			IC_Value2 = HAL_TIM_ReadCapturedValue(&htim2, TIM_CHANNEL_1); // 获取当前捕获寄存器的值
			tim2_up_cnt1 = tim2_up_cnt;
			__HAL_TIM_SET_CAPTUREPOLARITY(&htim2, TIM_CHANNEL_1, TIM_INPUTCHANNELPOLARITY_RISING);
		}
		else if (flag_tim == 2){ // 第二个上升沿
			HAL_TIM_IC_Stop_IT(&htim2,TIM_CHANNEL_1);
			IC_Value3 = HAL_TIM_ReadCapturedValue(&htim2, TIM_CHANNEL_1);
			tick = tim2_up_cnt * 65536 + IC_Value3 - IC_Value1;
			flag_tim = 0;
			process_data();
			HAL_TIM_IC_Start_IT(&htim2, TIM_CHANNEL_1);		
		}
	}
	 
}	

void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim){ // 溢出中断
	UNUSED(htim);
	
	if (htim == &htim2){
			tim2_up_cnt ++; // 统计溢出中断的次数
	}
}
/* USER CODE END 0 */

主函数:
注意:
溢出中断需要这样启动HAL_TIM_Base_Start_IT(&htim2);输入捕获中断需要这样启动:HAL_TIM_IC_Start_IT(&htim2, TIM_CHANNEL_1);。参考书。

int main(void){

  /* USER CODE BEGIN 1 */

  /* USER CODE END 1 */

  /* MPU Configuration--------------------------------------------------------*/
  MPU_Config();

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

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

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_TIM2_Init();
  MX_USART3_UART_Init();
  /* USER CODE BEGIN 2 */
	HAL_TIM_Base_Start_IT(&htim2); // 根据书,溢出中断需要这样启动
	HAL_TIM_IC_Start_IT(&htim2, TIM_CHANNEL_1); // 输入捕获中断需要这样启动

  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */
}

三、存在的问题

  • 测量范围:1-100kHz
  • 屏幕刷新速度太快不知道怎么改
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值