raw_frame.lua 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258
  1. --[[
  2. @module raw_frame
  3. @summary TCP 主站应用模块(原始帧方式)
  4. @version 1.0
  5. @date 2025.12.18
  6. @author 马梦阳
  7. @usage
  8. 本功能模块演示的内容为:
  9. 1、将设备配置为 modbus TCP 主站模式
  10. 2、与从站 1 和 从站 2 进行通信
  11. 1. 对从站 1 进行 2 秒一次的读取保持寄存器 0-1 操作
  12. 2. 对从站 2 进行 4 秒一次的写入保持寄存器 0-1 操作
  13. 注意事项:
  14. 1、该示例程序需要搭配 exmodbus 扩展库使用
  15. 2、本功能模块只适合使用非标准 modbus TCP 请求报文格式的用户
  16. 3、如果你使用的是标准 modbus TCP 请求报文格式,请参考 param_field 功能模块
  17. 本文件没有对外接口,直接在 main.lua 中 require "raw_frame" 就可以加载运行;
  18. ]]
  19. local exmodbus = require("exmodbus")
  20. -- 创建 TCP 主站配置参数
  21. -- 说明:创建 TCP 主站时只需要配置如下参数即可
  22. local create_config = {
  23. -- 网络参数配置
  24. mode = exmodbus.TCP_MASTER, -- 通信模式:TCP主站
  25. adapter = socket.LWIP_ETH, -- 网卡 ID:LwIP 协议栈的以太网卡
  26. ip_address = "192.168.1.100", -- 从站 IP 地址
  27. port = 6000, -- 从站端口号
  28. }
  29. -- 初始化从站 1 数据结构
  30. -- 用于记录从站 1 保持寄存器 0-1 的值;
  31. local slave1_data = {}
  32. -- 配置读取从站 1 保持寄存器 0-1 的值;
  33. local read_config = {
  34. raw_request = string.char(
  35. 0x00, 0x01, -- 事务标识符
  36. 0x00, 0x00, -- 协议标识符
  37. 0x00, 0x06, -- 长度
  38. 0x01, -- 单元标识符(从站地址)
  39. 0x03, -- 功能码:读取保持寄存器
  40. 0x00, 0x00, -- 寄存器起始地址
  41. 0x00, 0x02 -- 寄存器数量
  42. ),
  43. timeout = 1000 -- 超时时间 1000 ms
  44. }
  45. -- 配置写入从站 2 保持寄存器 0-1 的值;
  46. local write_config = {
  47. raw_request = string.char(
  48. 0x00, 0x02, -- 事务标识符
  49. 0x00, 0x00, -- 协议标识符
  50. 0x00, 0x0B, -- 长度
  51. 0x02, -- 单元标识符(从站地址)
  52. 0x10, -- 功能码:写入保持寄存器
  53. 0x00, 0x00, -- 寄存器起始地址
  54. 0x00, 0x02, -- 寄存器数量
  55. 0x04, -- 字节数量
  56. 0x00, 0x12, -- 寄存器 0 的值
  57. 0x00, 0x34 -- 寄存器 1 的值
  58. ),
  59. timeout = 1000 -- 超时时间 1000 ms
  60. }
  61. -- 创建 TCP 主站实例
  62. local tcp_master = exmodbus.create(create_config)
  63. -- 判断主站是否创建成功并记录日志
  64. if not tcp_master then
  65. log.info("exmodbus_test", "tcp_master 创建失败")
  66. else
  67. log.info("exmodbus_test", "tcp_master 创建成功")
  68. end
  69. -- 读取从站 1 保持寄存器数据的函数
  70. local function read_slave1_holding_registers()
  71. log.info("exmodbus_test", "开始读取从站 1 保持寄存器 0-1 的值")
  72. -- 执行读取操作
  73. local read_result = tcp_master:read(read_config)
  74. -- 根据返回状态处理结果
  75. if read_result.status == exmodbus.STATUS_SUCCESS then
  76. local resp = read_result.raw_response
  77. -- 特别说明:
  78. -- 接下来的判断是针对 modbus TCP 标准响应格式的应答原始帧来解析的
  79. -- 在实际项目中,应根据自己项目中的实际应答原始帧格式进行解析
  80. -- 如果实际格式与此处演示的格式不一致,需要修改接下来的解析代码
  81. -- 1. 检查总长度:必须为 13 字节(7 MBAP头 + 1 功能码 + 1 字节数 + 4 数据)
  82. if #resp ~= 13 then
  83. log.info("exmodbus_test", "响应长度错误,期望 13 字节,实际:", #resp)
  84. return
  85. end
  86. -- 2. 检查事务标识符是否与请求一致
  87. local req_trans_id = string.unpack(">I2", read_config.raw_request, 1)
  88. local resp_trans_id = string.unpack(">I2", resp, 1)
  89. if req_trans_id ~= resp_trans_id then
  90. log.info("exmodbus_test", "事务标识符不一致,期望:", req_trans_id, "实际:", resp_trans_id)
  91. return
  92. end
  93. -- 3. 检查协议标识符是否为 0x0000
  94. if string.unpack(">I2", resp, 3) ~= 0x0000 then
  95. log.info("exmodbus_test", "协议标识符错误,期望 0x0000,实际:", string.unpack(">I2", resp, 3))
  96. return
  97. end
  98. -- 4. 检查单元标识符(从站地址)是否与请求一致
  99. local req_unit_id = string.byte(read_config.raw_request, 7)
  100. local resp_unit_id = string.byte(resp, 7)
  101. if req_unit_id ~= resp_unit_id then
  102. log.info("exmodbus_test", "单元标识符不一致,期望:", req_unit_id, "实际:", resp_unit_id)
  103. return
  104. end
  105. -- 5. 检查功能码是否与请求一致
  106. local req_func_code = string.byte(read_config.raw_request, 8)
  107. local resp_func_code = string.byte(resp, 8)
  108. if req_func_code ~= resp_func_code then
  109. log.info("exmodbus_test", "功能码不一致,期望:", req_func_code, "实际:", resp_func_code)
  110. return
  111. end
  112. -- 6. 检查字节数字段是否正确
  113. local byte_count = string.byte(resp, 9)
  114. if byte_count ~= 4 then
  115. log.info("exmodbus_test", "字节数字段错误,期望 4 字节,实际:", byte_count)
  116. return
  117. end
  118. -- 7. 解析寄存器数据(从第 10 字节开始,大端序)
  119. local data1 = string.unpack(">I2", resp, 10) -- 寄存器 0,偏移 10
  120. local data2 = string.unpack(">I2", resp, 12) -- 寄存器 1,偏移 12
  121. -- 8. 记录数据
  122. slave1_data[0] = data1
  123. slave1_data[1] = data2
  124. -- 9. 记录日志
  125. log.info("exmodbus_test", "成功读取到从站 1 保持寄存器 0-1 的值,寄存器 0 数值为", slave1_data[0], ",寄存器 1 数值为", slave1_data[1])
  126. elseif read_result.status == exmodbus.STATUS_TIMEOUT then
  127. log.info("exmodbus_test", "未收到从站 1 的响应(超时)")
  128. elseif read_result.status == exmodbus.STATUS_PARAM_INVALID then
  129. log.info("exmodbus_test", "读取从站 1 参数无效")
  130. end
  131. end
  132. -- 写入从站 2 保持寄存器数据的函数
  133. local function write_slave2_holding_registers()
  134. log.info("exmodbus_test", "开始写入从站 2 保持寄存器 0-1 的值")
  135. -- 执行写入操作
  136. local write_result = tcp_master:write(write_config)
  137. -- 根据返回状态处理结果
  138. if write_result.status == exmodbus.STATUS_SUCCESS then
  139. local resp = write_result.raw_response
  140. -- 特别说明:
  141. -- 接下来的判断是针对 modbus TCP 标准响应格式的应答原始帧来解析的
  142. -- 在实际项目中,应根据自己项目中的实际应答原始帧格式进行解析
  143. -- 如果实际格式与此处演示的格式不一致,需要修改接下来的解析代码
  144. -- 1. 检查总长度:必须为 12 字节(7 MBAP头 + 1 功能码 + 2 起始地址 + 2 寄存器数量)
  145. if #resp ~= 12 then
  146. log.info("exmodbus_test", "响应长度错误,期望 12 字节,实际:", #resp)
  147. return
  148. end
  149. -- 2. 检查事务标识符是否与请求一致
  150. local req_trans_id = string.unpack(">I2", write_config.raw_request, 1)
  151. local resp_trans_id = string.unpack(">I2", resp, 1)
  152. if req_trans_id ~= resp_trans_id then
  153. log.info("exmodbus_test", "事务标识符不一致,期望:", req_trans_id, "实际:", resp_trans_id)
  154. return
  155. end
  156. -- 3. 检查协议标识符是否为 0x0000
  157. if string.unpack(">I2", resp, 3) ~= 0x0000 then
  158. log.info("exmodbus_test", "协议标识符错误,期望 0x0000,实际:", string.unpack(">I2", resp, 3))
  159. return
  160. end
  161. -- 4. 检查单元标识符(从站地址)是否与请求一致
  162. local req_unit_id = string.byte(write_config.raw_request, 7)
  163. local resp_unit_id = string.byte(resp, 7)
  164. if req_unit_id ~= resp_unit_id then
  165. log.info("exmodbus_test", "单元标识符不一致,期望:", req_unit_id, "实际:", resp_unit_id)
  166. return
  167. end
  168. -- 5. 检查功能码是否与请求一致
  169. local req_func_code = string.byte(write_config.raw_request, 8)
  170. local resp_func_code = string.byte(resp, 8)
  171. if req_func_code ~= resp_func_code then
  172. log.info("exmodbus_test", "功能码不一致,期望:", req_func_code, "实际:", resp_func_code)
  173. return
  174. end
  175. -- 6. 检查起始地址是否与请求一致
  176. local req_start_addr = string.unpack(">I2", write_config.raw_request, 9)
  177. local resp_start_addr = string.unpack(">I2", resp, 9)
  178. if req_start_addr ~= resp_start_addr then
  179. log.info("exmodbus_test", "起始地址不一致,期望:", req_start_addr, "实际:", resp_start_addr)
  180. return
  181. end
  182. -- 7. 检查寄存器数量是否与请求一致
  183. local req_reg_count = string.unpack(">I2", write_config.raw_request, 11)
  184. local resp_reg_count = string.unpack(">I2", resp, 11)
  185. if req_reg_count ~= resp_reg_count then
  186. log.info("exmodbus_test", "寄存器数量不一致,期望:", req_reg_count, "实际:", resp_reg_count)
  187. return
  188. end
  189. log.info("exmodbus_test", "成功写入从站 2 保持寄存器 0-1")
  190. elseif write_result.status == exmodbus.STATUS_TIMEOUT then
  191. log.info("exmodbus_test", "未收到从站 2 的响应(超时)")
  192. elseif write_result.status == exmodbus.STATUS_PARAM_INVALID then
  193. log.info("exmodbus_test", "写入从站 2 参数无效")
  194. end
  195. end
  196. -- 定时任务函数:每 2 秒调用一次读取函数,每 4 秒调用一次写入函数
  197. local function task()
  198. local count = 0 -- 计数器
  199. while true do
  200. if tcp_master then
  201. -- 每 2 秒调用一次读取函数
  202. read_slave1_holding_registers()
  203. if count == 0 then
  204. -- 每 4 秒调用一次写入函数
  205. write_slave2_holding_registers()
  206. end
  207. count = (count + 1) % 2
  208. else
  209. log.info("exmodbus_test", "tcp_master 未创建,无法执行 read_slave1_holding_registers()")
  210. end
  211. sys.wait(2000)
  212. end
  213. end
  214. -- 初始化任务
  215. sys.taskInit(task)