Warm tip: This article is reproduced from serverfault.com, please click

network programming-在 C 中使用 Modbus 创建一个简单的客户端/服务器

(network programming - Create a simple Client/Server using Modbus in C)

发布于 2015-04-13 10:04:25

我目前正在开展一个项目,该项目将允许不同的自动化进行通信。为此,我想创建一个客户端和一个服务器,它们将使用 modbus 协议进行通信。我不确定我现在是否要使用 ModBus/TCP、ModBus/RTU 或 ModBus/ASCII。

我在 C 中搜索了客户端/服务器示例,我可以找到库,但没有简单的通信示例。我想从头开始,所以库不是我想要的。

我要问的是,是否有人可以给我一个用 C 编写的简单代码,用于使用 Modbus 进行通信的客户端和/或服务器,因为我不确定我将使用什么类型的 Modbus 会有很大帮助(RTU/TCP/ASCII)。

越简单越好,我希望代码演示的是,例如:对服务器的初始化、请求、应答、关闭连接。

非常感谢你的宝贵时间。

Questioner
PiggyGenius
Viewed
11
matpop 2015-04-17 18:17:02

三件事:

  1. 当你正在开发自己的客户端和服务器组件时,我建议你仅在严格要求或方便的情况下使用 Modbus(即其他制造商必须能够通过标准化协议与你的客户端或服务器组件进行通信) - 和 Modbus 适合)。
  2. 请注意,Modbus TCP 不仅仅是 TCP/IP 上的 Modbus RTU(/ASCII)(仍然允许,当然也允许 UDP)。有一些重要的差异需要考虑。
  3. 我了解你需要更深入地了解 Modbus。那时,一旦你在 C 程序中有一个开放的串行通道或(侦听)TCP 套接字,你就可以从简单的 Modbus 请求/响应开始。

看看这个简短但非常完整的描述,以及这个不断更新的库的文档。


这是一个基于libmodbus的 Linux 超级简化 RTU 示例。
请允许我放松一下 C99 以保持紧凑。
在现实世界中,你还应该正确处理 SIGTERM 等信号…… Linux 内核 2.6.28 及更高版本
还有一个(RS232 与 RS485)功能。modbus_rtu_set_serial_mode你可能会发现其他库可以在你的平台上更轻松地使用 RS485。

主片段

//Create a new RTU context with proper serial parameters (in this example,
//device name /dev/ttyS0, baud rate 9600, no parity bit, 8 data bits, 1 stop bit)
modbus_t *ctx = modbus_new_rtu("/dev/ttyS0", 9600, 'N', 8, 1);
if (!ctx) {
    fprintf(stderr, "Failed to create the context: %s\n", modbus_strerror(errno));
    exit(1);
}

if (modbus_connect(ctx) == -1) {
    fprintf(stderr, "Unable to connect: %s\n", modbus_strerror(errno));
    modbus_free(ctx);
    exit(1);
}

//Set the Modbus address of the remote slave (to 3)
modbus_set_slave(ctx, 3);


uint16_t reg[5];// will store read registers values

//Read 5 holding registers starting from address 10
int num = modbus_read_registers(ctx, 10, 5, reg);
if (num != 5) {// number of read registers is not the one expected
    fprintf(stderr, "Failed to read: %s\n", modbus_strerror(errno));
}

modbus_close(ctx);
modbus_free(ctx);

从属片段

//Prepare a Modbus mapping with 30 holding registers
//(plus no output coil, one input coil and two input registers)
//This will also automatically set the value of each register to 0
modbus_mapping_t *mapping = modbus_mapping_new(0, 1, 30, 2);
if (!mapping) {
    fprintf(stderr, "Failed to allocate the mapping: %s\n", modbus_strerror(errno));
    exit(1);
}


//Example: set register 12 to integer value 623
mapping->tab_registers[12] = 623;


modbus_t *ctx = modbus_new_rtu("/dev/ttyS0", 9600, 'N', 8, 1);
if (!ctx) {
    fprintf(stderr, "Failed to create the context: %s\n", modbus_strerror(errno));
    exit(1);
}

//Set the Modbus address of this slave (to 3)
modbus_set_slave(ctx, 3);


if (modbus_connect(ctx) == -1) {
    fprintf(stderr, "Unable to connect: %s\n", modbus_strerror(errno));
    modbus_free(ctx);
    exit(1);
}


uint8_t req[MODBUS_RTU_MAX_ADU_LENGTH];// request buffer
int len;// length of the request/response

while(1) {
    len = modbus_receive(ctx, req);
    if (len == -1) break;

    len = modbus_reply(ctx, req, len, mapping);
    if (len == -1) break;
}
printf("Exit the loop: %s\n", modbus_strerror(errno));

modbus_mapping_free(mapping);
modbus_close(ctx);
modbus_free(ctx);