STM32F103基于HAL工程TIM4多通道PWM固定占空比连续和单脉冲输出

STM32F103基于HAL工程TIM4多通道PWM固定占空比输出


✨在测试过程中发现配置的2路PWM总是只有一个固定的通道有输出,于是将该功能输出引脚复用到其他引脚,输出波形有,确定是芯片引脚的问题,后面换了一块板测试,确定是芯片引脚确实有问题,刚开始一直没有怀疑是IO口本身的问题。
  • 📋通过STM32cubemx配置好定时器以及选择好通道后,在keil中只需要开启定时器以及设置占空比比较值就可以完成固定占空比的工程配置。
  • 🌿PB8 -> TIM4_CH3
    在这里插入图片描述
  • 🌿PB9 -> TIM4_CH4
    在这里插入图片描述

⛳注意事项

  • 🌿在使用定时器单脉冲输出模式下,STM32CUMX配置后,调用HAL_TIM_OnePulse_Start_ITHAL_TIM_OnePulse_Start前,还需要使能定时器,__HAL_TIM_ENABLE(&htim1)

🛠STM32CUMX定时器配置

  • 🌿选择定时器4,选择对应通道3和通道4
    在这里插入图片描述
  • 🔖如果上面使用TIM4的通道1和通道2,可以将其引脚映射到PB6和PB7引脚上,
    在这里插入图片描述
  • 🌿配置具体频率以及计数值相关参数
    在这里插入图片描述
void MX_TIM4_Init(void)
{

  /* USER CODE BEGIN TIM4_Init 0 */

  /* USER CODE END TIM4_Init 0 */

  TIM_ClockConfigTypeDef sClockSourceConfig = {0};
  TIM_MasterConfigTypeDef sMasterConfig = {0};
  TIM_OC_InitTypeDef sConfigOC = {0};

  /* USER CODE BEGIN TIM4_Init 1 */

  /* USER CODE END TIM4_Init 1 */
  htim4.Instance = TIM4;
  htim4.Init.Prescaler = 72-1;//分频系数72分频
  htim4.Init.CounterMode = TIM_COUNTERMODE_UP;//向上计数方式
  htim4.Init.Period = 500-1;
  htim4.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1;//时钟不分频
  htim4.Init.AutoReloadPreload = TIM_AUTORELOAD_PRELOAD_ENABLE;//分频系数72分频
  if (HAL_TIM_Base_Init(&htim4) != HAL_OK)//初始化定时器
  {
    Error_Handler();
  }
  sClockSourceConfig.ClockSource = TIM_CLOCKSOURCE_INTERNAL;
  if (HAL_TIM_ConfigClockSource(&htim4, &sClockSourceConfig) != HAL_OK)
  {
    Error_Handler();
  }
  if (HAL_TIM_PWM_Init(&htim4) != HAL_OK)
  {
    Error_Handler();
  }
  sMasterConfig.MasterOutputTrigger = TIM_TRGO_RESET;
  sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE;
  if (HAL_TIMEx_MasterConfigSynchronization(&htim4, &sMasterConfig) != HAL_OK)
  {
    Error_Handler();
  }
  sConfigOC.OCMode = TIM_OCMODE_PWM2;//TIMx_CNT>TIMx_CCRn时,输出有效电平
  sConfigOC.Pulse = 0;//占空比
  sConfigOC.OCPolarity = TIM_OCPOLARITY_HIGH; //输出极性:TIM输出比较极性高
  sConfigOC.OCFastMode = TIM_OCFAST_DISABLE;//快速比较使能
  if (HAL_TIM_PWM_ConfigChannel(&htim4, &sConfigOC, TIM_CHANNEL_3) != HAL_OK)
  {
    Error_Handler();
  }
  sConfigOC.OCMode = TIM_OCMODE_PWM1;//TIMx_CNT<TIMx_CCRn时,输出有效电平
  if (HAL_TIM_PWM_ConfigChannel(&htim4, &sConfigOC, TIM_CHANNEL_4) != HAL_OK)
  {
    Error_Handler();
  }
  /* USER CODE BEGIN TIM4_Init 2 */

  /* USER CODE END TIM4_Init 2 */
  HAL_TIM_MspPostInit(&htim4);

}

📑Keil中开启定时器4以及设置占空比

  1. 在tim.c中,定时器初始化函数中开启定时器计时。
void HAL_TIM_MspPostInit(TIM_HandleTypeDef* timHandle)
{

    GPIO_InitTypeDef GPIO_InitStruct = {0};
    if(timHandle->Instance == TIM4)
    {
        /* USER CODE BEGIN TIM4_MspPostInit 0 */

        /* USER CODE END TIM4_MspPostInit 0 */

        __HAL_RCC_GPIOB_CLK_ENABLE();
        /**TIM4 GPIO Configuration
        PB8     ------> TIM4_CH3
        PB9     ------> TIM4_CH4
        */
        GPIO_InitStruct.Pin = GPIO_PIN_8 | GPIO_PIN_9;
        GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
        GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
        HAL_GPIO_Init(GPIOB, &GPIO_InitStruct);

        /* USER CODE BEGIN TIM4_MspPostInit 1 */
        HAL_TIM_PWM_Start(timHandle, TIM_CHANNEL_3); //开启PWM输出通道3,PB8 -->2KHz
        HAL_TIM_PWM_Start(timHandle, TIM_CHANNEL_4); //开启PWM输出通道4,PB9 -->2KHz
        /* USER CODE END TIM4_MspPostInit 1 */
    }

}
  1. 在main主函数中设置或重新更新改变占空比的比较值。
int main(void)
{
  /* USER CODE BEGIN 1 */
//	uint8_t flag,i;
  /* USER CODE END 1 */

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_TIM4_Init();
  /* USER CODE BEGIN 2 */

  /* USER CODE END 2 */

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

    /* USER CODE BEGIN 3 */
		htim4.Instance->CCR4=250;//输出固定占空比
		htim4.Instance->CCR3=350;
//        __HAL_TIM_SET_COMPARE(&htim4, TIM_CHANNEL_4, 250); //设置CH4->PWM脉冲宽度,同上
//        __HAL_TIM_SET_COMPARE(&htim4, TIM_CHANNEL_3, (150)); //设置CH3->PWM脉冲宽度
    }
  /* USER CODE END 3 */
}

📓输出极性和电平状态关系

  • 🌿如果设置的输出极性为High,则函数__HAL_TIM_SetCompare(&htimx, TIM_CHANNEL_2, pluse)中的,第三个形参就代表输出的占空比值。
    在这里插入图片描述
 __HAL_TIM_SET_AUTORELOAD(&htim4, 5000 - 1); 
 __HAL_TIM_SetCompare(&htim4, TIM_CHANNEL_1, 4800);

在这里插入图片描述

📚程序源码
链接: https://pan.baidu.com/s/1kNZLOGtXkgX9th21qtxnPg
提取码: knm8

📗单脉冲输出

✨所谓的单脉冲输出方式,就是在启动执行 HAL_TIM_PWM_Start(&htim4, TIM_CHANNEL_1);函数后,对应的PWM输出通道上的引脚上只会产生一个脉冲信号。(固定占空比的PWM信号是由连续输出的脉冲信号组成的。)由于是单脉冲输出方式,信号每次产生时时,都需要调用 HAL_TIM_PWM_Start(&htim4, TIM_CHANNEL_1);然后延时一段超过脉冲信号宽度的时间,再调用HAL_TIM_PWM_Stop(&htim4, TIM_CHANNEL_1);以表示单脉冲信号输出结束。其中间所加的延时时间,如果短于所产生的脉宽信号宽度时,脉冲信号会直接截断,所以延时时间一定要预留足够。
  • 🔧在STM32CubeMX配置定时器是勾选One Pulse Mode.

在这里插入图片描述

  • 🔖对应生成的配置代码:
if (HAL_TIM_OnePulse_Init(&htim4, TIM_OPMODE_SINGLE) != HAL_OK)
  {
    Error_Handler();
  }
  • 📐调用方法:
__HAL_TIM_ENABLE(&htim4);//启动定时器
/*配置TIM1 单脉冲启动*/
  if (HAL_TIM_OnePulse_Start(&htim4, TIM_CHANNEL_1) != HAL_OK)
  {
    Error_Handler();
  }
    if (HAL_TIM_OnePulse_Start(&htim4, TIM_CHANNEL_2) != HAL_OK)
  {
    Error_Handler();
  }
  
  • 🔖另外一种启动方式:
        HAL_TIM_PWM_Start(&htim4, TIM_CHANNEL_1); //启动
        HAL_TIM_PWM_Start(&htim4, TIM_CHANNEL_2);
        HAL_Delay(500);  //中间的延时不能过短, 否则输出被截断                                                             
        HAL_TIM_PWM_Stop(&htim4, TIM_CHANNEL_1);
        HAL_TIM_PWM_Stop(&htim4, TIM_CHANNEL_2);
        HAL_Delay(1000);
  • ⚡下图中,白色波形是被截断不正常的波形:
    在这里插入图片描述
  • 1
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,我可以向您提供一些关于如何使用STM32F103HAL库来实现pwm控制蜂鸣器的指导。 首先,您需要配置使用的定时器并设置其为pwm模式。在HAL库中,您可以使用以下代码段来完成此操作: ```C // 配置TIM1为PWM模式,通道1 TIM_HandleTypeDef htim1; TIM_OC_InitTypeDef sConfigOC; htim1.Instance = TIM1; htim1.Init.Prescaler = 0; htim1.Init.CounterMode = TIM_COUNTERMODE_UP; htim1.Init.Period = 500; htim1.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1; HAL_TIM_PWM_Init(&htim1); sConfigOC.OCMode = TIM_OCMODE_PWM1; sConfigOC.Pulse = 250; HAL_TIM_PWM_ConfigChannel(&htim1, &sConfigOC, TIM_CHANNEL_1); ``` 在这里,我们使用TIM1和通道1来实现PWM控制。`sConfigOC.OCMode`设置了PWM模式,Pulse是占空比的值。 接下来,您需要打开定时器并设置蜂鸣器的控制引脚。您可以使用以下代码段来实现这一点: ```C // 打开TIM1,开始PWM信号输出 HAL_TIM_PWM_Start(&htim1, TIM_CHANNEL_1); // 设置Buzzer所在引脚为输出,初始值为0 GPIO_InitTypeDef GPIO_InitStruct; GPIO_InitStruct.Pin = GPIO_PIN_0; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; GPIO_InitStruct.Pull = GPIO_NOPULL; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); HAL_GPIO_WritePin(GPIOA, GPIO_PIN_0, GPIO_PIN_RESET); ``` 在这里,我们使用`GPIOA_PIN_0`来控制蜂鸣器。`HAL_TIM_PWM_Start`打开了PWM信号输出,`GPIO_InitStruct`用来设置蜂鸣器所在引脚的输出模式,`HAL_GPIO_Init`用来初始设置该引脚的输出状态。 最后,您可以使用以下代码段来实现PWM信号的控制: ```C // 设置蜂鸣器的占空比 sConfigOC.Pulse = 250; HAL_TIM_PWM_ConfigChannel(&htim1, &sConfigOC, TIM_CHANNEL_1); // 开启或关闭蜂鸣器 HAL_GPIO_WritePin(GPIOA, GPIO_PIN_0, GPIO_PIN_SET); // 开启 HAL_GPIO_WritePin(GPIOA, GPIO_PIN_0, GPIO_PIN_RESET); // 关闭 ``` 在这里,`sConfigOC.Pulse`用来设置占空比。若要使蜂鸣器发声,则需要将GPIO引脚设置为高电平,否则将其设置为低电平。 希望这些代码对您有所帮助!

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值