STM32驱动SG90舵机与HC_SR04超声波模块

前言


前言

这篇文章的内容主要对一个超声波+舵机小项目进行总结,记录学习到的知识以及遇到的问题,方便后续对知识的回溯与更新,如果能帮大家少走弯路也是极好的。如果有错误还请指正,谢谢~~


一、项目涉及的内容

本项实验的硬件组成有STM32F103C8T6开发板、USB转TTL串口模块(CH340)、超声波模块(HC-SR04)、舵机模块(SG90 180度),设计到的软件模块有GPIO和AFIO、通用定时器TIM2、中断、系统定时器SysTick,接下来对这些主要模块进行讲解,回顾项目的重点,希望大家也能有所收获。

项目简介

通过超声波模块进行测距,根据距离来操纵舵机运动实时反馈。


二、模块实操

1. SysTick系统定时器模块

我们使用 STM32 的内部 SysTick 来实现延时的,要进行延时的时候开启SysTick,延时时间到就直接把SysTick失能,这样就不需要一直占用CPU的执行。

代码如下:

void ms_delay(uint32_t ms)
{
    uint32_t i;
    SysTick_Config(72000); //一个循环相当于1毫秒
    for(i=0;i<ms;i++)
    {
        while( !((SysTick->CTRL)&(1<<16)) );//判断是否一次计数结束
        
    }
    SysTick->CTRL &= ~(SysTick_CTRL_ENABLE_Msk); //循环结束后让系统定时器失能  
}

我们先看一下两段对寄存器直接操作的代码,这段有点复杂难懂,但其实就是他们的作用也比较简单:

  1. while( !((SysTick->CTRL)&(1<<16)) );用来判断CTRL这个SysTick 控制及状态寄存器的第16位是否为1,不为1就在while循环体内持续等待直到一次计数周期结束。
  2. SysTick->CTRL &= ~(SysTick_CTRL_ENABLE_Msk);操作CTRL寄存器的第0位为0来关闭系统定时器,SysTick_CTRL_ENABLE_Msk是系统的宏定义,其实就是1<<0。

 【野火®】零死角玩转STM32—F103霸道.pdf-->19.2 SysTick 寄存器介绍-->表 19-2 SysTick 控制及状态寄存器

定时时间计算:

计数器每计数一次的时间为 1/SYSCLK,一般我们设置系统时钟 SYSCLK 等于 72M。当重装载数值寄存器的值递减到 0 的时候,系统定时器就产生一次中断,只要不把它在 SysTick 控制及状态寄存器中的使能位清除,就以此循环往复。

因此只要知道计数的次数就可以准确得到它的延时时间,STM32 的内核库已经提供了这个功能,只要配置内置的SysTick_Config()即可实现,SysTick_Config()函数的形参用来设置重装载寄存器的值。

SysTick_Config(72000)代表:72000*(1/72MHz) = 1/1000 = 1(ms)。即定时为1ms。

 同理,我们可以得出周期为1微秒的系统定时器的写法,72*(1/72MHz) = 1/1000000 = 0.001(ms) = 1(us)。

void us_delay(uint32_t us)
{
    uint32_t i;
    SysTick_Config(72); //一个循环相当于1微秒
    for(i=0;i<us;i++)
    {
        while( !((SysTick->CTRL)&(1<<16)) );
        
    }
    SysTick->CTRL &= ~(SysTick_CTRL_ENABLE_Msk); //循环结束后让系统定时器失能    
    
}

2. SG90 舵机模块

舵机是由PWM波来控制的一个器件,不同的PWM波的高电平时间对应的是舵机不同的位置,也就是说,你输出一个一定占空比的PWM波,对应的就是舵机确定好的一个位置。 </

  • 4
    点赞
  • 80
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
首先,你需要一个STM32开发板和一个SG90舵机模块。然后,你需要将开发板和舵机模块连接起来。 连接舵机模块的信号线到STM32的一个PWM引脚,比如PA0。接下来,将舵机模块的电源和地线连接到STM32的电源和地线。 接下来,你需要编写一个控制舵机模块的程序。以下是一个简单的示例代码: ``` #include "stm32f10x.h" #define SG90_PERIOD 20000 #define SG90_MIN_PULSE 1000 #define SG90_MAX_PULSE 2000 void delay_ms(uint32_t ms) { for (uint32_t i = 0; i < ms * 1000; i++); } int main(void) { RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_AFIO, ENABLE); RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); GPIO_InitTypeDef GPIO_InitStructure; GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOA, &GPIO_InitStructure); TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; TIM_TimeBaseStructure.TIM_Period = SG90_PERIOD - 1; TIM_TimeBaseStructure.TIM_Prescaler = 72 - 1; TIM_TimeBaseStructure.TIM_ClockDivision = 0; TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure); TIM_OCInitTypeDef TIM_OCInitStructure; TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; TIM_OCInitStructure.TIM_Pulse = SG90_MIN_PULSE; TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; TIM_OC1Init(TIM2, &TIM_OCInitStructure); TIM_Cmd(TIM2, ENABLE); while (1) { for (uint16_t i = SG90_MIN_PULSE; i <= SG90_MAX_PULSE; i += 10) { TIM_SetCompare1(TIM2, i); delay_ms(10); } for (uint16_t i = SG90_MAX_PULSE; i >= SG90_MIN_PULSE; i -= 10) { TIM_SetCompare1(TIM2, i); delay_ms(10); } } } ``` 这段代码使用了STM32的定时器和PWM模块来控制舵机模块。首先,它初始化了定时器和PWM模块,然后使用一个循环来让舵机缓慢转动。在循环中,每隔10毫秒改变一次PWM脉冲宽度,从而控制舵机的角度。 你可以根据需要修改SG90_MIN_PULSE和SG90_MAX_PULSE的值来控制舵机的最小和最大脉冲宽度,从而控制舵机的行程。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值