STM32CubeMx配置HAL库PWM

PWM简介

PWM(Pulse Width Modulation)是脉冲宽度调制的缩写,是一种利用微处理器的数字输出来对模拟电路进行控制的技术。PWM的原理是

通过调节占空比来调节脉冲宽度,从而改变输出电压的大小。波形图如下
image-20230822103209917

PWM的两个重要参数为频率占空比。频率是周期的导数即 f = 1 / T f=1/T f=1/T,占空比是指脉宽时间占周期的比例。

STM32中用定时器来输出PWM,其原理图如下

image-20230822103605225

定时器重复计数从0到ARR,设定比较值为CCRx,当定时器的计数值向上计数到和CCRx的值相等时,对应波形输出管脚电瓶翻转,

当定时器计数值计到ARR时,输出管脚波形再次翻转。从图中可以看出,定时器的一个计时周期对应一个 PWM周期,脉宽(高电平的宽

度)需要根据输出管脚的初始电平的不同或者为0—CCRx这一段,或者为CCRx—ARR,这一段。改变CCRx的值即可实现脉宽的控制。

PWM频率和占空比的影响因素有:

  • ARR :决定PWM周期(在系统时钟频率固定的情况下)
  • CCRx:决定PWM占空比(高低电平所占整个周期比例)

硬件

  • stm32f103c8t6
  • tb6612fng
  • 直流电机
硬件引脚作用
电机1APB12普通IO
电机1BPB13普通IO
电机2APB14普通IO
电机2BPB15普通IO
PWMAPA6TIM3_CH1
PWMBPA7TIM3_CH2

软件

STM32CubeMX

  • RCC->HSE->Crystal/Ceramic Resonator
  • SYS->Debug->Serial Wire
  • GPIO

image-20230822104850113

  • TIM3->Channel1->PWM Generation CH1
  • TIM3->Channel2->PWM Generation CH1

image-20230822105013360

预分频系数(PSC)设置为144-1,自动重装载值为1000-1,PWM模式1,向上计数,则计时器时钟频率为 72 M / 144 / 1000 = 500 H z 72M/144/1000=500Hz 72M/144/1000=500Hz,周期为 T = 1 / 500 = 2 m s T=1/500=2ms T=1/500=2ms

  • Clock Configuration->HCLK->72
  • GRNRATE CODE

CLion

在Core/Inc新建motor.h

#ifndef STM32_MOTOR_H
#define STM32_MOTOR_H
#include "main.h"
#include "tim.h"

#define AIN1_RESET HAL_GPIO_WritePin(AIN1_GPIO_Port, AIN1_Pin, GPIO_PIN_RESET)
#define AIN2_RESET HAL_GPIO_WritePin(AIN2_GPIO_Port, AIN2_Pin, GPIO_PIN_RESET)
#define AIN1_SET HAL_GPIO_WritePin(AIN1_GPIO_Port, AIN1_Pin, GPIO_PIN_SET)
#define AIN2_SET HAL_GPIO_WritePin(AIN2_GPIO_Port, AIN2_Pin, GPIO_PIN_SET)

#define BIN1_RESET HAL_GPIO_WritePin(BIN1_GPIO_Port, BIN1_Pin, GPIO_PIN_RESET)
#define BIN2_RESET HAL_GPIO_WritePin(BIN2_GPIO_Port, BIN2_Pin, GPIO_PIN_RESET)
#define BIN1_SET HAL_GPIO_WritePin(BIN1_GPIO_Port, BIN1_Pin, GPIO_PIN_SET)
#define BIN2_SET HAL_GPIO_WritePin(BIN2_GPIO_Port, BIN2_Pin, GPIO_PIN_SET)

void Limit_PWM(int *motorA, int *motorB);
void Load_PWM(int Left_Motor, int Right_Motor);
#endif /* MOTOR_MOTOR_H_ */

在Core/Inc新建motor.c

#include "motor.h"

/**
 *    @brief 控制电机进行正转、反转、停止
 *    @param None
 *    @retval None
 */
void LeftMotor_Go() //左电机正转 AIN输出相反电平  BIN也输出相反电平
{
    AIN1_RESET;
    AIN2_SET;
}
void LeftMotor_Back()  //左电机反转
{
    AIN1_SET;
    AIN2_RESET;
}
void LeftMotor_Stop()  //左电机停止 AIN和BIN输出相同电平
{
    AIN1_RESET;
    AIN2_RESET;
}
void RightMotor_Go() //右电机正转 AIN输出相反电平  BIN也输出相反电平
{
    BIN1_SET;
    BIN2_RESET;
}
void RightMotor_Back()  //右电机反转
{
    BIN1_RESET;
    BIN2_SET;
}
void RightMotor_Stop()  //右电机停止 AIN和BIN输出相同电平
{
    BIN1_RESET;
    BIN2_RESET;
}
/**
 *    @brief 绝对值
 *    @param int整数
 *    @retval int整数的绝对值
 */
int GFP_Abs(int p)
{
    int q;
    q=p>0?p:(-p);
    return q;
}

/**
 *    @brief 控制电机进行PWM限速
 *    @param 左右电机的PWM值
 *    @retval None
 */
void Limit_PWM(int *motorA, int *motorB)
{
    if(*motorA>PWM_MAX) *motorA=PWM_MAX;
    if(*motorA<PWM_MIN) *motorA=PWM_MIN;

    if(*motorB>PWM_MAX) *motorB=PWM_MAX;
    if(*motorB<PWM_MIN) *motorB=PWM_MIN;
}

/**
 *    @brief 控制电机进行速度方向控制
 *    @param 左右电机的PWM值
 *    @retval None
 */
void Load_PWM(int Left_Motor, int Right_Motor)
{
    if(Left_Motor>0) LeftMotor_Go();
    else LeftMotor_Back();
    __HAL_TIM_SET_COMPARE(&htim3, TIM_CHANNEL_2, GFP_Abs(Right_Motor));

    if(Right_Motor>0) RightMotor_Go();
    else RightMotor_Back();
    __HAL_TIM_SET_COMPARE(&htim3, TIM_CHANNEL_1, GFP_Abs(Left_Motor));
}

main.c

/* USER CODE BEGIN Includes */
#include "motor.h"
/* USER CODE END Includes */
int PWM_MAX = 1000, PWM_MIN = -1000;
void SystemClock_Config(void);
int main(void)
{
  HAL_Init();
  SystemClock_Config();
  MX_GPIO_Init();
  MX_TIM3_Init();
  /* USER CODE BEGIN 2 */
  HAL_TIM_PWM_Start(&htim3, TIM_CHANNEL_1);
  HAL_TIM_PWM_Start(&htim3, TIM_CHANNEL_2);

  /* USER CODE BEGIN WHILE */
  while (1)
  {
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
      Load_PWM(800, 800);
      HAL_Delay(2000);
      Load_PWM(500, 500);
      HAL_Delay(2000);
      Load_PWM(-800, -800);
      HAL_Delay(2000);
      Load_PWM(0, 0);
      HAL_Delay(2000);
  }
  /* USER CODE END 3 */
}
  • 1
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是使用STM32HAL控制电机的PWM示例代码: 1. 首先,需要在CubeMX配置PWM通道和定时器。在Pinout & Configuration选项卡中,选择需要使用的定时器和PWM通道,并将其配置PWM输出模式。然后在Configuration选项卡中,配置定时器的时钟源和频率,以及PWM的周期和占空比。 2. 在代码中,需要包含stm32f4xx_hal.h头文件,并初始化定时器和PWM通道。以下是一个示例代码: ```c #include "stm32f4xx_hal.h" TIM_HandleTypeDef htim2; TIM_OC_InitTypeDef sConfigOC; void MX_TIM2_Init(void) { TIM_MasterConfigTypeDef sMasterConfig = {0}; htim2.Instance = TIM2; htim2.Init.Prescaler = 0; htim2.Init.CounterMode = TIM_COUNTERMODE_UP; htim2.Init.Period = 999; htim2.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1; if (HAL_TIM_Base_Init(&htim2) != HAL_OK) { Error_Handler(); } sMasterConfig.MasterOutputTrigger = TIM_TRGO_RESET; sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE; if (HAL_TIMEx_MasterConfigSynchronization(&htim2, &sMasterConfig) != HAL_OK) { Error_Handler(); } } void HAL_TIM_PWM_MspInit(TIM_HandleTypeDef* tim_pwmHandle) { GPIO_InitTypeDef GPIO_InitStruct = {0}; if(tim_pwmHandle->Instance==TIM2) { __HAL_RCC_TIM2_CLK_ENABLE(); __HAL_RCC_GPIOA_CLK_ENABLE(); /**TIM2 GPIO Configuration PA0/WKUP ------> TIM2_CH1 */ GPIO_InitStruct.Pin = GPIO_PIN_0; GPIO_InitStruct.Mode = GPIO_MODE_AF_PP; GPIO_InitStruct.Pull = GPIO_NOPULL; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW; GPIO_InitStruct.Alternate = GPIO_AF1_TIM2; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); sConfigOC.OCMode = TIM_OCMODE_PWM1; sConfigOC.Pulse = 500; sConfigOC.OCPolarity = TIM_OCPOLARITY_HIGH; sConfigOC.OCFastMode = TIM_OCFAST_DISABLE; if (HAL_TIM_PWM_ConfigChannel(&htim2, &sConfigOC, TIM_CHANNEL_1) != HAL_OK) { Error_Handler(); } HAL_TIM_PWM_Start(&htim2, TIM_CHANNEL_1); } } int main(void) { HAL_Init(); MX_TIM2_Init(); while (1) { // 设置占空比 __HAL_TIM_SET_COMPARE(&htim2, TIM_CHANNEL_1, 500); HAL_Delay(1000); __HAL_TIM_SET_COMPARE(&htim2, TIM_CHANNEL_1, 250); HAL_Delay(1000); } } ``` 在上面的代码中,我们使用了TIM2通道1和PA0引脚来控制电机。我们使用HAL_TIM_PWM_ConfigChannel函数来配置PWM通道,并使用HAL_TIM_PWM_Start函数来启动PWM输出。在主循环中,我们使用__HAL_TIM_SET_COMPARE函数来设置PWM的占空比,从而控制电机的转速。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值