123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186 |
- # greaseweazle/tools/write.py
- #
- # Greaseweazle control script: Write Image to Disk.
- #
- # Written & released by Keir Fraser <keir.xen@gmail.com>
- #
- # This is free and unencumbered software released into the public domain.
- # See the file COPYING for more details, or visit <http://unlicense.org>.
- description = "Write a disk from the specified image file."
- import sys
- from greaseweazle.tools import util
- from greaseweazle import error, track
- from greaseweazle import usb as USB
- # Read and parse the image file.
- def open_image(args):
- cls = util.get_image_class(args.file)
- return cls.from_file(args.file)
- # write_from_image:
- # Writes the specified image file to floppy disk.
- def write_from_image(usb, args, image):
- # Measure drive RPM.
- # We will adjust the flux intervals per track to allow for this.
- drive = usb.read_track(2)
- del drive.list
- verified_count, not_verified_count = 0, 0
- for t in args.tracks:
- cyl, head = t.cyl, t.head
- track = image.get_track(cyl, head)
- if track is None and not args.erase_empty:
- continue
- print("\r%sing Track %u.%u..." %
- ("Writ" if track is not None else "Eras", cyl, head),
- end="", flush=True)
- usb.seek(t.physical_cyl, head)
- if track is None:
- usb.erase_track(drive.ticks_per_rev * 1.1)
- continue
- if args.precomp is not None:
- track.precomp = args.precomp.track_precomp(cyl)
- flux = track.flux_for_writeout()
- # @factor adjusts flux times for speed variations between the
- # read-in and write-out drives.
- factor = drive.ticks_per_rev / flux.index_list[0]
- # Convert the flux samples to Greaseweazle sample frequency.
- rem = 0.0
- flux_list = []
- for x in flux.list:
- y = x * factor + rem
- val = round(y)
- rem = y - val
- flux_list.append(val)
- # Encode the flux times for Greaseweazle, and write them out.
- verified = False
- for retry in range(args.retries+1):
- if retry != 0:
- print("T%u.%u: Verify Failure - Retry (%d)"
- % (cyl, head, retry))
- usb.write_track(flux_list = flux_list,
- cue_at_index = flux.index_cued,
- terminate_at_index = flux.terminate_at_index)
- try:
- no_verify = args.no_verify or track.verify is None
- except AttributeError: # track.verify undefined
- no_verify = True
- if no_verify:
- not_verified_count += 1
- verified = True
- break
- v_revs, v_ticks = track.verify_revs, 0
- if isinstance(v_revs, float):
- v_ticks = int(drive.ticks_per_rev * v_revs)
- v_revs = 2
- v_flux = usb.read_track(revs = v_revs, ticks = v_ticks)
- v_flux.scale(flux.time_per_rev / drive.time_per_rev)
- verified = track.verify.verify_track(v_flux)
- if verified:
- verified_count += 1
- break
- if retry == 0:
- print()
- error.check(verified, "Failed to verify Track %u.%u" % (cyl, head))
- print()
- if not_verified_count == 0:
- print("All tracks verified")
- else:
- if verified_count == 0:
- s = "No tracks verified "
- else:
- s = ("%d tracks verified; %d tracks *not* verified "
- % (verified_count, not_verified_count))
- s += ("(Reason: Verify %s)"
- % ("unavailable", "disabled")[args.no_verify])
- print(s)
- class PrecompSpec:
- def __str__(self):
- s = "Precomp %s" % track.Precomp.TYPESTRING[self.type]
- for e in self.list:
- s += ", %d-:%dns" % e
- return s
- def track_precomp(self, cyl):
- for c,s in reversed(self.list):
- if cyl >= c:
- return track.Precomp(self.type, s)
- return None
- def importspec(self, spec):
- self.list = []
- self.type = track.Precomp.MFM
- for x in spec.split(':'):
- k,v = x.split('=')
- if k == 'type':
- self.type = track.Precomp.TYPESTRING.index(v.upper())
- else:
- self.list.append((int(k), int(v)))
- self.list.sort()
- def __init__(self, spec):
- try:
- self.importspec(spec)
- except:
- raise ValueError
-
- def main(argv):
- parser = util.ArgumentParser(usage='%(prog)s [options] file')
- parser.add_argument("--device", help="greaseweazle device name")
- parser.add_argument("--drive", type=util.drive_letter, default='A',
- help="drive to write (A,B,0,1,2)")
- parser.add_argument("--tracks", type=util.TrackSet,
- help="which tracks to write")
- parser.add_argument("--erase-empty", action="store_true",
- help="erase empty tracks (default: skip)")
- parser.add_argument("--no-verify", action="store_true",
- help="disable verify")
- parser.add_argument("--retries", type=int, default=3,
- help="number of retries on verify failure")
- parser.add_argument("--precomp", type=PrecompSpec,
- help="write precompensation")
- parser.add_argument("file", help="input filename")
- parser.description = description
- parser.prog += ' ' + argv[1]
- args = parser.parse_args(argv[2:])
- try:
- usb = util.usb_open(args.device)
- image = open_image(args)
- tracks = util.TrackSet('c=0-81:h=0-1')
- if args.tracks is not None:
- tracks.update_from_trackspec(args.tracks.trackspec)
- args.tracks = tracks
- s = str(args.tracks)
- if args.precomp is not None:
- s += "; %s" % args.precomp
- print("Writing %s" % s)
- util.with_drive_selected(write_from_image, usb, args, image)
- except USB.CmdError as error:
- print("Command Failed: %s" % error)
- if __name__ == "__main__":
- main(sys.argv)
- # Local variables:
- # python-indent: 4
- # End:
|