赞
踩
在电机控制,PWM整流器,逆变器中都需要单片机输出SPWM波去控制.
图0
对于单相整流,逆变器有双极性调制,单极性调制,单极性倍频调制。
图1
直接将整个正弦波放在三角波里做比较。
图2
全桥输出波形(占空比按正弦幅值变化).这个波经过LC滤波后就变成正弦波了。
图3
matlab里搭出的双极性调制,后面写代码也是参考这个。
0.7是调制度。前面输进去的正弦波是从-1到0再到1的正弦波。将其抬高1,整个波形就只有正的,然后再除2就成了图1中的正弦波(归一化让正弦波在0到1内)。与三角波比较后输出调制波。
图4
图5
与双极性不同,单极性调制的输出是有正有负的。可以看到,这个波要比双极性调制更接近正弦波,所以谐波含量更少。
图6
按这种方式调制,那么全桥两边都是一会为高频开关一会为低频开关。而我们也可以只让一边做高频侧,另一边做低频侧(负责波形的正负)。
两半桥输出20K(比如)PWM,相减后得到40KPWM完成成倍频。
/**
* @brief 单极性倍频调制
*
* @param insignal 输入基波
* @param cnt 当时定时器cnt值
* @return unsigned int
*/
void unimp_modulation(float insignal, unsigned short cnt, spwm_t *ccr)
{
ccr->ch1_ccr = (1+insignal)/2 * (cnt-1);
ccr->ch2_ccr = (1-insignal)/2 * (cnt-1);
}
可以把调制简单的理解成,将正弦波与三角波混合。后面经过滤波器就能去除三角波得到正弦波。那么三角波(载波)其实不一定就是上面那种等腰三角形,你用只有一半的波也是可以的。
图7
后面加滤波器就能把这个半个的三角波滤去。
图8
可以看出,调制的核心是利用高频载波(三角波)与低频基波(正弦波)作比较得出。那么在stm上怎么实现这个过程?或者说在STM32里三角波和正弦波分别是什么,从哪里来?
stm32的pwm输出功能(中心对称计数)可以用下图形象的描述。
图9
三角波是定时器计数值按时间的变化,方波是输出的PWM波。
可以看出改变CCR就可以改变脉冲宽度。即CCR/CNT=占空比。(当然在另一个PWM模式里就与这里的相反了)
图10
(实际的三角波频率要比正弦波大很多,看不清楚,这里减小了三角波的频率为了看清比较过程。)
联系上面两张图,其实就是在每次到达CCR做比较来改变脉冲宽度。那么当CCR值按正弦变化不就实现了SPWM调制了嘛。
好了,我们现在知道了。定时器的计数值就是三角波。正弦波就是按正弦变化的CCR值。
比如我们要一个50HZ的正弦波,三角波频率20KHZ。可以知道一个正弦波里包含400个三角波,而一个三角波要与正弦波比较两次,所以我们需要800个CCR值。当然,我们不追求精度,可以让一个三角波的两次比较值都一样,也就是400个CCR值。甚至可以两次三角波的比较CCR都一样,这样就只需要200个CCR了,当然这么做会损失一些精度。最好的情况当然是三角波频率足够高,比较值也足够多。
常见的CCR获取方式有下面的两种。
早期的单片机由于运算性能不行,所以是先把这些CCR值存储在ROM里(RAM也行)做正弦波码表。然后这个码表可以在一些软件里生成。也可以自己提前算好。
有了码表,我们只需要每次计数器计到CCR触发中断时把CCR值更新就行了。
使用码表是因为运算性能不够,而stm32运算性能足够(使用dsp库,用C库的函数还是算起来有些慢的),可以自己去算正弦值。大致思路就是开一个定时器,每次进定时器中断的时间相同,累加就能得到时间.然后在定时器中断里计算sin(wt)
具体操作如下:
spwm_struct.w = 2*pi*50; //50HZ正弦波
spwm_struct.T = 0.00005; //20K的定时器中断,每次进入间隔50us
spwm_struct.uref = A*arm_sin_f32(spwm_struct.WT); //A是幅值
spwm_struct.WT += spwm_struct.w * spwm_struct.T;
if(spwm_struct.WT >= 2 * pi) spwm_struct.WT = 0; //WT在0到2pi变化
单片机:STM32F446RCT6
栅极驱动:IR2104。
一个2104高端接G1,低端接G2 另一个高端接G3,低端接G4
调制方式:双极性,定时器中断计算正弦波(查表实现的文章很多,这里就不展示了)
大致思路:用高级定时器输出PWM给一个2104去控制G1,G2(PWM信号与G1信号一样,2104会互补输出G2的信号)。G1和G3的信号又是互补的。所以生成互补PWM给另一个2104。
定时器中断频率20Khz,周期0.00005s.
定时器其他配置默认。
至于其他配置(下载,时钟等),就不展示了。
定时器中断
void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim) { if(htim->Instance==TIM1) { spwm_struct.jibo = spwm_struct.mod_dep*arm_cos_f32(spwm_struct.wt);//更新正弦波数据 bipolar_modulation(spwm_struct.jibo, tim_cnt, &spwm_struct); TIM1->CCR1 = spwm_struct.ch1_ccr; spwm_struct.wt += spwm_struct.w * spwm_struct.T;//更新相角wt if(spwm_struct.wt>=2*pi) spwm_struct.wt=0;//计满2π后归零,防止溢出 } }
双极性调制
/**
* @brief 双极性调制
*
* @param insignal 输入基波 (-1到1变化)
* @param cnt 当时定时器cnt值
* @return unsigned int
*/
void bipolar_modulation(float insignal, unsigned short cnt, spwm_t *ccr)
{
insignal = (insignal+1) / 2;
ccr->ch1_ccr = insignal * (cnt - 1);
}
主函数while1前放的初始化
void spwm_init() { /*参数初始化*/ spwm_struct.w = 2*pi*50; //2*pi*f spwm_struct.wt = 0; spwm_struct.fre = 50; spwm_struct.uref = 1; spwm_struct.rqd_flag = 1; spwm_struct.mod_dep = 0.5; //调制度 spwm_struct.T = 0.00005; /*2104使能*/ HAL_GPIO_WritePin(GPIOB,GPIO_PIN_5,GPIO_PIN_SET); /*PWM输出开启*/ HAL_TIM_Base_Start_IT(&htim1); HAL_TIM_PWM_Start(&htim1, TIM_CHANNEL_1); HAL_TIMEx_PWMN_Start(&htim1, TIM_CHANNEL_1); }
spwm结构体
typedef struct
{
float w; //角频率
float fre; //频率
float wt; //相角
float mod_dep; //调制度
short ch1_ccr; //ccr1
short ch2_ccr;
float jibo;
float T; //每次进入定时器中断的时间
}spwm_t;
欢迎加入扣扣交流群,群号:807477521
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。