找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 2564|回复: 1
收起左侧

基于单片机+VB上位机的调光系统设计(含PCB文件和程序源码)

[复制链接]
ID:78790 发表于 2018-12-28 22:55 | 显示全部楼层 |阅读模式
分享一个采用STC12单片机+485+232通信接口,完成调光系统的硬件设计,同时还设计了一个上位机用于控制系统的调光,整个调光电路的设计采用模拟方式完成,具有一定的参考价值。另外,上位机写了很久了,采用VB写的,仅供参考,现在也基本不用VB了,控制器硬件电路分为控制电路以及转接板电路,程序则分为上位机以及下位机单片机代码,详见附件。

Altium Designer画的原理图和PCB图如下:(51hei附件中可下载工程文件)
0.png 0.png 0.png 0.png

上位机程序+源码:
0.png

单片机源程序如下:
  1. #include "STC12C5A60S2.h"
  2. #include "main.h"
  3. #include "stdio.h"

  4. #define        HeadCode         0xFC

  5. #define SlaveAddr            2
  6. #define        EndCode                  0xed
  7. #define SCIDELAYTIME  22000


  8. u8 i = 0;
  9. unsigned char uart_receive[10];
  10. unsigned char uart_rc_num  = 0;
  11. unsigned char uart_rc_ok   = 0;

  12. unsigned char RX_Dat[10];
  13. unsigned char TX_Dat[10];
  14. unsigned char len;
  15. unsigned char CmdValid = 0;

  16. unsigned char  A_control;
  17. unsigned char  light;
  18. unsigned char  M_control;
  19. unsigned char  Auto=1;
  20. void sys_init()
  21. {
  22.     UartInit();                                                //初始化串口
  23.     InitADC();                      //初始化内部ADC转换器
  24.     pwm_init();                     //初始化内部PWM发生器
  25.     EA = 1;                         //开启总中断
  26. }
  27. //------------------------------------/
  28. //执行上一次串口接收到的指令
  29. //------------------------------------/
  30. void CmdRun(void)
  31. {
  32.     switch (RX_Dat[3])
  33.     {
  34.     case 0xAB:  //手动模式
  35.             M_control =  RX_Dat[5];
  36.                 set_pwm2(255-(M_control));
  37.                 Auto = 0;
  38.         break;
  39.     case 0xAC: //自动模式
  40.             A_control = RX_Dat[5];
  41.                 Auto = 1;        
  42.         break;
  43.     default:
  44.         break;
  45.     }
  46. }

  47. void main(void)
  48. {
  49.     sys_init();                                        //初始化单片机内部模块
  50.     while(1)
  51.              {
  52.                  if(Auto == 1)                        //自动调光打开
  53.                  {
  54.                          if((GetADCResult(1)>A_control)) //环境光比设定值弱
  55.                          {
  56.                                  if(light<255)                                 //限制pwm范围在1-255
  57.                                 {
  58.                                  light++;                                         //增加led亮度
  59.                                 }
  60.                                  set_pwm2(light);                         //pwm值装载到pwm模块
  61.                          }
  62.                          if((GetADCResult(1)<A_control))//环境光比设定值强
  63.                          {
  64.                                  if(light>1)                                        //限制pwm范围在1-255
  65.                                 {
  66.                                  light--;                                        //减弱led亮度
  67.                                 }
  68.                                  set_pwm2(light);                        //pwm值装载到pwm模块  
  69.                          }
  70.                 }

  71.         if(CmdValid==1)                                                 //收到了一条串口发来的指令
  72.         {                                                                    
  73.             CmdValid=0;                                                 //清除标志(指令仅执行一次)
  74.             CmdRun();                                                 //执行串口发来的指令
  75.         }
  76.         
  77.     }
  78. }

  79. // 鎵ц涓插彛鎸囦护

  80. /*----------------------------
  81.                 ===涓插彛涓柇===
  82. ----------------------------*/
  83. void Uart_Isr() interrupt 4 using 1                          //串口中断服务程序
  84. {
  85. /////////////////////////////////////////////////////////////////////////
  86. //// //串口中断服务程序
  87. /////////////////////////////////////////////////////////////////////////
  88.     unsigned char  i=0;
  89.     unsigned int j=0;

  90.     if(RI)                                          //串口接收中断
  91.     {   
  92.        EA = 0;                                //关闭总中断
  93. //                 ES = 0
  94.         RI=0;                                        //清空串口接收中断标志
  95.         RX_Dat[0]=SBUF;
  96.         if(RX_Dat[0] == HeadCode)//判断头码是否正确
  97.         {
  98.             for(i=1;; i++)
  99.             {
  100.                 while(!RI)
  101.                 {
  102.                     j++;
  103.                     if(j>SCIDELAYTIME)break;
  104.                 }
  105.                 if(j<SCIDELAYTIME)   //判断接收是否超时
  106.                 {
  107.                     RX_Dat[i]=SBUF;
  108.                     RI=0;
  109.                     j=0;
  110.                     if(i == 2)
  111.                     {
  112.                         //if((RX_Dat[1]!=SlaveAddr0))break;
  113.                     }
  114.                        if((RX_Dat[i]==EndCode)&&((RX_Dat[1]==SlaveAddr))||((RX_Dat[1]==0x00)))
  115.                     {
  116.                         len=i;
  117.                         CmdValid=1;
  118.                     }
  119.                 }
  120.                 else
  121.                 {
  122.                     //SBUF = 0XA5;//*********
  123.                     break;
  124.                 }
  125.             }
  126.         }
  127.         else
  128.         {
  129.             //            ErrCode = ErrCMD;
  130.             SBUF = 0X5A;//*********
  131.         }
  132.                 RI = 0;     //清除串口接收中断标志位
  133. //      ES = 1                //打开串口中断
  134.         EA = 1;                //打开总中断
  135.     }

  136. }


  137. /*----------------------------
  138. 初始化adc
  139. ----------------------------*/
  140. void InitADC()
  141. {
  142.     P1ASF = 0x03;                   //打开adc通道
  143.     ADC_RES = 0;                    //清楚之前的结果
  144.     ADC_CONTR = ADC_POWER | ADC_SPEEDLL;//ADC开机
  145.     Delay(2);                       //延迟
  146. }



  147. /*----------------------------
  148. 返回adc的值(反应光的强度)
  149. ----------------------------*/
  150. BYTE GetADCResult(BYTE ch)
  151. {
  152.     ADC_CONTR = ADC_POWER | ADC_SPEEDLL | ch | ADC_START;
  153.     Delay(2);
  154.     while (!(ADC_CONTR & ADC_FLAG));//等待AD转换完成标志
  155.     ADC_CONTR &= ~ADC_FLAG;         //关闭ADC
  156.     return ADC_RES;                 //返回adc的值
  157. }


  158. //-----------------pwm初始化 -----------------

  159. void pwm_init()
  160. {
  161.     CCON = 0;                       //初始化pca控制器
  162.     CH = CL= 0;                     //复位pca
  163.     CMOD = 0x01;                    //设置pac时钟
  164.     //不允许中断
  165.     CCAP0H = CCAP0L = 0x00;         //PWM0 初始占空比为0
  166.     CCAPM0 = 0x42;                  //PCA 8位无中断模式

  167.     CCAP1H = CCAP1L = 0x00;         //PWM1初始占空比为0
  168.     CCAPM1 = 0x42;                  //PCA 8位无中断模式

  169. // PCAPWM1 = 0x03;
  170.     CR = 1;                         //PCA 开启


  171. }

  172. void set_pwm2(u8 set_dat)
  173. {
  174.     CCAP1H = CCAP1L = set_dat;
  175. }
  176. /*------------------------
  177. // 串口配置初始化
  178. --------------------------*/
  179. void UartInit(void)                //9600bps@12.000MHz
  180. ……………………

  181. …………限于本文篇幅 余下代码请从51黑下载附件…………
复制代码

所有资料51hei提供下载:
光控上位机+下位机电路+代码.zip (6.6 MB, 下载次数: 63)

评分

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

查看全部评分

回复

使用道具 举报

ID:165792 发表于 2019-1-9 20:06 | 显示全部楼层
好资料,51黑有你更精彩!!!
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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