PWM实现电机的正反转和调速以及TIM定时器

pwm.c

#include "pwm.h"/*
PWM --- PA2 --TIM2_CH3
//将电机信号控制一根接GND,一根接在PA2(TIM2_CH3),
输出PWM控制电机快慢
TIM2挂在APB1 定时器频率:84MHZ*/
void Pwm_Init(void)
{GPIO_InitTypeDef  		GPIO_InitStruct;TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct;TIM_OCInitTypeDef 		TIM_OCInitStruct;//2、使能定时器2和相关IO口时钟。RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);//3、使能GPIOA时钟:RCC_AHB1PeriphClockCmd (RCC_AHB1Periph_GPIOA,ENABLE);GPIO_InitStruct.GPIO_Pin	= GPIO_Pin_2; 						//引脚2GPIO_InitStruct.GPIO_Mode	= GPIO_Mode_AF;       				//复用功能GPIO_InitStruct.GPIO_Speed	= GPIO_Speed_50MHz;					//速度GPIO_InitStruct.GPIO_OType	= GPIO_OType_PP;					//推挽GPIO_InitStruct.GPIO_PuPd	= GPIO_PuPd_UP;						//上拉//4、初始化IO口为复用功能输出。GPIO_Init(GPIOA, &GPIO_InitStruct);//5、GPIOF9复用映射到定时器2选择哪个复用功能GPIO_PinAFConfig(GPIOA,GPIO_PinSource2,GPIO_AF_TIM2); //定时器2挂在APB1(42MHZ)  所以定时器频率:84MHZTIM_TimeBaseInitStruct.TIM_Prescaler	= 84-1;  				//84分频  84MHZ/84=1MZ TIM_TimeBaseInitStruct.TIM_Period		= 500-1;				//计数500 用时500usTIM_TimeBaseInitStruct.TIM_CounterMode	= TIM_CounterMode_Up; 	//向上计数TIM_TimeBaseInitStruct.TIM_ClockDivision= TIM_CKD_DIV1;			//分频因子//2、初始化定时器,配置ARR,PSC。TIM_TimeBaseInit(TIM2,&TIM_TimeBaseInitStruct);TIM_OCInitStruct.TIM_OCMode			= TIM_OCMode_PWM1;  		//PWM模式1TIM_OCInitStruct.TIM_OCPolarity		= TIM_OCPolarity_High;		//极性电平 即输出的有效电平(电机需要的是高电平)TIM_OCInitStruct.TIM_OutputState    = TIM_OutputState_Enable; 	//通道使能//7、初始化输出比较参数: 0C3通道3TIM_OC3Init(TIM2, &TIM_OCInitStruct);//8、使能预装载寄存器: TIM_OC3PreloadConfig(TIM2, 	TIM_OCPreload_Enable); //9、使能自动重装载的预装载寄存器允许位	TIM_ARRPreloadConfig(TIM2,ENABLE);//10、使能定时器。TIM_Cmd(TIM2, ENABLE);}

在这里,使用了TIM定时器,只能说像EXTI外部中断、TIM定时器这些是基础呀~

之前忘了写time.c的总结

以下是利用TIM定时器控制led灯的
/*定时器4配置流程1、使能定时器时钟。RCC_APB1PeriphClockCmd();2、初始化定时器,配置ARR,PSC。TIM_TimeBaseInit();3、启定时器中断,配置NVIC。NVIC_Init();4、设置 TIM4_DIER  允许更新中断TIM_ITConfig();5、使能定时器。TIM_Cmd();6、编写中断服务函数。TIMx_IRQHandler();*/#include "time.h"/*定时器TIM3挂APB1总线下,时钟频率:84MHZTIM3为16位定时器  最大计数值为:65535*/
void Time3_Init(void)
{TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct;NVIC_InitTypeDef  		NVIC_InitStruct;//定义结构体名称//1、能定时器时钟。RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);//1ms产生中断TIM_TimeBaseInitStruct.TIM_Prescaler	= 840-1;  				//84分频  84MHZ/840 = 100KHZ  100 000HZTIM_TimeBaseInitStruct.TIM_Period		= 20000-1;				//计数1000 用时200msTIM_TimeBaseInitStruct.TIM_CounterMode	= TIM_CounterMode_Up; 	//向上计数TIM_TimeBaseInitStruct.TIM_ClockDivision= TIM_CKD_DIV1;			//分频因子//2、初始化定时器,配置ARR,PSC。TIM_TimeBaseInit(TIM3,&TIM_TimeBaseInitStruct);//3、启定时器中断,配置NVIC。NVIC_InitStruct.NVIC_IRQChannel						= TIM3_IRQn;  	//选择通道NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority	= 2;			//抢占优先级NVIC_InitStruct.NVIC_IRQChannelSubPriority			= 2; 			//响应优先级NVIC_InitStruct.NVIC_IRQChannelCmd					= ENABLE;		//通道使能	//5、配置中断分组(NVIC),并使能中断。NVIC_Init(&NVIC_InitStruct);//4、设置 TIM3_DIER  允许更新中断TIM_ITConfig(TIM3, TIM_IT_Update, ENABLE);//5、使能定时器。TIM_Cmd(TIM3, ENABLE);}/*定时器TIM4挂APB1总线下,时钟频率:84MHZTIM4为16位定时器  最大计数值为:65535*/
void Time4_Init(void)
{TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct;NVIC_InitTypeDef  		NVIC_InitStruct;//1、能定时器时钟。RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM4, ENABLE);//1ms产生中断
//	TIM_TimeBaseInitStruct.TIM_Prescaler	= 84-1;  				//84分频  84MHZ/84 = 1MHZ  1us数一个数
//	TIM_TimeBaseInitStruct.TIM_Period		= 1000-1;				//计数1000 用时1ms
//	TIM_TimeBaseInitStruct.TIM_CounterMode	= TIM_CounterMode_Up; 	//向上计数
//	TIM_TimeBaseInitStruct.TIM_ClockDivision= TIM_CKD_DIV1;			//分频因子
//	//2、初始化定时器,配置ARR,PSC。
//	TIM_TimeBaseInit(TIM4,&TIM_TimeBaseInitStruct);//练习3TIM_TimeBaseInitStruct.TIM_Prescaler	= 8400-1;  				//84分频    84MHZ/8400 = 10KHZ  10000HZTIM_TimeBaseInitStruct.TIM_Period		= 10000-1;				//计数10000 用时1sTIM_TimeBaseInitStruct.TIM_CounterMode	= TIM_CounterMode_Up; 	//向上计数TIM_TimeBaseInitStruct.TIM_ClockDivision= TIM_CKD_DIV1;			//分频因子//2、初始化定时器,配置ARR,PSC。TIM_TimeBaseInit(TIM4,&TIM_TimeBaseInitStruct);	//3、启定时器中断,配置NVIC。NVIC_InitStruct.NVIC_IRQChannel						= TIM4_IRQn;  	//选择通道NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority	= 2;			//抢占优先级NVIC_InitStruct.NVIC_IRQChannelSubPriority			= 2; 			//响应优先级NVIC_InitStruct.NVIC_IRQChannelCmd					= ENABLE;		//通道使能	//5、配置中断分组(NVIC),并使能中断。NVIC_Init(&NVIC_InitStruct);//4、设置 TIM4_DIER  允许更新中断TIM_ITConfig(TIM4, TIM_IT_Update, ENABLE);//5、使能定时器。TIM_Cmd(TIM4, ENABLE);}	//6、编写中断服务函数。1ms
void TIM3_IRQHandler(void)
{if(TIM_GetITStatus(TIM3, TIM_IT_Update) == SET){//定时器处理事件GPIO_ToggleBits(GPIOA,GPIO_Pin_6);}TIM_ClearITPendingBit(TIM3, TIM_IT_Update);
}//6、编写中断服务函数。1ms
void TIM4_IRQHandler(void)
{static int num=0;if(TIM_GetITStatus(TIM4, TIM_IT_Update) == SET){if(num == 0){GPIO_ResetBits(GPIOA, GPIO_Pin_7);//D31一灭4//GPIO_ToggleBits(GPIOF, GPIO_Pin_9);num++;}else{if(num == 4){num=0;}else{num++;} GPIO_SetBits(GPIOA, GPIO_Pin_7);//灭}//定时器处理事件//GPIO_ToggleBits(GPIOA,GPIO_Pin_7);}TIM_ClearITPendingBit(TIM4, TIM_IT_Update);
}

就是上两篇说到的,TIM定时器和EXTI外部中断都会用到NVIC和中断服务函数,配置流程差不多。

在PWM中TIM定时器的作用

TIM(定时器)在嵌入式系统中通常用于生成精确的时间延迟、定时触发等功能。

TIM2 被配置为 PWM 模式,用于生成 PWM 信号来控制电机的速度或位置。

具体来说,TIM 定时器的作用包括:

1. 生成定时的基准时钟信号:定时器可以生成一个基于系统时钟的定时信号,用于精确计时。这对于需要精确时间间隔的应用非常有用,比如周期性任务的触发

2. 生成 PWM 信号:定时器可以通过配置输出比较通道,生成 PWM(脉冲宽度调制)信号。PWM 信号的占空比可以通过定时器的参数配置来调整,从而控制电机的转速或位置

3. 产生精确的时间延迟:定时器可以用来产生精确的时间延迟,比如在需要精确控制时序的情况下,比如在通信协议中生成特定的时钟信号。

4. 实现定时触发功能:定时器可以配置为在达到特定时间后触发中断,用于执行定时任务。这对于需要周期性执行的任务非常有用,比如传感器数据的定时采集、周期性数据传输等。

总之,TIM 定时器在嵌入式系统中是非常重要的功能模块,它提供了精确的时间控制能力,可以满足各种定时、PWM 生成、延迟等需求。

在这个代码中,通过配置 TIM2 定时器和相关的输出比较通道,实现了 PWM 信号的生成,用于控制电机。

 

pwm.h

#ifndef __PWM_H
#define __PWM_H
#include "stm32f4xx.h"void Pwm_Init(void);#endif

 

main.c

#include "stm32f4xx.h"
#include "led.h"
#include "key.h"
#include "exti.h"
#include "delay.h"
#include "pwm.h"//这是一个主函数
int main(void)
{u32 count = 0;//NVIC分组 抢占优先级两位:0~3  响应优先级两位:0~3 NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);Delay_Init();Led_Init();Pwm_Init();//不断改变比较值CCRx,达到不同的占空比效果:TIM_SetCompare3(TIM2,499);//GPIO_ToggleBits(GPIOE,GPIO_Pin_14);while(1){for(count=499; count>200; count -= 20){TIM_SetCompare3(TIM2,count);delay_s(1);}}return 0;
}

反转只需将PA2和GND调换位置,从快速到慢速,或者从慢速到快速只需要更改count的数法。 

README

实现电机的正反转和调速只需要用到PWM和delay相关函数 (pwm.c、pwm.h、delay.c、delay.h还要main.c),其中在PWM中使用了TIM定时器、delay相关函数在上几篇(用systick定时器写的精准延时)。

我所实验成功的电机是这种:

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

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

相关文章

Django下载使用、文件介绍

【一】下载并使用 【1】下载框架 (1)注意事项 计算机名称不要出现中文python解释器版本不同可能会出现启动报错项目中所有的文件名称不要出现中文多个项目文件尽量不要嵌套,做到一项一夹 (2)下载 Django属于第三方模块&#…

张宇高数一学习笔记-1-第一讲-函数极限与连续(1)

1、函数的概念与特性 1.1、函数 1、y才是x的函数,而f()是加工法则。 2、定义域是基于加工法则f()存在的,也就是说加工法则f()对()内数据的限制范围就是定义域。而 3、自变量x的取值范围来源于:定义域对()内的含x表达式的限制下得出的x的取值…

【C2架构】

C2架构 C2 架构也就可以理解为,恶意软件通过什么样的方式获取资源和命令,以及通过什么样的方式将数据回传给攻击者 使用的通信方式: 传输协议层 TCP、UDP 网络协议层 icmp(ping) 应用协议层 HTTP/HTTPS、FTP、DNS、S…

[深度学习] 常见名称概念

SOTA SOTA是指"State-of-the-art"的缩写,意为"最先进的技术"。SOTA是指在某个领域或任务中,当前被认为是最好的技术或模型。随着技术的不断发展和进步,SOTA会随之变化。对于机器学习和人工智能领域而言,SOTA…

20.Ubuntu下安装GCC

文章目录 Ubuntu下安装GCC查看官方安装指导错误缺少gmp库缺少32位开发库libcg: error: gengtype-lex.c: No such file or directoryreference 欢迎访问个人网络日志🌹🌹知行空间🌹🌹 Ubuntu下安装GCC 为了支持新的c标准&#xff…

深入了解 Vue 组件

在 Vue.js 中,组件是构建用户界面的核心概念之一。通过组件,我们可以将界面分割成独立、可复用的模块,使得代码更加清晰、灵活,并且更易于维护。在本文中,我们将深入探讨 Vue 组件的基本概念、创建方法以及常见用法。 …

STM32微控制器中,如何处理多个同时触发的中断请求?

在STM32微控制器中,处理多个同时触发的中断请求需要一个明确的中断优先级策略,以确保关键任务能够及时得到响应。STM32的中断控制器(NVIC)支持优先级分组,允许开发者为不同的中断设置抢占优先级和子优先级。本文将详细…

uniapp 打包后缺少maps模块和share模块的解决方案

缺失maps模块 我的应用 | 高德控制台 缺失share模块 QQ互联管理中心 微信开放平台

星云小窝项目1.0——项目介绍(一)

星云小窝项目1.0——项目介绍(一) 文章目录 前言1. 介绍页面2. 首页2.1. 游客模式2.2. 注册用户后 3. 星云笔记3.1. 星云笔记首页3.2. 星云笔记 个人中心3.2. 星云笔记 系统管理3.3. 星云笔记 文章展示3.3. 星云笔记 新建文章 4. 数据中心5. 交流评论6. …

Linux cp、mv命令显示进度条

1.advcpmv 平常使用cp 拷贝大文件时,看不到多久可以完成,虽然加上-v参数也只能看到正在拷贝文件,那就使用以下方法实现 git clone https://github.com/jarun/advcpmv.git cd advcpmv/ bash install.shmv ./advcp /usr/local/bin/ mv ./advmv …

SpringBoot3+Vue3项目的阿里云部署--将后端以及前端项目打包

一、后端:在服务器上制作成镜像 1.准备Dockerfile文件 # 基础镜像 FROM openjdk:17-jdk-alpine # 作者 MAINTAINER lixuan # 工作目录 WORKDIR /usr/local/lixuan # 同步docker内部的时间 RUN ln -snf /usr/share/zoneinfo/$TZ /etc/localtime && echo $TZ…

HCIP【PPP协议CHAP验证】

目录 实验目的: 实验拓扑图: 实验过程: 一:按照图示配置所有设备的IP地址 (1)R1和R2使用PPP链路直连 (2)R2和R3把2条PPP链路捆绑为PPP MP直连 二:PPP协议的CHAP验证 (3)R2 对 R1 的 PPP 进行单向 c…

代码随想录Day57:回文子串、最长回文子序列

回文子串 class Solution { public:int countSubstrings(string s) {vector<vector<bool>> dp(s.size(), vector<bool>(s.size(), false));int res 0;for(int i s.size() - 1; i > 0; i--){for(int j i; j < s.size(); j){if(i j || (s[i] s[j] …

LeetCode——贪心算法

贪心思想 保证每次操作都是局部最优的&#xff0c;并且最后得到的结果是全局最优的——减少遍历的次数 1.买卖股票的最佳时机 121简单 这里的贪心思想是更新股票的最低价和最大利润&#xff0c;规则是先买后卖 class Solution {public int maxProfit(int[] prices) {// 记录…

C#执行命令行

效果图 主要代码方法 private Process p;public List<string> ExecuteCmd(string args){System.Diagnostics.Process p new System.Diagnostics.Process();p.StartInfo.FileName "cmd.exe";p.StartInfo.RedirectStandardInput true;p.StartInfo.RedirectSta…

权限提升-Windows权限提升篇数据库篇MYSQLMSSQLORACLE自动化项目

知识点 1、Web到Win-数据库提权-MSSQL 2、Web到Win-数据库提权-MYSQL 3、Web到Win-数据库提权-Oracle 章节点&#xff1a; 1、Web权限提升及转移 2、系统权限提升及转移 3、宿主权限提升及转移 4、域控权限提升及转移 基础点 0、为什么我们要学习权限提升转移技术&#xff1…

108、3D Gaussian Splatting for Real-Time Radiance Field Rendering

简介 官网 更少训练时间的同时实现最先进的视觉质量&#xff0c;能在1080p分辨率下实现高质量的实时(≥30 fps)新视图合成 NeRF使用隐式场景表示&#xff0c;体素&#xff0c;点云等属于显示建模方法&#xff0c;3DGS就是显示辐射场。它用3D高斯作为灵活高效的表示方法&…

vscode使用Runner插件将.exe文件统一放到一个目录下

找到右下角管理&#xff0c;点击扩展。 找到Code Runner插件&#xff0c;打开扩展设置。 向下翻&#xff0c;找到Executor Map&#xff0c;点击在settings.json中编辑。 在c和c的配置命令栏中增加\\\output\\即可。&#xff08;增加的目录不能自动创建&#xff0c;需要手动创建…

FPGA时钟资源详解——Clock-Capable Inputs

目录 一、概述 1.1 为什么使用CC 1.2 如何使用CC 二、Clock-Capable Inputs 2.1 SRCC 2.2 MRCC 2.3 其他用途 2.3.1 作为普通I/O使用 2.3.2 连接到CMT 一、概述 在 FPGA 设计中&#xff0c;将外部用户时钟引入 FPGA 是一项重要的任务&#xff0c;对整个系统的性能和稳…

常用ES标准

ES2015&#xff1a; 1.块级作用域const、let const声明对象可修改属性&#xff0c;但不能重新赋值对象。 2.解构赋值 const arr [a1, a2, a3]; const [a1, ...rest] arr; // rest [a2, a3];3.模板字符串 const date "星期一"; console.log(今天是${date};);4…