Home | History | Annotate | Line # | Download | only in dkctl
dkctl.c revision 1.1
      1 /*	$NetBSD: dkctl.c,v 1.1 2002/01/09 22:30:14 thorpej Exp $	*/
      2 
      3 /*
      4  * Copyright 2001 Wasabi Systems, Inc.
      5  * All rights reserved.
      6  *
      7  * Written by Jason R. Thorpe for Wasabi Systems, Inc.
      8  *
      9  * Redistribution and use in source and binary forms, with or without
     10  * modification, are permitted provided that the following conditions
     11  * are met:
     12  * 1. Redistributions of source code must retain the above copyright
     13  *    notice, this list of conditions and the following disclaimer.
     14  * 2. Redistributions in binary form must reproduce the above copyright
     15  *    notice, this list of conditions and the following disclaimer in the
     16  *    documentation and/or other materials provided with the distribution.
     17  * 3. All advertising materials mentioning features or use of this software
     18  *    must display the following acknowledgement:
     19  *	This product includes software developed for the NetBSD Project by
     20  *	Wasabi Systems, Inc.
     21  * 4. The name of Wasabi Systems, Inc. may not be used to endorse
     22  *    or promote products derived from this software without specific prior
     23  *    written permission.
     24  *
     25  * THIS SOFTWARE IS PROVIDED BY WASABI SYSTEMS, INC. ``AS IS'' AND
     26  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
     27  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
     28  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL WASABI SYSTEMS, INC
     29  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
     30  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
     31  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
     32  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
     33  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
     34  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
     35  * POSSIBILITY OF SUCH DAMAGE.
     36  */
     37 
     38 /*
     39  * dkctl(8) -- a program to manipulate disks.
     40  */
     41 
     42 #include <sys/param.h>
     43 #include <sys/ioctl.h>
     44 #include <sys/dkio.h>
     45 #include <err.h>
     46 #include <errno.h>
     47 #include <fcntl.h>
     48 #include <stdio.h>
     49 #include <stdlib.h>
     50 #include <string.h>
     51 #include <unistd.h>
     52 #include <util.h>
     53 
     54 struct command {
     55 	const char *cmd_name;
     56 	const char *arg_names;
     57 	void (*cmd_func)(int, char *[]);
     58 	int open_flags;
     59 };
     60 
     61 int	main(int, char *[]);
     62 void	usage(void);
     63 
     64 int	fd;				/* file descriptor for device */
     65 const	char *dvname;			/* device name */
     66 char	dvname_store[MAXPATHLEN];	/* for opendisk(3) */
     67 const	char *cmdname;			/* command user issued */
     68 const	char *argnames;			/* helpstring; expected arguments */
     69 
     70 void	disk_getcache(int, char *[]);
     71 void	disk_setcache(int, char *[]);
     72 void	disk_synccache(int, char *[]);
     73 
     74 struct command commands[] = {
     75 	{ "getcache",
     76 	  "",
     77 	  disk_getcache,
     78 	  O_RDONLY },
     79 
     80 	{ "setcache",
     81 	  "none | r | w | rw [save]",
     82 	  disk_setcache,
     83 	  O_RDWR },
     84 
     85 	{ "synccache",
     86 	  "[force]",
     87 	  disk_synccache,
     88 	  O_RDWR },
     89 
     90 	{ NULL,
     91 	  NULL,
     92 	  NULL,
     93 	  0 },
     94 };
     95 
     96 int
     97 main(int argc, char *argv[])
     98 {
     99 	int i;
    100 
    101 	/* Must have at least: device command */
    102 	if (argc < 3)
    103 		usage();
    104 
    105 	/* Skip program name, get and skip device name and command. */
    106 	dvname = argv[1];
    107 	cmdname = argv[2];
    108 	argv += 3;
    109 	argc -= 3;
    110 
    111 	/* Look up and call the command. */
    112 	for (i = 0; commands[i].cmd_name != NULL; i++)
    113 		if (strcmp(cmdname, commands[i].cmd_name) == 0)
    114 			break;
    115 	if (commands[i].cmd_name == NULL)
    116 		errx(1, "unknown command: %s", cmdname);
    117 
    118 	argnames = commands[i].arg_names;
    119 
    120 	/* Open the device. */
    121 	fd = opendisk(dvname, commands[i].open_flags, dvname_store,
    122 	    sizeof(dvname_store), 0);
    123 	if (fd == -1)
    124 		err(1, "%s", dvname);
    125 
    126 	dvname = dvname_store;
    127 
    128 	(*commands[i].cmd_func)(argc, argv);
    129 	exit(0);
    130 }
    131 
    132 void
    133 usage()
    134 {
    135 	int i;
    136 
    137 	fprintf(stderr, "Usage: %s device command [arg [...]]\n",
    138 	    getprogname());
    139 
    140 	fprintf(stderr, "   Available commands:\n");
    141 	for (i = 0; commands[i].cmd_name != NULL; i++)
    142 		fprintf(stderr, "\t%s %s\n", commands[i].cmd_name,
    143 		    commands[i].arg_names);
    144 
    145 	exit(1);
    146 }
    147 
    148 void
    149 disk_getcache(int argc, char *argv[])
    150 {
    151 	int bits;
    152 
    153 	if (ioctl(fd, DIOCGCACHE, &bits) == -1)
    154 		err(1, "%s: getcache", dvname);
    155 
    156 	if ((bits & (DKCACHE_READ|DKCACHE_WRITE)) == 0)
    157 		printf("%s: No caches enabled\n", dvname);
    158 	else {
    159 		if (bits & DKCACHE_READ)
    160 			printf("%s: read cache enabled\n", dvname);
    161 		if (bits & DKCACHE_WRITE)
    162 			printf("%s: write-back cache enabled\n", dvname);
    163 	}
    164 
    165 	printf("%s: read cache enable is %schangeable\n", dvname,
    166 	    (bits & DKCACHE_RCHANGE) ? "" : "not ");
    167 	printf("%s: write cache enable is %schangeable\n", dvname,
    168 	    (bits & DKCACHE_WCHANGE) ? "" : "not ");
    169 
    170 	printf("%s: cache parameters are %ssavable\n", dvname,
    171 	    (bits & DKCACHE_SAVE) ? "" : "not ");
    172 }
    173 
    174 void
    175 disk_setcache(int argc, char *argv[])
    176 {
    177 	int bits;
    178 
    179 	if (argc > 2 || argc == 0)
    180 		usage();
    181 
    182 	if (strcmp(argv[0], "none") == 0)
    183 		bits = 0;
    184 	else if (strcmp(argv[0], "r") == 0)
    185 		bits = DKCACHE_READ;
    186 	else if (strcmp(argv[0], "w") == 0)
    187 		bits = DKCACHE_WRITE;
    188 	else if (strcmp(argv[0], "rw") == 0)
    189 		bits = DKCACHE_READ|DKCACHE_WRITE;
    190 	else
    191 		usage();
    192 
    193 	if (argc == 2) {
    194 		if (strcmp(argv[1], "save") == 0)
    195 			bits |= DKCACHE_SAVE;
    196 		else
    197 			usage();
    198 	}
    199 
    200 	if (ioctl(fd, DIOCSCACHE, &bits) == -1)
    201 		err(1, "%s: setcache", dvname);
    202 }
    203 
    204 void
    205 disk_synccache(int argc, char *argv[])
    206 {
    207 	int force;
    208 
    209 	switch (argc) {
    210 	case 0:
    211 		force = 0;
    212 		break;
    213 
    214 	case 1:
    215 		if (strcmp(argv[0], "force") == 0)
    216 			force = 1;
    217 		else
    218 			usage();
    219 		break;
    220 
    221 	default:
    222 		usage();
    223 	}
    224 
    225 	if (ioctl(fd, DIOCCACHESYNC, &force) == -1)
    226 		err(1, "%s: sync cache", dvname);
    227 }
    228