123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166 |
- # greaseweazle/tools/read.py
- #
- # Greaseweazle control script: Read Disk to Image.
- #
- # 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 = "Read a disk to the specified image file."
- import sys
- import importlib
- from greaseweazle.tools import util
- from greaseweazle import error
- from greaseweazle import usb as USB
- from greaseweazle.flux import Flux
- def open_image(args, image_class):
- image = image_class.to_file(args.file)
- if args.rate is not None:
- image.bitrate = args.rate
- for opt, val in args.file_opts.items():
- error.check(hasattr(image, 'opts') and hasattr(image.opts, opt),
- "%s: Invalid file option: %s" % (args.file, opt))
- setattr(image.opts, opt, val)
- return image
- def read_and_normalise(usb, args, revs, ticks=0):
- flux = usb.read_track(revs=revs, ticks=ticks)
- if args.rpm is not None:
- flux.scale((60/args.rpm) / flux.time_per_rev)
- return flux
- def read_with_retry(usb, args, cyl, head, decoder):
- flux = read_and_normalise(usb, args, args.revs, args.ticks)
- if decoder is None:
- return flux
- dat = decoder(cyl, head, flux)
- if dat.nr_missing() != 0:
- for retry in range(3):
- print("T%u.%u: %s - %d sectors missing - Retrying (%d)"
- % (cyl, head, dat.summary_string(),
- dat.nr_missing(), retry+1))
- flux = read_and_normalise(usb, args, max(args.revs, 3))
- dat.decode_raw(flux)
- if dat.nr_missing() == 0:
- break
- return dat
- def print_summary(args, summary):
- s = 'Cyl-> '
- p = -1
- for c in args.tracks.cyls:
- s += ' ' if c//10==p else str(c//10)
- p = c//10
- print(s)
- s = 'H. S: '
- for c in args.tracks.cyls:
- s += str(c%10)
- print(s)
- tot_sec = good_sec = 0
- for head in args.tracks.heads:
- nsec = max(summary[x].nsec for x in summary if x[1] == head)
- for sec in range(nsec):
- print("%d.%2d: " % (head, sec), end="")
- for cyl in args.tracks.cyls:
- s = summary[cyl,head]
- if sec > s.nsec:
- print(" ", end="")
- else:
- tot_sec += 1
- if s.has_sec(sec): good_sec += 1
- print("." if s.has_sec(sec) else "X", end="")
- print()
- if tot_sec != 0:
- print("Found %d sectors of %d (%d%%)" %
- (good_sec, tot_sec, good_sec*100/tot_sec))
- def read_to_image(usb, args, image, decoder=None):
- """Reads a floppy disk and dumps it into a new image file.
- """
- args.ticks = 0
- if isinstance(args.revs, float):
- # Measure drive RPM.
- # We will adjust the flux intervals per track to allow for this.
- args.ticks = int(usb.read_track(2).ticks_per_rev * args.revs)
- args.revs = 2
- summary = dict()
- for t in args.tracks:
- cyl, head = t.cyl, t.head
- usb.seek(t.physical_cyl, head)
- dat = read_with_retry(usb, args, cyl, head, decoder)
- print("T%u.%u: %s" % (cyl, head, dat.summary_string()))
- summary[cyl,head] = dat
- image.emit_track(cyl, head, dat)
- if decoder is not None:
- print_summary(args, summary)
- 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 read (A,B,0,1,2)")
- parser.add_argument("--format", help="disk format")
- parser.add_argument("--revs", type=int,
- help="number of revolutions to read per track")
- parser.add_argument("--tracks", type=util.TrackSet,
- help="which tracks to read")
- parser.add_argument("--rate", type=int, help="data rate (kbit/s)")
- parser.add_argument("--rpm", type=int, help="convert drive speed to RPM")
- parser.add_argument("file", help="output filename")
- parser.description = description
- parser.prog += ' ' + argv[1]
- args = parser.parse_args(argv[2:])
- args.file, args.file_opts = util.split_opts(args.file)
- try:
- usb = util.usb_open(args.device)
- image_class = util.get_image_class(args.file)
- if not args.format and hasattr(image_class, 'default_format'):
- args.format = image_class.default_format
- decoder, def_tracks = None, None
- if args.format:
- try:
- mod = importlib.import_module('greaseweazle.codec.'
- + args.format)
- decoder = mod.decode_track
- except (ModuleNotFoundError, AttributeError) as ex:
- raise error.Fatal("Unknown format '%s'" % args.format) from ex
- def_tracks = util.TrackSet(mod.default_trackset)
- if args.revs is None: args.revs = mod.default_revs
- if def_tracks is None:
- def_tracks = util.TrackSet('c=0-81:h=0-1')
- if args.revs is None: args.revs = 3
- if args.tracks is not None:
- def_tracks.update_from_trackspec(args.tracks.trackspec)
- args.tracks = def_tracks
-
- print(("Reading %s revs=" % args.tracks) + str(args.revs))
- with open_image(args, image_class) as image:
- util.with_drive_selected(read_to_image, usb, args, image,
- decoder=decoder)
- except USB.CmdError as err:
- print("Command Failed: %s" % err)
- if __name__ == "__main__":
- main(sys.argv)
- # Local variables:
- # python-indent: 4
- # End:
|