Librería Modbus RTU Master para automatización industrial

Conoce qué es Modbus RTU y sus aplicaciones
18 de marzo de 2020 por
Librería Modbus RTU Master para automatización industrial
Boot & Work Corp. S.L., Fernandez Queralt Martinez

Introduction

En el área de automatización de Arduino, el protocolo Modbus RTU es un medio de comunicación que permite el intercambio de datos entre controladores lógicos programables (controlador PLC Arduino) y computadoras.

Los dispositivos electrónicos pueden intercambiar información a través de líneas en serie utilizando el protocolo Modbus.

En este post, vas a aprender cómo funciona Modbus RTU.

Index

1. ¿Qué es un Modbus RTU?

2. ¿Cómo funciona un Modbus RTU?

3. Estructura general del marco Modbus

4. Código de función Modbus

5. Tipos de objetos Modbus

6. Formato de datos Modbus

    6.1. (0x01) Leer Coils 

    6.2. (0x02) Leer entradas discretas

    6.3. (0x03) Lectura de registros de retención

    6.4. (0x04) Leer registro de entrada

    6.5. (0x05) Escribir 1 coil

    6.6. (0x06) Escribir 1 registro

    6.7. (0x0F) Escribir múltiples coils

    6.8. (0x10) Escribir múltiples registros

7. Creando nuestro mensaje Modbus RTU

8. Software

1. ¿Qué es un Modbus RTU?

Modbus es un protocolo de comunicación situado en los niveles 1, 2 y 7 del Modelo OSI, basado en la arquitectura maestro/esclavo, diseñado en 1979 por Modicon para su gama de PLCs.

Convertido en un protocolo de comunicaciones de facto en la industria, veamos algunas de sus principales características:

  1. Diseñado pensando en su uso en aplicaciones industriales
  2. Es público y gratis
  3. Es fácil de implementar y requiere un poco de desarrollo.
  4. Maneja bloques de datos sin asumir restricciones
  5. Cada uno de los mensajes incluye información redundante que garantiza su integridad en la recepción.
  6. Los comandos básicos de Modbus permiten controlar un dispositivo RTU para modificar el valor de cualquiera de sus registros o solicitar el contenido de los mismos.

2. ¿Cómo funciona un MODBUS RTU?

Modbus RTU es la implementación más común disponible para Modbus.

Modbus RTU se utiliza en la comunicación en serie y hace uso de una representación compacta y binaria de los datos para la comunicación del protocolo.

Los mensajes Modbus se dividen por periodos de inactividad, como se puede ver en la siguiente imagen.

Mensaje - Cómo funciona MODBUS RTU - Librería maestra Modbus RTU para la automatización industrial

Each device on a Modbus communication has a unique address

The Modbus RTU works by RS-485 which is a single cable multi-drop network, only the node assigned as the Master may initiate a command. All the other devices are slaves and answer requests and commands.

Un comando Modbus contiene la dirección Modbus del dispositivo al que va dirigido. Sólo el dispositivo al que va dirigido responderá y actuará sobre el comando, aunque otros dispositivos puedan recibirlo.

Además, es importante decir que todos los comandos Modbus contienen información de suma de comprobación para que el receptor pueda detectar errores de transmisión.

Pongamos un ejemplo. Imaginemos que tenemos una red serial Modbus, donde hay un maestro y hasta 31 esclavos, cada uno con una dirección de esclavo única.

El maestro sólo quiere enviar un mensaje al esclavo número 2 solicitando el valor de 6 registros de entrada.

Así, el maestro enviaría un mensaje y todos los esclavos lo recibirían, pero sólo el esclavo número 2 respondería y actuaría sobre la orden, aunque otros dispositivos podrían recibirla.

Con este ejemplo, vamos a crear un mensaje Modbus RTU a lo largo de este post. Mensaje Modbus por el momento: 02 (dirección del esclavo)

Maestro y Esclavo - Cómo funciona MODBUS RTU - Librería maestra Modbus RTU para la automatización industrial

3. Estructura general del marco Modbus

La unidad de datos de aplicación (ADU) de Modbus RTU consta de los elementos mostrados:

Estructura general del marco Modbus - Librería maestra Modbus RTU para la automatización industrial

Dirección: Establecemos la dirección del esclavo para el dispositivo al que queremos enviar el mensaje.
Código de función: El número del código de función. Puedes ver la tabla de los códigos de función en la sección "Códigos de función Modbus".
Data: El mensaje en sí. Puede variar según el código de la función.
CRC: El número de la comprobación de redundancia cíclica. Debe ser calculado.

De ellos, el código de función y los datos constituyen la unidad de datos de protocolo (PDU)


4.Código de función Modbus

MODBUS is a request/reply protocol and offers services specified by function codes. MODBUS function codes are elements of MODBUS request/reply PDUs. 

The function code field of a MODBUS data unit is coded in one byte. Valid codes are in the range of 1 to 255 decimal (the range 128 – 255 is reserved and used for exception responses). When a message is sent from a Client to a Server device the function code field tells the server what kind of action to perform. Function code "0" is not valid. Sub-function codes are added to some function codes to define multiple actions.  

A continuación puedes encontrar la lista de códigos de función y sus funciones:

Códigos de función Modbus - Librería maestra Modbus RTU para la automatización industrial

5. Tipos de objetos Modbus

En Modbus, los tipos de datos se pueden dividir principalmente en dos tipos: Coils y Registros. Las Coils pueden entenderse como digitales ya que sólo pueden estar en ON (1) o en OFF (0). Algunas coils pueden representar entradas y otras salidas.

Los registros son de 16 bits ( 2 bytes) sin signo y por lo tanto pueden tener valores de 0 a 65535 (0 a FFFF). Aunque tiene sus limitaciones como que no puede representar números negativos , números en coma flotante o valores con representación mayor a 65535. La siguiente tabla resume los tipos de objetos.

Las cuatro tablas principales son las siguientes:

TABLAS PRINCIPALESTIPO DE OBJECTOTIPO DECOMENTARIOS
Entradas discretas.
 (Entradas)
Un solo bitSólo lecturaEste tipo de datos será proporcionado por un sistema de E/S.
Coils
(Salidas)
Un solo bitLectura-EscrituraEste tipo de datos pueden ser alterados por un programa de aplicación.
Registros de entrada
(Entradas)
Palabra de 16 bitsSólo lecturaEste tipo de datos puede ser proporcionado por un sistema de E/S.
Registros de retención
(Salidas)
Palabra de 16 bitsLectura-EscrituraEste tipo de datos pueden ser alterados por un programa de aplicación.

6. Formato de datos Modbus

Descripción de los códigos de función

Las peticiones y respuestas de Modbus contienen una Unidad de Datos de Aplicación (ADU) que contiene una Unidad de Datos de Protocolo (PDU).

Descripción de los códigos de función - Formato de datos Modbus - Librería maestra Modbus RTU para la automatización industrial

6.1. (0x01) Leer Coils

Este código de función se utiliza para leer de 1 a 2000 estados contiguos de Coil en un dispositivo remoto.

Las Coilen el mensaje de respuesta se empaquetan como una Coil por bit del campo de datos. El estado se indica como 1: ON y 0: OFF. El LSB del primer byte de datos contiene la salida a la que se dirige la consulta. Las demás coils siguen hacia el extremo de orden alto de este byte, y de orden bajo a orden alto en los bytes siguientes.

Solicitud

Código de función1 Byte0x01
Dirección inicial2 Bytes0x0000 a 0xFFFF
Cantidad de Coils2 Bytes1 a 2000 (0x7D0)

Respuesta

Código de función1 Byte0x01
Cantidad de bytes1 ByteN*
Estado de la Coilsn Bytesn = N o N+1

*N =Cantidad de salidas / 8, si el resto es diferente de 0 => N = N+1


Ejemplo de solicitud de lectura de las salidas discretas 20 - 38: 

Solicitud
Respuesta
Nombre del campoHexNombre del campoHex
Función01Función01
Dirección inicial Hi00Cantidad de bytes03
Dirección inicial Lo13Estado de las salidas 27-20CD
Cantidad de salidas Hi00Estado de las salidas 35-286B
Cantidad de salidas Lo13Estados de las salidas05

*El CRC debe ser calculado.


6.2. (0x02) Leer Entradas Discretas

 Este código de función se utiliza para leer de 1 a 2000 estados contiguos de entradas discretas en un dispositivo remoto.

La PDU de solicitud especifica la dirección de inicio, es decir, la dirección de la primera entrada especificada, y el número de entradas. En la PDU, las entradas discretas se direccionan empezando por cero. Por lo tanto, las entradas discretas numeradas de 1 a 16 se direccionan como 0 a 15.

Las entradas discretas en el mensaje de respuesta se empaquetan como una entrada por bit del campo de datos. El estado se indica como 1= ON; 0= OFF. El LSB del primer byte de datos contiene la entrada dirigida en la consulta. Las demás entradas siguen hacia el final de orden alto de este byte, y de orden bajo a orden alto en los bytes siguientes.

Solicitud

Código de función1 Byte0x02
Dirección inicial2 Bytes0x0000 a 0xFFFF
Cantidad de entradas2 Bytes1 a 2000 (0x7D0)


Respuesta

Código de función1 Byte0x02
Recuento de bytes1 ByteN*
Estado de las entradasN* x 1 Byte


Ejemplo de petición de lectura de las entradas discretas 197 - 218:

Solicitud

Respuesta
Nombre del campoHexNombre del campoHex
Función02Función02
Dirección de salida Hi00Recuento de bytes03
Dirección de salida LoC4Estado de las salidas 27-20AD
Cantidad de salidas Hi00Estado de las salidas 35-28DB
Cantidad de salidas Lo16Estado de las salidas35


6.3. (0x03) Leer Registros de Retención

Este código de función se utiliza para leer el contenido de un bloque contiguo de registros de retención en un dispositivo remoto. La PDU de solicitud especifica la dirección de registro inicial y el número de registros. En la PDU los registros se direccionan empezando por cero. Por lo tanto, los registros numerados del 1 al 16 se direccionan como 0-15.

Los datos del registro en el mensaje de respuesta se empaquetan en dos bytes por registro, con el contenido binario justificado a la derecha dentro de cada byte. Para cada registro, el primer byte contiene los bits de orden alto y el segundo contiene los bits de orden bajo.

Solicitud

Código de función1 Byte0x03
Dirección inicial2 Bytes0x0000 a 0xFFFF
Cantidad de registros2 Bytes1 a 125 (0x7D0)

Respuesta

Código de función1 Byte0x03
Recuento de bytes1 Byte2 x N*
Valor de registroN* x 2 Bytes

*N = Cantidad de registros


Ejemplo de solicitud de lectura de los registros 108 - 110: 

Solicitud

Respuesta
Nombre del campoHexNombre del campoHex
Función03Función03
Dirección de salida Hi00Recuento de bytes06
Dirección de salida Lo6BValor del registro Hi (108)
Valor del registro Lo (108)
02

2B
No. de Registros Hi00Valor del registro Hi (109)
Valor del registro Lo (109)
00

00
No. de Registros Lo03Valor del registro Hi (110)
Valor del registro Lo (110)
00

64



6.4. (0x04) Leer Registros de Entradas

Este código de función se utiliza para leer de 1 a 125 registros de entrada contiguos en un dispositivo remoto. La PDU de solicitud especifica la dirección de registro inicial y el número de registros. En la PDU los registros se direccionan empezando por cero. Por lo tanto, los registros de entrada numerados del 1 al 16 se direccionan como 0-15.

Los datos del registro en el mensaje de respuesta se empaquetan en dos bytes por registro, con el contenido binario justificado a la derecha dentro de cada byte. Para cada registro, el primer byte contiene los bits de orden alto y el segundo contiene los bits de orden bajo.

Solicitud

Código de función1 Byte0x04
Dirección inicial2 Bytes0x0000 a 0xFFFF
Cantidad de registros de entrada2 Bytes0x0001 a 0x007D

Respuesta

Código de función1 Byte0x04
Recuento de bytes1 Byte2 x N*
Registros de entradaN* x 2 Bytes

*N = Cantidad de registros de entrada

Ejemplo de solicitud de lectura del registro de entrada 9

Solicitud

Respuesta
Nombre del campoHexNombre del campoHex
Función04Función04
Dirección de salida Hi00Recuento de bytes02
Dirección de salida Lo08Entrada Reg. 9 Hi00


Cantidad de registros de entrada Hi00Entrada Reg. 9 Lo
0A
Cantidad de registros de entrada Lo
01


6.5. (0x05) Escribir un solo coil

Este código de función se utiliza para escribir una única salida en ON u OFF en un dispositivo remoto.

El estado ON/OFF solicitado se especifica mediante una constante en el campo de datos de solicitud. Un valor de FF00 hex. solicita que la salida esté en ON. Un valor de 00 00 solicita que esté en OFF. Todos los demás valores son ilegales y no afectan a la salida.

La PDU de petición especifica la dirección de la Coil que se va a forzar. Las Coils se direccionan empezando por el cero. Por lo tanto, la coil número 1 se direcciona como 0. El estado ON/OFF solicitado se especifica mediante una constante en el campo Valor de la Coil. Un valor de 0XFF00 solicita que la Coil esté en ON. Un valor de 0X0000 solicita que la coil esté apagada. Todos los demás valores son ilegales y no afectan a la Coil.

La respuesta normal es un eco de la petición, que se devuelve después de escribir el estado de la Coil.

Solicitud

Código de función1 Byte0x05
Dirección de salida2 Bytes0x0000 a 0xFFFF
Valor de salida2 Bytes0x0000 a 0xFF00

Respuesta

Código de función1 Byte0x05
Recuento de bytes2 Byte0x0000 a 0xFFFF
Registros de entrada2 Bytes0x0000 o 0xFF00


Ejemplo de solicitud de escritura de la Coil 173 ON:

Solicitud
Respuesta
Nombre del campoHexNombre del campoHex
Función05Función05
Dirección inicial Hi00Dirección de salida Hi
00
Dirección inicial LoACDirección de salida Lo
AC


Cantidad de Reg. de Entrada HiFFValor de salida HiFF
Cantidad de Reg. de Entrada Lo
00Valor de salida Lo
00
 


6.6. (0x06) Escribir un solo registro

Este código de función se utiliza para escribir un único registro de retención en un dispositivo remoto.

La PDU de solicitud especifica la dirección del registro que se va a escribir. Los registros se direccionan empezando por cero. Por lo tanto, el registro número 1 se direcciona como 0.

La respuesta normal es un eco de la solicitud, que se devuelve después de que se haya escrito el contenido del registro.

Solicitud

Código de función1 Byte0x06
Dirección de registro2 Bytes0x0000 a 0xFFFF
Valor de registro2 Bytes0x0000 a 0xFFFF

Respuesta

Código de función1 Byte0x06
Dirección de registro2 Byte0x0000 a 0xFFFF
Valor de registro2 Bytes0x0000 o 0xFF00


Ejemplo de solicitud de escritura del registro 2 a 00 03 hex:

Solicitud
Respuesta
Nombre del campoHexNombre del campoHex
Función06Función06
Dirección inicial Hi00Dirección de salida Hi
00
Dirección inicial Lo01Dirección de salida Lo
01


Cantidad de Reg. de Entrada Hi00Valor de salida Hi00
Cantidad de Reg. de Entrada Lo
03Valor de salida Lo
03



6.7. (0x0F) Escribir Múltiples Coils

Este código de función se utiliza para forzar cada Coil de una secuencia de Coil a ON u OFF en un dispositivo remoto. La PDU de solicitud especifica las referencias de las coils que deben forzarse. Las Coils se direccionan empezando por el cero. Por lo tanto, la coil número 1 se direcciona como 0.

Los estados ON/OFF solicitados se especifican mediante el contenido del campo de datos de solicitud. Un '1' lógico en una posición de bit del campo solicita que la salida correspondiente esté en ON. Un '0' lógico solicita que esté en OFF.

La respuesta normal devuelve el código de función, la dirección de inicio y la cantidad de Coils forzadas.


Solicitud

Código de función1 Byte0x0F
Dirección inicial2 Bytes0x0000 a 0xFFFF
Cantidad de salidas2 Bytes0x0001 a 0x07B0
Recuento de bytes1 ByteN*
Valor de las salidasN* x 1 Byte 

*N = Cantidad de salidas / 8, si el resto es diferente de 0 => N = N+1


Respuesta

Código de función1 Byte0x0F
Dirección inicial2 Byte0x0000 a 0xFFFF
Cantidad de salidas2 Bytes0x0001 o 0x07B0


Ejemplo de petición de escritura del registro 2 a 00 03 hex:

Solicitud

Respuesta
Nombre del campoHexNombre del campoHex
Función0FFunción0F
Dirección de salida Hi00Dirección de salida Hi00
Dirección de salida Lo13Dirección de salida Lo13
Cantidad de salidas Hi00Cantidad de salidas Hi00
Cantidad de salidas Lo0ACantidad de salidas Lo0A
Recuento de bytes02

Valor de las salidas HiCD

Valor de las salidas Lo01



6.8. (0x10) Escribir Múltiples Registros

Este código de función se utiliza para escribir un bloque de registros contiguos (de 1 a 123 registros) en un dispositivo remoto.

Los valores escritos solicitados se especifican en el campo de datos de solicitud. Los datos se empaquetan en dos bytes por registro.

La respuesta normal devuelve el código de función, la dirección inicial y la cantidad de registros escritos.


Solicitud

Código de función1 Byte0x10
Dirección inicial2 Bytes0x0000 a 0xFFFF
Cantidad de registros2 Bytes0x0001 a 0x007B
Recuento de bytes1 Byte2 x N*
Valor de los registrosN* x 2 Bytesvalor

*N = Cantidad de registros


Respuesta

Código de función1 Byte0x10
Dirección inicial2 Byte0x0000 a 0xFFFF
Cantidad de registros2 Bytes0x123 o (0x7B)


Ejemplo de una solicitud para escribir dos registros que comienzan en 2 a 00 0A y 01 02 hex:

Solicitud

Respuesta
Nombre del campoHexNombre del campoHex
Función10Función10
Dirección de salida Hi00Dirección de salida Hi00
Dirección de salida Lo01Dirección de salida Lo01
Cantidad de registros Hi00Cantidad de salidas Hi00
Cantidad de registros Lo
02Cantidad de salidas Lo02
Recuento de bytes04

Valor de los registros Hi00

Valor de los registros Lo
0A

Valor de los registros Hi
01

Valor de los registros Lo

02





7. Creando nuestro mensaje Modbus RTU

Ahora que ya sabemos un poco más sobre Modbus RTU y su formato de trama, vamos a terminar nuestro mensaje Modbus del ejemplo que dimos al principio de esta entrada del post.

Queríamos que el maestro enviara un mensaje al esclavo número 2 solicitando el valor de 6 registros de entrada.

Nuestro mensaje Modbus RTU se ve así en este momento: 0204 (02 (Dirección del esclavo) + 04 (Código de función))

Como nuestro código de función es el número 04: Read Input Register, los datos deben contener: Dirección inicial Hi + Dirección inicial Lo + Cantidad de Reg. de Entrada Hi + Cantidad de Reg. de Entrada Lo. Lo + CRC.

Por lo tanto, vamos a llenar la solicitud ADU para obtener todo el mensaje:

Solicitud ADU
Nombre del campoHEX
Dirección del esclavo02
Código de función04
Dirección inicial Hi00
Dirección inicial Lo
00
Cantidad de Reg. de Entrada Hi00
Cantidad de Reg. de entrada Lo
06
CRC-
CRC-

Para calcular el CRC, basta con escribir el mensaje Modbus 020400000006 en esta web. Selecciona el tipo de entrada HEX y obtén el número CRC-16 (Modbus).

Como es LSB, lo invertiremos. Si el resultado del CRC es 0x3B70, ahora será: 703B.

Finalmente, así es como queda nuestro mensaje Modbus:

020400000006703B

8. Software

Modbus RTU Master conArduino IDE

El módulo maestro Modbus RTU implementa las capacidades del Modbus RTU Master. Vamos a trabajar con la función modbusrtumaster.h:

#include <ModbusRTUMaster.h>


Es posible utilizar cualquier secuencia de hardware Serial Arduino:

  • RS-485
#include <RS485.h>

ModbusRTUMaster master(RS485);


  • RS-232
#include <RS232.h>

ModbusRTUMaster master(RS232);


Antes de usarlo, es necesario llamar a la función de inicio en la configuración tanto para la variable serial como para la variable Modbus. Es una buena práctica establecer la velocidad de transmisión (valor predeterminado: 19200 bps) también en la variable Modbus para definir los tiempos de espera internos de Modbus.

RS485.begin(9600, HALFDUPLEX, SERIAL_8E1);
master.begin(9600);


Las funciones para leer y escribir los valores de los esclavos son:

readCoils(slave_address, address, quantity);
readDiscreteInputs(slave_address, address, quantity);
readHoldingRegisters(slave_address, address, quantity);
readInputRegisters(slave_address, address, quantity);
writeSingleCoil(slave_address, address, value);
writeSingleRegister(slave_address, address, value);
writeMultipleCoils(slave_address, address, values, quantity);
writeMultipleRegisters(slave_address, address, values, quantity);


Donde:

  • slave_address es la dirección del esclavo Modbus RTU.
  • address es la coil, entrada digital, registro de retención o dirección de registro de entrada. Por lo general, esta dirección es la coil, la entrada digital, el registro de retención o el número de registro de entrada menos 1: el número de registro de retención 40009 tiene la dirección 8.
    quantity es el número de coils, digitales, registros de retención o registros de entrada a leer/escribir.
  • value es el valor dado de la coil o los registros de retención en una operación de escritura. Dependiendo de la función, el tipo de datos cambia. Una coil está representada por un valor bool y un registro de retención está representado por un valor
    uint16_t.

En una función de lectura/escritura múltiple, el argumento de address es la primera dirección. En una función de escritura múltiple, el argumento values es una matriz de valores para escribir.

Es importante decir que estas funciones no son de bloqueo, por lo que no devuelven el valor leído. Devuelven true o false dependiendo del estado actual del módulo. Si hay una petición Modbus pendiente, devuelven false.  

// Read 5 holding registers from address 0x24 of slave with address 0x10
if (master.readHoldingRegisters(0x100x245)) {
	// OK, the request is being processedelse {
	// ERROR, the master is not in an IDLE state
}

Existe la función vailable() para verificar las respuestas del esclavo.

ModbusResponse response = master.available();
if (response) {
	// Process response
}


ModbusResponse implementa algunas funciones para obtener la información de respuesta:

hasError();
getErrorCode();
getSlave();
getFC();
isCoilSet(offset);
isDiscreteInputSet(offset);
isDiscreteSet(offset);
getRegister(offset);
ModbusResponse response = master.available();
if (response) {
	if (response.hasError()) {
		// There is an error. You can get the error code with response.getErrorCode()else {
		// Response ready: print the read holding registers
		for (int i = 0; i < 5; ++i) {
			Serial.println(response.getRegister(i));
		}
	}
}

 

Los posibles códigos de error son:

0x01 ILLEGAL FUNCTION
0x02 ILLEGAL DATA ADDRESS
0x03 ILLEGAL DATA VALUE
0x04 SERVER DEVICE FAILURE


Fuente

Más información >>>

Buscar en nuestro blog

Librería Modbus RTU Master para automatización industrial
Boot & Work Corp. S.L., Fernandez Queralt Martinez 18 de marzo de 2020
Compartir

¿Estás buscando tu Controlador Lógico Programable ideal?

Echa un vistazo a esta comparativa de producto de varios controladores industriales basados en Arduino.

Comparamos entradas, salidas, comunicaciones y otras especificaciones con las de los equipos de otras marcas destacadas.


Industrial PLC comparison >>>