Как читать из Modbus / TCP с помощью libmodbus

У меня есть moxa ioLogik E1214, и я хочу общаться с ним, используя modbus / tcp. Я нашел библиотеку, которая должна это делать - libmodbus. Сначала я скопировал образец кода с их веб-сайта и немного изменил его, чтобы он выглядел так:

#include <stdio.h>
#ifndef _MSC_VER
#include <unistd.h>
#include <sys/time.h>
#endif
#include <string.h>
#include <stdlib.h>
#include <time.h>
#include <errno.h>
#include <modbus.h>
#define G_MSEC_PER_SEC 1000

int main(int argc, char *argv[])
{
    modbus_t *mb;
    uint16_t tab_reg[30];
    int length = 6;

    mb = modbus_new_tcp("192.168.127.254", 502);
    if (modbus_connect(mb) == -1) {
        fprintf(stderr, "Connection failed: %s\n", modbus_strerror(errno));
        modbus_free(mb);
        return -1;
    }

    /* Read "length" registers from the address 0 */
    int read_val = modbus_read_registers(mb, 0, length, tab_reg);

    if(read_val==-1)
        printf("ERROR: %s\n", modbus_strerror(errno));
    else
    {
        printf("Read registers: %d\n", read_val);
        for(int i=0; i<length; i++)
        {
            printf("%d ", tab_reg[i]);
        }
        printf("\n");
    }
    modbus_close(mb);
    modbus_free(mb);
    return 0;
}

Соединение работает, но у меня проблемы с чтением регистров. В документации есть некоторые параметры с описанием, начальным адресом, длиной и т.д., это выглядит так: документация moxa

Итак, я понимаю, что использование int read_val = modbus_read_registers(mb, 0, length, tab_reg); должно считывать значение с адреса 0 и сохранять его в tab_reg. Но я получаю сообщение об ошибке "Недопустимый адрес данных". То же самое и с другими адресами, которые я пытался прочитать.

Итак, подведем итоги - все ли я правильно понимаю? Как мне использовать libmodbus для чтения регистров?


person P.B.    schedule 24.05.2019    source источник


Ответы (2)


Кажется, у вашего ведомого устройства нет регистров хранения по адресу 0, и поэтому он жалуется на ошибку illegal data address.

Вам нужно посмотреть на карту регистров Modbus, которая у вас есть, и соответственно использовать правильную функцию Modbus из libmodbus. Например, если вы хотите прочитать свою переменную RLY_pulseOnWidth, вы должны использовать:

int read_val = modbus_read_input_registers(mb, 52, length, tab_reg);

Но читать RLY_TotalCount:

int read_val = modbus_read_registers(mb, 64, length, tab_reg);

Здесь вы можете проверить все коды функций Modbus, относящиеся к столбцу «Тип точки»:

https://github.com/stephane/libmodbus/blob/ddac0cf909d9ab05423df7b9a66e6e60e8bb0fbb/src/modbus.h

person Marcos G.    schedule 24.05.2019

Вы пытаетесь использовать «регистры чтения» (значения int16), когда вам нужна «катушка чтения» (биты). Катушки и регистры хранятся в отдельных таблицах, и поэтому адресуются по-разному.

Используйте modbus_read_bits для чтения RLY_status, поскольку, как указано в вашей документации, это значение катушки.

Вам нужно будет читать регистры ввода и хранения иначе, чем катушки. Ваш примерный код должен быть удобен для чтения регистров хранения.

person Thomas Østergaard    schedule 12.06.2019