write.py 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186
  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, track
  13. from greaseweazle import usb as USB
  14. # Read and parse the image file.
  15. def open_image(args):
  16. cls = util.get_image_class(args.file)
  17. return cls.from_file(args.file)
  18. # write_from_image:
  19. # Writes the specified image file to floppy disk.
  20. def write_from_image(usb, args, image):
  21. # Measure drive RPM.
  22. # We will adjust the flux intervals per track to allow for this.
  23. drive = usb.read_track(2)
  24. del drive.list
  25. verified_count, not_verified_count = 0, 0
  26. for t in args.tracks:
  27. cyl, head = t.cyl, t.head
  28. track = image.get_track(cyl, head)
  29. if track is None and not args.erase_empty:
  30. continue
  31. print("\r%sing Track %u.%u..." %
  32. ("Writ" if track is not None else "Eras", cyl, head),
  33. end="", flush=True)
  34. usb.seek(t.physical_cyl, head)
  35. if track is None:
  36. usb.erase_track(drive.ticks_per_rev * 1.1)
  37. continue
  38. if args.precomp is not None:
  39. track.precomp = args.precomp.track_precomp(cyl)
  40. flux = track.flux_for_writeout()
  41. # @factor adjusts flux times for speed variations between the
  42. # read-in and write-out drives.
  43. factor = drive.ticks_per_rev / flux.index_list[0]
  44. # Convert the flux samples to Greaseweazle sample frequency.
  45. rem = 0.0
  46. flux_list = []
  47. for x in flux.list:
  48. y = x * factor + rem
  49. val = round(y)
  50. rem = y - val
  51. flux_list.append(val)
  52. # Encode the flux times for Greaseweazle, and write them out.
  53. verified = False
  54. for retry in range(args.retries+1):
  55. if retry != 0:
  56. print("T%u.%u: Verify Failure - Retry (%d)"
  57. % (cyl, head, retry))
  58. usb.write_track(flux_list = flux_list,
  59. cue_at_index = flux.index_cued,
  60. terminate_at_index = flux.terminate_at_index)
  61. try:
  62. no_verify = args.no_verify or track.verify is None
  63. except AttributeError: # track.verify undefined
  64. no_verify = True
  65. if no_verify:
  66. not_verified_count += 1
  67. verified = True
  68. break
  69. v_revs, v_ticks = track.verify_revs, 0
  70. if isinstance(v_revs, float):
  71. v_ticks = int(drive.ticks_per_rev * v_revs)
  72. v_revs = 2
  73. v_flux = usb.read_track(revs = v_revs, ticks = v_ticks)
  74. v_flux.scale(flux.time_per_rev / drive.time_per_rev)
  75. verified = track.verify.verify_track(v_flux)
  76. if verified:
  77. verified_count += 1
  78. break
  79. if retry == 0:
  80. print()
  81. error.check(verified, "Failed to verify Track %u.%u" % (cyl, head))
  82. print()
  83. if not_verified_count == 0:
  84. print("All tracks verified")
  85. else:
  86. if verified_count == 0:
  87. s = "No tracks verified "
  88. else:
  89. s = ("%d tracks verified; %d tracks *not* verified "
  90. % (verified_count, not_verified_count))
  91. s += ("(Reason: Verify %s)"
  92. % ("unavailable", "disabled")[args.no_verify])
  93. print(s)
  94. class PrecompSpec:
  95. def __str__(self):
  96. s = "Precomp %s" % track.Precomp.TYPESTRING[self.type]
  97. for e in self.list:
  98. s += ", %d-:%dns" % e
  99. return s
  100. def track_precomp(self, cyl):
  101. for c,s in reversed(self.list):
  102. if cyl >= c:
  103. return track.Precomp(self.type, s)
  104. return None
  105. def importspec(self, spec):
  106. self.list = []
  107. self.type = track.Precomp.MFM
  108. for x in spec.split(':'):
  109. k,v = x.split('=')
  110. if k == 'type':
  111. self.type = track.Precomp.TYPESTRING.index(v.upper())
  112. else:
  113. self.list.append((int(k), int(v)))
  114. self.list.sort()
  115. def __init__(self, spec):
  116. try:
  117. self.importspec(spec)
  118. except:
  119. raise ValueError
  120. def main(argv):
  121. parser = util.ArgumentParser(usage='%(prog)s [options] file')
  122. parser.add_argument("--device", help="greaseweazle device name")
  123. parser.add_argument("--drive", type=util.drive_letter, default='A',
  124. help="drive to write (A,B,0,1,2)")
  125. parser.add_argument("--tracks", type=util.TrackSet,
  126. help="which tracks to write")
  127. parser.add_argument("--erase-empty", action="store_true",
  128. help="erase empty tracks (default: skip)")
  129. parser.add_argument("--no-verify", action="store_true",
  130. help="disable verify")
  131. parser.add_argument("--retries", type=int, default=3,
  132. help="number of retries on verify failure")
  133. parser.add_argument("--precomp", type=PrecompSpec,
  134. help="write precompensation")
  135. parser.add_argument("file", help="input filename")
  136. parser.description = description
  137. parser.prog += ' ' + argv[1]
  138. args = parser.parse_args(argv[2:])
  139. try:
  140. usb = util.usb_open(args.device)
  141. image = open_image(args)
  142. tracks = util.TrackSet('c=0-81:h=0-1')
  143. if args.tracks is not None:
  144. tracks.update_from_trackspec(args.tracks.trackspec)
  145. args.tracks = tracks
  146. s = str(args.tracks)
  147. if args.precomp is not None:
  148. s += "; %s" % args.precomp
  149. print("Writing %s" % s)
  150. util.with_drive_selected(write_from_image, usb, args, image)
  151. except USB.CmdError as error:
  152. print("Command Failed: %s" % error)
  153. if __name__ == "__main__":
  154. main(sys.argv)
  155. # Local variables:
  156. # python-indent: 4
  157. # End: