STM32(八):定时器——输入捕获实验

news/2024/9/20 6:57:11/ 标签: stm32, 嵌入式硬件, 单片机

目录

输入捕获模式测频率:

结构图:

步骤: 

部分函数详解:

源码:

PWMI模式测频率占空比:

结构图:

​编辑

举例说明

源码: 


输入捕获模式测频率:

结构图:

    

步骤: 

        1. RCC开启时钟,把GPIO和TIM的时钟打开。

         2. GPIO初始化,把GPIO设置为输入模式,一般选择上拉输入或者浮空输入。

        3. 配置时基单元,让CNT计数器在内部时钟的驱动下自增运行。

        4. 配置输入捕获单元,包括滤波器、极性、直连通道/交叉通道、分频器(结构体配置)。

        5. 选择从模式的触发源 TI1FP1。

        6. 选择之后执行的操作Reset。

        7. 开启定时器,TIM_CMD。

部分函数详解:

void TIM_ICInit(TIM_TypeDef* TIMx, TIM_ICInitTypeDef* TIM_ICInitStruct); 配置输入捕获电路
void TIM_PWMIConfig(TIM_TypeDef* TIMx, TIM_ICInitTypeDef* TIM_ICInitStruct);可以快速配置两个通道。

void TIM_ICStructInit(TIM_ICInitTypeDef* TIM_ICInitStruct);可以给输入结构体赋一个初始值

void TIM_SelectInputTrigger(TIM_TypeDef* TIMx, uint16_t TIM_InputTriggerSource);选择输入触发源TRGI,调用这个函数可以设置从模式的触发源。

void TIM_SelectOutputTrigger(TIM_TypeDef* TIMx, uint16_t TIM_TRGOSource);选择输出触发源TRGO,调用这个函数可以设置主模式的触发源

void TIM_SelectSlaveMode(TIM_TypeDef* TIMx, uint16_t TIM_SlaveMode);选择从模式

void TIM_SetIC1Prescaler(TIM_TypeDef* TIMx, uint16_t TIM_ICPSC);
void TIM_SetIC2Prescaler(TIM_TypeDef* TIMx, uint16_t TIM_ICPSC);
void TIM_SetIC3Prescaler(TIM_TypeDef* TIMx, uint16_t TIM_ICPSC);
void TIM_SetIC4Prescaler(TIM_TypeDef* TIMx, uint16_t TIM_ICPSC);单独配置通道1、2、3、4的分频器;

输出比较模式下,CCR是写的,要用SetCompare写入;输入捕获模式下,CCR是只读的,要用GetCapure读出;

uint16_t TIM_GetCapture1(TIM_TypeDef* TIMx);
uint16_t TIM_GetCapture2(TIM_TypeDef* TIMx);
uint16_t TIM_GetCapture3(TIM_TypeDef* TIMx);
uint16_t TIM_GetCapture4(TIM_TypeDef* TIMx);

源码:

#include "stm32f10x.h"                  // Device headervoid IC_Init(void)
{//第一步开启时钟RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);//第二步配置GPIOGPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPU;//上拉输入GPIO_InitStructure.GPIO_Pin=GPIO_Pin_6;GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;GPIO_Init(GPIOA,&GPIO_InitStructure);//第三步配置时基单元TIM_InternalClockConfig(TIM3);//选择时基单元的时钟TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;//配置采样频率(滤波器),消除抖动;TIM_TimeBaseInitStructure.TIM_CounterMode=TIM_CounterMode_Up;//向上计数TIM_TimeBaseInitStructure.TIM_Period=65536-1;//ARR自动重装器的值TIM_TimeBaseInitStructure.TIM_Prescaler=36-1;//预分频器的值 计数标准频率TIM_TimeBaseInitStructure.TIM_RepetitionCounter=0;//重复计数器的值(高级计数器才需要,目前不需要用给0即可)TIM_TimeBaseInit(TIM3,&TIM_TimeBaseInitStructure);//时基单元配置完毕 更新事件和更新中断是同时发生的,//第四步配置输入捕获函数TIM_ICInitTypeDef TIM_ICInitStruct;TIM_ICInitStruct.TIM_Channel=TIM_Channel_1;//指定配置哪个通道TIM_ICInitStruct.TIM_ICFilter=0xF;TIM_ICInitStruct.TIM_ICPolarity=TIM_ICPolarity_Rising;//边沿检测极性选择,上升、下降沿触发TIM_ICInitStruct.TIM_ICPrescaler=TIM_ICPSC_DIV1;//分频器,每隔几次有效TIM_ICInitStruct.TIM_ICSelection=TIM_ICSelection_DirectTI;TIM_ICInit(TIM3,&TIM_ICInitStruct);//第五步 配置TRGI的触发源为TI1FP1TIM_SelectInputTrigger(TIM3, TIM_TS_TI1FP1);//第六步,配置从模式为ResetTIM_SelectSlaveMode(TIM3,TIM_SlaveMode_Reset);//第七步,开启定时器TIM_Cmd(TIM3,ENABLE);//开启定时器;}uint32_t IC_GetFreq(void){return 2000000/TIM_GetCapture1(TIM3);}

PWMI模式测频率占空比:

结构图:

看一下英文版本的TIM_ICSelection_IndirectTI的说明:

TIM Input 1, 2, 3 or 4 is selected to be connected to IC2, IC1, IC4 or IC3, respectively. 

设置通道2的交叉连接也就是TIM通道1输入和IC2信号连接,也就是说TIM_ICInit(TIM3,&TIM_ICInitStruct); 实际上是对ICx捕获电路进行的配置,而不是对TIM_CHx输入口进行的配置。

TIM_ICSelection_IndirectTI 这个参数是定时器输入捕获模式中的一个配置选项,用于指定输入捕获触发信号源通过复用映射连接到对应通道的输入捕获引脚的方式。具体来说,当选择TIM_ICSelection_IndirectTI时,输入捕获触发信号需要经过复用映射电路,然后再由复用映射电路将信号传递给定时器的输入捕获电路。

举例说明

假设有一个STM32微控制器,它拥有多个定时器输入捕获通道(如IC1、IC2、IC3、IC4)和多个定时器输入通道(如TI1、TI2、TI3、TI4)。这些通道之间通常可以通过直接连接或间接(复用映射)方式相连。

  1. 直接连接(TIM_ICSelection_DirectTI):
    • 假设我们选择TIM_ICSelection_DirectTI,那么输入捕获通道与输入通道之间的连接将是直接的,即IC1连接TI1,IC2连接TI2,以此类推。
    • 例如,如果TIMx_CH1(定时器x的通道1)的输入信号需要被IC1捕获,那么信号将直接通过TI1进入IC1,无需经过任何复用映射电路。
  2. 间接连接(TIM_ICSelection_IndirectTI):
    • 当选择TIM_ICSelection_IndirectTI时,输入捕获通道与输入通道之间的连接将不再是直接的,而是需要通过复用映射电路进行连接。
    • 例如,如果TIMx_CH1的输入信号需要被IC2捕获(而不是通常的IC1),那么信号首先会进入TI1,然后通过复用映射电路被重定向到IC2。这种连接方式允许将不同的输入通道信号映射到非对应的输入捕获通道上,从而增加了配置的灵活性。

源码: 

#include "stm32f10x.h"                  // Device headervoid IC_Init(void)
{//第一步开启时钟RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);//第二步配置GPIOGPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPU;//上拉输入GPIO_InitStructure.GPIO_Pin=GPIO_Pin_6;GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;GPIO_Init(GPIOA,&GPIO_InitStructure);//第三步配置时基单元TIM_InternalClockConfig(TIM3);//选择时基单元的时钟TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;//配置采样频率(滤波器),消除抖动;TIM_TimeBaseInitStructure.TIM_CounterMode=TIM_CounterMode_Up;//向上计数TIM_TimeBaseInitStructure.TIM_Period=65536-1;//ARR自动重装器的值TIM_TimeBaseInitStructure.TIM_Prescaler=72-1;//预分频器的值 计数标准频率TIM_TimeBaseInitStructure.TIM_RepetitionCounter=0;//重复计数器的值(高级计数器才需要,目前不需要用给0即可)TIM_TimeBaseInit(TIM3,&TIM_TimeBaseInitStructure);//时基单元配置完毕 更新事件和更新中断是同时发生的,//第四步配置输入捕获函数TIM_ICInitTypeDef TIM_ICInitStruct;TIM_ICInitStruct.TIM_Channel=TIM_Channel_1;//指定配置哪个通道TIM_ICInitStruct.TIM_ICFilter=0xF;TIM_ICInitStruct.TIM_ICPolarity=TIM_ICPolarity_Rising;//边沿检测极性选择,上升、下降沿触发TIM_ICInitStruct.TIM_ICPrescaler=TIM_ICPSC_DIV1;//分频器,每隔几次有效TIM_ICInitStruct.TIM_ICSelection=TIM_ICSelection_DirectTI;TIM_ICInit(TIM3,&TIM_ICInitStruct);TIM_ICInitStruct.TIM_Channel=TIM_Channel_2;//指定配置哪个通道TIM_ICInitStruct.TIM_ICFilter=0xF;TIM_ICInitStruct.TIM_ICPolarity=TIM_ICPolarity_Falling;//边沿检测极性选择,下降沿触发TIM_ICInitStruct.TIM_ICPrescaler=TIM_ICPSC_DIV1;//分频器,每隔几次有效TIM_ICInitStruct.TIM_ICSelection=TIM_ICSelection_IndirectTI;TIM_ICInit(TIM3,&TIM_ICInitStruct);//TIM_PWMIConfig(TIM3,&TIM_ICInitStruct);//第五步 配置TRGI的触发源为TI1FP1TIM_SelectInputTrigger(TIM3, TIM_TS_TI1FP1);//第六步,配置从模式为ResetTIM_SelectSlaveMode(TIM3,TIM_SlaveMode_Reset);//第七步,开启定时器TIM_Cmd(TIM3,ENABLE);//开启定时器;}uint32_t IC_GetFreq(void){return 1000000/(TIM_GetCapture1(TIM3)+1);}
uint32_t IC_GetDuty(void){return (TIM_GetCapture2(TIM3)+1)*100/(TIM_GetCapture1(TIM3)+1);}

 


http://www.ppmy.cn/news/1518921.html

相关文章

C#中List集合使用Remove方法详解——List使用Remove方法需要注意的坑?

目录 一、基本使用 1、简单类型的例子 2、复杂类型的例子 二、思考 三、深度解析 四、正确的使用方式 1、重写 Equals 和 GetHashCode 2、使用 LINQ 的 FirstOrDefault 方法 五、性能考虑 六、注意事项 总结 在C#中&#xff0c;List<T> 是一个常用的数据结构&…

第四章 Java核心类库 第三节 集合框架

1. 集合框架概述与结构 首先&#xff0c;我们来简单了解一下Java集合框架的概述和结构。 集合框架的定义&#xff1a;Java集合框架是一组用来存储和操作数据集合的接口和类。它提供了一种统一的标准方法来操作不同的数据集合&#xff0c;极大简化了编程任务。 集合框架的结构…

我的电脑/资源管理器里无法显示新硬盘?

前情提要 我新&#xff01;买了一个京东京造的SATA3硬盘&#xff0c;一个绿联的SATA3转USB读取 现在我的电脑里只能显示我本地的C盘和D盘&#xff0c;不能显示这个接入的SATA盘。 系统环境&#xff1a;windows11 问题描述 在我的电脑里&#xff0c;只能看到我原本的C和D&…

互联网平台大模型网络架构设计

字节跳动&#xff1a;大模型网络实践分享 自2019年起&#xff0c;字节跳动公司便开始着手白盒项目。2020年&#xff0c;推出了首款接入交换机——25G型号&#xff0c;随后逐步实现软硬件的自主研发。在当前一代产品中&#xff0c;已经实现了100G接入、25.6T400G互联&#xff0c…

扩展——双向搜索

1. 基本概念 单向搜索&#xff1a;传统的搜索算法&#xff08;如广度优先搜索 BFS、深度优先搜索 DFS&#xff09;通常从起点开始&#xff0c;逐步扩展搜索到目标节点。搜索的时间复杂度与图的大小和结构有关。 双向搜索&#xff1a;双向搜索则同时从起点和终点进行搜索&#…

分享8个Python自动化实战脚本!

1. Python自动化实战脚本 1.1 网络自动化 网络上有丰富的信息资源&#xff0c;Python可以帮我们自动化获取这些信息。 爬虫简介&#xff1a;爬虫是一种自动提取网页信息的程序。Python有许多优秀的爬虫库&#xff0c;如requests和BeautifulSoup。 案例&#xff1a;使用Pytho…

软件测试学习笔记丨静态测试与代码审计 SonarQube

本文转自测试人社区&#xff0c;原文链接&#xff1a;https://ceshiren.com/t/topic/32049 一&#xff0c;SonarQube 平台搭建 1.1&#xff0c; 介绍 Sonar 是一个用于代码质量管理的开放平台。通过插件机制&#xff0c;Sonar 可以集成不同的测试工具、代码分析工具&#xff…

Having trouble using OpenAI API

题意&#xff1a;"使用OpenAI API遇到困难" 问题背景&#xff1a; I am having trouble with this code. I want to implement AI using OpenAI API in my React.js project but I cannot seem to get what the issue is. I ask it a question in the search bar in…

大语言模型数据增强与模型蒸馏解决方案

背景 在人工智能和自然语言处理领域&#xff0c;大语言模型通过训练数百亿甚至上千亿参数&#xff0c;实现了出色的文本生成、翻译、总结等任务。然而&#xff0c;这些模型的训练和推理过程需要大量的计算资源&#xff0c;使得它们的实际开发应用成本非常高&#xff1b;其次&a…

线程间数据传递之ThreadLocal、InheritableThreadLocal、TransmittableThreadLocal

线程间数据传递之ThreadLocal、InheritableThreadLocal、TransmittableThreadLocal 1、ThreadLocal介绍 spring 中基于 ThreadLocal 来实现事务。 多线程 访问同一个共享变量的时候容易出现并发问题&#xff0c;ThreadLocal是除了加锁这种同步方式之外的一种保证 规避多线程…

【bug记录7】导入Lottie的json格式动画,获取不到相对路径下的图片

一、问题背景 在vue3项目中&#xff0c;想把Lottie依赖的图片放在其相对路径下&#xff0c;但是发现即使修改其中的u参数&#xff0c;也无法拿到其相对路径中的图片。因为json解析绝对路径&#xff0c;只能将图片放在项目根目录下的public文件夹应急。 二、解决方法 将Lottie…

Lua 代码编码规范

lua代码格式 vscode stylua 插件 配置文件stylua.toml column_width 240 line_endings “Unix” indent_type “Spaces” --使用空格 很重要&#xff0c;保证不同编辑器打开是一样的 indent_width 4 quote_style “AutoPreferDouble” --字符串引号样式双引号 call_paren…

c++关于字符串的练习

提示并输入一个字符串&#xff0c;统计该字符串中字母个数、数字个数、空格个数、其他字符的个数 #include <iostream> #include<string> using namespace std;int main() {string s1;int letter0,digit0,space0,other0;cout<<"请输入一个字符串:"…

ISO C++ 和 GNU C++ 的区别

C 的 ios 标准和 gnu 标准是两种编译器标准或模式&#xff0c;主要由编译器在编译 C 代码时所遵循的规范决定。它们之间的区别主要在于是否包含标准之外的扩展以及对特定功能的支持。 1. ISO C 标准 (-stdc11, -stdc14, -stdc17, 等) 定义: ISO C 标准是由国际标准化组织 (IS…

3D打印透气钢与传统透气钢的差异

透气钢作为一种集金属强度与透气性能于一体的特殊材料&#xff0c;在注塑模具领域扮演着关键角色&#xff0c;通过有效排除模具内困气&#xff0c;显著提升制品成型质量与生产效率。当前&#xff0c;市场上主流的透气钢产品多源自日本、美国&#xff0c;其高昂成本与技术壁垒限…

Java算法之选择排序(Selection Sort)

简介 选择排序是一种简单直观的排序算法&#xff0c;它的工作原理是每次从待排序的数据元素中选出最小&#xff08;或最大&#xff09;的一个元素&#xff0c;然后放到序列的起始位置。通过重复这个过程&#xff0c;直到所有元素都被排序。 算法步骤 在未排序序列中找到最小…

【Vue】Echart图表中的属性

目录 背景属性介绍1. title2. tooltip3. legend4. toolbox5. color6. grid7. xAxis / yAxis8. series9. visualMap10. dataZoom 示例 背景 最近Echart用的比较多&#xff0c;改动的展示效果其实也就那么些&#xff0c;而且很多案例、展示效果在Echart官网写的都很丰富&#xf…

数字化转型中的数据应用:挑战、机遇与追赶之路

在数字化时代的大潮中&#xff0c;数据已悄然从企业的边缘资源跃升为最宝贵的核心资产。然而&#xff0c;这场数据盛宴并未带来普遍的数据应用成熟&#xff0c;反而揭示了企业在数据利用上的巨大鸿沟。即便是全球500强企业&#xff0c;在数据应用的征途上&#xff0c;也仅仅是比…

android 14及android15 READ_EXTERNAL_STORAGE跟相册,视频权限的适配

最近在做Android15的适配&#xff0c;发现WRITE_EXTERNAL_STORAGE跟READ_EXTERNAL_STORAGE无法使用了&#xff0c;被弃用了 在android 13添加了外部细分权限&#xff0c;READ_MEDIA_IMAGES跟READ_MEDIA_VIDEO及 READ_MEDIA_AUDIO权限&#xff0c;而在应用内部的文件管理则不需要…

TCP与UDP对比

这两个都是运输层的协议&#xff0c;UDP是无连接不可靠的&#xff0c;而TCP是面向连接可靠的&#xff0c;相较而言&#xff0c;UDP要简单许多。两者对比做一个简要概述。 连接方式 1.UDP是无连接的&#xff0c;就是通信双方无需建立连接就可以随时发送数据。 2.而TCP在发送数…