学习工控值得收藏 51单片机与MCGS的modbusRTU通信
目前,工业组态软件已经广泛应用于工业控制领域。用户无需了解复杂的编程知识,就可以用工业组态软件在短时间内完成一个具备专业水准的计算机监控系统的开发工作。MCGS是一套基于Windows平台的、为工业过程控制和实时监测服务的32位全中文界面组态软件系统。为了与现场设备进行交互,它提供了国内外各种常用的工控设备的工控设备的驱动程序;但现场很多设备是厂家根据本厂要求定制的设备,这就需要设计定制新设备与MCGS的数据通讯程序。本文以STC89C52单片机为例,编写单片机与MSGS的modbus rtu通讯协议,本文只列出了MODBUS RTU协议的子程序。
sbit LED0 = P1^0; // 对应线圈0
sbit LED1 = P1^1; // 对应线圈1
sbit LED2 = P1^2; // 对应线圈2
sbit LED3 = P1^3; // 对应线圈3
sbit LED4 = P1^4; // 对应线圈4
sbit LED5 = P1^5; // 对应线圈5
sbit LED6 = P1^6; // 对应线圈6
sbit LED7 = P1^7; // 对应线圈7
sbit KEY0 = P3^2;
sbit KEY1 = P3^3;
sbit KEY2 = P3^4;
sbit KEY3 = P3^5;
//字地址 0 - 255 (只取低8位)
//位地址 0 - 255 (只取低8位)
/* CRC 高位字节值表 */
const UINT8 code 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 UINT8 code 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
} ;
UINT8 testCoil; //用于测试 位地址1
UINT8 localAddr = 1; //单片机控制板的地址
UINT16 testRegister0,// 测试寄存器
testRegister1,
testRegister2,
testRegister3,//存放当前温度
testRegister4,//读写寄存器 控制P1口输出
testRegister5,
testRegister6,
testRegister7,
testRegister8,
testRegister9;
UINT16 crc16(const UINT8 *puchMsg, UINT16 usDataLen)
{
UINT8 uchCRCHi = 0xFF ; /* 高CRC字节初始化 */
UINT8 uchCRCLo = 0xFF ; /* 低CRC 字节初始化 */
UINT32 uIndex ; /* CRC循环中的索引 */
while (usDataLen--) { /* 传输消息缓冲区 */
uIndex = uchCRCHi ^ *puchMsg++ ; /* 计算CRC */
uchCRCHi = uchCRCLo ^ auchCRCHi[uIndex] ;
uchCRCLo = auchCRCLo[uIndex] ;
}
return (uchCRCHi << 8 | uchCRCLo) ;
}
//开始发送
void beginSend(void)
{
UartSendBytes (sendBuf, sendCount);
}
//读线圈状态
void readCoil(void)
{
UINT8 addr;
UINT8 tempAddr;
UINT8 byteCount;
UINT8 bitCount;
UINT16 crcData;
UINT8 position;
UINT8 i, k;
UINT16 tempData;
UINT8 exit = 0;
//addr = (receBuf[2]<<8) + receBuf[3];
//tempAddr = addr & 0xfff;
addr = receBuf[3];
tempAddr = addr;
//bitCount = (receBuf[4]<<8) + receBuf[5]; //读取的位个数
bitCount = receBuf[5];
byteCount = bitCount / 8; //字节个数
if (bitCount % 8 != 0)
byteCount++;
for (k = 0; k < byteCount; k++) {
//字节位置
position = k + 3;
sendBuf[position] = 0;
for (i = 0; i < 8; i++) {
getCoilVal(tempAddr, &tempData);
sendBuf[position] |= (tempData << i);
tempAddr++;
if (tempAddr >= addr + bitCount) {
//读完
exit = 1;
break;
}
}
if (exit == 1)
break;
}
sendBuf[0] = localAddr;
sendBuf[1] = 0x01;
sendBuf[2] = byteCount;
byteCount += 3;
crcData = crc16(sendBuf, byteCount);
sendBuf[byteCount] = crcData >> 8;
byteCount++;
sendBuf[byteCount] = crcData & 0xff;
sendCount = byteCount + 1;
beginSend();
}
//读寄存器
void readRegisters(void)
{
UINT8 addr;
UINT8 tempAddr;
UINT16 crcData;
UINT8 readCount;
UINT8 byteCount;
UINT16 i;
UINT16 tempData = 0;
//addr = (receBuf[2]<<8) + receBuf[3];
//tempAddr = addr & 0xfff;
addr = receBuf[3];
tempAddr = addr;
//readCount = (receBuf[4]<<8) + receBuf[5]; //要读的个数
readCount = receBuf[5];
byteCount = readCount * 2;
for (i = 0; i < byteCount; i += 2, tempAddr++) {
getRegisterVal(tempAddr, &tempData);
sendBuf[i+3] = tempData >> 8;
sendBuf[i+4] = tempData & 0xff;
}
sendBuf[0] = localAddr;
sendBuf[1] = 3;
sendBuf[2] = byteCount;
byteCount += 3;
crcData = crc16(sendBuf, byteCount);
sendBuf[byteCount] = crcData >> 8;
byteCount++;
sendBuf[byteCount] = crcData & 0xff;
sendCount = byteCount + 1;
beginSend();
}//void readRegisters(void)
//强制单个线圈
void forceSingleCoil(void)
{
UINT8 addr;
UINT8 tempAddr;
UINT16 tempData;
UINT8 onOff;
UINT8 i;
//addr = (receBuf[2]<<8) + receBuf[3];
//tempAddr = addr & 0xfff;
addr = receBuf[3];
tempAddr = addr;
//onOff = (receBuf[4]<<8) + receBuf[5];
onOff = receBuf[4];
//if(onOff == 0xff00)
if (onOff == 0xff) {
//设为ON
tempData = 1;
}
//else if(onOff == 0x0000)
else if (onOff == 0x00) {
//设为OFF
tempData = 0;
}
setCoilVal(tempAddr, tempData);
for (i = 0; i < receCount; i++) {
sendBuf[i] = receBuf[i];
}
sendCount = receCount;
beginSend();
}
void presetSingleRegister(void) //设置单个寄存器
{
U8 addr;
U8 tempAddr;
U8 setCount;
U16 crcData;
U16 tempData;
//addr = (receBuf[2]<<8) + receBuf[3];
//tempAddr = addr & 0xfff;
addr = receBuf[3];
tempAddr = addr; //& 0xff
tempData = ( receBuf[4]<<8 ) + receBuf[5];
setRegisterVal(tempAddr,tempData);
sendBuf[0] = localAddr;
sendBuf[1] = 6;
sendBuf[2] = addr >> 8;
sendBuf[3] = addr & 0xff;
sendBuf[4] = receBuf[4];
sendBuf[5] = receBuf[5] ;
setCount = 6; //共6个字节
crcData = crc16(sendBuf,6);
sendBuf[6] = crcData >> 8;
sendBuf[7] = crcData & 0xff;
sendCount = 8;
beginSend();
}
//设置多个寄存器
void presetMultipleRegisters(void)
{
UINT8 addr;
UINT8 tempAddr;
UINT8 byteCount;
UINT8 setCount;
UINT16 crcData;
UINT16 tempData;
UINT8 i;
//addr = (receBuf[2]<<8) + receBuf[3];
//tempAddr = addr & 0xfff;
addr = receBuf[3];
tempAddr = addr & 0xff;
//setCount = (receBuf[4]<<8) + receBuf[5];
setCount = receBuf[5];
byteCount = receBuf[6];
for (i = 0; i < setCount; i++, tempAddr++) {
//SBUF = receBuf[i*2+7];
//SBUF = receBuf[i*2+8];
tempData = (receBuf[i*2+7] << 8) + receBuf[i*2+8];
setRegisterVal(tempAddr, tempData);
}
sendBuf[0] = localAddr;
sendBuf[1] = 16;
sendBuf[2] = addr >> 8;
sendBuf[3] = addr & 0xff;
sendBuf[4] = setCount >> 8;
sendBuf[5] = setCount & 0xff;
crcData = crc16(sendBuf, 6);
sendBuf[6] = crcData >> 8;
sendBuf[7] = crcData & 0xff;
sendCount = 8;
beginSend();
}
//取线圈状态 返回0表示成功
UINT16 getCoilVal(UINT16 addr, UINT16 *tempData)
{
UINT16 result = 0;
UINT16 tempAddr;
tempAddr = addr & 0xfff;
//只取低8位地址
switch( tempAddr ) //只取低8位地址 & 0xff
{
case 0:
if (KEY0)
*tempData = 1;
else
*tempData = 0;
break;
case 1:
if (KEY1)
*tempData = 1;
else
*tempData = 0;
break;
case 2:
if (KEY2)
*tempData = 1;
else
*tempData = 0;
break;
case 3:
if (KEY3)
*tempData = 1;
else
*tempData = 0;
break;
case 4:
break;
case 5:
break;
case 6:
break;
case 7:
break;
case 8:
break;
case 9:
break;
case 10:
break;
case 11:
break;
case 12:
break;
case 13:
break;
case 14:
break;
case 15:
break;
default:
break;
}
return result;
}
//设定线圈状态 返回0表示成功
UINT16 setCoilVal(UINT16 addr, UINT16 tempData)
{
UINT16 result = 0;
UINT16 tempAddr;
tempAddr = addr & 0xfff;
switch(tempAddr) // & 0xff
{
case 0:
if (tempData)
LED0 = 1;
else
LED0 = 0;
break;
case 1:
if (tempData)
LED1 = 1;
else
LED1 = 0;
break;
case 2:
if (tempData)
LED2 = 1;
else
LED2 = 0;
break;
case 3:
if (tempData)
LED3 = 1;
else
LED3 = 0;
break;
case 4:
if (tempData)
LED4 = 1;
else
LED4 = 0;
break;
case 5:
if (tempData)
LED5 = 1;
else
LED5 = 0;
break;
case 6:
if (tempData)
LED6 = 1;
else
LED6 = 0;
break;
case 7:
if (tempData)
LED7 = 1;
else
LED7 = 0;
break;
case 10:
break;
case 11:
break;
case 12:
break;
case 13:
break;
case 14:
break;
case 15:
break;
case 16:
break;
case 17:
break;
default:
break;
}
return result;
}
//取寄存器值 返回0表示成功
UINT16 getRegisterVal(UINT16 addr, UINT16 *tempData)
{
UINT16 result = 0;
UINT16 tempAddr;
tempAddr = addr & 0xfff;
switch(tempAddr) //& 0xff
{
case 0:
*tempData = testRegister0;
break;
case 1:
*tempData = testRegister1;
break;
case 2:
*tempData = testRegister2;
break;
case 3:
//testRegister3= sdate;
*tempData = testRegister3;
break;
case 4:
*tempData = testRegister4;
break;
case 5:
*tempData = testRegister5;
break;
case 6:
*tempData = testRegister6;
break;
case 7:
*tempData = testRegister7;
break;
case 8:
*tempData = testRegister8;
break;
case 9:
*tempData = testRegister9;
break;
case 10:
break;
case 11:
break;
case 12:
break;
case 13:
break;
case 14:
break;
case 15:
break;
case 16:
break;
default:
break;
}
return result;
}
//设置寄存器值 返回0表示成功
UINT16 setRegisterVal(UINT16 addr, UINT16 tempData)
{
UINT16 result = 0;
UINT16 tempAddr;
tempAddr = addr & 0xfff;
switch(tempAddr) //& 0xff
{
case 0:
testRegister0 = tempData;
break;
case 1:
testRegister1 = tempData;
break;
case 2:
testRegister2 = tempData;
break;
case 3:
testRegister3 = tempData;
break;
case 4:
testRegister4 = tempData;
P1=tempData ; //&& 0X00FF
break;
case 5:
testRegister5 = tempData;
break;
case 6:
testRegister6 = tempData;
break;
case 7:
testRegister7 = tempData;
break;
case 8:
testRegister8 = tempData;
break;
case 9:
testRegister9 = tempData;
break;
case 10:
break;
case 11:
break;
case 12:
break;
case 13:
break;
case 14:
break;
case 15:
break;
case 16:
break;
default:
break;
}
return result;
}
void ModbusDelay (unsigned int nDelay)
{
volatile unsigned int i, j;
for (i = 0; i < nDelay; i ++)
for (j = 0; j < 10; j ++);
}
//检查uart0数据
void checkComm0Modbus(void)
{
UINT16 crcData;
UINT16 tempData;
if (receCount > 4) {
switch (receBuf[1]) {
case 1://读取线圈状态(读取点 16位以内)
case 3://读取保持寄存器(一个或多个)
case 5://强制单个线圈
case 6://设置单个寄存器
if (receCount >= 8) {
//接收完成一组数据
//应该关闭接收中断
UART_DISABLE_INTERRUPT();
if (receBuf[0] == localAddr) {
ModbusDelay (10);
crcData = crc16(receBuf, 6);
if (crcData == receBuf[7] + (receBuf[6] << 8)) {
//校验正确
if (receBuf[1] == 1) {
//读取线圈状态(读取点 16位以内)
readCoil();
}
else if (receBuf[1] == 3) {
//读取保持寄存器(一个或多个)
readRegisters();
}
else if (receBuf[1] == 5) {
//强制单个线圈
forceSingleCoil();
}
else if (receBuf[1] == 6) {
//写单个寄存器
presetSingleRegister();
}
}
}
RI = 0;
receCount = 0;
UART_ENABLE_INTERRUPT();
}
break;
case 15://设置多个线圈
tempData = receBuf[6];
tempData += 9; //数据个数
if (receCount >= tempData) {
if (receBuf[0] == localAddr) {
crcData = crc16(receBuf, tempData - 2);
if (crcData == (receBuf[tempData-2] << 8) + receBuf[tempData-1]) {
//forceMultipleCoils();
}
}
receCount = 0;
}
break;
case 16://设置多个寄存器
tempData = (receBuf[4] << 8) + receBuf[5];
tempData = tempData * 2; //数据个数
tempData += 9;
if (receCount >= tempData) {
UART_DISABLE_INTERRUPT();
if (receBuf[0] == localAddr) {
crcData = crc16(receBuf, tempData - 2);
if (crcData == (receBuf[tempData-2] << 8) + receBuf[tempData-1]) {
presetMultipleRegisters();
}
}
RI = 0;
receCount = 0;
UART_ENABLE_INTERRUPT();
}
break;
default:
break;
}
}
}
PLC作主站,51单片机作从站,用ModBus协议进行通讯
本节主要完成PLC作主站,51单片机作从站,用ModBus协议进行通讯。PLC读取单片机保持寄存器区的数据。S7-200PLC程序主要通过调用Modubs RTU 主站指令库完成。
一、调用 Modbus RTU 主站初始化和控制子程序
使用 SM0.0 调用 MBUS_CTRL 完成主站的初始化,并启动其功能控制:
各参数意义如下:
a.
EN
使能:
必须保证每一扫描周期都被使能(使用 SM0.0)
b.
Mode
模式:
为 1 时,使能 Modbus 协议功能;为 0 时恢复为系统 PPI 协议
c.
Baud
波特率:
支持的通讯波特率为1200,2400,4800,9600,19200,38400,57600,115200。
d.
Parity
校验:
校验方式选择
0=无校验
1=奇较验
2=偶较验
e.
Timeout
超时:
主站等待从站响应的时间,以毫秒为单位,典型的设置值为 1000 毫秒(1 秒),允许设置的范围为 1 - 32767。
注意: 这个值必须设置足够大以保证从站有时间响应。
f.
Done
完成位:
初始化完成,此位会自动置1。可以用该位启动 MBUS_MSG 读写操作(见例程)
g.
Error
初始化错误代码(只有在 Done 位为1时有效):
0= 无错误
1= 校验选择非法
2= 波特率选择非法
3= 模式选择非法
二、调用 Modbus RTU 主站读写子程序MBUS_MSG,发送一个Modbus 请求;
各参数意义如下:
a.
EN
使能:
同一时刻只能有一个读写功能(即 MBUS_MSG)使能
注意:建议每一个读写功能(即 MBUS_MSG)都用上一个 MBUS_MSG 指令的 Done 完成位来激活,以保证所有读写指令循环进行(见例程)。
b.
First
读写请求位:
每一个新的读写请求必须使用脉冲触发
c.
Slave
从站地址:
可选择的范围 1 - 247
d.
RW
从站地址:
0 = 读, 1 = 写
注意:
1. 开关量输出和保持寄存器支持读和写功能
2. 开关量输入和模拟量输入只支持读功能
e.
Addr
读写从站的数据地址:
选择读写的数据类型
00001 至 0xxxx - 开关量输出
10001 至 1xxxx - 开关量输入
30001 至 3xxxx - 模拟量输入
40001 至 4xxxx - 保持寄存器
f.
Count
数据个数
通讯的数据个数(位或字的个数)
注意: Modbus主站可读/写的最大数据量为120个字(是指每一个 MBUS_MSG 指令)
g.
DataPtr
数据指针:
1. 如果是读指令,读回的数据放到这个数据区中
2. 如果是写指令,要写出的数据放到这个数据区中
h.
Done
完成位
读写功能完成位
i.
Error
错误代码:
只有在 Done 位为1时,错误代码才有效
0 = 无错误
1 = 响应校验错误
2 = 未用
3 = 接收超时(从站无响应)
4 = 请求参数错误(slave address, Modbus address, count, RW)
5 = Modbus/自由口未使能
6 = Modbus正在忙于其它请求
7 = 响应错误(响应不是请求的操作)
8 = 响应CRC校验和错误
-
101 = 从站不支持请求的功能
102 = 从站不支持数据地址
103 = 从站不支持此种数据类型
104 = 从站设备故障
105 = 从站接受了信息,但是响应被延迟
106 = 从站忙,拒绝了该信息
107 = 从站拒绝了信息
108 = 从站存储器奇偶错误
常见的错误:
如果多个 MBUS_MSG 指令同时使能会造成 6 号错误库存储区被程序其它地方复用,有时也会造成6 号错误从站 delay 参数设的时间过长会造成主站 3 号错误从站掉电或不运行,网络故障都会造成主站 3 号错误。
三、需要从站支持的功能及Modbus 保持寄存器地址映射
为了支持上述 Modbus 地址的读写,Modbus Master 协议库需要从站支持下列功能:
需要从站支持的功能
Modbus 地址
读/写
Modbus 从站须支持的功能
00001 - 09999
数字量输出
读
功能 1
写
功能 5:写单输出点
功能 15:写多输出点
10001 - 19999
数字量输入
读
功能 2
写
-
30001 - 39999
输入寄存器
读
功能 4
写
-
40001 - 49999
保持寄存器
读
功能 3
写
功能 6:写单寄存器单元
功能 16:写多寄存器单元
Modbus 保持寄存器地址映射举例:
四、S7-200PLC程序
五、单片机程序;STC11F04E单片机,9600波特率
START: MOV TMOD,#21H ;定时器1是8位再装入,定时器0为16位定时器
MOV TH1,#0FDH;预置初值(按照波特率9600BPS预置初值)
MOV TL1,#0FDH; 0FDH=9600=11.0592
MOV TH0, #0DCH;88H ;8800=12t,7000=stc1t
MOV TL0, #00H
ORL IE, #92H ;EA=1,ES=1;ET0=1
SETB PS ;串口中断优先
SETB TR1 ;启动定时器1
MOV 98H,#50H ;scon
MOV P1M0,#01000000b ; P1M0=0 P1M1=0双向口 P1M0=1 P1M1=0输入口 P1M0=0 P1M1=1推挽输出20ma
MOV P1M1,#10000000b
MOV WDT_CONTR ,#27H 看门狗设置使能
QL0: MOV A,#00H
MOV R0,#10H
MOV R2,#9BH ;10-ABH清零
CLEAR: MOV @R0,A
INC R0
DJNZ R2,CLEAR
CLR FLAG
CLR FLAG_0
SETB TR0 ;启动定时器0
;ANL AUX,#07FH ;p3.0p3.1当串口
ORL AUX,#80H ;p1.7,p1.6当串口
CLR P3.7 ;485芯片接收使能
WA1: ;MOV WDT_CONTR ,#37H;喂狗; SETB CW
JNB FLAG_0,WA1 ;FLAG_0=1表示已经接收到上位机数据
CLR TR0
MOV A,2CH ;检查设备地址是01h码,设本机地址码是1
MOV R2,A
XRL A,#01H
JNZ QL0
ACALL FSZJ ;FH: DB 01H,03H,16,00H,01H,02H,03H,04H,05H,06H,07H,08H,09H,10H,11H,12H,0DH,0EH,0FH,10H,11H,12H,13H,14H,15H,16H,17H,18H,19H,1AH,1BH,1CH,1DH,1EH,1FH;18
ACALL DELAY
CALL FZJ
AJMP QL0
FZJ: MOV R0,#2cH ;向主机发送数据子程序
FZJ0: MOV R2,#10H
FZJ1: CLR EA
ANL AUX,#07FH ;p3.0p3.1当串口
FZL1: MOV A,@R0
MOV SBUF,A
JNB TI,$
CLR TI
INC R0
DJNZ R2,FZL1
SETB EA
RET
FSZJ: MOV DPTR,#FH
MOV R2,#19;
ORL AUX,#80H
SETB P3.7 ;发送数据
MOV R0,#40H
FSZJA: MOV A,#0H
MOVC A,@A+DPTR
MOV @R0,A
INC R0
INC DPTR
DJNZ R2,FSZJA
MOV R0,#40H
MOV CRCCD,#19
LCALL CRC1
MOV R2,#21
MOV R0,#40H
FSZJ2: MOV A,@R0
MOV SBUF,A
JNB TI,$
CLR TI
INC R0
DJNZ R2,FSZJ2
SETB EA
RET
FH:DB 01H,03H,16,00H,01H,02H,03H,04H,05H,06H,07H,08H,09H,10H,11H,12H,0DH,0EH,0FH,10H,11H,12H,13H,14H,15H,16H,17H,18H,19H,1AH,1BH,1CH,1DH,1EH,1FH;18
用串口助手检测到的数据如下图。
相关问答
51单片机 怎么与触摸屏连接?实现51单机片与触摸屏连接有四个步骤:1触摸屏与单片机的硬件联接2建立触摸屏与单片机的内部存储器地址对应关系3触摸屏组态软件编辑4MODBUSRTU(远程...
51单片机 用什么波特率?51单片机串口通讯波特率可以通过相应定时器寄存器设置成多个值,典型的波特率有2400、4800、9600、19200、38400和115200,其中最常用的是9600和15200。具体使...
51单片机 如何与触摸屏连接?1、实现触摸屏与单片机的通信,主要是解决通信协议的问题。2、使用开放的Modbus通信协议,以触摸屏作主站,单片机作从站。3、eView触摸屏本身支持Modbus通信协...
自己编制的 MODBUS单片机 程序,与 MODBUS OPCserver通讯有问题?modscan32测试软件可以测试MODBUSRTU、ASCII以及MODBUSTCP。但modbusopcserver是按照OPC规范的,而且它是opcserver,那么你的MODBUS单...
三菱PLC与 单片机 怎么实现485通讯?有2种方案:1、PLC做从机,单片机做主机2、单片机做从机,PLC做主机对上面两种,PLC需要加485拓展模块或拓展小板,同时在PLC内部需要通过写程序设置好D8140、...
新手用 51单片机 做什么好呢?我认为作为新手用51单片机最好用PCB万能板焊接一个单片机实验板,一来这样能够提高自己的动手能力;二来可以锻炼自己的编程调试能力。同时我们在焊接时还要设计...
买成品的 单片机 板子开发实现仪表规约协议转换?你说的这个功能叫协议转换,你可以做一个专用的协议转换接口电路板,将不同公司的仪表的串口数据协议转成ModbusTCP协议。普通单片机就可以,用不着这个高大上的...
用上位机控制 单片机 然后实现数据传输,学习哪种上位机好呢?用上位机控制单片机然后实现数据传输,学习哪种上位机好呢?既然是控制单片机,那么通讯协议是可以自己编写实现的,要实现数据传输甚至不需要自己编写上位机软...
RS-485总线接口电路硬件如何设计?RS-485如何进行网络配置?RS485接口在工控行业中广泛应用,即可以走Modbus-RTU协议,又可以走ModbusTCP/IP协议,传输举例可达数公里,工控行业的集控系统用这个接口比较多。RS485接口的...
Rs485是通信协议还是物理接口啊?若只是一个接口,就是说通过这...楼主百度百科有介绍通信协议有modbus等等485是通信协议还是物理接口啊?若只是一个接口,就是说通过这个RS485接口可以利用什么通讯协议来传输数...