forked from atc1441/ATC_MiThermometer
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathATCtelink.py
398 lines (374 loc) · 12.4 KB
/
ATCtelink.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
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
#!/usr/bin/env python
### ComSwireWriter.py ###
### Autor: pvvx ###
### Edited: Aaron Christophel ATCnetz.de ###
### Edit : Pila ###
import sys
import struct
import serial
import platform
import time
import argparse
import os
import io
import serial.tools.list_ports
__progname__ = 'TLSR825x Flasher, modified from ComSwireReader Utility'
__filename__ = 'ComSwireReader'
__version__ = "00.00.01"
COMPORT_MIN_BAUD_RATE=340000
COMPORT_DEF_BAUD_RATE=921600
USBCOMPORT_BAD_BAUD_RATE=11700000
debug = False
bit8mask = 0x20
class FatalError(RuntimeError):
def __init__(self, message):
RuntimeError.__init__(self, message)
@staticmethod
def WithResult(message, result):
message += " (result was %s)" % hexify(result)
return FatalError(message)
def arg_auto_int(x):
return int(x, 0)
def hex_dump(addr, blk):
print('%06x: ' % addr, end='')
for i in range(len(blk)):
if (i+1) % 16 == 0:
print('%02x ' % blk[i])
if i < len(blk) - 1:
print('%06x: ' % (addr + i + 1), end='')
else:
print('%02x ' % blk[i], end='')
if len(blk) % 16 != 0:
print('')
# encode data (blk) into 10-bit swire words
def sws_encode_blk(blk):
pkt=[]
d = bytearray(10) # word swire 10 bits
d[0] = 0x80 # start bit byte cmd swire = 1
for el in blk:
m = 0x80 # mask bit
idx = 1
while m != 0:
if (el & m) != 0:
d[idx] = 0x80
else:
d[idx] = 0xfe
idx += 1
m >>= 1
d[9] = 0xfe # stop bit swire = 0
pkt += d
d[0] = 0xfe # start bit next byte swire = 0
return pkt
# decode 9 bit swire response to byte (blk)
def sws_decode_blk(blk):
if (len(blk) == 9) and ((blk[8] & 0xfe) == 0xfe):
bitmask = bit8mask
data = 0;
for el in range(8):
data <<= 1
if (blk[el] & bitmask) == 0:
data |= 1
bitmask = 0x10
#print('0x%02x' % data)
return data
#print('Error blk:', blk)
return None
# encode a part of the read-by-address command (before the data read start bit) into 10-bit swire words
def sws_rd_addr(addr):
return sws_encode_blk(bytearray([0x5a, (addr>>16)&0xff, (addr>>8)&0xff, addr & 0xff, 0x80]))
# encode command stop into 10-bit swire words
def sws_code_end():
return sws_encode_blk([0xff])
# encode the command for writing data into 10-bit swire words
def sws_wr_addr(addr, data):
return sws_encode_blk(bytearray([0x5a, (addr>>16)&0xff, (addr>>8)&0xff, addr & 0xff, 0x00]) + bytearray(data)) + sws_encode_blk([0xff])
# send block to USB-COM
def wr_usbcom_blk(serialPort, blk):
# USB-COM chips throttle the stream into blocks at high speed!
if serialPort.baudrate > USBCOMPORT_BAD_BAUD_RATE:
i = 0
s = 60
l = len(blk)
while i < l:
if l - i < s:
s = l - i
i += serialPort.write(blk[i:i+s])
return i
return serialPort.write(blk)
# send and receive block to USB-COM
def rd_wr_usbcom_blk(serialPort, blk):
i = wr_usbcom_blk(serialPort, blk)
return i == len(serialPort.read(i))
# send swire command write to USB-COM
def sws_wr_addr_usbcom(serialPort, addr, data):
return wr_usbcom_blk(serialPort, sws_wr_addr(addr, data))
# send and receive swire command write to USB-COM
def rd_sws_wr_addr_usbcom(serialPort, addr, data):
i = wr_usbcom_blk(serialPort, sws_wr_addr(addr, data))
return i == len(serialPort.read(i))
# send and receive swire command read to USB-COM
def sws_read_data(serialPort, addr, size):
# A serialPort.timeout must be set !
serialPort.timeout = 0.01
# send addr and flag read
rd_wr_usbcom_blk(serialPort, sws_rd_addr(addr))
out=[]
# read size bytes
for i in range(size):
# send bit start read byte
serialPort.write([0xfe])
# read 9 bits swire, decode read byte
blk = serialPort.read(9)
# Added retry reading for Prolific PL-2303HX and ...
if len(blk) < 9:
blk += serialPort.read(10-len(blk))
x = sws_decode_blk(blk)
if x != None:
out += [x]
else:
if debug:
print('\r\nDebug: read swire byte:')
hex_dump(addr+i, blk)
# send stop read
rd_wr_usbcom_blk(serialPort, sws_code_end())
out = None
break
# send stop read
rd_wr_usbcom_blk(serialPort, sws_code_end())
return out
# set sws speed according to clk frequency and serialPort baud
def set_sws_speed(serialPort, clk):
#--------------------------------
# Set register[0x00b2]
print('SWire speed for CLK %.1f MHz... ' % (clk/1000000), end='')
swsdiv = int(round(clk*2/serialPort.baudrate))
if swsdiv > 0x7f:
print('Low UART baud rate!')
return False
byteSent = sws_wr_addr_usbcom(serialPort, 0x00b2, [swsdiv])
# print('Test SWM/SWS %d/%d baud...' % (int(serialPort.baudrate/5),int(clk/5/swsbaud)))
read = serialPort.read(byteSent)
if len(read) != byteSent:
if serialPort.baudrate > USBCOMPORT_BAD_BAUD_RATE and byteSent > 64 and len(read) >= 64 and len(read) < byteSent:
print('\n\r!!!!!!!!!!!!!!!!!!!BAD USB-UART Chip!!!!!!!!!!!!!!!!!!!')
print('UART Output:')
hex_dump(0,sws_wr_addr(0x00b2, [swsdiv]))
print('UART Input:')
hex_dump(0,read)
print('!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!')
return False
print('\n\rError: Wrong RX-TX connection!')
return False
#--------------------------------
# Test read register[0x00b2]
x = sws_read_data(serialPort, 0x00b2, 1)
#print(x)
if x != None and x[0] == swsdiv:
print('ok.')
if debug:
print('Debug: UART-SWS %d baud. SW-CLK ~%.1f MHz' % (int(serialPort.baudrate/10), serialPort.baudrate*swsdiv/2000000))
print('Debug: swdiv = 0x%02x' % (swsdiv))
return True
#--------------------------------
# Set default register[0x00b2]
rd_sws_wr_addr_usbcom(serialPort, 0x00b2, 0x05)
print('no')
return False
# auto set sws speed according to serialport baud
def set_sws_auto_speed(serialPort):
#---------------------------------------------------
# swsbaud = Fclk/5/register[0x00b2]
# register[0x00b2] = Fclk/5/swsbaud
# swsbaud = serialPort.baudrate/10
# register[0x00b2] = Fclk*2/serialPort.baudrate
# Fclk = 16000000..48000000 Hz
# serialPort.baudrate = 460800..3000000 bits/s
# register[0x00b2] = swsdiv = 10..208
#---------------------------------------------------
serialPort.timeout = 0.01 # A serialPort.timeout must be set !
if debug:
swsdiv_def = int(round(24000000*2/serialPort.baudrate))
print('Debug: default swdiv for 24 MHz = %d (0x%02x)' % (swsdiv_def, swsdiv_def))
swsdiv = int(round(16000000*2/serialPort.baudrate))
if swsdiv > 0x7f:
print('Low UART baud rate!')
return False
swsdiv_max = int(round(48000000*2/serialPort.baudrate))
#bit8m = (bit8mask + (bit8mask<<1) + (bit8mask<<2))&0xff
bit8m = 0x80 #((~(bit8mask-1))<<1)&0xff
while swsdiv <= swsdiv_max:
# register[0x00b2] = swsdiv
rd_sws_wr_addr_usbcom(serialPort, 0x00b2, [swsdiv])
# send addr and flag read
rd_wr_usbcom_blk(serialPort, sws_rd_addr(0x00b2))
# start read data
serialPort.write([0xfe])
# read 9 bits data
blk = serialPort.read(9)
# Added retry reading for Prolific PL-2303HX and ...
if len(blk) < 9:
blk += serialPort.read(9-len(blk))
# send stop read
rd_wr_usbcom_blk(serialPort, sws_code_end())
if debug:
print('Debug (read data):')
hex_dump(swsdiv, blk)
if len(blk) == 9 and blk[8] == 0xfe:
cmp = sws_encode_blk([swsdiv])
if debug:
print('Debug (check data):')
hex_dump(swsdiv+0xccc00, sws_encode_blk([swsdiv]))
if (blk[0]&bit8m) == bit8m and blk[1] == cmp[2] and blk[2] == cmp[3] and blk[4] == cmp[5] and blk[6] == cmp[7] and blk[7] == cmp[8]:
print('UART-SWS %d baud. SW-CLK ~%.1f MHz(?)' % (int(serialPort.baudrate/10), serialPort.baudrate*swsdiv/2000000))
return True
swsdiv += 1
if swsdiv > 0x7f:
print('Low UART baud rate!')
break
#--------------------------------
# Set default register[0x00b2]
rd_sws_wr_addr_usbcom(serialPort, 0x00b2, 0x05)
return False
def activate(serialPort, tact_ms):
#--------------------------------
# issue reset-to-bootloader:
# RTS = either RESET (active low = chip in reset)
# DTR = active low
print('Reset module (RTS low)...')
serialPort.setDTR(True)
serialPort.setRTS(True)
time.sleep(0.05)
serialPort.setDTR(False)
serialPort.setRTS(False)
#--------------------------------
# Stop CPU|: [0x0602]=5
print('Activate (%d ms)...' % tact_ms)
sws_wr_addr_usbcom(serialPort, 0x06f, 0x20) # soft reset mcu
blk = sws_wr_addr(0x0602, 0x05)
if tact_ms > 0:
tact = tact_ms/1000.0
t1 = time.time()
while time.time()-t1 < tact:
for i in range(5):
wr_usbcom_blk(serialPort, blk)
serialPort.reset_input_buffer()
#--------------------------------
# Duplication with syncronization
time.sleep(0.01)
serialPort.reset_input_buffer()
rd_wr_usbcom_blk(serialPort, sws_code_end())
rd_wr_usbcom_blk(serialPort, blk)
time.sleep(0.01)
serialPort.reset_input_buffer()
def flash_send_cmds(serialPort,data):
serialPort.write(sws_wr_addr(0x0d, bytearray([0x00]))) # cns low
for b in data:
serialPort.write(sws_wr_addr(0x0c, bytearray([b])))
serialPort.write(sws_wr_addr(0x0d, bytearray([0x01]))) # cns high
def flash_write_enable(serialPort):
flash_send_cmds(serialPort,[0x06])
def flash_write_addr(serialPort,addr, data):
flash_send_cmds(serialPort,bytearray([0x02, (addr >> 16) & 0xffff, (addr >> 8) & 0xff, addr & 0xff]) + bytearray(data))
def flash_erase_sector(serialPort,addr):
flash_send_cmds(serialPort,[0x20, (addr >> 16) & 0xffff, (addr >> 8) & 0xff, addr & 0xff])
def flash_erase_all(serialPort):
flash_send_cmds(serialPort,[0x60])
def main():
ports = serial.tools.list_ports.comports()
comport_def_name = ports[0].device
parser = argparse.ArgumentParser(description='%s version %s' % (__progname__, __version__), prog=__filename__)
parser.add_argument(
'--port', '-p',
help='Serial port device (default: '+comport_def_name+')',
default=comport_def_name)
parser.add_argument(
'--tact','-t',
help='Time Activation ms (0-off, default: 0 ms)',
type=arg_auto_int,
default=0)
parser.add_argument(
'--clk','-c',
help='SWire CLK (default: auto, 0 - auto)',
type=arg_auto_int,
default=0)
parser.add_argument(
'--baud','-b',
help='UART Baud Rate (default: '+str(COMPORT_DEF_BAUD_RATE)+', min: '+str(COMPORT_MIN_BAUD_RATE)+')',
type=arg_auto_int,
default=COMPORT_DEF_BAUD_RATE)
parser.add_argument(
'--debug','-d',
help='Debug info',
action="store_true")
parser.add_argument(
'file',
help='Filename to load (default: floader.bin)')
args = parser.parse_args()
global debug
debug = args.debug
global bit8mask
if args.baud > 1000000:
bit8mask = 0x40
if args.baud > 3000000:
bit8mask = 0x80
print('=======================================================')
print('%s version %s' % (__progname__, __version__))
print('-------------------------------------------------------')
if(args.baud < COMPORT_MIN_BAUD_RATE):
print ('The minimum speed of the COM port is %d baud!' % COMPORT_MIN_BAUD_RATE)
sys.exit(1)
print ('Open %s, %d baud...' % (args.port, args.baud))
try:
serialPort = serial.Serial(args.port,args.baud)
serialPort.reset_input_buffer()
# serialPort.flushInput()
# serialPort.flushOutput()
serialPort.timeout = 0.05
except:
print ('Error: Open %s, %d baud!' % (args.port, args.baud))
sys.exit(1)
if args.tact != 0:
# activate
activate(serialPort, args.tact);
if args.clk == 0:
# auto speed
if not set_sws_auto_speed(serialPort):
print('Chip sleep? -> Use reset chip (RTS-RST): see option --tact')
sys.exit(1)
else:
# Set SWS Speed = CLK/5/[0xb2] bits/s
if not set_sws_speed(serialPort, args.clk * 1000000):
if not set_sws_speed(serialPort, 16000000):
if not set_sws_speed(serialPort, 24000000):
if not set_sws_speed(serialPort, 32000000):
if not set_sws_speed(serialPort, 48000000):
print('Chip sleep? -> Use reset chip (RTS-RST): see option --tact')
sys.exit(1)
serialPort.timeout = 0.05 # A serialPort.timeout must be set !
flash_write_enable(serialPort)
flash_erase_all(serialPort)
print('Erasing flash ...')
time.sleep(15)
bytes_ = open(args.file, 'rb').read()
print('Started writing {} bytes'.format(hex(len(bytes_))))
i = len(bytes_)
curr_addr = 0
pkt_length = 0
while curr_addr < len(bytes_):
outbuffer = bytearray(0x100)
if i >= 0x100:
pkt_length = 0x100
else:
pkt_length = i
for x in range(pkt_length):
outbuffer[x] = bytes_[curr_addr + x]
flash_write_enable(serialPort)
flash_write_addr(serialPort,curr_addr, outbuffer)
print('Writing at {} ({}%)'.format(hex(curr_addr), int(100*curr_addr/len(bytes_))))
curr_addr += pkt_length
i -= pkt_length
print('Done, resetting CPU')
serialPort.write(sws_wr_addr(0x006f, [0x22])) # Reset CPU
sys.exit(1)
if __name__ == '__main__':
main()