一、概念理解

PWM输入捕获模式是输入捕获模式的特例,自己理解如下

  1. 每个定时器有四个输入捕获通道IC1、IC2、IC3、IC4。且IC1 IC2一组,IC3 IC4一组。并且可是设置管脚和寄存器的对应关系。

  2. 同一个TIx输入映射了两个ICx信号。

  3. 这两个ICx信号分别在相反的极性边沿有效。

  4. 两个边沿信号中的一个被选为触发信号,并且从模式控制器被设置成复位模式。

  5. 当触发信号来临时,被设置成触发输入信号的捕获寄存器,捕获“一个PWM周期(即连续的两个上升沿或下降沿)”,它等于包含TIM

    时钟周期的个数(即捕获寄存器中捕获的为TIM的计数个数n)。

  6. 同样另一个捕获通道捕获触发信号和下一个相反极性的边沿信号的计数个数m,即(即高电平的周期或低电平的周期)

  7. 由此可以计算出PWM的时钟周期和占空比了

    frequency=f(TIM时钟频率)/n。

    duty cycle=(高电平计数个数/n),

    若m为高电平计数个数,则duty cycle=m/n

    若m为高电平计数个数,则duty cycle=(n-m)/n

    注:因为计数器为16位,所以一个周期最多技术65535个,所以测得的 最小频率= TIM时钟频率/65535。

    二、程序设计与分析

  8. 程序概述:选择TIM3作为PWM输入捕获。IC2设置为上升沿,并设置为有效的触发输入信号。所以IC2的捕获寄存器捕获PWM周期,

    IC1的捕获寄存器捕获PWM的高电平周期。

    2.程序代码如下:

    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); //时钟配置

    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7; //GPIO配置

    PIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;

    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

    GPIO_Init(GPIOA, &GPIO_InitStructure);

    NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn; //NVIC配置

    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_2; //通道选择

    TIM_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Rising; //上升沿触发

    TIM_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI; //管脚与寄存器对应关系

    TIM_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1; //输入预分频。意思是控制在多少个输入周期做一次捕获,如果

    //输入的信号频率没有变,测得的周期也不会变。比如选择4分频,则每四个输入周期才做一次捕获,这样在输入信号变化不频繁的情况下,

    //可以减少软件被不断中断的次数。

    TIM_ICInitStructure.TIM_ICFilter = 0x0; //滤波设置,经历几个周期跳变认定波形稳定0x0~0xF
    TIM_PWMIConfig(TIM3, &TIM_ICInitStructure); //根据参数配置TIM外设信息

    TIM_SelectInputTrigger(TIM3, TIM_TS_TI2FP2); //选择IC2为始终触发源

    TIM_SelectSlaveMode(TIM3, TI***aveMode_Reset);//TIM从模式:触发信号的上升沿重新初始化计数器和触发寄存器的更新事件

    TIM_SelectMasterSlaveMode(TIM3, TIM_MasterSlaveMode_Enable); //启动定时器的被动触发

    TIM_Cmd(TIM3, ENABLE); //启动TIM2

    TIM_ITConfig(TIM3, TIM_IT_CC2, ENABLE); //打开中断
    中断处理函数
    void TIM3_IRQHandler(void)
    {
    TIM_ClearITPendingBit(TIM3, TIM_IT_CC2); //清楚TIM的中断待处理位

    IC2Value = TIM_GetCapture2(TIM3); //读取IC2捕获寄存器的值,即为PWM周期的计数值
    if (IC2Value != 0)
    {
    DutyCycle = (TIM_GetCapture1(TIM3) * 100) / IC2Value; //读取IC1捕获寄存器的值,并计算占空比

    Frequency = 72000000 / IC2Value; //计算PWM频率。
    }
    else
    {
    DutyCycle = 0;
    Frequency = 0;
    }
    }

注(一):若想改变测量的PWM频率范围,可将TIM时钟频率做分频处理

TIM_TimeBaseStructure.TIM_Period = 0xFFFF; //周期0~FFFF

TIM_TimeBaseStructure.TIM_Prescaler = 5; //时钟分频,分频数为5+1即6分频

TIM_TimeBaseStructure.TIM_ClockDivision = 0; //时钟分割

TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;//模式

TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);//基本初始化

注注(二):定时器TIM的倍频器X1或X2。在APB分频为1时,倍频值为1,否则为2。

用的输入捕获功能

IO配置

GPIO_InitTypeDef GPIO_InitStructure; //

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7;

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;

GPIO_Init(GPIOB, &GPIO_InitStructure);

GPIO_ResetBits(GPIOB,GPIO_Pin_7);

中断配置

void NVIC_Configuration_Cap(void)

{

NVIC_InitTypeDef NVIC_InitStructure;

#ifdef VECT_TAB_RAM

NVIC_SetVectorTable(NVIC_VectTab_RAM, 0x0);
#else

NVIC_SetVectorTable(NVIC_VectTab_FLASH, 0x0);
#endif
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0); //占先优先级、副优先级的资源分配

NVIC_InitStructure.NVIC_IRQChannel = TIM4_IRQn; //指定中断源
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; //占先优先级设定
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; //副优先级设定
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);
}
时钟配置
void TIM4_Cap_Init(u16 arr, u16 psc)
{
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
TIM_ICInitTypeDef TIM_ICInitStructure;
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM4 , ENABLE);

TIM_DeInit(TIM4); //初始化TIM4为缺省值 0

TIM_TimeBaseStructure.TIM_Period = arr; //arr=10000
TIM_TimeBaseStructure.TIM_Prescaler = psc; //psc=71
TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //向上计数
TIM_TimeBaseInit(TIM4, &TIM_TimeBaseStructure);

TIM_ICInitStructure.TIM_Channel = TIM_Channel_2;
TIM_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Falling; //上升沿捕获
TIM_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI;
TIM_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1;
TIM_ICInitStructure.TIM_ICFilter = 0x0;
TIM_ICInit(TIM4, &TIM_ICInitStructure);
//TIM_SelectInputTrigger(TIM4, TIM_TS_TI2FP2 ); //选择时钟触发源,这里只能用1或2,意味着通道34不能用输入触发
//TIM_SelectSlaveMode(TIM4, TIM_SlaveMode_Reset);//触发方式
// TIM_SelectMasterSlaveMode(TIM4, TIM_MasterSlaveMode_Enable); //启动定时器的被动触发
TIM_Cmd(TIM4, ENABLE);
TIM_ITConfig(TIM4, TIM_IT_CC2, ENABLE); //允许捕获中断
TIM_ITConfig(TIM4,TIM_IT_Update, ENABLE); //允许更新中断

TIM_ClearITPendingBit(TIM4, TIM_IT_Update ); //清除更新中断标记
TIM_ClearITPendingBit(TIM4, TIM_IT_CC2); //清除捕获中断标记
}

中断服务函数

u16 STA=0;
u16 VAL;
u8 CAP_N=0;
void TIM4_IRQHandler(void)
{
if((STA&0x8000)==0) //捕获开启标志
{
if(TIM_GetITStatus(TIM4,TIM_IT_Update)!=RESET) //进入更新中断
{
if(CAP_N==1)
{
if((STA&0X3FFF)==0X3FFF)//高电平太长了
{STA|=0X8000;//标记成功捕获了一次
VAL=0XFFFF;
}
else STA++;
}
}
if(TIM_GetITStatus(TIM4,TIM_IT_CC2)!=RESET) //进入捕获中断
{
if(CAP_N==1)
{STA|=0x8000;//标志完成捕获过程,进入主函数处理部分}
VAL=TIM_GetCapture2(TIM4);
}
else
{STA=0;
VAL=0;
TIM_SetCounter(TIM4,0);
CAP_N=1;
}
}
}
TIM_ClearITPendingBit(TIM4, TIM_IT_Update|TIM_IT_CC2);
}

数据处理主函数

void TIM_CAP(void)
{
u32 CAP_temp=0;
NVIC_Configuration_Cap();
TIM4_Cap_Init(9999,71);

while(1)
{
delay_ms(10);
if(STA&0x8000)
{
CAP_temp=STA&0x3fff;
printf("\r\nSTA=%d",CAP_temp);
CAP_temp*=10000;
CAP_temp+=VAL;
//printf("\r\nVAL=%d",VAL);
printf("\r\n周期:%d us",CAP_temp);

CAP_N=0;
STA=0;
}
}
}
PWM输入功能

void Confiuration_PWM_CAP(void)
{
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
TIM_ICInitTypeDef TIM_ICInitStructure;
GPIO_InitTypeDef GPIO_InitStructure;
NVIC_InitTypeDef NVIC_InitStructure;

RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); //时钟配置
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7; //GPIO配置
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStructure);
NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn; //NVIC配置
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);

TIM_TimeBaseStructure.TIM_Period = 0xFFFF; //周期0~FFFF
TIM_TimeBaseStructure.TIM_Prescaler = 0; //时钟分频,分频数为5+1即6分频
TIM_TimeBaseStructure.TIM_ClockDivision = 0; //时钟分割
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;//模式
TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);//基本初始化

TIM_ICInitStructure.TIM_Channel = TIM_Channel_2; //通道选择
TIM_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Rising; //上升沿触发
TIM_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI; //管脚与寄存器对应关系
TIM_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1; //输入预分频。意思是控制在多少个输入周期做一次捕获,如果
//输入的信号频率没有变,测得的周期也不会变。比如选择4分频,则每四个输入周期才做一次捕获,这样在输入信号变化不频繁的情况下,
//可以减少软件被不断中断的次数。
TIM_ICInitStructure.TIM_ICFilter = 0x0; //滤波设置,经历几个周期跳变认定波形稳定0x0~0xF
TIM_PWMIConfig(TIM3, &TIM_ICInitStructure); //根据参数配置TIM外设信息

TIM_SelectInputTrigger(TIM3, TIM_TS_TI2FP2); //选择IC2为始终触发源
TIM_SelectSlaveMode(TIM3, TIM_SlaveMode_Reset);//TIM从模式:触发信号的上升沿重新初始化计数器和触发寄存器的更新事件
TIM_SelectMasterSlaveMode(TIM3, TIM_MasterSlaveMode_Enable); //启动定时器的被动触发

TIM_Cmd(TIM3, ENABLE); //启动TIM2
TIM_ITConfig(TIM3, TIM_IT_CC2, ENABLE); //打开中断

}

void PWM_CAP(void)
{
Confiuration_PWM_CAP();
while(1)
{
printf("\r\nFrequency=%d",Frequency);
printf("\r\nDutyCycle=%f",DutyCycle);
}

}

中断服务函数

float IC2Value;
float DutyCycle;
u16 Frequency;
void TIM3_IRQHandler(void)
{

TIM_ClearITPendingBit(TIM3, TIM_IT_CC2); //清除TIM的中断待处理位

IC2Value = TIM_GetCapture2(TIM3); //读取IC2捕获寄存器的值,即为PWM周期的计数值
if (IC2Value != 0)
{

DutyCycle = (TIM_GetCapture1(TIM3) * 100) / IC2Value; //读取IC1捕获寄存器的值,并计算占空比
Frequency = 72000000 / IC2Value; //计算PWM频率。
}
else
{
DutyCycle = 0;
Frequency = 0;
}
}

stm32PWM输入捕获模式详解相关推荐

  1. Spotify敏捷模式详解三部曲第二篇:研发过程

    本文转自:Scrum 中文网 引言 在本系列文章的第一篇,我们介绍了Spotify的敏捷研发团队,以及它独特的组织架构.Spotify的研发团队采用的是一种非常独特的组织架构,如下图所示: 整个研发组 ...

  2. Spotify敏捷模式详解三部曲第一篇:研发团队

    本文转自:Scrum中文网 引言 2018年4月,来自北欧瑞典的音乐流媒体公司.百亿美元独角兽Spotify创造了历史,它成为了当代上市公司当中,第一家通过"直接上市"的方式在美国 ...

  3. 23种设计模式之简单工厂模式,工厂方法模式,抽象工厂模式详解

    工厂模式详解 1. 简单工厂模式 1.1 需求分析 1.2 使用传统方式实现 1.2.1 类图 1.2.2 代码实现 1.2.2.1 新建pizza抽象类 1.2.2.2 希腊披萨实现类 1.2.2. ...

  4. 敏捷开发系列学习总结(14)——Spotify敏捷模式详解三部曲第二篇:研发过程

    分享一个大神的人工智能教程.零基础!通俗易懂!风趣幽默!还带黄段子!希望你也加入到人工智能的队伍中来!点击浏览教程 摘要 在本系列文章的第一篇,我们介绍了Spotify的敏捷研发团队,以及它独特的组织 ...

  5. 敏捷开发系列学习总结(13)——Spotify敏捷模式详解三部曲第一篇:研发团队

    分享一个大神的人工智能教程.零基础!通俗易懂!风趣幽默!还带黄段子!希望你也加入到人工智能的队伍中来!点击浏览教程 引言 2018年4月,来自北欧瑞典的音乐流媒体公司.百亿美元独角兽Spotify创造 ...

  6. Solr系列二:solr-部署详解(solr两种部署模式介绍、独立服务器模式详解、SolrCloud分布式集群模式详解)...

    一.solr两种部署模式介绍 Standalone Server 独立服务器模式:适用于数据规模不大的场景 SolrCloud  分布式集群模式:适用于数据规模大,高可靠.高可用.高并发的场景 二.独 ...

  7. IoT:电子密本ECB和DES模式详解

    1.ECB模式详解 转自:https://blog.csdn.net/sunqiujing/article/details/75066924 DES参考这篇博文:写的好 https://blog.cs ...

  8. STM32:GPIO的8种输入输出模式深入详解

    STM32的8种GPIO输入输出模式深入详解 输入模式 -输入浮空(GPIO_Mode_IN_FLOATING) -输入上拉(GPIO_Mode_IPU) -输入下拉(GPIO_Mode_IPD) - ...

  9. IoC与DI工厂、单例、原型模式详解

    1.工厂模式 1.1 工厂模式的由来 在现实生活中我们都知道 原始社会自给自足(没有工厂) 农耕社会有了小作坊(简单工厂,如民间酒坊) 工业革命后有了流水线(工厂方法,自产自销) 现代产业链中有代工厂 ...

最新文章

  1. F - Weakness and Poorness CodeForces - 578C
  2. 反思PHP多个字符串函数
  3. 【渝粤教育】 国家开放大学2020年春季 1080工程数学(本) 参考试题
  4. 用c语言程序算自己的年龄,新手编的小程序:计算年龄和出生后经历的天数的小程序...
  5. 微软 Edge bug 导致黑客窃取用户在任意站点的机密信息,颁发2万美元奖金
  6. Python判断文件是否存在、访问
  7. 坯子库曲面推拉教程_SU曲面建模太简单?还是网友技术太强大...
  8. python mock server_python学习笔记6--mockserver
  9. IIS 7 为 URL Rewrite 模块创建重写规则
  10. 师傅带徒弟学:Python游戏开发引擎cocos2d-python-关东升-专题视频课程
  11. 中国科学院大学毕业典礼致辞全文
  12. 新浪微博Android客户端开发之OAuth认证篇
  13. 机柜系统:数据和业务的幕后英雄
  14. 在html页面中引入jquery
  15. android11最新版本是多少,Android11正式发布 安卓11正式版系统发布更新时间
  16. 信息安全数学基础——模重复平方计算法(两种方法实现C+JAVA)
  17. 有什么软件可以裁剪html文件,有什么软件可以裁剪视频?要手机版的
  18. 超级计算机国产cpu,中国的天河一号超级计算机用的是国产CPU还是尽...-卓优商学问答...
  19. 电信4G添加ctlte接入点提高网速
  20. mysql 货币显示_MySQL查询设置货币记录

热门文章

  1. Codeforces Round #222 (Div. 2): C. Maze(BFS)
  2. 关系代数操作应用、关系元组运算应用
  3. bzoj 3385: [Usaco2004 Nov]Lake Counting 数池塘(DFS)
  4. 荣耀10 原版android,荣耀V10新内置壁纸
  5. Java设计模式应用到数据库_Java设计模式在数据库编程中的应用研究
  6. css代码颜色十六进制大全
  7. XCode6 ,iOS之PCH文件配置
  8. python之函数用法capitalize()
  9. 九度OJ 1193:矩阵转置 (矩阵计算)
  10. WPF采用MVVM模式(绑定:纯前台、命令:触发器绑定命令)