2020-05-29 16:45:25 +02:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2020 PHYTEC Messtechnik GmbH
|
2021-03-03 14:58:39 +01:00
|
|
|
* Copyright (c) 2021 Nordic Semiconductor ASA
|
2020-05-29 16:45:25 +02:00
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Parts of this file are based on mb.h from uC/Modbus Stack.
|
|
|
|
*
|
|
|
|
* uC/Modbus
|
|
|
|
* The Embedded Modbus Stack
|
|
|
|
*
|
|
|
|
* Copyright 2003-2020 Silicon Laboratories Inc. www.silabs.com
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: APACHE-2.0
|
|
|
|
*
|
|
|
|
* This software is subject to an open source license and is distributed by
|
|
|
|
* Silicon Laboratories Inc. pursuant to the terms of the Apache License,
|
|
|
|
* Version 2.0 available at www.apache.org/licenses/LICENSE-2.0.
|
|
|
|
*/
|
|
|
|
|
2021-03-02 19:26:04 +01:00
|
|
|
#ifndef ZEPHYR_INCLUDE_MODBUS_INTERNAL_H_
|
|
|
|
#define ZEPHYR_INCLUDE_MODBUS_INTERNAL_H_
|
2020-05-29 16:45:25 +02:00
|
|
|
|
includes: prefer <zephyr/kernel.h> over <zephyr/zephyr.h>
As of today <zephyr/zephyr.h> is 100% equivalent to <zephyr/kernel.h>.
This patch proposes to then include <zephyr/kernel.h> instead of
<zephyr/zephyr.h> since it is more clear that you are including the
Kernel APIs and (probably) nothing else. <zephyr/zephyr.h> sounds like a
catch-all header that may be confusing. Most applications need to
include a bunch of other things to compile, e.g. driver headers or
subsystem headers like BT, logging, etc.
The idea of a catch-all header in Zephyr is probably not feasible
anyway. Reason is that Zephyr is not a library, like it could be for
example `libpython`. Zephyr provides many utilities nowadays: a kernel,
drivers, subsystems, etc and things will likely grow. A catch-all header
would be massive, difficult to keep up-to-date. It is also likely that
an application will only build a small subset. Note that subsystem-level
headers may use a catch-all approach to make things easier, though.
NOTE: This patch is **NOT** removing the header, just removing its usage
in-tree. I'd advocate for its deprecation (add a #warning on it), but I
understand many people will have concerns.
Signed-off-by: Gerard Marull-Paretas <gerard.marull@nordicsemi.no>
2022-08-25 09:58:46 +02:00
|
|
|
#include <zephyr/kernel.h>
|
2022-05-06 11:12:04 +02:00
|
|
|
#include <zephyr/drivers/gpio.h>
|
|
|
|
#include <zephyr/modbus/modbus.h>
|
2020-05-29 16:45:25 +02:00
|
|
|
|
2021-03-02 19:26:04 +01:00
|
|
|
#ifdef CONFIG_MODBUS_FP_EXTENSIONS
|
|
|
|
#define MODBUS_FP_EXTENSIONS_ADDR 5000
|
2020-05-29 16:45:25 +02:00
|
|
|
#else
|
2021-03-02 19:26:04 +01:00
|
|
|
#define MODBUS_FP_EXTENSIONS_ADDR UINT16_MAX
|
2020-05-29 16:45:25 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#define MODBUS_RTU_MTU 256
|
|
|
|
|
|
|
|
/* Modbus function codes */
|
|
|
|
#define MODBUS_FC01_COIL_RD 1
|
|
|
|
#define MODBUS_FC02_DI_RD 2
|
|
|
|
#define MODBUS_FC03_HOLDING_REG_RD 3
|
|
|
|
#define MODBUS_FC04_IN_REG_RD 4
|
|
|
|
#define MODBUS_FC05_COIL_WR 5
|
|
|
|
#define MODBUS_FC06_HOLDING_REG_WR 6
|
|
|
|
#define MODBUS_FC08_DIAGNOSTICS 8
|
|
|
|
#define MODBUS_FC15_COILS_WR 15
|
|
|
|
#define MODBUS_FC16_HOLDING_REGS_WR 16
|
|
|
|
|
|
|
|
/* Diagnostic sub-function codes */
|
|
|
|
#define MODBUS_FC08_SUBF_QUERY 0
|
|
|
|
#define MODBUS_FC08_SUBF_CLR_CTR 10
|
|
|
|
#define MODBUS_FC08_SUBF_BUS_MSG_CTR 11
|
|
|
|
#define MODBUS_FC08_SUBF_BUS_CRC_CTR 12
|
|
|
|
#define MODBUS_FC08_SUBF_BUS_EXCEPT_CTR 13
|
|
|
|
#define MODBUS_FC08_SUBF_SERVER_MSG_CTR 14
|
|
|
|
#define MODBUS_FC08_SUBF_SERVER_NO_RESP_CTR 15
|
|
|
|
|
|
|
|
/* Modbus RTU (ASCII) constants */
|
|
|
|
#define MODBUS_COIL_OFF_CODE 0x0000
|
|
|
|
#define MODBUS_COIL_ON_CODE 0xFF00
|
|
|
|
#define MODBUS_RTU_MIN_MSG_SIZE 4
|
|
|
|
#define MODBUS_CRC16_POLY 0xA001
|
|
|
|
#define MODBUS_ASCII_MIN_MSG_SIZE 11
|
|
|
|
#define MODBUS_ASCII_START_FRAME_CHAR ':'
|
|
|
|
#define MODBUS_ASCII_END_FRAME_CHAR1 '\r'
|
|
|
|
#define MODBUS_ASCII_END_FRAME_CHAR2 '\n'
|
|
|
|
|
2021-03-11 00:42:21 +01:00
|
|
|
/* Modbus ADU constants */
|
|
|
|
#define MODBUS_ADU_PROTO_ID 0x0000
|
2020-05-29 16:45:25 +02:00
|
|
|
|
2021-03-03 14:58:39 +01:00
|
|
|
struct modbus_serial_config {
|
|
|
|
/* UART device */
|
|
|
|
const struct device *dev;
|
|
|
|
/* RTU timeout (maximum inter-frame delay) */
|
|
|
|
uint32_t rtu_timeout;
|
|
|
|
/* Pointer to current position in buffer */
|
|
|
|
uint8_t *uart_buf_ptr;
|
|
|
|
/* Pointer to driver enable (DE) pin config */
|
2021-03-17 16:58:05 +01:00
|
|
|
struct gpio_dt_spec *de;
|
2021-03-03 14:58:39 +01:00
|
|
|
/* Pointer to receiver enable (nRE) pin config */
|
2021-03-17 16:58:05 +01:00
|
|
|
struct gpio_dt_spec *re;
|
2021-03-03 14:58:39 +01:00
|
|
|
/* RTU timer to detect frame end point */
|
|
|
|
struct k_timer rtu_timer;
|
|
|
|
/* Number of bytes received or to send */
|
|
|
|
uint16_t uart_buf_ctr;
|
|
|
|
/* Storage of received characters or characters to send */
|
|
|
|
uint8_t uart_buf[CONFIG_MODBUS_BUFFER_SIZE];
|
|
|
|
};
|
|
|
|
|
2021-03-02 19:26:04 +01:00
|
|
|
#define MODBUS_STATE_CONFIGURED 0
|
2020-05-29 16:45:25 +02:00
|
|
|
|
2021-03-02 19:26:04 +01:00
|
|
|
struct modbus_context {
|
2020-12-28 16:19:19 +01:00
|
|
|
/* Interface name */
|
|
|
|
const char *iface_name;
|
2021-03-11 00:42:21 +01:00
|
|
|
union {
|
|
|
|
/* Serial line configuration */
|
|
|
|
struct modbus_serial_config *cfg;
|
|
|
|
/* RAW TX callback */
|
2022-07-31 12:53:37 +02:00
|
|
|
struct modbus_raw_cb rawcb;
|
2021-03-11 00:42:21 +01:00
|
|
|
};
|
2021-03-01 14:36:35 +01:00
|
|
|
/* MODBUS mode */
|
|
|
|
enum modbus_mode mode;
|
2020-05-29 16:45:25 +02:00
|
|
|
/* True if interface is configured as client */
|
|
|
|
bool client;
|
|
|
|
/* Amount of time client is willing to wait for response from server */
|
|
|
|
uint32_t rxwait_to;
|
|
|
|
/* Pointer to user server callbacks */
|
2021-03-02 18:48:30 +01:00
|
|
|
struct modbus_user_callbacks *mbs_user_cb;
|
2020-05-29 16:45:25 +02:00
|
|
|
/* Interface state */
|
|
|
|
atomic_t state;
|
|
|
|
|
|
|
|
/* Client's mutually exclusive access */
|
|
|
|
struct k_mutex iface_lock;
|
|
|
|
/* Wait for response semaphore */
|
|
|
|
struct k_sem client_wait_sem;
|
|
|
|
/* Server work item */
|
|
|
|
struct k_work server_work;
|
|
|
|
/* Received frame */
|
2021-03-03 15:34:43 +01:00
|
|
|
struct modbus_adu rx_adu;
|
2020-05-29 16:45:25 +02:00
|
|
|
/* Frame to transmit */
|
2021-03-03 15:34:43 +01:00
|
|
|
struct modbus_adu tx_adu;
|
2020-05-29 16:45:25 +02:00
|
|
|
|
|
|
|
/* Records error from frame reception, e.g. CRC error */
|
2021-03-03 15:34:43 +01:00
|
|
|
int rx_adu_err;
|
2020-05-29 16:45:25 +02:00
|
|
|
|
2021-03-02 19:26:04 +01:00
|
|
|
#ifdef CONFIG_MODBUS_FC08_DIAGNOSTIC
|
2020-05-29 16:45:25 +02:00
|
|
|
uint16_t mbs_msg_ctr;
|
|
|
|
uint16_t mbs_crc_err_ctr;
|
|
|
|
uint16_t mbs_except_ctr;
|
|
|
|
uint16_t mbs_server_msg_ctr;
|
|
|
|
uint16_t mbs_noresp_ctr;
|
|
|
|
#endif
|
2023-03-04 07:38:45 +01:00
|
|
|
/* A linked list of function code, handler pairs */
|
|
|
|
sys_slist_t user_defined_cbs;
|
2021-03-03 15:34:43 +01:00
|
|
|
/* Unit ID */
|
|
|
|
uint8_t unit_id;
|
2020-05-29 16:45:25 +02:00
|
|
|
|
|
|
|
};
|
|
|
|
|
2021-03-10 18:41:26 +01:00
|
|
|
/**
|
|
|
|
* @brief Get Modbus interface context.
|
|
|
|
*
|
|
|
|
* @param ctx Modbus interface context
|
|
|
|
*
|
|
|
|
* @retval Pointer to interface context or NULL
|
|
|
|
* if interface not available or not configured;
|
|
|
|
*/
|
2021-03-03 15:34:43 +01:00
|
|
|
struct modbus_context *modbus_get_context(const uint8_t iface);
|
2021-03-10 18:41:26 +01:00
|
|
|
|
2021-03-11 00:42:21 +01:00
|
|
|
/**
|
|
|
|
* @brief Get Modbus interface index.
|
|
|
|
*
|
|
|
|
* @param ctx Pointer to Modbus interface context
|
|
|
|
*
|
|
|
|
* @retval Interface index or negative error value.
|
|
|
|
*/
|
|
|
|
int modbus_iface_get_by_ctx(const struct modbus_context *ctx);
|
|
|
|
|
2021-03-10 18:41:26 +01:00
|
|
|
/**
|
|
|
|
* @brief Send ADU.
|
|
|
|
*
|
|
|
|
* @param ctx Modbus interface context
|
|
|
|
*/
|
2021-03-03 15:34:43 +01:00
|
|
|
void modbus_tx_adu(struct modbus_context *ctx);
|
2021-03-10 18:41:26 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Send ADU and wait certain time for response.
|
|
|
|
*
|
|
|
|
* @param ctx Modbus interface context
|
|
|
|
*
|
|
|
|
* @retval 0 If the function was successful,
|
|
|
|
* -ENOTSUP if Modbus mode is not supported,
|
|
|
|
* -ETIMEDOUT on timeout,
|
|
|
|
* -EMSGSIZE on length error,
|
|
|
|
* -EIO on CRC error.
|
|
|
|
*/
|
2021-03-09 12:00:32 +01:00
|
|
|
int modbus_tx_wait_rx_adu(struct modbus_context *ctx);
|
2020-05-29 16:45:25 +02:00
|
|
|
|
2021-03-10 18:41:26 +01:00
|
|
|
/**
|
|
|
|
* @brief Let server handle the received ADU.
|
|
|
|
*
|
|
|
|
* @param ctx Modbus interface context
|
|
|
|
*
|
|
|
|
* @retval True if the server has prepared a response ADU
|
|
|
|
* that should be sent.
|
|
|
|
*/
|
2021-03-03 15:34:43 +01:00
|
|
|
bool modbus_server_handler(struct modbus_context *ctx);
|
2021-03-10 18:41:26 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Reset server stats.
|
|
|
|
*
|
|
|
|
* @param ctx Modbus interface context
|
|
|
|
*/
|
2021-03-03 15:34:43 +01:00
|
|
|
void modbus_reset_stats(struct modbus_context *ctx);
|
2020-05-29 16:45:25 +02:00
|
|
|
|
2021-03-10 18:41:26 +01:00
|
|
|
/**
|
|
|
|
* @brief Disable serial line reception.
|
|
|
|
*
|
|
|
|
* @param ctx Modbus interface context
|
|
|
|
*/
|
2021-03-03 13:43:48 +01:00
|
|
|
void modbus_serial_rx_disable(struct modbus_context *ctx);
|
2021-03-10 18:41:26 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Enable serial line reception.
|
|
|
|
*
|
|
|
|
* @param ctx Modbus interface context
|
|
|
|
*/
|
2021-03-03 13:43:48 +01:00
|
|
|
void modbus_serial_rx_enable(struct modbus_context *ctx);
|
2021-03-10 18:41:26 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Assemble ADU from serial line RX buffer
|
|
|
|
*
|
|
|
|
* @param ctx Modbus interface context
|
|
|
|
*
|
|
|
|
* @retval 0 If the function was successful,
|
|
|
|
* -ENOTSUP if serial line mode is not supported,
|
|
|
|
* -EMSGSIZE on length error,
|
|
|
|
* -EIO on CRC error.
|
|
|
|
*/
|
2021-03-03 15:34:43 +01:00
|
|
|
int modbus_serial_rx_adu(struct modbus_context *ctx);
|
2021-03-10 18:41:26 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Assemble ADU from serial line RX buffer
|
|
|
|
*
|
|
|
|
* @param ctx Modbus interface context
|
|
|
|
*
|
|
|
|
* @retval 0 If the function was successful,
|
|
|
|
* -ENOTSUP if serial line mode is not supported.
|
|
|
|
*/
|
2021-03-03 15:34:43 +01:00
|
|
|
int modbus_serial_tx_adu(struct modbus_context *ctx);
|
2021-03-10 18:41:26 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Initialize serial line support.
|
|
|
|
*
|
|
|
|
* @param ctx Modbus interface context
|
2021-03-10 19:19:12 +01:00
|
|
|
* @param param Configuration parameter of the interface
|
2021-03-10 18:41:26 +01:00
|
|
|
*
|
|
|
|
* @retval 0 If the function was successful.
|
|
|
|
*/
|
2021-03-03 13:43:48 +01:00
|
|
|
int modbus_serial_init(struct modbus_context *ctx,
|
2021-03-10 19:19:12 +01:00
|
|
|
struct modbus_iface_param param);
|
2021-03-10 18:41:26 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Disable serial line support.
|
|
|
|
*
|
|
|
|
* @param ctx Modbus interface context
|
|
|
|
*/
|
2021-03-03 13:43:48 +01:00
|
|
|
void modbus_serial_disable(struct modbus_context *ctx);
|
|
|
|
|
2021-03-11 00:42:21 +01:00
|
|
|
int modbus_raw_rx_adu(struct modbus_context *ctx);
|
|
|
|
int modbus_raw_tx_adu(struct modbus_context *ctx);
|
|
|
|
int modbus_raw_init(struct modbus_context *ctx,
|
|
|
|
struct modbus_iface_param param);
|
|
|
|
|
2021-03-02 19:26:04 +01:00
|
|
|
#endif /* ZEPHYR_INCLUDE_MODBUS_INTERNAL_H_ */
|