附MODBUS协议
#include "modbus.h"
#include "usart.h"
#include "delay.h"
//********************************************************************************/
//modbus协议支持485总线
//修改日期:2013/12/9
//版本:V1.0
//Copyright(C) 象牙塔 All rights reserved
//Email:cronus_skl@163.com QQ:374199080
//********************************************************************************/
u8 MODBUS_SEND_SBUF[64];
u8 MODBUS_RECEIVE_SBUF[256];
u8 Defaultspec=0;
u8 baseAddress=0;
//字地址 0 - 255 (只取低8位)
//位地址 0 - 255 (只取低8位)
/* CRC 高位字节值表 */
const u8 auchCRCHi[] = {
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0/**/,
0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0,
0x80, 0x41, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1,
0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1,
0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,
0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40,
0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1,
0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,
0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40,
0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0,
0x80, 0x41, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,
0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0,
0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0,
0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40,
0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1,
0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,
0x80, 0x41, 0x00, 0xC1, 0x81, 0x40
};
/* CRC低位字节值表*/
const u8 auchCRCLo[] = {
0x00, 0xC0, 0xC1, 0x01, 0xC3, 0x03, 0x02, 0xC2, 0xC6, 0x06/**/,
0x07, 0xC7, 0x05, 0xC5, 0xC4, 0x04, 0xCC, 0x0C, 0x0D, 0xCD,
0x0F, 0xCF, 0xCE, 0x0E, 0x0A, 0xCA, 0xCB, 0x0B, 0xC9, 0x09,
0x08, 0xC8, 0xD8, 0x18, 0x19, 0xD9, 0x1B, 0xDB, 0xDA, 0x1A,
0x1E, 0xDE, 0xDF, 0x1F, 0xDD, 0x1D, 0x1C, 0xDC, 0x14, 0xD4,
0xD5, 0x15, 0xD7, 0x17, 0x16, 0xD6, 0xD2, 0x12, 0x13, 0xD3,
0x11, 0xD1, 0xD0, 0x10, 0xF0, 0x30, 0x31, 0xF1, 0x33, 0xF3,
0xF2, 0x32, 0x36, 0xF6, 0xF7, 0x37, 0xF5, 0x35, 0x34, 0xF4,
0x3C, 0xFC, 0xFD, 0x3D, 0xFF, 0x3F, 0x3E, 0xFE, 0xFA, 0x3A,
0x3B, 0xFB, 0x39, 0xF9, 0xF8, 0x38, 0x28, 0xE8, 0xE9, 0x29,
0xEB, 0x2B, 0x2A, 0xEA, 0xEE, 0x2E, 0x2F, 0xEF, 0x2D, 0xED,
0xEC, 0x2C, 0xE4, 0x24, 0x25, 0xE5, 0x27, 0xE7, 0xE6, 0x26,
0x22, 0xE2, 0xE3, 0x23, 0xE1, 0x21, 0x20, 0xE0, 0xA0, 0x60,
0x61, 0xA1, 0x63, 0xA3, 0xA2, 0x62, 0x66, 0xA6, 0xA7, 0x67,
0xA5, 0x65, 0x64, 0xA4, 0x6C, 0xAC, 0xAD, 0x6D, 0xAF, 0x6F,
0x6E, 0xAE, 0xAA, 0x6A, 0x6B, 0xAB, 0x69, 0xA9, 0xA8, 0x68,
0x78, 0xB8, 0xB9, 0x79, 0xBB, 0x7B, 0x7A, 0xBA, 0xBE, 0x7E,
0x7F, 0xBF, 0x7D, 0xBD, 0xBC, 0x7C, 0xB4, 0x74, 0x75, 0xB5,
0x77, 0xB7, 0xB6, 0x76, 0x72, 0xB2, 0xB3, 0x73, 0xB1, 0x71,
0x70, 0xB0, 0x50, 0x90, 0x91, 0x51, 0x93, 0x53, 0x52, 0x92,
0x96, 0x56, 0x57, 0x97, 0x55, 0x95, 0x94, 0x54, 0x9C, 0x5C,
0x5D, 0x9D, 0x5F, 0x9F, 0x9E, 0x5E, 0x5A, 0x9A, 0x9B, 0x5B,
0x99, 0x59, 0x58, 0x98, 0x88, 0x48, 0x49, 0x89, 0x4B, 0x8B,
0x8A, 0x4A, 0x4E, 0x8E, 0x8F, 0x4F, 0x8D, 0x4D, 0x4C, 0x8C,
0x44, 0x84, 0x85, 0x45, 0x87, 0x47, 0x46, 0x86, 0x82, 0x42,
0x43, 0x83, 0x41, 0x81, 0x80, 0x40
};
/***************************CRC校验码生成函数 ********************************
*函数功能:生成CRC校验码
*本代码中使用查表法,以提高运算速度
****************************************************************************/
u16 crc16(u8 *puchMsg, u16 usDataLen)
{
u8 uchCRCHi = 0xFF ; /* 高CRC字节初始化 */
u8 uchCRCLo = 0xFF ; /* 低CRC 字节初始化 */
u16 uIndex ; /* CRC循环中的索引 */
while (usDataLen--) /* 传输消息缓冲区 */
{
uIndex = uchCRCHi ^ *puchMsg++ ; /* 计算CRC */
uchCRCHi = uchCRCLo ^ auchCRCHi[uIndex] ;
uchCRCLo = auchCRCLo[uIndex] ;
}
return (uchCRCLo << 8 | uchCRCHi) ;
}
/********************************读写线圈*********************************/
/***************主要功能码:读线圈,写单个线圈,写多个线圈****************/
/*************************************************************************
*SendReadCoilCommand();读线圈(0X01):最多读256个线圈
*SD:地址(1)+功能码(1)+起始地址(2)+线圈数量(2)+CRC 发送读线圈命令
*RD:地址(1)+功能码(1)+字节数N(1)+状态(N)+CRC 接受读线圈数据
*输入:StartingAddress:起始地址;CoilNumber:线圈数量
*输出:无
**************************************************************************/
//发送命令
//最多可读256个线圈
void SendReadCoilCommand(u8 StartingAddress,u8 CoilNumber)
{
u16 crcData;
MODBUS_SEND_SBUF[0] = baseAddress;//地址
MODBUS_SEND_SBUF[1] = 0X01;//功能码
MODBUS_SEND_SBUF[2] = 0X00;//读地址只有48个0X2C,远小于0XFF个
MODBUS_SEND_SBUF[3] = StartingAddress;
MODBUS_SEND_SBUF[4] = 0X00;
MODBUS_SEND_SBUF[5] = CoilNumber;
crcData = crc16(MODBUS_SEND_SBUF,6);
MODBUS_SEND_SBUF[6] = crcData & 0xff; // CRC代码低位在前
MODBUS_SEND_SBUF[7] = crcData >> 8; //高位在后
RS485_Send_Data(MODBUS_SEND_SBUF,8);
// MODBUS_SEND_SBUF[]=0; 需不需要清零
}
//返回数据
u8 ReceiveReadCoilData(void)
{
u8 result=0;
u16 crcData;
RS485_Receive_Data(MODBUS_RECEIVE_SBUF,&RS485_RX_CNT);
crcData = crc16(MODBUS_RECEIVE_SBUF,RS485_RX_CNT-1);
if(MODBUS_RECEIVE_SBUF[RS485_RX_CNT-1] == ( crcData&0xff ) || MODBUS_RECEIVE_SBUF[RS485_RX_CNT] == ( crcData >> 8 ))
{
RS485_RX_CNT=0; //长度清零
return result=0;
}
else
{
RS485_RX_CNT=0; //长度清零
printf("读线圈出错!");
result=0X01;
return result;
}
}
/***************************************************************************
*写单个线圈(0X05): 可写线圈地址0~31
*SD: 地址(1)+功能码(1)+输出地址(2)+输出值(2)+CRC
*RD: 地址(1)+功能码(1)+输出地址(2)+输出值(2)+CRC
*输入:ExportAddress:输出地址;ExportData:输出值
*
***************************************************************************/
//发送命令
//ExportData 只能是0XFF 或 0X00
//地址范围0-31个
void SendWriteSingleCommand(u8 ExportAddress,u8 ExportData)
{
u16 crcData;
u8 i;
MODBUS_SEND_SBUF[0] = baseAddress;//地址
MODBUS_SEND_SBUF[1] = 0X05;//功能码
MODBUS_SEND_SBUF[2] = 0X00;//读地址只有48个0X2C,远小于0XFF个
MODBUS_SEND_SBUF[3] = ExportAddress;
MODBUS_SEND_SBUF[4] = ExportData;
MODBUS_SEND_SBUF[5] = 0X00;
crcData = crc16(MODBUS_SEND_SBUF,6);
MODBUS_SEND_SBUF[6] = crcData & 0xff; // CRC代码低位在前
MODBUS_SEND_SBUF[7] = crcData >> 8 ; //高位在后
RS485_Send_Data(MODBUS_SEND_SBUF,8);
for(i=0;i<8;i++)
{
printf("\n%2X\n\r",MODBUS_SEND_SBUF);
}
// MODBUS_SEND_SBUF[]=0; 需不需要清零
}
//返回数据
u8 ReceiveWriteSingleData(void)
{
u8 result=0;
u16 crcData;
RS485_Receive_Data(MODBUS_RECEIVE_SBUF,&RS485_RX_CNT);
crcData = crc16(MODBUS_RECEIVE_SBUF,RS485_RX_CNT-1);
if(MODBUS_RECEIVE_SBUF[RS485_RX_CNT-1] == ( crcData&0xff ) || MODBUS_RECEIVE_SBUF[RS485_RX_CNT] == ( crcData >> 8 ))
{
RS485_RX_CNT=0; //长度清零
return result=0;
}
else
{
RS485_RX_CNT=0; //长度清零
printf("写单个线圈出错!");
result=0X05;
return result;
}
}
/***************************************************************************
*写多个线圈(0X0F): 可写线圈地址0~31
*SD:地址(1)+功能码(1)+起始地址(2)+输出数量(2)+字节数量N(1)+输出值(N字节)+CRC
*RD: 地址(1)+功能码(1)+起始地址(2)+输出数量(2)+CRC
*查询——0X11 0X0F 0X00 0X13 0X00 0X0A 0X02 0XCD 0X01 0XBF 0X0B
*从机地址-功能码-寄存器地址高字节-寄存器地址低字节-寄存器数量高字节-寄存器数量
*低字节-字节数-数据1-数据2-CRC校验高字节-CRC校验低字节
* 001AH 0019H 0018H 0017H 0016H 0015H 0014H 0013H
* 1 1 0 0 1 1 0 1
* 0022H 0021H 0020H 001FH 001EH 001DH 001CH 001BH
* 0 0 0 0 0 0 0 1
*传输的第一个字节CDH对应线圈为0013H到001AH,LSB(最低位)对应0013H
*输入:StartAddress:起始地址 ExportNumber:输出数量 ByteNumber:字节数量
* ExportData:输出值(本代码里,最多4个字节)
***************************************************************************/
//发送命令
//地址范围0-31个,即最多32个地址,因此字节数是4Bit
void SendWriteMulCoilCommand(u8 StartAddress,u8 ExportNumber,u8 ByteNumber,u32 ExportData)
{
u16 crcData;
u8 i;
MODBUS_SEND_SBUF[0] = baseAddress;//地址
MODBUS_SEND_SBUF[1] = 0X0F;//功能码
MODBUS_SEND_SBUF[2] = 0X00;//起始地址高,写地址只有32个0X2C,远小于0XFF个
MODBUS_SEND_SBUF[3] = StartAddress;//起始地址低位
MODBUS_SEND_SBUF[4] = 0X00; //输出数量高位
MODBUS_SEND_SBUF[5] = ExportNumber;//输出数量低位
MODBUS_SEND_SBUF[6] = ByteNumber;//字节数
// if((ByteNumber>=0)&&(ByteNumber<8)){i=1;}
// else if((8<=ByteNumber)&&(ByteNumber<16)){i=2;}
// else if((16<=ByteNumber)&&(ByteNumber<24)){i=3;}
// else if((24<=ByteNumber)&&(ByteNumber<32)){i=4;}
// else{i=5;}
switch(ByteNumber)
{
case 1:
MODBUS_SEND_SBUF[7]=ExportData&0XFF;
crcData = crc16(MODBUS_SEND_SBUF,8);
MODBUS_SEND_SBUF[8] = crcData & 0xff; // CRC代码低位在前
MODBUS_SEND_SBUF[9] = crcData >> 8 ; //高位在后
RS485_Send_Data(MODBUS_SEND_SBUF,10);
for(i=0;i<10;i++)
{
printf("\n%2X\n\r",MODBUS_SEND_SBUF);
}
break;
case 2:
MODBUS_SEND_SBUF[7]=ExportData&0XFF;
MODBUS_SEND_SBUF[8]=(ExportData&0XFFFF)>>8;
crcData = crc16(MODBUS_SEND_SBUF,9);
MODBUS_SEND_SBUF[9] = crcData & 0xff; // CRC代码低位在前
MODBUS_SEND_SBUF[10] = crcData >> 8 ; //高位在后
RS485_Send_Data(MODBUS_SEND_SBUF,11);
for(i=0;i<11;i++)
{
printf("\n%2X\n\r",MODBUS_SEND_SBUF);
}
break;
case 3:
MODBUS_SEND_SBUF[7]=ExportData&0XFF;
MODBUS_SEND_SBUF[8]=(ExportData&0XFFFF)>>8;
MODBUS_SEND_SBUF[9]=(ExportData&0XFFFFFF)>>16;
crcData = crc16(MODBUS_SEND_SBUF,10);
MODBUS_SEND_SBUF[10] = crcData & 0xff; // CRC代码低位在前
MODBUS_SEND_SBUF[11] = crcData >> 8 ; //高位在后
RS485_Send_Data(MODBUS_SEND_SBUF,12);
for(i=0;i<12;i++)
{
printf("\n%2X\n\r",MODBUS_SEND_SBUF);
}
break;
case 4:
MODBUS_SEND_SBUF[7]=ExportData&0XFF;
MODBUS_SEND_SBUF[8]=(ExportData&0XFFFF)>>8;
MODBUS_SEND_SBUF[9]=(ExportData&0XFFFFFF)>>16;
MODBUS_SEND_SBUF[10]=ExportData>>24;
crcData = crc16(MODBUS_SEND_SBUF,11);
MODBUS_SEND_SBUF[11] = crcData & 0xff; // CRC代码低位在前
MODBUS_SEND_SBUF[12] = crcData >> 8 ; //高位在后
RS485_Send_Data(MODBUS_SEND_SBUF,13);
for(i=0;i<13;i++)
{
printf("\n%2X\n\r",MODBUS_SEND_SBUF);
}
break;
case 5:
/***********************/
break;
}
// MODBUS_SEND_SBUF[]=0; 需不需要清零
}
//返回数据
u8 ReceiveWriteMulCoilData(void)
{
u8 result=0;
u16 crcData;
RS485_Receive_Data(MODBUS_RECEIVE_SBUF,&RS485_RX_CNT);
crcData = crc16(MODBUS_RECEIVE_SBUF,RS485_RX_CNT-1);
if(MODBUS_RECEIVE_SBUF[RS485_RX_CNT-1] == ( crcData&0xff ) || MODBUS_RECEIVE_SBUF[RS485_RX_CNT] == ( crcData >> 8 ))
{
RS485_RX_CNT=0; //长度清零
return result=0;
}
else
{
RS485_RX_CNT=0; //长度清零
printf("写多线圈出错!");
result=0X0F;
return result;
}
}
/*****************************读写保持寄存器*********************************/
/*********主要功能码:读保持寄存器,写单个寄存器,写多个寄存器***************/
/****************************************************************************
*256套规范,每套规范预留256个寄存器,现有128个参数,每个参数2个字节。
*保持寄存器偏移量=规范号*256+参数号。
*规范号:0~255 参数号:0~127
*0<=保持寄存器偏移量<=255*256+127=65407=0XFF7F
*0XFF7F=65535
*get_MN(),动态定义规范
*****************************************************************************/
//取值范围是0~255
u8 get_MN()
{
/***********测试使用************/
// printf("\n%2X\n\r",Defaultspec);
return Defaultspec;
}
/*****************************************************************************
*读保持寄存器(0X03): 每次最多读取125个
*SD:地址(1)+功能码(1)+起始地址(2)+寄存器数量(2)+CRC
*RD:地址(1)+功能码(1)+字节数N(1)+寄存器值(N*2)+CRC
*输入:ParameterNum 参数号 RegNumber 寄存器数量
*****************************************************************************/
void SendReadRegCommand(u8 ParameterNum,u8 RegNumber)
{
u16 crcData,StartAddress;
u8 i;
StartAddress=256*mn+ParameterNum;
MODBUS_SEND_SBUF[0] = baseAddress;//地址
MODBUS_SEND_SBUF[1] = 0X03;//功能码
MODBUS_SEND_SBUF[2] = StartAddress>>8;
MODBUS_SEND_SBUF[3] = StartAddress&0XFF;
MODBUS_SEND_SBUF[4] = 0X00;
MODBUS_SEND_SBUF[5] = RegNumber;//不超过 7D(125)
crcData = crc16(MODBUS_SEND_SBUF,6);
MODBUS_SEND_SBUF[6] = crcData & 0xff; // CRC代码低位在前
MODBUS_SEND_SBUF[7] = crcData >> 8; //高位在后
RS485_Send_Data(MODBUS_SEND_SBUF,8);
// MODBUS_SEND_SBUF[]=0; // 需不需要清零
// for(i=0;i<8;i++)
// {
// printf("\n%2X\n\r",MODBUS_SEND_SBUF);
// }
}
u8 ReceiveReadRegData(void)
{
u8 result=0;
u16 crcData;
RS485_Receive_Data(MODBUS_RECEIVE_SBUF,&RS485_RX_CNT);
crcData = crc16(MODBUS_RECEIVE_SBUF,RS485_RX_CNT-1);
if(MODBUS_RECEIVE_SBUF[RS485_RX_CNT-1] == ( crcData&0xff ) || MODBUS_RECEIVE_SBUF[RS485_RX_CNT] == ( crcData >> 8 ))
{
RS485_RX_CNT=0; //长度清零
return result=0;
}
else
{
RS485_RX_CNT=0; //长度清零
printf("读寄存器出错!");
result=0X03;
return result;
}
}
/*************************************************************************
*SendWriteRegisterCommand();写单个寄存器(0X06):
*SD:地址(1)+功能码(1)+寄存器地址(2)+寄存器值(2)+CRC
*RD:地址(1)+功能码(1)+寄存器地址(2)+寄存器值(2)+CRC
*输入:ParameterNum:寄存器地址;SinRegswitch:ON或OFF
*输出:无
**************************************************************************/
//发送命令
void SendWriteSinRegCommand(u8 ParameterNum,u8 SinRegswitch)
{
u16 crcData,RegisterAddress;
u8 i;
RegisterAddress=256*mn+ParameterNum;
MODBUS_SEND_SBUF[0] = baseAddress;//地址
MODBUS_SEND_SBUF[1] = 0X06;//功能码
MODBUS_SEND_SBUF[2] = RegisterAddress>>8;
MODBUS_SEND_SBUF[3] = RegisterAddress&0XFF;
MODBUS_SEND_SBUF[4] = 0X00;
MODBUS_SEND_SBUF[5] = SinRegswitch;
crcData = crc16(MODBUS_SEND_SBUF,6);
MODBUS_SEND_SBUF[6] = crcData & 0xff; // CRC代码低位在前
MODBUS_SEND_SBUF[7] = crcData >> 8; //高位在后
RS485_Send_Data(MODBUS_SEND_SBUF,8);
// MODBUS_SEND_SBUF[]=0; 需不需要清零
for(i=0;i<8;i++)
{
printf("\n%2X\n\r",MODBUS_SEND_SBUF);
}
}
u8 ReceiveWriteSinRegData(void)
{
u8 result=0;
u16 crcData;
RS485_Receive_Data(MODBUS_RECEIVE_SBUF,&RS485_RX_CNT);
crcData = crc16(MODBUS_RECEIVE_SBUF,RS485_RX_CNT-1);
if(MODBUS_RECEIVE_SBUF[RS485_RX_CNT-1] == ( crcData&0xff ) || MODBUS_RECEIVE_SBUF[RS485_RX_CNT] == ( crcData >> 8 ))
{
RS485_RX_CNT=0; //长度清零
return result=0;
}
else
{
RS485_RX_CNT=0; //长度清零
printf("写单个寄存器出错!");
result=0X06;
return result;
}
}
/*************************************************************************
*写多个寄存器(0X10): 每次最多写123个
*SD:地址(1)+功能码(1)+起始地址(2)+寄存器数量(2)+字节数N(1)+寄存器值(2*N)+CRC
*RD:地址(1)+功能码(1)+起始地址(2)+寄存器数量(2)+CRC
*输入:StartAddress:寄存器地址;SinRegswitch:ON或OFF
*输出:无
**************************************************************************/
// void SendWriteMulRegCommand( u8 ParameterNum, )
// {
// u16 crcData,StartAddress;
// u8 i;
// StartAddress=256*mn+ParameterNum;
// MODBUS_SEND_SBUF[0] = baseAddress;//地址
// MODBUS_SEND_SBUF[1] = 0X10;//功能码
// MODBUS_SEND_SBUF[2] = StartAddress>>8;
// MODBUS_SEND_SBUF[3] = StartAddress&0XFF;
// }
/*************************************************************************
*读输入寄存器(0X04):读内存
*SD:地址(2)+命令(2)+起始地(2)址+寄存器数量(2)
*RD:地址(2)+命令(2)+字节数N(2)+数据内容(N*2)
*输入:StartAddress:起始地址0X0000~0XFFFF
*输入:RegisterNum:0X0001~0X007D(125)
*输出:无
**************************************************************************/
void SendReadEnterRegCommand( u16 StartAddress,u8 RegisterNum )
{
u16 crcData;
MODBUS_SEND_SBUF[0] = baseAddress;//地址
MODBUS_SEND_SBUF[1] = 0X04;//功能码
MODBUS_SEND_SBUF[2] = StartAddress>>8;
MODBUS_SEND_SBUF[3] = StartAddress&0XFF;
MODBUS_SEND_SBUF[4] = 0X00;
MODBUS_SEND_SBUF[5] = RegisterNum;
crcData = crc16(MODBUS_SEND_SBUF,6);
MODBUS_SEND_SBUF[6] = crcData & 0xff; // CRC代码低位在前
MODBUS_SEND_SBUF[7] = crcData >> 8; //高位在后
RS485_Send_Data(MODBUS_SEND_SBUF,8);
}
u8 ReceiveReadEnterRegData(void)
{
u8 result=0;
u16 crcData;
RS485_Receive_Data(MODBUS_RECEIVE_SBUF,&RS485_RX_CNT);
crcData = crc16(MODBUS_RECEIVE_SBUF,RS485_RX_CNT-1);
if(MODBUS_RECEIVE_SBUF[RS485_RX_CNT-1] == ( crcData&0xff ) || MODBUS_RECEIVE_SBUF[RS485_RX_CNT] == ( crcData >> 8 ))
{
RS485_RX_CNT=0; //长度清零
return result=0;
}
else
{
RS485_RX_CNT=0; //长度清零
printf("读输入寄存器出错!");
result=0X04;
return result;
}
}
/*************************************************************************
*读焊接历史记录(功能码:0X41)
*焊接记录,最多960条记录;记录读取方式只有0和1,1代表读记录( 从最早的记录
*开始读),0代表重读记录。每次最多读取3条记录。每个记录体包含64个字节(低
*字节在先)。
*SD: 地址(1)+功能码(1)+读取方式(1)+CRC
*RD: 地址(1)+功能码(1)+读取方式(1)+回复的记录个数(1)+记录体(64)+CRC
**************************************************************************/
void SendReadWeldingHistoryCommand( u8 ReadMode )
{
u16 crcData;
// u8 i;
MODBUS_SEND_SBUF[0] = baseAddress;//地址
MODBUS_SEND_SBUF[1] = 0X41;//功能码
MODBUS_SEND_SBUF[2] = ReadMode;
crcData = crc16(MODBUS_SEND_SBUF,3);
MODBUS_SEND_SBUF[3] = crcData & 0xff; // CRC代码低位在前
MODBUS_SEND_SBUF[4] = crcData >> 8; //高位在后
RS485_Send_Data(MODBUS_SEND_SBUF,5);
// for(i=0;i<5;i++)
// {
// printf("\n%d\r",MODBUS_SEND_SBUF);
// }
}
u8 ReceiveWeldingHistoryData(void)
{
u8 result=0;
u16 crcData;
RS485_Receive_Data(MODBUS_RECEIVE_SBUF,&RS485_RX_CNT);
crcData = crc16(MODBUS_RECEIVE_SBUF,RS485_RX_CNT-1);
if(MODBUS_RECEIVE_SBUF[RS485_RX_CNT-1] == ( crcData&0xff ) || MODBUS_RECEIVE_SBUF[RS485_RX_CNT] == ( crcData >> 8 ))
{
RS485_RX_CNT=0; //长度清零
return result=0;
}
else
{
RS485_RX_CNT=0; //长度清零
printf("读焊接历史记录错误!");
result=0X41;
return result;
}
}
void SendReadWriteTimeCommand(u8 TimeCommand , u8 TimeNumber[7] )
{
u16 crcData;
u8 i;
MODBUS_SEND_SBUF[0] = baseAddress;//地址
MODBUS_SEND_SBUF[1] = 0X44;//功能码
MODBUS_SEND_SBUF[2] = TimeCommand;
if(TimeCommand==0)
{
crcData = crc16(MODBUS_SEND_SBUF,3);
MODBUS_SEND_SBUF[3] = crcData & 0xff; // CRC代码低位在前
MODBUS_SEND_SBUF[4] = crcData >> 8; //高位在后
RS485_Send_Data(MODBUS_SEND_SBUF,5);
}
else
{
delay_ms(5);
for(i=0;i<7;i++)
{
MODBUS_SEND_SBUF[3+i] = TimeNumber;
// printf("\n\r%d\n\r",TimeNumber);
}
crcData = crc16(MODBUS_SEND_SBUF,10);
MODBUS_SEND_SBUF[10] = crcData & 0xff; // CRC代码低位在前
MODBUS_SEND_SBUF[11] = crcData >> 8; //高位在后
RS485_Send_Data(MODBUS_SEND_SBUF,12);
}
// MODBUS_SEND_SBUF[3] = RecordPointer&0XFF;
// MODBUS_SEND_SBUF[4] = RecordNumber;
//
}
u8 ReceiveReadWriteTimeData( void )
{
u8 result=0;
u16 crcData;
RS485_Receive_Data(MODBUS_RECEIVE_SBUF,&RS485_RX_CNT);
crcData = crc16(MODBUS_RECEIVE_SBUF,RS485_RX_CNT-1);
if(MODBUS_RECEIVE_SBUF[RS485_RX_CNT-1] == ( crcData&0xff ) || MODBUS_RECEIVE_SBUF[RS485_RX_CNT] == ( crcData >> 8 ))
{
RS485_RX_CNT=0; //长度清零
return result=0;
}
else
{
RS485_RX_CNT=0; //长度清零
printf("读写时钟错误!");
result=0X44;
return result;
}
}
/*************************************************************************
* 读设备标识(0X2B)
* 表:10:设备标识
* 对象ID 对象名称 类型
* 0X00 厂商名称 ASCII字符串 只读
* 0X01 产品代码 ASCII字符串 只读
* 0X02 主次版本号 ASCII字符串 只读
* SD:地址(1)+功能码(1)+MEI类型(1)+读设备ID码(1)+对象ID(1)
SD: **+2B+0E+01+00+CRC
* RD:地址(1)+功能码(1)+MEI类型(1)+设备ID码(1)+一致性等级[conformity
* level](1)+00(1)+下一个设备ID码(1)+对象数量n (1)+对象1 ID(1)+对象1
* 长度N(1)+对象内容(N)+… … +对象n ID(1)+对象n长度M(1)+对象内容(M)+CRC
*************************************************************************/
void SendReadDeviceIdentifineCommand( void )
{
u16 crcData;
MODBUS_SEND_SBUF[0] = baseAddress;//地址
MODBUS_SEND_SBUF[1] = 0X2B;//功能码
MODBUS_SEND_SBUF[2] = 0X0E;
MODBUS_SEND_SBUF[3] = 0X01;
MODBUS_SEND_SBUF[4] = 0X00;
crcData = crc16(MODBUS_SEND_SBUF,5);
MODBUS_SEND_SBUF[5] = crcData & 0xff; // CRC代码低位在前
MODBUS_SEND_SBUF[6] = crcData >> 8; //高位在后
RS485_Send_Data(MODBUS_SEND_SBUF,7);
}
u8 ReceiveDeviceIdentifineData( void )
{
u8 result=0;
u16 crcData;
RS485_Receive_Data(MODBUS_RECEIVE_SBUF,&RS485_RX_CNT);
crcData = crc16(MODBUS_RECEIVE_SBUF,RS485_RX_CNT-1);
if(MODBUS_RECEIVE_SBUF[RS485_RX_CNT-1] == ( crcData&0xff ) || MODBUS_RECEIVE_SBUF[RS485_RX_CNT] == ( crcData >> 8 ))
{
RS485_RX_CNT=0; //长度清零
return result=0;
}
else
{
RS485_RX_CNT=0; //长度清零
printf("读设备标识错误!");
result=0X2B;
return result;
}
}
|