initial
This commit is contained in:
245
Mudbus.cpp
Normal file
245
Mudbus.cpp
Normal file
@ -0,0 +1,245 @@
|
||||
/*
|
||||
Mudbus.cpp - an Arduino library for a Modbus TCP slave.
|
||||
Copyright (C) 2011 Dee Wykoff
|
||||
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#include "Mudbus.h"
|
||||
|
||||
// For Arduino 0022
|
||||
// Server MbServer(MB_PORT);
|
||||
// For Arduino 1.0
|
||||
WiFiServer MbServer = WiFiServer(MB_PORT);
|
||||
|
||||
Mudbus::Mudbus()
|
||||
{
|
||||
}
|
||||
|
||||
void Mudbus::Run()
|
||||
{
|
||||
static bool serverStarted = false;
|
||||
if (! serverStarted) {
|
||||
MbServer.begin();
|
||||
Serial.println("Modbus server started.");
|
||||
serverStarted = true;
|
||||
}
|
||||
|
||||
|
||||
Runs = 1 + Runs * (Runs < 999);
|
||||
|
||||
//****************** Read from socket ****************
|
||||
// For Arduino 0022
|
||||
// Client client = MbServer.available();
|
||||
// For Arduino 1.0
|
||||
WiFiClient client = MbServer.available();
|
||||
if(client.available())
|
||||
{
|
||||
Reads = 1 + Reads * (Reads < 999);
|
||||
int i = 0;
|
||||
while(client.available())
|
||||
{
|
||||
ByteArray[i] = client.read();
|
||||
i++;
|
||||
// Serial.println("read something from net");
|
||||
}
|
||||
SetFC(ByteArray[7]); //Byte 7 of request is FC
|
||||
if(!Active)
|
||||
{
|
||||
Active = true;
|
||||
PreviousActivityTime = millis();
|
||||
#ifdef MbDebug
|
||||
Serial.println("Mb active");
|
||||
#endif
|
||||
}
|
||||
}
|
||||
if(millis() > (PreviousActivityTime + 60000))
|
||||
{
|
||||
if(Active)
|
||||
{
|
||||
Active = false;
|
||||
#ifdef MbDebug
|
||||
Serial.println("Mb not active");
|
||||
#endif
|
||||
}
|
||||
}
|
||||
|
||||
int Start, WordDataLength, ByteDataLength, CoilDataLength, MessageLength;
|
||||
|
||||
//****************** Read Coils **********************
|
||||
if(FC == MB_FC_READ_COILS)
|
||||
{
|
||||
Start = word(ByteArray[8],ByteArray[9]);
|
||||
CoilDataLength = word(ByteArray[10],ByteArray[11]);
|
||||
ByteDataLength = CoilDataLength / 8;
|
||||
if(ByteDataLength * 8 < CoilDataLength) ByteDataLength++;
|
||||
CoilDataLength = ByteDataLength * 8;
|
||||
#ifdef MbDebug
|
||||
Serial.print(" MB_FC_READ_COILS S=");
|
||||
Serial.print(Start);
|
||||
Serial.print(" L=");
|
||||
Serial.println(CoilDataLength);
|
||||
#endif
|
||||
ByteArray[5] = ByteDataLength + 3; //Number of bytes after this one.
|
||||
ByteArray[8] = ByteDataLength; //Number of bytes after this one (or number of bytes of data).
|
||||
for(int i = 0; i < ByteDataLength ; i++)
|
||||
{
|
||||
for(int j = 0; j < 8; j++)
|
||||
{
|
||||
bitWrite(ByteArray[9 + i], j, C[Start + i * 8 + j]);
|
||||
}
|
||||
}
|
||||
MessageLength = ByteDataLength + 9;
|
||||
client.write(ByteArray, MessageLength);
|
||||
Writes = 1 + Writes * (Writes < 999);
|
||||
FC = MB_FC_NONE;
|
||||
}
|
||||
|
||||
//****************** Read Registers ******************
|
||||
if(FC == MB_FC_READ_REGISTERS)
|
||||
{
|
||||
Start = word(ByteArray[8],ByteArray[9]);
|
||||
WordDataLength = word(ByteArray[10],ByteArray[11]);
|
||||
ByteDataLength = WordDataLength * 2;
|
||||
#ifdef MbDebug
|
||||
Serial.print(" MB_FC_READ_REGISTERS S=");
|
||||
Serial.print(Start);
|
||||
Serial.print(" L=");
|
||||
Serial.println(WordDataLength);
|
||||
#endif
|
||||
ByteArray[5] = ByteDataLength + 3; //Number of bytes after this one.
|
||||
ByteArray[8] = ByteDataLength; //Number of bytes after this one (or number of bytes of data).
|
||||
for(int i = 0; i < WordDataLength; i++)
|
||||
{
|
||||
ByteArray[ 9 + i * 2] = highByte(R[Start + i]);
|
||||
ByteArray[10 + i * 2] = lowByte(R[Start + i]);
|
||||
}
|
||||
MessageLength = ByteDataLength + 9;
|
||||
client.write(ByteArray, MessageLength);
|
||||
Writes = 1 + Writes * (Writes < 999);
|
||||
FC = MB_FC_NONE;
|
||||
}
|
||||
|
||||
//****************** Write Coil **********************
|
||||
if(FC == MB_FC_WRITE_COIL)
|
||||
{
|
||||
Start = word(ByteArray[8],ByteArray[9]);
|
||||
C[Start] = word(ByteArray[10],ByteArray[11]) > 0;
|
||||
#ifdef MbDebug
|
||||
Serial.print(" MB_FC_WRITE_COIL C");
|
||||
Serial.print(Start);
|
||||
Serial.print("=");
|
||||
Serial.println(C[Start]);
|
||||
#endif
|
||||
ByteArray[5] = 2; //Number of bytes after this one.
|
||||
MessageLength = 8;
|
||||
client.write(ByteArray, MessageLength);
|
||||
Writes = 1 + Writes * (Writes < 999);
|
||||
FC = MB_FC_NONE;
|
||||
}
|
||||
|
||||
//****************** Write Register ******************
|
||||
if(FC == MB_FC_WRITE_REGISTER)
|
||||
{
|
||||
Start = word(ByteArray[8],ByteArray[9]);
|
||||
R[Start] = word(ByteArray[10],ByteArray[11]);
|
||||
#ifdef MbDebug
|
||||
Serial.print(" MB_FC_WRITE_REGISTER R");
|
||||
Serial.print(Start);
|
||||
Serial.print("=");
|
||||
Serial.println(R[Start]);
|
||||
#endif
|
||||
ByteArray[5] = 6; //Number of bytes after this one.
|
||||
MessageLength = 12;
|
||||
client.write(ByteArray, MessageLength);
|
||||
Writes = 1 + Writes * (Writes < 999);
|
||||
FC = MB_FC_NONE;
|
||||
}
|
||||
|
||||
|
||||
//****************** Write Multiple Coils **********************
|
||||
//Function codes 15 & 16 by Martin Pettersson http://siamect.com
|
||||
if(FC == MB_FC_WRITE_MULTIPLE_COILS)
|
||||
{
|
||||
Start = word(ByteArray[8],ByteArray[9]);
|
||||
CoilDataLength = word(ByteArray[10],ByteArray[11]);
|
||||
ByteDataLength = CoilDataLength / 8;
|
||||
if(ByteDataLength * 8 < CoilDataLength) ByteDataLength++;
|
||||
CoilDataLength = ByteDataLength * 8;
|
||||
#ifdef MbDebug
|
||||
Serial.print(" MB_FC_WRITE_MULTIPLE_COILS S=");
|
||||
Serial.print(Start);
|
||||
Serial.print(" L=");
|
||||
Serial.println(CoilDataLength);
|
||||
#endif
|
||||
ByteArray[5] = ByteDataLength + 5; //Number of bytes after this one.
|
||||
for(int i = 0; i < ByteDataLength ; i++)
|
||||
{
|
||||
for(int j = 0; j < 8; j++)
|
||||
{
|
||||
C[Start + i * 8 + j] = bitRead( ByteArray[13 + i], j);
|
||||
}
|
||||
}
|
||||
MessageLength = 12;
|
||||
client.write(ByteArray, MessageLength);
|
||||
Writes = 1 + Writes * (Writes < 999);
|
||||
FC = MB_FC_NONE;
|
||||
}
|
||||
|
||||
|
||||
//****************** Write Multiple Registers ******************
|
||||
//Function codes 15 & 16 by Martin Pettersson http://siamect.com
|
||||
if(FC == MB_FC_WRITE_MULTIPLE_REGISTERS)
|
||||
{
|
||||
Start = word(ByteArray[8],ByteArray[9]);
|
||||
WordDataLength = word(ByteArray[10],ByteArray[11]);
|
||||
ByteDataLength = WordDataLength * 2;
|
||||
#ifdef MbDebug
|
||||
Serial.print(" MB_FC_READ_REGISTERS S=");
|
||||
Serial.print(Start);
|
||||
Serial.print(" L=");
|
||||
Serial.println(WordDataLength);
|
||||
#endif
|
||||
ByteArray[5] = ByteDataLength + 3; //Number of bytes after this one.
|
||||
for(int i = 0; i < WordDataLength; i++)
|
||||
{
|
||||
R[Start + i] = word(ByteArray[ 13 + i * 2],ByteArray[14 + i * 2]);
|
||||
}
|
||||
MessageLength = 12;
|
||||
client.write(ByteArray, MessageLength);
|
||||
Writes = 1 + Writes * (Writes < 999);
|
||||
FC = MB_FC_NONE;
|
||||
}
|
||||
|
||||
#ifdef MbDebugX
|
||||
Serial.print("Mb runs: ");
|
||||
Serial.print(Runs);
|
||||
Serial.print(" reads: ");
|
||||
Serial.print(Reads);
|
||||
Serial.print(" writes: ");
|
||||
Serial.print(Writes);
|
||||
Serial.println();
|
||||
#endif
|
||||
}
|
||||
|
||||
|
||||
void Mudbus::SetFC(int fc)
|
||||
{
|
||||
if(fc == 1) FC = MB_FC_READ_COILS;
|
||||
if(fc == 3) FC = MB_FC_READ_REGISTERS;
|
||||
if(fc == 5) FC = MB_FC_WRITE_COIL;
|
||||
if(fc == 6) FC = MB_FC_WRITE_REGISTER;
|
||||
if(fc == 15) FC = MB_FC_WRITE_MULTIPLE_COILS;
|
||||
if(fc == 16) FC = MB_FC_WRITE_MULTIPLE_REGISTERS;
|
||||
}
|
Reference in New Issue
Block a user