summaryrefslogtreecommitdiff
path: root/usr.sbin/bad144/bad144.c
diff options
context:
space:
mode:
Diffstat (limited to 'usr.sbin/bad144/bad144.c')
-rw-r--r--usr.sbin/bad144/bad144.c250
1 files changed, 102 insertions, 148 deletions
diff --git a/usr.sbin/bad144/bad144.c b/usr.sbin/bad144/bad144.c
index 5a19c907e5a..d49cfa407b6 100644
--- a/usr.sbin/bad144/bad144.c
+++ b/usr.sbin/bad144/bad144.c
@@ -1,3 +1,4 @@
+/* $OpenBSD: bad144.c,v 1.14 2002/06/10 00:40:53 miod Exp $ */
/*
* Copyright (c) 1980, 1986, 1988, 1993
* The Regents of the University of California. All rights reserved.
@@ -39,7 +40,7 @@ static char copyright[] =
#ifndef lint
/*static char sccsid[] = "from: @(#)bad144.c 8.1 (Berkeley) 6/6/93";*/
-static char *rcsid = "$Id: bad144.c,v 1.13 2002/06/09 02:38:11 itojun Exp $";
+static char *rcsid = "$Id: bad144.c,v 1.14 2002/06/10 00:40:53 miod Exp $";
#endif /* not lint */
/*
@@ -68,10 +69,11 @@ static char *rcsid = "$Id: bad144.c,v 1.13 2002/06/09 02:38:11 itojun Exp $";
#include <stdio.h>
#include <paths.h>
#include <string.h>
+#include <err.h>
#define RETRIES 10 /* number of retries on reading old sectors */
-int fflag, add, copy, verbose, nflag;
+int add, copy, verbose, nflag;
int dups;
int badfile = -1; /* copy of badsector table to use, -1 if any */
#define MAXSECSIZE 1024
@@ -91,6 +93,7 @@ int checkold(void);
int compare(const void *, const void *);
daddr_t getold(int, struct dkbad *);
void shift(int, int, int);
+void usage(void);
int
main(argc, argv)
@@ -98,57 +101,38 @@ main(argc, argv)
char *argv[];
{
struct bt_bad *bt;
- daddr_t sn, bn[126];
- int i, f, nbad, new, bad, errs;
-
- argc--, argv++;
- while (argc > 0 && **argv == '-') {
- (*argv)++;
- while (**argv) {
- switch (**argv) {
-#if __vax__
- case 'f':
- fflag++;
- break;
-#endif
- case 'a':
- add++;
- break;
- case 'c':
- copy++;
- break;
- case 'v':
- verbose++;
- break;
- case 'n':
- nflag++;
- verbose++;
- break;
- default:
- if (**argv >= '0' && **argv <= '4') {
- badfile = **argv - '0';
- break;
- }
- goto usage;
- }
- (*argv)++;
+ daddr_t sn, bn[NBT_BAD];
+ int i, f, nbad, new, bad, errs, ch;
+
+ while ((ch = getopt(argc, argv, "acnv01234")) != -1) {
+ switch(ch) {
+ case 'a':
+ add = 1;
+ break;
+ case 'c':
+ copy = 1;
+ break;
+ case 'n':
+ nflag = 1;
+ /* FALLTHROUGH */
+ case 'v':
+ verbose = 1;
+ break;
+ case '0':
+ case '1':
+ case '2':
+ case '3':
+ case '4':
+ badfile = ch - '0';
+ break;
}
- argc--, argv++;
- }
- if (argc < 1) {
-usage:
- fprintf(stderr,
- "usage: bad144 [ -f ] disk [ snum [ bn ... ] ]\n");
- fprintf(stderr,
- "to read or overwrite bad-sector table, e.g.: bad144 hp0\n");
- fprintf(stderr,
- "or bad144 -a [ -f ] [ -c ] disk bn ...\n");
- fprintf(stderr, "where options are:\n");
- fprintf(stderr, "\t-a add new bad sectors to the table\n");
- fprintf(stderr, "\t-f reformat listed sectors as bad\n");
- fprintf(stderr, "\t-c copy original sector to replacement\n");
- exit(1);
}
+ argc -= optind;
+ argv += optind;
+
+ if (argc < 1)
+ usage();
+
if (argv[0][0] != '/')
(void)snprintf(name, sizeof(name), "%sr%s%c", _PATH_DEV,
argv[0], 'a' + RAW_PART);
@@ -156,37 +140,25 @@ usage:
strlcpy(name, argv[0], sizeof(name));
f = open(name, argc == 1? O_RDONLY : O_RDWR);
if (f < 0)
- Perror(name);
-#ifdef was
- if (read(f, label, sizeof(label)) < 0)
- Perror("read");
- for (dp = (struct disklabel *)(label + LABELOFFSET);
- dp < (struct disklabel *)
- (label + sizeof(label) - sizeof(struct disklabel));
- dp = (struct disklabel *)((char *)dp + 64))
- if (dp->d_magic == DISKMAGIC && dp->d_magic2 == DISKMAGIC)
- break;
-#else
+ err(4, "can't open %s", name);
+
/* obtain label and adjust to fit */
dp = (struct disklabel *)&label;
if (ioctl(f, DIOCGDINFO, dp) < 0)
- Perror("ioctl DIOCGDINFO");
-#endif
+ err(4, "DIOCGDINFO");
+
if (dp->d_magic != DISKMAGIC || dp->d_magic2 != DISKMAGIC
/* dkcksum(lp) != 0 */ ) {
- fprintf(stderr, "Bad pack magic number (pack is unlabeled)\n");
- exit(1);
+ errx(1, "bad pack magic number (pack is unlabeled)");
}
if (dp->d_secsize > MAXSECSIZE || dp->d_secsize <= 0) {
- fprintf(stderr, "Disk sector size too large/small (%d)\n",
+ errx(7, "Disk sector size too large/small (%d)",
dp->d_secsize);
- exit(7);
}
-#ifdef __i386__
if (dp->d_type == DTYPE_SCSI) {
- fprintf(stderr, "SCSI disks don't use bad144!\n");
- exit(1);
+ errx(1, "SCSI disks don't need bad144!");
}
+#ifdef __i386__
/* are we inside a DOS partition? */
if (dp->d_partitions[0].p_offset) {
/* yes, rules change. assume bad tables at end of partition C,
@@ -195,6 +167,7 @@ usage:
} else
#endif
size = dp->d_nsectors * dp->d_ntracks * dp->d_ncylinders;
+
argc--;
argv++;
if (argc == 0) {
@@ -216,7 +189,7 @@ usage:
break;
}
bt = oldbad.bt_bad;
- for (i = 0; i < 126; i++) {
+ for (i = 0; i < NBT_BAD; i++) {
bad = (bt->bt_cyl<<16) + bt->bt_trksec;
if (bad < 0)
break;
@@ -237,22 +210,21 @@ usage:
i = checkold();
if (verbose)
printf("Had %d bad sectors, adding %d\n", i, argc);
- if (i + argc > 126) {
- printf("bad144: not enough room for %d more sectors\n",
- argc);
- printf("limited to 126 by information format\n");
- exit(1);
+ if (i + argc > NBT_BAD) {
+ errx(1, "not enough room for %d more sectors, "
+ "limited to %d by information format",
+ argc, NBT_BAD);
}
curbad = oldbad;
} else {
- curbad.bt_csn = atoi(*argv++);
+ curbad.bt_csn = strtol(*argv++, NULL, 10);
argc--;
curbad.bt_mbz = 0;
curbad.bt_flag = DKBAD_MAGIC;
- if (argc > 126) {
- printf("bad144: too many bad sectors specified\n");
- printf("limited to 126 by information format\n");
- exit(1);
+ if (argc > NBT_BAD) {
+ errx(1, "too many bad sectors specified, "
+ "limited to %d by information format",
+ NBT_BAD);
}
i = 0;
}
@@ -277,7 +249,7 @@ usage:
if (errs)
exit(1);
nbad = i;
- while (i < 126) {
+ while (i < NBT_BAD) {
curbad.bt_bad[i].bt_trksec = -1;
curbad.bt_bad[i].bt_cyl = -1;
i++;
@@ -291,9 +263,8 @@ usage:
qsort((char *)curbad.bt_bad, nbad, sizeof (struct bt_bad),
compare);
if (dups) {
- fprintf(stderr,
-"bad144: bad sectors have been duplicated; can't add existing sectors\n");
- exit(3);
+ errx(3, "bad sectors have been duplicated; "
+ "can't add existing sectors");
}
shift(f, nbad, nbad-new);
}
@@ -304,37 +275,26 @@ usage:
for (; i < 10 && i < dp->d_nsectors; i += 2) {
if (lseek(f, dp->d_secsize * (size - dp->d_nsectors + i),
SEEK_SET) < 0)
- Perror("lseek");
+ err(4, "lseek");
if (verbose)
printf("write badsect file at %d\n",
size - dp->d_nsectors + i);
if (nflag == 0 && write(f, (caddr_t)&curbad, sizeof(curbad)) !=
sizeof(curbad)) {
- char msg[80];
- (void)snprintf(msg, sizeof msg,
- "bad144: write bad sector file %d",
- i/2);
- perror(msg);
+ warn("write bad sector file %d", i/2);
}
if (badfile != -1)
break;
}
-#ifdef vax
- if (nflag == 0 && fflag)
- for (i = nbad - new; i < nbad; i++)
- format(f, bn[i]);
-#endif
#ifdef DIOCSBAD
if (nflag == 0 && ioctl(f, DIOCSBAD, (caddr_t)&curbad) < 0)
- fprintf(stderr,
- "Can't sync bad-sector file; reboot for changes to take effect\n");
+ warn("can't sync bad-sector file");
+ warnx("reboot for changes to take effect");
#endif
if ((dp->d_flags & D_BADSECT) == 0 && nflag == 0) {
dp->d_flags |= D_BADSECT;
if (ioctl(f, DIOCWDINFO, dp) < 0) {
- perror("label");
- fprintf(stderr, "Can't write label to enable bad sector handling\n");
- exit(1);
+ err(1, "can't write label to enable bad sector handling");
}
}
exit(0);
@@ -342,11 +302,11 @@ usage:
daddr_t
getold(f, bad)
-struct dkbad *bad;
+ int f;
+ struct dkbad *bad;
{
int i;
daddr_t sn;
- char msg[80];
if (badfile == -1)
i = 0;
@@ -355,20 +315,17 @@ struct dkbad *bad;
for (; i < 10 && i < dp->d_nsectors; i += 2) {
sn = size - dp->d_nsectors + i;
if (lseek(f, sn * dp->d_secsize, SEEK_SET) < 0)
- Perror("lseek");
+ err(4, "lseek");
if (read(f, (char *) bad, dp->d_secsize) == dp->d_secsize) {
if (i > 0)
printf("Using bad-sector file %d\n", i/2);
return(sn);
}
- (void)snprintf(msg, sizeof msg,
- "bad144: read bad sector file at sn %d", sn);
- perror(msg);
+ warn("read bad sector file at sn %d", sn);
if (badfile != -1)
break;
}
- fprintf(stderr, "bad144: %s: can't read bad block info\n", name);
- exit(1);
+ errx(1, "can't read bad block info from %s", name);
/*NOTREACHED*/
}
@@ -381,24 +338,22 @@ checkold()
int errors = 0, warned = 0;
if (oldbad.bt_flag != DKBAD_MAGIC) {
- fprintf(stderr, "bad144: %s: bad flag in bad-sector table\n",
- name);
+ warnx("bad flag in bad-sector table %s", name);
errors++;
}
if (oldbad.bt_mbz != 0) {
- fprintf(stderr, "bad144: %s: bad magic number\n", name);
+ warnx("%s: bad magic number", name);
errors++;
}
bt = oldbad.bt_bad;
- for (i = 0; i < 126; i++, bt++) {
+ for (i = 0; i < NBT_BAD; i++, bt++) {
if (bt->bt_cyl == 0xffff && bt->bt_trksec == 0xffff)
break;
if ((bt->bt_cyl >= dp->d_ncylinders) ||
((bt->bt_trksec >> 8) >= dp->d_ntracks) ||
((bt->bt_trksec & 0xff) >= dp->d_nsectors)) {
- fprintf(stderr,
- "bad144: cyl/trk/sect out of range in existing entry: ");
- fprintf(stderr, "sn=%d, cn=%d, tn=%d, sn=%d\n",
+ warnx("cyl/trk/sect out of range in existing entry: "
+ "sn=%d, cn=%d, tn=%d, sn=%d",
badsn(bt), bt->bt_cyl, bt->bt_trksec>>8,
bt->bt_trksec & 0xff);
errors++;
@@ -407,14 +362,12 @@ checkold()
(bt->bt_trksec >> 8)) *
dp->d_nsectors + (bt->bt_trksec & 0xff);
if (i > 0 && sn < lsn && !warned) {
- fprintf(stderr,
- "bad144: bad sector file is out of order\n");
+ warnx("bad sector file is out of order");
errors++;
warned++;
}
if (i > 0 && sn == lsn) {
- fprintf(stderr,
- "bad144: bad sector file contains duplicates (sn %d)\n",
+ warnx("bad sector file contains duplicates (sn %d)",
sn);
errors++;
}
@@ -432,6 +385,7 @@ checkold()
*/
void
shift(f, new, old)
+ int f, new, old;
{
daddr_t repl;
@@ -469,37 +423,33 @@ char *buf;
*/
int
blkcopy(f, s1, s2)
-daddr_t s1, s2;
+ int f;
+ daddr_t s1, s2;
{
int tries, n;
- if (buf == (char *)NULL) {
+ if (buf == NULL) {
buf = malloc((unsigned)dp->d_secsize);
- if (buf == (char *)NULL) {
- fprintf(stderr, "Out of memory\n");
- exit(20);
+ if (buf == NULL) {
+ err(20, NULL);
}
}
for (tries = 0; tries < RETRIES; tries++) {
if (lseek(f, dp->d_secsize * s1, SEEK_SET) < 0)
- Perror("lseek");
+ err(4, "lseek");
if ((n = read(f, buf, dp->d_secsize)) == dp->d_secsize)
break;
}
if (n != dp->d_secsize) {
- fprintf(stderr, "bad144: can't read sector, %d: ", s1);
- if (n < 0)
- perror((char *)0);
+ warnx("can't read sector %d", s1);
return(0);
}
if (lseek(f, dp->d_secsize * s2, SEEK_SET) < 0)
- Perror("lseek");
+ err(4, "lseek");
if (verbose)
printf("copying %d to %d\n", s1, s2);
if (nflag == 0 && write(f, buf, dp->d_secsize) != dp->d_secsize) {
- fprintf(stderr,
- "bad144: can't write replacement sector, %d: ", s2);
- perror((char *)0);
+ warn("can't write replacement sector %d", s2);
return(0);
}
return(1);
@@ -509,24 +459,22 @@ char *zbuf;
void
blkzero(f, sn)
-daddr_t sn;
+ int f;
+ daddr_t sn;
{
- if (zbuf == (char *)NULL) {
+ if (zbuf == NULL) {
zbuf = malloc((unsigned)dp->d_secsize);
- if (zbuf == (char *)NULL) {
- fprintf(stderr, "Out of memory\n");
- exit(20);
+ if (zbuf == NULL) {
+ err(20, NULL);
}
}
if (lseek(f, dp->d_secsize * sn, SEEK_SET) < 0)
- Perror("lseek");
+ err(4, "lseek");
if (verbose)
printf("zeroing %d\n", sn);
if (nflag == 0 && write(f, zbuf, dp->d_secsize) != dp->d_secsize) {
- fprintf(stderr,
- "bad144: can't write replacement sector, %d: ", sn);
- perror((char *)0);
+ warn("can't write replacement sector %d", sn);
}
}
@@ -703,10 +651,16 @@ format(fd, blk)
#endif
void
-Perror(op)
- const char *op;
+usage()
{
- fprintf(stderr, "bad144: ");
- perror(op);
- exit(4);
+ fprintf(stderr,
+ "usage: bad144 disk [ snum [ bn ... ] ]\n");
+ fprintf(stderr,
+ "to read or overwrite bad-sector table, e.g.: bad144 hp0\n");
+ fprintf(stderr,
+ "or bad144 -a [ -c ] disk bn ...\n");
+ fprintf(stderr, "where options are:\n");
+ fprintf(stderr, "\t-a add new bad sectors to the table\n");
+ fprintf(stderr, "\t-c copy original sector to replacement\n");
+ exit(1);
}