用Python实现Modbus-RTU协议及串口调试(一)

用Python实现Modbus-RTU协议及串口调试(一)

最近由于要测试几块客户使用的现场仪表的通信(Modbus-RTU协议),就用Python写了个Modbus-RTU协议的串口调试模块,主要涉及了bytes类型字节串的使用,串口模块pyserial的使用,循环冗余校验CRC计算模块crcmod的使用,以及struct内置模块的使用。如果没有安装以上模块请按下面命令安装。

pip install pyserial
pip install crcmod

实现CRC16校验

首先按照Modbus-RTU协议的规范,所有通信帧末尾都要有2个字节的CRC16字节的校验码,以保证通信的可靠性。所以首先要实现CRC16校验算法,我们可以直接使用crcmod模块的已有算法,代码如下:

import crcmod
# CRC16校验,返回整型数
def crc16(veritydata):
    if not veritydata:
        return
    crc16 = crcmod.mkCrcFun(0x18005, rev=True, initCrc=0xFFFF, xorOut=0x0000)
    return crc16(veritydata)

其中函数参数veritydata是一个字节串bytes变量,表示要进行校验的数据。
当然你也可以自己按Modbus-RTU协议给出的c语言算法自己用Python写一个。
由于客户仪表主要使用了Modbus_RTU协议的03和04功能号,通过03和04功能号主机读取仪表从机的实时数据,用Python主要实现这两个功能号的协议,03和04功能号协议具体可以参考Modbus-RTU标准文档。

用函数实现Modbus-RTU的03和04功能主机->从机的命令帧

代码如下:

def mmodbus03or04(add, startregadd, regnum, funcode=3):
    if add < 0 or add > 0xFF or startregadd < 0 or startregadd > 0xFFFF or regnum < 1 or regnum > 0x7D:
        print("Error: parameter error")
        return
    if funcode != 3 and funcode != 4:
        print("Error: parameter error")
        return
    sendbytes = add.to_bytes(1, byteorder="big", signed=False)
    sendbytes = sendbytes + funcode.to_bytes(1, byteorder="big", signed=False) + startregadd.to_bytes(2, byteorder="big", signed=False) + \
                regnum.to_bytes(2, byteorder="big", signed=False)
    crcres = crc16(sendbytes)
    crc16bytes = crcres.to_bytes(2, byteorder="little", signed=False)
    sendbytes = sendbytes + crc16bytes
    return sendbytes

此函数一共有4个参数,含义如下。
add:Modbus从站地址(仪表地址)
startregadd:要读取的开始寄存器地址(从0开始的绝对地址)
regnum:要读取的寄存器个数
funcode:功能号,默认值是3
此函数首先判断了从站地址是否超限,开始寄存器地址是否超限,寄存器个数是否超限,功能号是否正确等。
然后将几个参数以及CRC16的校验码形成一个可以发送串口的字节串后返回。注意int整形的to_bytes成员函数,此函数用于将整形数转换为字节码,第一个参数是转换字节码的字节个数,第二个参数是转换是按大端模式(big)还是小端模式(little)。第三个参数标识整形数按有符号整形转换还是无符号整形转换。

将Modbus从机返回的数据帧的解析为数值的函数

代码如下:

def smodbus03or04(recvdata, valueformat=0, intsigned=False):
    if not recvdata:
        print("Error: data error")
        return
    if not checkcrc(recvdata):
        print("Error: crc error")
        return
    datalist = list(recvdata)
    if datalist[1] != 0x3 and datalist[1] != 0x4:
        print("Error: recv data funcode error")
        return
    bytenums = datalist[2]
    if bytenums % 2 != 0:
        print("Error: recv data reg data error")
        return
    retdata = []
    if valueformat == 0:
        floatnums = bytenums / 4
        print("float nums: ", str(floatnums))
        floatlist = [0, 0, 0, 0]
        for i in range(int(floatnums)):
            floatlist[1] = datalist[3+i*4]
            floatlist[0] = datalist[4+i*4]
            floatlist[3] = datalist[5+i*4]
            floatlist[2] = datalist[6+i*4]
            bfloatdata = bytes(floatlist)
            [fvalue] = struct.unpack('f', bfloatdata)
            retdata.append(fvalue)
            print(f'Data{i+1}: {fvalue:.3f}')
    elif valueformat == 1:
        shortintnums = bytenums / 2
        print("short int nums: ", str(shortintnums))
        for i in range(int(shortintnums)):
            btemp = recvdata[3+i*2:5+i*2]
            shortvalue = int.from_bytes(btemp, byteorder="big", signed=intsigned)
            retdata.append(shortvalue)
            print(f"Data{i+1}: {shortvalue}")
    return retdata

此函数的3个参数含义如下:
recvdata:Modbus-RTU从站在接收03和04功能号命令帧后的发回主站的数据帧,bytes字节串类型。
valueformat:寄存器中值的格式,0代表用2个寄存器4个字节表示一个单精度浮点数,1代表1个寄存器(2字节)存放1个16位整形值,默认为0,(仪表用的是单精度浮点数)
intsigned:当寄存器数据值格式是整形时,true则按照有符号整形转换,false则按无符号整形转换。
首先是对接收的数据帧的各种错误判断,CRC16校验值核对等判断。如果数据值格式是单精度浮点数则按照仪表规定的浮点数格式进行转换,先将接收的字节串转换为整形列表,然后根据浮点数个数进行迭代,将接收的数据转换为浮点数,并放入列表中返回,要注意的是如何将bytes类型转换为一个浮点数,这里使用了struct模块的unpack函数。将浮点数转换为bytes类型则使用struct模块的pack函数。具体使用方法请参考说明文档。当数据值格式是16位整形时,则将切片出来每个值对应的bytes字节串转换为整形数,并放到列表中返回。

用串口读取仪表数据

串口读写使用pyserial模块,代码如下:

if __name__ == '__main__':
    slaveadd = 1 	# modbus从站地址
    startreg = 0 	# 开始寄存器地址
    regnums = 40 	# 寄存器个数
    send_data = mmodbus03or04(slaveadd, startreg, regnums)
    print("send data : ", send_data.hex())
    com = serial.Serial("com3", 9600, timeout=0.8)
    starttime = time.time()
    com.write(send_data)
    recv_data = com.read(regnums*2+5)
    endtime = time.time()
    if len(recv_data) > 0:
        print("recv: ", recv_data.hex())
    print(f"used time: {endtime-starttime:.3f}")
    com.close()
    smodbus03or04(recv_data)

serial模块的Serial函数常用的3个参数是:串口号、波特率、超时时间(s)。这里在发送了命令帧后,根据命令帧中的寄存器个数计算出需要读取多少个字节的串口数据。读取后用解析函数得到具体数值。

完整的代码如下:

import serial
import crcmod
import time
import struct

# CRC16校验,返回整型数
def crc16(veritydata):
    if not veritydata:
        return
    crc16 = crcmod.mkCrcFun(0x18005, rev=True, initCrc=0xFFFF, xorOut=0x0000)
    return crc16(veritydata)

# 校验数据帧的CRC码是否正确
def checkcrc(data):
    if not data:
    	return False
    if len(data) <= 2:
        return False
    nocrcdata = data[:-2]
    oldcrc16 = data[-2:]
    oldcrclist = list(oldcrc16)
    crcres = crc16(nocrcdata)
    crc16byts = crcres.to_bytes(2, byteorder="little", signed=False)
    # print("CRC16:", crc16byts.hex())
    crclist = list(crc16byts)
    if oldcrclist[0] != crclist[0] or oldcrclist[1] != crclist[1]:
        return False
    return True

# Modbus-RTU协议的03或04读取保存或输入寄存器功能主-》从命令帧
def mmodbus03or04(add, startregadd, regnum, funcode=3):
    if add < 0 or add > 0xFF or startregadd < 0 or startregadd > 0xFFFF or regnum < 1 or regnum > 0x7D:
        print("Error: parameter error")
        return
    if funcode != 3 and funcode != 4:
        print("Error: parameter error")
        return
    sendbytes = add.to_bytes(1, byteorder="big", signed=False)
    sendbytes = sendbytes + funcode.to_bytes(1, byteorder="big", signed=False) + startregadd.to_bytes(2, byteorder="big", signed=False) + \
                regnum.to_bytes(2, byteorder="big", signed=False)
    crcres = crc16(sendbytes)
    crc16bytes = crcres.to_bytes(2, byteorder="little", signed=False)
    sendbytes = sendbytes + crc16bytes
    return sendbytes

# Modbus-RTU协议的03或04读取保持或输入寄存器功能从-》主的数据帧解析(浮点数2,1,4,3格式,16位短整形(定义正负数))
def smodbus03or04(recvdata, valueformat=0, intsigned=False):
    if not recvdata:
        print("Error: data error")
        return
    if not checkcrc(recvdata):
        print("Error: crc error")
        return
    datalist = list(recvdata)
    if datalist[1] != 0x3 and datalist[1] != 0x4:
        print("Error: recv data funcode error")
        return
    bytenums = datalist[2]
    if bytenums % 2 != 0:
        print("Error: recv data reg data error")
        return
    retdata = []
    if valueformat == 0:
        floatnums = bytenums / 4
        print("float nums: ", str(floatnums))
        floatlist = [0, 0, 0, 0]
        for i in range(int(floatnums)):
            floatlist[1] = datalist[3+i*4]
            floatlist[0] = datalist[4+i*4]
            floatlist[3] = datalist[5+i*4]
            floatlist[2] = datalist[6+i*4]
            bfloatdata = bytes(floatlist)
            [fvalue] = struct.unpack('f', bfloatdata)
            retdata.append(fvalue)
            print(f'Data{i+1}: {fvalue:.3f}')
    elif valueformat == 1:
        shortintnums = bytenums / 2
        print("short int nums: ", str(shortintnums))
        for i in range(int(shortintnums)):
            btemp = recvdata[3+i*2:5+i*2]
            shortvalue = int.from_bytes(btemp, byteorder="big", signed=intsigned)
            retdata.append(shortvalue)
            print(f"Data{i+1}: {shortvalue}")
    return retdata

if __name__ == '__main__':
    slaveadd = 1
    startreg = 0
    regnums = 40
    send_data = mmodbus03or04(slaveadd, startreg, regnums)
    print("send data : ", send_data.hex())
    com = serial.Serial("com3", 9600, timeout=0.8)
    starttime = time.time()
    com.write(send_data)
    recv_data = com.read(regnums*2+5)
    endtime = time.time()
    if len(recv_data) > 0:
        print("recv: ", recv_data.hex())
    print(f"used time: {endtime-starttime:.3f}")
    com.close()
    smodbus03or04(recv_data)

码字不易,如果本文对您有用请随手点个赞,谢谢!^_^

  • 94
    点赞
  • 178
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值