stm32 PWM命令

问题描述 投票:0回答:1

我是 stm32 的初学者,也是这个社区的新人 我正在尝试以下程序:

-运行 DC 电机(使用 PWM 命令)以速度“1”并等待 5 秒,然后以速度“2”运行并等待 5 秒,然后电机停止。

问题在于电机处于循环状态:开始转动约 1 秒然后停止。

    #include "main.h"
    TIM_HandleTypeDef htim3;
    int puls ;
    float duty ;
    UART_HandleTypeDef huart2;
    void SystemClock_Config(void);
    static void MX_GPIO_Init(void);
    static void MX_USART2_UART_Init(void);
    static void MX_TIM3_Init(void);
    
    int main(void)
    {
      HAL_Init();
      SystemClock_Config();
      MX_GPIO_Init();
      MX_USART2_UART_Init();
      MX_TIM3_Init();
      /* USER CODE BEGIN 2 */
    HAL_TIM_Base_Start(&htim3);
    HAL_TIM_PWM_Start(&htim3, TIM_CHANNEL_2);
      while (1)
      {puls=150 ; // motor with speed 1
       duty =(puls*100)/31999; 
            MX_TIM3_Init();
            HAL_TIM_PWM_Start(&htim3, TIM_CHANNEL_2);
            HAL_Delay(5000);
    
      puls=300 ; //motor with  speed  2
        duty =(puls*100)/31999;
        MX_TIM3_Init();
        HAL_TIM_PWM_Start(&htim3, TIM_CHANNEL_2);
        HAL_Delay(5000);
        HAL_TIM_PWM_Stop(&htim3, TIM_CHANNEL_2);

    
      }
    
    }
static void MX_TIM3_Init(void)
{
  TIM_ClockConfigTypeDef sClockSourceConfig = {0};
  TIM_MasterConfigTypeDef sMasterConfig = {0};
  TIM_OC_InitTypeDef sConfigOC = {0};
  htim3.Instance = TIM3;
  htim3.Init.Prescaler = 31999;
  htim3.Init.CounterMode = TIM_COUNTERMODE_UP;
  htim3.Init.Period = 65535;
  htim3.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1;
  htim3.Init.AutoReloadPreload = TIM_AUTORELOAD_PRELOAD_DISABLE;
  if (HAL_TIM_Base_Init(&htim3) != HAL_OK)
  {
    Error_Handler();
  }
  sClockSourceConfig.ClockSource = TIM_CLOCKSOURCE_INTERNAL;
  if (HAL_TIM_ConfigClockSource(&htim3, &sClockSourceConfig) != HAL_OK)
  {
    Error_Handler();
  }
  if (HAL_TIM_PWM_Init(&htim3) != HAL_OK)
  {
    Error_Handler();
  }
  sMasterConfig.MasterOutputTrigger = TIM_TRGO_RESET;
  sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE;
  if (HAL_TIMEx_MasterConfigSynchronization(&htim3, &sMasterConfig) != HAL_OK)
  {
    Error_Handler();
  }
  sConfigOC.OCMode = TIM_OCMODE_PWM1;
  sConfigOC.Pulse = puls;
  sConfigOC.OCPolarity = TIM_OCPOLARITY_HIGH;
  sConfigOC.OCFastMode = TIM_OCFAST_DISABLE;
  if (HAL_TIM_PWM_ConfigChannel(&htim3, &sConfigOC, TIM_CHANNEL_2) != HAL_OK)
  {
    Error_Handler();
  }
  
  HAL_TIM_MspPostInit(&htim3);

}

我使用的是stm32f4。

电机引脚 C7 .

我已经使用 STM32CubeIDE 配置了我的项目。

感谢您的帮助。

c embedded microcontroller stm32f4 pwm
1个回答
1
投票

将初始化与占空比设置分开。变化:

sConfigOC.Pulse = puls ;

sConfigOC.Pulse = 0 ;

然后添加一个设置占空比的函数,例如:

int setDutyCycle( unsigned duty_cycle_pecent_X10 )
{
    TIM_OC_InitTypeDef sConfigOC;

    sConfigOC.OCMode = TIM_OCMODE_PWM1;
    sConfigOC.Pulse = ((htim3.Init.Period + 1) * duty_cycle_pecent_X10) / 1000 ;
    sConfigOC.OCPolarity = TIM_OCPOLARITY_HIGH;
    sConfigOC.OCFastMode = TIM_OCFAST_DISABLE;

    if (HAL_TIM_PWM_ConfigChannel(&htim3, &sConfigOC, TIM_CHANNEL_2) != HAL_OK)
    {
        Error_Handler();
    }
}

不再需要全局变量

puls
。事实上,不需要全局变量,也不是一个好主意; CubeMX 生成依赖于全局变量的代码(例如本例中的
htim3
)这一事实令人沮丧,但这是一个不同的问题 - 没有必要用你自己的方法来加剧问题。

HAL_TIM_PWM_Stop()
上的输出状态未定义,并且可以使电机保持全速。这在您的代码中不会发生,因为停止循环后会进行迭代并立即再次启动它。停止电机运行的正确方法是将占空比设置为零。

然后你的

main()
就可以被简化:

int main (void)
{
    HAL_Init ();
    SystemClock_Config ();
    MX_GPIO_Init ();
    MX_USART2_UART_Init ();
    MX_TIM3_Init ();
    
    while (1)
    {
        // Duty cycle 20% for 5 seconds
        speed_percent_x10 = 200u ;           
        setDutyCycle( speed_percent_x10 ) ;
        HAL_Delay (5000u);
        
        // Duty cycle 50% for 5 seconds
        speed_percent_x10 = 500u ;           
        setDutyCycle( speed_percent_x10 ) ;
        HAL_Delay (5000u);
        
        // Duty cycle 0% (stop) for 5 seconds
        speed_percent_x10 = 0u ;             
        setDutyCycle( speed_percent_x10 ) ;
        HAL_Delay (5000u);
    }
    
    return 0 ;
}

这可能无法解决代码的所有问题;我无法测试它,但我严重怀疑你的 PWM 频率是否足够高以正确稳定地驱动电机。

CubeMX 代码在这方面特别差,我建议修改

MX_TIM3_Init()
如下:

htim3.Init.Prescaler = 0 ;  // run at SystemCoreClock / 2 (APB1 bus clock)
htim3.Init.Period = SystemCoreClock / (2 * PWM_FREQ) ;

其中

PWM_FREQ
是明智的,例如:

#define PWM_FREQ 20000u

如果您有精确的电机数据,理论上您可以计算出最佳频率,但通常足以确保听不到它。当然不是您的设置似乎使用的低于 10Hz(取决于

SystemCoreClock
)。

© www.soinside.com 2019 - 2024. All rights reserved.