你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

STM32互补PWM输出使能控制

[复制链接]
STMCU小助手 发布时间:2023-1-6 16:29
STM32的高级定时器TIM1可以产生互补的PWM,并且可以通过相关寄存器的设置使能或关闭PWM的输出。在编写BLDC的驱动程序时,本人利用TIM1的channel1,2,3三个通道生成了三路互补的PWM波形,定时器驱动程序如下:
  1. void TIM1_Init(u16 arr, u16 psc)
  2. {
  3. GPIO_InitTypeDef         GPIO_InitStructure;
  4. TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;
  5. TIM_OCInitTypeDef       TIM_OCInitStructure;
  6. TIM_BDTRInitTypeDef      TIM_BDTRInitStructure;

  7. RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1 | RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB,ENABLE);  

  8. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8 | GPIO_Pin_9 | GPIO_Pin_10;
  9. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;  
  10. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  11. GPIO_Init(GPIOA,&GPIO_InitStructure);

  12. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15;
  13. GPIO_Init(GPIOB,&GPIO_InitStructure);

  14. TIM_DeInit(TIM1);
  15. TIM_TimeBaseInitStructure.TIM_Period = arr;     
  16. TIM_TimeBaseInitStructure.TIM_Prescaler = psc;  
  17. TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;   
  18. TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;
  19. TIM_TimeBaseInit(TIM1,&TIM_TimeBaseInitStructure);

  20. TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;
  21. TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;
  22. TIM_OCInitStructure.TIM_OutputNState = TIM_OutputNState_Enable;
  23. TIM_OCInitStructure.TIM_Pulse = 0;
  24. TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;
  25. TIM_OCInitStructure.TIM_OCNPolarity = TIM_OCNPolarity_High;
  26. TIM_OCInitStructure.TIM_OCIdleState = TIM_OCIdleState_Reset;     
  27. TIM_OCInitStructure.TIM_OCNIdleState = TIM_OCIdleState_Reset;

  28. TIM_OC1Init(TIM1, &TIM_OCInitStructure);
  29. TIM_OC2Init(TIM1, &TIM_OCInitStructure);
  30. TIM_OC3Init(TIM1, &TIM_OCInitStructure);

  31. TIM_OC1PreloadConfig(TIM1,TIM_OCPreload_Enable);
  32. TIM_OC2PreloadConfig(TIM1,TIM_OCPreload_Enable);
  33. TIM_OC3PreloadConfig(TIM1,TIM_OCPreload_Enable);

  34. TIM_BDTRInitStructure.TIM_OSSRState = TIM_OSSRState_Enable;
  35. TIM_BDTRInitStructure.TIM_OSSIState = TIM_OSSIState_Enable;
  36. TIM_BDTRInitStructure.TIM_LOCKLevel = TIM_LOCKLevel_OFF;
  37. TIM_BDTRInitStructure.TIM_DeadTime = 0;
  38. TIM_BDTRInitStructure.TIM_Break = TIM_Break_Disable;               
  39. TIM_BDTRInitStructure.TIM_BreakPolarity = TIM_BreakPolarity_Low;
  40. TIM_BDTRInitStructure.TIM_AutomaticOutput = TIM_AutomaticOutput_Disable;
  41. TIM_BDTRConfig(TIM1, &TIM_BDTRInitStructure);

  42. TIM_Cmd(TIM1, ENABLE);
  43. TIM_CCPreloadControl(TIM1,ENABLE);
  44. TIM_CtrlPWMOutputs(TIM1, ENABLE);
  45. }
复制代码

该函数配置了PWM的GPIO,定时器以及死区控制等。产生的波形如图1所示

20170104200150086.jpg

当我想要控制某些通道的输出(比如使能通道1的输出CH1,但是关闭其互补输出CH1N),利用下面两个函数发现TIM1依旧输出两路互补的PWM。这个函数最终操作的寄存器是捕获/比较使能寄存器TIMx->CCER。
TIM_CCxCmd(TIM1, TIM_Channel_1, TIM_CCx_Enable);
TIM_CCxNCmd(TIM1, TIM_Channel_1, TIM_CCxN_Disable);

经过调试发现在定时器初始化函数中一个函数的调用:TIM_CCPreloadControl(TIM1,ENABLE);
注释掉这句之后,或者将ENABLE改为DISABLE,才能实现关闭通道1的互补输出,如图2所示:

20170104201137481.jpg

此时通道1CH1输出PWM波形,其互补通道CH1N输出低电平。

分析发现,TIM_CCPreloadControl()函数最终操作的是控制寄存器 2(TIMx_CR2)的第0位CCPC。查阅STM32的数据手册,描述如下:
CCPC:捕获/比较预装载控制位 (Capture/compare preloaded control)
0: CCxE, CCxNE和OCxM位不是预装载的;
1: CCxE, CCxNE和OCxM位是预装载的;设置该位后,它们只在设置了COM位后被更新。
注:该位只对具有互补输出的通道起作用。

也就是说如果使能了预装载,只有在COM事件发生后值才能写入影子寄存器(影子寄存器才是真正起作用的寄存器),产生作用。同时该位只对互补输出起作用,因此对于普通定时器来说,无论该位是否被设置,都有可以通过TIM_CCxCmd()函数使能或者关闭PWM的输出。

想要让定时器的PWM输出低电平(高电平),也可以将相对的 寄存器TIMx->CCR1置零(大于arr),但是此时互补通道只能为高电平(低电平)。而通过关闭PWM的输出,可以让两个互补的通道输出更多的电压组合。
————————————————
版权声明:GroupHuang


收藏 评论0 发布时间:2023-1-6 16:29

举报

0个回答

所属标签

相似分享

官网相关资源

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版