STM32控制步进电机:基于定时器中断的ULN2003驱动器/步进电机驱动程序

修改

2023.04.24 修改内容:完善了下程序,添加了工作原理中的一些内容。
看了评论区一位朋友指出的问题,然后我对文章中的工作原理、和程序做了些小修改(修改位置为:一、1 2 ;三、1 ;五),方便之前收藏了的朋友查阅,程序源文件也重新上传了。该篇文章中的一圈所需脉冲计算为4096个脉冲,但是其实我个人测试时为2048个脉冲,驱动器输出用了示波器测了也都是正常的(一、2),目前还不清楚原因。为了不误导其他后来的朋友,先在这里说明一下。如果有大佬知道是什么原因欢迎批评指正。

一、ULN2003驱动器

1、工作原理

下图为ULN2003驱动器原理图。
在这里插入图片描述
此驱动器的原理即为步进电机的工作原理,此篇文章有介绍到:STM32控制步进电机:工作原理及库函数(标准库) / HAL库控制程序(不定期更新)
通过一个接一个的引脚驱动电机的4个相,使得步进电机转动,该驱动器输入为低电平时,对应输出引脚输出为高电平,反之,输入为高电平时,输出为低电平。因此要使得某些引脚输出为高电平时,应当置其对应的引脚为低电平,其他引脚为高电平。

2、步距角以及一圈所需步数的计算

本篇文章使用的4相5线步进电机步距角为5.625/64,因此步进电机转一圈需要走360°/步距角的步数,即(360/5.625)*64 = 4096步
虽然计算是一圈4096个脉冲,但是实际上我使用的时候是2048个脉冲电机转一圈,于是我使用了示波器测试了程序是否有问题。
如下所示白圈中所示:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

二、硬件连接

如图所示,记得共地。4个控制引脚在下方程序的.h和.c文件中有定义和引用到。
在这里插入图片描述

三、STM32F103定时器中断控制步进电机程序

此程序效果为正转1周后再反转1周回到原点。程序如下所示。

1、.c文件

以下为步进电机驱动的motor.c文件

#include "motor.h"

//num用于对引脚的索引,j用于计算步数,fx为电机旋转方向
unsigned short int num=0,j,fx;

void motor_GPIO_Init(void)
{
    
    
 
 GPIO_InitTypeDef  GPIO_InitStructure;
 	
 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);	 //使能PB端口时钟
	
 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5 | GPIO_Pin_6 | GPIO_Pin_7 | GPIO_Pin_8;
 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; 		 //推挽输出
 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;		 //IO口速度为50MHz
 GPIO_Init(GPIOB, &GPIO_InitStructure);					 //根据设定参数初始化GPIOB
 GPIO_ResetBits(GPIOB,GPIO_Pin_5 | GPIO_Pin_6 | GPIO_Pin_7 | GPIO_Pin_8);						 //PB.5/6/7/8 输出低电平
}

void TIM3_Int_Init(u16 arr,u16 psc)
{
    
    
  TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
	NVIC_InitTypeDef NVIC_InitStructure;

	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); //时钟使能

	TIM_TimeBaseStructure.TIM_Period = arr; //设置在下一个更新事件装入活动的自动重装载寄存器周期的值 计数到5000为500ms
	TIM_TimeBaseStructure.TIM_Prescaler =psc; //设置用来作为定时器时钟频率出书的预分频值 10KHZ的计数频率
	TIM_TimeBaseStructure.TIM_ClockDivision = 0; //设置时钟分割:TDTS = Tck_tim
	TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  //TIM向上计数模式
	TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位

	TIM_ITConfig(  //使能或失能指定的TIM中断
		TIM3, //TIM
		TIM_IT_Update ,
		ENABLE  //使能
		);
	NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn;  //TIM3中断
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;  //先占优先级1
	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;  //从优先级3
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道使能
	NVIC_Init(&NVIC_InitStructure);  //初始化外设NVIC寄存器

	TIM_Cmd(TIM3, ENABLE);  //使能定时器外设
							 
}

static uint8_t GPIO_list[] = {
    
    0x01,0x02,0x04,0x08};     //对应驱动器4引脚,即电机4相

void TIM3_IRQHandler(void)   //TIM3中断(2ms)
{
    
    
	if (TIM_GetITStatus(TIM3, TIM_IT_Update) != RESET) //检查指定的TIM中断发生与否,TIM中断源
		{
    
    
			TIM_ClearITPendingBit(TIM3, TIM_IT_Update);  //清除TIMx的中断处理位
			if(judge == 0)
			{
    
    
			  judge=0;
			}
			if(judge == 1)
			{
    
    
				 if(fx == 0)  //fx为电机旋转方向,fx=0时电机正转,fx=1时电机反转
		     {
    
    
					 motor_GPIO1 = ~(GPIO_list[num]&GPIO_list[0])>>0;  //判断是否为引脚1,然后将其数值向右移动0位至第1位,得到unsigned int类型时的1或0
					 motor_GPIO2 = ~(GPIO_list[num]&GPIO_list[1])>>1;  //判断是否为引脚2,然后将其数值向右移动1位至第1位,得到unsigned int类型时的1或0
					 motor_GPIO3 = ~(GPIO_list[num]&GPIO_list[2])>>2;  //判断是否为引脚3,然后将其数值向右移动1位至第1位,得到unsigned int类型时的1或0
					 motor_GPIO4 = ~(GPIO_list[num]&GPIO_list[3])>>3;  //判断是否为引脚4,然后将其数值向右移动1位至第1位,得到unsigned int类型时的1或0
			 	 }
				 if(fx == 1)
				 {
    
    
					 motor_GPIO4 = ~(GPIO_list[num]&GPIO_list[0])>>0;  //上述的反转
					 motor_GPIO3 = ~(GPIO_list[num]&GPIO_list[1])>>1;
					 motor_GPIO2 = ~(GPIO_list[num]&GPIO_list[2])>>2;
					 motor_GPIO1 = ~(GPIO_list[num]&GPIO_list[3])>>3;
				 }
				 num += 1;  //num用于对引脚的索引
				 j += 1;  //j用于计算步数
				 if(num == 4)  //到第4个GPIO后回到第1个GPIO
				 {
    
    
				   num = 0;
				 }
		     if(j == 2048&fx == 0)  //走完一圈同时是正转结束,对参数进行修改
		     {
    
    
					 j = 0;
					 fx = 1;
					 num = 0;
		     }
				 if(j == 2048&fx == 1)  //走完一圈同时是正反转结束,对参数进行修改
				 {
    
    
					 j = 0;
					 fx = 0;
					 start = 0;
					 num = 0;
				 }
			}
		}
}

2、.h文件

以下为步进电机驱动的motor.h文件

#ifndef __MOTOR_H
#define __MOTOR_H

#include "sys.h"
#include "delay.h"

void motor_GPIO_Init(void);//引脚初始化
void TIM3_Int_Init(u16 arr,u16 psc); //定时器初始化

#define motor_GPIO1 PBout(5)  //引脚定义
#define motor_GPIO2 PBout(6)  //引脚定义
#define motor_GPIO3 PBout(7)  //引脚定义
#define motor_GPIO4 PBout(8)  //引脚定义

extern u8 start;  //start为1时启动电机程序,为0时关闭
extern u8 judge;  //judge为1时电机开始旋转,为0时停止

#endif

3、main.c部分程序


u8 judge = 0;  //judge为1时电机开始旋转,为0时停止
u8 start = 0;  //start为1时启动电机程序,为0时关闭

void run(void)  //步进电机启动函数
{
    
    
	if(start == 1) {
    
    judge = 1;}
	else {
    
    judge = 0;}
}

//初始化后,只要给start赋值、把run()放进main里即可,也可在上述start里添加一些步进电机以外的程序

四、效果演示

如下视频所示:

SMT32串口控制ULN2003驱动器驱动步进电机

五、程序链接

程序已经打包好上传到csdn的资源里了。
CSDN:库函数(标准库)STM32F103C8T6基于定时器中断的ULN2003驱动器/步进电机驱动程序
也可以通过以下链接下载:

链接:链接:https://pan.baidu.com/s/1rpUggpOruBFhwOHcfRon2w
提取码:l70o

本人是一名学生,目前正在学习中,本篇文章也算是我的学习笔记,如有错误的话还请指正。

猜你喜欢

转载自blog.csdn.net/xztli/article/details/127158444