| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138 |
- PROJECT = "modbus_rtu"
- VERSION = "001.000.000"
- sys = require("sys")
- -- mobile.simid(2)
- -- Air780E的AT固件默认会为开机键防抖, 导致部分用户刷机很麻烦
- if rtos.bsp() == "EC618" and pm and pm.PWK_MODE then
- pm.power(pm.PWK_MODE, false)
- end
- -- 如果你用的是合宙DTU整机系列,才需要打开,否则按自己设计的PCB来
- -- gpio.setup(1, 1) -- 485转TTL芯片供电打开
- -- gpio.setup(24, 1) -- 外置供电电源打开
- modbus_rtu = require("modbus_rtu")
- -- 初始化modbus_rtu
- modbus_rtu.init({
- uartid = 1, -- 接收/发送数据的串口id
- baudrate = 4800, -- 波特率
- gpio_485 = 25, -- 转向GPIO编号
- tx_delay = 50000 -- 转向延迟时间,单位us
- -- 下面这些数据不填也行,不填底层默认为如下参数
- -- databits = 8,--数据位
- -- stopbits = 1,--停止位
- -- parity = uart.None,--校验位
- -- endianness = uart.LSB,--大小端
- -- buffer_size = 1024,--缓冲区大小
- -- rx_level = 0,--rx初始电平
- })
- -- 定义modbus_rtu数据接收回调
- local function on_modbus_rtu_receive(frame)
- log.info("modbus_rtu frame received:", json.encode(frame))
- if frame.fun == 0x03 then -- 功能码0x03表示读取保持寄存器
- local byte = frame.byte
- local payload = frame.payload
- -- log.info("modbus_rtu payload (hex):", payload:toHex())
- -- 解析数据(假设数据为16位寄存器值)
- local values_big = {} -- 大端序解析结果
- for i = 1, #payload, 2 do
- local msb = payload:byte(i)
- local lsb = payload:byte(i + 1)
- -- 大端序解析
- local result_big = (msb * 256) + lsb
- table.insert(values_big, result_big)
- end
- -- 输出大端序的解析结果
- log.info("输出大端序的解析结果:", table.concat(values_big, ", "))
- -- 第一个寄存器是湿度,第二个是温度,除以10以获取实际值
- if #values_big == 2 then
- log.info("测试同款485温湿度计")
- local humidity = values_big[1] / 10
- local temperature = values_big[2] / 10
- -- 打印湿度和温度
- log.info(string.format("湿度: %.1f%%", humidity))
- log.info(string.format("温度: %.1f°C", temperature))
- -- 发布湿度和温度
- sys.publish("modbus_rtu_data", {humidity, temperature})
- else
- log.info("用户自己的485下位机,共有" .. #values_big .. "组数据")
- for index, value in ipairs(values_big) do
- log.info(string.format("寄存器 %d: %d (大端序)", index, value))
- end
- end
- else
- log.info("功能码不是03")
- end
- end
- -- 设置modbus_rtu数据接收回调
- modbus_rtu.set_receive_callback(1, on_modbus_rtu_receive)
- local function send_modbus_rtu_command()
- local addr = 0x01 -- 设备地址,此处填客户自己的
- local fun = 0x03 -- 功能码(03为读取保持寄存器),此处填客户自己的
- local data = string.char(0x00, 0x00, 0x00, 0x02) -- 起始地址和寄存器数量(此处填客户自己的起始地址进而寄存器数量)
- -- modbus_rtu.send_command(1, addr, fun, data) -- 只发送一次命令并等待响应处理
- modbus_rtu.send_command(1, addr, fun, data, 5000) -- 循环5S发送一次
- end
- sys.taskInit(function()
- sys.wait(5000)
- send_modbus_rtu_command()
- end)
- -- local modbus_tcp = require("modbus_tcp")
- -- function modbus_tcp_test()
- -- sys.waitUntil("IP_READY")
- -- -- 连接到 Modbus TCP 服务器
- -- local netc, err = modbus_tcp.connect("112.125.89.8", 42514)
- -- if not netc then
- -- log.error("Modbus TCP", "连接失败:", err)
- -- return
- -- end
- -- -- 示例 1: 读取保持寄存器
- -- local response, err = modbus_tcp.send_request(netc, 1, 0x03, 0, 10) -- 读取从站地址为 1 的保持寄存器,起始地址为 0,数量为 10
- -- if not response then
- -- log.error("Modbus TCP", "读取保持寄存器失败:", err)
- -- else
- -- log.info("Modbus TCP", "读取保持寄存器响应数据:", response.data)
- -- end
- -- -- 示例 2: 写入单个寄存器
- -- local write_response, err = modbus_tcp.send_request(netc, 1, 0x06, 5, 1234) -- 向从站地址为 1 的寄存器地址 5 写入值 1234
- -- if not write_response then
- -- log.error("Modbus TCP", "写入单个寄存器失败:", err)
- -- else
- -- log.info("Modbus TCP", "写入单个寄存器响应数据:", write_response.data)
- -- end
- -- -- 关闭连接
- -- -- modbus_tcp.close(netc)
- -- end
- -- sys.taskInit()
- -- log.info("mem.lua", rtos.meminfo())
- -- log.info("mem.sys", rtos.meminfo("sys"))
- -- 用户代码已结束---------------------------------------------
- -- 结尾总是这一句
- sys.run()
- -- sys.run()之后后面不要加任何语句!!!!!
|