libmodbus:写一个modbusTCP服务

初级代码游戏的专栏介绍与文章目录-CSDN博客

我的github:codetoys,所有代码都将会位于ctfc库中。已经放入库中我会指出在库中的位置。

这些代码大部分以Linux为目标但部分代码是纯C++的,可以在任何平台上使用。

源码指引:github源码指引_初级代码游戏的博客-CSDN博客


        libmodbus很好用,不过多是写客户端。为了测试客户端,一般会用物理设备或模拟程序,不过既然libmodbus支持写服务端,为什么不直接写一个服务端用来测试呢?(串口当然可能受数量限制,TCP就没有任何限制了)

目录

一、主要过程

1.1 创建上下文对象,设定参数

1.1.1 坑:Ubuntu上无法打开低端口

1.2 数据映射

1.3 启动服务

1.4 接受连接

1.5 接收请求

1.6 返回应答

1.7 清理

二、完整代码

三、处理多个连接


一、主要过程

1.1 创建上下文对象,设定参数

MODBUS_API modbus_t* modbus_new_tcp(const char *ip_address, int port);

        非常简单,指定地址端口就可以了。地址NULL则使用任何地址,标准端口是502。

1.1.1 坑:Ubuntu上无法打开低端口

        这个坑好大,我试了好久程序都不正确,在后面modbus_receive的时候挂了,开始以为是内存错误,后来老老实实每步检查返回值才发现是modbus_tcp_listen这一步就失败了,提示“无权操作”,用了su也不行,于是想到会不会是低端口保护,改成高端口就正常了(比如10502)。

        低端口0-1023由国际组织分配,Ubuntu限制应用程序不能使用是可以理解的。

1.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;MODBUS_API modbus_mapping_t* modbus_mapping_new(int nb_bits, int nb_input_bits,int nb_registers, int nb_input_registers);

        根据给定的四种数据的数量创建存储结构,返回的结构里面对每种数据都包含三个值:

  1. 数据个数,最大数据量
  2. 起始modbus地址,数据对应的modbus地址可以不从0开始,比如只提供【100-120】
  3. 数据指针,存储实际数据,可以根据需要直接修改每个数据的值(但是不要动这个指针,这是内部创建的,用另一个函数释放)

1.3 启动服务

MODBUS_API int modbus_tcp_listen(modbus_t *ctx, int nb_connection);

        这会根据之前设置的参数来启动服务,nb_connection是一般TCP编程里面的等待连接队列长度。

        返回值是服务socket的值,如果成功返回值应该大于0。服务端口要自行用close来关闭。

1.4 接受连接

MODBUS_API int modbus_tcp_accept(modbus_t *ctx, int *s);

        这一步的参数s就是前一步的返回值,也就是服务socket。

        返回值是新socket,同时新socket也会存储在上下文中,后续收发操作使用上下文中存储的socket。

1.5 接收请求

MODBUS_API int modbus_receive(modbus_t *ctx, uint8_t *req);

        这个函数接收一个请求并存储在req里面,返回值是数据长度:

  • 大于0 有效的请求
  • 等于0 忽略的请求,比如从站号不匹配(本例程并未设置从站号)
  • -1 出错

        循环调用此函数接受请求,并可以在接收之后进行一些处理,然后再发送应答。

1.6 返回应答

MODBUS_API int modbus_reply(modbus_t *ctx, const uint8_t *req,int req_length, modbus_mapping_t *mb_mapping);

        如果没什么别的要求,直接调用这个函数返回应答就可以了。调用之前可以修改数据映射的数据。

1.7 清理

				if (s != -1){close(s);}modbus_mapping_free(mb_mapping);modbus_close(ctx);modbus_free(ctx);

        服务端口需要关闭,数据映射和上下文需要释放。

二、完整代码

			modbus_t * ctx = modbus_new_tcp(NULL, 10503);//ubuntu上开启低端口会报权限不足,su也不行modbus_mapping_t * mb_mapping = modbus_mapping_new(100, 100, 100, 100);if (mb_mapping == NULL){fprintf(stderr, "Failed to allocate the mapping: %s\n", modbus_strerror(errno));modbus_free(ctx);return -1;}//设置初值{mb_mapping->start_registers = 0;for (int i = 0; i < 10; ++i){mb_mapping->tab_registers[mb_mapping->start_registers + i] = i;}}while (CMyProcess::isProcessLive(parent_pid)){int s = modbus_tcp_listen(ctx, 5);if (s < 0){thelog << "modbus_tcp_listen error : " << modbus_strerror(errno) << endi;SleepSeconds(1);continue;}modbus_tcp_accept(ctx, &s); thelog << "s:" << s << endi;while (CMyProcess::isProcessLive(parent_pid)){uint8_t query[512];int rc = modbus_receive(ctx, query);if (rc > 0){modbus_reply(ctx, query, rc, mb_mapping);}else if (rc == -1){break;}//改变数据for (int i = 0; i < 10; ++i){++mb_mapping->tab_registers[mb_mapping->start_registers + i];}}thelog << "对方断开或出错 " << modbus_strerror(errno) << endi;if (s != -1){close(s);}}modbus_mapping_free(mb_mapping);modbus_close(ctx);modbus_free(ctx);

        CMyProcess::isProcessLive(parent_pid)判断父进程是否存在,换成死循环就可以了。

        专门对保持寄存器的前十个值做了设置,因为测试只用了这几个值。

        一次只能处理一个连接,这个连接断开才会处理下一个连接。因为客户socket是存储在上下文的,所以并行处理多个连接不方便。实际上写这个代码的目的是程序连接到自身来进行回归测试的。

三、处理多个连接

        额外有个函数modbus_set_socket用来改变上下文中保存的客户连接,可以接受多个连接,然后用select来选择可以操作的连接,然后先设置modbus_set_socket再modbus_receive。

        因为我没有试,所以没有示例代码。


(这里是文档结束)

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

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

相关文章

如何使用 ONNX 结合 GPU 加速推理(CUDA 与 cuDNN 简明指南)

前言 在深度学习模型推理中,使用 GPU 进行加速是提升模型推理速度的关键方式之一。 本文将带大家一步步了解如何使用 ONNX Runtime 结合 NVIDIA 的 CUDA 和 cuDNN 进行 GPU 加速。 一、查找ONNX、CUDA与cuDNN之间的对应版本 首先,我们需要确保 ONNX Runtime 与 CUDA 和 cu…

量化投资策略_因子打分选股的案例实现

一&#xff1a;因子打分选股的介绍 因子打分选股是一种量化投资策略&#xff0c;它通过选取多个与股票收益率相关的因子&#xff0c;对股票进行综合评分&#xff0c;然后根据评分来选择股票构建投资组合。以下是构建多因子打分选股模型的一般步骤&#xff1a; 数据预处理&…

Redis——常用数据类型hash

目录 hash常用命令hsethgethdelhkeyshvalshgetallhmgethlenhsetnxhincrbyhdecrby 哈希的编码方式哈希的应用 hash 常用命令 hset HSET key field value [field value ...]//时间复杂度O(1) //返回值&#xff1a;设置成功的键值对的个数hget HGET key field//hdel HDEL key…

【SSRF漏洞】——http协议常见绕过

改变的确很难&#xff0c;但结果值得冒险 本文如有错误之处&#xff0c;还请各位师傅指正 一.ssrf概述 SSRF全称为Server-side Request Fogery,中文含义服务器端请求伪造 SSRF是一种由攻击者构造形成由目标服务端发起请求的一个安全漏洞。一般情况下&#xff0c;SSRF攻击的目标…

Linux 防火墙:iptables (二)

文章目录 SNAT 原理与应用SNAT 应用环境SNAT 原理SNAT 转换前提条件SNAT 格式SNAT 转换规则配置 DNAT 原理与应用DNAT 应用环境DNAT 原理DNAT 转换前提条件DNAT 格式DNAT 转换规则配置 iptables 规则的备份和还原导出&#xff08;备份&#xff09;所有表的规则导入&#xff08;…

PCL 点云基于曲率大小渲染颜色

目录 一、概述 1.1原理 1.2实现步骤 1.3应用场景 二、代码实现 2.1关键函数 2.2完整代码 三、实现效果 3.1原始点云 3.2处理后点云 PCL点云算法汇总及实战案例汇总的目录地址链接&#xff1a; PCL点云算法与项目实战案例汇总&#xff08;长期更新&#xff09; 一、概…

Django笔记一:搭建Django环境与URL路径访问

博主之前学从Java后端开发&#xff0c;后面获取到读研资格&#xff0c;想着未来转算法岗&#xff0c;初学Python&#xff0c;发现Python还挺有趣的&#xff0c;由于之前所学后端缘故&#xff0c;有点后端情节&#xff0c;想学习一下Django框架&#xff08;python的web框架&…

人工智能和机器学习:探讨人工智能和机器学习的最新发展、应用、挑战和未来趋势

人工智能和机器学习是当前科技领域的热点话题&#xff0c;其最新发展、应用、挑战和未来趋势备受关注。 最新发展&#xff1a; 人工智能和机器学习技术在近年来得到了快速发展&#xff0c;尤其是深度学习技术的广泛应用。例如&#xff0c;深度学习在图像识别、语音识别、自然语…

react 基础语法

前置知识 类的回顾 通过class关键字定义一个类 类名首字母大写 class类有constructor构造器 new 一个类得到一个实例 类还有方法&#xff0c;该方法也会在其原型上 static静态数据&#xff0c;访问静态属性通过 类名.id getter和setter getter&#xff1a;定义一个属性&…

网络学习-eNSP配置VRRP

虚拟路由冗余协议(Virtual Router Redundancy Protocol&#xff0c;简称VRRP) VRRP广泛应用在边缘网络中&#xff0c;是一种路由冗余协议&#xff0c;它的设计目标是支持特定情况下IP数据流量失败转移不会引起混乱&#xff0c;允许主机使用单路由器&#xff0c;以及即使在实际…

全球NAND原厂闪存市场格局变化

根据市场研究机构TrendForce的最新跟踪报告&#xff0c;三星&#xff08;Samsung&#xff09;和SK海力士&#xff08;SK hynix-Solidigm&#xff09;在过去的一个季度中扩大了他们在NAND闪存市场的份额&#xff0c;这主要得益于抢占了铠侠&#xff08;Kioxia&#xff09;与西部…

小目标检测顶会新思路!最新成果刷爆遥感SOTA,参数小了18倍

遥感领域的小目标检测一直是个具有挑战性和趣味性的研究方向&#xff0c;同时也是顶会顶刊的常客。但不得不说&#xff0c;今年关于遥感小目标检测的研究热情尤其高涨&#xff0c;已经出现了很多非常优秀的成果。 比如SuperYOLO方法&#xff0c;通过融合多模态数据并执行高分辨…

【重学 MySQL】二十八、SQL99语法新特性之自然连接和 using 连接

【重学 MySQL】二十八、SQL99语法新特性之自然连接和 using 连接 自然连接&#xff08;NATURAL JOIN&#xff09;USING连接总结 SQL99语法在SQL92的基础上引入了一些新特性&#xff0c;其中自然连接&#xff08;NATURAL JOIN&#xff09;和USING连接是较为显著的两个特性。 自…

数据结构(14)——哈希表(1)

欢迎来到博主的专栏&#xff1a;数据结构 博主ID&#xff1a;代码小豪 文章目录 哈希表的思想映射方法&#xff08;哈希函数&#xff09;除留余数法 哈希表insert闭散列负载因子扩容find和erase 哈希表的思想 在以往的线性表中&#xff0c;查找速度取决于线性表是否有序&#…

知识库管理系统在企业数字化转型中的作用

引言 在数字化转型的浪潮中&#xff0c;企业正以前所未有的速度重塑其业务模式、运营流程和组织架构&#xff0c;以适应快速变化的市场环境和客户需求。这一过程中&#xff0c;知识库管理系统作为信息整合与知识共享的核心平台&#xff0c;发挥着举足轻重的作用&#xff0c;不…

【解决】AnimationCurve 运行时丢失数据问题

开发平台&#xff1a;Unity 2022 编程平台&#xff1a;Visual Studio 编程语言&#xff1a;CSharp   一、问题背景 如上图所示的 GracityComponent 组件中&#xff0c;引用 AnimationCurve 作为可调属性。但在实际使用中出现数据丢失问题。大致为以下两种情况&#xff1a; 运…

【重学 MySQL】二十七、七种 join 连接

【重学 MySQL】二十七、七种 join 连接 union 的使用UNION 的基本用法示例UNION ALL 的用法 七种 join 连接代码实现语法格式小结 union 的使用 UNION 在 SQL 中用于合并两个或多个 SELECT 语句的结果集&#xff0c;并默认去除重复的行。如果希望包含重复行&#xff0c;可以使…

RNN发展(RNN/LSTM/GRU/GNMT/transformer/RWKV)

RNN到GRU参考&#xff1a; https://blog.csdn.net/weixin_36378508/article/details/115101779 tRANSFORMERS参考&#xff1a; seq2seq到attention到transformer理解 GNMT 2016年9月 谷歌,基于神经网络的翻译系统&#xff08;GNMT&#xff09;&#xff0c;并宣称GNMT在多个主…

java程序员入行科目一之CRUD轻松入门教程(二)

封装工具类 封装获取连接&释放资源操作 在实际使用JDBC的时候&#xff0c;很多操作都是固定的&#xff0c;没有必要每次都去注册驱动&#xff0c;获取链接对象等等。 同样&#xff0c;释放资源的close操作也可以封装一下 下面是封装好的具体工具类 package com.jimihua.u…

海外云手机是否适合运营TikTok?

随着科技的迅猛发展&#xff0c;海外云手机逐渐成为改变工作模式的重要工具。这种基于云端技术的虚拟手机&#xff0c;不仅提供了更加便捷、安全的使用体验&#xff0c;还在电商引流和海外社媒管理等领域展示了其巨大潜力。那么&#xff0c;海外云手机究竟能否有效用于运营TikT…