完善资料让更多小伙伴认识你,还能领取20积分哦, 立即完善>
1.关于DSP28335的内部ePWM模块
PWM的4要素:周期,脉宽,脉冲相位,脉冲个数。 DSP中共有6个EPWM模块,每个EPWM模块有两路输出ePWMxA和ePWMxB。它们两者之间可以是两路独立单边沿PWM输出;可以是两路独立对称的双边沿PWM输出;可以是一对双边沿非对称PWM输出。 每组ePWM包括时基TB;计数比较模块CC;动作模块AQ,死区产生模块DB,PWM斩波模块PC,错误联防TZ,时基触发ET。 简要介绍模块的功能: TB: 确定TBCLK,即与载波的周期有关 工作模式:增,减,增减 产生事件:CTR=PRD;CTR=0、 相位关系 管理ePWM模块间的同步性 如何使用:使能ePWM式中PCLKER;禁用时基时钟TBCLKSYNC=0;设置分频值(HSPCLKDIV,CLKDIV)和ePWM工作模式CTRMODE;启用时基时钟TBCLKSYNC=1; CC: 输入是时基计数器的值,将其与CMPA,CMPB比较。主要是影子寄存器和活动寄存器的使用。一种是直接加载,一种是CTR=PRD,CTR=ZERO加载。 AQ: 根据事件选择对应的ePWMxA,ePWMxB是高还是低。有ZRO,CAU,CAD,CBU,CBD。Clear和set就行。 DB: 死区控制。选择不同的模式,对ePWMxA和ePWMxB的上下沿做一个延时处理。 PC: 斩波,可以降低开关管的损耗。相当于与,将一个高电平,分成很多小份,可以降低开关管的损耗,此模块可以设置第一个小份脉冲的宽度,打开开关管。然后后边的小份目的在于维持管子的导通。 TZ: 错误联防,是6个GPIO口,可以是周期触发也可以是单次触发,两种方式均可以产生中断,周期触发的中断标志可以自动清,单次触发的中断标志需要手动清。一旦错误发生,TZCTL寄存器中的值立刻输出到ePWMxA和ePWMxB引脚上。 ET: 事件触发。 2.关于SPWM 什么是调制波和正弦波? 一般SPWM里,调制波=正弦波,载波=三角波 橙色线模拟的就是DSP中的TB模块“增减”计数模式。清晰可见它不是从0开始的,峰值为TBPRD。因此需要改变调制波的值。原本幅值为1的正弦波,使用公式 注意TBPRD是峰值。橙色三角波的顶点值。 什么是调制比? 调制比是调制波幅值和载波幅值之比,同时调制所得到的基波分量的幅值也与调制比成正比,在双极性SPWM下,得到基波分量的幅值等于调制比M与输入直流电的二分之一的乘积。因此调制比是一个很重要的概念,它决定了所得到正弦基波的幅值。补充一下,还有一个重要的概念是载波比,载波比决定最低谐波次数,单极性与双极性得到的最低谐波次数是不同的。 3.PWM的实现 #include "DSP2833x_Device.h" // DSP2833x Headerfile Include File #include "DSP2833x_Examples.h" // DSP2833x Examples Include File /******************************************************************************* * 函 数 名 : main * 函数功能 : 主函数 * 输 入 : 无 * 输 出 : 无 *******************************************************************************/ void main() { InitSysCtrl(); InitPieCtrl(); IER = 0x0000; IFR = 0x0000; InitPieVectTable(); // LED_Init(); EPWM6_Init(500); EPwm6A_SetCompare(300); //300/500=0.6 EPwm6B_SetCompare(300); //i值最大可以取499,因为ARR最大值是499. } void EPWM6_Init(Uint16 tbprd) { EALLOW; SysCtrlRegs.PCLKCR0.bit.TBCLKSYNC = 0; // Disable TBCLK within the ePWM SysCtrlRegs.PCLKCR1.bit.EPWM6ENCLK = 1; // 使能epwm外设时钟 EDIS; InitEPwm6Gpio(); // Setup Sync EPwm6Regs.TBCTL.bit.SYNCOSEL = TB_SYNC_DISABLE; // Pass through // Allow each timer to be sync'ed EPwm6Regs.TBCTL.bit.PHSEN = TB_DISABLE; EPwm6Regs.TBPHS.half.TBPHS = 0; EPwm6Regs.TBCTR = 0x0000; // Clear counter EPwm6Regs.TBPRD = tbprd; EPwm6Regs.TBCTL.bit.CTRMODE = TB_COUNT_UP; // Count up EPwm6Regs.TBCTL.bit.HSPCLKDIV=TB_DIV1; //不分频即TBCLK=150M EPwm6Regs.TBCTL.bit.CLKDIV=TB_DIV1; // Setup shadow register load on ZERO EPwm6Regs.CMPCTL.bit.SHDWAMODE = CC_SHADOW; EPwm6Regs.CMPCTL.bit.SHDWBMODE = CC_SHADOW; EPwm6Regs.CMPCTL.bit.LOADAMODE = CC_CTR_ZERO; EPwm6Regs.CMPCTL.bit.LOADBMODE = CC_CTR_ZERO; // Set Compare values EPwm6Regs.CMPA.half.CMPA = 0; // Set compare A value EPwm6Regs.CMPB = 0; // Set Compare B value // Set actions EPwm6Regs.AQCTLA.bit.ZRO = AQ_CLEAR; // Set PWM1A on Zero EPwm6Regs.AQCTLA.bit.CAU = AQ_SET; // Clear PWM1A on event A, up count EPwm6Regs.AQCTLB.bit.ZRO = AQ_CLEAR; // Set PWM1B on Zero EPwm6Regs.AQCTLB.bit.CBU = AQ_SET; // Clear PWM1B on event B, up count EPwm6Regs.ETSEL.bit.INTSEL = ET_CTR_ZERO; // Select INT on Zero event EPwm6Regs.ETSEL.bit.INTEN = 1; // Enable INT EPwm6Regs.ETPS.bit.INTPRD = ET_1ST; // Generate INT on 1st event EALLOW; SysCtrlRegs.PCLKCR0.bit.TBCLKSYNC = 1; // Start all the timers synced EDIS; } void EPwm6A_SetCompare(Uint16 val) { EPwm6Regs.CMPA.half.CMPA = val; //设置占空比 } void EPwm6B_SetCompare(Uint16 val) { EPwm6Regs.CMPB = val; //设置占空比 } 采用增计数模式,150M的TBCLK,150个是1us,现在500个,就是(1/150)*500us,换成频率就是300M。此处周期500,占空比300,所以0.6的占空比。更改CLKDIV,HSPCLKDIV可以将周期变长。如CLKDIV=2时,频率为150Mkhz 4.SPWM的实现 利用中断,每次进中断,都改变CMPA或CMPB的值(根据调整后的正弦表,前边提到过(公式))。 #include "DSP2833x_Device.h" // DSP2833x Headerfile Include File #include "DSP2833x_Examples.h" // DSP2833x Examples Include File #include "Math.h" interrupt void epwm6_timer_isr(void); //函数的声明,因为函数体的定义在最后,这个是自己定义的 void EPWM6_Init(Uint16 tbprd); int k,N=128; float m=1;//设置调制度 /******************************************************************************* * 函 数 名 : main * 函数功能 : 主函数 * 输 入 : 无 * 输 出 : 无 *******************************************************************************/ void main() { InitSysCtrl(); InitPieCtrl(); IER = 0x0000; IFR = 0x0000; InitPieVectTable(); EALLOW; PieVectTable.EPWM6_INT =&epwm6_timer_isr; EDIS; // LED_Init(); EPWM6_Init(7500); IER |= M_INT3; PieCtrlRegs.PIEIER3.bit.INTx6 = 1; //外部中断 使能第三组中断的第6个小中断 EINT; //开CPU级中断响应 ERTM; } void EPWM6_Init(Uint16 tbprd) { EALLOW; SysCtrlRegs.PCLKCR0.bit.TBCLKSYNC = 0; // Disable TBCLK within the ePWM SysCtrlRegs.PCLKCR1.bit.EPWM6ENCLK = 1; // ePWM6 EDIS; InitEPwm6Gpio(); // Setup Sync EPwm6Regs.TBCTL.bit.SYNCOSEL = TB_SYNC_DISABLE; // Pass through // Allow each timer to be sync'ed EPwm6Regs.TBCTL.bit.PHSEN = TB_DISABLE; EPwm6Regs.TBPHS.half.TBPHS = 0; EPwm6Regs.TBCTR = 0x0000; // Clear counter EPwm6Regs.TBPRD = tbprd; EPwm6Regs.TBCTL.bit.CTRMODE = TB_COUNT_UPDOWN; // Count up EPwm6Regs.TBCTL.bit.HSPCLKDIV=TB_DIV1; EPwm6Regs.TBCTL.bit.CLKDIV=TB_DIV1; // Setup shadow register load on ZERO EPwm6Regs.CMPCTL.bit.SHDWAMODE = CC_SHADOW; EPwm6Regs.CMPCTL.bit.SHDWBMODE = CC_SHADOW; EPwm6Regs.CMPCTL.bit.LOADAMODE = CC_CTR_ZERO; EPwm6Regs.CMPCTL.bit.LOADBMODE = CC_CTR_ZERO; // Set Compare values EPwm6Regs.CMPA.half.CMPA = 0; // Set compare A value EPwm6Regs.CMPB = 0; // Set Compare B value // Set actions EPwm6Regs.AQCTLA.bit.CAU = AQ_CLEAR; // Set PWM1A on Zero EPwm6Regs.AQCTLA.bit.CAD = AQ_SET; // Clear PWM1A on event A, up count EPwm6Regs.AQCTLB.bit.CBU = AQ_CLEAR; // Set PWM1B on Zero EPwm6Regs.AQCTLB.bit.CBD = AQ_SET; // Clear PWM1B on event B, up count EPwm6Regs.ETSEL.bit.INTSEL = ET_CTR_ZERO; // Select INT on Zero event EPwm6Regs.ETSEL.bit.INTEN = 1; // Enable INT EPwm6Regs.ETPS.bit.INTPRD = ET_1ST; // Generate INT on 1st event EALLOW; SysCtrlRegs.PCLKCR0.bit.TBCLKSYNC = 1; // Start all the timers synced EDIS; } interrupt void epwm6_timer_isr(void) //中断服务程序 { if(k >= N) { k = 0; } EPwm6Regs.CMPA.half.CMPA = EPwm6Regs.TBPRD*(1+m*sin(2*3.14159*k/N))/2; EPwm6Regs.CMPB=EPwm6Regs.TBPRD*(1+m*sin(2*3.14159*k/N))/2; k++; EPwm6Regs.ETCLR.bit.INT = 1; //中断标志位清零才能进入下一次的中断,在PWM的ET模块中 PieCtrlRegs.PIEACK.all = PIEACK_GROUP3; //使能中断 PIEACK_GROUP3 } TBCLK是150Mhz,tbprd=7500;增减计数。计数150次是1us,得到周期为1/150*(7500*2)us,频率为10k。 改变N,改变调制波的频率。 |
|
|
|
只有小组成员才能发言,加入小组>>
3320 浏览 9 评论
2998 浏览 16 评论
3496 浏览 1 评论
9066 浏览 16 评论
4089 浏览 18 评论
1186浏览 3评论
612浏览 2评论
const uint16_t Tab[10]={0}; const uint16_t *p; p = Tab;//报错是怎么回事?
602浏览 2评论
用NUC131单片机UART3作为打印口,但printf没有输出东西是什么原因?
2339浏览 2评论
NUC980DK61YC启动随机性出现Err-DDR是为什么?
1899浏览 2评论
小黑屋| 手机版| Archiver| 电子发烧友 ( 湘ICP备2023018690号 )
GMT+8, 2024-12-26 07:19 , Processed in 0.999246 second(s), Total 46, Slave 37 queries .
Powered by 电子发烧友网
© 2015 bbs.elecfans.com
关注我们的微信
下载发烧友APP
电子发烧友观察
版权所有 © 湖南华秋数字科技有限公司
电子发烧友 (电路图) 湘公网安备 43011202000918 号 电信与信息服务业务经营许可证:合字B2-20210191 工商网监 湘ICP备2023018690号