找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 1988|回复: 22
打印 上一主题 下一主题
收起左侧

STC8单片机做逆变50hz,这里的例子程序,要怎么改?

  [复制链接]
跳转到指定楼层
楼主
一个SPWM带有死区的程序 用STC8A8K64S4A12单片机编写的SPWM
  1. //一个SPWM带有死区的程序 用STC8A8K64S4A12单片机编写的SPWM
  2. //介绍了一种高性能微机控制的单相 SPWM 变频调速系统
  3. // SPWM信号经低通滤波后可变换为正弦波,窄带范围内的方波经低通滤波后可变换为相应频率的正弦波
  4. //程序能用,如果不是你想要的频率,那就 自己改一下数据 记得同时要替换数组里面的值
  5. #define         MAIN_Fosc                24000000L        //定义主时钟
  6. #include        "STC8.H"
  7. unsigned int code T_SinTable[]={
  8. 1220, 1256, 1292, 1328, 1364, 1400, 1435, 1471, 1506, 1541,
  9. 1575, 1610, 1643, 1677, 1710, 1742, 1774, 1805, 1836, 1866,
  10. 1896, 1925, 1953, 1981, 2007, 2033, 2058, 2083, 2106, 2129,
  11. 2150, 2171, 2191, 2210, 2228, 2245, 2261, 2275, 2289, 2302,
  12. 2314, 2324, 2334, 2342, 2350, 2356, 2361, 2365, 2368, 2369,
  13. 2370, 2369, 2368, 2365, 2361, 2356, 2350, 2342, 2334, 2324,
  14. 2314, 2302, 2289, 2275, 2261, 2245, 2228, 2210, 2191, 2171,
  15. 2150, 2129, 2106, 2083, 2058, 2033, 2007, 1981, 1953, 1925,
  16. 1896, 1866, 1836, 1805, 1774, 1742, 1710, 1677, 1643, 1610,
  17. 1575, 1541, 1506, 1471, 1435, 1400, 1364, 1328, 1292, 1256,
  18. 1220, 1184, 1148, 1112, 1076, 1040, 1005, 969, 934, 899,
  19. 865, 830, 797, 763, 730, 698, 666, 635, 604, 574,
  20. 544, 515, 487, 459, 433, 407, 382, 357, 334, 311,
  21. 290, 269, 249, 230, 212, 195, 179, 165, 151, 138,
  22. 126, 116, 106, 98, 90, 84, 79, 75, 72, 71,
  23. 70, 71, 72, 75, 79, 84, 90, 98, 106, 116,
  24. 126, 138, 151, 165, 179, 195, 212, 230, 249, 269,
  25. 290, 311, 334, 357, 382, 407, 433, 459, 487, 515,
  26. 544, 574, 604, 635, 666, 698, 730, 763, 797, 830,
  27. 865, 899, 934, 969, 1005, 1040, 1076, 1112, 1148, 1184,
  28. };
  29. #define        PwmClk_1T        0
  30. #define        PwmClk_2T        1
  31. #define        PwmClk_3T        2
  32. #define        PwmClk_4T        3
  33. #define        PwmClk_5T        4
  34. #define        PwmClk_6T        5
  35. #define        PwmClk_7T        6
  36. #define        PwmClk_8T        7
  37. #define        PwmClk_9T        8
  38. #define        PwmClk_10T        9
  39. #define        PwmClk_11T        10
  40. #define        PwmClk_12T        11
  41. #define        PwmClk_13T        12
  42. #define        PwmClk_14T        13
  43. #define        PwmClk_15T        14
  44. #define        PwmClk_16T        15
  45. #define        PwmClk_T2        16
  46. #define        EAXSFR()                P_SW2 |=  0x80        /* MOVX A,@DPTR/MOVX @DPTR,A指令的操作对象为扩展SFR(XSFR) */
  47. #define        EAXRAM()                P_SW2 &= ~0x80        /* MOVX A,@DPTR/MOVX @DPTR,A指令的操作对象为扩展RAM(XRAM) */
  48. #define        PWM_Enable()        PWMCR |=  0x80        /* 使能PWM波形发生器,PWM计数器开始计数 */
  49. #define        PWM_Disable()        PWMCR &= ~0x80        /* 关闭PWM波形发生器 */
  50. #define P2n_standard(bitn)                        P2M1 &= ~(bitn),        P2M0 &= ~(bitn)
  51. #define P2n_push_pull(bitn)                        P2M1 &= ~(bitn),        P2M0 |=  (bitn)
  52. #define P2n_pure_input(bitn)                P2M1 |=  (bitn),        P2M0 &= ~(bitn)
  53. #define P2n_open_drain(bitn)                P2M1 |=  (bitn),        P2M0 |=  (bitn)
  54. #define                PWM_VECTOR                22
  55. #define                PWM_DeadZone        12        /* 死区时钟数, 6 ~ 24之间  */
  56. typedef unsigned char         u8;
  57. typedef unsigned int          u16;
  58. typedef unsigned long         u32;
  59. sbit OE245 =P4^1;
  60. sbit LED_OUT =P3^3;
  61. #define Port P1                         //设置CPU和模块的通讯接口
  62. u8        PWM_Index;        //SPWM查表索引
  63. void delay(unsigned long d)
  64. {
  65.   while(d--);
  66. }
  67. //========================================================================
  68. // 函数: void        PWM_config(void)
  69. // 描述: PWM配置函数。
  70. // 参数: none.
  71. // 返回: none.
  72. // 版本: VER1.0
  73. // 日期:
  74. // 备注:
  75. //========================================================================
  76. void        PWM_config(void)
  77. {
  78.   EAXSFR();                        // 访问XFR
  79.   PWM3T1 = 65; // 第一个翻转计数
  80.   PWM3T2=1220;         // 第二个翻转计数
  81.   PWM3CR =0;         // PWM3输出选择P2.3, 无中断
  82.   PWM3CR  |=  0x80;        // 相应PWM通道的端口为PWM输出口,受PWM波形发生器控制 ENC30
  83.   PWM3CR &= ~0x40;        // 设置PWM输出端口的初始电平为0 C3INI
  84.   P23 = 0;
  85.   P2n_push_pull(1<<3);        //IO初始化, 上电时为高阻
  86.   PWM4T1= 65-PWM_DeadZone;                        // 第一个翻转计数低字节
  87.   PWM4T2 = (1220+PWM_DeadZone); // 第二个翻转计数高字节
  88.   PWM4CR = 0;        // PWM4输出选择P2.2, 无中断
  89.   PWM4CR  |=  0x80;        // 相应PWM通道的端口为PWM输出口,受PWM波形发生器控制 ENC40
  90.   //        PWMCFG &= ~0x04;        // 设置PWM输出端口的初始电平为0  C4INI
  91.   PWM4CR |=  0x40;        // 设置PWM输出端口的初始电平为1
  92.   P24 = 1;
  93.   P2n_push_pull(1<<4);        //IO初始化, 上电时为高阻
  94.   PWMC = 2400;        // PWM计数器的高字节
  95.   PWMCKS = PwmClk_1T;        // 时钟源: PwmClk_1T,PwmClk_2T, ... PwmClk_16T, PwmClk_Timer2
  96.   EAXRAM();                        // 恢复访问XRAM
  97.   PWMCR |= ENPWM;                // 使能PWM波形发生器,PWM计数器开始计数
  98.   //        PWMCR &= ~ECBI;                // 禁止PWM计数器归零中断
  99.   PWMCR |=  ECBI;                // 允许PWM计数器归零中断
  100.   //        PWMFDCR = ENFD | FLTFLIO | FDIO;        //PWM失效中断控制,  ENFD | FLTFLIO | EFDI | FDCMP | FDIO
  101. }
  102. /**********************************************/
  103. void main(void)
  104. {
  105.   unsigned char i;
  106.   OE245 =0;
  107.   LED_OUT =1;
  108.   PWM_config();        //初始化PWM
  109.   EA = 1;                //允许全局中断
  110.   while (1)
  111.   {
  112.     for(i=1;i!=0;i<<=1)  // 0000 0000
  113.     {
  114.       Port =~i; //0111 1111
  115.       delay(200000);
  116.     }
  117.   }
  118. }
  119. /********************* PWM中断函数************************/
  120. void PWM_int (void) interrupt PWM_VECTOR
  121. {
  122.   u16        j;
  123.   u8        SW2_tmp;
  124.   if(PWMCFG & CBIF)        //PWM计数器归零中断标志 PWMCFG
  125.   {
  126.     PWMCFG &= ~CBIF;        //清除中断标志
  127.     SW2_tmp = P_SW2;        //保存SW2设置
  128.     EAXSFR();                //访问XFR
  129.     j = T_SinTable[PWM_Index];
  130.     PWM3T2H = (u8)(j >> 8);        //第二个翻转计数高字节
  131.     PWM3T2L  = (u8)j;                        //第二个翻转计数低字节
  132.     j += PWM_DeadZone;                //死区
  133.     // 指向PWM4
  134.     PWM4T2H = (u8)(j >> 8);        //第二个翻转计数高字节
  135.     PWM4T2L = (u8)j;                        //第二个翻转计数低字节
  136.     P_SW2 = SW2_tmp;        //恢复SW2设置
  137.     if(++PWM_Index >= 200)        PWM_Index = 0;
  138.   }
  139. }
复制代码


单片机SPWM波 产生正弦波的源程序及工具
http://www.51hei.com/bbs/dpj-55510-1.html
分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏3 分享淘帖 顶 踩
回复

使用道具 举报

沙发
ID:675799 发表于 2021-11-24 16:05 | 只看该作者
这个 并没有直接指定 频率, 那要 50HZ, 或则45HZ, 要怎么修改。
回复

使用道具 举报

板凳
ID:675799 发表于 2021-11-25 14:03 | 只看该作者
估计这个 挂这个1年,总有回复1下答案吧。
回复

使用道具 举报

地板
ID:624769 发表于 2021-11-25 14:49 | 只看该作者
回复 +1
要回复容易,这就给你回复了。
问题是,别人如何我不知道,我是因为你的题目看不懂,所以,帮不了你…… 只好一直不回复……
回复

使用道具 举报

5#
ID:123289 发表于 2021-11-25 16:06 | 只看该作者
1、你想达成什么目的?假设是A。
2、你给出的例子,达成了什么目的?假设是B。
3、A与B有什么关联,你想利用B的哪些东东,来达成A。
4、从B到A,哪几项不会,需要帮助?
拧得清?


回复

使用道具 举报

6#
ID:675799 发表于 2021-11-25 16:34 | 只看该作者
继续问,直到有人会, 估计也没几个会。  会皮毛的,估计也不多。
回复

使用道具 举报

7#
ID:96682 发表于 2021-11-27 05:34 来自手机 | 只看该作者
弱弱的问一句,上面的 《……T_SinTable[]={……》怎么来的?
回复

使用道具 举报

8#
ID:675799 发表于 2021-11-29 10:04 | 只看该作者
wc86110 发表于 2021-11-27 05:34
弱弱的问一句,上面的 《……T_SinTable[]={……》怎么来的?

定义1个 正弦波sin数组, 数值,是他们用软件生成,终值1220
回复

使用道具 举报

9#
ID:844772 发表于 2021-11-29 17:39 | 只看该作者
1.这里是用积分表示SIN函数,你可以理解把一个周期切成200片,通过占空比不同,用矩形波代替每片的函数,然后再通过低通就是正弦波了。
2.用200个PWM(不同占空比的波)表示一个正弦波。
3.所以就很明显了,你sin函数的频率就是PWM频率除以200,要调整频率就几种了,一种是调整PWM,一种是改变函数取样数量(目前是200),还有就是用几个PWM处理一片(这样会更平滑)。
不知道我说明白没,好多人不懂其实是数学问题。

评分

参与人数 1黑币 +20 收起 理由
admin + 20 回帖助人的奖励!

查看全部评分

回复

使用道具 举报

10#
ID:675799 发表于 2021-11-30 08:46 | 只看该作者
24000000L /200点 /2400计数器=50HZ,

如果 要输出 45,   240000000/200 /2666=45hz   
PWMC = 2666;        // PWM计数器的高字节 ,  2666为45HZ
  PWMCKS = PwmClk_1T;        // 时钟源: PwmClk_1T,PwmClk_2T, ... PwmClk_16T, PwmClk_Timer2
回复

使用道具 举报

11#
ID:844772 发表于 2021-11-30 09:08 | 只看该作者
hww22 发表于 2021-11-30 08:46
24000000L /200点 /2400计数器=50HZ,

如果 要输出 45,   240000000/200 /2666=45hz   

对啊,我说的就是这个意思。
回复

使用道具 举报

12#
ID:675799 发表于 2021-11-30 15:04 | 只看该作者
glinfei 发表于 2021-11-30 09:08
对啊,我说的就是这个意思。

如果要检测,输出的4个管子,过流,然后,降低脉宽(功率),要怎么写代码? P1.2端口,低电平表示过流。 因为浴霸启动电流大,所以要低功率,启动1下。
回复

使用道具 举报

13#
ID:844772 发表于 2021-12-1 08:56 | 只看该作者
hww22 发表于 2021-11-30 15:04
如果要检测,输出的4个管子,过流,然后,降低脉宽(功率),要怎么写代码? P1.2端口,低电平表示过流。  ...

不过是调整PWM2_2的T1T2,减小占空比而已,要注意是动态调,不过流就增大占空比,使得一直处于快要过流状态。
回复

使用道具 举报

14#
ID:675799 发表于 2021-12-3 11:36 | 只看该作者
glinfei 发表于 2021-12-1 08:56
不过是调整PWM2_2的T1T2,减小占空比而已,要注意是动态调,不过流就增大占空比,使得一直处于快要过流状 ...

oid main(void)
{
  unsigned char i;
  OE245 =0;
  LED_OUT =1;
  PWM_config();        //初始化PWM
  EA = 1;                //允许全局中断
  while (1)
  {
    for(i=1;i!=0;i<<=1)  // 0000 0000
    {///////////
if (过流)
          { PWM3T1 = 65; // 第一个翻转计数 改 35  ??
  PWM3T2=1220;         // 第二个翻转计数 改 610  ???
PWM4T1= 65-PWM_DeadZone;                        // 第一个翻转计数低字节  降级脉冲 65改35??
  PWM4T2 = (1220+PWM_DeadZone); // 第二个翻转计数高字节 1220改610 ??

}
if(不过流)
   { PWM3T1 = 65; // 第一个翻转计数 改 35  ??
  PWM3T2=1220;         // 第二个翻转计数 改 610  ???
PWM4T1= 65-PWM_DeadZone;                        // 第一个翻转计数低字节  降级脉冲 65改35??
  PWM4T2 = (1220+PWM_DeadZone); // 第二个翻转计数高字节 1220改610 ??

}//////////////
      Port =~i; //0111 1111
      delay(200000);
    }
  }
}
回复

使用道具 举报

15#
ID:675799 发表于 2021-12-6 10:46 | 只看该作者
while (1)
  {
    for(i=1;i!=0;i<<=1)  // 0000 0000
    {///////////
if (过流)
          { PWM3T1 = 65; // 第一个翻转计数 改 35  ??
  PWM3T2=1220;         // 第二个翻转计数 改 610  ???
PWM4T1= 65-PWM_DeadZone;                        // 第一个翻转计数低字节  降级脉冲 65改35??
  PWM4T2 = (1220+PWM_DeadZone); // 第二个翻转计数高字节 1220改610 ??

}
if(轻负荷)
{ PWM3T1 = 65; // 第一个翻转计数 改 30  ??
  PWM3T2=1220;         // 第二个翻转计数 改 405  ???
PWM4T1= 65-PWM_DeadZone;                        // 第一个翻转计数低字节  降级脉冲 65改30??
  PWM4T2 = (1220+PWM_DeadZone); // 第二个翻转计数高字节 1220改610 ??

}//////////////
if(不过流)
   { PWM3T1 = 65; // 第一个翻转计数 改 35  ??
  PWM3T2=1220;         // 第二个翻转计数 改 610  ???
PWM4T1= 65-PWM_DeadZone;                        // 第一个翻转计数低字节  降级脉冲 65改35??
  PWM4T2 = (1220+PWM_DeadZone); // 第二个翻转计数高字节 1220改610 ??

}//////////////
回复

使用道具 举报

16#
ID:507641 发表于 2021-12-7 19:27 | 只看该作者
写程序要与硬件配合的,都没有实质的科目怎么讨论?
回复

使用道具 举报

17#
ID:675799 发表于 2021-12-27 15:27 | 只看该作者
SURUBU1976 发表于 2021-12-7 19:27
写程序要与硬件配合的,都没有实质的科目怎么讨论?

实际上,电路 很多是1边输入 正弦波脉冲, 1边输入 同极性的 电平信号。
回复

使用道具 举报

18#
ID:675799 发表于 2021-12-27 15:37 | 只看该作者
glinfei 发表于 2021-11-30 09:08
对啊,我说的就是这个意思。

为了降低 脉冲突变, 所以现在是 2个上管,轮流送 正弦波脉冲, 2个下管轮流送同级性的电平信号。
都是在 读取的 0值, (最小值0)。 进行 下管切换换的。

程序要怎么写才合理。
回复

使用道具 举报

19#
ID:675799 发表于 2021-12-27 15:50 | 只看该作者
由于要在0值 切换 下管 的极性,所以代码 不好理解, 上面的 正弦波值, 都没有0

22222.jpg (17.39 KB, 下载次数: 80)

22222.jpg
回复

使用道具 举报

20#
ID:844772 发表于 2021-12-27 17:49 | 只看该作者
本帖最后由 glinfei 于 2021-12-27 18:09 编辑

你过流时,应该不动T1,只改T2就行啊,根据过流情况给 T_SinTable[PWM_Index]乘以一个百分比整体降压;你是在做H桥逆变吧?带了死区,没啥特别建议,分别看看波形再上负载;
你是在直流上做正弦波,零在T_SinTable[]=1220,实在不行你就利用PWM控制你说的上下管导通(带了死区);利用PWM_Index控制左右管导通?
另外,调整T_SinTable[],很容易做出你图一的波形。
回复

使用道具 举报

21#
ID:675799 发表于 2021-12-28 08:34 | 只看该作者
本帖最后由 hww22 于 2021-12-28 08:57 编辑
glinfei 发表于 2021-12-27 17:49
你过流时,应该不动T1,只改T2就行啊,根据过流情况给 T_SinTable乘以一个百分比整体降压;你是在做H桥逆变 ...

直流310V , H 桥 的。       自己做来玩玩, 驱动 卷帘门,之类。 停电神器。
T_SinTable 乘以一个百分比整体降压, 放在中断函数 里面,是吧。
这么说 我把2个 下管 轮流送 方波信号, 就可以了。  
if (j==1220)   //0值, 就切换方波 比如 P1.6    P1.7
{P1.6 =0;
P1.7 =1; }
if (j==???)   //0值, 就切换方波 比如 P1.6    P1.7
{P1.6 =1;
P1.7 =0; }


问题是  1个周期有 4个 0值吧。  其余3个 呢?
回复

使用道具 举报

22#
ID:675799 发表于 2021-12-28 08:57 | 只看该作者
本帖最后由 hww22 于 2021-12-28 09:27 编辑

********************* PWM中断函数************************/
void PWM_int (void) interrupt PWM_VECTOR
{
  u16        j;
  u8        SW2_tmp;
  if(PWMCFG & CBIF)        //PWM计数器归零中断标志 PWMCFG
  {
    PWMCFG &= ~CBIF;        //清除中断标志
    SW2_tmp = P_SW2;        //保存SW2设置
    EAXSFR();                //访问XFR
    j = T_SinTable[PWM_Index];
    if (过流)             ////////////     过流 降压   
j=j*0.6              // 降压为 原来0.6                              //////////////////////////
    PWM3T2H = (u8)(j >> 8);        //第二个翻转计数高字节
    PWM3T2L  = (u8)j;                        //第二个翻转计数低字节
    j += PWM_DeadZone;                //死区if(j==1220)                    ///////////    2个下管 轮流50HZ方波  p2.6 p.27初值,  要与 上管, 的初值 相反。
{  P 2.6=~P 2.6;
   P2.7=~P2.7;
}
    // 指向PWM4   PWM4T2H = (u8)(j >> 8);        //第二个翻转计数高字节
    PWM4T2L = (u8)j;                        //第二个翻转计数低字节
    P_SW2 = SW2_tmp;        //恢复SW2设置
    if(++PWM_Index >= 200)   
    PWM_Index = 0;
  
  }
}


回复

使用道具 举报

23#
ID:844772 发表于 2021-12-28 09:09 | 只看该作者
你仔细研究T_SinTable[]就发现,它有两个1220,按你4个0的说法,T_SinTable[]第一个1220是上个波末和这个波初的两个零,中间一个1220时同一个波上半周和下半周的两个零。
你务必用示波器看一下,特别是看一下死区对不对,再加负载啊,觉得程序还不很成熟。
另外驱动电饭锅也就算了,卷帘门是感性电路啊
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表