summaryrefslogtreecommitdiffstats
path: root/sbin/vinum/list.c
diff options
context:
space:
mode:
Diffstat (limited to 'sbin/vinum/list.c')
-rw-r--r--sbin/vinum/list.c1502
1 files changed, 0 insertions, 1502 deletions
diff --git a/sbin/vinum/list.c b/sbin/vinum/list.c
deleted file mode 100644
index ee5602a..0000000
--- a/sbin/vinum/list.c
+++ /dev/null
@@ -1,1502 +0,0 @@
-/* list.c: vinum interface program, list routines
- */
-/*-
- * Copyright (c) 1997, 1998
- * Nan Yang Computer Services Limited. All rights reserved.
- *
- * Parts copyright (c) 1997, 1998 Cybernet Corporation, NetMAX project.
- *
- * Written by Greg Lehey
- *
- * This software is distributed under the so-called ``Berkeley
- * License'':
- *
- * 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 by Nan Yang Computer
- * Services Limited.
- * 4. Neither the name of the Company nor the names of its contributors
- * may be used to endorse or promote products derived from this software
- * without specific prior written permission.
- *
- * This software is provided ``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 the company or contributors 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.
- *
- * $Id: list.c,v 1.32 2003/04/28 06:19:06 grog Exp $
- * $FreeBSD$
- */
-
-#include "vext.h"
-#include <sys/utsname.h>
-#include <dev/vinum/request.h>
-#include <devstat.h>
-
-/*
- * When a subdisk is reviving or initializing, we
- * check to see whether it is still progressing
- * and print a warning if not. We check every 50
- * ms, up to a maximum of 5 seconds. This is the
- * counter value.
- */
-#define STALLCOUNT 100
-
-/*
- * Take a size in sectors and return a pointer to
- * a string which represents the size best. If lj
- * is != 0, return left justified, otherwise in a
- * fixed 10 character field suitable for columnar
- * printing.
- *
- * Note this uses a static string: it's only
- * intended to be used immediately for printing.
- */
-char *
-roughlength(int64_t bytes, int lj)
-{
- static char description[16];
-
- if (bytes > (int64_t) MEGABYTE * 10000) /* gigabytes */
- sprintf(description, lj ? "%lld GB" : "%10d GB", bytes / GIGABYTE);
- else if (bytes > KILOBYTE * 10000) /* megabytes */
- sprintf(description, lj ? "%lld MB" : "%10d MB", bytes / MEGABYTE);
- else if (bytes > 10000) /* kilobytes */
- sprintf(description, lj ? "%lld kB" : "%10d kB", bytes / KILOBYTE);
- else /* bytes */
- sprintf(description, lj ? "%lld B" : "%10d B", bytes);
- return description;
-}
-
-void
-vinum_list(int argc, char *argv[], char *argv0[])
-{
- int object;
- int i;
- enum objecttype type;
-
- if (sflag & (!vflag)) /* just summary stats, */
- printf("Object\t\t Reads\t\tBytes\tAverage\tRecover\t Writes"
- "\t\tBytes\tAverage\t Mblock Mstripe\n\n");
- if (argc == 0)
- listconfig(); /* list everything */
- else {
- for (i = 0; i < argc; i++) {
- object = find_object(argv[i], &type); /* look for it */
- if (vinum_li(object, type))
- fprintf(stderr, "Can't find object: %s\n", argv[i]);
- }
- }
-}
-
-/* List an object */
-int
-vinum_li(int object, enum objecttype type)
-{
- switch (type) {
- case drive_object:
- vinum_ldi(object, recurse);
- break;
-
- case sd_object:
- vinum_lsi(object, recurse);
- break;
-
- case plex_object:
- vinum_lpi(object, recurse);
- break;
-
- case volume_object:
- vinum_lvi(object, recurse);
- break;
-
- default:
- return -1;
- }
- return 0;
-}
-
-void
-vinum_ldi(int driveno, int recurse)
-{
- time_t t; /* because Bruce says so */
- int sdno; /* for recursion */
-
- get_drive_info(&drive, driveno);
- if (drive.state != drive_unallocated) {
- if (vflag) {
- printf("Drive %s:\tDevice %s\n",
- drive.label.name,
- drive.devicename);
- t = drive.label.date_of_birth.tv_sec;
- printf("\t\tCreated on %s at %s",
- drive.label.sysname,
- ctime(&t));
- t = drive.label.last_update.tv_sec;
- printf("\t\tConfig last updated %s", /* care: \n at end */
- ctime(&t));
- printf("\t\tSize: %16lld bytes (%lld MB)\n\t\tUsed: %16lld bytes (%lld MB)\n"
- "\t\tAvailable: %11qd bytes (%d MB)\n",
- (long long) drive.label.drive_size, /* bytes used */
- (long long) (drive.label.drive_size / MEGABYTE),
- (long long) (drive.label.drive_size - drive.sectors_available
- * DEV_BSIZE),
- (long long) (drive.label.drive_size - drive.sectors_available
- * DEV_BSIZE) / MEGABYTE,
- (long long) drive.sectors_available * DEV_BSIZE,
- (int) (drive.sectors_available * DEV_BSIZE / MEGABYTE));
- printf("\t\tState: %s\n", drive_state(drive.state));
- if (drive.lasterror != 0)
- printf("\t\tLast error: %s\n", strerror(drive.lasterror));
- else
- printf("\t\tLast error: none\n");
- printf("\t\tActive requests:\t%d\n\t\tMaximum active:\t\t%d\n",
- drive.active,
- drive.maxactive);
- if (Verbose) { /* print the free list */
- int fe; /* freelist entry */
- struct drive_freelist freelist;
- struct ferq { /* request to pass to ioctl */
- int driveno;
- int fe;
- } *ferq = (struct ferq *) &freelist;
-
- printf("\t\tFree list contains %d entries:\n\t\t Offset\t Size\n",
- drive.freelist_entries);
- for (fe = 0; fe < drive.freelist_entries; fe++) {
- ferq->driveno = drive.driveno;
- ferq->fe = fe;
- if (ioctl(superdev, VINUM_GETFREELIST, &freelist) < 0) {
- fprintf(stderr,
- "Can't get free list element %d: %s\n",
- fe,
- strerror(errno));
- longjmp(command_fail, -1);
- }
- printf("\t\t%9lld\t%9lld\n",
- (long long) freelist.offset,
- (long long) freelist.sectors);
- }
- }
- } else if (!sflag) {
- printf("D %-21s State: %s\t%s\tA: %lld/%lld MB",
- drive.label.name,
- drive_state(drive.state),
- drive.devicename,
- (long long) drive.sectors_available * DEV_BSIZE / MEGABYTE,
- (long long) (drive.label.drive_size / MEGABYTE));
- if (drive.label.drive_size != 0)
- printf(" (%d%%)",
- (int) ((drive.sectors_available * 100 * DEV_BSIZE)
- / (drive.label.drive_size - (DATASTART * DEV_BSIZE))));
- }
- if (sflag) {
- if (vflag || Verbose) {
- printf("\t\tReads: \t%16lld\n\t\tBytes read:\t%16lld (%s)\n",
- (long long) drive.reads,
- (long long) drive.bytes_read,
- roughlength(drive.bytes_read, 1));
- if (drive.reads != 0)
- printf("\t\tAverage read:\t%16lld bytes\n",
- (long long) drive.bytes_read / drive.reads);
- printf("\t\tWrites: \t%16lld\n\t\tBytes written:\t%16lld (%s)\n",
- (long long) drive.writes,
- (long long) drive.bytes_written,
- roughlength(drive.bytes_written, 1));
- if (drive.writes != 0)
- printf("\t\tAverage write:\t%16lld bytes\n",
- (long long) (drive.bytes_written / drive.writes));
- } else { /* non-verbose stats */
- printf("%-15s\t%7lld\t%15lld\t",
- drive.label.name,
- (long long) drive.reads,
- (long long) drive.bytes_read);
- if (drive.reads != 0)
- printf("%7lld\t\t",
- (long long) (drive.bytes_read / drive.reads));
- else
- printf("\t\t");
- printf("%7lld\t%15lld\t",
- (long long) drive.writes,
- (long long) drive.bytes_written);
- if (drive.writes != 0)
- printf("%7lld",
- (long long) (drive.bytes_written / drive.writes));
- }
- }
- if (recurse) {
- printf("\n");
- for (sdno = 0; sdno < vinum_conf.subdisks_allocated; sdno++) {
- get_sd_info(&sd, sdno);
- if ((sd.state != sd_unallocated)
- && (sd.driveno == drive.driveno))
- vinum_lsi(sd.sdno, 0);
- }
- }
- printf("\n");
- }
-}
-
-void
-vinum_ld(int argc, char *argv[], char *argv0[])
-{
- int i;
- int driveno;
- enum objecttype type;
-
- if (ioctl(superdev, VINUM_GETCONFIG, &vinum_conf) < 0) {
- perror("Can't get vinum config");
- return;
- }
- if (argc == 0) {
- for (driveno = 0; driveno < vinum_conf.drives_allocated; driveno++)
- vinum_ldi(driveno, recurse);
- } else {
- for (i = 0; i < argc; i++) {
- driveno = find_object(argv[i], &type);
- if (type == drive_object)
- vinum_ldi(driveno, recurse);
- else
- fprintf(stderr, "%s is not a drive\n", argv[i]);
- }
- }
-}
-
-void
-vinum_lvi(int volno, int recurse)
-{
- get_volume_info(&vol, volno);
- if (vol.state != volume_unallocated) {
- if (vflag) {
- printf("Volume %s:\tSize: %lld bytes (%lld MB)\n"
- "\t\tState: %s\n\t\tFlags: %s%s%s\n",
- vol.name,
- ((long long) vol.size) * DEV_BSIZE,
- ((long long) vol.size) * DEV_BSIZE / MEGABYTE,
- volume_state(vol.state),
- vol.flags & VF_OPEN ? "open " : "",
- (vol.flags & VF_WRITETHROUGH ? "writethrough " : ""),
- (vol.flags & VF_RAW ? "raw" : ""));
- printf("\t\t%d plexes\n\t\tRead policy: ", vol.plexes);
- if (vol.preferred_plex < 0) /* round robin */
- printf("round robin\n");
- else {
- get_plex_info(&plex, vol.plex[vol.preferred_plex]);
- printf("plex %d (%s)\n", vol.preferred_plex, plex.name);
- }
- } else if (!sflag) /* brief */
- printf("V %-21s State: %s\tPlexes: %7d\tSize: %s\n",
- vol.name,
- volume_state(vol.state),
- vol.plexes,
- roughlength(vol.size << DEV_BSHIFT, 0));
- if (sflag) {
- if (vflag || Verbose) {
- printf("\t\tReads: \t%16lld\n\t\tRecovered:\t%16lld\n\t\tBytes read:\t%16lld (%s)\n",
- (long long) vol.reads,
- (long long) vol.recovered_reads,
- (long long) vol.bytes_read,
- roughlength(vol.bytes_read, 1));
- if (vol.reads != 0)
- printf("\t\tAverage read:\t%16lld bytes\n",
- (long long) (vol.bytes_read / vol.reads));
- printf("\t\tWrites: \t%16lld\n\t\tBytes written:\t%16lld (%s)\n",
- (long long) vol.writes,
- (long long) vol.bytes_written,
- roughlength(vol.bytes_written, 1));
- if (vol.writes != 0)
- printf("\t\tAverage write:\t%16lld bytes\n",
- (long long) (vol.bytes_written / vol.writes));
- printf("\t\tActive requests:\t%8d\n", vol.active);
- } else { /* brief stats listing */
- printf("%-15s\t%7lld\t%15lld\t",
- vol.name,
- (long long) vol.reads,
- (long long) vol.bytes_read);
- if (vol.reads != 0)
- printf("%7lld\t",
- (long long) (vol.bytes_read / vol.reads));
- else
- printf("\t");
- printf("%7lld\t", (long long) vol.recovered_reads);
- printf("%7lld\t%15lld\t",
- (long long) vol.writes,
- vol.bytes_written);
- if (vol.writes != 0)
- printf("%7lld\n",
- (long long) (vol.bytes_written / vol.writes));
- else
- printf("\n");
- }
- }
- if (vol.plexes > 0) {
- int plexno;
- if (Verbose) { /* brief list */
- for (plexno = 0; plexno < vol.plexes; plexno++) {
- get_plex_info(&plex, vol.plex[plexno]);
- /* Just a brief summary here */
- printf("\t\tPlex %2d:\t%s\t(%s), %s\n",
- plexno,
- plex.name,
- plex_org(plex.organization),
- roughlength(plex.length << DEV_BSHIFT, 0));
- }
- }
- if (recurse) {
- for (plexno = 0; plexno < vol.plexes; plexno++)
- vinum_lpi(vol.plex[plexno], 0); /* first show the plexes */
- for (plexno = 0; plexno < vol.plexes; plexno++) { /* then the subdisks */
- get_plex_info(&plex, vol.plex[plexno]);
- if (plex.subdisks > 0) {
- int sdno;
-
- for (sdno = 0; sdno < plex.subdisks; sdno++) {
- get_plex_sd_info(&sd, vol.plex[plexno], sdno);
- vinum_lsi(sd.sdno, 0);
- }
- }
- }
- printf("\n");
- }
- }
- }
-}
-
-void
-vinum_lv(int argc, char *argv[], char *argv0[])
-{
- int i;
- int volno;
- enum objecttype type;
-
- if (ioctl(superdev, VINUM_GETCONFIG, &vinum_conf) < 0) {
- perror("Can't get vinum config");
- return;
- }
- if (argc == 0)
- for (volno = 0; volno < vinum_conf.volumes_allocated; volno++)
- vinum_lvi(volno, recurse);
- else {
- for (i = 0; i < argc; i++) {
- volno = find_object(argv[i], &type);
- if (type == volume_object)
- vinum_lvi(volno, recurse);
- else
- fprintf(stderr, "%s is not a volume\n", argv[i]);
- }
- }
-}
-
-void
-vinum_lpi(int plexno, int recurse)
-{
- get_plex_info(&plex, plexno);
- if (plex.state != plex_unallocated) {
- if (vflag) {
- printf("Plex %s:\tSize:\t%9lld bytes (%lld MB)\n\t\tSubdisks: %8d\n",
- plex.name,
- (long long) plex.length * DEV_BSIZE,
- (long long) plex.length * DEV_BSIZE / MEGABYTE,
- plex.subdisks);
- printf("\t\tState: %s\n\t\tOrganization: %s",
- plex_state(plex.state),
- plex_org(plex.organization));
- if (isstriped((&plex)))
- printf("\tStripe size: %s\n", roughlength(plex.stripesize * DEV_BSIZE, 1));
- else
- printf("\n");
- if ((isparity((&plex)))
- && (plex.checkblock != 0))
- printf("\t\tCheck block pointer:\t\t%s (%d%%)\n",
- roughlength((plex.checkblock << DEV_BSHIFT) * (plex.subdisks - 1), 0),
- (int) (((u_int64_t) (plex.checkblock * 100)) * (plex.subdisks - 1) / plex.length));
- if (plex.volno >= 0) {
- get_volume_info(&vol, plex.volno);
- printf("\t\tPart of volume %s\n", vol.name);
- }
- } else if (!sflag) { /* non-verbose list */
- char *org = ""; /* organization */
-
- switch (plex.organization) {
- case plex_disorg: /* disorganized */
- org = "??";
- break;
- case plex_concat: /* concatenated plex */
- org = "C";
- break;
- case plex_striped: /* striped plex */
- org = "S";
- break;
- case plex_raid4: /* RAID4 plex */
- org = "R4";
- break;
- case plex_raid5: /* RAID5 plex */
- org = "R5";
- break;
- }
- printf("P %-18s %2s State: %s\tSubdisks: %5d\tSize: %s",
- plex.name,
- org,
- plex_state(plex.state),
- plex.subdisks,
- roughlength(plex.length << DEV_BSHIFT, 0));
- }
- if (sflag) {
- if (vflag || Verbose) {
- printf("\t\tReads: \t%16lld\n\t\tBytes read:\t%16lld (%s)\n",
- (long long) plex.reads,
- (long long) plex.bytes_read,
- roughlength(plex.bytes_read, 1));
- if (plex.reads != 0)
- printf("\t\tAverage read:\t%16lld bytes\n",
- (long long) (plex.bytes_read / plex.reads));
- printf("\t\tWrites: \t%16lld\n\t\tBytes written:\t%16lld (%s)\n",
- (long long) plex.writes,
- (long long) plex.bytes_written,
- roughlength(plex.bytes_written, 1));
- if (plex.writes != 0)
- printf("\t\tAverage write:\t%16lld bytes\n",
- (long long) (plex.bytes_written / plex.writes));
- if (((plex.reads + plex.writes) > 0)
- && isstriped((&plex)))
- printf("\t\tMultiblock:\t%16lld (%d%%)\n"
- "\t\tMultistripe:\t%16lld (%d%%)\n",
- (long long) plex.multiblock,
- (int) (plex.multiblock * 100 / (plex.reads + plex.writes)),
- (long long) plex.multistripe,
- (int) (plex.multistripe * 100 / (plex.reads + plex.writes)));
- if (plex.recovered_reads)
- printf("\t\tRecovered reads:%16lld\n",
- (long long) plex.recovered_reads);
- if (plex.degraded_writes)
- printf("\t\tDegraded writes:%16lld\n",
- (long long) plex.degraded_writes);
- if (plex.parityless_writes)
- printf("\t\tParityless writes:%14lld\n",
- (long long) plex.parityless_writes);
- } else {
- printf("%-15s\t%7lld\t%15lld\t",
- plex.name,
- (long long) plex.reads,
- (long long) plex.bytes_read);
- if (plex.reads != 0)
- printf("%7lld\t",
- (long long) (plex.bytes_read / plex.reads));
- else
- printf("\t");
- printf("%7lld\t", (long long) plex.recovered_reads);
- printf("%7lld\t%15lld\t",
- (long long) plex.writes,
- (long long) plex.bytes_written);
- if (plex.writes != 0)
- printf("%7lld\t",
- (long long) (plex.bytes_written / plex.writes));
- else
- printf("\t");
- printf("%7lld\t%7lld\n",
- (long long) plex.multiblock,
- (long long) plex.multistripe);
- }
- }
- if (plex.subdisks > 0) {
- int sdno;
-
- if (Verbose) {
- printf("\n");
- for (sdno = 0; sdno < plex.subdisks; sdno++) {
- get_plex_sd_info(&sd, plexno, sdno);
- printf("\t\tSubdisk %d:\t%s\n\t\t state: %s\tsize %11lld (%lld MB)\n",
- sdno,
- sd.name,
- sd_state(sd.state),
- (long long) sd.sectors * DEV_BSIZE,
- (long long) sd.sectors * DEV_BSIZE / MEGABYTE);
- if (plex.organization == plex_concat)
- printf("\t\t\toffset %9ld (0x%lx)\n",
- (long) sd.plexoffset,
- (long) sd.plexoffset);
- }
- }
- if (recurse) {
- printf("\n");
- for (sdno = 0; sdno < plex.subdisks; sdno++) {
- get_plex_sd_info(&sd, plexno, sdno);
- vinum_lsi(sd.sdno, 0);
- }
- }
- }
- printf("\n");
- }
-}
-
-void
-vinum_lp(int argc, char *argv[], char *argv0[])
-{
- int i;
- int plexno;
- enum objecttype type;
-
- if (ioctl(superdev, VINUM_GETCONFIG, &vinum_conf) < 0) {
- perror("Can't get vinum config");
- return;
- }
- if (argc == 0) {
- for (plexno = 0; plexno < vinum_conf.plexes_allocated; plexno++)
- vinum_lpi(plexno, recurse);
- } else {
- for (i = 0; i < argc; i++) {
- plexno = find_object(argv[i], &type);
- if (type == plex_object)
- vinum_lpi(plexno, recurse);
- else
- fprintf(stderr, "%s is not a plex\n", argv[i]);
- }
- }
-}
-
-void
-vinum_lsi(int sdno, int recurse)
-{
- long long revived; /* keep an eye on revive progress */
- int times;
-
- get_sd_info(&sd, sdno);
- if (sd.state != sd_unallocated) {
- get_drive_info(&drive, sd.driveno);
-
- if (vflag) {
- printf("Subdisk %s:\n\t\tSize: %16lld bytes (%lld MB)\n\t\tState: %s\n",
- sd.name,
- (long long) sd.sectors * DEV_BSIZE,
- (long long) sd.sectors / (MEGABYTE / DEV_BSIZE),
- sd_state(sd.state));
- if (sd.flags & VF_RETRYERRORS)
- printf("\t\tretryerrors\n");
- if (sd.plexno >= 0) {
- get_plex_info(&plex, sd.plexno);
- printf("\t\tPlex %s", plex.name);
- printf(" at offset %lld (%s)\n",
- (long long) sd.plexoffset * DEV_BSIZE,
- roughlength((long long) sd.plexoffset * DEV_BSIZE, 1));
- }
- if (sd.state == sd_reviving) {
- if (sd.reviver == 0)
- printf("\t\t*** Start subdisk with 'start' command ***\n");
- else {
- printf("\t\tReviver PID:\t%d\n", sd.reviver);
- if (kill(sd.reviver, 0) == -1) {
- if (errno == ESRCH) /* no process */
- printf("\t\t*** Revive process has died ***\n");
- /* Don't report a problem that "can't happen" */
- } else {
- revived = sd.revived; /* note how far we were */
-
- /*
- * Wait for up to a second until we
- * see some progress with the revive.
- * Do it like this so we don't have
- * annoying delays in the listing.
- */
- for (times = 0; times < STALLCOUNT; times++) {
- get_sd_info(&sd, sdno);
- if (sd.revived != revived) /* progress? */
- break;
- usleep(50000);
- }
- if (times == STALLCOUNT)
- printf("\t\t*** Revive has stalled ***\n");
- }
- }
- printf("\t\tRevive pointer:\t\t%s (%d%%)\n",
- roughlength(sd.revived << DEV_BSHIFT, 0),
- (int) (((u_int64_t) (sd.revived * 100)) / sd.sectors));
- printf("\t\tRevive blocksize:\t%s\n"
- "\t\tRevive interval:\t%10d seconds\n",
- roughlength(sd.revive_blocksize, 0),
- sd.revive_interval);
- }
- if (sd.state == sd_initializing) {
- printf("\t\tInitialize pointer:\t%s (%d%%)\n",
- roughlength(sd.initialized << DEV_BSHIFT, 0),
- (int) (((u_int64_t) (sd.initialized * 100)) / sd.sectors));
- printf("\t\tInitialize blocksize:\t%s\n"
- "\t\tInitialize interval:\t%10d seconds\n",
- roughlength(sd.init_blocksize, 0),
- sd.init_interval);
- }
- if (sd.driveoffset < 0)
- printf("\t\tDrive %s (%s), no offset\n",
- drive.label.name,
- drive.devicename);
- else if (drive.devicename[0] != '\0') /* has a name */
- printf("\t\tDrive %s (%s) at offset %lld (%s)\n",
- drive.label.name,
- drive.devicename,
- (long long) (sd.driveoffset * DEV_BSIZE),
- roughlength(sd.driveoffset * DEV_BSIZE, 1));
- else
- printf("\t\tDrive %s (*missing*) at offset %lld (%s)\n",
- drive.label.name,
- (long long) (sd.driveoffset * DEV_BSIZE),
- roughlength(sd.driveoffset * DEV_BSIZE, 1));
- } else if (!sflag) { /* brief listing, no stats */
- if (sd.state == sd_reviving)
- printf("S %-21s State: R %d%%\t",
- sd.name,
- (int) (((u_int64_t) (sd.revived * 100)) / sd.sectors));
- else if (sd.state == sd_initializing)
- printf("S %-21s State: I %d%%\t",
- sd.name,
- (int) (((u_int64_t) (sd.initialized * 100)) / sd.sectors));
- else
- printf("S %-21s State: %s\t",
- sd.name,
- sd_state(sd.state));
- printf("D: %-12s Size: %s\n",
- drive.label.name,
- roughlength(sd.sectors << DEV_BSHIFT, 0));
- if (sd.state == sd_reviving) {
- if (sd.reviver == 0)
- printf("\t\t\t*** Start %s with 'start' command ***\n",
- sd.name);
- else if (kill(sd.reviver, 0) == -1) {
- if (errno == ESRCH) /* no process */
- printf("\t\t\t*** Revive process for %s has died ***\n",
- sd.name);
- /* Don't report a problem that "can't happen" */
- } else {
- revived = sd.revived; /* note how far we were */
-
- /*
- * Wait for up to a second until we
- * see some progress with the revive.
- * Do it like this so we don't have
- * annoying delays in the listing.
- */
- for (times = 0; times < STALLCOUNT; times++) {
- get_sd_info(&sd, sdno);
- if (sd.revived != revived) /* progress? */
- break;
- usleep(50000);
- }
- if (times == STALLCOUNT)
- printf("\t\t\t*** Revive of %s has stalled ***\n",
- sd.name);
- }
- }
- }
- if (sflag) {
- if (vflag || Verbose) {
- printf("\t\tReads: \t%16lld\n\t\tBytes read:\t%16lld (%s)\n",
- (long long) sd.reads,
- (long long) sd.bytes_read,
- roughlength(sd.bytes_read, 1));
- if (sd.reads != 0)
- printf("\t\tAverage read:\t%16lld bytes\n",
- (long long) (sd.bytes_read / sd.reads));
- printf("\t\tWrites: \t%16lld\n\t\tBytes written:\t%16lld (%s)\n",
- (long long) sd.writes,
- (long long) sd.bytes_written,
- roughlength(sd.bytes_written, 1));
- if (sd.writes != 0)
- printf("\t\tAverage write:\t%16lld bytes\n",
- (long long) (sd.bytes_written / sd.writes));
- } else {
- printf("%-15s\t%7lld\t%15lld\t",
- sd.name,
- (long long) sd.reads,
- (long long) sd.bytes_read);
- if (sd.reads != 0)
- printf("%7lld\t\t",
- (long long) (sd.bytes_read / sd.reads));
- else
- printf("\t\t");
- printf("%7lld\t%15lld\t",
- (long long) sd.writes,
- (long long) sd.bytes_written);
- if (sd.writes != 0)
- printf("%7lld\n",
- (long long) (sd.bytes_written / sd.writes));
- else
- printf("\n");
- }
- }
- if (recurse)
- vinum_ldi(sd.driveno, 0);
- if (vflag)
- printf("\n"); /* make it more readable */
- }
-}
-
-void
-vinum_ls(int argc, char *argv[], char *argv0[])
-{
- int i;
- int sdno;
-
- /* Structures to read kernel data into */
- struct __vinum_conf vinum_conf;
- enum objecttype type;
-
- if (ioctl(superdev, VINUM_GETCONFIG, &vinum_conf) < 0) {
- perror("Can't get vinum config");
- return;
- }
- if (argc == 0) {
- for (sdno = 0; sdno < vinum_conf.subdisks_allocated; sdno++)
- vinum_lsi(sdno, recurse);
- } else { /* specific subdisks */
- for (i = 0; i < argc; i++) {
- sdno = find_object(argv[i], &type);
- if (type == sd_object)
- vinum_lsi(sdno, recurse);
- else
- fprintf(stderr, "%s is not a subdisk\n", argv[i]);
- }
- }
-}
-
-
-/* List the complete configuration.
-
- * XXX Change this to specific lists */
-void
-listconfig()
-{
- if (ioctl(superdev, VINUM_GETCONFIG, &vinum_conf) < 0) {
- perror("Can't get vinum config");
- return;
- }
- printf("%d drives:\n", vinum_conf.drives_used);
- if (vinum_conf.drives_used > 0) {
- vinum_ld(0, NULL, NULL);
- printf("\n");
- }
- printf("%d volumes:\n", vinum_conf.volumes_used);
- if (vinum_conf.volumes_used > 0) {
- vinum_lv(0, NULL, NULL);
- printf("\n");
- }
- printf("%d plexes:\n", vinum_conf.plexes_used);
- if (vinum_conf.plexes_used > 0) {
- vinum_lp(0, NULL, NULL);
- printf("\n");
- }
- printf("%d subdisks:\n", vinum_conf.subdisks_used);
- if (vinum_conf.subdisks_used > 0)
- vinum_ls(0, NULL, NULL);
-}
-
-/* Convert a timeval to Tue Oct 13 13:54:14.0434324
- * Return pointer to text */
-char *
-timetext(struct timeval *time)
-{
- static char text[30];
- time_t t; /* to keep Bruce happy */
-
- t = time->tv_sec;
- strcpy(text, ctime(&t)); /* to the second */
- sprintf(&text[19], ".%06ld", time->tv_usec); /* and the microseconds */
- return &text[11];
-}
-
-/* Return the difference in microseconds between two timevals. */
-inline struct timeval
-timediff(struct timeval then, struct timeval now)
-{
- struct timeval diff;
-
- diff.tv_sec = now.tv_sec - then.tv_sec;
- diff.tv_usec = now.tv_usec - then.tv_usec;
- if (diff.tv_usec < 0) {
- diff.tv_usec += 1000000;
- diff.tv_sec--;
- }
- return diff;
-}
-
-void
-vinum_info(int argc, char *argv[], char *argv0[])
-{
- struct meminfo meminfo;
- struct mc malloced;
- int i;
- struct rqinfo rq;
- struct timeval lasttime; /* time of previous request */
- struct timeval diff; /* difference from now */
-
- if (ioctl(superdev, VINUM_GETCONFIG, &vinum_conf) < 0) {
- perror("Can't get vinum config");
- return;
- }
- if ((vinum_conf.flags & VF_HASDEBUG) == 0)
- fprintf(stderr, "Kernel module does not have debug support\n");
- else {
- printf("Flags: 0x%x\n", vinum_conf.flags);
- if (ioctl(superdev, VINUM_MEMINFO, &meminfo) < 0) {
- perror("Can't get information");
- return;
- }
- printf("Total of %d blocks malloced, total memory: %d\n"
- "Maximum allocs: %8d, malloc table at 0x%08lx\n",
- meminfo.mallocs,
- meminfo.total_malloced,
- meminfo.highwater,
- (unsigned long) meminfo.malloced);
-
- printf("%d requests active, maximum %d active\n",
- vinum_conf.active,
- vinum_conf.maxactive);
- if (vflag && (!Verbose))
- for (i = 0; i < meminfo.mallocs; i++) {
- malloced.seq = i;
- if (ioctl(superdev, VINUM_MALLOCINFO, &malloced) < 0) {
- perror("Can't get information");
- return;
- }
- if (!(i & 63))
- printf("Block\tSequence\t size\t address\t line\t\tfile\n\n");
- printf("%6d\t%6d\t\t%6d\t0x%08lx\t%6d\t\t%s\n",
- i,
- malloced.seq,
- malloced.size,
- (unsigned long) malloced.address,
- malloced.line,
- (char *) &malloced.file);
- }
- if (Verbose) {
- if (Verbose > 1) {
- lasttime.tv_sec = 0;
- lasttime.tv_usec = 0;
- printf("\n Time\t\t Event\t Buf\tDev\t Offset\t"
- "Bytes\tSD\tSDoff\tDoffset\tGoffset\n\n");
- } else
- printf("\nTime\t\t Event\t Buf\tDev\t Offset\tBytes\tSD"
- "\tSDoff\tDoffset\tGoffset\n\n");
- for (i = RQINFO_SIZE - 1; i >= 0; i--) { /* go through the request list in order */
- *((int *) &rq) = i;
- if (ioctl(superdev, VINUM_RQINFO, &rq) < 0) {
- perror("Can't get information");
- return;
- }
- if (rq.type != loginfo_unused) {
- switch (Verbose) {
- case 2:
- if ((lasttime.tv_usec != 0) || (lasttime.tv_sec != 0)) {
- diff = timediff(lasttime, rq.timestamp);
- if (diff.tv_sec != 0)
- printf("\n+ %d.%06d sec:\n ", diff.tv_sec, diff.tv_usec);
- else
- printf("+%6d µs ", diff.tv_usec);
- } else
- printf(" ");
- break;
-
- case 3:
- if ((lasttime.tv_usec != 0) || (lasttime.tv_sec != 0))
- diff = timediff(lasttime, rq.timestamp);
- if (diff.tv_sec != 0)
- printf("\n+ %d.%06d sec:\n ", diff.tv_sec, diff.tv_usec);
- else if (rq.type == loginfo_iodone) {
- diff = timediff(rq.info.rqe.launchtime, rq.timestamp);
- printf("+%6d µs ",
- diff.tv_usec);
- } else
- printf(" ");
- break;
-
- default:
- break;
- }
-
- /* Compress devminor into something printable. */
- rq.devminor = (rq.devminor & 0xff)
- | ((rq.devminor & 0xfff0000) >> 8);
- switch (rq.type) {
- case loginfo_user_bp: /* this is the bp when strategy is called */
- printf("%s %dVS %s %p\t%2d.%-6d 0x%9llx\t%d\n",
- timetext(&rq.timestamp),
- rq.type,
- rq.info.b.b_iocmd == BIO_READ ? "Read " : "Write",
- rq.bp,
- rq.devmajor,
- rq.devminor,
- rq.info.b.b_blkno,
- rq.info.b.b_bcount);
- break;
-
- case loginfo_sdiol: /* subdisk I/O launch */
- case loginfo_user_bpl: /* and this is the bp at launch time */
- printf("%s %dLR %s %p\t%2d.%-6d 0x%9llx\t%ld\n",
- timetext(&rq.timestamp),
- rq.type,
- rq.info.b.b_iocmd == BIO_READ ? "Read " : "Write",
- rq.bp,
- rq.devmajor,
- rq.devminor,
- rq.info.b.b_blkno,
- rq.info.b.b_bcount);
- break;
-
- case loginfo_rqe: /* user RQE */
- /*
- * We have two timestamps
- * in this request, and
- * they might not agree by
- * one or two µs. Make
- * them agree by force.
- */
- rq.timestamp = rq.info.rqe.launchtime;
- printf("%s 3RQ %s %p\t%2d.%-6d 0x%9llx\t%ld\t%d\t%6x\t%6x\t%x\n",
- timetext(&rq.timestamp),
- rq.info.rqe.b.b_iocmd == BIO_READ ? "Read " : "Write",
- rq.bp,
- rq.devmajor,
- rq.devminor,
- rq.info.rqe.b.b_blkno,
- rq.info.rqe.b.b_bcount,
- rq.info.rqe.sdno,
- rq.info.rqe.sdoffset,
- rq.info.rqe.dataoffset,
- rq.info.rqe.groupoffset);
- break;
-
- case loginfo_iodone: /* iodone called */
- printf("%s 4DN %s %p\t%2d.%-6d 0x%9llx\t%ld\t%d\t%6x\t%6x\t%x\n",
- timetext(&rq.timestamp),
- rq.info.rqe.b.b_iocmd == BIO_READ ? "Read " : "Write",
- rq.bp,
- rq.devmajor,
- rq.devminor,
- rq.info.rqe.b.b_blkno,
- rq.info.rqe.b.b_bcount,
- rq.info.rqe.sdno,
- rq.info.rqe.sdoffset,
- rq.info.rqe.dataoffset,
- rq.info.rqe.groupoffset);
- break;
-
- case loginfo_raid5_data: /* RAID-5 write data block */
- printf("%s 5RD %s %p\t%2d.%-6d 0x%9llx\t%ld\t%d\t%6x\t%6x\t%x\n",
- timetext(&rq.timestamp),
- rq.info.rqe.b.b_iocmd == BIO_READ ? "Read " : "Write",
- rq.bp,
- rq.devmajor,
- rq.devminor,
- rq.info.rqe.b.b_blkno,
- rq.info.rqe.b.b_bcount,
- rq.info.rqe.sdno,
- rq.info.rqe.sdoffset,
- rq.info.rqe.dataoffset,
- rq.info.rqe.groupoffset);
- break;
-
- case loginfo_raid5_parity: /* RAID-5 write parity block */
- printf("%s 6RP %s %p\t%2d.%-6d 0x%9llx\t%ld\t%d\t%6x\t%6x\t%x\n",
- timetext(&rq.timestamp),
- rq.info.rqe.b.b_iocmd == BIO_READ ? "Read " : "Write",
- rq.bp,
- rq.devmajor,
- rq.devminor,
- rq.info.rqe.b.b_blkno,
- rq.info.rqe.b.b_bcount,
- rq.info.rqe.sdno,
- rq.info.rqe.sdoffset,
- rq.info.rqe.dataoffset,
- rq.info.rqe.groupoffset);
- break;
-
- case loginfo_sdio: /* subdisk I/O */
- printf("%s %dVS %s %p\t\t 0x%9llx\t%ld\t%d\n",
- timetext(&rq.timestamp),
- rq.type,
- rq.info.b.b_iocmd == BIO_READ ? "Read " : "Write",
- rq.bp,
- rq.info.b.b_blkno,
- rq.info.b.b_bcount,
- rq.devminor);
- break;
-
- case loginfo_sdiodone: /* subdisk I/O done */
- printf("%s %dSD %s %p\t\t 0x%9llx\t%ld\t%d\n",
- timetext(&rq.timestamp),
- rq.type,
- rq.info.b.b_iocmd == BIO_READ ? "Read " : "Write",
- rq.bp,
- rq.info.b.b_blkno,
- rq.info.b.b_bcount,
- rq.devminor);
- break;
-
- case loginfo_lockwait:
- printf("%s Lockwait %p\t 0x%x\n",
- timetext(&rq.timestamp),
- rq.bp,
- rq.info.lockinfo.stripe);
- break;
-
- case loginfo_lock:
- printf("%s Lock %p\t 0x%x\n",
- timetext(&rq.timestamp),
- rq.bp,
- rq.info.lockinfo.stripe);
- break;
-
- case loginfo_unlock:
- printf("%s Unlock\t %p\t 0x%x\n",
- timetext(&rq.timestamp),
- rq.bp,
- rq.info.lockinfo.stripe);
- break;
- default:
- printf("*** invalid log type: %d ***\n", rq.type);
- }
- if (Verbose > 1)
- lasttime = rq.timestamp;
- }
- }
- }
- }
-}
-
-/*
- * Print config file to a file. This is a userland version
- * of kernel format_config
- */
-void
-vinum_printconfig(int argc, char *argv[], char *argv0[])
-{
- FILE *of;
-
- if (argc > 1) {
- fprintf(stderr, "usage: \tprintconfig [<outfile>]\n");
- return;
- } else if (argc == 1)
- of = fopen(argv[0], "w");
- else
- of = stdout;
- if (of == NULL) {
- fprintf(stderr, "Can't open %s: %s\n", argv[0], strerror(errno));
- return;
- }
- printconfig(of, "");
- if (argc == 1)
- fclose(of);
-}
-
-/*
- * The guts of printconfig. This is called from
- * vinum_printconfig and from vinum_create when
- * called without an argument, in order to give
- * the user something to edit.
- */
-void
-printconfig(FILE * of, char *comment)
-{
- struct utsname uname_s;
- time_t now;
- int i;
- struct _volume vol;
- struct _plex plex;
- struct _sd sd;
- struct _drive drive;
-
- if (ioctl(superdev, VINUM_GETCONFIG, &vinum_conf) < 0) {
- perror("Can't get vinum config");
- return;
- }
- uname(&uname_s); /* get our system name */
- time(&now); /* and the current time */
- fprintf(of,
- "# Vinum configuration of %s, saved at %s",
- uname_s.nodename,
- ctime(&now)); /* say who did it */
-
- if (comment[0] != 0) /* abuse this for commented version */
- fprintf(of, "# Current configuration:\n");
- for (i = 0; i < vinum_conf.drives_allocated; i++) {
- get_drive_info(&drive, i);
- if (drive.state != drive_unallocated) {
- fprintf(of,
- "%sdrive %s device %s\n",
- comment,
- drive.label.name,
- drive.devicename);
- }
- }
-
- for (i = 0; i < vinum_conf.volumes_allocated; i++) {
- get_volume_info(&vol, i);
- if (vol.state != volume_unallocated) {
- if (vol.preferred_plex >= 0) { /* preferences, */
- get_plex_info(&plex, vol.plex[vol.preferred_plex]);
- fprintf(of,
- "%svolume %s readpol prefer %s\n",
- comment,
- vol.name,
- plex.name);
- } else /* default round-robin */
- fprintf(of, "%svolume %s\n", comment, vol.name);
- }
- }
-
- /* Then the plex configuration */
- for (i = 0; i < vinum_conf.plexes_allocated; i++) {
- get_plex_info(&plex, i);
- if (plex.state != plex_unallocated) {
- fprintf(of, "%splex name %s org %s ",
- comment,
- plex.name,
- plex_org(plex.organization));
- if (isstriped((&plex)))
- fprintf(of, "%ds ", (int) plex.stripesize);
- if (plex.volno >= 0) { /* we have a volume */
- get_volume_info(&vol, plex.volno);
- fprintf(of, "vol %s ", vol.name);
- if ((vol.preferred_plex >= 0) /* has a preferred plex */
- &&vol.plex[vol.preferred_plex] == i) /* and it's us */
- fprintf(of, "preferred ");
- } else
- fprintf(of, "detached ");
- fprintf(of, "\n");
- }
- }
-
- /* And finally the subdisk configuration */
- for (i = 0; i < vinum_conf.subdisks_allocated; i++) {
- get_sd_info(&sd, i);
- if (sd.state != sd_unallocated) {
- get_drive_info(&drive, sd.driveno);
- if (sd.plexno >= 0) {
- get_plex_info(&plex, sd.plexno);
- fprintf(of,
- "%ssd name %s drive %s plex %s len %llds driveoffset %llds plexoffset %llds\n",
- comment,
- sd.name,
- drive.label.name,
- plex.name,
- (long long) sd.sectors,
- (long long) sd.driveoffset,
- (long long) sd.plexoffset);
- } else
- fprintf(of,
- "%ssd name %s drive %s detached len %llds driveoffset %llds\n",
- comment,
- sd.name,
- drive.label.name,
- (long long) sd.sectors,
- (long long) sd.driveoffset);
- }
- }
-}
-
-void
-list_defective_objects()
-{
- int o; /* object */
- int heading_needed = 1;
-
- if (ioctl(superdev, VINUM_GETCONFIG, &vinum_conf) < 0) {
- perror("Can't get vinum config");
- return;
- }
- for (o = 0; o < vinum_conf.drives_allocated; o++) {
- get_drive_info(&drive, o);
- if ((drive.state != drive_unallocated) /* drive exists */
- &&(drive.state != drive_up)) { /* but it's not up */
- if (heading_needed) {
- printf("Warning: defective objects\n\n");
- heading_needed = 0;
- }
- vinum_ldi(o, 0); /* print info */
- }
- }
-
- for (o = 0; o < vinum_conf.volumes_allocated; o++) {
- get_volume_info(&vol, o);
- if ((vol.state != volume_unallocated) /* volume exists */
- &&(vol.state != volume_up)) { /* but it's not up */
- if (heading_needed) {
- printf("Warning: defective objects\n\n");
- heading_needed = 0;
- }
- vinum_lvi(o, 0); /* print info */
- }
- }
-
- for (o = 0; o < vinum_conf.plexes_allocated; o++) {
- get_plex_info(&plex, o);
- if ((plex.state != plex_unallocated) /* plex exists */
- &&(plex.state != plex_up)) { /* but it's not up */
- if (heading_needed) {
- printf("Warning: defective objects\n\n");
- heading_needed = 0;
- }
- vinum_lpi(o, 0); /* print info */
- }
- }
-
- for (o = 0; o < vinum_conf.subdisks_allocated; o++) {
- get_sd_info(&sd, o);
- if ((sd.state != sd_unallocated) /* sd exists */
- &&(sd.state != sd_up)) { /* but it's not up */
- if (heading_needed) {
- printf("Warning: defective objects\n\n");
- heading_needed = 0;
- }
- vinum_lsi(o, 0); /* print info */
- }
- }
-}
-
-/* Dump config from specified disk drives */
-void
-vinum_dumpconfig(int argc, char *argv[], char *argv0[])
-{
- int i;
-
- if (argc == 0) { /* dump everything */
- int devs = devstat_getnumdevs(NULL);
- struct statinfo statinfo;
- char *namelist;
- char *enamelist; /* end of name list */
- int i;
- char **token; /* list of tokens */
- int tokens; /* and their number */
-
- bzero(&statinfo, sizeof(struct statinfo));
- statinfo.dinfo = malloc(devs * sizeof(struct statinfo));
- namelist = malloc(devs * (DEVSTAT_NAME_LEN + 8));
- token = malloc((devs + 1) * sizeof(char *));
- if ((statinfo.dinfo == NULL) || (namelist == NULL) || (token == NULL)) {
- fprintf(stderr, "Can't allocate memory for drive list\n");
- return;
- }
- bzero(statinfo.dinfo, sizeof(struct devinfo));
-
- tokens = 0; /* no tokens yet */
- if (devstat_getdevs(NULL, &statinfo) < 0) { /* find out what devices we have */
- perror("Can't get device list");
- return;
- }
- namelist[0] = '\0'; /* start with empty namelist */
- enamelist = namelist; /* point to the end of the list */
-
- for (i = 0; i < devs; i++) {
- struct devstat *stat = &statinfo.dinfo->devices[i];
-
- if (((stat->device_type & DEVSTAT_TYPE_MASK) == DEVSTAT_TYPE_DIRECT) /* disk device */
- &&((stat->device_type & DEVSTAT_TYPE_PASS) == 0) /* and not passthrough */
- &&((stat->device_name[0] != '\0'))) { /* and it has a name */
- sprintf(enamelist, "/dev/%s%d", stat->device_name, stat->unit_number);
- token[tokens] = enamelist; /* point to it */
- tokens++; /* one more token */
- enamelist = &enamelist[strlen(enamelist) + 1]; /* and start beyond the end */
- }
- }
- free(statinfo.dinfo); /* don't need the list any more */
- for (i = 0; i < tokens; i++)
- dumpconfig(token[i]);
- free(namelist);
- free(token);
- } else { /* list specified drives */
- for (i = 0; i < argc; i++)
- dumpconfig(argv[i]);
- }
-}
-
-#define DEVLEN 5
-void
-dumpconfig(char *part)
-{
- char partname[MAXPATHLEN];
- char *partid;
- char partition; /* UNIX partition */
- int slice;
- int founddrive; /* flag when we find a vinum drive */
- struct disklabel label; /* label of this drive */
- int driveno; /* fd of drive */
- int found;
- u_int64_t drivelength;
-
- if (memcmp(part, "/dev/", DEVLEN) == 0) /* starts with /dev */
- memcpy(partname, part, MAXPATHLEN);
- else { /* prepend */
- strcpy(partname, "/dev/");
- strncat(&partname[DEVLEN], part, MAXPATHLEN - DEVLEN);
- }
- partid = &partname[strlen(partname)];
- founddrive = 0; /* no vinum drive found yet on this spindle */
- /* first try the partition table */
- for (slice = 1; slice < 5; slice++) {
- sprintf(partid, "s%dc", slice); /* c partition */
- driveno = open(partname, O_RDONLY);
- if (driveno < 0) {
- if (errno != ENOENT)
- fprintf(stderr, "Can't open %s: %s (%d)\n", partname, strerror(errno), errno);
- continue;
- }
- if (ioctl(driveno, DIOCGDINFO, &label) < 0) {
- if ((errno != EINVAL) || vflag)
- fprintf(stderr, "Can't get label from %s: %s (%d)\n", partname, strerror(errno), errno);
- continue;
- }
- for (partition = 'a'; partition < 'i'; partition++) {
- if ((partition != 'c') /* it's not the c partition */
- &&((label.d_partitions[partition - 'a'].p_fstype == FS_VINUM) /* and it's a Vinum partition */
- ||Verbose)) { /* or we're just plain curious */
- sprintf(partid, "s%d%c", slice, partition);
- found = check_drive(partname); /* try to open it */
- founddrive |= found; /* and note if we were successful at all */
- if (label.d_partitions[partition - 'a'].p_fstype == FS_VINUM) { /* it's a Vinum partition */
- drivelength = ((u_int64_t) label.d_partitions[partition - 'a'].p_size) * DEV_BSIZE;
- printf("Drive %s: %s (%lld bytes)\n",
- partname,
- roughlength(drivelength, 1),
- drivelength);
- if ((!found) && vflag) /* we're talkative */
- printf("*** no configuration found ***\n");
- }
- }
- }
- }
- if (founddrive == 0) { /* didn't find anything, */
- sprintf(partid, "c"); /* c partition */
- driveno = open(partname, O_RDONLY);
- if (driveno < 0) {
- if (errno != ENOENT)
- fprintf(stderr, "Can't open %s: %s (%d)\n", partname, strerror(errno), errno);
- return;
- }
- if (ioctl(driveno, DIOCGDINFO, &label) < 0) {
- fprintf(stderr, "Can't get label from %s: %s (%d)\n", partname, strerror(errno), errno);
- return;
- }
- for (partition = 'a'; partition < 'i'; partition++) { /* try the compatibility partition */
- if ((partition != 'c') /* it's not the c partition */
- &&((label.d_partitions[partition - 'a'].p_fstype == FS_VINUM) /* and it's a Vinum partition */
- ||Verbose)) { /* or we're just plain curious */
- sprintf(partid, "%c", partition);
- found = check_drive(partname); /* try to open it */
- founddrive |= found; /* and note if we were successful at all */
- if (label.d_partitions[partition - 'a'].p_fstype == FS_VINUM) { /* it's a Vinum partition */
- drivelength = ((u_int64_t) label.d_partitions[partition - 'a'].p_size) * DEV_BSIZE;
- printf("Drive %s: %s (%lld bytes)\n",
- partname,
- roughlength(drivelength, 1),
- drivelength);
- if ((!found) && vflag) /* we're talkative */
- printf("*** no configuration found ***\n");
- }
- }
- }
- }
-}
-
-/*
- * Check a drive for a Vinum header. If found,
- * print configuration information from the drive.
- *
- * Return 1 if Vinum config found.
- */
-int
-check_drive(char *devicename)
-{
- int fd;
- char vinumlabel[DEV_BSIZE]; /* one sector for label */
- struct vinum_hdr *hdr = (struct vinum_hdr *) vinumlabel; /* with this structure */
- char *config_text; /* read the config info from disk into here */
- time_t t;
-
- fd = open(devicename, O_RDONLY);
- if (fd >= 0) {
- if (lseek(fd, VINUM_LABEL_OFFSET, SEEK_SET) < 0) {
- fprintf(stderr,
- "Can't seek label for %s: %s (%d)\n",
- devicename,
- strerror(errno),
- errno);
- close(fd);
- return 0;
- }
- if (read(fd, vinumlabel, DEV_BSIZE) != DEV_BSIZE) {
- if (errno != EINVAL)
- fprintf(stderr,
- "Can't read label from %s: %s (%d)\n",
- devicename,
- strerror(errno),
- errno);
- close(fd);
- return 0;
- }
- if ((hdr->magic == VINUM_MAGIC)
- || (vflag && (hdr->magic == VINUM_NOMAGIC))) {
- printf("Drive %s:\tDevice %s\n",
- hdr->label.name,
- devicename);
- if (hdr->magic == VINUM_NOMAGIC)
- printf("*** Drive has been obliterated ***\n");
- t = hdr->label.date_of_birth.tv_sec;
- printf("\t\tCreated on %s at %s",
- hdr->label.sysname,
- ctime(&t));
- t = hdr->label.last_update.tv_sec;
- printf("\t\tConfig last updated %s", /* care: \n at end */
- ctime(&t));
- printf("\t\tSize: %16lld bytes (%lld MB)\n",
- (long long) hdr->label.drive_size, /* bytes used */
- (long long) (hdr->label.drive_size / MEGABYTE));
- config_text = (char *) malloc(MAXCONFIG);
- if (config_text == NULL)
- fprintf(stderr, "Can't allocate memory\n");
- else {
- if (read(fd, config_text, MAXCONFIG) != MAXCONFIG)
- fprintf(stderr,
- "Can't read config from %s: %s (%d)\n",
- devicename,
- strerror(errno),
- errno);
- else
- puts(config_text);
- free(config_text);
- }
- }
- close(fd);
- return 1;
- }
- return 0;
-}
-
-int
-checkupdates()
-{
- int options;
-
- if (ioctl(superdev, VINUM_GETDAEMON, &options) < 0)
- fprintf(stderr, "Can't get daemon options: %s (%d)\n", strerror(errno), errno);
- if (options & daemon_noupdate) {
- fprintf(stderr, "*** Warning: configuration updates are disabled. ***\n");
- return 1;
- } else
- return 0;
-}
-
-/* Local Variables: */
-/* fill-column: 50 */
-/* End: */
OpenPOWER on IntegriCloud