2021-06-12 19:35:06 +00:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
|
2021-06-30 06:14:22 +00:00
|
|
|
import argparse, serial, struct, os
|
2021-06-12 19:35:06 +00:00
|
|
|
from typing import *
|
|
|
|
|
2021-06-21 14:22:01 +00:00
|
|
|
def auto_int(x):
|
|
|
|
return int(x, 0)
|
|
|
|
|
2021-06-12 19:55:09 +00:00
|
|
|
class RTOpt(NamedTuple):
|
2021-06-28 19:19:46 +00:00
|
|
|
type: Callable[[Any], Any]
|
2021-06-12 19:55:09 +00:00
|
|
|
optid: int
|
|
|
|
desc: str
|
|
|
|
|
2021-06-21 14:22:01 +00:00
|
|
|
supportmap = {
|
|
|
|
1: "CMSIS-DAP",
|
|
|
|
2: "UART",
|
|
|
|
4: "I2C-Tiny-USB",
|
|
|
|
8: "Temperature sensor",
|
|
|
|
|
|
|
|
0x80: "stdio USB-CDC debug interface"
|
|
|
|
}
|
|
|
|
|
2021-06-12 19:55:09 +00:00
|
|
|
option_table = {
|
2021-06-21 14:22:01 +00:00
|
|
|
'ctsrts': RTOpt(bool, 1, "Enable or disable CTS/RTS flow control (--ctsrts [true|false])"),
|
2021-06-28 21:08:00 +00:00
|
|
|
'i2ctemp': RTOpt(auto_int, 2,
|
|
|
|
"Control the builtin I2C temperature controller: get (0), disable (-1/0xff) "
|
|
|
|
"or set/enable (other) the current status and I2C bus address"),
|
2021-06-21 14:22:01 +00:00
|
|
|
'support': RTOpt(str, 0xff, "Get list of supported/implemented functionality"),
|
2021-06-12 19:55:09 +00:00
|
|
|
}
|
|
|
|
|
2021-06-12 19:35:06 +00:00
|
|
|
S_ACK = b'\x06'
|
|
|
|
S_NAK = b'\x15'
|
|
|
|
|
|
|
|
S_CMD_NOP = b'\x00'
|
|
|
|
S_CMD_Q_IFACE = b'\x01'
|
|
|
|
S_CMD_Q_CMDMAP = b'\x02'
|
|
|
|
S_CMD_Q_PGMNAME = b'\x03'
|
|
|
|
S_CMD_SYNCNOP = b'\x10'
|
|
|
|
S_CMD_MAGIC_SETTINGS = b'\x53'
|
|
|
|
|
2021-06-28 21:08:00 +00:00
|
|
|
|
2021-06-12 19:55:09 +00:00
|
|
|
def val2byte(t, v) -> int:
|
|
|
|
if t == bool:
|
|
|
|
return 1 if v else 0
|
2021-06-21 14:22:01 +00:00
|
|
|
if t == int or t == auto_int:
|
|
|
|
return 0xff if v < 0 else (v & 0xff)
|
|
|
|
if t == str:
|
|
|
|
return 0
|
2021-06-12 19:55:09 +00:00
|
|
|
|
|
|
|
assert False, "unimplemented type %s" % str(t)
|
|
|
|
|
2021-06-28 21:08:00 +00:00
|
|
|
|
2021-06-12 19:55:09 +00:00
|
|
|
def do_xfer(args, cmd:int, arg:int, port: str, baudrate:int=115200) -> Optional[int]:
|
2021-06-12 19:35:06 +00:00
|
|
|
with serial.Serial(port, baudrate, timeout=1) as ser:
|
|
|
|
cmdmap = [0]*32
|
|
|
|
syncok = False
|
|
|
|
for i in range(8):
|
|
|
|
ser.write(S_CMD_SYNCNOP)
|
|
|
|
a = ser.read()
|
|
|
|
b = ser.read()
|
|
|
|
if a == S_NAK and b == S_ACK:
|
|
|
|
syncok = True
|
|
|
|
break
|
|
|
|
|
|
|
|
if not syncok:
|
|
|
|
print("sync failed")
|
|
|
|
return None
|
|
|
|
|
|
|
|
ser.write(S_CMD_NOP)
|
|
|
|
if ser.read() != S_ACK:
|
|
|
|
print("nop failed")
|
|
|
|
return None
|
|
|
|
|
|
|
|
ser.write(S_CMD_Q_IFACE)
|
|
|
|
if ser.read() != S_ACK:
|
|
|
|
print("q_iface failed")
|
|
|
|
return None
|
|
|
|
serprogver = struct.unpack('<H', ser.read(2))[0]
|
|
|
|
if serprogver != 1:
|
|
|
|
print("unknown serprog protocol version %d" % serprogver)
|
|
|
|
return None
|
|
|
|
|
|
|
|
ser.write(S_CMD_Q_CMDMAP)
|
|
|
|
if ser.read() != S_ACK:
|
|
|
|
print("q_cmdmap failed")
|
|
|
|
return None
|
|
|
|
else:
|
|
|
|
cmdmap = ser.read(32)
|
|
|
|
|
|
|
|
if (cmdmap[S_CMD_MAGIC_SETTINGS[0] >> 3] & (1<<(S_CMD_MAGIC_SETTINGS[0]&7))) == 0:
|
|
|
|
print("serprog programmer has no S_CMD_MAGIC_SETTINGS")
|
|
|
|
return None
|
|
|
|
|
|
|
|
if (cmdmap[S_CMD_Q_PGMNAME[0] >> 3] & (1<<(S_CMD_Q_PGMNAME[0]&7))) != 0:
|
|
|
|
ser.write(S_CMD_Q_PGMNAME)
|
|
|
|
if ser.read() != S_ACK:
|
|
|
|
print("q_pgmname failed")
|
|
|
|
else:
|
|
|
|
name = ser.read(16).decode('utf-8')
|
2021-06-12 19:55:09 +00:00
|
|
|
if args.verbose: print("programmer is '%s'" % name)
|
2021-06-12 19:35:06 +00:00
|
|
|
|
|
|
|
ser.write(S_CMD_MAGIC_SETTINGS)
|
|
|
|
ser.write(bytes([cmd,arg]))
|
|
|
|
acknak = ser.read()
|
|
|
|
|
|
|
|
if acknak == S_ACK:
|
|
|
|
return ser.read()[0]
|
|
|
|
else:
|
|
|
|
print("settings command failed")
|
|
|
|
return None
|
|
|
|
|
2021-06-28 21:08:00 +00:00
|
|
|
|
2021-06-12 19:55:09 +00:00
|
|
|
def main():
|
2021-06-28 21:08:00 +00:00
|
|
|
parser = argparse.ArgumentParser(
|
|
|
|
prog="dbctl",
|
|
|
|
description="Runtime configuration control for DapperMime-JTAG")
|
2021-06-12 19:55:09 +00:00
|
|
|
|
2021-06-28 21:08:00 +00:00
|
|
|
parser.add_argument('tty', type=str, nargs='?', default="/dev/dragnbus-serprog",
|
|
|
|
help="Path to DapperMime-JTAG Serprog UART device [/dev/dragnbus-serprog]")
|
2021-06-12 19:55:09 +00:00
|
|
|
|
|
|
|
parser.add_argument('-v', '--verbose', default=False, action='store_true',
|
|
|
|
help="Verbose logging (for this utility)")
|
|
|
|
|
2021-06-30 06:14:22 +00:00
|
|
|
parser.add_argument('-f', '--flash', default=False, action='store_true',
|
|
|
|
help="Flash firmware to attached pico")
|
|
|
|
|
2021-06-12 19:55:09 +00:00
|
|
|
for k, v in option_table.items():
|
2021-06-21 14:22:01 +00:00
|
|
|
if k == "support":
|
|
|
|
parser.add_argument('--%s'%k, default=None, action='store_true',
|
|
|
|
help=v.desc)
|
|
|
|
else:
|
|
|
|
parser.add_argument('--%s'%k, type=v.type, nargs='?', default=None,
|
|
|
|
help=v.desc)
|
2021-06-12 19:55:09 +00:00
|
|
|
|
|
|
|
args = parser.parse_args()
|
|
|
|
|
2021-06-30 06:14:22 +00:00
|
|
|
if args.flash:
|
|
|
|
os.system("picotool load -x /usr/share/dragnbus/dragnbus.uf2")
|
|
|
|
return
|
|
|
|
|
2021-06-12 19:55:09 +00:00
|
|
|
for k, v in option_table.items():
|
|
|
|
if args.__dict__[k] is not None:
|
2021-06-28 21:08:00 +00:00
|
|
|
resp = do_xfer(args, v.optid, val2byte(v.type, args.__dict__[k]), args.tty)
|
2021-06-12 19:55:09 +00:00
|
|
|
if resp is None:
|
|
|
|
return 1
|
2021-06-21 14:22:01 +00:00
|
|
|
if k == "support":
|
|
|
|
print(", ".join(kvp[1] for kvp in supportmap.items() if (kvp[0] & resp) != 0))
|
|
|
|
else:
|
|
|
|
#if args.verbose:
|
|
|
|
print("-> %d" % resp)
|
2021-06-12 19:55:09 +00:00
|
|
|
|
|
|
|
return 0
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|