1644 lines
52 KiB
C
Raw Normal View History

/**
* @file agile_modbus.c
* @brief Agile Modbus
* @author (2544047213@qq.com)
* @date 2022-07-28
*
@verbatim
使
`` `` ``
-
1. `agile_modbus_rtu_init` / `agile_modbus_tcp_init` `RTU/TCP`
2. `agile_modbus_set_slave`
3. ``
4. `agile_modbus_serialize_xxx`
5. ``
6. ``
7. `agile_modbus_deserialize_xxx`
8.
-
1. `agile_modbus_slave_callback_t`
2. `agile_modbus_rtu_init` / `agile_modbus_tcp_init` `RTU/TCP`
3. `agile_modbus_set_slave`
4. ``
5. `agile_modbus_slave_handle`
6. `` ()
7. ``
@endverbatim
*
* @attention
*
* <h2><center>&copy; Copyright (c) 2021 Ma Longwei.
* All rights reserved.</center></h2>
*
*/
#include "agile_modbus.h"
#include <string.h>
/** @defgroup COMMON Common
* @{
*/
/** @defgroup COMMON_Private_Constants Common Private Constants
* @{
*/
#define AGILE_MODBUS_MSG_LENGTH_UNDEFINED -1 /**< 对应功能码数据长度未定义 */
/**
* @}
*/
/** @defgroup COMMON_Private_Functions Common Private Functions
* @{
*/
/**
* @brief
@verbatim
---------- Request Indication ----------
| Client | ---------------------->| Server |
---------- Confirmation Response ----------
03
---------- ------ --------------- ---------
| header | | 03 | | 00 00 00 01 | | CRC16 |
---------- ------ --------------- ---------
----------
| header |
----------
RTU:
TCP: | |
---------------
| 00 00 00 01 |
---------------
: 03
@endverbatim
* @param ctx modbus
* @param function
* @param msg_type
* @return
*/
static uint8_t agile_modbus_compute_meta_length_after_function(
agile_modbus_t* ctx,
int function,
agile_modbus_msg_type_t msg_type) {
int length;
if (msg_type == AGILE_MODBUS_MSG_INDICATION) {
if (function <= AGILE_MODBUS_FC_WRITE_SINGLE_REGISTER) {
length = 4;
} else if (function == AGILE_MODBUS_FC_WRITE_MULTIPLE_COILS ||
function == AGILE_MODBUS_FC_WRITE_MULTIPLE_REGISTERS) {
length = 5;
} else if (function == AGILE_MODBUS_FC_MASK_WRITE_REGISTER) {
length = 6;
} else if (function == AGILE_MODBUS_FC_WRITE_AND_READ_REGISTERS) {
length = 9;
} else {
/* MODBUS_FC_READ_EXCEPTION_STATUS, MODBUS_FC_REPORT_SLAVE_ID */
length = 0;
if (ctx->compute_meta_length_after_function)
length = ctx->compute_meta_length_after_function(ctx, function,
msg_type);
}
} else {
/* MSG_CONFIRMATION */
switch (function) {
case AGILE_MODBUS_FC_READ_COILS:
case AGILE_MODBUS_FC_READ_DISCRETE_INPUTS:
case AGILE_MODBUS_FC_READ_HOLDING_REGISTERS:
case AGILE_MODBUS_FC_READ_INPUT_REGISTERS:
case AGILE_MODBUS_FC_REPORT_SLAVE_ID:
case AGILE_MODBUS_FC_WRITE_AND_READ_REGISTERS:
length = 1;
break;
case AGILE_MODBUS_FC_WRITE_SINGLE_COIL:
case AGILE_MODBUS_FC_WRITE_SINGLE_REGISTER:
case AGILE_MODBUS_FC_WRITE_MULTIPLE_COILS:
case AGILE_MODBUS_FC_WRITE_MULTIPLE_REGISTERS:
length = 4;
break;
case AGILE_MODBUS_FC_MASK_WRITE_REGISTER:
length = 6;
break;
default:
length = 1;
if (ctx->compute_meta_length_after_function)
length = ctx->compute_meta_length_after_function(
ctx, function, msg_type);
}
}
return length;
}
/**
* @brief
@verbatim
---------- Request Indication ----------
| Client | ---------------------->| Server |
---------- Confirmation Response ----------
03
---------- ------ ------ --------- ---------
| header | | 03 | | 02 | | 00 00 | | CRC16 |
---------- ------ ------ --------- ---------
----------
| header |
----------
RTU:
TCP: | |
------
| 02 |
------
:
---------
| 00 00 |
---------
@endverbatim
* @param ctx modbus
* @param msg
* @param msg_length
* @param msg_type
* @return
*/
static int agile_modbus_compute_data_length_after_meta(
agile_modbus_t* ctx,
uint8_t* msg,
int msg_length,
agile_modbus_msg_type_t msg_type) {
int function = msg[ctx->backend->header_length];
int length;
if (msg_type == AGILE_MODBUS_MSG_INDICATION) {
switch (function) {
case AGILE_MODBUS_FC_WRITE_MULTIPLE_COILS:
case AGILE_MODBUS_FC_WRITE_MULTIPLE_REGISTERS:
length = msg[ctx->backend->header_length + 5];
break;
case AGILE_MODBUS_FC_WRITE_AND_READ_REGISTERS:
length = msg[ctx->backend->header_length + 9];
break;
default:
length = 0;
if (ctx->compute_data_length_after_meta)
length = ctx->compute_data_length_after_meta(
ctx, msg, msg_length, msg_type);
}
} else {
/* MSG_CONFIRMATION */
if (function <= AGILE_MODBUS_FC_READ_INPUT_REGISTERS ||
function == AGILE_MODBUS_FC_REPORT_SLAVE_ID ||
function == AGILE_MODBUS_FC_WRITE_AND_READ_REGISTERS) {
length = msg[ctx->backend->header_length + 1];
} else {
length = 0;
if (ctx->compute_data_length_after_meta)
length = ctx->compute_data_length_after_meta(
ctx, msg, msg_length, msg_type);
}
}
length += ctx->backend->checksum_length;
return length;
}
/**
* @brief
* @param ctx modbus
* @param msg
* @param msg_length
* @param msg_type
* @return >0:modbus ; :
*/
static int agile_modbus_receive_msg_judge(agile_modbus_t* ctx,
uint8_t* msg,
int msg_length,
agile_modbus_msg_type_t msg_type) {
int remain_len = msg_length;
remain_len -= (ctx->backend->header_length + 1);
if (remain_len < 0)
return -1;
remain_len -= agile_modbus_compute_meta_length_after_function(
ctx, msg[ctx->backend->header_length], msg_type);
if (remain_len < 0)
return -1;
remain_len -= agile_modbus_compute_data_length_after_meta(
ctx, msg, msg_length, msg_type);
if (remain_len < 0)
return -1;
return ctx->backend->check_integrity(ctx, msg, msg_length - remain_len);
}
/**
* @}
*/
/** @defgroup COMMON_Exported_Functions Common Exported Functions
* @{
*/
/**
* @brief modbus
* @param ctx modbus
* @param send_buf
* @param send_bufsz
* @param read_buf
* @param read_bufsz
*/
void agile_modbus_common_init(agile_modbus_t* ctx,
uint8_t* send_buf,
int send_bufsz,
uint8_t* read_buf,
int read_bufsz) {
memset(ctx, 0, sizeof(agile_modbus_t));
ctx->slave = -1;
ctx->send_buf = send_buf;
ctx->send_bufsz = send_bufsz;
ctx->read_buf = read_buf;
ctx->read_bufsz = read_bufsz;
}
/**
* @brief
* @param ctx modbus
* @param slave
* @return 0:
*/
int agile_modbus_set_slave(agile_modbus_t* ctx, int slave) {
return ctx->backend->set_slave(ctx, slave);
}
/**
* @brief modbus
* @param ctx modbus
* @param cb
* @see agile_modbus_compute_meta_length_after_function
*/
void agile_modbus_set_compute_meta_length_after_function_cb(
agile_modbus_t* ctx,
uint8_t (*cb)(agile_modbus_t* ctx,
int function,
agile_modbus_msg_type_t msg_type)) {
ctx->compute_meta_length_after_function = cb;
}
/**
* @brief modbus
* @param ctx modbus
* @param cb
* @see agile_modbus_compute_data_length_after_meta
*/
void agile_modbus_set_compute_data_length_after_meta_cb(
agile_modbus_t* ctx,
int (*cb)(agile_modbus_t* ctx,
uint8_t* msg,
int msg_length,
agile_modbus_msg_type_t msg_type)) {
ctx->compute_data_length_after_meta = cb;
}
/**
* @brief
* @note API modbus 8 modbus + 2
* 8
* @param ctx modbus
* @param msg_length
* @param msg_type
* @return >0:modbus ; :
*/
int agile_modbus_receive_judge(agile_modbus_t* ctx,
int msg_length,
agile_modbus_msg_type_t msg_type) {
if ((msg_length <= 0) || (msg_length > ctx->read_bufsz))
return -1;
int rc = agile_modbus_receive_msg_judge(ctx, ctx->read_buf, msg_length,
msg_type);
return rc;
}
/**
* @}
*/
/** @defgroup Modbus_Master Modbus Master
* @{
*/
/** @defgroup Master_Private_Functions Master Private Functions
* @{
*/
/**
* @brief
* @note AGILE_MODBUS_MSG_LENGTH_UNDEFINED
* agile_modbus_check_confirmation API
* AGILE_MODBUS_MSG_LENGTH_UNDEFINED
* @param ctx modbus
* @param req
* @return
*/
static int agile_modbus_compute_response_length_from_request(
agile_modbus_t* ctx,
uint8_t* req) {
int length;
const int offset = ctx->backend->header_length;
switch (req[offset]) {
case AGILE_MODBUS_FC_READ_COILS:
case AGILE_MODBUS_FC_READ_DISCRETE_INPUTS: {
/* Header + nb values (code from write_bits) */
int nb = (req[offset + 3] << 8) | req[offset + 4];
length = 2 + (nb / 8) + ((nb % 8) ? 1 : 0);
} break;
case AGILE_MODBUS_FC_WRITE_AND_READ_REGISTERS:
case AGILE_MODBUS_FC_READ_HOLDING_REGISTERS:
case AGILE_MODBUS_FC_READ_INPUT_REGISTERS:
/* Header + 2 * nb values */
length = 2 + 2 * (req[offset + 3] << 8 | req[offset + 4]);
break;
case AGILE_MODBUS_FC_WRITE_SINGLE_COIL:
case AGILE_MODBUS_FC_WRITE_SINGLE_REGISTER:
case AGILE_MODBUS_FC_WRITE_MULTIPLE_COILS:
case AGILE_MODBUS_FC_WRITE_MULTIPLE_REGISTERS:
length = 5;
break;
case AGILE_MODBUS_FC_MASK_WRITE_REGISTER:
length = 7;
break;
default:
/* The response is device specific (the header provides the
length) */
return AGILE_MODBUS_MSG_LENGTH_UNDEFINED;
}
return offset + length + ctx->backend->checksum_length;
}
/**
* @brief
* @param ctx modbus
* @param req
* @param rsp
* @param rsp_length
* @return >=0:( 03 );
* : (-1 `-128 - $` )
*/
static int agile_modbus_check_confirmation(agile_modbus_t* ctx,
uint8_t* req,
uint8_t* rsp,
int rsp_length) {
int rc;
int rsp_length_computed;
const int offset = ctx->backend->header_length;
const int function = rsp[offset];
if (ctx->backend->pre_check_confirmation) {
rc = ctx->backend->pre_check_confirmation(ctx, req, rsp, rsp_length);
if (rc < 0)
return -1;
}
rsp_length_computed =
agile_modbus_compute_response_length_from_request(ctx, req);
/* Exception code */
if (function >= 0x80) {
if (rsp_length == (offset + 2 + (int)ctx->backend->checksum_length) &&
req[offset] == (rsp[offset] - 0x80))
return (-128 - rsp[offset + 1]);
else
return -1;
}
/* Check length */
if ((rsp_length == rsp_length_computed ||
rsp_length_computed == AGILE_MODBUS_MSG_LENGTH_UNDEFINED) &&
function < 0x80) {
int req_nb_value;
int rsp_nb_value;
/* Check function code */
if (function != req[offset])
return -1;
/* Check the number of values is corresponding to the request */
switch (function) {
case AGILE_MODBUS_FC_READ_COILS:
case AGILE_MODBUS_FC_READ_DISCRETE_INPUTS:
/* Read functions, 8 values in a byte (nb
* of values in the request and byte count in
* the response. */
req_nb_value = (req[offset + 3] << 8) + req[offset + 4];
req_nb_value =
(req_nb_value / 8) + ((req_nb_value % 8) ? 1 : 0);
rsp_nb_value = rsp[offset + 1];
break;
case AGILE_MODBUS_FC_WRITE_AND_READ_REGISTERS:
case AGILE_MODBUS_FC_READ_HOLDING_REGISTERS:
case AGILE_MODBUS_FC_READ_INPUT_REGISTERS:
/* Read functions 1 value = 2 bytes */
req_nb_value = (req[offset + 3] << 8) + req[offset + 4];
rsp_nb_value = (rsp[offset + 1] / 2);
break;
case AGILE_MODBUS_FC_WRITE_MULTIPLE_COILS:
case AGILE_MODBUS_FC_WRITE_MULTIPLE_REGISTERS:
/* N Write functions */
req_nb_value = (req[offset + 3] << 8) + req[offset + 4];
rsp_nb_value = (rsp[offset + 3] << 8) | rsp[offset + 4];
break;
case AGILE_MODBUS_FC_REPORT_SLAVE_ID:
/* Report slave ID (bytes received) */
req_nb_value = rsp_nb_value = rsp[offset + 1];
break;
default:
/* 1 Write functions & others */
req_nb_value = rsp_nb_value = 1;
}
if (req_nb_value == rsp_nb_value)
rc = rsp_nb_value;
else
rc = -1;
} else
rc = -1;
return rc;
}
/**
* @}
*/
/** @defgroup Master_Common_Operation_Functions Master Common Operation
Functions
* @brief modbus
@verbatim
API
- agile_modbus_serialize_xxx
:
>0:
:
- agile_modbus_deserialize_xxx
:
>=0:( 03 )
: (-1 `-128 - $` )
@endverbatim
* @{
*/
int agile_modbus_serialize_read_bits(agile_modbus_t* ctx, int addr, int nb) {
int min_req_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length;
if (ctx->send_bufsz < min_req_length)
return -1;
if (nb > AGILE_MODBUS_MAX_READ_BITS)
return -1;
int req_length = 0;
req_length = ctx->backend->build_request_basis(
ctx, AGILE_MODBUS_FC_READ_COILS, addr, nb, ctx->send_buf);
req_length = ctx->backend->send_msg_pre(ctx->send_buf, req_length);
return req_length;
}
int agile_modbus_deserialize_read_bits(agile_modbus_t* ctx,
int msg_length,
uint8_t* dest) {
int min_req_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length;
if (ctx->send_bufsz < min_req_length)
return -1;
if ((msg_length <= 0) || (msg_length > ctx->read_bufsz))
return -1;
int rc = agile_modbus_receive_msg_judge(ctx, ctx->read_buf, msg_length,
AGILE_MODBUS_MSG_CONFIRMATION);
if (rc < 0)
return -1;
rc = agile_modbus_check_confirmation(ctx, ctx->send_buf, ctx->read_buf, rc);
if (rc < 0)
return rc;
int i, temp, bit;
int pos = 0;
int offset;
int offset_end;
int nb;
offset = ctx->backend->header_length + 2;
offset_end = offset + rc;
nb = (ctx->send_buf[ctx->backend->header_length + 3] << 8) +
ctx->send_buf[ctx->backend->header_length + 4];
for (i = offset; i < offset_end; i++) {
/* Shift reg hi_byte to temp */
temp = ctx->read_buf[i];
for (bit = 0x01; (bit & 0xff) && (pos < nb);) {
dest[pos++] = (temp & bit) ? 1 : 0;
bit = bit << 1;
}
}
return nb;
}
int agile_modbus_serialize_read_input_bits(agile_modbus_t* ctx,
int addr,
int nb) {
int min_req_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length;
if (ctx->send_bufsz < min_req_length)
return -1;
if (nb > AGILE_MODBUS_MAX_READ_BITS)
return -1;
int req_length = 0;
req_length = ctx->backend->build_request_basis(
ctx, AGILE_MODBUS_FC_READ_DISCRETE_INPUTS, addr, nb, ctx->send_buf);
req_length = ctx->backend->send_msg_pre(ctx->send_buf, req_length);
return req_length;
}
int agile_modbus_deserialize_read_input_bits(agile_modbus_t* ctx,
int msg_length,
uint8_t* dest) {
int min_req_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length;
if (ctx->send_bufsz < min_req_length)
return -1;
if ((msg_length <= 0) || (msg_length > ctx->read_bufsz))
return -1;
int rc = agile_modbus_receive_msg_judge(ctx, ctx->read_buf, msg_length,
AGILE_MODBUS_MSG_CONFIRMATION);
if (rc < 0)
return -1;
rc = agile_modbus_check_confirmation(ctx, ctx->send_buf, ctx->read_buf, rc);
if (rc < 0)
return rc;
int i, temp, bit;
int pos = 0;
int offset;
int offset_end;
int nb;
offset = ctx->backend->header_length + 2;
offset_end = offset + rc;
nb = (ctx->send_buf[ctx->backend->header_length + 3] << 8) +
ctx->send_buf[ctx->backend->header_length + 4];
for (i = offset; i < offset_end; i++) {
/* Shift reg hi_byte to temp */
temp = ctx->read_buf[i];
for (bit = 0x01; (bit & 0xff) && (pos < nb);) {
dest[pos++] = (temp & bit) ? 1 : 0;
bit = bit << 1;
}
}
return nb;
}
int agile_modbus_serialize_read_registers(agile_modbus_t* ctx,
int addr,
int nb) {
int min_req_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length;
if (ctx->send_bufsz < min_req_length)
return -1;
if (nb > AGILE_MODBUS_MAX_READ_REGISTERS)
return -1;
int req_length = 0;
req_length = ctx->backend->build_request_basis(
ctx, AGILE_MODBUS_FC_READ_HOLDING_REGISTERS, addr, nb, ctx->send_buf);
req_length = ctx->backend->send_msg_pre(ctx->send_buf, req_length);
return req_length;
}
int agile_modbus_deserialize_read_registers(agile_modbus_t* ctx,
int msg_length,
uint16_t* dest) {
int min_req_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length;
if (ctx->send_bufsz < min_req_length)
return -1;
if ((msg_length <= 0) || (msg_length > ctx->read_bufsz))
return -1;
int rc = agile_modbus_receive_msg_judge(ctx, ctx->read_buf, msg_length,
AGILE_MODBUS_MSG_CONFIRMATION);
if (rc < 0)
return -1;
rc = agile_modbus_check_confirmation(ctx, ctx->send_buf, ctx->read_buf, rc);
if (rc < 0)
return rc;
int offset;
int i;
offset = ctx->backend->header_length;
for (i = 0; i < rc; i++) {
/* shift reg hi_byte to temp OR with lo_byte */
dest[i] = (ctx->read_buf[offset + 2 + (i << 1)] << 8) |
ctx->read_buf[offset + 3 + (i << 1)];
}
return rc;
}
int agile_modbus_serialize_read_input_registers(agile_modbus_t* ctx,
int addr,
int nb) {
int min_req_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length;
if (ctx->send_bufsz < min_req_length)
return -1;
if (nb > AGILE_MODBUS_MAX_READ_REGISTERS)
return -1;
int req_length = 0;
req_length = ctx->backend->build_request_basis(
ctx, AGILE_MODBUS_FC_READ_INPUT_REGISTERS, addr, nb, ctx->send_buf);
req_length = ctx->backend->send_msg_pre(ctx->send_buf, req_length);
return req_length;
}
int agile_modbus_deserialize_read_input_registers(agile_modbus_t* ctx,
int msg_length,
uint16_t* dest) {
int min_req_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length;
if (ctx->send_bufsz < min_req_length)
return -1;
if ((msg_length <= 0) || (msg_length > ctx->read_bufsz))
return -1;
int rc = agile_modbus_receive_msg_judge(ctx, ctx->read_buf, msg_length,
AGILE_MODBUS_MSG_CONFIRMATION);
if (rc < 0)
return -1;
rc = agile_modbus_check_confirmation(ctx, ctx->send_buf, ctx->read_buf, rc);
if (rc < 0)
return rc;
int offset;
int i;
offset = ctx->backend->header_length;
for (i = 0; i < rc; i++) {
/* shift reg hi_byte to temp OR with lo_byte */
dest[i] = (ctx->read_buf[offset + 2 + (i << 1)] << 8) |
ctx->read_buf[offset + 3 + (i << 1)];
}
return rc;
}
int agile_modbus_serialize_write_bit(agile_modbus_t* ctx,
int addr,
int status) {
int min_req_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length;
if (ctx->send_bufsz < min_req_length)
return -1;
int req_length = 0;
req_length = ctx->backend->build_request_basis(
ctx, AGILE_MODBUS_FC_WRITE_SINGLE_COIL, addr, status ? 0xFF00 : 0,
ctx->send_buf);
req_length = ctx->backend->send_msg_pre(ctx->send_buf, req_length);
return req_length;
}
int agile_modbus_deserialize_write_bit(agile_modbus_t* ctx, int msg_length) {
int min_req_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length;
if (ctx->send_bufsz < min_req_length)
return -1;
if ((msg_length <= 0) || (msg_length > ctx->read_bufsz))
return -1;
int rc = agile_modbus_receive_msg_judge(ctx, ctx->read_buf, msg_length,
AGILE_MODBUS_MSG_CONFIRMATION);
if (rc < 0)
return -1;
rc = agile_modbus_check_confirmation(ctx, ctx->send_buf, ctx->read_buf, rc);
return rc;
}
int agile_modbus_serialize_write_register(agile_modbus_t* ctx,
int addr,
const uint16_t value) {
int min_req_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length;
if (ctx->send_bufsz < min_req_length)
return -1;
int req_length = 0;
req_length = ctx->backend->build_request_basis(
ctx, AGILE_MODBUS_FC_WRITE_SINGLE_REGISTER, addr, (int)value,
ctx->send_buf);
req_length = ctx->backend->send_msg_pre(ctx->send_buf, req_length);
return req_length;
}
int agile_modbus_deserialize_write_register(agile_modbus_t* ctx,
int msg_length) {
int min_req_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length;
if (ctx->send_bufsz < min_req_length)
return -1;
if ((msg_length <= 0) || (msg_length > ctx->read_bufsz))
return -1;
int rc = agile_modbus_receive_msg_judge(ctx, ctx->read_buf, msg_length,
AGILE_MODBUS_MSG_CONFIRMATION);
if (rc < 0)
return -1;
rc = agile_modbus_check_confirmation(ctx, ctx->send_buf, ctx->read_buf, rc);
return rc;
}
int agile_modbus_serialize_write_bits(agile_modbus_t* ctx,
int addr,
int nb,
const uint8_t* src) {
int min_req_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length;
if (ctx->send_bufsz < min_req_length)
return -1;
if (nb > AGILE_MODBUS_MAX_WRITE_BITS)
return -1;
int i;
int byte_count;
int req_length;
int bit_check = 0;
int pos = 0;
req_length = ctx->backend->build_request_basis(
ctx, AGILE_MODBUS_FC_WRITE_MULTIPLE_COILS, addr, nb, ctx->send_buf);
byte_count = (nb / 8) + ((nb % 8) ? 1 : 0);
min_req_length += (1 + byte_count);
if (ctx->send_bufsz < min_req_length)
return -1;
ctx->send_buf[req_length++] = byte_count;
for (i = 0; i < byte_count; i++) {
int bit;
bit = 0x01;
ctx->send_buf[req_length] = 0;
while ((bit & 0xFF) && (bit_check++ < nb)) {
if (src[pos++])
ctx->send_buf[req_length] |= bit;
else
ctx->send_buf[req_length] &= ~bit;
bit = bit << 1;
}
req_length++;
}
req_length = ctx->backend->send_msg_pre(ctx->send_buf, req_length);
return req_length;
}
int agile_modbus_deserialize_write_bits(agile_modbus_t* ctx, int msg_length) {
int min_req_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length;
if (ctx->send_bufsz < min_req_length)
return -1;
if ((msg_length <= 0) || (msg_length > ctx->read_bufsz))
return -1;
int rc = agile_modbus_receive_msg_judge(ctx, ctx->read_buf, msg_length,
AGILE_MODBUS_MSG_CONFIRMATION);
if (rc < 0)
return -1;
rc = agile_modbus_check_confirmation(ctx, ctx->send_buf, ctx->read_buf, rc);
return rc;
}
int agile_modbus_serialize_write_registers(agile_modbus_t* ctx,
int addr,
int nb,
const uint16_t* src) {
int min_req_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length;
if (ctx->send_bufsz < min_req_length)
return -1;
if (nb > AGILE_MODBUS_MAX_WRITE_REGISTERS)
return -1;
int i;
int req_length;
int byte_count;
req_length = ctx->backend->build_request_basis(
ctx, AGILE_MODBUS_FC_WRITE_MULTIPLE_REGISTERS, addr, nb, ctx->send_buf);
byte_count = nb * 2;
min_req_length += (1 + byte_count);
if (ctx->send_bufsz < min_req_length)
return -1;
ctx->send_buf[req_length++] = byte_count;
for (i = 0; i < nb; i++) {
ctx->send_buf[req_length++] = src[i] >> 8;
ctx->send_buf[req_length++] = src[i] & 0x00FF;
}
req_length = ctx->backend->send_msg_pre(ctx->send_buf, req_length);
return req_length;
}
int agile_modbus_deserialize_write_registers(agile_modbus_t* ctx,
int msg_length) {
int min_req_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length;
if (ctx->send_bufsz < min_req_length)
return -1;
if ((msg_length <= 0) || (msg_length > ctx->read_bufsz))
return -1;
int rc = agile_modbus_receive_msg_judge(ctx, ctx->read_buf, msg_length,
AGILE_MODBUS_MSG_CONFIRMATION);
if (rc < 0)
return -1;
rc = agile_modbus_check_confirmation(ctx, ctx->send_buf, ctx->read_buf, rc);
return rc;
}
int agile_modbus_serialize_mask_write_register(agile_modbus_t* ctx,
int addr,
uint16_t and_mask,
uint16_t or_mask) {
int min_req_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length + 2;
if (ctx->send_bufsz < min_req_length)
return -1;
int req_length = 0;
req_length = ctx->backend->build_request_basis(
ctx, AGILE_MODBUS_FC_MASK_WRITE_REGISTER, addr, 0, ctx->send_buf);
/* HACKISH, count is not used */
req_length -= 2;
ctx->send_buf[req_length++] = and_mask >> 8;
ctx->send_buf[req_length++] = and_mask & 0x00ff;
ctx->send_buf[req_length++] = or_mask >> 8;
ctx->send_buf[req_length++] = or_mask & 0x00ff;
req_length = ctx->backend->send_msg_pre(ctx->send_buf, req_length);
return req_length;
}
int agile_modbus_deserialize_mask_write_register(agile_modbus_t* ctx,
int msg_length) {
int min_req_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length;
if (ctx->send_bufsz < min_req_length)
return -1;
if ((msg_length <= 0) || (msg_length > ctx->read_bufsz))
return -1;
int rc = agile_modbus_receive_msg_judge(ctx, ctx->read_buf, msg_length,
AGILE_MODBUS_MSG_CONFIRMATION);
if (rc < 0)
return -1;
rc = agile_modbus_check_confirmation(ctx, ctx->send_buf, ctx->read_buf, rc);
return rc;
}
int agile_modbus_serialize_write_and_read_registers(agile_modbus_t* ctx,
int write_addr,
int write_nb,
const uint16_t* src,
int read_addr,
int read_nb) {
int min_req_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length;
if (ctx->send_bufsz < min_req_length)
return -1;
if (write_nb > AGILE_MODBUS_MAX_WR_WRITE_REGISTERS)
return -1;
if (read_nb > AGILE_MODBUS_MAX_WR_READ_REGISTERS)
return -1;
int req_length;
int i;
int byte_count;
req_length = ctx->backend->build_request_basis(
ctx, AGILE_MODBUS_FC_WRITE_AND_READ_REGISTERS, read_addr, read_nb,
ctx->send_buf);
byte_count = write_nb * 2;
min_req_length += (5 + byte_count);
if (ctx->send_bufsz < min_req_length)
return -1;
ctx->send_buf[req_length++] = write_addr >> 8;
ctx->send_buf[req_length++] = write_addr & 0x00ff;
ctx->send_buf[req_length++] = write_nb >> 8;
ctx->send_buf[req_length++] = write_nb & 0x00ff;
ctx->send_buf[req_length++] = byte_count;
for (i = 0; i < write_nb; i++) {
ctx->send_buf[req_length++] = src[i] >> 8;
ctx->send_buf[req_length++] = src[i] & 0x00FF;
}
req_length = ctx->backend->send_msg_pre(ctx->send_buf, req_length);
return req_length;
}
int agile_modbus_deserialize_write_and_read_registers(agile_modbus_t* ctx,
int msg_length,
uint16_t* dest) {
int min_req_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length;
if (ctx->send_bufsz < min_req_length)
return -1;
if ((msg_length <= 0) || (msg_length > ctx->read_bufsz))
return -1;
int rc = agile_modbus_receive_msg_judge(ctx, ctx->read_buf, msg_length,
AGILE_MODBUS_MSG_CONFIRMATION);
if (rc < 0)
return -1;
rc = agile_modbus_check_confirmation(ctx, ctx->send_buf, ctx->read_buf, rc);
if (rc < 0)
return rc;
int offset;
int i;
offset = ctx->backend->header_length;
for (i = 0; i < rc; i++) {
/* shift reg hi_byte to temp OR with lo_byte */
dest[i] = (ctx->read_buf[offset + 2 + (i << 1)] << 8) |
ctx->read_buf[offset + 3 + (i << 1)];
}
return rc;
}
int agile_modbus_serialize_report_slave_id(agile_modbus_t* ctx) {
int min_req_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length;
if (ctx->send_bufsz < min_req_length)
return -1;
int req_length = 0;
req_length = ctx->backend->build_request_basis(
ctx, AGILE_MODBUS_FC_REPORT_SLAVE_ID, 0, 0, ctx->send_buf);
/* HACKISH, addr and count are not used */
req_length -= 4;
req_length = ctx->backend->send_msg_pre(ctx->send_buf, req_length);
return req_length;
}
int agile_modbus_deserialize_report_slave_id(agile_modbus_t* ctx,
int msg_length,
int max_dest,
uint8_t* dest) {
int min_req_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length;
if (ctx->send_bufsz < min_req_length)
return -1;
if ((msg_length <= 0) || (msg_length > ctx->read_bufsz))
return -1;
if (max_dest <= 0)
return -1;
int rc = agile_modbus_receive_msg_judge(ctx, ctx->read_buf, msg_length,
AGILE_MODBUS_MSG_CONFIRMATION);
if (rc < 0)
return -1;
rc = agile_modbus_check_confirmation(ctx, ctx->send_buf, ctx->read_buf, rc);
if (rc < 0)
return rc;
int i;
int offset;
offset = ctx->backend->header_length + 2;
/* Byte count, slave id, run indicator status and
additional data. Truncate copy to max_dest. */
for (i = 0; i < rc && i < max_dest; i++) {
dest[i] = ctx->read_buf[offset + i];
}
return rc;
}
/**
* @}
*/
/** @defgroup Master_Raw_Operation_Functions Master Raw Operation Functions
* @{
*/
/**
* @brief
* @param ctx modbus
* @param raw_req (PDU + Slave address)
* @param raw_req_length
* @return >0:; :
*/
int agile_modbus_serialize_raw_request(agile_modbus_t* ctx,
const uint8_t* raw_req,
int raw_req_length) {
if (raw_req_length < 2) {
/* The raw request must contain function and slave at least and
must not be longer than the maximum pdu length plus the slave
address. */
return -1;
}
int min_req_length = ctx->backend->header_length + 1 +
ctx->backend->checksum_length + raw_req_length - 2;
if (ctx->send_bufsz < min_req_length)
return -1;
agile_modbus_sft_t sft;
int req_length;
sft.slave = raw_req[0];
sft.function = raw_req[1];
/* The t_id is left to zero */
sft.t_id = 0;
/* This response function only set the header so it's convenient here */
req_length = ctx->backend->build_response_basis(&sft, ctx->send_buf);
if (raw_req_length > 2) {
/* Copy data after function code */
memcpy(ctx->send_buf + req_length, raw_req + 2, raw_req_length - 2);
req_length += raw_req_length - 2;
}
req_length = ctx->backend->send_msg_pre(ctx->send_buf, req_length);
return req_length;
}
/**
* @brief
* @param ctx modbus
* @param msg_length
* @return >=0:( 03 );
* : (-1 `-128 - $` )
*/
int agile_modbus_deserialize_raw_response(agile_modbus_t* ctx, int msg_length) {
int min_req_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length;
if (ctx->send_bufsz < min_req_length)
return -1;
if ((msg_length <= 0) || (msg_length > ctx->read_bufsz))
return -1;
int rc = agile_modbus_receive_msg_judge(ctx, ctx->read_buf, msg_length,
AGILE_MODBUS_MSG_CONFIRMATION);
if (rc < 0)
return -1;
rc = agile_modbus_check_confirmation(ctx, ctx->send_buf, ctx->read_buf, rc);
return rc;
}
/**
* @}
*/
/**
* @}
*/
/** @defgroup Modbus_Slave Modbus Slave
* @{
*/
/** @defgroup Slave_Private_Functions Slave Private Functions
* @{
*/
/**
* @brief
* @param ctx modbus
* @param sft modbus
* @param exception_code
* @return
*/
static int agile_modbus_serialize_response_exception(agile_modbus_t* ctx,
agile_modbus_sft_t* sft,
int exception_code) {
int rsp_length;
/* Build exception response */
sft->function = sft->function + 0x80;
rsp_length = ctx->backend->build_response_basis(sft, ctx->send_buf);
ctx->send_buf[rsp_length++] = exception_code;
return rsp_length;
}
/**
* @}
*/
/** @defgroup Slave_Operation_Functions Slave Operation Functions
* @{
*/
/**
* @brief IO
* @param buf IO
* @param index IO ( IO)
* @param status IO
*/
void agile_modbus_slave_io_set(uint8_t* buf, int index, int status) {
int offset = index / 8;
int shift = index % 8;
if (status)
buf[offset] |= (0x01 << shift);
else
buf[offset] &= ~(0x01 << shift);
}
/**
* @brief IO
* @param buf IO
* @param index IO ( IO)
* @return IO (1/0)
*/
uint8_t agile_modbus_slave_io_get(uint8_t* buf, int index) {
int offset = index / 8;
int shift = index % 8;
uint8_t status = (buf[offset] & (0x01 << shift)) ? 1 : 0;
return status;
}
/**
* @brief
* @param buf
* @param index ()
* @param data
*/
void agile_modbus_slave_register_set(uint8_t* buf, int index, uint16_t data) {
buf[index * 2] = data >> 8;
buf[index * 2 + 1] = data & 0xFF;
}
/**
* @brief
* @param buf
* @param index ()
* @return
*/
uint16_t agile_modbus_slave_register_get(uint8_t* buf, int index) {
uint16_t data = (buf[index * 2] << 8) + buf[index * 2 + 1];
return data;
}
/**
* @brief
* @param ctx modbus
* @param msg_length
* @param slave_strict
* @arg 0:
* @arg 1:
* @param slave_cb
* @param slave_data
* @param frame_length modbus
* @return >=0:; :
*/
int agile_modbus_slave_handle(agile_modbus_t* ctx,
int msg_length,
uint8_t slave_strict,
agile_modbus_slave_callback_t slave_cb,
const void* slave_data,
int* frame_length) {
int min_rsp_length =
ctx->backend->header_length + 5 + ctx->backend->checksum_length;
if (ctx->send_bufsz < min_rsp_length)
return -1;
int req_length = agile_modbus_receive_judge(ctx, msg_length,
AGILE_MODBUS_MSG_INDICATION);
if (req_length < 0)
return -1;
if (frame_length)
*frame_length = req_length;
int offset;
int slave;
int function;
uint16_t address;
int rsp_length = 0;
int exception_code = 0;
agile_modbus_sft_t sft;
uint8_t* req = ctx->read_buf;
uint8_t* rsp = ctx->send_buf;
memset(rsp, 0, ctx->send_bufsz);
offset = ctx->backend->header_length;
slave = req[offset - 1];
function = req[offset];
address = (req[offset + 1] << 8) + req[offset + 2];
sft.slave = slave;
sft.function = function;
sft.t_id = ctx->backend->prepare_response_tid(req, &req_length);
struct agile_modbus_slave_info slave_info = {0};
slave_info.sft = &sft;
slave_info.rsp_length = &rsp_length;
slave_info.address = address;
if (slave_strict) {
if ((slave != ctx->slave) && (slave != AGILE_MODBUS_BROADCAST_ADDRESS))
return 0;
}
switch (function) {
case AGILE_MODBUS_FC_READ_COILS:
case AGILE_MODBUS_FC_READ_DISCRETE_INPUTS: {
int nb = (req[offset + 3] << 8) + req[offset + 4];
if (nb < 1 || AGILE_MODBUS_MAX_READ_BITS < nb) {
exception_code = AGILE_MODBUS_EXCEPTION_ILLEGAL_DATA_VALUE;
break;
}
int end_address = (int)address + nb - 1;
if (end_address > 0xFFFF) {
exception_code = AGILE_MODBUS_EXCEPTION_ILLEGAL_DATA_ADDRESS;
break;
}
rsp_length = ctx->backend->build_response_basis(&sft, rsp);
slave_info.nb = (nb / 8) + ((nb % 8) ? 1 : 0);
rsp[rsp_length++] = slave_info.nb;
slave_info.send_index = rsp_length;
rsp_length += slave_info.nb;
slave_info.nb = nb;
if (ctx->send_bufsz <
(int)(rsp_length + ctx->backend->checksum_length)) {
exception_code = AGILE_MODBUS_EXCEPTION_NEGATIVE_ACKNOWLEDGE;
break;
}
} break;
case AGILE_MODBUS_FC_READ_HOLDING_REGISTERS:
case AGILE_MODBUS_FC_READ_INPUT_REGISTERS: {
int nb = (req[offset + 3] << 8) + req[offset + 4];
if (nb < 1 || AGILE_MODBUS_MAX_READ_REGISTERS < nb) {
exception_code = AGILE_MODBUS_EXCEPTION_ILLEGAL_DATA_VALUE;
break;
}
int end_address = (int)address + nb - 1;
if (end_address > 0xFFFF) {
exception_code = AGILE_MODBUS_EXCEPTION_ILLEGAL_DATA_ADDRESS;
break;
}
rsp_length = ctx->backend->build_response_basis(&sft, rsp);
slave_info.nb = nb << 1;
rsp[rsp_length++] = slave_info.nb;
slave_info.send_index = rsp_length;
rsp_length += slave_info.nb;
slave_info.nb = nb;
if (ctx->send_bufsz <
(int)(rsp_length + ctx->backend->checksum_length)) {
exception_code = AGILE_MODBUS_EXCEPTION_NEGATIVE_ACKNOWLEDGE;
break;
}
} break;
case AGILE_MODBUS_FC_WRITE_SINGLE_COIL: {
//! warning: comparison is always false due to limited range of data type
//! [-Wtype-limits]
#if 0
if (address > 0xFFFF) {
exception_code = AGILE_MODBUS_EXCEPTION_ILLEGAL_DATA_ADDRESS;
break;
}
#endif
int data = (req[offset + 3] << 8) + req[offset + 4];
if (data == 0xFF00 || data == 0x0)
data = data ? 1 : 0;
else {
exception_code = AGILE_MODBUS_EXCEPTION_ILLEGAL_DATA_VALUE;
break;
}
slave_info.buf = (uint8_t*)&data;
rsp_length = req_length;
if (ctx->send_bufsz <
(int)(rsp_length + ctx->backend->checksum_length)) {
exception_code = AGILE_MODBUS_EXCEPTION_NEGATIVE_ACKNOWLEDGE;
break;
}
memcpy(rsp, req, req_length);
} break;
case AGILE_MODBUS_FC_WRITE_SINGLE_REGISTER: {
//! warning: comparison is always false due to limited range of data type
//! [-Wtype-limits]
#if 0
if (address > 0xFFFF) {
exception_code = AGILE_MODBUS_EXCEPTION_ILLEGAL_DATA_ADDRESS;
break;
}
#endif
int data = (req[offset + 3] << 8) + req[offset + 4];
slave_info.buf = (uint8_t*)&data;
rsp_length = req_length;
if (ctx->send_bufsz <
(int)(rsp_length + ctx->backend->checksum_length)) {
exception_code = AGILE_MODBUS_EXCEPTION_NEGATIVE_ACKNOWLEDGE;
break;
}
memcpy(rsp, req, req_length);
} break;
case AGILE_MODBUS_FC_WRITE_MULTIPLE_COILS: {
int nb = (req[offset + 3] << 8) + req[offset + 4];
int nb_bits = req[offset + 5];
if (nb < 1 || AGILE_MODBUS_MAX_WRITE_BITS < nb ||
nb_bits * 8 < nb) {
exception_code = AGILE_MODBUS_EXCEPTION_ILLEGAL_DATA_VALUE;
break;
}
int end_address = (int)address + nb - 1;
if (end_address > 0xFFFF) {
exception_code = AGILE_MODBUS_EXCEPTION_ILLEGAL_DATA_ADDRESS;
break;
}
rsp_length = ctx->backend->build_response_basis(&sft, rsp);
slave_info.nb = nb;
slave_info.buf = &req[offset + 6];
if (ctx->send_bufsz <
(int)(rsp_length + ctx->backend->checksum_length + 4)) {
exception_code = AGILE_MODBUS_EXCEPTION_NEGATIVE_ACKNOWLEDGE;
break;
}
/* 4 to copy the bit address (2) and the quantity of bits */
memcpy(rsp + rsp_length, req + rsp_length, 4);
rsp_length += 4;
} break;
case AGILE_MODBUS_FC_WRITE_MULTIPLE_REGISTERS: {
int nb = (req[offset + 3] << 8) + req[offset + 4];
int nb_bytes = req[offset + 5];
if (nb < 1 || AGILE_MODBUS_MAX_WRITE_REGISTERS < nb ||
nb_bytes != nb * 2) {
exception_code = AGILE_MODBUS_EXCEPTION_ILLEGAL_DATA_VALUE;
break;
}
int end_address = (int)address + nb - 1;
if (end_address > 0xFFFF) {
exception_code = AGILE_MODBUS_EXCEPTION_ILLEGAL_DATA_ADDRESS;
break;
}
rsp_length = ctx->backend->build_response_basis(&sft, rsp);
slave_info.nb = nb;
slave_info.buf = &req[offset + 6];
if (ctx->send_bufsz <
(int)(rsp_length + ctx->backend->checksum_length + 4)) {
exception_code = AGILE_MODBUS_EXCEPTION_NEGATIVE_ACKNOWLEDGE;
break;
}
/* 4 to copy the address (2) and the no. of registers */
memcpy(rsp + rsp_length, req + rsp_length, 4);
rsp_length += 4;
} break;
case AGILE_MODBUS_FC_REPORT_SLAVE_ID: {
int str_len;
int byte_count_pos;
slave_cb = NULL;
rsp_length = ctx->backend->build_response_basis(&sft, rsp);
/* Skip byte count for now */
byte_count_pos = rsp_length++;
rsp[rsp_length++] = ctx->slave;
/* Run indicator status to ON */
rsp[rsp_length++] = 0xFF;
str_len = strlen(AGILE_MODBUS_VERSION_STRING);
if (ctx->send_bufsz <
(int)(rsp_length + ctx->backend->checksum_length + str_len)) {
exception_code = AGILE_MODBUS_EXCEPTION_NEGATIVE_ACKNOWLEDGE;
break;
}
memcpy(rsp + rsp_length, AGILE_MODBUS_VERSION_STRING, str_len);
rsp_length += str_len;
rsp[byte_count_pos] = rsp_length - byte_count_pos - 1;
} break;
case AGILE_MODBUS_FC_READ_EXCEPTION_STATUS:
exception_code = AGILE_MODBUS_EXCEPTION_ILLEGAL_FUNCTION;
break;
case AGILE_MODBUS_FC_MASK_WRITE_REGISTER: {
//! warning: comparison is always false due to limited range of data type
//! [-Wtype-limits]
#if 0
if (address > 0xFFFF) {
exception_code = AGILE_MODBUS_EXCEPTION_ILLEGAL_DATA_ADDRESS;
break;
}
#endif
slave_info.buf = &req[offset + 3];
rsp_length = req_length;
if (ctx->send_bufsz <
(int)(rsp_length + ctx->backend->checksum_length)) {
exception_code = AGILE_MODBUS_EXCEPTION_NEGATIVE_ACKNOWLEDGE;
break;
}
memcpy(rsp, req, req_length);
} break;
case AGILE_MODBUS_FC_WRITE_AND_READ_REGISTERS: {
int nb = (req[offset + 3] << 8) + req[offset + 4];
uint16_t address_write = (req[offset + 5] << 8) + req[offset + 6];
int nb_write = (req[offset + 7] << 8) + req[offset + 8];
int nb_write_bytes = req[offset + 9];
if (nb_write < 1 ||
AGILE_MODBUS_MAX_WR_WRITE_REGISTERS < nb_write || nb < 1 ||
AGILE_MODBUS_MAX_WR_READ_REGISTERS < nb ||
nb_write_bytes != nb_write * 2) {
exception_code = AGILE_MODBUS_EXCEPTION_ILLEGAL_DATA_VALUE;
break;
}
int end_address = (int)address + nb - 1;
int end_address_write = (int)address_write + nb_write - 1;
if (end_address > 0xFFFF || end_address_write > 0xFFFF) {
exception_code = AGILE_MODBUS_EXCEPTION_ILLEGAL_DATA_ADDRESS;
break;
}
rsp_length = ctx->backend->build_response_basis(&sft, rsp);
rsp[rsp_length++] = nb << 1;
slave_info.buf = &req[offset + 3];
slave_info.send_index = rsp_length;
rsp_length += (nb << 1);
if (ctx->send_bufsz <
(int)(rsp_length + ctx->backend->checksum_length)) {
exception_code = AGILE_MODBUS_EXCEPTION_NEGATIVE_ACKNOWLEDGE;
break;
}
} break;
default: {
if (slave_cb == NULL)
exception_code = AGILE_MODBUS_EXCEPTION_ILLEGAL_FUNCTION;
else {
rsp_length = ctx->backend->build_response_basis(&sft, rsp);
slave_info.send_index = rsp_length;
slave_info.buf = &req[offset + 1];
slave_info.nb = req_length - offset - 1;
}
} break;
}
if (exception_code)
rsp_length = agile_modbus_serialize_response_exception(ctx, &sft,
exception_code);
else {
if (slave_cb) {
int ret = slave_cb(ctx, &slave_info, slave_data);
if (ret < 0) {
if (ret == -AGILE_MODBUS_EXCEPTION_UNKNOW)
rsp_length = 0;
else
rsp_length = agile_modbus_serialize_response_exception(
ctx, &sft, -ret);
}
}
}
if (rsp_length) {
if ((ctx->backend->backend_type == AGILE_MODBUS_BACKEND_TYPE_RTU) &&
(slave == AGILE_MODBUS_BROADCAST_ADDRESS))
return 0;
rsp_length = ctx->backend->send_msg_pre(rsp, rsp_length);
}
return rsp_length;
}
/**
* @}
*/
/**
* @}
*/
/**
* @}
*/