找回密码
 立即注册

QQ登录

只需一步,快速开始

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

modbus 30路晶体管可编程控制器上位机与单片机下位机源码等全套资料下载

[复制链接]
跳转到指定楼层
楼主
本30路晶体管可编程模块的modbus上位机运行界面如图(附件中有vb.net源码下载)支持rs485与rs232通讯. 可做快递柜 电磁阀的控制


完整版modbus30路完善版本(看门狗)232+485单片机源程序如下:
  1. /******************************
  2. 程序功能:    modbus RTU 模式设置读取十六个继电器状态,以及内部保持寄存器的设置读取
  3. 硬件测试环境:单片机stc89C52RC十六继电器485接口控制板  
  4. 通信协议:    晶振:11.0592  波特率:9600  8位数据 1位停止位 偶校验   485通位接口P3.7控制方向端
  5. 控制板地址:  修改localAddr(变量)
  6. 线圈个数:16个  线圈地址范围:0x0000~0x000F
  7. 保持寄存器个数:16个(字节型) 寄存器地址:0x0000~0x000F
  8. \\******************************\\
  9. 功能码简介:
  10. 02:读取单个线圈状态
  11. 03:读取多个保持寄存器
  12. 05:设置单个线圈状态
  13. 06:设置单个寄存器值
  14. 0F:设置多个线圈
  15. 10:设置多个保持寄存器
  16. *******************************/
  17. #include "hader\\main.h"
  18. uint32        dwTickCount,dwIntTick;        //时钟
  19. uint8        idata sendBuf[32],receBuf[16]; //发送接收缓冲区
  20. uint8        idata checkoutError;        // ==2 偶校验错  
  21. uint8        idata receTimeOut;                //接收超时
  22. uint8        idata c10ms;                        //10ms 计时
  23. uint8   idata c200ms;
  24. bit                b1ms,bt1ms,b10ms,bt10ms,b100ms,bt100ms;        //定时标志位


  25. //定时处理
  26. void timeProc(void)
  27. {        
  28.     b1ms = 0;
  29.         if(bt1ms)        //如果1ms到
  30.         {
  31.                 bt1ms = 0;
  32.                 b1ms = 1;
  33.         if(receTimeOut>0)        //如果接收超时值>0
  34.         {
  35.             receTimeOut--;        //接收超时-1(1ms减1次)
  36.             if(receTimeOut==0 && receCount>0)   //判断通讯接收是否超时
  37.             {
  38.             //    b485Send = 0;       //将485置为接收状态
  39.                 receCount = 0;      //将接收地址偏移寄存器清零
  40.                                 checkoutError = 0;
  41.             }
  42.         }
  43.         }
  44. }   // void TimerProc(void)
  45. //初始化
  46. void initInt(void)
  47. {
  48.     SCON = 0xd0;
  49.         TMOD = 0x21;
  50.         PCON = 0;
  51.         TH0 = TIMER_HIGHT;
  52.         TL0 = TIMER_LOW;
  53.         TH1=  0xfd;        
  54.         TL1 = 0xfd;                             //波特率 9600
  55.    
  56.         TR0 = 1;
  57.         TR1=1;                        
  58.     ET0 = 1;
  59.         ES = 1;
  60.         EA = 1;        

  61.         //串口2设置        
  62.         S2CON = 0xd0;                          //方式1,9位数据,波特率不可变  S2TB8 偶校验位
  63.         BRT=0XFD;                                  //设置波特率9600
  64.         AUXR=0x10;                                 //启动串口1波特率发生器
  65.         IP=0x00;                                 //优先级默认                                         //开串口1中断
  66.         IE2=0x01;                                 //开串口2中断               
  67. }
  68. //初始化
  69. void initProg(void)
  70. {        

  71.    P4SW|=0x20;                   //配置P4.5为IO口
  72.    P4M0|=0x10;                   //配置P4.4为IO口
  73.    P4M1|=0x10;
  74.    P0=P1=P2=P3=0xff;
  75.    P4|=0x30;
  76.         initInt(); //初始化定时器
  77. //        b485Send = 0;
  78. }
  79. //上电时读取上次线圈状态,并设置
  80. void forceMultipleCoils1()
  81. {
  82.         uint8 tempAddr;
  83.         uint8 i,k;
  84.     uint8 Data;
  85.         uint8  exit = 0;                                                        
  86.         for(k=0;k<4;k++)
  87.         {
  88.            switch(k)
  89.           {        
  90.             case 0:Data=coilreg1; break;
  91.                 case 1:Data=coilreg2; break;
  92.                 case 2:Data=coilreg3; break;
  93.                 case 3:Data=coilreg4; break;
  94.            }
  95.                 for(i=0;i<8;i++)
  96.                 {
  97.                     if(        Data &0x01==1)
  98.                            setCoilVal(tempAddr,0);
  99.                         else
  100.                            setCoilVal(tempAddr,1);

  101.                         Data=Data>>1;        
  102.                         tempAddr++;
  103.                         if(tempAddr >=32)
  104.                         {
  105.                                 exit = 1;
  106.                                 break;
  107.                         }        
  108.                 }
  109.                 if(exit==1)
  110.                         break;
  111.         }
  112.         }
  113. void main(void)
  114. {
  115.    
  116.         initProg();
  117.         localAddr=EEPROMReadByte(0);     //从EERPOM的相对0地址读取数据
  118.     coilreg1=EEPROMReadByte(1);
  119.         coilreg2=EEPROMReadByte(2);
  120.         coilreg3=EEPROMReadByte(3);
  121.         coilreg4=EEPROMReadByte(4);
  122.         forceMultipleCoils1();
  123.         if(localAddr>=10)
  124.         {
  125.           localAddr=1;
  126.           EEPROMSectorErase(0);       //从EEPROM的相对0地址扇区擦除
  127.       EEPROMWriteByte(0,localAddr);
  128.           EEPROMWriteByte(1,coilreg1);
  129.           EEPROMWriteByte(2,coilreg2);
  130.           EEPROMWriteByte(3,coilreg1);
  131.           EEPROMWriteByte(4,coilreg2);
  132.         
  133.         }
  134.         WDT_CONTR =0x32;   //大概284.4ms
  135.         while(1)
  136.         {
  137.                 timeProc();
  138.                 checkComm0Modbus();
  139.         }
  140. }
  141. //定时器0 1ms 中断
  142. void timer0IntProc() interrupt 1
  143. {
  144.         TL0 = TIMER_LOW;
  145.     TH0 = TIMER_HIGHT;
  146.     dwIntTick++;
  147.         bt1ms = 1;
  148.     c10ms++;
  149.         c200ms++;
  150.     if(c10ms >= 10)
  151.     {
  152.         c10ms = 0;      //20ms计时器清零
  153.         bt10ms = 1;
  154.             WDT_CONTR =0x32;        //   
  155.     }
  156. }   // void Timer0IntProc()
  157. // 串行中断1程序
  158. void commIntProc() interrupt 4
  159. {
  160.         if(TI)
  161.         {
  162.                 TI = 0;
  163.                 if(sendPosi < sendCount) //如果发送位置小于发送计数,那么继续发送
  164.                 {
  165.                         sendPosi++;
  166.                         ACC = sendBuf[sendPosi];
  167.                         TB8 = P;        //加上校验位
  168.                         SBUF = sendBuf[sendPosi];
  169.                 }
  170.                 else //否则发送完毕,置接收状态
  171.                 {
  172.                 //        b485Send = 0;    //发送完后将485置于接收状态
  173.                         receCount = 0;   //清接收地址偏移寄存器
  174.                         checkoutError = 0;
  175.                 }
  176.         }
  177.         else if(RI)
  178.         {
  179.                 RI = 0;
  180.                 receTimeOut = 10;    //通讯超时值
  181.                 receBuf[receCount] = SBUF;
  182.                 ACC = receBuf[receCount];
  183.                 if(P != RB8)
  184.                         checkoutError = 2;        //偶校验出错
  185.                 receCount++;          //接收地址偏移寄存器加1
  186.                 receCount &= 0x0f;    //最多一次只能接收16个字节
  187.         }
  188. }   // void CommIntProc()
  189. //串口2中断
  190. void uart2_isr()  interrupt 8
  191. {
  192.   
  193.         if( S2CON & 0x02 )
  194.         {
  195.                 S2CON &= ~0x02;
  196.                    if(sendPosi<receCount) //如果发送位置小于发送计数,那么继续发送
  197.                 {
  198.                         sendPosi++;
  199.                         ACC = receBuf[sendPosi];
  200.                         if(P)
  201.                            S2CON|=0x08;         //                                7      6      5      4      3      2     1     0        Reset Value
  202.                              //sfr S2CON = 0x9A; //S2 Control  S2SM0  S2SM1  S2SM2  S2REN  S2TB8  S2RB8  S2TI  S2RI      00000000B
  203.                         else
  204.                            S2CON&=0xf7;
  205.                         S2BUF =receBuf[sendPosi];
  206.                 }
  207.                 else //否则发送完毕,置接收状态
  208.                 {
  209. //                        b485Send = 0;    //发送完后将485置于接收状态
  210.                         receCount = 0;   //清接收地址偏移寄存器
  211.                         checkoutError = 0;
  212.                 }
  213.          }
  214.         else if( S2CON & 0x01)//接收
  215.         {
  216.             S2CON &= ~0x01;
  217.                 receTimeOut = 10;    //通讯超时值
  218.                 receBuf[receCount] =S2BUF;
  219.                 ACC = receBuf[receCount];
  220. //                if(P != RB8)
  221. //                        checkoutError = 2;        //偶校验出错
  222.                 receCount++;          //接收地址偏移寄存器加1
  223.                 receCount &= 0x0f;    //最多一次只能接收16个字节
  224.         }
  225. }
复制代码

简要说明:
一、 尺寸:长180mmX110mmX22mm
二、 主要芯片:STC12C60S2支持双串口(支持51系列DIP40封装单片机
三、 工作电压:直流6~24供电可控制负载6~24V
四、 串口1COM1)下载程序,(RS485口使用单机机内部串口2均可与上位机通讯)
五、输入输出采用独立供电,也可以公用一个电源,独立供电可减少负载对单片机电路的影响
六、 特点:
             1、具有电源指示。
             2、三十路带光电隔离输出控制场效应管。
              3、标准的11.0592M晶振。(便于设置串口波特率)
             4、具有上电复位和手动复位。
             5、支持51系列DIP40封装单片机。  
       5、输出三十路场效应管的供电采用独立供电 (可避免主控电路与被控电路干扰)        
             6、输出三十路具有LED指示。
              7、三十路IRF1205场效应管
             8、可控制直流36V以内 50W负载设备。最大输出电流6A
             9、具有RS232和RS485通信功能(可以同时和上位机通信)。
             10、单片机无加密,可插拔更换,可随意更改程序。
使用说明:
【标注说明】

【接线图】


【应用举例接线图】



全部资料51hei下载地址vb.net上位机与单片机下位源程序与电路原理图等全套开发资料

0.png (56.57 KB, 下载次数: 62)

0.png

30路晶体管可编程模块485及232产品使用手册.doc

4.95 MB, 下载次数: 50, 下载积分: 黑币 -5

GYJ-0077_30路场效应管可编程模块485及232.rar

11.91 MB, 下载次数: 63, 下载积分: 黑币 -5

定时发送命令30路源码安装包下位机.7z

17.22 MB, 下载次数: 62, 下载积分: 黑币 -5

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

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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