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

程序代码篇---STM舵机控制


文章目录

  • 前言
  • 1. 头文件和全局变量
  • 2. 舵机GPIO初始化函数
  • 3. TIM7初始化函数
  • 4. 舵机增量偏移计算函数
  • 5. 舵机引脚电平设置函数
  • 6. TIM7中断服务函数
  • 7. 舵机初始化函数
  • 8. 舵机角度设置函数
  • 9. 舵机停止运动函数
  • 10. 舵机偏差设置函数
  • 总结


前言

本次代码实现了基于STM32的舵机控制功能,主要包括舵机的GPIO初始化、PWM信号生成、舵机角度设置、停止运动以及偏差设置等功能。下面我将逐部分详细解释这段代码。


1. 头文件和全局变量

#include "y_servo/y_servo.h"

pwmServo_t pwmServo_angle[SERVO_NUM];

#include “y_servo/y_servo.h”:包含了舵机相关的头文件,可能定义了舵机的配置和函数声明。
pwmServo_t pwmServo_angle[SERVO_NUM]:定义了一个全局数组,用于存储每个舵机的控制参数(目标角度、当前角度、增量、时间等)。

2. 舵机GPIO初始化函数

static void servo_gpio_init(void)
{
    u8 i;
    GPIO_InitTypeDef GPIO_InitStructure;

    RCC_APB2PeriphClockCmd(SERVO0_GPIO_CLK | SERVO1_GPIO_CLK | SERVO2_GPIO_CLK | SERVO3_GPIO_CLK | SERVO4_GPIO_CLK | SERVO5_GPIO_CLK, ENABLE); /* 使能 舵机 端口时钟 */

    GPIO_InitStructure.GPIO_Pin = SERVO0_PIN;         /* 配置引脚 */
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; /* IO翻转50MHZ */
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;  /* 推挽输出 */
    GPIO_Init(SERVO0_GPIO_PORT, &GPIO_InitStructure);

    GPIO_InitStructure.GPIO_Pin = SERVO1_PIN;
    GPIO_Init(SERVO1_GPIO_PORT, &GPIO_InitStructure);

    GPIO_InitStructure.GPIO_Pin = SERVO2_PIN;
    GPIO_Init(SERVO2_GPIO_PORT, &GPIO_InitStructure);

    GPIO_InitStructure.GPIO_Pin = SERVO3_PIN;
    GPIO_Init(SERVO3_GPIO_PORT, &GPIO_InitStructure);

    GPIO_InitStructure.GPIO_Pin = SERVO4_PIN;
    GPIO_Init(SERVO4_GPIO_PORT, &GPIO_InitStructure);

    GPIO_InitStructure.GPIO_Pin = SERVO5_PIN;
    GPIO_Init(SERVO5_GPIO_PORT, &GPIO_InitStructure);

    for (i = 0; i < SERVO_NUM; i++)
    {
        pwmServo_angle[i].aim = 1500;
        pwmServo_angle[i].current = 1500;
        pwmServo_angle[i].increment = 0;
        pwmServo_angle[i].time = 5000;
        pwmServo_angle[i].bias = 0;
    }
}

功能:初始化舵机的GPIO引脚,配置为推挽输出模式,并初始化舵机控制参数。
关键点:
RCC_APB2PeriphClockCmd:使能舵机GPIO端口的时钟。
GPIO_InitStructure:配置GPIO引脚为推挽输出模式,速度为50MHz。
for循环:初始化每个舵机的控制参数,默认角度为1500(舵机中位),时间5000ms,偏差为0。

3. TIM7初始化函数

static void TIM7_Init(void)
{
    TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
    NVIC_InitTypeDef NVIC_InitStructure;
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM7, ENABLE); //使能定时器的时钟
    TIM_TimeBaseStructure.TIM_Prescaler = 71;            // 预分频器
    TIM_TimeBaseStructure.TIM_Period = 9;                //设定计数器自动重装值
    TIM_TimeBaseInit(TIM7, &TIM_TimeBaseStructure);
    TIM_ClearFlag(TIM7, TIM_FLAG_Update);               //清除TIM的更新标志位
    TIM_ITConfig(TIM7, TIM_IT_Update, ENABLE);

    //中断优先级NVIC设置
    NVIC_InitStructure.NVIC_IRQChannel = TIM7_IRQn;              //TIM中断
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;    //先占优先级0级
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;           //从优先级1级
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;              //IRQ通道被使能
    NVIC_Init(&NVIC_InitStructure);                              //初始化NVIC寄存器

    TIM_Cmd(TIM7, ENABLE);
}

功能:初始化TIM7定时器,用于生成PWM信号控制舵机。
关键点:
TIM_TimeBaseStructure.TIM_Prescaler = 71:设置预分频器,将系统时钟分频
TIM_TimeBaseStructure.TIM_Period = 9:设置定时器自动重装值,控制PWM周期。
TIM_ITConfig(TIM7, TIM_IT_Update, ENABLE):使能定时器更新中断。
NVIC_Init:配置TIM7中断优先级

4. 舵机增量偏移计算函数

static void servo_increment_offset(uint8_t index)
{
    int aim_temp;

    if (pwmServo_angle[index].increment != 0)
    {
        aim_temp = pwmServo_angle[index].aim + pwmServo_angle[index].bias;
        if (aim_temp > 2490)
            aim_temp = 2490;
        else if (aim_temp < 500)
            aim_temp = 500;

        if (abs(aim_temp - pwmServo_angle[index].current) <= fabs(pwmServo_angle[index].increment + pwmServo_angle[index].increment))
        {
            pwmServo_angle[index].current = aim_temp;
            pwmServo_angle[index].increment = 0;
        }
        else
        {
            pwmServo_angle[index].current += pwmServo_angle[index].increment;
        }
    }
}

功能:根据目标角度和当前角度计算舵机的增量偏移,逐步调整舵机角度。
关键点:
aim_temp:计算目标角度加上偏差后的值。
if (aim_temp > 2490):限制舵机角度在500到2490之间。
pwmServo_angle[index].current += pwmServo_angle[index].increment;:逐步调整当前角度。

5. 舵机引脚电平设置函数

static void servo_pin_set(u8 index, BitAction level)
{
    switch (index)
    {
    case 0:
        SERVO0_PIN_SET(level);
        break;
    case 1:
        SERVO1_PIN_SET(level);
        break;
    case 2:
        SERVO2_PIN_SET(level);
        break;
    case 3:
        SERVO3_PIN_SET(level);
        break;
    case 4:
        SERVO4_PIN_SET(level);
        break;
    case 5:
        SERVO5_PIN_SET(level);
        break;
    default:
        break;
    }
}

功能:根据舵机索引设置对应引脚的电平
关键点:
SERVO0_PIN_SET(level):设置舵机引脚电平(高或低)。

6. TIM7中断服务函数

void TIM7_IRQHandler(void)
{
    static u8 flag = 0;
    static u8 duoji_index1 = 0;
    int temp;

    if (TIM_GetITStatus(TIM7, TIM_IT_Update) != RESET) //检查TIM更新中断发生与否
    {
        /* 通过改变重装载值和舵机下标索引,每个舵机定时2500(2.5ms),执行8个舵机后完成一个周期20000(20ms) */
        if (duoji_index1 == 8)
        {
            duoji_index1 = 0;
        }

        if (flag == 0)
        {
            TIM7->ARR = ((unsigned int)(pwmServo_angle[duoji_index1].current));
            servo_pin_set(duoji_index1, Bit_SET);
            servo_increment_offset(duoji_index1);
        }
        else
        {
            temp = 2500 - (unsigned int)(pwmServo_angle[duoji_index1].current);
            TIM7->ARR = temp;
            servo_pin_set(duoji_index1, Bit_RESET);
            duoji_index1++;
        }
        flag = !flag;

        TIM_ClearITPendingBit(TIM7, TIM_IT_Update);    //清除TIMx更新中断标志
    }
}

功能:TIM7中断服务函数,用于生成PWM信号控制舵机。
关键点:

TIM7->ARR:设置定时器自动重装值,控制PWM占空比。
servo_pin_set:设置舵机引脚电平。
servo_increment_offset:逐步调整舵机角度。

7. 舵机初始化函数

void pwmServo_init(void)
{
    servo_gpio_init();
    TIM7_Init();
}

功能:初始化舵机GPIO和TIM7定时器。

8. 舵机角度设置函数

void pwmServo_angle_set(uint8_t index, int aim, int time)
{
    uint8_t i;

    //角度数据超出范围,退出
    if ((aim > 2500) || (aim < 500))
        return;

    if (index == 255)
    {
        for (i = 0; i < SERVO_NUM; i++)
        {
            pwmServo_angle[i].aim = aim;
            pwmServo_angle[i].increment = (float)((pwmServo_angle[i].aim - pwmServo_angle[i].current) / (pwmServo_angle[i].time / 20.000));
            pwmServo_angle[i].time = time;
        }
        return;
    }

    /* 限制输入值大小 */
    if (index >= SERVO_NUM)
        return;

    if (time > 10000)
        time = 10000;

    if (time < 20) /* 执行时间太短,舵机直接以最快速度运动 */
    {
        pwmServo_angle[index].aim = aim;
        pwmServo_angle[index].current = aim;
        pwmServo_angle[index].increment = 0;
    }
    else
    {
        pwmServo_angle[index].aim = aim;
        pwmServo_angle[index].time = time;
        pwmServo_angle[index].increment = (float)((pwmServo_angle[index].aim - pwmServo_angle[index].current) / (pwmServo_angle[index].time / 20.000));
    }
}

功能:设置舵机的目标角度和执行时间。
关键点:
if (index == 255):如果索引为255,设置所有舵机的角度。
pwmServo_angle[index].increment:计算舵机角度的增量。

9. 舵机停止运动函数

void pwmServo_stop_motion(uint8_t index)
{
    uint8_t i;

    if (index == 255)
    {
        for (i = 0; i < SERVO_NUM; i++)
        {
            pwmServo_angle[i].aim = pwmServo_angle[index].current;
            pwmServo_angle[i].increment = 0.001;
        }
        return;
    }

    /* 限制输入值大小 */
    if (index >= SERVO_NUM)
        return;

    pwmServo_angle[index].aim = pwmServo_angle[index].current;
    pwmServo_angle[index].increment = 0.001;
}

功能:停止舵机运动。

10. 舵机偏差设置函数

void pwmServo_bias_set(uint8_t index, int bias)
{
    /* 限制输入值大小 */
    if (index >= SERVO_NUM)
        return;

    pwmServo_angle[index].bias = bias;
    pwmServo_angle[index].increment = 0.001;
}

功能:设置舵机的偏差值。

总结

这段代码实现了基于STM32的舵机控制功能,通过TIM7定时器生成PWM信号,控制舵机的角度和运动。代码结构清晰,功能模块化,便于维护和扩展。



http://www.kler.cn/a/582478.html

相关文章:

  • uniapp+微信小程序+最简单局部下拉刷新实现
  • 【Linux系统】进程优先级:进程间的权力游戏
  • Android : Camera之CHI API
  • 学习MDA规范_5.统一建模语言(UML)
  • Chrome 浏览器 133 版本新特性
  • 本地Git仓库搭建(DevStar)与Git基本命令
  • 20 | 如何添加单元测试用例
  • 含k个3的数(信息学奥赛一本通-1090)
  • 深度学习之目标检测/对象检测
  • Scala编程_实现Rational的基本操作
  • rust语言闭包trait类型编译器推导总结
  • Matlab深度学习ResNet、迁移学习多标签图像分类及超分辨率处理Grad CAM可视化分析COCO、手写数字数据集...
  • 大模型在甲状腺肿瘤预测及治疗方案制定中的应用研究
  • 探索DEHP暴露对小鼠心脏发育的影响:AbMole助力揭示线粒体功能障碍的奥秘
  • 每周一篇——PLG(Promtail+Loki+Grafana)轻量日志方案
  • [JAVASE] 注解
  • 大白话 Vue 中的keep - alive组件,它的作用是什么?在什么场景下使用?
  • APK文件结构与逆向工具链深度解析
  • 【BUG分析】微服务无法读取Nacos中的共享配置
  • AI模型的构建过程是怎样的(下)