xref: /freebsd/usr.sbin/fdformat/fdformat.c (revision b3e7694832e81d7a904a10f525f8797b753bf0d3)
11de7b4b8SPedro F. Giffuni /*-
2*4d846d26SWarner Losh  * SPDX-License-Identifier: BSD-2-Clause
31de7b4b8SPedro F. Giffuni  *
4a5cef3b4SJoerg Wunsch  * Copyright (C) 1992-1994,2001 by Joerg Wunsch, Dresden
54f81180dSAndrey A. Chernov  * All rights reserved.
64f81180dSAndrey A. Chernov  *
74f81180dSAndrey A. Chernov  * Redistribution and use in source and binary forms, with or without
84f81180dSAndrey A. Chernov  * modification, are permitted provided that the following conditions
94f81180dSAndrey A. Chernov  * are met:
104f81180dSAndrey A. Chernov  * 1. Redistributions of source code must retain the above copyright
114f81180dSAndrey A. Chernov  *    notice, this list of conditions and the following disclaimer.
124f81180dSAndrey A. Chernov  * 2. Redistributions in binary form must reproduce the above copyright
134f81180dSAndrey A. Chernov  *    notice, this list of conditions and the following disclaimer in the
144f81180dSAndrey A. Chernov  *    documentation and/or other materials provided with the distribution.
154f81180dSAndrey A. Chernov  *
16538788f3SJoerg Wunsch  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR(S) ``AS IS'' AND ANY EXPRESS OR
17538788f3SJoerg Wunsch  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
18538788f3SJoerg Wunsch  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
19538788f3SJoerg Wunsch  * DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR(S) BE LIABLE FOR ANY DIRECT,
20538788f3SJoerg Wunsch  * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
21538788f3SJoerg Wunsch  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
22538788f3SJoerg Wunsch  * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23538788f3SJoerg Wunsch  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
24538788f3SJoerg Wunsch  * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
25538788f3SJoerg Wunsch  * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
26538788f3SJoerg Wunsch  * POSSIBILITY OF SUCH DAMAGE.
274f81180dSAndrey A. Chernov  */
284f81180dSAndrey A. Chernov 
291a6bed68SJoerg Wunsch #include <sys/types.h>
301a6bed68SJoerg Wunsch #include <sys/fdcio.h>
311a6bed68SJoerg Wunsch #include <sys/stat.h>
324f81180dSAndrey A. Chernov 
336c9d5885SPhilippe Charnier #include <ctype.h>
346c9d5885SPhilippe Charnier #include <err.h>
35e42ad56fSJoerg Wunsch #include <errno.h>
366c9d5885SPhilippe Charnier #include <fcntl.h>
371a37aa56SDavid E. O'Brien #include <paths.h>
384f81180dSAndrey A. Chernov #include <stdio.h>
394f81180dSAndrey A. Chernov #include <stdlib.h>
40467a0b06SMike Barcroft #include <string.h>
414f81180dSAndrey A. Chernov #include <strings.h>
421a6bed68SJoerg Wunsch #include <sysexits.h>
436c9d5885SPhilippe Charnier #include <unistd.h>
444f81180dSAndrey A. Chernov 
45e42ad56fSJoerg Wunsch #include "fdutil.h"
46e42ad56fSJoerg Wunsch 
474f81180dSAndrey A. Chernov static void
format_track(int fd,int cyl,int secs,int head,int rate,int gaplen,int secsize,int fill,int interleave,int offset)484f81180dSAndrey A. Chernov format_track(int fd, int cyl, int secs, int head, int rate,
491a6bed68SJoerg Wunsch 	     int gaplen, int secsize, int fill, int interleave,
501a6bed68SJoerg Wunsch 	     int offset)
514f81180dSAndrey A. Chernov {
524f81180dSAndrey A. Chernov 	struct fd_formb f;
531a6bed68SJoerg Wunsch 	int i, j, il[FD_MAX_NSEC + 1];
54bef69b45SAndrey A. Chernov 
55bef69b45SAndrey A. Chernov 	memset(il, 0, sizeof il);
561a6bed68SJoerg Wunsch 	for(j = 0, i = 1 + offset; i <= secs + offset; i++) {
571a6bed68SJoerg Wunsch 	    while(il[(j % secs) + 1])
581a6bed68SJoerg Wunsch 		    j++;
59bef69b45SAndrey A. Chernov 	    il[(j % secs) + 1] = i;
60bef69b45SAndrey A. Chernov 	    j += interleave;
61bef69b45SAndrey A. Chernov 	}
624f81180dSAndrey A. Chernov 
634f81180dSAndrey A. Chernov 	f.format_version = FD_FORMAT_VERSION;
644f81180dSAndrey A. Chernov 	f.head = head;
654f81180dSAndrey A. Chernov 	f.cyl = cyl;
664f81180dSAndrey A. Chernov 	f.transfer_rate = rate;
674f81180dSAndrey A. Chernov 
684f81180dSAndrey A. Chernov 	f.fd_formb_secshift = secsize;
694f81180dSAndrey A. Chernov 	f.fd_formb_nsecs = secs;
704f81180dSAndrey A. Chernov 	f.fd_formb_gaplen = gaplen;
714f81180dSAndrey A. Chernov 	f.fd_formb_fillbyte = fill;
724f81180dSAndrey A. Chernov 	for(i = 0; i < secs; i++) {
734f81180dSAndrey A. Chernov 		f.fd_formb_cylno(i) = cyl;
744f81180dSAndrey A. Chernov 		f.fd_formb_headno(i) = head;
75bef69b45SAndrey A. Chernov 		f.fd_formb_secno(i) = il[i+1];
764f81180dSAndrey A. Chernov 		f.fd_formb_secsize(i) = secsize;
774f81180dSAndrey A. Chernov 	}
78bcaa6ebcSRebecca Cran 	(void)ioctl(fd, FD_FORM, (caddr_t)&f);
794f81180dSAndrey A. Chernov }
804f81180dSAndrey A. Chernov 
814f81180dSAndrey A. Chernov static int
verify_track(int fd,int track,int tracksize)824f81180dSAndrey A. Chernov verify_track(int fd, int track, int tracksize)
834f81180dSAndrey A. Chernov {
841a6bed68SJoerg Wunsch 	static char *buf;
851a6bed68SJoerg Wunsch 	static int bufsz;
86538788f3SJoerg Wunsch 	int fdopts = -1, ofdopts, rv = 0;
87538788f3SJoerg Wunsch 
88538788f3SJoerg Wunsch 	if (ioctl(fd, FD_GOPTS, &fdopts) < 0)
896c9d5885SPhilippe Charnier 		warn("warning: ioctl(FD_GOPTS)");
90538788f3SJoerg Wunsch 	else {
91538788f3SJoerg Wunsch 		ofdopts = fdopts;
92538788f3SJoerg Wunsch 		fdopts |= FDOPT_NORETRY;
93538788f3SJoerg Wunsch 		(void)ioctl(fd, FD_SOPTS, &fdopts);
94538788f3SJoerg Wunsch 	}
954f81180dSAndrey A. Chernov 
961a6bed68SJoerg Wunsch 	if (bufsz < tracksize)
971a6bed68SJoerg Wunsch 		buf = realloc(buf, bufsz = tracksize);
98960536b6SMarcelo Araujo 	if (buf == NULL)
991a6bed68SJoerg Wunsch 		errx(EX_UNAVAILABLE, "out of memory");
1004f81180dSAndrey A. Chernov 	if (lseek (fd, (long) track * tracksize, 0) < 0)
101538788f3SJoerg Wunsch 		rv = -1;
102538788f3SJoerg Wunsch 	/* try twice reading it, without using the normal retrier */
103538788f3SJoerg Wunsch 	else if (read (fd, buf, tracksize) != tracksize
104538788f3SJoerg Wunsch 		 && read (fd, buf, tracksize) != tracksize)
105538788f3SJoerg Wunsch 		rv = -1;
106538788f3SJoerg Wunsch 	if (fdopts != -1)
107538788f3SJoerg Wunsch 		(void)ioctl(fd, FD_SOPTS, &ofdopts);
108538788f3SJoerg Wunsch 	return (rv);
1094f81180dSAndrey A. Chernov }
1104f81180dSAndrey A. Chernov 
1114f81180dSAndrey A. Chernov static void
usage(void)112538788f3SJoerg Wunsch usage (void)
1134f81180dSAndrey A. Chernov {
1141a6bed68SJoerg Wunsch 	errx(EX_USAGE,
1151a6bed68SJoerg Wunsch 	     "usage: fdformat [-F fill] [-f fmt] [-s fmtstr] [-nqvy] device");
1164f81180dSAndrey A. Chernov }
1174f81180dSAndrey A. Chernov 
1184f81180dSAndrey A. Chernov static int
yes(void)119538788f3SJoerg Wunsch yes (void)
1204f81180dSAndrey A. Chernov {
1214f81180dSAndrey A. Chernov 	char reply[256], *p;
1224f81180dSAndrey A. Chernov 
1234f81180dSAndrey A. Chernov 	reply[sizeof(reply) - 1] = 0;
1244f81180dSAndrey A. Chernov 	for (;;) {
1254f81180dSAndrey A. Chernov 		fflush(stdout);
1264f81180dSAndrey A. Chernov 		if (!fgets (reply, sizeof(reply) - 1, stdin))
1274f81180dSAndrey A. Chernov 			return (0);
1284f81180dSAndrey A. Chernov 		for (p=reply; *p==' ' || *p=='\t'; ++p)
1294f81180dSAndrey A. Chernov 			continue;
1304f81180dSAndrey A. Chernov 		if (*p=='y' || *p=='Y')
1314f81180dSAndrey A. Chernov 			return (1);
1324f81180dSAndrey A. Chernov 		if (*p=='n' || *p=='N' || *p=='\n' || *p=='\r')
1334f81180dSAndrey A. Chernov 			return (0);
1344f81180dSAndrey A. Chernov 		printf("Answer `yes' or `no': ");
1354f81180dSAndrey A. Chernov 	}
1364f81180dSAndrey A. Chernov }
1374f81180dSAndrey A. Chernov 
1384f81180dSAndrey A. Chernov int
main(int argc,char ** argv)1394f81180dSAndrey A. Chernov main(int argc, char **argv)
1404f81180dSAndrey A. Chernov {
1411a6bed68SJoerg Wunsch 	enum fd_drivetype type;
1421a6bed68SJoerg Wunsch 	struct fd_type fdt, newft, *fdtp;
1431a6bed68SJoerg Wunsch 	struct stat sb;
144e42ad56fSJoerg Wunsch #define MAXPRINTERRS 10
145e42ad56fSJoerg Wunsch 	struct fdc_status fdcs[MAXPRINTERRS];
1461a6bed68SJoerg Wunsch 	int format, fill, quiet, verify, verify_only, confirm;
1471a6bed68SJoerg Wunsch 	int fd, c, i, track, error, tracks_per_dot, bytes_per_track, errs;
14881f5cd99SJoerg Wunsch 	int flags;
1491a6bed68SJoerg Wunsch 	char *fmtstring, *device;
1501a6bed68SJoerg Wunsch 	const char *name, *descr;
1514f81180dSAndrey A. Chernov 
1521a6bed68SJoerg Wunsch 	format = quiet = verify_only = confirm = 0;
1531a6bed68SJoerg Wunsch 	verify = 1;
1541a6bed68SJoerg Wunsch 	fill = 0xf6;
1551a6bed68SJoerg Wunsch 	fmtstring = 0;
1561a6bed68SJoerg Wunsch 
1571a6bed68SJoerg Wunsch 	while((c = getopt(argc, argv, "F:f:nqs:vy")) != -1)
1584f81180dSAndrey A. Chernov 		switch(c) {
1591a6bed68SJoerg Wunsch 		case 'F':	/* fill byte */
1601a6bed68SJoerg Wunsch 			if (getnum(optarg, &fill)) {
1611a6bed68SJoerg Wunsch 				fprintf(stderr,
1621a6bed68SJoerg Wunsch 			"Bad argument %s to -F option; must be numeric\n",
1631a6bed68SJoerg Wunsch 					optarg);
1641a6bed68SJoerg Wunsch 				usage();
1651a6bed68SJoerg Wunsch 			}
1661a6bed68SJoerg Wunsch 			break;
1671a6bed68SJoerg Wunsch 
1684f81180dSAndrey A. Chernov 		case 'f':	/* format in kilobytes */
1691a6bed68SJoerg Wunsch 			if (getnum(optarg, &format)) {
1701a6bed68SJoerg Wunsch 				fprintf(stderr,
1711a6bed68SJoerg Wunsch 			"Bad argument %s to -f option; must be numeric\n",
1721a6bed68SJoerg Wunsch 					optarg);
1731a6bed68SJoerg Wunsch 				usage();
1741a6bed68SJoerg Wunsch 			}
1754f81180dSAndrey A. Chernov 			break;
1764f81180dSAndrey A. Chernov 
1771a6bed68SJoerg Wunsch 		case 'n':	/* don't verify */
1784f81180dSAndrey A. Chernov 			verify = 0;
1794f81180dSAndrey A. Chernov 			break;
1804f81180dSAndrey A. Chernov 
1811a6bed68SJoerg Wunsch 		case 'q':	/* quiet */
1821a6bed68SJoerg Wunsch 			quiet = 1;
1831a6bed68SJoerg Wunsch 			break;
1841a6bed68SJoerg Wunsch 
1851a6bed68SJoerg Wunsch 		case 's':	/* format string with detailed options */
1861a6bed68SJoerg Wunsch 			fmtstring = optarg;
1871a6bed68SJoerg Wunsch 			break;
1881a6bed68SJoerg Wunsch 
1891a6bed68SJoerg Wunsch 		case 'v':	/* verify only */
1904f81180dSAndrey A. Chernov 			verify = 1;
1914f81180dSAndrey A. Chernov 			verify_only = 1;
1924f81180dSAndrey A. Chernov 			break;
1934f81180dSAndrey A. Chernov 
1941a6bed68SJoerg Wunsch 		case 'y':	/* confirm */
1951a6bed68SJoerg Wunsch 			confirm = 1;
1961a6bed68SJoerg Wunsch 			break;
1971a6bed68SJoerg Wunsch 
1981a6bed68SJoerg Wunsch 		default:
1994f81180dSAndrey A. Chernov 			usage();
2004f81180dSAndrey A. Chernov 		}
2014f81180dSAndrey A. Chernov 
2024f81180dSAndrey A. Chernov 	if(optind != argc - 1)
2034f81180dSAndrey A. Chernov 		usage();
2044f81180dSAndrey A. Chernov 
2051a6bed68SJoerg Wunsch 	if (stat(argv[optind], &sb) == -1 && errno == ENOENT) {
2061a6bed68SJoerg Wunsch 		/* try prepending _PATH_DEV */
207624a58b6SEric Anholt 		device = malloc(strlen(argv[optind]) + sizeof(_PATH_DEV) + 1);
208960536b6SMarcelo Araujo 		if (device == NULL)
2091a6bed68SJoerg Wunsch 			errx(EX_UNAVAILABLE, "out of memory");
2101a6bed68SJoerg Wunsch 		strcpy(device, _PATH_DEV);
2111a6bed68SJoerg Wunsch 		strcat(device, argv[optind]);
2121a6bed68SJoerg Wunsch 		if (stat(device, &sb) == -1) {
2131a6bed68SJoerg Wunsch 			free(device);
2141a6bed68SJoerg Wunsch 			device = argv[optind]; /* let it fail below */
2151a6bed68SJoerg Wunsch 		}
2161a6bed68SJoerg Wunsch 	} else {
2171a6bed68SJoerg Wunsch 		device = argv[optind];
2184f81180dSAndrey A. Chernov 	}
2194f81180dSAndrey A. Chernov 
2201a6bed68SJoerg Wunsch 	if ((fd = open(device, O_RDWR | O_NONBLOCK)) < 0)
2211a6bed68SJoerg Wunsch 		err(EX_OSERR, "open(%s)", device);
2224f81180dSAndrey A. Chernov 
2231a6bed68SJoerg Wunsch 	/*
2241a6bed68SJoerg Wunsch 	 * Device initialization.
2251a6bed68SJoerg Wunsch 	 *
2261a6bed68SJoerg Wunsch 	 * First, get the device type descriptor.  This tells us about
2271a6bed68SJoerg Wunsch 	 * the media geometry data we need to format a medium.  It also
2281a6bed68SJoerg Wunsch 	 * lets us know quickly whether the device name actually points
2291a6bed68SJoerg Wunsch 	 * to a floppy disk drive.
2301a6bed68SJoerg Wunsch 	 *
2311a6bed68SJoerg Wunsch 	 * Then, obtain any drive options.  We're mainly interested to
2321a6bed68SJoerg Wunsch 	 * see whether we're currently working on a device with media
2331a6bed68SJoerg Wunsch 	 * density autoselection (FDOPT_AUTOSEL).  Then, we add the
2341a6bed68SJoerg Wunsch 	 * device option to tell the kernel not to log media errors,
2351a6bed68SJoerg Wunsch 	 * since we can handle them ourselves.  If the device does
2361a6bed68SJoerg Wunsch 	 * media density autoselection, we then need to set the device
2371a6bed68SJoerg Wunsch 	 * type appropriately, since by opening with O_NONBLOCK we
2381a6bed68SJoerg Wunsch 	 * told the driver to bypass media autoselection (otherwise we
2391a6bed68SJoerg Wunsch 	 * wouldn't stand a chance to format an unformatted or damaged
2401a6bed68SJoerg Wunsch 	 * medium).  We do not attempt to set the media type on any
2411a6bed68SJoerg Wunsch 	 * other devices since this is a privileged operation.  For the
2421a6bed68SJoerg Wunsch 	 * same reason, specifying -f and -s options is only possible
2431a6bed68SJoerg Wunsch 	 * for autoselecting devices.
2441a6bed68SJoerg Wunsch 	 *
2451a6bed68SJoerg Wunsch 	 * Finally, we are ready to turn off O_NONBLOCK, and start to
2461a6bed68SJoerg Wunsch 	 * actually format something.
2471a6bed68SJoerg Wunsch 	 */
2486c9d5885SPhilippe Charnier 	if(ioctl(fd, FD_GTYPE, &fdt) < 0)
2491a6bed68SJoerg Wunsch 		errx(EX_OSERR, "not a floppy disk: %s", device);
2501a6bed68SJoerg Wunsch 	if (ioctl(fd, FD_GDTYPE, &type) == -1)
2511a6bed68SJoerg Wunsch 		err(EX_OSERR, "ioctl(FD_GDTYPE)");
2521a6bed68SJoerg Wunsch 	if (format) {
2531a6bed68SJoerg Wunsch 		getname(type, &name, &descr);
2541a6bed68SJoerg Wunsch 		fdtp = get_fmt(format, type);
255960536b6SMarcelo Araujo 		if (fdtp == NULL)
2561a6bed68SJoerg Wunsch 			errx(EX_USAGE,
2571a6bed68SJoerg Wunsch 			    "unknown format %d KB for drive type %s",
2581a6bed68SJoerg Wunsch 			     format, name);
2591a6bed68SJoerg Wunsch 		fdt = *fdtp;
2604f81180dSAndrey A. Chernov 	}
2611a6bed68SJoerg Wunsch 	if (fmtstring) {
2621a6bed68SJoerg Wunsch 		parse_fmt(fmtstring, type, fdt, &newft);
2631a6bed68SJoerg Wunsch 		fdt = newft;
2641a6bed68SJoerg Wunsch 	}
2651a6bed68SJoerg Wunsch 	if (ioctl(fd, FD_STYPE, &fdt) < 0)
2661a6bed68SJoerg Wunsch 		err(EX_OSERR, "ioctl(FD_STYPE)");
2671a6bed68SJoerg Wunsch 	if ((flags = fcntl(fd, F_GETFL, 0)) == -1)
2681a6bed68SJoerg Wunsch 		err(EX_OSERR, "fcntl(F_GETFL)");
2691a6bed68SJoerg Wunsch 	flags &= ~O_NONBLOCK;
2701a6bed68SJoerg Wunsch 	if (fcntl(fd, F_SETFL, flags) == -1)
2711a6bed68SJoerg Wunsch 		err(EX_OSERR, "fcntl(F_SETFL)");
2724f81180dSAndrey A. Chernov 
2731a6bed68SJoerg Wunsch 	bytes_per_track = fdt.sectrac * (128 << fdt.secsize);
274d0fd62ecSKATO Takenori 
275d0fd62ecSKATO Takenori 	/* XXX  20/40 = 0.5 */
276d0fd62ecSKATO Takenori 	tracks_per_dot = (fdt.tracks * fdt.heads + 20) / 40;
2774f81180dSAndrey A. Chernov 
2784f81180dSAndrey A. Chernov 	if (verify_only) {
2794f81180dSAndrey A. Chernov 		if(!quiet)
2804f81180dSAndrey A. Chernov 			printf("Verify %dK floppy `%s'.\n",
2814f81180dSAndrey A. Chernov 				fdt.tracks * fdt.heads * bytes_per_track / 1024,
2824f0b1b78SJoerg Wunsch 				device);
2834f81180dSAndrey A. Chernov 	}
284d2c1dda3SPoul-Henning Kamp 	else if(!quiet && !confirm) {
2854f81180dSAndrey A. Chernov 		printf("Format %dK floppy `%s'? (y/n): ",
2864f81180dSAndrey A. Chernov 			fdt.tracks * fdt.heads * bytes_per_track / 1024,
2874f0b1b78SJoerg Wunsch 			device);
2884f81180dSAndrey A. Chernov 		if(!yes()) {
2894f81180dSAndrey A. Chernov 			printf("Not confirmed.\n");
2901a6bed68SJoerg Wunsch 			return (EX_UNAVAILABLE);
2914f81180dSAndrey A. Chernov 		}
2924f81180dSAndrey A. Chernov 	}
2934f81180dSAndrey A. Chernov 
2944f81180dSAndrey A. Chernov 	/*
2954f81180dSAndrey A. Chernov 	 * Formatting.
2964f81180dSAndrey A. Chernov 	 */
2974f81180dSAndrey A. Chernov 	if(!quiet) {
2984f81180dSAndrey A. Chernov 		printf("Processing ");
299d0fd62ecSKATO Takenori 		for (i = 0; i < (fdt.tracks * fdt.heads) / tracks_per_dot; i++)
300d0fd62ecSKATO Takenori 			putchar('-');
301d0fd62ecSKATO Takenori 		printf("\rProcessing ");
3024f81180dSAndrey A. Chernov 		fflush(stdout);
3034f81180dSAndrey A. Chernov 	}
3044f81180dSAndrey A. Chernov 
3054f81180dSAndrey A. Chernov 	error = errs = 0;
3064f81180dSAndrey A. Chernov 
3074f81180dSAndrey A. Chernov 	for (track = 0; track < fdt.tracks * fdt.heads; track++) {
3084f81180dSAndrey A. Chernov 		if (!verify_only) {
3096ae4c621SRodney W. Grimes 			format_track(fd, track / fdt.heads, fdt.sectrac,
3106ae4c621SRodney W. Grimes 				track % fdt.heads, fdt.trans, fdt.f_gap,
3111a6bed68SJoerg Wunsch 				fdt.secsize, fill, fdt.f_inter,
3121a6bed68SJoerg Wunsch 				track % fdt.heads? fdt.offset_side2: 0);
3134f81180dSAndrey A. Chernov 			if(!quiet && !((track + 1) % tracks_per_dot)) {
3144f81180dSAndrey A. Chernov 				putchar('F');
3154f81180dSAndrey A. Chernov 				fflush(stdout);
3164f81180dSAndrey A. Chernov 			}
3174f81180dSAndrey A. Chernov 		}
3184f81180dSAndrey A. Chernov 		if (verify) {
319e42ad56fSJoerg Wunsch 			if (verify_track(fd, track, bytes_per_track) < 0) {
320e42ad56fSJoerg Wunsch 				error = 1;
321e42ad56fSJoerg Wunsch 				if (errs < MAXPRINTERRS && errno == EIO) {
322e42ad56fSJoerg Wunsch 					if (ioctl(fd, FD_GSTAT, fdcs + errs) ==
323e42ad56fSJoerg Wunsch 					    -1)
3241a6bed68SJoerg Wunsch 						errx(EX_IOERR,
325e42ad56fSJoerg Wunsch 					"floppy IO error, but no FDC status");
326e42ad56fSJoerg Wunsch 					errs++;
327e42ad56fSJoerg Wunsch 				}
328e42ad56fSJoerg Wunsch 			}
3294f81180dSAndrey A. Chernov 			if(!quiet && !((track + 1) % tracks_per_dot)) {
3304f81180dSAndrey A. Chernov 				if (!verify_only)
3314f81180dSAndrey A. Chernov 					putchar('\b');
3324f81180dSAndrey A. Chernov 				if (error) {
3334f81180dSAndrey A. Chernov 					putchar('E');
3344f81180dSAndrey A. Chernov 					error = 0;
3354f81180dSAndrey A. Chernov 				}
3364f81180dSAndrey A. Chernov 				else
3374f81180dSAndrey A. Chernov 					putchar('V');
3384f81180dSAndrey A. Chernov 				fflush(stdout);
3394f81180dSAndrey A. Chernov 			}
3404f81180dSAndrey A. Chernov 		}
3414f81180dSAndrey A. Chernov 	}
3424f81180dSAndrey A. Chernov 	if(!quiet)
3434f81180dSAndrey A. Chernov 		printf(" done.\n");
3444f81180dSAndrey A. Chernov 
345e42ad56fSJoerg Wunsch 	if (!quiet && errs) {
346e42ad56fSJoerg Wunsch 		fflush(stdout);
347e42ad56fSJoerg Wunsch 		fprintf(stderr, "Errors encountered:\nCyl Head Sect   Error\n");
348e42ad56fSJoerg Wunsch 		for (i = 0; i < errs && i < MAXPRINTERRS; i++) {
349e42ad56fSJoerg Wunsch 			fprintf(stderr, " %2d   %2d   %2d   ",
350e42ad56fSJoerg Wunsch 				fdcs[i].status[3], fdcs[i].status[4],
351e42ad56fSJoerg Wunsch 				fdcs[i].status[5]);
352e42ad56fSJoerg Wunsch 			printstatus(fdcs + i, 1);
353e42ad56fSJoerg Wunsch 			putc('\n', stderr);
3544f81180dSAndrey A. Chernov 		}
355e42ad56fSJoerg Wunsch 		if (errs >= MAXPRINTERRS)
356e42ad56fSJoerg Wunsch 			fprintf(stderr, "(Further errors not printed.)\n");
357e42ad56fSJoerg Wunsch 	}
358e42ad56fSJoerg Wunsch 
359e42ad56fSJoerg Wunsch 	return errs != 0;
360e42ad56fSJoerg Wunsch }
361