基于STM32同一个定时器TIM1产生多路不同频率的时钟信号。


前言

有时我们可能会遇到需要输出多个不同频率的波形,而外设资源又没法使用多个定时器的情况。此时我们可以使用定时器比较输出的电平翻转模式,利用这个模式即可实现单个定时器产生最多四个通道的不同频率的方波。现在就通过多通道输出比较实验来讲解如何使用。

一、实验环境

1、开发板:无
2、开发工具:STM32CubeMX、MDK5.36.0.0
3、单片机型号:STM32F103ZET6

二、使用步骤

1.STM32CubeMX配置

本实验采用TIM1的4个输出通道,配置4路频率。
TIM1采用内部时钟72分频,则TIM1的时钟CK_CNT = 1MHz = 1000000Hz。
通道x频率计算:fx = 1000000 / OC_PULSEx / 2。
注意:,OC_PULSEx 等4个参数在本实验中不可配置为倍数关系。因为中断回调函数逻辑关系会造成某些通道在运行时不能有效更新通道的pulse值,致使输出频率错误。所以本实验中的OC_PULSEx 采用了质数。

具体的配置如下图:

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

2.代码

mian.c:

int main(void)
{
  /* USER CODE BEGIN 1 */

  /* 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_TIM1_Init();
  /* USER CODE BEGIN 2 */
  HAL_TIM_OC_Start_IT(&htim1, TIM_CHANNEL_1);
  HAL_TIM_OC_Start_IT(&htim1, TIM_CHANNEL_2);
  HAL_TIM_OC_Start_IT(&htim1, TIM_CHANNEL_3);
  HAL_TIM_OC_Start_IT(&htim1, TIM_CHANNEL_4);
  /* USER CODE END 2 */

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

    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */
}

tim.c比较非阻塞模式下的回调

void HAL_TIM_OC_DelayElapsedCallback(TIM_HandleTypeDef *htim)
{
  uint16_t count;	
  
  /*获取当前计数值*/
  count = __HAL_TIM_GET_COUNTER(htim);
	if(count > 0xff00)
		__NOP();
  
  /*判断触发中断的输入通道并设置新的比较值*/
  if(htim->Channel == HAL_TIM_ACTIVE_CHANNEL_1)
  {
		if(count + OC_PULSE1 <= 65535)
		{
			__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_1, count + OC_PULSE1);
		}
		else
		{
			__HAL_TIM_SET_COUNTER(&htim1, 0);
			__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_1, OC_PULSE1);
			if(OC_PULSE2 - 65535 - count >= 0)
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_2, OC_PULSE2 - 65535 - count);
			}
			else
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_2, OC_PULSE2);
			}
			if(OC_PULSE3 - 65535 - count >= 0)
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_3, OC_PULSE3 - 65535 - count);
			}
			else
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_3, OC_PULSE3);
			}
			if(OC_PULSE4 - 65535 - count >= 0)
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_4, OC_PULSE4 - 65535 - count);
			}
			else
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_4, OC_PULSE4);
			}
		}
  }
  if(htim->Channel == HAL_TIM_ACTIVE_CHANNEL_2)
  {
		if(count + OC_PULSE2 <= 65535)
		{
			__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_2, count + OC_PULSE2);
		}
		else
		{
			__HAL_TIM_SET_COUNTER(&htim1, 0);
			__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_2, OC_PULSE2);
			if(OC_PULSE1 - 65535 - count >= 0)
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_1, OC_PULSE1 - 65535 - count);
			}
			else
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_1, OC_PULSE1);
			}
			if(OC_PULSE3 - 65535 - count >= 0)
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_3, OC_PULSE3 - 65535 - count);
			}
			else
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_3, OC_PULSE3);
			}
			if(OC_PULSE4 - 65535 - count >= 0)
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_4, OC_PULSE4 - 65535 - count);
			}
			else
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_4, OC_PULSE4);
			}
		}
  }
  if(htim->Channel == HAL_TIM_ACTIVE_CHANNEL_3)
  {
		if(count + OC_PULSE3 <= 65535)
		{
			__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_3, count + OC_PULSE3);
		}
		else
		{
			__HAL_TIM_SET_COUNTER(&htim1, 0);
			__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_3, OC_PULSE3);
			if(OC_PULSE1 - 65535 - count >= 0)
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_1, OC_PULSE1 - 65535 - count);
			}
			else
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_1, OC_PULSE1);
			}
			if(OC_PULSE2 - 65535 - count >= 0)
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_2, OC_PULSE2 - 65535 - count);
			}
			else
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_2, OC_PULSE2);
			}
			if(OC_PULSE4 - 65535 - count >= 0)
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_4, OC_PULSE4 - 65535 - count);
			}
			else
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_4, OC_PULSE4);
			}
		}
  }
  if(htim->Channel == HAL_TIM_ACTIVE_CHANNEL_4)
  {
		if(count + OC_PULSE4 <= 65535)
		{
			__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_4, count + OC_PULSE4);
		}
    else
		{
			__HAL_TIM_SET_COUNTER(&htim1, 0);
			__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_4, OC_PULSE4);
			if(OC_PULSE1 - 65535 - count >= 0)
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_1, OC_PULSE1 - 65535 - count);
			}
			else
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_1, OC_PULSE1);
			}
			if(OC_PULSE2 - 65535 - count >= 0)
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_2, OC_PULSE2 - 65535 - count);
			}
			else
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_2, OC_PULSE2);
			}
			if(OC_PULSE3 - 65535 - count >= 0)
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_3, OC_PULSE3 - 65535 - count);
			}
			else
			{
				__HAL_TIM_SET_COMPARE(htim, TIM_CHANNEL_3, OC_PULSE3);
			}
		}
  }
}

三、MDK仿真配置及实验结果

本实验采用模拟仿真,需要通过MDK的逻辑分析仪观察实验结果。特别注意:MDK版本逻辑分析仪目前不支持F4系列,其他系列未验证。
具体步骤及配置如下:

1、调试设置如下图

特别注意步骤4的修改,否则无法正常在逻辑分析仪中添加引脚。
在这里插入图片描述

2、MDK逻辑分析仪配置如下图

第4步可以直接输入PORTE.14,按回车后MDK自动变为第4步上面输入的格式。
在这里插入图片描述
最后一步设置如下,设置关闭后单击close按键,然后全速运行代码即可在逻辑分析仪中查看最终波形。
在这里插入图片描述

2、MDK逻辑分析仪输出的实验结果如下:

在这里插入图片描述

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
可以使用STM32F1的多个定时器来输出多路不同频率及占空比的PWM信号。 首先,需要配置定时器时钟源和分频系数。然后,设置定时器的计数值和自动重载值,以确定PWM信号频率。接着,设置定时器的比较输出模式和比较值,以确定PWM信号的占空比。最后,使能定时器的输出通道,即可输出PWM信号。 以下是一个使用STM32F1的定时器1和2输出两路不同频率及占空比的PWM信号的示例代码: ```c #include "stm32f10x.h" void TIM1_PWM_Init(uint16_t arr, uint16_t psc) { GPIO_InitTypeDef GPIO_InitStructure; TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; TIM_OCInitTypeDef TIM_OCInitStructure; // 打开GPIO和定时器时钟 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_AFIO | RCC_APB2Periph_TIM1, ENABLE); // 配置PA8和PA9为推挽输出模式 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8 | GPIO_Pin_9; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOA, &GPIO_InitStructure); // 定时器时钟源为APB2时钟,分频系数为psc TIM_TimeBaseStructure.TIM_Period = arr; TIM_TimeBaseStructure.TIM_Prescaler = psc; TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; TIM_TimeBaseInit(TIM1, &TIM_TimeBaseStructure); // 配置TIM1_CH1和TIM1_CH2为PWM输出模式 TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; TIM_OCInitStructure.TIM_Pulse = arr / 2; // 初始占空比设为50% TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; TIM_OC1Init(TIM1, &TIM_OCInitStructure); TIM_OC2Init(TIM1, &TIM_OCInitStructure); // 使能TIM1_CH1和TIM1_CH2的输出通道 TIM_CtrlPWMOutputs(TIM1, ENABLE); // 启动定时器1 TIM_Cmd(TIM1, ENABLE); } void TIM2_PWM_Init(uint16_t arr, uint16_t psc) { GPIO_InitTypeDef GPIO_InitStructure; TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; TIM_OCInitTypeDef TIM_OCInitStructure; // 打开GPIO和定时器时钟 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_AFIO | RCC_APB1Periph_TIM2, ENABLE); // 配置PA0和PA1为推挽输出模式 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOA, &GPIO_InitStructure); // 定时器时钟源为APB1时钟,分频系数为psc TIM_TimeBaseStructure.TIM_Period = arr; TIM_TimeBaseStructure.TIM_Prescaler = psc; TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure); // 配置TIM2_CH1和TIM2_CH2为PWM输出模式 TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; TIM_OCInitStructure.TIM_Pulse = arr / 4; // 初始占空比设为25% TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; TIM_OC1Init(TIM2, &TIM_OCInitStructure); TIM_OC2Init(TIM2, &TIM_OCInitStructure); // 使能TIM2_CH1和TIM2_CH2的输出通道 TIM_CtrlPWMOutputs(TIM2, ENABLE); // 启动定时器2 TIM_Cmd(TIM2, ENABLE); } int main(void) { // 配置定时器1和2输出不同频率及占空比的PWM信号 TIM1_PWM_Init(999, 71); // PWM频率为72MHz/(999+1)/(71+1)=100Hz,占空比50% TIM2_PWM_Init(1999, 719); // PWM频率为72MHz/(1999+1)/(719+1)=10Hz,占空比25% while (1); } ``` 注意,定时器的计数值和自动重载值、PWM信号频率和占空比的计算方式可能会因为STM32F1的型号和配置而有所不同,具体可以参考相关的芯片手册和数据手册。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值