util.py 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198
  1. # greaseweazle/tools/util.py
  2. #
  3. # Greaseweazle control script: Utility functions.
  4. #
  5. # Written & released by Keir Fraser <keir.xen@gmail.com>
  6. #
  7. # This is free and unencumbered software released into the public domain.
  8. # See the file COPYING for more details, or visit <http://unlicense.org>.
  9. import argparse, os, sys, serial, struct, time
  10. import serial.tools.list_ports
  11. from greaseweazle import version
  12. from greaseweazle import error
  13. from greaseweazle import usb as USB
  14. from greaseweazle.image.scp import SCP
  15. from greaseweazle.image.hfe import HFE
  16. from greaseweazle.image.ipf import IPF
  17. class CmdlineHelpFormatter(argparse.ArgumentDefaultsHelpFormatter):
  18. def _get_help_string(self, action):
  19. help = action.help
  20. if '%no_default' in help:
  21. return help.replace('%no_default', '')
  22. if ('%(default)' in help
  23. or action.default is None
  24. or action.default is False
  25. or action.default is argparse.SUPPRESS):
  26. return help
  27. return help + ' (default: %(default)s)'
  28. class ArgumentParser(argparse.ArgumentParser):
  29. def __init__(self, formatter_class=CmdlineHelpFormatter, *args, **kwargs):
  30. return super().__init__(formatter_class=formatter_class,
  31. *args, **kwargs)
  32. def drive_letter(letter):
  33. types = {
  34. 'A': (USB.BusType.IBMPC, 0),
  35. 'B': (USB.BusType.IBMPC, 1),
  36. '0': (USB.BusType.Shugart, 0),
  37. '1': (USB.BusType.Shugart, 1),
  38. '2': (USB.BusType.Shugart, 2)
  39. }
  40. if not letter.upper() in types:
  41. raise argparse.ArgumentTypeError("invalid drive letter: '%s'" % letter)
  42. return types[letter.upper()]
  43. def get_image_class(name):
  44. image_types = { '.scp': SCP, '.hfe': HFE, '.ipf': IPF }
  45. _, ext = os.path.splitext(name)
  46. error.check(ext.lower() in image_types,
  47. "%s: Unrecognised file suffix '%s'" % (name, ext))
  48. return image_types[ext.lower()]
  49. def with_drive_selected(fn, usb, args, *_args, **_kwargs):
  50. usb.set_bus_type(args.drive[0])
  51. try:
  52. usb.drive_select(args.drive[1])
  53. usb.drive_motor(args.drive[1], True)
  54. fn(usb, args, *_args, **_kwargs)
  55. except KeyboardInterrupt:
  56. print()
  57. usb.reset()
  58. usb.ser.close()
  59. usb.ser.open()
  60. raise
  61. finally:
  62. usb.drive_motor(args.drive[1], False)
  63. usb.drive_deselect()
  64. def valid_ser_id(ser_id):
  65. return ser_id and ser_id.upper().startswith("GW")
  66. def find_port(old_port=None):
  67. # If we are reopening, and we know the location of the old port, require
  68. # to match on location.
  69. if old_port and old_port.location:
  70. for x in serial.tools.list_ports.comports():
  71. if x.location and x.location == old_port.location:
  72. return x.device
  73. raise serial.SerialException('Could not reopen Greaseweazle device')
  74. # Score each serial port
  75. best_score, best_port = 0, None
  76. for x in serial.tools.list_ports.comports():
  77. score = 0
  78. if x.manufacturer == "Keir Fraser" and x.product == "Greaseweazle":
  79. score = 20
  80. elif x.vid == 0x1209 and x.pid == 0x4d69:
  81. # Our very own properly-assigned PID. Guaranteed to be us.
  82. score = 20
  83. elif x.vid == 0x1209 and x.pid == 0x0001:
  84. # Our old shared Test PID. It's not guaranteed to be us.
  85. score = 10
  86. if score > 0 and valid_ser_id(x.serial_number):
  87. if not old_port or not valid_ser_id(old_port.serial_number):
  88. score = 20
  89. elif x.serial_number == old_port.serial_number:
  90. score = 30
  91. else:
  92. score = 0
  93. if score > best_score:
  94. best_score, best_port = score, x
  95. if best_port:
  96. return best_port.device
  97. raise serial.SerialException('Could not auto-probe Greaseweazle device')
  98. def port_info(devname):
  99. for x in serial.tools.list_ports.comports():
  100. if x.device == devname:
  101. return x
  102. return None
  103. def usb_reopen(usb, is_update):
  104. mode = { False: 1, True: 0 }
  105. try:
  106. usb.switch_fw_mode(mode[is_update])
  107. except (serial.SerialException, struct.error):
  108. # Mac and Linux raise SerialException ("... returned no data")
  109. # Win10 pyserial returns a short read which fails struct.unpack
  110. pass
  111. usb.ser.close()
  112. for i in range(10):
  113. time.sleep(0.5)
  114. try:
  115. devicename = find_port(usb.port_info)
  116. new_ser = serial.Serial(devicename)
  117. except serial.SerialException:
  118. # Device not found
  119. pass
  120. else:
  121. new_usb = USB.Unit(new_ser)
  122. new_usb.port_info = port_info(devicename)
  123. return new_usb
  124. raise serial.SerialException('Could not reopen port after mode switch')
  125. def usb_open(devicename, is_update=False, mode_check=True):
  126. if devicename is None:
  127. devicename = find_port()
  128. usb = USB.Unit(serial.Serial(devicename))
  129. usb.port_info = port_info(devicename)
  130. if not mode_check:
  131. return usb
  132. print("** %s v%u.%u [F%u], Host Tools v%u.%u"
  133. % (("Greaseweazle", "Bootloader")[usb.update_mode],
  134. usb.major, usb.minor, usb.hw_model,
  135. version.major, version.minor))
  136. if usb.update_mode and not is_update:
  137. if usb.hw_model == 7 and not usb.update_jumpered:
  138. usb = usb_reopen(usb, is_update)
  139. if not usb.update_mode:
  140. return usb
  141. print("Greaseweazle is in Firmware Update Mode:")
  142. print(" The only available action is \"update\" of main firmware")
  143. if usb.update_jumpered:
  144. print(" Remove the Update Jumper for normal operation")
  145. else:
  146. print(" Main firmware is erased: You *must* perform an update!")
  147. sys.exit(1)
  148. if is_update and not usb.update_mode:
  149. if usb.hw_model == 7:
  150. usb = usb_reopen(usb, is_update)
  151. error.check(usb.update_mode, """\
  152. Greaseweazle F7 did not change to Firmware Update Mode as requested.
  153. If the problem persists, install the Update Jumper (across RX/TX).""")
  154. return usb
  155. print("Greaseweazle is in Normal Mode:")
  156. print(" To \"update\" you must install the Update Jumper")
  157. sys.exit(1)
  158. if not usb.update_mode and usb.update_needed:
  159. print("Firmware is out of date: Require v%u.%u"
  160. % (version.major, version.minor))
  161. if usb.hw_model == 7:
  162. print("Run \"update <update_file>\"")
  163. else:
  164. print("Install the Update Jumper and \"update <update_file>\"")
  165. sys.exit(1)
  166. return usb
  167. # Local variables:
  168. # python-indent: 4
  169. # End: