Air001 高级定时器输入捕获功能测量脉宽和频率

news2025/1/11 7:42:24

Air001 高级定时器输入捕获功能测量脉宽和频率


  • ✨Air001只有1个16位高级定时器,经实际测试发现,通道1用于输入捕获功能失效,不确定是否是IO引脚存在问题还是硬件bug,折腾了好久,最后切换到通道2使用,就可以捕获到了,其余通道3和通道4没有做有效性测试。
  • 🌿对于ARM 32位的M0+内核单片机,同STM32同内核架构的单片机,在定时器的使用上都一样,代码基本可以照搬。
  • 🔰本文主要是将STM32代码移植到该芯片工程中,验证功能。
  • 🌿测试了采用高级定时器通道1和通道2,复位模式,测量脉宽信号,发现也是不行的,在STM32f030上使用都正常的。《STM32 HAL库定时器输入捕获SlaveMode脉宽测量》
    在这里插入图片描述
  • 🔖Air001价格比较便宜,硬件上还是有很多暗坑或设计缺陷的,在选择移植切换方案时,最好提前做好功能验证。
  • 📑定时器输入捕获配置函数:
void MX_TIM1_Init(void)
{

  /* USER CODE BEGIN TIM1_Init 0 */

  /* USER CODE END TIM1_Init 0 */

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

  /* USER CODE BEGIN TIM1_Init 1 */

  /* USER CODE END TIM1_Init 1 */
  htim1.Instance = TIM1;
  htim1.Init.Prescaler = 23;//1MHz
  htim1.Init.CounterMode = TIM_COUNTERMODE_UP;
  htim1.Init.Period = 65535;
  htim1.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1;
  htim1.Init.RepetitionCounter = 0;
  htim1.Init.AutoReloadPreload = TIM_AUTORELOAD_PRELOAD_DISABLE;
  if (HAL_TIM_Base_Init(&htim1) != HAL_OK)
  {
    Error_Handler();
  }
  sClockSourceConfig.ClockSource = TIM_CLOCKSOURCE_INTERNAL;
  if (HAL_TIM_ConfigClockSource(&htim1, &sClockSourceConfig) != HAL_OK)
  {
    Error_Handler();
  }
  if (HAL_TIM_IC_Init(&htim1) != HAL_OK)
  {
    Error_Handler();
  }
  sMasterConfig.MasterOutputTrigger = TIM_TRGO_RESET;
  sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE;
  if (HAL_TIMEx_MasterConfigSynchronization(&htim1, &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(&htim1, &sConfigIC, TIM_CHANNEL_2) != HAL_OK)//通道2
  {
    Error_Handler();
  }
	  /* TIM1使能启动,并使能中断 */
  if (HAL_TIM_Base_Start_IT(&htim1) != HAL_OK)
  {
    Error_Handler();
  }
}

  • 🔧定时器输入捕获引脚配置以及中断优先级配置
void HAL_TIM_Base_MspInit(TIM_HandleTypeDef* tim_baseHandle)
{
	GPIO_InitTypeDef GPIO_InitStruct = {0};
  if(tim_baseHandle->Instance==TIM1)
  {
  /* USER CODE BEGIN TIM1_MspInit 0 */

  /* USER CODE END TIM1_MspInit 0 */
    /* TIM1 clock enable */
    __HAL_RCC_TIM1_CLK_ENABLE();
//		__HAL_RCC_GPIOA_CLK_ENABLE();
    __HAL_RCC_GPIOB_CLK_ENABLE();
    /**TIM1 GPIO Configuration
    PA3   GPIO_AF13_TIM1  ------> TIM1_CH1
		PB3  GPIO_AF1_TIM1  ------> TIM1_CH2
	
    */
    GPIO_InitStruct.Pin = GPIO_PIN_3;
    GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
    GPIO_InitStruct.Pull = GPIO_NOPULL;
    GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH;
    GPIO_InitStruct.Alternate = GPIO_AF1_TIM1;
    HAL_GPIO_Init(GPIOB, &GPIO_InitStruct);

    /* TIM1 interrupt Init */
		
    HAL_NVIC_SetPriority(TIM1_BRK_UP_TRG_COM_IRQn, 2, 0);
    HAL_NVIC_EnableIRQ(TIM1_BRK_UP_TRG_COM_IRQn);
    HAL_NVIC_SetPriority(TIM1_CC_IRQn, 1, 0);//输入捕获优先级>更新溢出中断
    HAL_NVIC_EnableIRQ(TIM1_CC_IRQn);
  /* USER CODE BEGIN TIM1_MspInit 1 */

  /* USER CODE END TIM1_MspInit 1 */
  }
  if(tim_baseHandle->Instance==TIM16)//用于产生PWM信号
  {
  /* USER CODE BEGIN TIM16_MspInit 0 */

  /* USER CODE END TIM16_MspInit 0 */
    /* TIM16 clock enable */
    __HAL_RCC_TIM16_CLK_ENABLE();
  /* USER CODE BEGIN TIM16_MspInit 1 */

  /* USER CODE END TIM16_MspInit 1 */
  }
}
  • 🌿系统时钟频率配置(本例程测试用于内部时钟,频率24MHz)
/**
  * @brief   配置系统时钟
  * @param   HSICLKSource_SET:选择HSI时钟频率
  *            @arg @ref RCC_HSICALIBRATION_8MHz:8M时钟
  *            @arg @ref RCC_HSICALIBRATION_16MHz:16M时钟
  *            @arg @ref RCC_HSICALIBRATION_22p12MHz:22.12M时钟
  *            @arg @ref RCC_HSICALIBRATION_24MHz:24M时钟
  * @retval  无
  */
void SystemClock_Config(uint32_t HSICLKSource_SET)
{
    RCC_OscInitTypeDef RCC_OscInitStruct = {0};
    RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};

//  RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;
//  RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSI;
    /* 配置HSI时钟 */
    RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSI;
    RCC_OscInitStruct.HSIState = RCC_HSI_ON;                                      /* 使能HSI */
    RCC_OscInitStruct.HSIDiv =    RCC_HSI_DIV1;                                   /* HSI预分频 */
    RCC_OscInitStruct.HSICalibrationValue = HSICLKSource_SET;
    /* 设置HSI输出时钟库会设置校准值 */
    if(HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK)                           /* 配置时钟 */
    {
        Error_Handler();
    }

    /* 初始化AHB,APB总线时钟 */
    RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK | RCC_CLOCKTYPE_SYSCLK | RCC_CLOCKTYPE_PCLK1;
    RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_HSI;                         /* 配置AHB时钟源 */
    RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;                             /* 设置AHB预分频 */
    RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV1;                              /* 设置APB1预分频 */

    if(HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_1) != HAL_OK)         /* 配置总线 */
    {
        Error_Handler();
    }
}
  • 🌿定时器中断回调函数
/**
  * @brief 输入捕获回调函数
  * @retval None
  */
void HAL_TIM_IC_CaptureCallback(TIM_HandleTypeDef* htim)
{
    static uint8_t RisingEdge_count = 0;
    if(htim->Channel == HAL_TIM_ACTIVE_CHANNEL_2)
    {
        __HAL_TIM_CLEAR_IT(&htim1, TIM_IT_CC2);//清零中断标志位
        if(capture_flag & 0x40)										//0X40是0100 0000,高电平期间捕获到下降沿
        {
            capture_flag &= 0x3F;										//0X3F是0011 1111,清除捕获到上升沿的标记位和捕获完成的标记位
            value2 = HAL_TIM_ReadCapturedValue(htim, TIM_CHANNEL_2);	//获取当前的捕获值
            __HAL_TIM_DISABLE(htim);        						//关闭定时器
            __HAL_TIM_SET_COUNTER(htim, value2);						//以value2为基准重新计数
            TIM_RESET_CAPTUREPOLARITY(htim, TIM_CHANNEL_2);      	//复位极性选择才能进行下行配置
            /*输入捕获功能的重配与开启,硬件启动会产生几个时钟的延迟*/
            TIM_SET_CAPTUREPOLARITY(htim, TIM_CHANNEL_2, TIM_ICPOLARITY_RISING);	//下次上升沿捕获
            __HAL_TIM_ENABLE(htim);									//重开定时器
        }
        else	            								 		//捕获到上升沿
        {
            capture_flag |= 0x40;										//0X40是0100 0000,标记捕捉到了一次上升沿
            RisingEdge_count++;
            if((RisingEdge_count % 2 == 0))								//每捕获两次相同跳变沿表示过了一个PWM周期
            {
                value3 = HAL_TIM_ReadCapturedValue(htim, TIM_CHANNEL_2); //检测完一个周期的那个上升沿为value3
                capture_flag |= 0x80;								 //标记捕获完了一个周期
                Pulse_Width = (value2 + OverflowCount_high * TIM1_Period_Value - value1 + 7)/2;	//7 时钟补偿
                PWM_Period_ARR[PWM_Period_CNT++] = value3 + OverflowCount_high * TIM1_Period_Value + OverflowCount_low * TIM1_Period_Value - value1 - 3;
                if(PWM_Period_CNT == 5)//5次读取,取出现频率最高的值
                {
                    PWM_Period = findMostFrequentNum(PWM_Period_ARR, 5)/2;//2MHZ计数
                    PWM_Period_CNT = 0;
                }
                OverflowCount_high = OverflowCount_low = 0;//清空溢出计数

            }
            else				 								   //正在检测PWM信号的第一个上升沿,意味着下次捕获下降沿
            {
                capture_flag &= 0x7F;								 //0X7F是0111 1111,清除PWM捕获完成标志,开始新一轮PWM周期捕获
                value1 = HAL_TIM_ReadCapturedValue(htim, TIM_CHANNEL_2); //第一个上升沿是value1
            }
            __HAL_TIM_DISABLE(htim);
            __HAL_TIM_SET_COUNTER(htim, value1);					//以value1为基准重新计数
            TIM_RESET_CAPTUREPOLARITY(htim, TIM_CHANNEL_2);  	//复位极性选择才能进行下行配置
            TIM_SET_CAPTUREPOLARITY(htim, TIM_CHANNEL_2, TIM_ICPOLARITY_FALLING);	//下次下降沿捕获
            __HAL_TIM_ENABLE(htim);								//重开定时器
        }
    }
}

/**
  * @brief 更新溢出回调函数
  * @retval None
  */
void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef* htim)
{
	__HAL_TIM_CLEAR_IT(&htim1, TIM_IT_UPDATE);//清零中断标志位
    if((capture_flag & 0X80) == 0)			//PWM的一个周期没检测完
    {
        if(capture_flag & 0x40)				//在高电平期间溢出M次
        {
            OverflowCount_high++;
        }
        else								//在低电平期间溢出N次
        {
            OverflowCount_low++;
        }
    }
    else								   //PWM的一个周期检测完了
    {

        OverflowCount_high = 0;
        OverflowCount_low = 0;
    }
}

  • 📝main函数
volatile uint32_t  ARR_Value, PWM_f;
volatile char capture_flag = 0;				//捕获状态标记变量,0x80最高位标记捕获完一个周期,0x40表示捕获到了上升沿
volatile uint8_t OverflowCount_high = 0;		//高电平期间溢出次数
volatile uint8_t OverflowCount_low = 0;		//低电平期间溢出次数
volatile uint32_t value1, value2, value3;	  //三个边沿中的值
volatile uint32_t Pulse_Width = 0;		  //脉宽
volatile uint32_t PWM_Period = 0;				//周期
uint32_t PWM_Period_ARR[5] = {0};
uint8_t PWM_Period_CNT = 0;
int main(void)
{
    uint32_t TimerUART;
    uint8_t USART_TX_Buff[40] = {0};
    /* 初始化所有外设,Flash接口,SysTick */
    HAL_Init();
    SystemClock_Config(RCC_HSICALIBRATION_24MHz);
    APP_GpioConfig();
    HAL_TIM_Base_MspInit(&htim1);
    MX_TIM1_Init();
    MX_TIM16_Init();
    MX_USARTx_UART_Init();
//    __HAL_TIM_SET_AUTORELOAD(&htim16, plusewidth - 1); //调整分频系数,TIM16->ARR
//    __HAL_TIM_SET_COMPARE(&htim16, TIM_CHANNEL_2, plusedelay); //PWM脉冲宽度,TIM16->CCR1修改占空比比较值
    HAL_TIM_PWM_Start(&htim16, TIM_CHANNEL_1); //开启PWM输出通道:PA6
    HAL_TIM_Base_Start_IT(&htim1);
//		__HAL_TIM_ENABLE_IT(&htim1, TIM_IT_UPDATE); //启动更新中断
    __HAL_TIM_URS_ENABLE(&htim1);               //更新中断只有溢出时触发
    /* 清零中断标志位 */
    __HAL_TIM_CLEAR_IT(&htim1, TIM_IT_UPDATE);
    /* 使能定时器的更新事件中断 */
    __HAL_TIM_ENABLE_IT(&htim1, TIM_IT_UPDATE);//使能更新中断
    /* 使能输入捕获 */
    HAL_TIM_IC_Start_IT(&htim1, TIM_CHANNEL_2);
    TimerUART = HAL_GetTick();
    printf("Air001 SysClockFreq:%d \r\n", HAL_RCC_GetSysClockFreq());
    while(1)
    {
//        HAL_Delay(1000);
        if((HAL_GetTick() - TimerUART) > 1000)
        {
            /* LED翻转 */
            HAL_GPIO_TogglePin(GPIOB, GPIO_PIN_0);
            HAL_GPIO_TogglePin(GPIOB, GPIO_PIN_1);
            PWM_f = 1000000l / PWM_Period;

            sprintf((char*)USART_TX_Buff, "Pulse_Width:%dus,PWM_Period:%d,PWM_f:%dHz", Pulse_Width, PWM_Period, PWM_f);
            printf("%s \n", USART_TX_Buff);
            memset((char*)USART_TX_Buff, '\0', strlen((char*)USART_TX_Buff)); //清空数组
            TimerUART = HAL_GetTick();
        }
    }
}
  • 🌿当被测量信号频率越高时,误差就越大。
    在这里插入图片描述

在这里插入图片描述

  • 🌿被测信号频率比较低时,测量结果:
    在这里插入图片描述

📚测试工程

链接:https://pan.baidu.com/s/1OIf_XiWwRuBVPgidumX_SQ 
提取码:93ev

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

使用 PyAudio、语音识别、pyttsx3 和 SerpApi 构建简单的基于 CLI 的语音助手

德米特里祖布☀️ 一、介绍 正如您从标题中看到的,这是一个演示项目,显示了一个非常基本的语音助手脚本,可以根据 Google 搜索结果在终端中回答您的问题。 您可以在 GitHub 存储库中找到完整代码:dimitryzub/serpapi-demo-project…

fork() 详解,返回2次什么意思?

在 Unix-like 系统中,fork() 函数是用于创建新进程的系统调用。当调用 fork() 时,会创建一个新的子进程,该子进程是原始进程的副本。fork() 函数在父进程和子进程中返回不同的值,具体如下: 对于父进程:for…

万界星空科技云MES系统产品追溯功能介绍

制造业工厂产品质量贯穿于产品的整个生命周期,也是企业参与市场竞争求生存求发展的基础,而制造过程中出现的产品质量问题则是产品最终质量的基石。 MES系统是一种信息化管理系统,它可以实现产品的全生命周期的监控和管理,同时&am…

openGauss Meetup(天津站)精彩回顾 | openGauss天津用户组正式成立

由openGauss社区、天开发展集团、天津市软件行业协会、天大智图(天津)科技有限公司联合主办的“openGauss Meetup • 天津站”已于10月13日落下帷幕,此次活动邀请到众多业内技术专家,从技术创新、学术创新、发展创新、以及生态共建…

【面试系列】Vue

引言:下面是一些常见的 Vue 面试题和对应的答案 目录 1. Vue 是什么?它有哪些特点?2. Vue 的生命周期有哪些?请简要描述每个生命周期的作用。3. Vue 组件间通信的有哪些方式?4. Vue 的 computed 和 watch 的区别是什么…

【LeetCode:2530. 执行 K 次操作后的最大分数 | 堆】

🚀 算法题 🚀 🌲 算法刷题专栏 | 面试必备算法 | 面试高频算法 🍀 🌲 越难的东西,越要努力坚持,因为它具有很高的价值,算法就是这样✨ 🌲 作者简介:硕风和炜,…

javaEE - 1(9000字详解多线程)

一:认识线程 1.1 线程的概念 线程是操作系统中执行的最小单位,它是进程中的一个实体。一个进程可以包含多个线程,并且这些线程共享进程的资源,如内存、文件句柄等,但每个线程有自己的独立执行流程和栈空间。 线程在…

关于vant 的tabbar功能

1、想要实现tabbar页面A,其他的页面B(非tabbar页面)。 从A页面进入B页面,底部的active选中效果应该被取消掉,但是还是选中A。 按照官网的说法有两个方法 一、根据path路径 二、自定义的model 但是!但是…

孩子近视了用什么台灯好呢?对视力友好的护眼台灯推荐

对于很多家里有孩子写作业、或者夜晚需要面对电脑或手机屏幕加班写文案、搞剪辑等工作的朋友,一款“好”的台灯不仅可以保护眼睛、缓解视疲劳,而且还能够更贴心与智能,提升工作效率。最近总结了一期非常好用的护眼台灯,希望可以给…

如何恢复红米手机删除的照片/文件?(亲测有效的6 种方式)

如何恢复红米手机删除的照片/文件?(亲测有效的6 种方式) 凭借出色的相机和实惠的价格,小米红米系列已成为全球知名品牌。但是,最近有人抱怨说他们的红米手机丢失了很多珍贵的照片或视频,希望知道如何从小米…

Scrum敏捷开发方法

什么是Scrum敏捷开发方法? Scrum是一种广泛使用的敏捷开发方法,旨在提高软件开发和项目管理的效率。Scrum强调迭代、协作、自组织和透明度,使团队能够更好地应对不断变化的需求和复杂性。Scrum方法的核心思想是通过一系列短期周期来交付功能…

创建线程的四种方法(Java)

目录 一、继承 Thread类 二、实现Runnable接口 三、实现Callable接口 四、使用线程池 一、继承 Thread类 创建一个类 Thread 类,并重写run()方法,通过start()启动线程。以继承的方式创建的线程可以使用当前类来获取线程的名称、状态、优先级等相关信…

PXIE板卡,4口QSFP+,PCIE GEN3 X8,XILINX FPGA XCVU3P设计

PXIE板卡,4口QSFP,PCIE GEN3 X8,基于XILINX FPGA XCVU3P设计。 1:电路拓扑 ● 支持利用 EEPROM 存储数据; ● 电源时序控制和总功耗监控; 2:电路调试 3:测试 PCIE gen3 x8&#…

通讯协议学习之路:SPI协议理论

通讯协议之路主要分为两部分,第一部分从理论上面讲解各类协议的通讯原理以及通讯格式,第二部分从具体运用上讲解各类通讯协议的具体应用方法。 后续文章会同时发表在个人博客(jason1016.club)、CSDN;视频会发布在bilibili(UID:399951374) 序、…

同一网段内两台电脑ping不通,诡异问题记录

今天发现了个诡异的问题,和锐捷技术倒腾了半天,记录一下。 症状: 如图(简单画了下),pc1 到 网络打印机 10.64.253.184 ,就是ping不通... 但是在交换机上,在路由器上,在…

企业数据安全组织建设

PS:内容均为个人学习及实践积累所得! 前言&&背景 企业数据安全组织团队的建设非常重要。 首先,数据是企业最重要的资产和核心竞争力。如果数据泄露或遭到破坏,将直接威胁企业的生存。因此,数据安全事关企业的生存和发展。建立专业的数据安全团队,可以系统…

多云系列|10个关键的多云战略:云计算成本的完整可视性

了解和控制你的云计算支出是每个企业都会遇到的一个难题。根据IDC最近的一项研究,IT决策者面临的首要挑战之一是控制云计算成本。对于大多数企业来说,问题在于他们所有的云资源没有完全的可视性。这可能是由于其他部门绕过IT部门为其开发项目进行支出&am…

论坛介绍|COSCon'23 区块链(B)

众多开源爱好者翘首期盼的开源盛会:第八届中国开源年会(COSCon23)将于10月28-29日在四川成都市高新区菁蓉汇举办。本次大会的主题是:“开源:川流不息、山海相映”!各位新老朋友们,欢迎到成都&am…

【论文复现】基于多模态深度学习方法的单细胞多组学数据聚类(【生物信息学】实验二:多组学数据融合:scMDC)

目录 一、实验介绍 1. 论文:基于多模态深度学习方法的单细胞多组学数据聚类 Abstract 2. Github链接 二、实验环境 0. 作者要求 1. 环境复现 实验一 实验二(本实验) 2. 库版本介绍 实验一 实验二 3. IDE 三、实验内容 1. 用法…

Linux高性能服务器编程——ch4笔记

第4章 TCP/IP 通信案例:访问 Internet 上的Web 服务器 4.1 实例总图 4.2 部署代理服务器 客户端和目标服务器之间可能存在多个代理服务器。 正向代理:要求客户端自己设置代理服务器的地址。 反向代理:设置在服务器端。 透明代理&#xff1…