1. STM32H743定时器与PWM基础
STM32H743作为高性能微控制器,其定时器模块功能强大,特别适合电机控制场景。我最早接触这块芯片时,就被它TIM定时器的灵活性惊艳到了——尤其是高级定时器TIM1/TIM8,不仅能生成普通PWM,还能输出带死区的互补PWM,这对驱动三相电机简直是刚需。
先说说什么是互补PWM。想象一下控制电机的H桥电路:上下两个MOS管就像跷跷板的两端,绝对不能同时导通,否则电源直接短路。互补PWM就是让两路PWM信号始终保持相反状态,当CH1为高电平时CH1N自动为低电平,反之亦然。但现实中MOS管开关需要时间,所以必须插入死区时间(Dead Time),确保一个管子完全关闭后,另一个才开启。
注意:死区时间过短会导致桥臂直通烧毁MOS管,过长则会降低电机效率,需要精确计算
2. CubeMX配置互补PWM
2.1 时钟树配置
打开CubeMX后,首先配置时钟树。TIM8挂在APB2总线上,在我的工程里APB2时钟设为240MHz(对应STM32H743的默认配置)。这里有个坑:如果使用外部晶振,务必检查PLL配置是否正确,我有次因为PLL倍频算错,导致实际PWM频率只有预期值的一半。
2.2 定时器基础参数
在TIM8配置界面:
- 选择"Clock Source"为内部时钟
- 模式选择"PWM Generation CHxN"(注意带N的才是互补通道)
- 预分频器(Prescaler)设为0,计数器周期(Counter Period)设为23999
- 计算公式:PWM频率 = 240MHz/(23999+1)/(0+1) = 10kHz
- 脉冲宽度(Pulse)先设为11999,即50%占空比
2.3 死区时间配置
这是最关键的步骤!在"Parameter Settings"标签页找到"Dead Time"选项:
- Dead Time值:根据MOS管规格书计算,比如IR2104驱动芯片需要约500ns死区时间
- 计算公式:Dead Time = (DTG[7:0] * Tdts) + Tdtg
- 其中Tdts为时钟周期,当CK_INT=240MHz时,Tdts=4.17ns
- 假设需要500ns:设置DTG=0x5F(计算过程:500ns/4.17ns≈120→0x78)
实测时我用示波器抓波形发现死区实际为520ns,说明计算时还要考虑硬件电路延迟。建议先用保守值,再逐步优化。
3. 电机驱动实战技巧
3.1 无刷电机控制
驱动BLDC电机时,我通常这样配置三组互补PWM:
// 启动所有通道 HAL_TIM_PWM_Start(&htim8, TIM_CHANNEL_1); HAL_TIMEx_PWMN_Start(&htim8, TIM_CHANNEL_1); // 互补通道 // 重复上述代码配置CH2/CH3遇到过的一个典型问题:电机启动时抖动严重。后来发现是死区时间不足导致MOS管短暂直通,解决方法是在初始化后动态调整死区:
__HAL_TIM_SET_AUTORELOAD(&htim8, 23999); // 重设周期 __HAL_TIM_SET_COMPARE(&htim8, TIM_CHANNEL_1, 11999); MODIFY_REG(htim8.Instance->BDTR, TIM_BDTR_DTG, 0x78); // 动态修改死区3.2 安全保护机制
STM32H743的刹车功能(Break)一定要用起来!我在电路里加了过流检测芯片,一旦触发故障信号:
- 硬件刹车:通过BKIN引脚直接关闭PWM输出
- 软件保护:在中断服务函数中调用
HAL_TIMEx_ConfigBreakDeadTime(&htim8, TIM_BREAK_ENABLE | TIM_BREAKPOLARITY_HIGH);
4. 调试与优化
4.1 示波器观测技巧
用差分探头测量电机相线电压时,建议这样验证:
- 先单独测试每对互补PWM,确认死区时间
- 观察六步换向时的波形过渡是否平滑
- 满负载时检查PWM是否有畸变(我曾因PCB走线过长导致振铃)
4.2 性能优化
当PWM频率超过20kHz时,发现CPU负载明显上升。解决方法:
- 启用DMA传输PWM占空比数据
- 使用TIM8的重复计数器减少中断频率
- 关键代码移到RAM执行(通过__attribute__((section(".RamFunc"))))
有一次为了追求高动态响应,我把PWM频率提到50kHz,结果MOS管发热严重。后来用热像仪发现是死区损耗导致,最终折中选用30kHz频率+自适应死区算法。