write.py 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163
  1. # greaseweazle/tools/write.py
  2. #
  3. # Greaseweazle control script: Write Image to Disk.
  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. description = "Write a disk from the specified image file."
  10. import sys
  11. from greaseweazle.tools import util
  12. from greaseweazle import error
  13. from greaseweazle import usb as USB
  14. # Read and parse the image file.
  15. def open_image(args):
  16. image_class = util.get_image_class(args.file)
  17. if hasattr(image_class, 'from_filename'):
  18. image = image_class.from_filename(args.file)
  19. else:
  20. with open(args.file, "rb") as f:
  21. image = image_class.from_file(f.read())
  22. return image
  23. class Formatter:
  24. def __init__(self):
  25. self.length = 0
  26. def print(self, s):
  27. self.erase()
  28. self.length = len(s)
  29. print(s, end="", flush=True)
  30. def erase(self):
  31. l = self.length
  32. print("\b"*l + " "*l + "\b"*l, end="", flush=True)
  33. self.length = 0
  34. # write_from_image:
  35. # Writes the specified image file to floppy disk.
  36. def write_from_image(usb, args, image):
  37. # @drive_ticks is the time in Greaseweazle ticks between index pulses.
  38. # We will adjust the flux intervals per track to allow for this.
  39. flux = usb.read_track(2)
  40. drive_ticks = (flux.index_list[0] + flux.index_list[1]) / 2
  41. del flux
  42. verified_count, not_verified_count = 0, 0
  43. for cyl in range(args.scyl, args.ecyl+1):
  44. for side in range(0, args.nr_sides):
  45. track = image.get_track(cyl, side, writeout=True)
  46. if track is None and not args.erase_empty:
  47. continue
  48. print("\r%sing Track %u.%u..." %
  49. ("Writ" if track is not None else "Eras", cyl, side),
  50. end="", flush=True)
  51. usb.seek((cyl, cyl*2)[args.double_step], side)
  52. if track is None:
  53. usb.erase_track(drive_ticks * 1.1)
  54. continue
  55. flux = track.flux_for_writeout()
  56. # @factor adjusts flux times for speed variations between the
  57. # read-in and write-out drives.
  58. factor = drive_ticks / flux.index_list[0]
  59. # Convert the flux samples to Greaseweazle sample frequency.
  60. rem = 0.0
  61. flux_list = []
  62. for x in flux.list:
  63. y = x * factor + rem
  64. val = int(round(y))
  65. rem = y - val
  66. flux_list.append(val)
  67. # Encode the flux times for Greaseweazle, and write them out.
  68. formatter = Formatter()
  69. verified = False
  70. for retry in range(3):
  71. usb.write_track(flux_list, flux.terminate_at_index)
  72. try:
  73. no_verify = args.no_verify or track.verify is None
  74. except AttributeError: # track.verify undefined
  75. no_verify = True
  76. if no_verify:
  77. not_verified_count += 1
  78. verified = True
  79. break
  80. v_revs = 1 if track.splice == 0 else 2
  81. v_flux = usb.read_track(v_revs)
  82. v_flux.scale(flux.mean_index_time / v_flux.mean_index_time)
  83. verified = track.verify.verify_track(v_flux)
  84. if verified:
  85. verified_count += 1
  86. break
  87. formatter.print(" Retry %d" % (retry + 1))
  88. formatter.erase()
  89. error.check(verified, "Failed to write Track %u.%u" % (cyl, side))
  90. print()
  91. if not_verified_count == 0:
  92. print("All tracks verified")
  93. else:
  94. if verified_count == 0:
  95. s = "No tracks verified "
  96. else:
  97. s = ("%d tracks verified; %d tracks *not* verified "
  98. % (verified_count, not_verified_count))
  99. s += ("(Reason: Verify %s)"
  100. % ("unavailable", "disabled")[args.no_verify])
  101. print(s)
  102. def main(argv):
  103. parser = util.ArgumentParser(usage='%(prog)s [options] file')
  104. parser.add_argument("--device", help="greaseweazle device name")
  105. parser.add_argument("--drive", type=util.drive_letter, default='A',
  106. help="drive to write (A,B,0,1,2)")
  107. parser.add_argument("--scyl", type=int, default=0,
  108. help="first cylinder to write")
  109. parser.add_argument("--ecyl", type=int, default=81,
  110. help="last cylinder to write")
  111. parser.add_argument("--single-sided", action="store_true",
  112. help="single-sided write")
  113. parser.add_argument("--double-step", action="store_true",
  114. help="double-step drive heads")
  115. parser.add_argument("--erase-empty", action="store_true",
  116. help="erase empty tracks (default: skip)")
  117. parser.add_argument("--no-verify", action="store_true",
  118. help="disable verify")
  119. parser.add_argument("file", help="input filename")
  120. parser.description = description
  121. parser.prog += ' ' + argv[1]
  122. args = parser.parse_args(argv[2:])
  123. args.nr_sides = 1 if args.single_sided else 2
  124. try:
  125. usb = util.usb_open(args.device)
  126. image = open_image(args)
  127. try:
  128. util.with_drive_selected(write_from_image, usb, args, image)
  129. except:
  130. print()
  131. raise
  132. except USB.CmdError as error:
  133. print("Command Failed: %s" % error)
  134. if __name__ == "__main__":
  135. main(sys.argv)
  136. # Local variables:
  137. # python-indent: 4
  138. # End: