main.lua 6.8 KB

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