-
Notifications
You must be signed in to change notification settings - Fork 28
/
uModBusFunctions.py
78 lines (53 loc) · 2.96 KB
/
uModBusFunctions.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
#Source: https://github.com/pycom/pycom-modbus/tree/master/uModbus (2018-07-16)
import uModBusConst as Const
import struct
def read_coils(starting_address, quantity):
if not (1 <= quantity <= 2000):
raise ValueError('invalid number of coils')
return struct.pack('>BHH', Const.READ_COILS, starting_address, quantity)
def read_discrete_inputs(starting_address, quantity):
if not (1 <= quantity <= 2000):
raise ValueError('invalid number of discrete inputs')
return struct.pack('>BHH', Const.READ_DISCRETE_INPUTS, starting_address, quantity)
def read_holding_registers(starting_address, quantity):
if not (1 <= quantity <= 125):
raise ValueError('invalid number of holding registers')
return struct.pack('>BHH', Const.READ_HOLDING_REGISTERS, starting_address, quantity)
def read_input_registers(starting_address, quantity):
if not (1 <= quantity <= 125):
raise ValueError('invalid number of input registers')
return struct.pack('>BHH', Const.READ_INPUT_REGISTER, starting_address, quantity)
def write_single_coil(output_address, output_value):
if output_value not in [0x0000, 0xFF00]:
raise ValueError('Illegal coil value')
return struct.pack('>BHH', Const.WRITE_SINGLE_COIL, output_address, output_value)
def write_single_register(register_address, register_value, signed=True):
fmt = 'h' if signed else 'H'
return struct.pack('>BH' + fmt, Const.WRITE_SINGLE_REGISTER, register_address, register_value)
def write_multiple_coils(starting_address, value_list):
sectioned_list = [value_list[i:i + 8] for i in range(0, len(value_list), 8)]
output_value=[]
for index, byte in enumerate(sectioned_list):
output = sum(v << i for i, v in enumerate(byte))
output_value.append(output)
fmt = 'B' * len(output_value)
return struct.pack('>BHHB' + fmt, Const.WRITE_MULTIPLE_COILS, starting_address,
len(value_list), (len(value_list) // 8) + 1, *output_value)
def write_multiple_registers(starting_address, register_values, signed=True):
quantity = len(register_values)
if not (1 <= quantity <= 123):
raise ValueError('invalid number of registers')
fmt = ('h' if signed else 'H') * quantity
return struct.pack('>BHHB' + fmt, Const.WRITE_MULTIPLE_REGISTERS, starting_address,
quantity, quantity * 2, *register_values)
def validate_resp_data(data, function_code, address, value=None, quantity=None, signed = True):
if function_code in [Const.WRITE_SINGLE_COIL, Const.WRITE_SINGLE_REGISTER]:
fmt = '>H' + ('h' if signed else 'H')
resp_addr, resp_value = struct.unpack(fmt, data)
if (address == resp_addr) and (value == resp_value):
return True
elif function_code in [Const.WRITE_MULTIPLE_COILS, Const.WRITE_MULTIPLE_REGISTERS]:
resp_addr, resp_qty = struct.unpack('>HH', data)
if (address == resp_addr) and (quantity == resp_qty):
return True
return False