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