江协科技STM32学习- P18 实验-PWM输入捕获测频率&PWMI输入捕获模式测频率和占空比

     🚀write in front🚀  
🔎大家好,我是黄桃罐头,希望你看完之后,能对你有所帮助,不足请指正!共同学习交流
🎁欢迎各位→点赞👍 + 收藏⭐️ + 留言📝​ 

💬本系列哔哩哔哩江科大STM32的视频为主以及自己的总结梳理📚 

🚀Projeet source code🚀   

💾工程代码放在了本人的Gitee仓库:iPickCan (iPickCan) - Gitee.com

引用:

STM32入门教程-2023版 细致讲解 中文字幕_哔哩哔哩_bilibili

Keil5 MDK版 下载与安装教程(STM32单片机编程软件)_mdk528-CSDN博客

STM32之Keil5 MDK的安装与下载_keil5下载程序到单片机stm32-CSDN博客

0. 江协科技/江科大-STM32入门教程-各章节详细笔记-查阅传送门-STM32标准库开发_江协科技stm32笔记-CSDN博客

【STM32】江科大STM32学习笔记汇总(已完结)_stm32江科大笔记-CSDN博客

江科大STM32学习笔记(上)_stm32博客-CSDN博客

STM32学习笔记一(基于标准库学习)_电平输出推免-CSDN博客

STM32 MCU学习资源-CSDN博客

术语:

英文缩写描述
GPIO:General Purpose Input Onuput通用输入输出
AFIO:Alternate Function Input Output复用输入输出
AO:Analog Output模拟输出
DO:Digital Output数字输出
内部时钟源 CK_INT:Clock Internal内部时钟源
外部时钟源 ETR:External clock 时钟源 External clock 
外部时钟源 ETR:External clock mode 1外部时钟源 Extern Input pin 时钟模式1
外部时钟源 ETR:External clock mode 2外部时钟源 Extern Trigger 时钟模式2
外部时钟源 ITRx:Internal trigger inputs外部时钟源,ITRx (Internal trigger inputs)内部触发输入
外部时钟源 TIx:external input pin 外部时钟源 TIx (external input pin)外部输入引脚
CCR:Capture/Comapre Register捕获/比较寄存器
OC:Output Compare输出比较
IC:Input Capture输入捕获
TI1FP1:TI1 Filter Polarity 1Extern Input 1 Filter Polarity 1,外部输入1滤波极性1
TI1FP2:TI1 Filter Polarity 2Extern Input 1 Filter Polarity 2,外部输入1滤波极性2

正文:

0. 概述

从 2024/06/12 定下计划开始学习下江协科技STM32课程,接下来将会按照哔站上江协科技STM32的教学视频来学习入门STM32 开发,本文是视频教程 P2 STM32简介一讲的笔记。


定时器共四个部分,分为八个小节笔记。本小节为第一部分第一节。

🌳在第一部分,是定时器的基本定时的功能:定时中断功能、内外时钟源选择

🌳在第二部分,是定时器的输出比较功能,最常见的用途是产生PWM波形,用于驱动电机等设备

🌳在第三部分,是定时器的输入捕获功能和主从触发模式,来实现测量方波频率

🌳在第四部分,是定时器的编码器接口,能够更加方便读取正交编码器的输出波形,编码电机测速


1. 🚢输入捕获

定时器输入捕获相关的 stm32f10x_tim.h 中的函数原型:

函数原型功能
void TIM_ICInit(TIM_TypeDef* TIMx, TIM_ICInitTypeDef* TIM_ICInitStruct);TIM定时器输入捕获初始化
void TIM_ICStructInit(TIM_ICInitTypeDef* TIM_ICInitStruct);TIM定时器输入捕获结构体初始化为默认值
void TIM_PWMIConfig(TIM_TypeDef* TIMx, TIM_ICInitTypeDef* TIM_ICInitStruct);这个函数和TIM_ICInit类似都是用于初始化输入捕获单元的,但是TIM_ICInit函数只是单一地配置一个通道而TIM_PWMIConfig可以快速配置两个通道的PWMI模式(自动将另一个通道配置为相反的模式)
void TIM_PrescalerConfig(TIM_TypeDef* TIMx, uint16_t Prescaler, uint16_t TIM_PSCReloadMode);TIM定时器预分频器设置
void TIM_SelectInputTrigger(TIM_TypeDef* TIMx, uint16_t TIM_InputTriggerSource);选择输入(从模式)触发源TRGI
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);
TIM定时器输入捕获通道IC1,IC2,IC3,IC4预分频器配置
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);
uint16_t TIM_GetCounter(TIM_TypeDef* TIMx);
TIM定时器获取输出捕获值CCR
void TIM_SelectOutputTrigger(TIM_TypeDef* TIMx, uint16_t TIM_TRGOSource);选择输出(主模式)触发源TRGO,选择主模式触发的触发源
void TIM_SelectSlaveMode(TIM_TypeDef* TIMx, uint16_t TIM_SlaveMode);选择从模式需要执行的操作
void TIM_SelectMasterSlaveMode(TIM_TypeDef* TIMx, uint16_t TIM_MasterSlaveMode);TIM定时器主从模式,定时器级联

2. 🚢实验1-输入捕获测频率

 两个代码的接线图都一样,如下

 测量信号的输入引脚是PA6,信号从PA6进来,待测的PWM信号也是STM32自己生成的,输出引脚是PA0。

需要配置电路连接图示如下:

 

  • 🌾第一步,RCC开启时钟,把GPIO和TIM的时钟打开

  • 🌾第二步,GPIO初始化,把GPIO配置成输入模式(一般选择上拉输入或浮空输入模式

  • 🌾第三步,配置时基单元,让CNT计数器在内部时钟的驱动下自增运行,和之前代码一

  • 🌾第四步,配置输入捕获单元,包括滤波器、极性、直连通道、交叉通道、分频器这些参数,用一个结构体就可以统一进行配置了

  • 🌾第五步,选择从模式的触发源,触发源选择为TI1FP1,这里调用一个库函数给一个参数就行了

  • 🌾第六步,选择触发之后执行的操作,执行Reset操作,这里调用一个库函数就行了
    最后,当这些电路都配置好之后,调用TIM_Cmd函数,开启定时器。这样所有的电路就能配合起来了,按照我们的要求工作了。当我们需要读取最新一个周期的频率时,直接读取CCR寄存器,然后按照fc/N,计算一下就行了,这就是整个程序的思路

      注意滤波器和分频器的区别:虽然它俩都是计次,但是滤波器计次并不会改变信号的原有频率,一般滤波器的采样频率都会远高于信号频率,所以滤波器只会滤除高频噪声使信号更平滑,1KHz滤波之后仍然是1KHz,信号频率不会变化;而分频器就是对信号本身进行计次,会改变频率,1KHz,2分频之后就是500Hz,4分频就是250Hz。 

源码

PWM.c

#include "stm32f10x.h"                  // Device header
#include "PWM.h"

void PWM_Init(TIM_TypeDef* TIMx)
{
	//1. RCC开启时钟,把要用的TIM外设和GPIO外设的时钟打开
	//2.配置时基单元,包括时钟源选择,预分频器,自动重装载器,时基单元就配置好了
	//3.配置输出比较单元,包括CCR的值,输出比较模式,极性选择,输出使能这些参数
	//4.初始化GPIO,把PWM的OC输出GPIO引脚配置为复用输出模式
	//5.运行控制,启动计数器
	
	//Setp 1.
	//RCC APB1的外设时钟控制,因为TIM2在STM32的APB1外设总线上
	if(TIMx == TIM2)
		RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);
	
	//GPIO初始化
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
	GPIO_InitTypeDef GPIO_InitStruct;
	GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AF_PP;	//复用推挽输出
	GPIO_InitStruct.GPIO_Pin = GPIO_Pin_0;
	GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA, &GPIO_InitStruct);
	
	//Setp 2.
	//选择时基单元的时钟,使用内部RCC时钟 CLK_INT (Clock_Internal)
	TIM_InternalClockConfig(TIMx);
	
	//Setp 3.
	//配置时基单元
	TIM_TimeBaseInitTypeDef TimeBaseInitStruct;
	TimeBaseInitStruct.TIM_ClockDivision = TIM_CKD_DIV1;		//时钟信号滤波使用,滤波的采样频率,采样点数
	TimeBaseInitStruct.TIM_CounterMode = TIM_CounterMode_Up;	//计数器向上计数
	TimeBaseInitStruct.TIM_Period = 100 - 1;					//ARR, Auto-Reload Register 自动重装载寄存器的值,记得需要减一
	TimeBaseInitStruct.TIM_Prescaler = 720 - 1;					//PSC, 预分频器的值,记得需要减一
	TimeBaseInitStruct.TIM_RepetitionCounter = 0;				//重复计数器的值	
	TIM_TimeBaseInit(TIMx, &TimeBaseInitStruct);
	
	
	
	//输出表通道1
	TIM_OCInitTypeDef TIM_OCInitStruct;
	TIM_OCStructInit(&TIM_OCInitStruct);
	TIM_OCInitStruct.TIM_OCMode = TIM_OCMode_PWM1;
	TIM_OCInitStruct.TIM_OCPolarity = TIM_OCPolarity_High;
	TIM_OCInitStruct.TIM_OutputState = TIM_OutputState_Enable;
	TIM_OCInitStruct.TIM_Pulse = 10;		//CCR
	TIM_OC1Init(TIMx, &TIM_OCInitStruct);
	
	//输出PWM目标,PWM频率1KHz,PWM占空比 50%,PWM分辨率 1%
	//PWM频率=72MHz/(PSC+1)(ARR+1) =>PSC 720
	//PWM占空比=CCR/(ARR+1) ==>CRR 50
	//PWM分辨率=1/(ARR+1)  ==>ARR 100
	//
	
	//Setp 6.
	//定时器启动
	TIM_Cmd(TIMx, ENABLE);
}

void PWM_SetCompare1(TIM_TypeDef* TIMx, uint16_t Compare)
{
	TIM_SetCompare1(TIMx, Compare);
}

void PWM_SetPrescale(uint16_t Prescale)
{
	TIM_PrescalerConfig(TIM2, Prescale, TIM_PSCReloadMode_Update);
}

PWM.h

#ifndef __PWM_H__
#define __PWM_H__

void PWM_Init(TIM_TypeDef* TIMx);
void PWM_SetCompare1(TIM_TypeDef* TIMx, uint16_t Compare);
void PWM_SetPrescale(uint16_t Prescale);

#endif 

IC.c

#include "stm32f10x.h"                  // Device header
#include "InputCapture.h"


void IC_Init(void)
{
	//使用定时器TIM3 CH1
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
	
	//使用定时器TIM3 CH1,在stm32功能表中对应PA6引脚
	//GPIO初始化
	GPIO_InitTypeDef GPIO_InitStruct;
	GPIO_InitStruct.GPIO_Mode = GPIO_Mode_IPU;	//上拉输入
	GPIO_InitStruct.GPIO_Pin = GPIO_Pin_6;
	GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA, &GPIO_InitStruct);
	
	//Setp 2.
	//选择时基单元的时钟,使用内部RCC时钟 CLK_INT (Clock_Internal)
	TIM_InternalClockConfig(TIM3);
	
	//Setp 3.
	//配置时基单元
	TIM_TimeBaseInitTypeDef TimeBaseInitStruct;
	TimeBaseInitStruct.TIM_ClockDivision = TIM_CKD_DIV1;		//时钟信号滤波使用,滤波的采样频率,采样点数
	TimeBaseInitStruct.TIM_CounterMode = TIM_CounterMode_Up;	//计数器向上计数
	TimeBaseInitStruct.TIM_Period = 65536 - 1;					//ARR, Auto-Reload Register 自动重装载寄存器的值,记得需要减一
	TimeBaseInitStruct.TIM_Prescaler = 72 - 1;					//PSC, 预分频器的值,记得需要减一
	TimeBaseInitStruct.TIM_RepetitionCounter = 0;				//重复计数器的值	
	TIM_TimeBaseInit(TIM3, &TimeBaseInitStruct);
	
	//TIM3定时器
	//计数器频率=72MHz/(PSC+1)=72MHz/72=1MHz
	//
	
	
	//输入捕获功能配置
	TIM_ICInitTypeDef TIM_ICInitStruct;
	TIM_ICStructInit(&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);
	
	//选择输入Trigger
	TIM_SelectInputTrigger(TIM3, TIM_TS_TI1FP1);
	
	//TIM Slave功能
	TIM_SelectSlaveMode(TIM3, TIM_SlaveMode_Reset);
	
	
	//定时器启动
	TIM_Cmd(TIM3, ENABLE);
}

uint16_t IC_GetRate(void)
{
	return 1000000/(TIM_GetCapture1(TIM3) + 1);
}

IC.h

#ifndef __INPUT_CAPTURE_H__
#define __INPUT_CAPTURE_H__

void IC_Init(void);
uint16_t IC_GetRate(void);

#endif

man.c

#include "stm32f10x.h"                  // Device header
#include "oled.h"
#include "PWM.h"
#include "Delay.h"
#include "InputCapture.h"

uint8_t KeyNum = 0;

int main(int argc, char *argv[])
{	
	int i = 0;
	OLED_Init();
	OLED_ShowString(1, 1, "PWM IC");
	
	PWM_Init(TIM2);
	PWM_SetCompare1(TIM2, 80);
	PWM_SetPrescale(360-1);
	
	IC_Init();
	
	
	while(1)
	{
		OLED_ShowNum(2,1,IC_GetRate(), 5);
	}
	
	return 1;
}

2.1 实验结果 

逻辑分析仪抓取到的频率为2000Hz,占空比Duty=80%。

3.🚢实验2-PWMI输入捕获测频率和占空比

IC.c文件改进使用 PWMI 输入捕获模式测量输入PWM信号的频率和占空比。

PWM.c

#include "stm32f10x.h"                  // Device header
#include "InputCapture.h"


void IC_Init(void)
{
	//使用定时器TIM3 CH1
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
	
	//使用定时器TIM3 CH1,在stm32功能表中对应PA6引脚
	//GPIO初始化
	GPIO_InitTypeDef GPIO_InitStruct;
	GPIO_InitStruct.GPIO_Mode = GPIO_Mode_IPU;	//上拉输入
	GPIO_InitStruct.GPIO_Pin = GPIO_Pin_6;
	GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA, &GPIO_InitStruct);
	
	//Setp 2.
	//选择时基单元的时钟,使用内部RCC时钟 CLK_INT (Clock_Internal)
	TIM_InternalClockConfig(TIM3);
	
	//Setp 3.
	//配置时基单元
	TIM_TimeBaseInitTypeDef TimeBaseInitStruct;
	TimeBaseInitStruct.TIM_ClockDivision = TIM_CKD_DIV1;		//时钟信号滤波使用,滤波的采样频率,采样点数
	TimeBaseInitStruct.TIM_CounterMode = TIM_CounterMode_Up;	//计数器向上计数
	TimeBaseInitStruct.TIM_Period = 65536 - 1;					//ARR, Auto-Reload Register 自动重装载寄存器的值,记得需要减一
	TimeBaseInitStruct.TIM_Prescaler = 72 - 1;					//PSC, 预分频器的值,记得需要减一
	TimeBaseInitStruct.TIM_RepetitionCounter = 0;				//重复计数器的值	
	TIM_TimeBaseInit(TIM3, &TimeBaseInitStruct);
	
	//TIM3定时器
	//计数器频率=72MHz/(PSC+1)=72MHz/72=1MHz
	//
	
	
	//输入捕获功能配置
	TIM_ICInitTypeDef TIM_ICInitStruct;
	TIM_ICStructInit(&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_ICStructInit(&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);
	
	//使用PWMI输入捕获设置函数
	TIM_PWMIConfig(TIM3, &TIM_ICInitStruct);
	
	
	//选择输入Trigger
	TIM_SelectInputTrigger(TIM3, TIM_TS_TI1FP1);
	
	//TIM Slave功能
	TIM_SelectSlaveMode(TIM3, TIM_SlaveMode_Reset);
	
	
	//定时器启动
	TIM_Cmd(TIM3, ENABLE);
}

uint16_t IC_GetRate(void)
{
	return 1000000/(TIM_GetCapture1(TIM3) + 1);
}

uint16_t IC_GetDuty(void)
{
	return (TIM_GetCapture2(TIM3) + 1)*100/TIM_GetCapture1(TIM3);
}

3.1 实验结果

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值