第一次接触STM32F103的TIM1定时器时,我被它复杂的功能吓到了。但实际用起来发现,只要掌握几个关键点,这个"高级"定时器其实很友好。TIM1是STM32F103系列中功能最强大的定时器,相比通用定时器多了互补输出、死区插入等电机控制专用功能。
最让我印象深刻的是TIM1的时钟配置。记得刚开始调试时,PWM输出死活不正常,后来发现是没理解清楚时钟树。STM32F103的TIM1挂载在APB2总线上,默认情况下如果APB2预分频系数不为1,定时器时钟会倍频。比如系统时钟72MHz时,APB2分频系数如果是2(36MHz),TIM1实际得到的时钟会是72MHz。
提示:使用CubeMX配置时,时钟树页面会直观显示各定时器的实际工作频率,建议新手从这里开始。
TIM1的时基单元包含三个关键寄存器:
计算PWM频率的公式看起来复杂,其实拆解后很简单:
code复制PWM频率 = 定时器时钟 / [(PSC+1) * (ARR+1)]
比如要生成20kHz PWM(电机常用频率),使用72MHz时钟时,可以设PSC=71,ARR=49,这样实际频率就是72MHz/(72*50)=20kHz。
配置TIM1输出PWM的过程就像搭积木,需要按步骤设置各个功能模块。我总结了一个万用配置流程,适用于大多数电机控制场景。
首先初始化GPIO时要注意,TIM1的通道1~3对应PA8/PA9/PA10,通道4是PA11。这些引脚必须配置为复用推挽输出模式:
c复制GPIO_InitStruct.Pin = GPIO_PIN_8|GPIO_PIN_11;
GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH;
HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);
时基单元配置有个坑我踩过:高级定时器的TIM_TimeBaseInitTypeDef结构体里有个RepetitionCounter成员,这个只有TIM1/TIM8才有。如果没初始化,可能导致PWM输出异常。正确配置如下:
c复制TIM_HandleTypeDef htim1;
htim1.Instance = TIM1;
htim1.Init.Prescaler = 71;
htim1.Init.CounterMode = TIM_COUNTERMODE_UP;
htim1.Init.Period = 49;
htim1.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1;
htim1.Init.RepetitionCounter = 0; // 必须显式初始化
HAL_TIM_PWM_Init(&htim1);
输出比较单元配置决定了PWM的关键特性。这里有个实用技巧:使用TIM_OCMode_PWM2模式时,极性设置更符合常规思维(高电平有效时占空比越大输出越高):
c复制TIM_OC_InitTypeDef sConfigOC;
sConfigOC.OCMode = TIM_OCMODE_PWM2;
sConfigOC.Pulse = 25; // 初始占空比50%
sConfigOC.OCPolarity = TIM_OCPOLARITY_HIGH;
sConfigOC.OCFastMode = TIM_OCFAST_DISABLE;
HAL_TIM_PWM_ConfigChannel(&htim1, &sConfigOC, TIM_CHANNEL_1);
最后一定记得使能主输出(MOE),这是高级定时器特有的安全机制:
c复制__HAL_TIM_MOE_ENABLE(&htim1);
HAL_TIM_PWM_Start(&htim1, TIM_CHANNEL_1);
驱动电机H桥电路时,互补输出和死区时间是避免炸管的关键。TIM1的BRK(刹车)功能我曾在紧急停止场景中救过整个驱动板。
配置互补输出需要特别注意:
死区时间计算公式为:
code复制T_deadtime = DTG[7:0] * T_dts
其中T_dts是定时器时钟经过CKD分频后的周期。比如72MHz时钟,CKD=DIV1时:
实际配置代码示例:
c复制TIM_BreakDeadTimeConfigTypeDef sBreakDeadTimeConfig;
sBreakDeadTimeConfig.OffStateRunMode = TIM_OSSR_DISABLE;
sBreakDeadTimeConfig.OffStateIDLEMode = TIM_OSSI_DISABLE;
sBreakDeadTimeConfig.LockLevel = TIM_LOCKLEVEL_OFF;
sBreakDeadTimeConfig.DeadTime = 54; // 约750ns死区时间
sBreakDeadTimeConfig.BreakState = TIM_BREAK_ENABLE;
sBreakDeadTimeConfig.BreakPolarity = TIM_BREAKPOLARITY_HIGH;
sBreakDeadTimeConfig.AutomaticOutput = TIM_AUTOMATICOUTPUT_ENABLE;
HAL_TIMEx_ConfigBreakDeadTime(&htim1, &sBreakDeadTimeConfig);
刹车功能配置建议:
在实际电机驱动项目中,我发现几个教科书上不会讲的实用技巧。首先是软启动实现,通过逐步增加ARR值可以避免电机启动时的电流冲击。
一个典型的软启动流程:
代码实现参考:
c复制void SoftStart(TIM_HandleTypeDef *htim, uint32_t channel, uint32_t target_arr, uint32_t target_ccr)
{
__HAL_TIM_SET_AUTORELOAD(htim, 10);
__HAL_TIM_SET_COMPARE(htim, channel, 5);
HAL_TIM_PWM_Start(htim, channel);
for(uint32_t arr=10; arr<target_arr; arr+=5) {
__HAL_TIM_SET_AUTORELOAD(htim, arr);
uint32_t ccr = arr * target_ccr / target_arr;
__HAL_TIM_SET_COMPARE(htim, channel, ccr);
HAL_Delay(10);
}
}
PWM占空比动态调整时,为避免毛刺建议使用预装载功能。通过设置TIMx_CR1寄存器的ARPE位,可以确保ARR和CCR的更新在下一个周期生效:
c复制TIM1->CR1 |= TIM_CR1_ARPE; // 启用ARR预装载
__HAL_TIM_SET_COMPARE(&htim1, TIM_CHANNEL_1, new_ccr); // 平滑更新
电机驱动常见问题排查:
当项目需要更高性能时,TIM1的一些高级功能就派上用场了。比如使用DMA自动更新CCR值,可以实现无CPU干预的复杂PWM波形生成。
DMA配置示例(使用TIM1_CH1触发):
c复制// DMA流配置
__HAL_RCC_DMA1_CLK_ENABLE();
hdma_tim1_ch1.Instance = DMA1_Channel2;
hdma_tim1_ch1.Init.Direction = DMA_MEMORY_TO_PERIPH;
hdma_tim1_ch1.Init.PeriphInc = DMA_PINC_DISABLE;
hdma_tim1_ch1.Init.MemInc = DMA_MINC_ENABLE;
hdma_tim1_ch1.Init.PeriphDataAlignment = DMA_PDATAALIGN_HALFWORD;
hdma_tim1_ch1.Init.MemDataAlignment = DMA_MDATAALIGN_HALFWORD;
hdma_tim1_ch1.Init.Mode = DMA_CIRCULAR;
hdma_tim1_ch1.Init.Priority = DMA_PRIORITY_HIGH;
HAL_DMA_Init(&hdma_tim1_ch1);
// 绑定DMA到TIM1_CH1
__HAL_LINKDMA(&htim1, hdma[TIM_DMA_ID_CC1], hdma_tim1_ch1);
HAL_TIM_PWM_Start_DMA(&htim1, TIM_CHANNEL_1, (uint32_t*)pwm_buffer, buffer_length);
定时器同步功能可以精确协调多个PWM输出。我曾用这个特性实现步进电机的微步控制:
编码器接口模式是另一个实用功能,配合霍尔传感器可以直接读取电机转速:
c复制TIM_Encoder_InitTypeDef sEncoderConfig;
sEncoderConfig.EncoderMode = TIM_ENCODERMODE_TI12;
sEncoderConfig.IC1Polarity = TIM_ICPOLARITY_RISING;
sEncoderConfig.IC1Selection = TIM_ICSELECTION_DIRECTTI;
sEncoderConfig.IC1Prescaler = TIM_ICPSC_DIV1;
sEncoderConfig.IC1Filter = 0;
sEncoderConfig.IC2Polarity = TIM_ICPOLARITY_RISING;
sEncoderConfig.IC2Selection = TIM_ICSELECTION_DIRECTTI;
sEncoderConfig.IC2Prescaler = TIM_ICPSC_DIV1;
sEncoderConfig.IC2Filter = 0;
HAL_TIM_Encoder_Init(&htim1, &sEncoderConfig);
HAL_TIM_Encoder_Start(&htim1, TIM_CHANNEL_ALL);
调试TIM1时,逻辑分析仪是最得力的工具。我习惯先捕获PWM波形,检查三个关键参数:频率、占空比和死区时间。
几个典型的波形异常及解决方法:
使用STM32CubeIDE的调试视图可以实时监控寄存器值:
一个实用的调试技巧:通过强制输出模式(TIMx_CCMRx寄存器的OCxM位)可以手动控制输出电平,快速验证硬件连接:
c复制// 强制通道1输出高电平
TIM1->CCMR1 |= TIM_CCMR1_OC1M_2 | TIM_CCMR1_OC1M_1;
// 强制通道1输出低电平
TIM1->CCMR1 |= TIM_CCMR1_OC1M_2;
// 恢复PWM模式
TIM1->CCMR1 &= ~(TIM_CCMR1_OC1M_2 | TIM_CCMR1_OC1M_1);
TIM1->CCMR1 |= TIM_CCMR1_OC1M_1 | TIM_CCMR1_OC1M_2;
电源管理方面,当电机停止时,建议关闭TIM1时钟以降低功耗:
c复制__HAL_RCC_TIM1_CLK_DISABLE(); // 关闭时钟
// 需要重新启用时
__HAL_RCC_TIM1_CLK_ENABLE();
HAL_TIM_PWM_Start(&htim1, TIM_CHANNEL_1);