找回密码
 立即注册

QQ登录

只需一步,快速开始

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

上位机控制单片机pwm程序与Proteus仿真

[复制链接]
跳转到指定楼层
楼主
通过串口助手和虚拟串口,实现仿真,上位机发送1/2/3.....控制占空比。
仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)


单片机源程序如下:
  1. #include<reg52.h>
  2. typedef unsigned char uint8;
  3. typedef unsigned int  uint16;
  4. #define uint unsigned int
  5. uint pp;
  6. char num=2,dis;
  7. #define jingzhen     11059200UL                         /*使用晶体*/         
  8. #define botelv   9600UL                     /*波特率定义为9600*/
  9. unsigned char zifuchuan[]="清翔电子开发板是您最好的选择!\n";                        //待显示字符。
  10. volatile unsigned char sending;
  11. sbit down = P3^5;
  12. sbit up = P3^6;
  13. sbit  led=P1^0;
  14. uint8 t = 0;
  15. uint8 PWM_T = 0;   //占空比控制变
  16. unsigned char posit=0;
  17. sbit W1=P2^4;
  18. sbit W2=P2^5;
  19. sbit W3=P2^6;
  20. sbit W4=P2^7;

  21. unsigned char buf;
  22. unsigned char code table[]=
  23. {0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90,0xBF,0x00};
  24. unsigned char disbuff[4]={0,0,0,0};        
  25. void delayms(unsigned int z) //延时 z ms   //这里用来修改是12延时还是51 ,带xy的是12
  26. {        
  27.         int i;
  28.         while(z--)
  29.         for(i=0;i<110;i++);        
  30. }
  31. void keyscan()
  32. {
  33.                    if(!up)
  34.                          {
  35.                                 if(PWM_T<100)
  36.                                         {
  37.                                           PWM_T=PWM_T+10;
  38.                                                 
  39.                                         }num++;
  40.                                         if(num>9) num=9;
  41.                                 while(up==0);
  42.                         
  43.                         }
  44.                  if(!down)
  45.                          {
  46.                                 if(PWM_T>0)
  47.                                         {
  48.                                           PWM_T=PWM_T-10;
  49.                                                 
  50.                                         }num--;
  51.                                         if(num<0) num=0;
  52.                                         while(down==0);
  53.                         
  54.                         }

  55.         
  56. }
  57. void play()   //显示数字
  58. {
  59.          P0=0XFF;
  60.         
  61.          if(posit==0)P0=table[1];               
  62.          if(posit==1)P0=table[0];
  63.          if(posit==2){P0=table[num];}
  64.                 switch(posit)
  65.                 {
  66.                         case 0 : W1=1;W2=0;W3=0;W4=0; break;
  67.                         case 1 : W1=0;W2=1;W3=0;W4=0; break;
  68.                         case 2 : W1=0;W2=0;W3=1;W4=0; break;
  69.                         //case 3 : W1=0;W2=0;W3=0;W4=1; break;
  70.                 }
  71.                 delayms(5);
  72.                 posit++;
  73.                 if(posit>2)                //每进一次显示函数,变量加1
  74.                         posit=0;         
  75. }
  76. void init()
  77. {
  78.         SCON = 0x50;                        // SCON: 方式 1, 8-bit, 允许接收数据
  79.         TMOD |= 0x20;               // TMOD: 设置定时器1工作在方式2, 8-bit 自动重装
  80.         TH1 = 0xFD;               // TH1:  初始值为0xFD  波特率:9600 晶振频率:11.0592MHz  
  81.         TL1 = 0x0;
  82.         TR1 = 1;                  // TR1:  开启定时器1                        
  83.         EA  = 1;                  //打开总中断
  84.         ES  = 1;                  //打开串口中断
  85. }
  86. // 发送一个字节数据
  87. void uart_send_byte(unsigned char dat)
  88. {
  89.         SBUF = dat;
  90.         while(!TI);
  91.         TI = 0;
  92. }

  93. // 发送字符串
  94. void uart_send_str(unsigned char *s)
  95. {
  96.         while(*s != '\0')
  97.         {
  98.                 uart_send_byte(*s);
  99.                 s++;
  100.         }
  101. }

  102. void main()
  103. {
  104.         TMOD = 0x02;   //定时器0,工作模式2,8位定时模式
  105.         TH0=210;     //写入预置初值(取值1-255,数越大PWM频率越高)
  106.         TL0=210;     //写入预置值 (取值1-255,数越大PWM频率越高)
  107.         TR0=1;       //启动定时器
  108.         ET0=1;       //允许定时器0中断
  109.         EA=1;        //允许总中断
  110.         led=1;          //初始化P1,输出端口
  111.         PWM_T=30;
  112.         init();
  113.         while(1)
  114.         {
  115.                 keyscan();

  116.            play();//显示num当前值,占空比为num/10。
  117.         }
  118. }

  119. /****************************************************
  120.                /定时器0中断模拟PWM
  121. ****************************************************/
  122. timer0() interrupt 1  
  123. {
  124.          t++;    //每次定时器溢出加1
  125.          
  126.          if(t==100)   //PWM周期 100个单位
  127.                  {
  128.                           t=0;  //使t=0,开始新的PWM周期
  129.                           led=0;  //输出端口
  130.                  }
  131.          
  132.          if(PWM_T==t)  //按照当前占空比切换输出为高电平
  133.                  {  
  134.                          led=1;        //  
  135.                  }

  136. }

  137. void uart_interrupt(void) interrupt 4                 //也叫串行中断服务程序
  138. {
  139.         unsigned char recv_data;// 用来存放接收到的数据
  140.         unsigned char send_data[] = "I received.\n";// 要发送的信息
  141.         
  142.         if(RI) //接收数据(1字节)完毕,RI会被硬件置1
  143.         {
  144.                 RI = 0;                            // 将 接收中断标志位 清零(让串口可以继续接收数据)
  145.                 recv_data = SBUF;                   //读取接收到的数据,并存放到data
  146.         
  147.                
  148.                 if(recv_data == '0'){num=0;PWM_T=0;}               
  149.                 if(recv_data == '1'){num=1;PWM_T=10;}               
  150.                 if(recv_data == '2'){num=2;PWM_T=20;}               
  151.                 if(recv_data == '3'){num=3;PWM_T=30;}               
  152.                 if(recv_data == '4'){num=4;PWM_T=40;}               
  153.           if(recv_data == '5'){num=5;PWM_T=50;}
  154.                 if(recv_data == '6'){num=6;PWM_T=60;}               
  155.                 if(recv_data == '7'){num=7;PWM_T=70;}               
  156.                 if(recv_data == '8'){num=8;PWM_T=80;}
  157.                 if(recv_data == '9'){num=9;PWM_T=90;}
  158.                 play();
  159.           uart_send_str(send_data);
  160.         }
  161.         if(TI)
  162.         {
  163.                 TI = 0;
  164.         }
  165. }

  166. /*void  serial() interrupt 4
  167. {
  168.    ES = 0;                //关闭串行中断
  169.    RI = 0;                //清除串行接受标志位
  170.    buf = SBUF;            //从串口缓冲区取得数据
  171.   switch(buf)
  172.    {
  173.       case '0':  num=0;PWM_T=0;break;  
  174.                   case 0x32:  num=1;PWM_T=10;break;         
  175.       case 0x33:  num=2;PWM_T=20;break;         
  176.       case 0x34:  num=3;PWM_T=30;break;         
  177.       case 0x35:  num=4;PWM_T=40;break;        
  178.       case 0x36:  num=5;PWM_T=50;break;              
  179.       case 0x37:  num=6;PWM_T=60;break;                     
  180.       case 0x38:  num=7;PWM_T=70;break;  
  181.             case 0x39:  num=8;PWM_T=80;break;  
  182.                   case 0x40:  num=9;PWM_T=90;break;  
  183.             default:    num=10; break;         
  184.    }
  185.          play();
  186.    ES = 1;    //允许串口中断
  187. }*/
复制代码

串口助手下载:http://www.51hei.com/bbs/dpj-199058-1.html
vspd虚拟串口助手可以到百度搜索或者51hei下载

所有资料51hei提供下载:
上位机控制pwm.zip (4.97 MB, 下载次数: 24)


评分

参与人数 1黑币 +50 收起 理由
admin + 50 共享资料的黑币奖励!

查看全部评分

分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏 分享淘帖 顶 踩
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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