| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170 | # 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 sysimport importlibfrom greaseweazle.tools import utilfrom greaseweazle import errorfrom greaseweazle import usb as USBfrom greaseweazle.flux import Fluxdef 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 imagedef 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 fluxdef 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(args.retries):            print("T%u.%u: %s - Retrying (%d)"                  % (cyl, head, dat.summary_string(), retry+1))            flux = read_and_normalise(usb, args, max(args.revs, 3))            dat.decode_raw(flux)            if dat.nr_missing() == 0:                break    return datdef 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)        s = "T%u.%u: %s" % (cyl, head, dat.summary_string())        if hasattr(dat, 'nr_missing') and dat.nr_missing() != 0:            s += " - Giving up"        print(s)        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("--retries", type=int, default=3,                        help="number of retries on decode failure")    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:
 |