stm32 定时,互补输出,输入捕获,输出比较

本文根据野火电机写的个人笔记

Prescaler:定时器预分频器设置,时钟源经该预分频器才是定时器时钟,它设定 TIMx_PSC
寄存器的值。可设置范围为 0 至 65535,实现 1 至 65536 分频。

• CounterMode:定时器计数方式,可是在为向上计数、向下计数以及三种中心对齐模式。基
本定时器只能是向上计数,即 TIMx_CNT 只能从 0 开始递增,并且无需初始化。

• Period:定时器周期,实际就是设定自动重载寄存器的值,在事件生成时更新到影子寄存
器。可设置范围为 0 至 65535。 • ClockDivision:时钟分频,设置定时器时钟 CK_INT 频率与数字滤波器采样时钟频率分频
比,基本定时器没有此功能,不用设置。

ClockDivision:时钟分频,设置定时器时钟 CK_INT 频率与数字滤波器采样时钟频率分频
比,基本定时器没有此功能,不用设置

• RepetitionCounter:重复计数器,属于高级控制寄存器专用寄存器位,利用它可以非常容
易控制输出 PWM 的个数。这里不用设置。

led 0.5翻转电平信号

TIMxCLK=HCLK/2=SystemCoreClock/2=84MHz

Prescaler = 8400-1

设定定时器频率为 =TIMxCLK/(TIM_Prescaler+1)=10000Hz

Period = 5000-1

即1/10000 = 0.0001s

0.0001s * 5000 = 0.5s 中断0.5s触发一次
计数器CNT数到5000与Period比较,相同产生中断

高级控制定时器 (TIM1 和 TIM8) 和通用定时器在基本定时器的基础上引入了外部引脚,可以
输入捕获和输出比较功能。高级控制定时器比通用定时器增加了可编程死区互补输出、重复计数
器、带刹车 (断路) 功能。

互补输出

定时器的PWM互补输出,
用在驱动器、逆变器方面。主要是用在全桥MOS管的控制上。
如下图,如果MOS管相同,PWM不反向的话,势必造成两个MOS都导通。相当于电源和地都直连
在这里插入图片描述
另外,这里还涉及到一个概念,死区时间。
由于两个MOS管的参数不一定完全相同,关断速度不一样,可能有那么几个us 或者MS 是上一个管马上要关闭,下一个已经打开了,造成了两个MOS都打开的短暂现象,所以一般需要加个死区时间。

TIM_OCInitStructure.OCPolarity = TIM_OCPOLARITY_HIGH; //正常通道级性

TIM_OCInitStructure.OCNPolarity = TIM_OCNPOLARITY_HIGH; //互补通道级性

TIM_OCInitStructure.OCIdleState = TIM_OCIDLESTATE_SET; //空闲状态下

TIM_OCInitStructure.OCNIdleState = TIM_OCNIDLESTATE_RESET; //互补空闲状态下
在这里插入图片描述

/* 定时器通道 1 输出 PWM /
HAL_TIM_PWM_Start(&TIM_TimeBaseStructure,TIM_CHANNEL_1);
/
定时器通道 1 互补输出 PWM */
HAL_TIMEx_PWMN_Start(&TIM_TimeBaseStructure,TIM_CHANNEL_1);

死区时间计算

TIM_BDTRInitStructure.DeadTime = 0xff;
TIM_BDTRInitStructure.BreakState = TIM_BREAK_ENABLE; //使能断路

在这里插入图片描述
0xff = 1111 1111
DTG[4:0] = 后5位 5个1 = 31
Tdtg = 16* tDTS
tDTS是通过分频器得到的

在这里插入图片描述在这里插入图片描述

因为 TIM_TimeBaseStructure.Init.ClockDivision=TIM_CLOCKDIVISION_DIV1; 我们不分频
所以是 00:tDTS = tCK_INT 也就168Mhz 也就是168Mhz/1

0xff;即DTG[7:5]=111 => DT = (32+DTG[4:0]) * Tdtg

即DT = (168000000/1 * 16) * (32 + 31) = 0.000006s

0.000006s * 1000 = 6ms
大概死区时间为6ms

输入捕获

pwm输入捕获模式

PWM 频率
Period = 10000-1;
Prescaler = 84-1;
TIMxCLK/(TIM_Prescaler+1)=84000000/84=1000000
pwm频率=1000000/Period = 100hz
100/1 = 10ms ,周期为 10ms

占空比为:(Pulse+1)/(Period+1) = 50%。

PWM 输入模式,需要使用两个捕获通道,占用两个捕获寄存器。由输入通道 TI1 输
入的信号会分成 TI1FP1 和 TI1FP2****,具体选择哪一路信号作为捕获触发信号决定着哪个捕获通
道测量的是周期
。这里我们选择 TI1FP1 作为捕获的触发信号,那 PWM 信号的周期则存储在
CCR1 寄存器中,剩下的另外一路信号 TI1FP2 则进入 IC2,CCR2 寄存器存储的是脉冲宽度

测量脉冲宽度我们选择捕获通道 2,即 IC2,设置捕获信号的极性,这里我们配置为下降沿,我们
需要对捕获信号的每个有效边沿(即我们设置的下降沿)都捕获

IC2 作为间接输入模式,我们需要配置他的从模式,即从模式复位模式,定时器触发源为
TIM_TS_TI1FP1,最后使用函数 HAL_TIM_SlaveConfigSynchronization 进行配置。
在这里插入图片描述在这里插入图片描述

![在这里插入图片描述](https://img-blog.csdnimg.cn/2021012013501054.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3dlaXhpbl80Njk0MjQxNw==,size_16,color_FFFFFF,t_70#pic_center

HAL_TIM_IC_Init(&TIM_PWMINPUT_Handle);
/* IC1 捕获:上升沿触发 TI1FP1 /
TIM_ICInitStructure.ICPolarity = TIM_ICPOLARITY_RISING;
TIM_ICInitStructure.ICSelection = TIM_ICSELECTION_DIRECTTI;
TIM_ICInitStructure.ICPrescaler = TIM_ICPSC_DIV1;
TIM_ICInitStructure.ICFilter = 0x0;
HAL_TIM_IC_ConfigChannel(&TIM_PWMINPUT_Handle,
&TIM_ICInitStructure,ADVANCE_IC1PWM_CHANNEL);
/
IC2 捕获:下降沿触发 TI1FP2 */
TIM_ICInitStructure.ICPolarity = TIM_ICPOLARITY_FALLING;
TIM_ICInitStructure.ICSelection = TIM_ICSELECTION_INDIRECTTI;
TIM_ICInitStructure.ICPrescaler = TIM_ICPSC_DIV1;
TIM_ICInitStructure.ICFilter = 0x0;

HAL_TIM_IC_ConfigChannel(&TIM_PWMINPUT_Handle,
&TIM_ICInitStructure,ADVANCE_IC2PWM_CHANNEL);
/* 选择从模式: 复位模式 /
TIM_SlaveConfigStructure.SlaveMode = TIM_SLAVEMODE_RESET;
/
选择定时器输入触发: TI1FP1 /
TIM_SlaveConfigStructure.InputTrigger = TIM_TS_TI1FP1;
HAL_TIM_SlaveConfigSynchronization(&TIM_PWMINPUT_Handle,
&TIM_SlaveConfigStructure);
/
使能捕获/比较 2 中断请求 */
HAL_TIM_IC_Start_IT(&TIM_PWMINPUT_Handle,TIM_CHANNEL_1);
HAL_TIM_IC_Start_IT(&TIM_PWMINPUT_Handle,TIM_CHANNEL_2);

  • 获取输入捕获值 */
    6 IC1Value = HAL_TIM_ReadCapturedValue(&TIM_PWMINPUT_Handle,ADVANCE_
    ,→IC1PWM_CHANNEL);
    7 IC2Value = HAL_TIM_ReadCapturedValue(&TIM_PWMINPUT_Handle,ADVANCE_
    ,→IC2PWM_CHANNEL);

我们获取 CCR1 和 CCR2 寄存器中的值,当 CCR1 的值不为 0 时,
说明有效捕获到了一个周期,然后计算出频率和占空比。
如果是第一个上升沿中断,计数器会被复位,锁存到 CCR1 寄存器的值是 0,CCR2 寄存器的值
也是 0,无法计算频率和占空比。当第二次上升沿到来的时候,CCR1 和 CCR2 捕获到的才是有
效的值

/* 占空比计算 /
11 DutyCycle = (float)((IC2Value+1) * 100) / (IC1Value+1);
12
13 /
频率计算 */
14 Frequency = 168000000/168/(float)(IC1Value+1)

重要函数
HAL_TIM_IC_Init(&TIM_PWMINPUT_Handle);

HAL_TIM_IC_ConfigChannel(&TIM_PWMINPUT_Handle,
&TIM_ICInitStructure,ADVANCE_IC(1/2)PWM_CHANNEL);

HAL_TIM_SlaveConfigSynchronization(&TIM_PWMINPUT_Handle,
&TIM_SlaveConfigStructure);

HAL_TIM_IC_Start_IT(&TIM_PWMINPUT_Handle,TIM_CHANNEL_(1/2));

多通道输出比较
在这里插入图片描述

一开始我设置输出高电平,设置ARR为最大oxff。第一次时设置CCR为50000,当到了50000我进入中断使状态翻转并且使CCR的值加30000,因为寄存器是16位数据肯定溢出。最后CCR的数值是50000+30000-65536=14464。那么当计数器计数到了14464时,我又进入中断,使状态翻转并且将CCR的值加50000。。。那么它的频率就是1/计数80000的时间,占空比为50000/80000。

  • PWM 模式配置–这里配置为输出比较模式 /
    TIM_OCInitStructure.OCMode = TIM_OCMODE_TOGGLE;
    /
    比较输出的计数值 /
    TIM_OCInitStructure.Pulse = OC_Pulse_num_Channel1;
    /
    当定时器计数值小于 CCR1_Val 时为高电平 /
    TIM_OCInitStructure.OCPolarity = TIM_OCPOLARITY_HIGH;
    /
    设置互补通道输出的极性 /
    TIM_OCInitStructure.OCNPolarity = TIM_OCNPOLARITY_LOW;
    /
    快速模式设置 /
    TIM_OCInitStructure.OCFastMode = TIM_OCFAST_DISABLE;
    /
    空闲电平 /
    TIM_OCInitStructure.OCIdleState = TIM_OCIDLESTATE_RESET;
    /
    互补通道设置 */
    TIM_OCInitStructure.OCNIdleState = TIM_OCNIDLESTATE_RESET;
    HAL_TIM_OC_ConfigChannel(&TIM_AdvanceHandle, &TIM_OCInitStructure, ADVANCE_
    ,→TIM_CHANNEL_1);

/* 启动比较输出并使能中断 */
HAL_TIM_OC_Start_IT(&TIM_AdvanceHandle, ADVANCE_TIM_CHANNEL_1);

/* 使能比较通道 1 */
TIM_CCxChannelCmd(ADVANCE_TIM, ADVANCE_TIM_CHANNEL_1, TIM_CCx_ENABLE);

怎么理解呢
假设计数值为100,到125反转一次到150又反转了一次,即形成了pwm,那怎么做?在中断回调设置新的比较值即可

  • 4
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值