-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathhwrcommunication.py
189 lines (152 loc) · 5.6 KB
/
hwrcommunication.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
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
'''
'''
import serial
import time
import os, sys
import re
import struct
import logging
# create logger instance
logger = logging.getLogger(__name__)
logger.setLevel(logging.INFO)
# create console handler and set level to debug
ch = logging.StreamHandler()
ch.setLevel(logging.DEBUG)
# create formatter
formatter = logging.Formatter('%(levelname)s - %(message)s')
# add formatter to ch
ch.setFormatter(formatter)
# add ch to logger
logger.addHandler(ch)
class HWRCommunication:
def __init__(self, comport, baudrate, commands_fn):
# create commands dict from C definitions
self.commands_dict = self.commands_to_dict(commands_fn)
#print(self.commands_dict)
# set start byte
self.start_byte = b'\x55'
# open serial port
try:
self.ser = serial.Serial(comport,
baudrate=baudrate,
timeout=0.5)
except:
print('Aborted')
def commands_to_dict(self, commands_fn):
dict_out = dict()
# Read file
file = open(commands_fn, 'r')
enum_in = file.read()
file.close()
# regular expression
data_re = re.compile(r' {4}(?P<name>\w+) = (?P<data>\d+)')
data_re_iter = data_re.finditer(enum_in)
for data in data_re_iter:
if data:
#print('{} = {}'.format(data.group('name'), data.group('data')))
dict_out[data.group('name')] = int(data.group('data'))
return dict_out
def flaotToBytes(self, float_val):
return bytes(struct.pack('f', float_val))
def bytesToFloat(self, bytes_val):
return struct.unpack('f', bytes_val)
def intToBytes(self, float_val):
return bytes(struct.pack('I', int(float_val)))
def bytesToInt(self, bytes_val):
return struct.unpack('I', bytes_val)
def packFrame(self, command_str, data_bytes = b'\x00\x00\x00\x00'):
command_byte = self.commands_dict[command_str].to_bytes(1, 'big')
length_byte = len(data_bytes).to_bytes(1, 'big')
subframe = self.start_byte + command_byte + length_byte + data_bytes
# perform checksum calculation
checksum = 0
for b in subframe:
checksum = checksum ^ int(b)
frame_bytes = subframe + checksum.to_bytes(1, 'big')
return frame_bytes
def unpackFrame(self, frame_bytes):
command = int(frame_bytes[1])
length = int(frame_bytes[2])
data_bytes = frame_bytes[3:-1]
# perform checksum calculation
checksum = 0
for b in frame_bytes[:-1]:
checksum = checksum ^ b
data = None
if checksum == frame_bytes[-1]:
if length == 4:
data = self.bytesToInt(data_bytes)[0]
else:
print('receive length error on cmd {}'.format(command))
else:
print('receive checksum error')
return data
def transmitReceiveFrame(self, frame_bytes):
if len(frame_bytes) < 128:
# while(self.ser.out_waiting > 0):
# pass
self.ser.write(frame_bytes)
time.sleep(.5)
msg = self.ser.read(len(frame_bytes))
else:
msg = b''
print('Frame too long')
logger.debug('TXmsg: 0x{}, RXmsg: 0x{}'.format(frame_bytes.hex(), msg.hex()))
return bytes(msg)
def send(self, command, data=None):
if any(command in s for s in list(self.commands_dict.keys())):
if data is not None:
frame_bytes = self.packFrame(command, self.intToBytes(float(data)))
else:
frame_bytes = self.packFrame(command)
print(frame_bytes.hex())
response_bytes = self.transmitReceiveFrame(frame_bytes)
print(response_bytes.hex())
ret = self.unpackFrame(response_bytes)
print('response of cammand {}'.format(ret))
else:
print('Command error!')
ret = None
return ret
def getAll(hwr):
strng = ''
for command in hwr.commands_dict:
if 'get' in str(command):
#print('perform command {}'.format(command))
resp = hwr.send(command)
if resp is not None:
strng = strng + '{0} = {1}{2:2.3f}\n'.format(command,(30-len(command))*' ', resp)
#time.sleep(0.01)
os.system('cls')
print(strng)
def run(hwr):
while (1):
resp = hwr.send('setFrequency', '200')
print('{} = {}'.format('setFrequency', resp))
time.sleep(1)
def main():
dir_path = os.path.dirname(os.path.realpath(__file__))
hwr = HWRCommunication(comport='COM8',
baudrate=9600,
commands_fn=(("%s/commands.h") % dir_path ))
if len(sys.argv) > 1:
if len(sys.argv) == 2:
if '-cmds' == sys.argv[1]:
for cmd in hwr.commands_dict.keys():
print(cmd)
elif '-run' == sys.argv[1]:
run(hwr)
else:
resp = hwr.send(sys.argv[1])
print('{} = {}'.format(str(sys.argv[1]), resp))
if len(sys.argv) == 3:
resp = hwr.send(sys.argv[1], sys.argv[2])
print('{} = {}'.format(str(sys.argv[1]), resp))
else:
#testMulti(hwr)
#getAll(hwr)
while (True):
getAll(hwr)
time.sleep(1)
if __name__ == "__main__":
main()