找回密码
 立即注册

QQ登录

只需一步,快速开始

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

六路输入输出工控板完整版modbus-RTU程序带(看门狗)232+485通讯源码分享

[复制链接]
跳转到指定楼层
楼主
ID:284341 发表于 2023-7-3 11:01 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
单片机采用51系列 STC12C5A60S2单片机  
整个程序源码在附件中。
/******************************
程序功能:    modbus RTU 模式设置读取十六个继电器状态,以及内部保持寄存器的设置读取
硬件测试环境:单片机stc89C52RC十六继电器485接口控制板  
通信协议:    晶振:11.0592  波特率:9600  8位数据 1位停止位 偶校验   485通位接口P3.7控制方向端
控制板地址:  修改localAddr(变量)
线圈个数:16个  线圈地址范围:0x0000~0x000F
保持寄存器个数:16个(字节型) 寄存器地址:0x0000~0x000F
\\******************************\\
功能码简介:
02:读取单个线圈状态
03:读取多个保持寄存器
05:设置单个线圈状态
06:设置单个寄存器值
0F:设置多个线圈
10:设置多个保持寄存器
*******************************/
#include "hader\\main.h"
uint32        dwTickCount,dwIntTick;        //时钟
uint8        idata sendBuf[32],receBuf[16]; //发送接收缓冲区
uint8        idata checkoutError;        // ==2 偶校验错  
uint8        idata receTimeOut;                //接收超时
uint8        idata c10ms;                        //10ms 计时
uint8   idata c200ms;
bit                b1ms,bt1ms,b10ms,bt10ms,b100ms,bt100ms;        //定时标志位


//定时处理
void timeProc(void)
{       
    b1ms = 0;
        if(bt1ms)        //如果1ms到
        {
                bt1ms = 0;
                b1ms = 1;
        if(receTimeOut>0)        //如果接收超时值>0
        {
            receTimeOut--;        //接收超时-1(1ms减1次)
            if(receTimeOut==0 && receCount>0)   //判断通讯接收是否超时
            {
            //    b485Send = 0;       //将485置为接收状态
                receCount = 0;      //将接收地址偏移寄存器清零
                                checkoutError = 0;
            }
        }
        }
}   // void TimerProc(void)
//初始化
void initInt(void)
{
    SCON = 0xd0;
        TMOD = 0x21;
        PCON = 0;
        TH0 = TIMER_HIGHT;
        TL0 = TIMER_LOW;
        TH1=  0xfd;       
        TL1 = 0xfd;                             //波特率 9600

        TR0 = 1;
        TR1=1;                       
    ET0 = 1;
        ES = 1;
        EA = 1;       

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

   P4SW|=0x20;                   //配置P4.5为IO口
   P4M0|=0x10;                   //配置P4.4为IO口
   P4M1|=0x10;
   P0=P1=P2=P3=0xff;
   P4|=0x30;
        initInt(); //初始化定时器
//        b485Send = 0;
}
//上电时读取上次线圈状态,并设置
void forceMultipleCoils1()
{
        uint8 tempAddr;
        uint8 i,k;
    uint8 Data;
        uint8  exit = 0;                                                       
        for(k=0;k<4;k++)
        {
           switch(k)
          {       
            case 0:Data=coilreg1; break;
                case 1:Data=coilreg2; break;
                case 2:Data=coilreg3; break;
                case 3:Data=coilreg4; break;
           }
                for(i=0;i<8;i++)
                {
                    if(        Data &0x01==1)
                           setCoilVal(tempAddr,0);
                        else
                           setCoilVal(tempAddr,1);

                        Data=Data>>1;       
                        tempAddr++;
                        if(tempAddr >=32)
                        {
                                exit = 1;
                                break;
                        }       
                }
                if(exit==1)
                        break;
        }
        }
void main(void)
{

        initProg();
        localAddr=EEPROMReadByte(0);     //从EERPOM的相对0地址读取数据
    coilreg1=EEPROMReadByte(1);
        coilreg2=EEPROMReadByte(2);
        coilreg3=EEPROMReadByte(3);
        coilreg4=EEPROMReadByte(4);
        forceMultipleCoils1();
        if(localAddr>=10)
        {
          localAddr=1;
          EEPROMSectorErase(0);       //从EEPROM的相对0地址扇区擦除
      EEPROMWriteByte(0,localAddr);
          EEPROMWriteByte(1,coilreg1);
          EEPROMWriteByte(2,coilreg2);
          EEPROMWriteByte(3,coilreg1);
          EEPROMWriteByte(4,coilreg2);
       
        }
        WDT_CONTR =0x32;   //大概284.4ms
        while(1)
        {
                timeProc();
                checkComm0Modbus();
        }
}
//定时器0 1ms 中断
void timer0IntProc() interrupt 1
{
        TL0 = TIMER_LOW;
    TH0 = TIMER_HIGHT;
    dwIntTick++;
        bt1ms = 1;
    c10ms++;
        c200ms++;
    if(c10ms >= 10)
    {
        c10ms = 0;      //20ms计时器清零
        bt10ms = 1;
            WDT_CONTR =0x32;        //   
    }
}   // void Timer0IntProc()
// 串行中断1程序
void commIntProc() interrupt 4
{
        if(TI)
        {
                TI = 0;
                if(sendPosi < sendCount) //如果发送位置小于发送计数,那么继续发送
                {
                        sendPosi++;
                        ACC = sendBuf[sendPosi];
                        TB8 = P;        //加上校验位
                        SBUF = sendBuf[sendPosi];
                }
                else //否则发送完毕,置接收状态
                {
                //        b485Send = 0;    //发送完后将485置于接收状态
                        receCount = 0;   //清接收地址偏移寄存器
                        checkoutError = 0;
                }
        }
        else if(RI)
        {
                RI = 0;
                receTimeOut = 10;    //通讯超时值
                receBuf[receCount] = SBUF;
                ACC = receBuf[receCount];
                if(P != RB8)
                        checkoutError = 2;        //偶校验出错
                receCount++;          //接收地址偏移寄存器加1
                receCount &= 0x0f;    //最多一次只能接收16个字节
        }
}   // void CommIntProc()
//串口2中断
void uart2_isr()  interrupt 8
{

        if( S2CON & 0x02 )
        {
                S2CON &= ~0x02;
                   if(sendPosi<receCount) //如果发送位置小于发送计数,那么继续发送
                {
                        sendPosi++;
                        ACC = receBuf[sendPosi];
                        if(P)
                           S2CON|=0x08;         //                                7      6      5      4      3      2     1     0        Reset Value
                             //sfr S2CON = 0x9A; //S2 Control  S2SM0  S2SM1  S2SM2  S2REN  S2TB8  S2RB8  S2TI  S2RI      00000000B
                        else
                           S2CON&=0xf7;
                        S2BUF =receBuf[sendPosi];
                }
                else //否则发送完毕,置接收状态
                {
//                        b485Send = 0;    //发送完后将485置于接收状态
                        receCount = 0;   //清接收地址偏移寄存器
                        checkoutError = 0;
                }
         }
        else if( S2CON & 0x01)//接收
        {
            S2CON &= ~0x01;
                receTimeOut = 10;    //通讯超时值
                receBuf[receCount] =S2BUF;
                ACC = receBuf[receCount];
//                if(P != RB8)
//                        checkoutError = 2;        //偶校验出错
                receCount++;          //接收地址偏移寄存器加1
                receCount &= 0x0f;    //最多一次只能接收16个字节
        }
}

0040 完整版modbus6路完善版本(看门狗)232 485.zip

1.2 MB, 下载次数: 45, 下载积分: 黑币 -5

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

使用道具 举报

沙发
ID:1097786 发表于 2024-1-30 13:19 | 只看该作者
有原理图和pcb文件吗
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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