mirror of
https://gitee.com/Lyon1998/pikapython.git
synced 2025-01-22 17:12:55 +08:00
429 lines
14 KiB
C
429 lines
14 KiB
C
/**
|
|
* @file agile_modbus.h
|
|
* @brief Agile Modbus 软件包通用头文件
|
|
* @author 马龙伟 (2544047213@qq.com)
|
|
* @date 2022-07-28
|
|
*
|
|
* @attention
|
|
*
|
|
* <h2><center>© Copyright (c) 2021 Ma Longwei.
|
|
* All rights reserved.</center></h2>
|
|
*
|
|
*/
|
|
|
|
#ifndef __PKG_AGILE_MODBUS_H
|
|
#define __PKG_AGILE_MODBUS_H
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
#include <stdint.h>
|
|
|
|
/** @addtogroup COMMON
|
|
* @{
|
|
*/
|
|
|
|
/** @defgroup COMMON_Exported_Constants Common Exported Constants
|
|
* @{
|
|
*/
|
|
|
|
/** @defgroup Modbus_Function_Codes Modbus Function Codes
|
|
* @{
|
|
*/
|
|
#define AGILE_MODBUS_FC_READ_COILS 0x01
|
|
#define AGILE_MODBUS_FC_READ_DISCRETE_INPUTS 0x02
|
|
#define AGILE_MODBUS_FC_READ_HOLDING_REGISTERS 0x03
|
|
#define AGILE_MODBUS_FC_READ_INPUT_REGISTERS 0x04
|
|
#define AGILE_MODBUS_FC_WRITE_SINGLE_COIL 0x05
|
|
#define AGILE_MODBUS_FC_WRITE_SINGLE_REGISTER 0x06
|
|
#define AGILE_MODBUS_FC_READ_EXCEPTION_STATUS 0x07
|
|
#define AGILE_MODBUS_FC_WRITE_MULTIPLE_COILS 0x0F
|
|
#define AGILE_MODBUS_FC_WRITE_MULTIPLE_REGISTERS 0x10
|
|
#define AGILE_MODBUS_FC_REPORT_SLAVE_ID 0x11
|
|
#define AGILE_MODBUS_FC_MASK_WRITE_REGISTER 0x16
|
|
#define AGILE_MODBUS_FC_WRITE_AND_READ_REGISTERS 0x17
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/** @defgroup Modbus_Constants Modbus Constants
|
|
* @{
|
|
*/
|
|
#define AGILE_MODBUS_VERSION_STRING "AMB_1.1.0" /**< Agile Modbus 版本号 */
|
|
|
|
#define AGILE_MODBUS_BROADCAST_ADDRESS 0 /**< Modbus 广播地址 */
|
|
|
|
/** @name Quantity limit of Coils
|
|
@verbatim
|
|
Modbus_Application_Protocol_V1_1b.pdf (chapter 6 section 1 page 12)
|
|
Quantity of Coils to read (2 bytes): 1 to 2000 (0x7D0)
|
|
(chapter 6 section 11 page 29)
|
|
Quantity of Coils to write (2 bytes): 1 to 1968 (0x7B0)
|
|
|
|
@endverbatim
|
|
* @{
|
|
*/
|
|
#define AGILE_MODBUS_MAX_READ_BITS 2000
|
|
#define AGILE_MODBUS_MAX_WRITE_BITS 1968
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/** @name Quantity limit of Registers
|
|
@verbatim
|
|
Modbus_Application_Protocol_V1_1b.pdf (chapter 6 section 3 page 15)
|
|
Quantity of Registers to read (2 bytes): 1 to 125 (0x7D)
|
|
(chapter 6 section 12 page 31)
|
|
Quantity of Registers to write (2 bytes) 1 to 123 (0x7B)
|
|
(chapter 6 section 17 page 38)
|
|
Quantity of Registers to write in R/W registers (2 bytes) 1 to 121 (0x79)
|
|
|
|
@endverbatim
|
|
* @{
|
|
*/
|
|
#define AGILE_MODBUS_MAX_READ_REGISTERS 125
|
|
#define AGILE_MODBUS_MAX_WRITE_REGISTERS 123
|
|
#define AGILE_MODBUS_MAX_WR_WRITE_REGISTERS 121
|
|
#define AGILE_MODBUS_MAX_WR_READ_REGISTERS 125
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/**
|
|
@verbatim
|
|
The size of the MODBUS PDU is limited by the size constraint inherited from
|
|
the first MODBUS implementation on Serial Line network (max. RS485 ADU = 256
|
|
bytes). Therefore, MODBUS PDU for serial line communication = 256 - Server
|
|
address (1 byte) - CRC (2 bytes) = 253 bytes.
|
|
|
|
@endverbatim
|
|
*/
|
|
#define AGILE_MODBUS_MAX_PDU_LENGTH 253
|
|
|
|
/**
|
|
@verbatim
|
|
Consequently:
|
|
- RTU MODBUS ADU = 253 bytes + Server address (1 byte) + CRC (2 bytes) = 256
|
|
bytes.
|
|
- TCP MODBUS ADU = 253 bytes + MBAP (7 bytes) = 260 bytes.
|
|
so the maximum of both backend in 260 bytes. This size can used to allocate
|
|
an array of bytes to store responses and it will be compatible with the two
|
|
backends.
|
|
|
|
@endverbatim
|
|
*/
|
|
#define AGILE_MODBUS_MAX_ADU_LENGTH 260
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/** @defgroup COMMON_Exported_Types Common Exported Types
|
|
* @{
|
|
*/
|
|
|
|
/**
|
|
* @brief Modbus 异常码
|
|
*/
|
|
enum {
|
|
AGILE_MODBUS_EXCEPTION_ILLEGAL_FUNCTION = 0x01,
|
|
AGILE_MODBUS_EXCEPTION_ILLEGAL_DATA_ADDRESS,
|
|
AGILE_MODBUS_EXCEPTION_ILLEGAL_DATA_VALUE,
|
|
AGILE_MODBUS_EXCEPTION_SLAVE_OR_SERVER_FAILURE,
|
|
AGILE_MODBUS_EXCEPTION_ACKNOWLEDGE,
|
|
AGILE_MODBUS_EXCEPTION_SLAVE_OR_SERVER_BUSY,
|
|
AGILE_MODBUS_EXCEPTION_NEGATIVE_ACKNOWLEDGE,
|
|
AGILE_MODBUS_EXCEPTION_MEMORY_PARITY,
|
|
AGILE_MODBUS_EXCEPTION_NOT_DEFINED,
|
|
AGILE_MODBUS_EXCEPTION_GATEWAY_PATH,
|
|
AGILE_MODBUS_EXCEPTION_GATEWAY_TARGET,
|
|
AGILE_MODBUS_EXCEPTION_UNKNOW = 0xff
|
|
};
|
|
|
|
/**
|
|
* @brief Modbus 后端类型
|
|
*/
|
|
typedef enum {
|
|
AGILE_MODBUS_BACKEND_TYPE_RTU = 0, /**< RTU */
|
|
AGILE_MODBUS_BACKEND_TYPE_TCP /**< TCP */
|
|
} agile_modbus_backend_type_t;
|
|
|
|
/**
|
|
* @brief Modbus 收到消息类型
|
|
*
|
|
@verbatim
|
|
---------- Request Indication ----------
|
|
| Client | ---------------------->| Server |
|
|
---------- Confirmation Response ----------
|
|
|
|
@endverbatim
|
|
*/
|
|
typedef enum {
|
|
AGILE_MODBUS_MSG_INDICATION, /**< 主机端的请求消息 */
|
|
AGILE_MODBUS_MSG_CONFIRMATION /**< 服务器端的请求消息 */
|
|
} agile_modbus_msg_type_t;
|
|
|
|
/**
|
|
* @brief 包含 modbus 头部参数结构体
|
|
*/
|
|
typedef struct agile_modbus_sft {
|
|
int slave; /**< 从机地址 */
|
|
int function; /**< 功能码 */
|
|
int t_id; /**< 事务标识符 */
|
|
} agile_modbus_sft_t;
|
|
|
|
typedef struct agile_modbus agile_modbus_t; /**< Agile Modbus 结构体 */
|
|
|
|
/**
|
|
* @brief Agile Modbus 后端接口结构体
|
|
*/
|
|
typedef struct agile_modbus_backend {
|
|
uint32_t backend_type; /**< 后端类型 */
|
|
uint32_t header_length; /**< 头部长度,不包含功能码 */
|
|
uint32_t checksum_length; /**< 校验数据长度 */
|
|
uint32_t max_adu_length; /**< 后端 ADU 长度 */
|
|
int (*set_slave)(agile_modbus_t* ctx, int slave); /**< 设置地址接口 */
|
|
int (*build_request_basis)(agile_modbus_t* ctx,
|
|
int function,
|
|
int addr,
|
|
int nb,
|
|
uint8_t* req); /**< 构建基础请求报文接口 */
|
|
int (*build_response_basis)(agile_modbus_sft_t* sft,
|
|
uint8_t* rsp); /**< 构建基础响应报文接口 */
|
|
int (*prepare_response_tid)(const uint8_t* req,
|
|
int* req_length); /**< 准备响应接口 */
|
|
int (*send_msg_pre)(uint8_t* req, int req_length); /**< 预发送数据接口 */
|
|
int (*check_integrity)(agile_modbus_t* ctx,
|
|
uint8_t* msg,
|
|
const int msg_length); /**< 检查接收数据完整性接口 */
|
|
int (*pre_check_confirmation)(agile_modbus_t* ctx,
|
|
const uint8_t* req,
|
|
const uint8_t* rsp,
|
|
int rsp_length); /**< 预检查确认接口 */
|
|
} agile_modbus_backend_t;
|
|
|
|
/**
|
|
* @brief Agile Modbus 结构体
|
|
*/
|
|
struct agile_modbus {
|
|
int slave; /**< 从机地址 */
|
|
uint8_t* send_buf; /**< 发送缓冲区 */
|
|
int send_bufsz; /**< 发送缓冲区大小 */
|
|
uint8_t* read_buf; /**< 接收缓冲区 */
|
|
int read_bufsz; /**< 接收缓冲区大小 */
|
|
uint8_t (*compute_meta_length_after_function)(
|
|
agile_modbus_t* ctx,
|
|
int function,
|
|
agile_modbus_msg_type_t msg_type); /**< 自定义计算数据元长度接口 */
|
|
int (*compute_data_length_after_meta)(
|
|
agile_modbus_t* ctx,
|
|
uint8_t* msg,
|
|
int msg_length,
|
|
agile_modbus_msg_type_t msg_type); /**< 自定义计算数据长度接口 */
|
|
const agile_modbus_backend_t* backend; /**< 后端接口 */
|
|
void* backend_data; /**< 后端数据,指向 RTU 或 TCP 结构体 */
|
|
};
|
|
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/** @addtogroup Modbus_Slave
|
|
* @{
|
|
*/
|
|
|
|
/** @defgroup Slave_Exported_Types Slave Exported Types
|
|
* @{
|
|
*/
|
|
|
|
/**
|
|
* @brief Agile Modbus 从机信息结构体
|
|
*/
|
|
struct agile_modbus_slave_info {
|
|
agile_modbus_sft_t* sft; /**< sft 结构体指针 */
|
|
int* rsp_length; /**< 响应数据长度指针 */
|
|
int address; /**< 寄存器地址 */
|
|
int nb; /**< 数目 */
|
|
uint8_t* buf; /**< 不同功能码需要使用的数据域 */
|
|
int send_index; /**< 发送缓冲区当前索引 */
|
|
};
|
|
|
|
/**
|
|
* @brief 从机回调函数
|
|
* @param ctx modbus 句柄
|
|
* @param slave_info 从机信息体
|
|
* @param data 私有数据
|
|
* @return =0:正常;
|
|
* <0:异常
|
|
* (-AGILE_MODBUS_EXCEPTION_UNKNOW(-255):
|
|
* 未知异常,从机不会打包响应数据) (其他负数异常码: 从机会打包异常响应数据)
|
|
*/
|
|
typedef int (*agile_modbus_slave_callback_t)(
|
|
agile_modbus_t* ctx,
|
|
struct agile_modbus_slave_info* slave_info,
|
|
const void* data);
|
|
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/** @addtogroup COMMON_Exported_Functions
|
|
* @{
|
|
*/
|
|
void agile_modbus_common_init(agile_modbus_t* ctx,
|
|
uint8_t* send_buf,
|
|
int send_bufsz,
|
|
uint8_t* read_buf,
|
|
int read_bufsz);
|
|
int agile_modbus_set_slave(agile_modbus_t* ctx, int slave);
|
|
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));
|
|
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));
|
|
int agile_modbus_receive_judge(agile_modbus_t* ctx,
|
|
int msg_length,
|
|
agile_modbus_msg_type_t msg_type);
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/** @addtogroup Modbus_Master
|
|
* @{
|
|
*/
|
|
|
|
/** @addtogroup Master_Common_Operation_Functions
|
|
* @{
|
|
*/
|
|
int agile_modbus_serialize_read_bits(agile_modbus_t* ctx, int addr, int nb);
|
|
int agile_modbus_deserialize_read_bits(agile_modbus_t* ctx,
|
|
int msg_length,
|
|
uint8_t* dest);
|
|
int agile_modbus_serialize_read_input_bits(agile_modbus_t* ctx,
|
|
int addr,
|
|
int nb);
|
|
int agile_modbus_deserialize_read_input_bits(agile_modbus_t* ctx,
|
|
int msg_length,
|
|
uint8_t* dest);
|
|
int agile_modbus_serialize_read_registers(agile_modbus_t* ctx,
|
|
int addr,
|
|
int nb);
|
|
int agile_modbus_deserialize_read_registers(agile_modbus_t* ctx,
|
|
int msg_length,
|
|
uint16_t* dest);
|
|
int agile_modbus_serialize_read_input_registers(agile_modbus_t* ctx,
|
|
int addr,
|
|
int nb);
|
|
int agile_modbus_deserialize_read_input_registers(agile_modbus_t* ctx,
|
|
int msg_length,
|
|
uint16_t* dest);
|
|
int agile_modbus_serialize_write_bit(agile_modbus_t* ctx, int addr, int status);
|
|
int agile_modbus_deserialize_write_bit(agile_modbus_t* ctx, int msg_length);
|
|
int agile_modbus_serialize_write_register(agile_modbus_t* ctx,
|
|
int addr,
|
|
const uint16_t value);
|
|
int agile_modbus_deserialize_write_register(agile_modbus_t* ctx,
|
|
int msg_length);
|
|
int agile_modbus_serialize_write_bits(agile_modbus_t* ctx,
|
|
int addr,
|
|
int nb,
|
|
const uint8_t* src);
|
|
int agile_modbus_deserialize_write_bits(agile_modbus_t* ctx, int msg_length);
|
|
int agile_modbus_serialize_write_registers(agile_modbus_t* ctx,
|
|
int addr,
|
|
int nb,
|
|
const uint16_t* src);
|
|
int agile_modbus_deserialize_write_registers(agile_modbus_t* ctx,
|
|
int msg_length);
|
|
int agile_modbus_serialize_mask_write_register(agile_modbus_t* ctx,
|
|
int addr,
|
|
uint16_t and_mask,
|
|
uint16_t or_mask);
|
|
int agile_modbus_deserialize_mask_write_register(agile_modbus_t* ctx,
|
|
int msg_length);
|
|
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 agile_modbus_deserialize_write_and_read_registers(agile_modbus_t* ctx,
|
|
int msg_length,
|
|
uint16_t* dest);
|
|
int agile_modbus_serialize_report_slave_id(agile_modbus_t* ctx);
|
|
int agile_modbus_deserialize_report_slave_id(agile_modbus_t* ctx,
|
|
int msg_length,
|
|
int max_dest,
|
|
uint8_t* dest);
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/** @addtogroup Master_Raw_Operation_Functions
|
|
* @{
|
|
*/
|
|
int agile_modbus_serialize_raw_request(agile_modbus_t* ctx,
|
|
const uint8_t* raw_req,
|
|
int raw_req_length);
|
|
int agile_modbus_deserialize_raw_response(agile_modbus_t* ctx, int msg_length);
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/** @addtogroup Modbus_Slave
|
|
* @{
|
|
*/
|
|
|
|
/** @addtogroup Slave_Operation_Functions
|
|
* @{
|
|
*/
|
|
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);
|
|
void agile_modbus_slave_io_set(uint8_t* buf, int index, int status);
|
|
uint8_t agile_modbus_slave_io_get(uint8_t* buf, int index);
|
|
void agile_modbus_slave_register_set(uint8_t* buf, int index, uint16_t data);
|
|
uint16_t agile_modbus_slave_register_get(uint8_t* buf, int index);
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/* Include RTU and TCP module */
|
|
#include "agile_modbus_rtu.h"
|
|
#include "agile_modbus_tcp.h"
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
#endif /* __PKG_AGILE_MODBUS_H */
|