当前位置: 首页 > news >正文

STM32用PWM驱动步进电机

硬件介绍:

连线:

注意这里stp连的是pwm脉冲,dir连的是方向到时候代码pwm波形就是从这里来的,具体接线根据你的代码来

注意要点:步进电机和舵机驱动是不一样的,它是根据步长来移动的,所以要开一个中断,来计算步长,具体代码,可以看看。

代码部分:

#include "stm32f10x.h"
#include "math.h"

// 步进电机参数配置
#define STEPS_PER_REVOLUTION 200      // 每转步数 (1.8°/步)
#define MICROSTEPS           16       // 微步细分

// 全局变量
volatile uint32_t step_count = 0;
float current_angle = 0.0f;

// 初始化函数
void Stepper_Init(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
TIM_OCInitTypeDef TIM_OCInitStructure;
NVIC_InitTypeDef NVIC_InitStructure;

    // 1. 使能时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);

    // 2. 配置PWM输出引脚 (PA1 - TIM2_CH2)
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStructure);

    // 3. 配置方向控制引脚 (PA2)
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_Init(GPIOA, &GPIO_InitStructure);

    // 4. 配置定时器时基
TIM_TimeBaseStructure.TIM_Period = 7200 - 1;     // 初始频率约10Hz @72MHz
TIM_TimeBaseStructure.TIM_Prescaler = 72 - 1;
TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);

    // 5. PWM 输出配置
TIM_OCStructInit(&TIM_OCInitStructure);
TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;
TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;
TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;
TIM_OCInitStructure.TIM_Pulse = 3600; // 50% 占空比
TIM_OC2Init(TIM2, &TIM_OCInitStructure);

    // 6. 配置更新中断
NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);

    TIM_ITConfig(TIM2, TIM_IT_Update, ENABLE);
}

// 启动电机
void Start_Motor(void)
{
step_count = 0;
TIM_ClearITPendingBit(TIM2, TIM_IT_Update);
TIM_Cmd(TIM2, ENABLE);
}

// 停止电机
void Stop_Motor(void)
{
TIM_Cmd(TIM2, DISABLE);
TIM_SetCompare2(TIM2, 0);
TIM_ClearITPendingBit(TIM2, TIM_IT_Update);
}

// 设置电机速度 (RPM)
void Set_Motor_Speed(float rpm)
{
if (rpm <= 0) return;

    uint32_t freq = (uint32_t)(rpm * STEPS_PER_REVOLUTION * MICROSTEPS / 60.0f);
if (freq == 0) return;

    uint32_t arr = (72000000UL / (72UL * freq)) - 1;
if (arr < 10) arr = 10; // 防止太小导致溢出

    TIM_SetAutoreload(TIM2, arr);
TIM_SetCompare2(TIM2, arr / 2); // 保持50%占空比
}

// 旋转到指定角度(度)
void Rotate_To_Angle(float target_angle, float speed_rpm)
{
float angle_diff = target_angle - current_angle;

    while(angle_diff > 180.0f) angle_diff -= 360.0f;
while(angle_diff < -180.0f) angle_diff += 360.0f;

    int32_t required_steps = (int32_t)(angle_diff * STEPS_PER_REVOLUTION * MICROSTEPS / 360.0f);

    if(required_steps >= 0) {
GPIO_SetBits(GPIOA, GPIO_Pin_2);   // 正转
} else {
GPIO_ResetBits(GPIOA, GPIO_Pin_2); // 反转
required_steps = -required_steps;
}

    Set_Motor_Speed(speed_rpm);

    uint32_t total_steps = (uint32_t)required_steps;
step_count = 0;

    Start_Motor();

    while(step_count < total_steps) {
__NOP();
}

    Stop_Motor();

    current_angle = target_angle;
}

// 中断服务函数
void TIM2_IRQHandler(void)
{
if (TIM_GetITStatus(TIM2, TIM_IT_Update) != RESET)
{
step_count++;
TIM_ClearITPendingBit(TIM2, TIM_IT_Update);
}
}

这是pwm部分

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

// 主函数
int main(void)
{
__enable_irq();                     // 开启全局中断

    Stepper_Init();                     // 初始化步进电机驱动

    // 第一次旋转到90度
Rotate_To_Angle(90.0f, 60.0f);
Delay_ms(500);

    // 第二次旋转到-90度
Rotate_To_Angle(-90.0f, 30.0f);
Delay_ms(2000);

    // 进入主循环
while(1)
{
// 可以继续添加其他操作
}
}
这是主函数

注意要是想要同时驱动两个步进电机,代码还是要稍微修改的。后续代码再发

http://www.dtcms.com/a/275963.html

相关文章:

  • 快捷跑通ultralytics下的yolo系列
  • 算法第三十一天:贪心算法part05(第八章)
  • 回溯算法-数据结构与算法
  • Pythone第二次作业
  • brpc 介绍与安装
  • Redis过期策略与内存淘汰机制面试笔记
  • 数据库连接池及其核心特点
  • AI编程下的需求规格文档的问题及新规范
  • ADSP-1802这颗ADI的最新DSP应该怎么做开发(一)
  • 【Redis实战】Widnows本地模拟Redis集群的2种方法
  • Syntax Error: TypeError: Cannot set properties of undefined (setting ‘parent‘)
  • Unity URP + XR 自定义 Skybox 在真机变黑问题全解析与解决方案(支持 Pico、Quest 等一体机)
  • Cookie、Session、Token 有什么区别?
  • Spring Boot 中使用 Lombok 进行依赖注入的示例
  • 【离线数仓项目】——电商域DWD层开发实战
  • 【C++ STL 库】解析stack、queue、priority_queue类
  • 中文多智能体金融交易决策框架-TradingAgents-CN
  • 本地安装ClaudeCode全攻略
  • 【Python】多线程详解:从基础概念到实战应用
  • 免费尝试claude code的安利,截至今天可用(7/12)
  • openGauss数据库管理实战指南——基本常用操作总结
  • AI:机器人未来的形态是什么?
  • Cisco ACI 生成Postman CSV 脚本场景
  • 死锁的避免
  • Spring Boot 应用中,配置的加载优先级
  • 锁相环初探
  • CTFHub————Web{信息泄露[Git泄露(Stash、Index)]}
  • Java 接口详解:从基础到高级,掌握面向对象设计的核心契约
  • 使用FastAdmin框架开发二
  • ollama - sqlcoder模型:面向提示词编程(根据用户信息生成sql语句并执行返回结果)