main.lua 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166
  1. -- LuaTools需要PROJECT和VERSION这两个信息
  2. PROJECT = "modbus_master_tcp"
  3. VERSION = "1.0.0"
  4. log.style(1)
  5. log.info("main", PROJECT, VERSION)
  6. -- sys库是标配
  7. _G.sys = require("sys")
  8. _G.sysplus = require("sysplus")
  9. --初始化网络
  10. log.info("ch390", "打开LDO供电")
  11. gpio.setup(20, 1) --打开lan供电
  12. mcu.hardfault(0) -- 死机后停机,一般用于调试状态
  13. require "lan"
  14. -- 开启调试模式
  15. modbus.debug(1)
  16. -- -- 关闭调试模式
  17. -- modbus.debug(0)
  18. -- 创建主站设备,TCP模式
  19. -- 设置连接方式为socket.LWIP_ETH
  20. -- 设置通讯间隔时间,主站将按每隔 设置时间 的频率向从站问询数据(默认100ms),当添加了多个从站后,主站向每个从站问询的时间间隔将叠加
  21. -- 设置通讯超时时间和消息发送超时重发次数,当主站未在 设置的时间 内接收到从站数据,将向从站再次发送问询(问询次数按设置的 消息超时重发次数 发送,默认1)
  22. -- 设置断线重连时间间隔,当从站与主站断连后,主站将在设置时间内重新连接从站(默认5000ms)
  23. mb_tcp = modbus.create_master(modbus.MODBUS_TCP, socket.LWIP_ETH,3000,1,5000)
  24. -- 为主站添加从站,从站ID为1,ip地址为 192.168.0.104,端口号为 6000,最多可添加247个从站
  25. mb_slave1 = modbus.add_slave(mb_tcp, 1, "192.168.4.100", 6000)
  26. -- 为主站添加从站,从站ID为2,ip地址为 192.168.0.104,端口号为 6001
  27. -- mb_slave2 = modbus.add_slave(mb_tcp, 2, "192.168.4.100", 6001)
  28. -- 为从站1创建数据存储区,并创建通讯消息,默认为自动loop模式
  29. slave1_msg1_buf = zbuff.create(1)
  30. mb_slave1_msg1 = modbus.create_msg(mb_tcp, mb_slave1, modbus.REGISTERS, modbus.READ, 0, 10, slave1_msg1_buf)
  31. slave1_msg1_buf:clear()
  32. -- -- 为从站1创建数据存储区,并创建通讯消息,如需要使用手动模式,须在这里设置为手动模式
  33. -- slave1_msg1_buf = zbuff.create(1)
  34. -- mb_slave1_msg1 = modbus.create_msg(mb_tcp, mb_slave1, modbus.REGISTERS, modbus.READ, 0, 10, slave1_msg1_buf,1,modbus.EXEC)
  35. -- slave1_msg1_buf:clear()
  36. -- 为从站2创建数据存储区,并创建通讯消息
  37. -- slave2_msg1_buf = zbuff.create(1)
  38. -- mb_slave2_msg1 = modbus.create_msg(mb_tcp, mb_slave2, modbus.REGISTERS, modbus.WRITE, 0, 10, slave2_msg1_buf)
  39. -- slave2_msg1_buf:clear()
  40. -- 启动Modubs设备
  41. modbus.master_start(mb_tcp)
  42. log.info("start modbus master")
  43. -- -- 设置通讯间隔时间,设置后主站将按每隔 设置时间 的频率向从站问询数据,当添加了多个从站后,主站向每个从站问询的时间间隔将叠加
  44. -- modbus.set_comm_interval_time(mb_tcp,3000)
  45. -- -- 设置通讯超时时间,当主站未在 设置的时间 内接收到从站数据,将向从站再次发送问询(问询次数按设置的 消息超时重发次数 发送)
  46. -- modbus.set_comm_timeout(mb_tcp, 3000)
  47. -- -- 设置消息发送失败、超时重发次数,如果主站在设置超时时间内未接收到数据,将按设置次数问询数据
  48. -- modbus.set_comm_resend_count(mb_tcp,1)
  49. -- 设置断线重连时间间隔,若从站与主站断连,主站将在设置时间内重新连接从站
  50. -- modbus.set_comm_reconnection_time(mb_tcp, 5000)
  51. -- -- 设置消息通讯周期,搭配modbus.create_master/modbus.set_comm_interval_time(mb_tcp,3000)设置通讯时间使用,若设置通讯周期为2次,将在2倍的通讯时间后向从站问询数据
  52. -- modbus.set_msg_comm_period(mb_slave1_msg1, 2)
  53. -- 获取所有从站状态,如果所有从站状态为正常,返回true,其他情况返回false,将在每隔5秒的时间获取所有从站状态,并在日志中打印状态(仅方便调试使用,量产时可删除)
  54. sys.timerLoopStart(function()
  55. local status = modbus.get_all_slave_state(mb_tcp)
  56. log.info("modbus", status)
  57. end, 5000)
  58. -- 获取从站1的状态,每隔5秒获取从站状态并在日志打印出来(仅方便调试使用,量产时可删除)
  59. sys.timerLoopStart(function()
  60. local status = modbus.get_slave_state(mb_slave1)
  61. log.info("modbus1", status)
  62. end, 5000)
  63. -- -- 获取从站2的状态,每隔5秒获取从站状态并在日志打印出来(仅方便调试使用,量产时可删除)
  64. -- sys.timerLoopStart(function()
  65. -- local status = modbus.get_slave_state(mb_slave2)
  66. -- log.info("modbus2", status)
  67. -- end, 5000)
  68. -- -- 每隔5秒执行一次 mb_slave1_msg1 消息,使用modbus.exec(master_handler, msg_handler)接口须先在modbus.set_msg_comm_period(msg_handler, comm_period)接口中设置为手动模式;成功返回true,其他情况返回false
  69. -- sys.timerLoopStart(function()
  70. -- local status=modbus.exec(mb_tcp, mb_slave1_msg1)
  71. -- log.info("msg",status)
  72. -- end,5000)
  73. -- -- 测试删除一个从站对象,并删除与之相关的通讯消息句柄。需在主站停止时(modbus.master_stop)执行该操作,否则无效。
  74. -- -- 将在3分钟后删除从站2(主站已关闭),删除与之相关的通讯消息句柄,并在5秒后重启主站,可以观察从站是否删除成功。
  75. -- sys.timerStart(function()
  76. -- local status = modbus.remove_slave(mb_tcp, mb_slave2)
  77. -- log.info("modbus", "slave2 remove after 3 minutes")
  78. -- log.info("remove", status)
  79. -- -- 移除从站后,5秒后重新启动Modbus主站
  80. -- sys.timerStart(function()
  81. -- modbus.master_start(mb_tcp)
  82. -- log.info("modbus", "Modbus master restarted after slave removal")
  83. -- end, 5000)
  84. -- end, 180000)
  85. -- 获取从站1的状态,每1秒获取一次数据并转换为JSON
  86. sys.timerLoopStart(function()
  87. -- 检查从站状态
  88. local status = modbus.get_slave_state(mb_slave1)
  89. if status == 0 then -- 0表示正常
  90. -- 读取缓冲区数据
  91. slave1_msg1_buf:seek(0) -- 重置指针到起始位置
  92. -- 读取4个寄存器的值(每个寄存器2字节)
  93. local reg1 = slave1_msg1_buf:readU16()
  94. local reg2 = slave1_msg1_buf:readU16()
  95. local reg3 = slave1_msg1_buf:readU16()
  96. local reg4 = slave1_msg1_buf:readU16()
  97. -- 创建数据表
  98. local data = {
  99. addr = 1, -- 从站地址
  100. fun = 3, -- 功能码03
  101. reg1 = reg1 / 10, -- 假设原始数据需要除以10得到实际值
  102. reg2 = reg2 / 10,
  103. reg3 = reg3 / 10,
  104. reg4 = reg4 / 10,
  105. timestamp = os.time() -- 添加时间戳
  106. }
  107. -- 转换为JSON
  108. local json_str = json.encode(data)
  109. log.info("Modbus数据转JSON:", json_str)
  110. else
  111. log.warn("从站1状态异常:", status)
  112. end
  113. end, 1000)
  114. -- -- 将在主站开启2分钟后停止modbus主站
  115. -- sys.timerStart(function()
  116. -- modbus.master_stop(mb_tcp)
  117. -- log.info("modbus", "Modbus stopped after 2 minutes")
  118. -- end, 120000)
  119. -- 用户代码已结束---------------------------------------------
  120. -- 结尾总是这一句
  121. sys.run()
  122. -- sys.run()之后后面不要加任何语句!!!!!