基于STM32驱动SG90舵机

1.舵机介绍

这边对SG90舵机进行一个简短的文字介绍:

舵机的控制一般需要一个20ms左右的时基脉冲,该脉冲的高电平部分一般为0.5ms-2.5ms范围,总间隔为2ms。脉冲的宽度将决定马达转动的距离。例如:1.5毫秒的脉冲,电机将转向90度的位置(通常称为中立位置,对于180°舵机来说,就是90°位置)。如果脉冲宽度小于1.5毫秒,那么电机轴向朝向0度方向。如果脉冲宽度大于1.5毫秒,轴向就朝向180度方向。以180度舵机为例,对应的控制关系是这样的:

0.5ms————-0度;

1.0ms————45度;

1.5ms————90度;

2.0ms———–135度;

2.5ms———–180度;

2.程序框架

程序由bsp_config.h/c文件,bsp_advancetime1.h/c文件,main.c构成

bsp_config文件包含一个头文件与延时和位操作等程序。

bsp_advancetime1文件就是对高级定时器1的定义与PWM波产生程序的书写还有对GPIO引脚的初始化。

main.c函数不用多说

3.bsp_config.h与.c文件

#ifndef BSP_CONFIG_H
#define BSP_CONFIG_H

#include "stm32f10x.h"

//位带宏定义
#define BITBAND(addr, bitnum)   ((addr&0xF0000000) + 0x2000000 + ((addr&0xFFFFF)<<5) + (bitnum<<2))
#define MEM_ADDR(addr)          *((volatile unsigned long  *)(addr))
#define BIT_ADDR(addr, bitnum)  MEM_ADDR(BITBAND(addr, bitnum))

//IO口地址位带映射
#define GPIOA_ODR_Addr    (GPIOA_BASE+12) //GPIOA输出数据寄存器地址0x4001080C
#define GPIOB_ODR_Addr    (GPIOB_BASE+12) //GPIOB输出数据寄存器地址0x40010C0C
#define GPIOC_ODR_Addr    (GPIOC_BASE+12) //GPIOC输出数据寄存器地址0x4001100C
#define GPIOD_ODR_Addr    (GPIOD_BASE+12) //GPIOD输出数据寄存器地址0x4001140C
#define GPIOE_ODR_Addr    (GPIOE_BASE+12) //GPIOE输出数据寄存器地址0x4001180C
#define GPIOF_ODR_Addr    (GPIOF_BASE+12) //GPIOF输出数据寄存器地址0x40011A0C
#define GPIOG_ODR_Addr    (GPIOG_BASE+12) //GPIOG输出数据寄存器地址0x40011E0C

#define GPIOA_IDR_Addr    (GPIOA_BASE+8) //GPIOA输入数据寄存器地址0x40010808
#define GPIOB_IDR_Addr    (GPIOB_BASE+8) //GPIOB输入数据寄存器地址0x40010C08
#define GPIOC_IDR_Addr    (GPIOC_BASE+8) //GPIOC输入数据寄存器地址0x40011008
#define GPIOD_IDR_Addr    (GPIOD_BASE+8) //GPIOD输入数据寄存器地址0x40011408
#define GPIOE_IDR_Addr    (GPIOE_BASE+8) //GPIOE输入数据寄存器地址0x40011808
#define GPIOF_IDR_Addr    (GPIOF_BASE+8) //GPIOF输入数据寄存器地址0x40011A08
#define GPIOG_IDR_Addr    (GPIOG_BASE+8) //GPIOG输入数据寄存器地址0x40011E08

//单个IO口位带操作
#define PA_OUT(n)   BIT_ADDR(GPIOA_ODR_Addr,n)  //PAx输出
#define PA_IN(n)    BIT_ADDR(GPIOA_IDR_Addr,n)  //PAx输入

#define PB_OUT(n)   BIT_ADDR(GPIOB_ODR_Addr,n)  //PBx输出
#define PB_IN(n)    BIT_ADDR(GPIOB_IDR_Addr,n)  //PBx输入

#define PC_OUT(n)   BIT_ADDR(GPIOC_ODR_Addr,n)  //PCx输出
#define PC_IN(n)    BIT_ADDR(GPIOC_IDR_Addr,n)  //PCx输入

#define PD_OUT(n)   BIT_ADDR(GPIOD_ODR_Addr,n)  //PDx输出
#define PD_IN(n)    BIT_ADDR(GPIOD_IDR_Addr,n)  //PDx输入

#define PE_OUT(n)   BIT_ADDR(GPIOE_ODR_Addr,n)  //PEx输出
#define PE_IN(n)    BIT_ADDR(GPIOE_IDR_Addr,n)  //PEx输入

#define PF_OUT(n)   BIT_ADDR(GPIOF_ODR_Addr,n)  //PFx输出
#define PF_IN(n)    BIT_ADDR(GPIOF_IDR_Addr,n)  //PFx输入

#define PG_OUT(n)   BIT_ADDR(GPIOG_ODR_Addr,n)  //PGx输出
#define PG_IN(n)    BIT_ADDR(GPIOG_IDR_Addr,n)  //PGx输入



void delay_us(uint16_t us);
void delay_ms(uint16_t ms);

#endif
#include "bsp_config.h"


void delay_Quality_us(void)
{
	__ASM("nop");
	__ASM("nop");
	__ASM("nop");
	__ASM("nop");
	__ASM("nop");
	__ASM("nop");
	__ASM("nop");
	__ASM("nop");
}

void delay_us(uint16_t us)
{
	while(us--)
	{
		delay_Quality_us();
		delay_Quality_us();
		delay_Quality_us();
		delay_Quality_us();
	}
}

void delay_ms(uint16_t ms)
{
	while(ms--)
	{
		delay_us(1000);
	}
}

4.bsp_advancetime1.h与.c文件

#ifndef BSP_ADVANCETIMER_H
#define BSP_ADVANCETIMER_H

#include "bsp_config.h"

void TIME1Init(uint16_t arr, uint16_t psc);
void servo_angle(uint16_t angle);
	
#endif
#include "bsp_advancetimer1.h"

void TIME1Init(uint16_t arr, uint16_t psc)
{
	GPIO_InitTypeDef         GPIO_InitStructure;
	TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
	TIM_OCInitTypeDef        TIM_OCInitStructure;
 
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1, ENABLE); 
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);                                	
 
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8; 
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;  
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA, &GPIO_InitStructure);
	
	TIM_TimeBaseStructure.TIM_Period = arr-1; 
	TIM_TimeBaseStructure.TIM_Prescaler = psc-1; 
	TIM_TimeBaseStructure.TIM_ClockDivision = 0; 
	TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  
	TIM_TimeBaseInit(TIM1, &TIM_TimeBaseStructure);
 
	TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; 
	TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; 
	TIM_OCInitStructure.TIM_Pulse = 0; 
	TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; 
	TIM_OC1Init(TIM1, &TIM_OCInitStructure);  
 
	TIM_CtrlPWMOutputs(TIM1,ENABLE);	
	
	TIM_Cmd(TIM1, ENABLE); 
}

void servo_angle(uint16_t angle)
{
	uint16_t pluse; 
	if(angle <= 5)    angle = 5;
	if(angle >= 175)  angle = 175;
	//为90°时pluse等于1500us=1.5ms即转90度
	pluse = (uint16_t)((50 + angle * 100/90.0)*10);
	TIM_SetCompare1(TIM1, pluse);
  
}

5.main.c文件

#include "bsp_config.h"
#include "bsp_advancetimer1.h"

int main(void)
{
    uint8_t i;
	//20000/(72M/72)=20ms
    TIME1Init(20000,72);//(72M/72)=1us加一次,这个需要细品一下和舵机计算角度有关,一个周期为20ms
	while(1)
	{
		servo_angle(170);
	    delay_ms(1000);
        servo_angle(30);
		delay_ms(1000);
	}
}

6.结束

如果要使用PWM2来产生PWM信号的话

TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;

TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;

需要对这两行程序进行修改

PWM1->PWM2

HIGN->LOW

STM32 驱动 SG90 舵机的过程相对简单,以下是详细的步骤和原理: ### 一、SG90 舵机简介 SG90 是一种常见的小型舵机,常用于机器人、自动化设备等领域。它通过 PWM (脉宽调制)信号控制其角度位置。 #### 主要特性: - 工作电压:4.8V - 6V DC - 控制线接口通常有三根电线: - 红色:电源正极 VCC - 棕色或黑色:接地 GND - 黄色或橙色:PWM 输入信号线 - 角度范围大约是 0° 到 180° #### PWM 控制信号说明: - 转速较慢(约 0.1 秒/60 度) - 标准周期为 20ms 的方波信号; - 占空比决定了转动的角度,默认范围内占空比从 5% 至 7.5%,对应着全左转至中间位置;超过此区间则会继续向右旋转直到最大值处停止。 | 目标角度 | 方波宽度(ms) | | :-----------: | :----------: | | 中间位 | 1.5 | | 最小角 | 0.5 | | 最大角 | 2.5 | ### 二、硬件连接 将 STM32 开发板与 SG90 连接起来,具体的线路图如下所示: - 把棕色或黑色地线连到开发板的地线上; - 使用黄色或橙色的数据线接到任意一个定时器通道上对应的 GPIO 引脚(如 PA0),该引脚需要具备 PWM 功能才能正常工作。 **注意:** 如果供电不足可能会导致舵机无法启动或者动作不稳定,请确保提供的电流足够支持所有外设同时运行。 ### 三、软件编程配置 接下来是在 STM32 上编写程序以生成正确的 PWM 波形来精确地操控舵机的位置。这里给出基于 HAL 库的一个简化示例代码框架: ```c #include "main.h" #define SERVO_MIN_PULSEWIDTH 500 // 定义最小脉冲宽度 us #define SERVO_MAX_PULSEWIDTH 2500// 定义最大脉冲宽度 us #define SERVO_FREQUENCY 50 // 设置频率为 50Hz ,即每秒更新一次命令 TIM_HandleTypeDef htim3; void SystemClock_Config(void); static void MX_GPIO_Init(void); static void MX_TIM3_PWM_Init(TIM_HandleTypeDef*htim); int main(void) { HAL_Init(); SystemClock_Config(); // 初始化系统时钟 MX_GPIO_Init(); __HAL_RCC_TIM3_CLK_ENABLE(); // 启用 TIM3 时钟源 MX_TIM3_PWM_Init(&htim3); //初始化 TIM3 while (1){ SetServoPosition(&htim3,90);//设置舵机角度为 90 度 HAL_Delay(1000); SetServoPosition(&htim3,0); //设置舵机角度为 0 度 HAL_Delay(1000); } } /** * @brief 发送指定角度给舵机,并转换成相应的高低电平时间 */ void SetServoPosition(TIM_HandleTypeDef *htim,uint8_t angle){ if(angle < 0 || angle > 180)return; uint32_t pulse_width = map(angle,SERVO_MIN_PULSEWIDTH ,SERVO_MAX_PULSEWIDTH ); __HAL_TIM_SET_COMPARE(htim,TIM_CHANNEL_1,pulse_width); } ``` 其中 `map` 函数可以自定义实现为简单的比例计算函数,例如: ```c uint32_t map(uint8_t x, uint32_t in_min, uint32_t in_max){ return ((x-(in_min)) *(SERVO_MAX_PULSEWIDTH-SERVO_MIN_PULSEWIDTH)/(in_max-in_min)+SERVO_MIN_PULSEWIDTH)/2 ; } ``` 需要注意的是实际应用中最好加入一些误差修正机制提高精度并且避免超出有效范围造成损坏。 --- 以上就是如何利用 STM32 板卡去驱动并调节 SG90 类型的小型数字伺服电机的基础知识啦!
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值