631 lines
14 KiB
C
631 lines
14 KiB
C
/* $NetBSD: dkctl.c,v 1.14 2006/03/17 19:18:33 dsl Exp $ */
|
|
|
|
/*
|
|
* Copyright 2001 Wasabi Systems, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* Written by Jason R. Thorpe for Wasabi Systems, Inc.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
* must display the following acknowledgement:
|
|
* This product includes software developed for the NetBSD Project by
|
|
* Wasabi Systems, Inc.
|
|
* 4. The name of Wasabi Systems, Inc. may not be used to endorse
|
|
* or promote products derived from this software without specific prior
|
|
* written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY WASABI SYSTEMS, INC. ``AS IS'' AND
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
|
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL WASABI SYSTEMS, INC
|
|
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
/*
|
|
* dkctl(8) -- a program to manipulate disks.
|
|
*/
|
|
#include <sys/cdefs.h>
|
|
|
|
#ifndef lint
|
|
__RCSID("$NetBSD: dkctl.c,v 1.14 2006/03/17 19:18:33 dsl Exp $");
|
|
#endif
|
|
|
|
|
|
#include <sys/param.h>
|
|
#include <sys/ioctl.h>
|
|
#include <sys/dkio.h>
|
|
#include <sys/disk.h>
|
|
#include <sys/queue.h>
|
|
#include <err.h>
|
|
#include <errno.h>
|
|
#include <fcntl.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <util.h>
|
|
|
|
#define YES 1
|
|
#define NO 0
|
|
|
|
/* I don't think nl_langinfo is suitable in this case */
|
|
#define YES_STR "yes"
|
|
#define NO_STR "no"
|
|
#define YESNO_ARG YES_STR " | " NO_STR
|
|
|
|
#ifndef PRIdaddr
|
|
#define PRIdaddr PRId64
|
|
#endif
|
|
|
|
struct command {
|
|
const char *cmd_name;
|
|
const char *arg_names;
|
|
void (*cmd_func)(int, char *[]);
|
|
int open_flags;
|
|
};
|
|
|
|
void usage(void);
|
|
|
|
int fd; /* file descriptor for device */
|
|
const char *dvname; /* device name */
|
|
char dvname_store[MAXPATHLEN]; /* for opendisk(3) */
|
|
const char *cmdname; /* command user issued */
|
|
const char *argnames; /* helpstring; expected arguments */
|
|
|
|
int yesno(const char *);
|
|
|
|
void disk_getcache(int, char *[]);
|
|
void disk_setcache(int, char *[]);
|
|
void disk_synccache(int, char *[]);
|
|
void disk_keeplabel(int, char *[]);
|
|
void disk_badsectors(int, char *[]);
|
|
|
|
void disk_addwedge(int, char *[]);
|
|
void disk_delwedge(int, char *[]);
|
|
void disk_getwedgeinfo(int, char *[]);
|
|
void disk_listwedges(int, char *[]);
|
|
void disk_strategy(int, char *[]);
|
|
|
|
struct command commands[] = {
|
|
{ "getcache",
|
|
"",
|
|
disk_getcache,
|
|
O_RDONLY },
|
|
|
|
{ "setcache",
|
|
"none | r | w | rw [save]",
|
|
disk_setcache,
|
|
O_RDWR },
|
|
|
|
{ "synccache",
|
|
"[force]",
|
|
disk_synccache,
|
|
O_RDWR },
|
|
|
|
{ "keeplabel",
|
|
YESNO_ARG,
|
|
disk_keeplabel,
|
|
O_RDWR },
|
|
|
|
{ "badsector",
|
|
"flush | list | retry",
|
|
disk_badsectors,
|
|
O_RDWR },
|
|
|
|
{ "addwedge",
|
|
"name startblk blkcnt ptype",
|
|
disk_addwedge,
|
|
O_RDWR },
|
|
|
|
{ "delwedge",
|
|
"dk",
|
|
disk_delwedge,
|
|
O_RDWR },
|
|
|
|
{ "getwedgeinfo",
|
|
"",
|
|
disk_getwedgeinfo,
|
|
O_RDONLY },
|
|
|
|
{ "listwedges",
|
|
"",
|
|
disk_listwedges,
|
|
O_RDONLY },
|
|
|
|
{ "strategy",
|
|
"[name]",
|
|
disk_strategy,
|
|
O_RDWR },
|
|
|
|
{ NULL,
|
|
NULL,
|
|
NULL,
|
|
0 },
|
|
};
|
|
|
|
int
|
|
main(int argc, char *argv[])
|
|
{
|
|
int i;
|
|
|
|
/* Must have at least: device command */
|
|
if (argc < 3)
|
|
usage();
|
|
|
|
/* Skip program name, get and skip device name and command. */
|
|
dvname = argv[1];
|
|
cmdname = argv[2];
|
|
argv += 3;
|
|
argc -= 3;
|
|
|
|
/* Look up and call the command. */
|
|
for (i = 0; commands[i].cmd_name != NULL; i++)
|
|
if (strcmp(cmdname, commands[i].cmd_name) == 0)
|
|
break;
|
|
if (commands[i].cmd_name == NULL)
|
|
errx(1, "unknown command: %s", cmdname);
|
|
|
|
argnames = commands[i].arg_names;
|
|
|
|
/* Open the device. */
|
|
fd = opendisk(dvname, commands[i].open_flags, dvname_store,
|
|
sizeof(dvname_store), 0);
|
|
if (fd == -1)
|
|
err(1, "%s", dvname);
|
|
|
|
dvname = dvname_store;
|
|
|
|
(*commands[i].cmd_func)(argc, argv);
|
|
exit(0);
|
|
}
|
|
|
|
void
|
|
usage(void)
|
|
{
|
|
int i;
|
|
|
|
fprintf(stderr, "usage: %s device command [arg [...]]\n",
|
|
getprogname());
|
|
|
|
fprintf(stderr, " Available commands:\n");
|
|
for (i = 0; commands[i].cmd_name != NULL; i++)
|
|
fprintf(stderr, "\t%s %s\n", commands[i].cmd_name,
|
|
commands[i].arg_names);
|
|
|
|
exit(1);
|
|
}
|
|
|
|
void
|
|
disk_strategy(int argc, char *argv[])
|
|
{
|
|
struct disk_strategy odks;
|
|
struct disk_strategy dks;
|
|
|
|
memset(&dks, 0, sizeof(dks));
|
|
if (ioctl(fd, DIOCGSTRATEGY, &odks) == -1) {
|
|
err(EXIT_FAILURE, "%s: DIOCGSTRATEGY", dvname);
|
|
}
|
|
|
|
memset(&dks, 0, sizeof(dks));
|
|
switch (argc) {
|
|
case 0:
|
|
/* show the buffer queue strategy used */
|
|
printf("%s: %s\n", dvname, odks.dks_name);
|
|
return;
|
|
case 1:
|
|
/* set the buffer queue strategy */
|
|
strlcpy(dks.dks_name, argv[0], sizeof(dks.dks_name));
|
|
if (ioctl(fd, DIOCSSTRATEGY, &dks) == -1) {
|
|
err(EXIT_FAILURE, "%s: DIOCSSTRATEGY", dvname);
|
|
}
|
|
printf("%s: %s -> %s\n", dvname, odks.dks_name, argv[0]);
|
|
break;
|
|
default:
|
|
usage();
|
|
/* NOTREACHED */
|
|
}
|
|
}
|
|
|
|
void
|
|
disk_getcache(int argc, char *argv[])
|
|
{
|
|
int bits;
|
|
|
|
if (ioctl(fd, DIOCGCACHE, &bits) == -1)
|
|
err(1, "%s: getcache", dvname);
|
|
|
|
if ((bits & (DKCACHE_READ|DKCACHE_WRITE)) == 0)
|
|
printf("%s: No caches enabled\n", dvname);
|
|
else {
|
|
if (bits & DKCACHE_READ)
|
|
printf("%s: read cache enabled\n", dvname);
|
|
if (bits & DKCACHE_WRITE)
|
|
printf("%s: write-back cache enabled\n", dvname);
|
|
}
|
|
|
|
printf("%s: read cache enable is %schangeable\n", dvname,
|
|
(bits & DKCACHE_RCHANGE) ? "" : "not ");
|
|
printf("%s: write cache enable is %schangeable\n", dvname,
|
|
(bits & DKCACHE_WCHANGE) ? "" : "not ");
|
|
|
|
printf("%s: cache parameters are %ssavable\n", dvname,
|
|
(bits & DKCACHE_SAVE) ? "" : "not ");
|
|
}
|
|
|
|
void
|
|
disk_setcache(int argc, char *argv[])
|
|
{
|
|
int bits;
|
|
|
|
if (argc > 2 || argc == 0)
|
|
usage();
|
|
|
|
if (strcmp(argv[0], "none") == 0)
|
|
bits = 0;
|
|
else if (strcmp(argv[0], "r") == 0)
|
|
bits = DKCACHE_READ;
|
|
else if (strcmp(argv[0], "w") == 0)
|
|
bits = DKCACHE_WRITE;
|
|
else if (strcmp(argv[0], "rw") == 0)
|
|
bits = DKCACHE_READ|DKCACHE_WRITE;
|
|
else
|
|
usage();
|
|
|
|
if (argc == 2) {
|
|
if (strcmp(argv[1], "save") == 0)
|
|
bits |= DKCACHE_SAVE;
|
|
else
|
|
usage();
|
|
}
|
|
|
|
if (ioctl(fd, DIOCSCACHE, &bits) == -1)
|
|
err(1, "%s: setcache", dvname);
|
|
}
|
|
|
|
void
|
|
disk_synccache(int argc, char *argv[])
|
|
{
|
|
int force;
|
|
|
|
switch (argc) {
|
|
case 0:
|
|
force = 0;
|
|
break;
|
|
|
|
case 1:
|
|
if (strcmp(argv[0], "force") == 0)
|
|
force = 1;
|
|
else
|
|
usage();
|
|
break;
|
|
|
|
default:
|
|
usage();
|
|
}
|
|
|
|
if (ioctl(fd, DIOCCACHESYNC, &force) == -1)
|
|
err(1, "%s: sync cache", dvname);
|
|
}
|
|
|
|
void
|
|
disk_keeplabel(int argc, char *argv[])
|
|
{
|
|
int keep;
|
|
int yn;
|
|
|
|
if (argc != 1)
|
|
usage();
|
|
|
|
yn = yesno(argv[0]);
|
|
if (yn < 0)
|
|
usage();
|
|
|
|
keep = yn == YES;
|
|
|
|
if (ioctl(fd, DIOCKLABEL, &keep) == -1)
|
|
err(1, "%s: keep label", dvname);
|
|
}
|
|
|
|
|
|
void
|
|
disk_badsectors(int argc, char *argv[])
|
|
{
|
|
struct disk_badsectors *dbs, *dbs2, buffer[200];
|
|
SLIST_HEAD(, disk_badsectors) dbstop;
|
|
struct disk_badsecinfo dbsi;
|
|
daddr_t blk, totbad, bad;
|
|
u_int32_t count;
|
|
struct stat sb;
|
|
u_char *block;
|
|
time_t tm;
|
|
|
|
if (argc != 1)
|
|
usage();
|
|
|
|
if (strcmp(argv[0], "list") == 0) {
|
|
/*
|
|
* Copy the list of kernel bad sectors out in chunks that fit
|
|
* into buffer[]. Updating dbsi_skip means we don't sit here
|
|
* forever only getting the first chunk that fit in buffer[].
|
|
*/
|
|
dbsi.dbsi_buffer = (caddr_t)buffer;
|
|
dbsi.dbsi_bufsize = sizeof(buffer);
|
|
dbsi.dbsi_skip = 0;
|
|
dbsi.dbsi_copied = 0;
|
|
dbsi.dbsi_left = 0;
|
|
|
|
do {
|
|
if (ioctl(fd, DIOCBSLIST, (caddr_t)&dbsi) == -1)
|
|
err(1, "%s: badsectors list", dvname);
|
|
|
|
dbs = (struct disk_badsectors *)dbsi.dbsi_buffer;
|
|
for (count = dbsi.dbsi_copied; count > 0; count--) {
|
|
tm = dbs->dbs_failedat.tv_sec;
|
|
printf("%s: blocks %" PRIdaddr " - %" PRIdaddr " failed at %s",
|
|
dvname, dbs->dbs_min, dbs->dbs_max,
|
|
ctime(&tm));
|
|
dbs++;
|
|
}
|
|
dbsi.dbsi_skip += dbsi.dbsi_copied;
|
|
} while (dbsi.dbsi_left != 0);
|
|
|
|
} else if (strcmp(argv[0], "flush") == 0) {
|
|
if (ioctl(fd, DIOCBSFLUSH) == -1)
|
|
err(1, "%s: badsectors flush", dvname);
|
|
|
|
} else if (strcmp(argv[0], "retry") == 0) {
|
|
/*
|
|
* Enforce use of raw device here because the block device
|
|
* causes access to blocks to be clustered in a larger group,
|
|
* making it impossible to determine which individual sectors
|
|
* are the cause of a problem.
|
|
*/
|
|
if (fstat(fd, &sb) == -1)
|
|
err(1, "fstat");
|
|
|
|
if (!S_ISCHR(sb.st_mode)) {
|
|
fprintf(stderr, "'badsector retry' must be used %s\n",
|
|
"with character device");
|
|
exit(1);
|
|
}
|
|
|
|
SLIST_INIT(&dbstop);
|
|
|
|
/*
|
|
* Build up a copy of the in-kernel list in a number of stages.
|
|
* That the list we build up here is in the reverse order to
|
|
* the kernel's is of no concern.
|
|
*/
|
|
dbsi.dbsi_buffer = (caddr_t)buffer;
|
|
dbsi.dbsi_bufsize = sizeof(buffer);
|
|
dbsi.dbsi_skip = 0;
|
|
dbsi.dbsi_copied = 0;
|
|
dbsi.dbsi_left = 0;
|
|
|
|
do {
|
|
if (ioctl(fd, DIOCBSLIST, (caddr_t)&dbsi) == -1)
|
|
err(1, "%s: badsectors list", dvname);
|
|
|
|
dbs = (struct disk_badsectors *)dbsi.dbsi_buffer;
|
|
for (count = dbsi.dbsi_copied; count > 0; count--) {
|
|
dbs2 = malloc(sizeof *dbs2);
|
|
if (dbs2 == NULL)
|
|
err(1, NULL);
|
|
*dbs2 = *dbs;
|
|
SLIST_INSERT_HEAD(&dbstop, dbs2, dbs_next);
|
|
dbs++;
|
|
}
|
|
dbsi.dbsi_skip += dbsi.dbsi_copied;
|
|
} while (dbsi.dbsi_left != 0);
|
|
|
|
/*
|
|
* Just calculate and print out something that will hopefully
|
|
* provide some useful information about what's going to take
|
|
* place next (if anything.)
|
|
*/
|
|
bad = 0;
|
|
totbad = 0;
|
|
if ((block = calloc(1, DEV_BSIZE)) == NULL)
|
|
err(1, NULL);
|
|
SLIST_FOREACH(dbs, &dbstop, dbs_next) {
|
|
bad++;
|
|
totbad += dbs->dbs_max - dbs->dbs_min + 1;
|
|
}
|
|
|
|
printf("%s: bad sector clusters %"PRIdaddr
|
|
" total sectors %"PRIdaddr"\n", dvname, bad, totbad);
|
|
|
|
/*
|
|
* Clear out the kernel's list of bad sectors, ready for us
|
|
* to test all those it thought were bad.
|
|
*/
|
|
if (ioctl(fd, DIOCBSFLUSH) == -1)
|
|
err(1, "%s: badsectors flush", dvname);
|
|
|
|
printf("%s: bad sectors flushed\n", dvname);
|
|
|
|
/*
|
|
* For each entry we obtained from the kernel, retry each
|
|
* individual sector recorded as bad by seeking to it and
|
|
* attempting to read it in. Print out a line item for each
|
|
* bad block we verify.
|
|
*
|
|
* PRIdaddr is used here because the type of dbs_max is daddr_t
|
|
* and that may be either a 32bit or 64bit number(!)
|
|
*/
|
|
SLIST_FOREACH(dbs, &dbstop, dbs_next) {
|
|
printf("%s: Retrying %"PRIdaddr" - %"
|
|
PRIdaddr"\n", dvname, dbs->dbs_min, dbs->dbs_max);
|
|
|
|
for (blk = dbs->dbs_min; blk <= dbs->dbs_max; blk++) {
|
|
if (lseek(fd, (off_t)blk * DEV_BSIZE,
|
|
SEEK_SET) == -1) {
|
|
warn("%s: lseek block %" PRIdaddr "",
|
|
dvname, blk);
|
|
continue;
|
|
}
|
|
printf("%s: block %"PRIdaddr" - ", dvname, blk);
|
|
if (read(fd, block, DEV_BSIZE) != DEV_BSIZE)
|
|
printf("failed\n");
|
|
else
|
|
printf("ok\n");
|
|
fflush(stdout);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
disk_addwedge(int argc, char *argv[])
|
|
{
|
|
struct dkwedge_info dkw;
|
|
char *cp;
|
|
daddr_t start;
|
|
uint64_t size;
|
|
|
|
if (argc != 4)
|
|
usage();
|
|
|
|
/* XXX Unicode. */
|
|
if (strlen(argv[0]) > sizeof(dkw.dkw_wname) - 1)
|
|
errx(1, "Wedge name too long; max %zd characters",
|
|
sizeof(dkw.dkw_wname) - 1);
|
|
strcpy(dkw.dkw_wname, argv[0]);
|
|
|
|
if (strlen(argv[3]) > sizeof(dkw.dkw_ptype) - 1)
|
|
errx(1, "Wedge partition type too long; max %zd characters",
|
|
sizeof(dkw.dkw_ptype) - 1);
|
|
strcpy(dkw.dkw_ptype, argv[3]);
|
|
|
|
errno = 0;
|
|
start = strtoll(argv[1], &cp, 0);
|
|
if (*cp != '\0')
|
|
errx(1, "Invalid start block: %s", argv[1]);
|
|
if (errno == ERANGE && (start == LLONG_MAX ||
|
|
start == LLONG_MIN))
|
|
errx(1, "Start block out of range.");
|
|
if (start < 0)
|
|
errx(1, "Start block must be >= 0.");
|
|
|
|
errno = 0;
|
|
size = strtoull(argv[2], &cp, 0);
|
|
if (*cp != '\0')
|
|
errx(1, "Invalid block count: %s", argv[2]);
|
|
if (errno == ERANGE && (size == ULLONG_MAX))
|
|
errx(1, "Block count out of range.");
|
|
|
|
dkw.dkw_offset = start;
|
|
dkw.dkw_size = size;
|
|
|
|
if (ioctl(fd, DIOCAWEDGE, &dkw) == -1)
|
|
err(1, "%s: addwedge", dvname);
|
|
}
|
|
|
|
void
|
|
disk_delwedge(int argc, char *argv[])
|
|
{
|
|
struct dkwedge_info dkw;
|
|
|
|
if (argc != 1)
|
|
usage();
|
|
|
|
if (strlen(argv[0]) > sizeof(dkw.dkw_devname) - 1)
|
|
errx(1, "Wedge dk name too long; max %zd characters",
|
|
sizeof(dkw.dkw_devname) - 1);
|
|
strcpy(dkw.dkw_devname, argv[0]);
|
|
|
|
if (ioctl(fd, DIOCDWEDGE, &dkw) == -1)
|
|
err(1, "%s: delwedge", dvname);
|
|
}
|
|
|
|
void
|
|
disk_getwedgeinfo(int argc, char *argv[])
|
|
{
|
|
struct dkwedge_info dkw;
|
|
|
|
if (argc != 0)
|
|
usage();
|
|
|
|
if (ioctl(fd, DIOCGWEDGEINFO, &dkw) == -1)
|
|
err(1, "%s: getwedgeinfo", dvname);
|
|
|
|
printf("%s at %s: %s\n", dkw.dkw_devname, dkw.dkw_parent,
|
|
dkw.dkw_wname); /* XXX Unicode */
|
|
printf("%s: %"PRIu64" blocks at %"PRId64", type: %s\n",
|
|
dkw.dkw_devname, dkw.dkw_size, dkw.dkw_offset, dkw.dkw_ptype);
|
|
}
|
|
|
|
void
|
|
disk_listwedges(int argc, char *argv[])
|
|
{
|
|
struct dkwedge_info *dkw;
|
|
struct dkwedge_list dkwl;
|
|
size_t bufsize;
|
|
u_int i;
|
|
|
|
if (argc != 0)
|
|
usage();
|
|
|
|
dkw = NULL;
|
|
dkwl.dkwl_buf = dkw;
|
|
dkwl.dkwl_bufsize = 0;
|
|
|
|
for (;;) {
|
|
if (ioctl(fd, DIOCLWEDGES, &dkwl) == -1)
|
|
err(1, "%s: listwedges", dvname);
|
|
if (dkwl.dkwl_nwedges == dkwl.dkwl_ncopied)
|
|
break;
|
|
bufsize = dkwl.dkwl_nwedges * sizeof(*dkw);
|
|
if (dkwl.dkwl_bufsize < bufsize) {
|
|
dkw = realloc(dkwl.dkwl_buf, bufsize);
|
|
if (dkw == NULL)
|
|
errx(1, "%s: listwedges: unable to "
|
|
"allocate wedge info buffer", dvname);
|
|
dkwl.dkwl_buf = dkw;
|
|
dkwl.dkwl_bufsize = bufsize;
|
|
}
|
|
}
|
|
|
|
if (dkwl.dkwl_nwedges == 0) {
|
|
printf("%s: no wedges configured\n", dvname);
|
|
return;
|
|
}
|
|
|
|
printf("%s: %u wedge%s:\n", dvname, dkwl.dkwl_nwedges,
|
|
dkwl.dkwl_nwedges == 1 ? "" : "s");
|
|
for (i = 0; i < dkwl.dkwl_nwedges; i++) {
|
|
printf("%s: %s, %"PRIu64" blocks at %"PRId64", type: %s\n",
|
|
dkw[i].dkw_devname,
|
|
dkw[i].dkw_wname, /* XXX Unicode */
|
|
dkw[i].dkw_size, dkw[i].dkw_offset, dkw[i].dkw_ptype);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* return YES, NO or -1.
|
|
*/
|
|
int
|
|
yesno(const char *p)
|
|
{
|
|
|
|
if (!strcmp(p, YES_STR))
|
|
return YES;
|
|
if (!strcmp(p, NO_STR))
|
|
return NO;
|
|
return -1;
|
|
}
|