1 /*- 2 * Copyright (c) 1997, 1998, 1999 3 * Nan Yang Computer Services Limited. All rights reserved. 4 * 5 * Written by Greg Lehey 6 * 7 * This software is distributed under the so-called ``Berkeley 8 * License'': 9 * 10 * Redistribution and use in source and binary forms, with or without 11 * modification, are permitted provided that the following conditions 12 * are met: 13 * 1. Redistributions of source code must retain the above copyright 14 * notice, this list of conditions and the following disclaimer. 15 * 2. Redistributions in binary form must reproduce the above copyright 16 * notice, this list of conditions and the following disclaimer in the 17 * documentation and/or other materials provided with the distribution. 18 * 3. All advertising materials mentioning features or use of this software 19 * must display the following acknowledgement: 20 * This product includes software developed by Nan Yang Computer 21 * Services Limited. 22 * 4. Neither the name of the Company nor the names of its contributors 23 * may be used to endorse or promote products derived from this software 24 * without specific prior written permission. 25 * 26 * This software is provided ``as is'', and any express or implied 27 * warranties, including, but not limited to, the implied warranties of 28 * merchantability and fitness for a particular purpose are disclaimed. 29 * In no event shall the company or contributors be liable for any 30 * direct, indirect, incidental, special, exemplary, or consequential 31 * damages (including, but not limited to, procurement of substitute 32 * goods or services; loss of use, data, or profits; or business 33 * interruption) however caused and on any theory of liability, whether 34 * in contract, strict liability, or tort (including negligence or 35 * otherwise) arising in any way out of the use of this software, even if 36 * advised of the possibility of such damage. 37 * 38 * $Id: vinumutil.c,v 1.14 1999/12/30 07:04:02 grog Exp grog $ 39 * $FreeBSD: src/sys/dev/vinum/vinumutil.c,v 1.15 2000/02/29 06:16:44 grog Exp $ 40 * $DragonFly: src/sbin/vinum/vinumutil.c,v 1.3 2003/08/07 21:17:10 dillon Exp $ 41 */ 42 43 /* This file contains utility routines used both in kernel and user context */ 44 45 #include "vinumhdr.h" 46 #include "statetexts.h" 47 #ifndef _KERNEL 48 #include <stdio.h> 49 extern jmp_buf command_fail; /* return on a failed command */ 50 #endif 51 52 static char numeric_state[32]; /* temporary buffer for ASCII conversions */ 53 #define STATECOUNT(x) (sizeof (x##statetext) / sizeof (char *)) 54 /* Return drive state as a string */ 55 char * 56 drive_state(enum drivestate state) 57 { 58 if (((unsigned) state) >= STATECOUNT(drive)) { 59 sprintf(numeric_state, "Invalid state %d", (int) state); 60 return numeric_state; 61 } else 62 return drivestatetext[state]; 63 } 64 65 /* Return volume state as a string */ 66 char * 67 volume_state(enum volumestate state) 68 { 69 if (((unsigned) state) >= STATECOUNT(vol)) { 70 sprintf(numeric_state, "Invalid state %d", (int) state); 71 return numeric_state; 72 } else 73 return volstatetext[state]; 74 } 75 76 /* Return plex state as a string */ 77 char * 78 plex_state(enum plexstate state) 79 { 80 if (((unsigned) state) >= STATECOUNT(plex)) { 81 sprintf(numeric_state, "Invalid state %d", (int) state); 82 return numeric_state; 83 } else 84 return plexstatetext[state]; 85 } 86 87 /* Return plex organization as a string */ 88 char * 89 plex_org(enum plexorg org) 90 { 91 switch (org) { 92 case plex_disorg: /* disorganized */ 93 return "disorg"; 94 break; 95 96 case plex_concat: /* concatenated plex */ 97 return "concat"; 98 break; 99 100 case plex_striped: /* striped plex */ 101 return "striped"; 102 break; 103 104 case plex_raid4: /* RAID-4 plex */ 105 return "raid4"; 106 107 case plex_raid5: /* RAID-5 plex */ 108 return "raid5"; 109 break; 110 111 default: 112 sprintf(numeric_state, "Invalid org %d", (int) org); 113 return numeric_state; 114 } 115 } 116 117 /* Return sd state as a string */ 118 char * 119 sd_state(enum sdstate state) 120 { 121 if (((unsigned) state) >= STATECOUNT(sd)) { 122 sprintf(numeric_state, "Invalid state %d", (int) state); 123 return numeric_state; 124 } else 125 return sdstatetext[state]; 126 } 127 128 /* Now convert in the other direction */ 129 /* 130 * These are currently used only internally, 131 * so we don't do too much error checking 132 */ 133 enum drivestate 134 DriveState(char *text) 135 { 136 int i; 137 for (i = 0; i < STATECOUNT(drive); i++) 138 if (strcmp(text, drivestatetext[i]) == 0) /* found it */ 139 return (enum drivestate) i; 140 return -1; 141 } 142 143 enum sdstate 144 SdState(char *text) 145 { 146 int i; 147 for (i = 0; i < STATECOUNT(sd); i++) 148 if (strcmp(text, sdstatetext[i]) == 0) /* found it */ 149 return (enum sdstate) i; 150 return -1; 151 } 152 153 enum plexstate 154 PlexState(char *text) 155 { 156 int i; 157 for (i = 0; i < STATECOUNT(plex); i++) 158 if (strcmp(text, plexstatetext[i]) == 0) /* found it */ 159 return (enum plexstate) i; 160 return -1; 161 } 162 163 enum volumestate 164 VolState(char *text) 165 { 166 int i; 167 for (i = 0; i < STATECOUNT(vol); i++) 168 if (strcmp(text, volstatetext[i]) == 0) /* found it */ 169 return (enum volumestate) i; 170 return -1; 171 } 172 173 /* 174 * Take a number with an optional scale factor and convert 175 * it to a number of bytes. 176 * 177 * The scale factors are: 178 * 179 * s sectors (of 512 bytes) 180 * b blocks (of 512 bytes). This unit is deprecated, 181 * because it's confusing, but maintained to avoid 182 * confusing Veritas users. 183 * k kilobytes (1024 bytes) 184 * m megabytes (of 1024 * 1024 bytes) 185 * g gigabytes (of 1024 * 1024 * 1024 bytes) 186 */ 187 u_int64_t 188 sizespec(char *spec) 189 { 190 u_int64_t size; 191 char *s; 192 int sign = 1; /* -1 if negative */ 193 194 size = 0; 195 if (spec != NULL) { /* we have a parameter */ 196 s = spec; 197 if (*s == '-') { /* negative, */ 198 sign = -1; 199 s++; /* skip */ 200 } 201 if ((*s >= '0') && (*s <= '9')) { /* it's numeric */ 202 while ((*s >= '0') && (*s <= '9')) /* it's numeric */ 203 size = size * 10 + *s++ - '0'; /* convert it */ 204 switch (*s) { 205 case '\0': 206 return size * sign; 207 208 case 'B': 209 case 'b': 210 case 'S': 211 case 's': 212 return size * sign * 512; 213 214 case 'K': 215 case 'k': 216 return size * sign * 1024; 217 218 case 'M': 219 case 'm': 220 return size * sign * 1024 * 1024; 221 222 case 'G': 223 case 'g': 224 return size * sign * 1024 * 1024 * 1024; 225 } 226 } 227 #ifdef _KERNEL 228 throw_rude_remark(EINVAL, "Invalid length specification: %s", spec); 229 #else 230 fprintf(stderr, "Invalid length specification: %s", spec); 231 longjmp(command_fail, -1); 232 #endif 233 } 234 #ifdef _KERNEL 235 throw_rude_remark(EINVAL, "Missing length specification"); 236 #else 237 fprintf(stderr, "Missing length specification"); 238 longjmp(command_fail, -1); 239 #endif 240 /* NOTREACHED */ 241 return -1; 242 } 243 244 /* 245 * Extract the volume number from a device number. 246 * Perform no checking. 247 */ 248 int 249 Volno(dev_t dev) 250 { 251 return (minor(dev) & MASK(VINUM_VOL_WIDTH)) >> VINUM_VOL_SHIFT; 252 } 253 254 /* 255 * Extract a plex number from a device number. 256 * Don't check the major number, but check the 257 * type. Return -1 for invalid types. 258 */ 259 int 260 Plexno(dev_t dev) 261 { 262 switch (DEVTYPE(dev)) { 263 case VINUM_VOLUME_TYPE: 264 case VINUM_DRIVE_TYPE: 265 case VINUM_SUPERDEV_TYPE: /* ordinary super device */ 266 case VINUM_RAWSD_TYPE: 267 return -1; 268 269 case VINUM_PLEX_TYPE: 270 case VINUM_SD_TYPE: 271 return VOL[Volno(dev)].plex[(minor(dev) >> VINUM_PLEX_SHIFT) & (MASK(VINUM_PLEX_WIDTH))]; 272 273 case VINUM_RAWPLEX_TYPE: 274 return ((minor(dev) & MASK(VINUM_VOL_WIDTH)) >> VINUM_VOL_SHIFT) /* low order 8 bits */ 275 |((minor(dev) >> VINUM_RAWPLEX_SHIFT) 276 & (MASK(VINUM_RAWPLEX_WIDTH) 277 << (VINUM_VOL_SHIFT + VINUM_VOL_WIDTH))); /* upper 12 bits */ 278 } 279 return 0; /* compiler paranoia */ 280 } 281 282 /* 283 * Extract a subdisk number from a device number. 284 * Don't check the major number, but check the 285 * type. Return -1 for invalid types. 286 */ 287 int 288 Sdno(dev_t dev) 289 { 290 switch (DEVTYPE(dev)) { 291 case VINUM_VOLUME_TYPE: 292 case VINUM_DRIVE_TYPE: 293 case VINUM_SUPERDEV_TYPE: /* ordinary super device */ 294 case VINUM_PLEX_TYPE: 295 case VINUM_RAWPLEX_TYPE: 296 return -1; 297 298 case VINUM_SD_TYPE: 299 return PLEX[Plexno(dev)].sdnos[(minor(dev) >> VINUM_SD_SHIFT) & (MASK(VINUM_SD_WIDTH))]; 300 301 case VINUM_RAWSD_TYPE: 302 return ((minor(dev) & MASK(VINUM_VOL_WIDTH)) >> VINUM_VOL_SHIFT) /* low order 8 bits */ 303 |((minor(dev) >> VINUM_RAWPLEX_SHIFT) & (MASK(VINUM_RAWPLEX_WIDTH) 304 << (VINUM_VOL_SHIFT + VINUM_VOL_WIDTH))); /* upper 12 bits */ 305 } 306 return -1; /* compiler paranoia */ 307 } 308