使用libmodbus库开发modbusTcp从站(支持多个主站连接)

使用libmodbus库开发modbusTcp从站(支持多个主站连接)

  • Chapter1 使用libmodbus库开发modbusTcp从站(支持多个主站连接)
    • rdsmodbusslave.h
    • rdsmodbusslave.cpp
    • main.cpp

Chapter1 使用libmodbus库开发modbusTcp从站(支持多个主站连接)

参考链接:https://blog.csdn.net/v6543210/article/details/127426450

https://blog.csdn.net/qq_38158479/article/details/120928043

当我们需要自己搞一个C/C++版的 modbus Server时,总想像C#里面借助个好用的库来实现,但是libmodbus这个库封装的并不好用,从官方的源码中连个example都没有,能参考的也就tests目录下有几个可以借鉴。

但是仔细看了一下,random-test-server.c 还是会阻塞的,单线程。与拿来即用的标准相差甚远。

如果需要实现对多个客户端提供服务,需要参考 bandwidth-server-many-up.c

本文借鉴这篇文章,进行了一点优化,实现了可以为多个客户端提供服务的modbus tcp Server,可以拿来即用。

使用libmodbus库开发modbusTcp从站(支持多个主站连接)_酸菜。的博客-CSDN博客_libmodbus tcp

如果需要自己实现逻辑可以直接在另一个线程函数中对modbus的变量进行修改。

rdsmodbusslave.h

#ifndef RDSMODBUSSLAVE_H
#define RDSMODBUSSLAVE_H#include <iostream>
#include <thread>
#include <stdlib.h>
#include <iostream>
#include <mutex>
#include <string>
using namespace std;
/*如果是windows平台则要加载相应的静态库和头文件*/
#ifdef _WIN32
#define _WINSOCK_DEPRECATED_NO_WARNINGS
#include <winsock2.h>
#include <windows.h>
//#include <modbus.h>
#pragma comment(lib, "Ws2_32.lib")
//#pragma comment(lib, "modbus.lib")
/*linux平台*/
#else
//#include <modbus/modbus.h>
#include <unistd.h>
#include <error.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#include <sys/time.h>
#include <sys/select.h>
#endif//#define  MAX_POINT  50000
#include <QObject>
#include <QThread>
#include <stdio.h>
#include <libmodbus/config.h>
#include <libmodbus/modbus.h>
#include <libmodbus/modbus-rtu.h>
#include <QTimer>
#include <QDebug>
#include <QStringList>
#include <QSerialPortInfo>
#include <QSerialPort>class RDSModbusSlave : public QObject
{Q_OBJECT
public:explicit RDSModbusSlave(QObject *parent = nullptr);RDSModbusSlave(string host="0.0.0.0", uint16_t port=502);~RDSModbusSlave();public:void recieveMessages();bool modbus_set_slave_id(int id);bool initModbus(std::string Host_Ip, int port, bool debugging);uint8_t getTab_Input_Bits(int NumBit);bool setTab_Input_Bits(int NumBit, uint8_t Value);uint16_t getHoldingRegisterValue(int registerStartaddress);float getHoldingRegisterFloatValue(int registerStartaddress);bool setHoldingRegisterValue(int registerStartaddress, uint16_t Value);bool setHoldingRegisterValue(int registerStartaddress, float Value);bool setInputRegisterValue(int registerStartaddress, uint16_t Value);bool setInputRegisterValue(int registerStartaddress, float Value);private:std::mutex slavemutex;int m_errCount{ 0 };int m_modbusSocket{ -1 };bool m_initialized{ false };modbus_t* ctx{ nullptr };modbus_mapping_t* mapping{ nullptr };/*Mapping*/int m_numBits{ 60000 };int m_numInputBits{ 60000 };int m_numRegisters{ 60000 };int m_numInputRegisters{ 60000 };public:void loadFromConfigFile();void run();signals:};/*annotation:
(1)https://www.jianshu.com/p/0ed380fa39eb
(2)typedef struct _modbus_mapping_t
{int nb_bits;                //线圈int start_bits;int nb_input_bits;          //离散输入int start_input_bits;int nb_input_registers;     //输入寄存器int start_input_registers;int nb_registers;           //保持寄存器int start_registers;uint8_t *tab_bits;uint8_t *tab_input_bits;uint16_t *tab_input_registers;uint16_t *tab_registers;
}modbus_mapping_t;*/#endif // RDSMODBUSSLAVE_H

rdsmodbusslave.cpp

#include "rdsmodbusslave.h"#ifdef _WIN32
typedef int socklen_t;
#endifRDSModbusSlave::RDSModbusSlave(QObject *parent) : QObject(parent)
{}/**************************************************************** @file       RDSModbusSlave.cpp* @author     seer-txj* @brief      Constructor* @version    v1* @return     null* @date       2021/10/6**************************************************************/
RDSModbusSlave::RDSModbusSlave(string host, uint16_t port)
{initModbus(host, port, false);//TODO:this->setHoldingRegisterValue(0, (uint16_t)0x1122);this->setHoldingRegisterValue(3, (uint16_t)0x3022);this->setHoldingRegisterValue(6, (uint16_t)0x6022);
}/**************************************************************** @file       RDSModbusSlave.cpp* @author     seer-txj* @brief      Destructor* @version    v1* @return     null* @date       2021/10/6**************************************************************/
RDSModbusSlave::~RDSModbusSlave()
{modbus_mapping_free(mapping);modbus_close(ctx);modbus_free(ctx);
}/**************************************************************** @file       RDSModbusSlave.cpp* @author     seer-txj* @brief      支持多个master同时连接* @version    v1* @return     null* @date       2021/10/6**************************************************************/
void RDSModbusSlave::recieveMessages()
{uint8_t query[MODBUS_TCP_MAX_ADU_LENGTH];int master_socket;int rc;fd_set refset;fd_set rdset;/* Maximum file descriptor number */int fdmax;/* Clear the reference set of socket */FD_ZERO(&refset);/* Add the server socket */FD_SET(m_modbusSocket, &refset);/* Keep track of the max file descriptor */fdmax = m_modbusSocket;while( true ){rdset = refset;if (select(fdmax+1, &rdset, NULL, NULL, NULL) == -1){perror("Server select() failure.");break;}/* Run through the existing connections looking for data to be* read */for (master_socket = 0; master_socket <= fdmax; master_socket++){if (!FD_ISSET(master_socket, &rdset)){continue;}if (master_socket == m_modbusSocket){/* A client is asking a new connection */socklen_t addrlen;struct sockaddr_in clientaddr;int newfd;/* Handle new connections */addrlen = sizeof(clientaddr);memset(&clientaddr, 0, sizeof(clientaddr));newfd = accept(m_modbusSocket, (struct sockaddr *)&clientaddr, &addrlen);if (newfd == -1){perror("Server accept() error");} else{FD_SET(newfd, &refset);if (newfd > fdmax){/* Keep track of the maximum */fdmax = newfd;}printf("New connection from %s:%d on socket %d\n", inet_ntoa(clientaddr.sin_addr), clientaddr.sin_port, newfd);}} else{modbus_set_socket(ctx, master_socket);rc = modbus_receive(ctx, query);if (rc > 0){modbus_reply(ctx, query, rc, mapping);} else if (rc == -1){/* This example server in ended on connection closing or* any errors. */printf("Connection closed on socket %d\n", master_socket);#ifdef _WIN32closesocket(master_socket);#elseclose(master_socket);#endif/* Remove from reference set */FD_CLR(master_socket, &refset);if (master_socket == fdmax){fdmax--;}}}}}m_initialized = false;
}/**************************************************************** @file       RDSModbusSlave.cpp* @author     seer-txj* @brief      modbus_set_slave_id* @param      id* @version    v1* @return     null* @date       2021/10/19**************************************************************/
bool RDSModbusSlave::modbus_set_slave_id(int id)
{int rc = modbus_set_slave(ctx, id);if (rc == -1){fprintf(stderr, "Invalid slave id\n");modbus_free(ctx);return false;}return true;
}/**************************************************************** @file       RDSModbusSlave.cpp* @author     seer-txj* @brief      modbus initialization* @param      IP/PORT/debugflag* @version    v1* @return     null* @date       2021/10/6**************************************************************/
bool RDSModbusSlave::initModbus(std::string Host_Ip = "127.0.0.1", int port = 502, bool debugging = false)
{ctx = modbus_new_tcp(Host_Ip.c_str(), port);modbus_set_debug(ctx, debugging);if (ctx == NULL){fprintf(stderr, "There was an error allocating the modbus\n");throw - 1;}m_modbusSocket = modbus_tcp_listen(ctx, 1);/*设置线圈, 离散输入, 输入寄存器, 保持寄存器个数(数组元素个数))*/mapping = modbus_mapping_new(m_numBits, m_numInputBits, m_numInputRegisters, m_numRegisters);if (mapping == NULL){fprintf(stderr, "Unable to assign mapping:%s\n", modbus_strerror(errno));modbus_free(ctx);m_initialized = false;return false;}m_initialized = true;return true;
}/**************************************************************** @file       RDSModbusSlave.cpp* @author     seer-txj* @brief      getTab_Input_Bits(获取输入寄存器某一位的值)* @param      NumBit(输入寄存器相应的bit位)* @version    v1* @return     null* @date       2021/10/8**************************************************************/
uint8_t RDSModbusSlave::getTab_Input_Bits(int NumBit)
{if (!m_initialized){return -1;}return mapping->tab_input_bits[NumBit];
}/**************************************************************** @file       RDSModbusSlave.cpp* @author     seer-txj* @brief      setTab_Input_Bits(设置输入寄存器某一位的值)* @param      NumBit(输入寄存器的起始地址)* @param      Value(输入寄存器的值)* @version    v1* @return     null* @date       2021/10/8**************************************************************/
bool RDSModbusSlave::setTab_Input_Bits(int NumBit, uint8_t Value)
{if (NumBit > (m_numInputBits - 1)){return false;}slavemutex.lock();mapping->tab_input_bits[NumBit] = Value;slavemutex.unlock();return true;
}/**************************************************************** @file       RDSModbusSlave.cpp* @author     seer-txj* @brief      getRegisterValue(获取保存寄存器的值)* @param      registerStartaddress(保存寄存器的起始地址)* @version    v1* @return     null* @date       2021/10/6**************************************************************/
uint16_t RDSModbusSlave::getHoldingRegisterValue(int registerStartaddress)
{if (!m_initialized){return -1;}return mapping->tab_registers[registerStartaddress];
}/**************************************************************** @file       RDSModbusSlave.cpp* @author     seer-txj* @brief      获取寄存器里的浮点数* @param      registerStartaddress寄存器起始地址* @version    v1* @return     两个uint16_t拼接而成的浮点值* @date       2021/10/6**************************************************************/
float RDSModbusSlave::getHoldingRegisterFloatValue(int registerStartaddress)
{if (!m_initialized){return -1.0f;}return modbus_get_float_badc(&mapping->tab_registers[registerStartaddress]);
}/**************************************************************** @file       RDSModbusSlave.cpp* @author     seer-txj* @brief      setRegisterValue(设置保存寄存器的值,类型为uint16_t)* @param      registerStartaddress(保存寄存器的起始地址)* @param      Value(写入到保存寄存器里的值)* @version    v1* @return     null* @date       2021/10/6**************************************************************/
bool RDSModbusSlave::setHoldingRegisterValue(int registerStartaddress, uint16_t Value)
{if (registerStartaddress > (m_numRegisters - 1)){return false;}slavemutex.lock();mapping->tab_registers[registerStartaddress] = Value;slavemutex.unlock();return true;
}/**************************************************************** @file       RDSModbusSlave.cpp* @author     seer-txj* @brief      setRegisterFloatValue(设置浮点值)* @param      (Value:浮点值,registerStartaddress寄存器起始地址)* @version    v1* @return     null* @date       2021/10/8**************************************************************/
bool RDSModbusSlave::setHoldingRegisterValue(int registerStartaddress, float Value)
{if (registerStartaddress > (m_numRegisters - 2)){return false;}/*小端模式*/slavemutex.lock();modbus_set_float(Value, &mapping->tab_registers[registerStartaddress]);slavemutex.unlock();return true;
}bool RDSModbusSlave::setInputRegisterValue(int registerStartaddress, uint16_t Value)
{if (registerStartaddress > (m_numRegisters - 1)){return false;}slavemutex.lock();mapping->tab_input_registers[registerStartaddress] = Value;slavemutex.unlock();return true;
}bool RDSModbusSlave::setInputRegisterValue(int registerStartaddress, float Value)
{if (registerStartaddress > (m_numRegisters - 2)){return false;}/*小端模式*/slavemutex.lock();modbus_set_float(Value, &mapping->tab_input_registers[registerStartaddress]);slavemutex.unlock();return true;
}/**************************************************************** @file       RDSModbusSlave.cpp* @author     seer-txj* @brief      loadFromConfigFile* @version    v1* @return     null* @date       2021/10/18**************************************************************/
void RDSModbusSlave::loadFromConfigFile()
{}/**************************************************************** @file       RDSModbusSlave.cpp* @author     seer-txj* @brief      run* @version    v1* @return     null* @date       2021/10/18**************************************************************/
void RDSModbusSlave::run()
{std::thread loop([this](){while (true){if (m_initialized){recieveMessages();}else{m_initialized = true;}}});loop.detach();return;
}

main.cpp

#include "mainwindow.h"#include <QApplication>//#include "rdsmodbusslave.h"using namespace std;void modbusRunner(RDSModbusSlave* server)
{server->recieveMessages();
}RDSModbusSlave modServer("127.0.0.1", 502);int main(int argc, char *argv[])
{QApplication a(argc, argv);MainWindow w;w.show();std::thread modServerThread(modbusRunner, &modServer);modServerThread.join();return a.exec();
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/95934.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

合并不同年度收入数据-外连接

案例需求&#xff1a;统计2019年和2020年的客户销售收入并求和 思路&#xff1a;使用pandas读取excel数据横向连接&#xff0c;使用sum()含函数求和 代码如下&#xff1a; 1、使用pandas读取excel数据 2、由于两个表一个是"客户名称"&#xff0c;一个是客户描述,使…

基于ChatGPT快速入门体验NLP词云

基于ChatGPT快速入门体验NLP词云 一、什么是自然语言处理二、自然语言处理和词云的关系三、Python环境准备四、基于ChatGpt制作词云4.1 ChatGPT生成初级词云代码4.2 ChatGPT生成进阶词云代码4.3 基于ChatGPT解决代码问题4.4 基于ChatGPT建议修改问题代码 一、什么是自然语言处理…

最强中间件!Kafka快速入门(Kafka理论+SpringBoot集成Kafka实践)

自媒体文章上下架 需求分析 媒体端下架文章同时app端也下架文章的实现可以通过feign去调用&#xff0c;但这种实现耦合度太高&#xff0c;这里使用MQ进行解耦 自媒体端一旦上下架文章就发送消息给MQ&#xff0c;文章微服务在去读取消息根据消息内容上下架文章 MQ还可以流量削…

24 Python的sqlite3模块

概述 在上一节&#xff0c;我们介绍了Python的shutil模块&#xff0c;包括&#xff1a;shutil模块中一些常用的函数。在这一节&#xff0c;我们将介绍Python的sqlite3模块。sqlite3模块是Python中的内置模块&#xff0c;用于与SQLite数据库交互。SQLite是一个轻量级的磁盘数据库…

windows server 2012 服务器打开系统远程功能

服务器上开启远程功能 进入服务器&#xff0c;选择“添加角色和功能” 需要选择安装的服务器类型&#xff0c;如图所示 然后在服务器池中选择你需要使用的服务器。 选择完成后&#xff0c;在图示列表下勾选“远程桌面服务” 再选择需要安装的功能和角色服务。 选择完成确认内容…

CTFHUB - SSRF

目录 SSRF漏洞 攻击对象 攻击形式 产生漏洞的函数 file_get_contents() fsockopen() curl_exec() 提高危害 利用的伪协议 file dict gopher 内网访问 伪协议读取文件 端口扫描 POST请求 总结 上传文件 总结 FastCGI协议 CGI和FastCGI的区别 FastCGI协议 …

MyBatis-plus使用

1 基础介绍 MyBatis-Plus (opens new window)&#xff08;简称 MP&#xff09;是一个 MyBatis (opens new window)的增强工具&#xff0c;在 MyBatis 的基础上只做增强不做改变&#xff0c;为简化开发、提高效率而生。 它已经封装好了一些crud方法&#xff0c;我们不需要再写…

Leetcode hot 100之双指针(快慢指针、滑动窗口)

目录 数组 有序的平方仍有序 删除/覆盖元素 移动零&#xff1a;交换slow和fast 滑动窗口&#xff1a;最短的连续子串&#xff08;r可行解->l--最短解&#xff09; 最小长度的子数组 求和&#xff1a;sort、l i 1, r len - 1 三数之和abctarget 四数之和abcdtarg…

Linux中Too many open files

Linux中Too many open files 问题分析和解决_e929: too many viminfo temp files-CSDN博客 too many open files 出现这句提示的原因是程序打开的文件/socket连接数量超过系统设定值。 查看每个用户最大允许打开文件数量 ulimit -a fdipzoneubuntu:~$ ulimit -a core file …

修改 ModelScope 默认缓存路径

修改 ModelScope 默认缓存路径 设置 MODELSCOPE_CACHE 和 MODELSCOPE_MODULES_CACHE 两个环境变量。 export MODELSCOPE_CACHE<your_favourite_path>/hub export MODELSCOPE_MODULES_CACHE<your_favourite_path>/modelscope_modules完结&#xff01;

VUE3照本宣科——package.json与vite.config.js

VUE3照本宣科——package.json与vite.config.js VUE3照本宣科系列导航 前言一、package.json1.name2.version3.private4.scripts5.dependencies6.devDependencies 二、vite.config.js1.plugins2.resolve.alias3.base4.mode 三、VUE3照本宣科系列总结 VUE3照本宣科系列导航 1.VU…

大数据Doris(五):开始编译 Doris

文章目录 开始编译 Doris 一、下载Doris的安装包 二、解压缩 三、上传配置文件

commons-collections4工具常用方法

commons-collections4是Apache Commons项目中的一个模块&#xff0c;提供了一系列处理集合和映射的工具类、接口和算法。它是在commons-collections的基础上进行了改进和增强&#xff0c;为Java开发者提供了更多集合操作的功能和便利性。 引入依赖 <dependency><group…

嵌入式操作系统服务机制

欢迎关注博主 Mindtechnist 或加入【智能科技社区】一起学习和分享Linux、C、C、Python、Matlab&#xff0c;机器人运动控制、多机器人协作&#xff0c;智能优化算法&#xff0c;滤波估计、多传感器信息融合&#xff0c;机器学习&#xff0c;人工智能等相关领域的知识和技术。搜…

三一充填泵:煤矿矸石无害化充填,煤炭绿色高效开采的破局利器

富煤贫油少气是我国的能源禀赋特征&#xff0c;决定了我国以煤炭为主的能源结构&#xff0c;煤炭为国民经济发展提供了重要的基础。煤炭开采过程会对土地、地下水、空气等环境造成较大的污染&#xff0c;但大宗固废煤矸石无害化充填的技术手段可以有效改善这样的情况&#xff0…

【Linux】线程详解完结篇——信号量 + 线程池 + 单例模式 + 读写锁

线程详解第四篇 前言正式开始信号量引例信号量的本质信号量相关的四个核心接口生产消费者模型用环形队列实现生产者消费者模型基于环形队列的生产消费模型的原理代码演示单生产者单消费者多生产者多消费者 计数器的意义 线程池基本概念代码 单例模式STL,智能指针和线程安全STL中…

移动D频段频点的计算

移动D频段的频率范围是2515MHz ~ 2675MHz&#xff0c;用于TDD-LTE制式的通信。在D频段中&#xff0c;D1频点的中心频率为37900 MHz。这个中心频点的计算方式如下&#xff1a; 首先需要知道&#xff0c;在TDD-LTE的通信中&#xff0c;频段是被分成多个子带进行使用的。在移动D频…

软考知识汇总-软件工程

软件工程 1 能力成熟度模型&#xff08;CMM&#xff09;2 能力成熟度模型集成&#xff08;CMMI&#xff09;2.1阶段式模型2.2 连续式模型 3 软件过程模型 1 能力成熟度模型&#xff08;CMM&#xff09; 将软件工程成熟度分为5个级别 初始级&#xff1a;杂乱无章&#xff0c;很…

设计模式——11. 享元模式

1. 说明 享元模式(Flyweight Pattern)是一种结构型设计模式,它旨在减少系统中相似对象的内存占用或计算开销,通过共享相同的对象来达到节省资源的目的。 享元模式的核心思想是将对象的状态分为内部状态(Intrinsic State)和外部状态(Extrinsic State): 内部状态是对象…

基于Uniswap V3的去中心化前端现货交易平台Oku正式登陆Moonbeam

波卡上的Uniswap v3合约由Moonbeam智能合约、Oku前端&#xff0c;以及Wormhole远程路由技术共同实现。 跨链互连应用的最佳去中心化开发平台Moonbeam宣布Uniswap现已正式登陆。此次是Uniswap产品作为一个主流的DEX首次涉足Polkadot生态。用户可以通过新的、易于使用的Oku界面与…