1 1.86 rillig /* $NetBSD: raidctl.c,v 1.86 2024/11/05 20:07:41 rillig Exp $ */ 2 1.18 thorpej 3 1.1 oster /*- 4 1.1 oster * Copyright (c) 1996, 1997, 1998 The NetBSD Foundation, Inc. 5 1.1 oster * All rights reserved. 6 1.1 oster * 7 1.1 oster * This code is derived from software contributed to The NetBSD Foundation 8 1.1 oster * by Greg Oster 9 1.1 oster * 10 1.1 oster * Redistribution and use in source and binary forms, with or without 11 1.1 oster * modification, are permitted provided that the following conditions 12 1.1 oster * are met: 13 1.1 oster * 1. Redistributions of source code must retain the above copyright 14 1.1 oster * notice, this list of conditions and the following disclaimer. 15 1.1 oster * 2. Redistributions in binary form must reproduce the above copyright 16 1.1 oster * notice, this list of conditions and the following disclaimer in the 17 1.1 oster * documentation and/or other materials provided with the distribution. 18 1.1 oster * 19 1.1 oster * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS 20 1.1 oster * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED 21 1.1 oster * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 22 1.1 oster * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS 23 1.1 oster * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 24 1.1 oster * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 25 1.1 oster * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 26 1.1 oster * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 27 1.1 oster * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 28 1.1 oster * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 29 1.1 oster * POSSIBILITY OF SUCH DAMAGE. 30 1.1 oster */ 31 1.1 oster 32 1.85 rillig /* 33 1.85 rillig * This program is a re-write of the original rf_ctrl program 34 1.18 thorpej * distributed by CMU with RAIDframe 1.1. 35 1.18 thorpej * 36 1.18 thorpej * This program is the user-land interface to the RAIDframe kernel 37 1.18 thorpej * driver in NetBSD. 38 1.1 oster */ 39 1.33 agc #include <sys/cdefs.h> 40 1.33 agc 41 1.33 agc #ifndef lint 42 1.86 rillig __RCSID("$NetBSD: raidctl.c,v 1.86 2024/11/05 20:07:41 rillig Exp $"); 43 1.33 agc #endif 44 1.33 agc 45 1.1 oster 46 1.1 oster #include <sys/param.h> 47 1.1 oster #include <sys/ioctl.h> 48 1.1 oster #include <sys/stat.h> 49 1.18 thorpej #include <sys/disklabel.h> 50 1.18 thorpej 51 1.1 oster #include <ctype.h> 52 1.1 oster #include <err.h> 53 1.2 mjacob #include <errno.h> 54 1.25 lukem #include <fcntl.h> 55 1.25 lukem #include <stdio.h> 56 1.25 lukem #include <stdlib.h> 57 1.1 oster #include <string.h> 58 1.59 christos #include <inttypes.h> 59 1.3 oster #include <unistd.h> 60 1.25 lukem #include <util.h> 61 1.3 oster 62 1.29 oster #include <dev/raidframe/raidframevar.h> 63 1.29 oster #include <dev/raidframe/raidframeio.h> 64 1.29 oster #include "rf_configure.h" 65 1.50 pooka #include "prog_ops.h" 66 1.1 oster 67 1.79 oster #ifndef RAIDFRAME_REMOVE_COMPONENT 68 1.79 oster #define RAIDFRAME_REMOVE_COMPONENT RAIDFRAME_REMOVE_HOT_SPARE 69 1.79 oster #endif 70 1.79 oster 71 1.78 kre #define CONFIGURE_TEST 1 /* must be different from any raidframe ioctl */ 72 1.78 kre 73 1.36 xtraeme void do_ioctl(int, u_long, void *, const char *); 74 1.36 xtraeme static void rf_configure(int, char*, int); 75 1.36 xtraeme static const char *device_status(RF_DiskStatus_t); 76 1.36 xtraeme static void rf_get_device_status(int); 77 1.36 xtraeme static void rf_output_configuration(int, const char *); 78 1.36 xtraeme static void get_component_number(int, char *, int *, int *); 79 1.36 xtraeme static void rf_fail_disk(int, char *, int); 80 1.53 joerg __dead static void usage(void); 81 1.36 xtraeme static void get_component_label(int, char *); 82 1.36 xtraeme static void set_component_label(int, char *); 83 1.36 xtraeme static void init_component_labels(int, int); 84 1.36 xtraeme static void set_autoconfig(int, int, char *); 85 1.36 xtraeme static void add_hot_spare(int, char *); 86 1.79 oster static void remove_component(int, char *); 87 1.36 xtraeme static void rebuild_in_place(int, char *); 88 1.36 xtraeme static void check_status(int,int); 89 1.36 xtraeme static void check_parity(int,int, char *); 90 1.36 xtraeme static void do_meter(int, u_long); 91 1.36 xtraeme static void get_bar(char *, double, int); 92 1.68 mrg static void get_time_string(char *, size_t, int); 93 1.42 jld static void rf_output_pmstat(int, int); 94 1.42 jld static void rf_pm_configure(int, int, char *, int[]); 95 1.80 oster static void rf_simple_create(int, int, char *[]); 96 1.59 christos static unsigned int xstrtouint(const char *); 97 1.10 oster 98 1.18 thorpej int verbose; 99 1.1 oster 100 1.57 christos static const char *rootpart[] = { "No", "Force", "Soft", "*invalid*" }; 101 1.57 christos 102 1.72 mlelstv static void 103 1.72 mlelstv get_comp(char *buf, char *arg, size_t bufsz) 104 1.72 mlelstv { 105 1.72 mlelstv if (getfsspecname(buf, bufsz, arg) == NULL) 106 1.72 mlelstv errx(1,"%s",buf); 107 1.72 mlelstv } 108 1.72 mlelstv 109 1.1 oster int 110 1.36 xtraeme main(int argc,char *argv[]) 111 1.1 oster { 112 1.42 jld int ch, i; 113 1.1 oster int num_options; 114 1.1 oster unsigned long action; 115 1.1 oster char config_filename[PATH_MAX]; 116 1.1 oster char dev_name[PATH_MAX]; 117 1.1 oster char name[PATH_MAX]; 118 1.5 oster char component[PATH_MAX]; 119 1.12 oster char autoconf[10]; 120 1.42 jld char *parityconf = NULL; 121 1.42 jld int parityparams[3]; 122 1.27 lukem int do_output; 123 1.1 oster int do_recon; 124 1.7 oster int do_rewrite; 125 1.1 oster int raidID; 126 1.5 oster int serial_number; 127 1.1 oster struct stat st; 128 1.1 oster int fd; 129 1.5 oster int force; 130 1.30 simonb int openmode; 131 1.64 christos int last_unit; 132 1.80 oster struct timeval tv; 133 1.85 rillig 134 1.1 oster num_options = 0; 135 1.1 oster action = 0; 136 1.27 lukem do_output = 0; 137 1.1 oster do_recon = 0; 138 1.7 oster do_rewrite = 0; 139 1.38 lukem serial_number = 0; 140 1.5 oster force = 0; 141 1.64 christos last_unit = 0; 142 1.30 simonb openmode = O_RDWR; /* default to read/write */ 143 1.1 oster 144 1.80 oster if (argc > 5) { 145 1.80 oster /* we have at least 5 args, so it might be a simplified config */ 146 1.85 rillig 147 1.80 oster strlcpy(name, argv[1], sizeof(name)); 148 1.80 oster fd = opendisk(name, openmode, dev_name, sizeof(dev_name), 0); 149 1.80 oster if (fd != -1) { 150 1.80 oster /* we were able to open the device... */ 151 1.80 oster if (fstat(fd, &st) == -1) 152 1.80 oster err(1, "stat failure on: %s", dev_name); 153 1.80 oster if (!S_ISBLK(st.st_mode) && !S_ISCHR(st.st_mode)) 154 1.80 oster err(1, "invalid device: %s", dev_name); 155 1.85 rillig 156 1.80 oster raidID = DISKUNIT(st.st_rdev); 157 1.80 oster if (strncmp(argv[2],"create",6)==0) { 158 1.80 oster rf_simple_create(fd,argc-3,&argv[3]); 159 1.85 rillig 160 1.80 oster /* set serial number, set autoconfig, init parity */ 161 1.80 oster 162 1.80 oster if (gettimeofday(&tv,NULL) == -1) { 163 1.80 oster serial_number = 12345777; 164 1.80 oster } else { 165 1.80 oster serial_number = tv.tv_sec; 166 1.80 oster } 167 1.80 oster init_component_labels(fd, serial_number); 168 1.80 oster strlcpy(autoconf, "yes", sizeof(autoconf)); 169 1.80 oster set_autoconfig(fd, raidID, autoconf); 170 1.85 rillig 171 1.80 oster } else 172 1.80 oster usage(); 173 1.85 rillig 174 1.80 oster close(fd); 175 1.80 oster exit(0); 176 1.80 oster } 177 1.80 oster 178 1.80 oster /* otherwise we go back to regular parsing */ 179 1.80 oster } 180 1.85 rillig 181 1.78 kre while ((ch = getopt(argc, argv, 182 1.84 rillig "a:A:c:C:f:F:g:GiI:l:LmM:r:R:sSpPt:uU:v")) != -1) 183 1.78 kre switch (ch) { 184 1.5 oster case 'a': 185 1.5 oster action = RAIDFRAME_ADD_HOT_SPARE; 186 1.72 mlelstv get_comp(component, optarg, sizeof(component)); 187 1.5 oster num_options++; 188 1.5 oster break; 189 1.12 oster case 'A': 190 1.12 oster action = RAIDFRAME_SET_AUTOCONFIG; 191 1.32 itojun strlcpy(autoconf, optarg, sizeof(autoconf)); 192 1.12 oster num_options++; 193 1.12 oster break; 194 1.1 oster case 'c': 195 1.6 oster action = RAIDFRAME_CONFIGURE; 196 1.32 itojun strlcpy(config_filename, optarg, 197 1.32 itojun sizeof(config_filename)); 198 1.6 oster force = 0; 199 1.1 oster num_options++; 200 1.1 oster break; 201 1.1 oster case 'C': 202 1.32 itojun strlcpy(config_filename, optarg, 203 1.32 itojun sizeof(config_filename)); 204 1.6 oster action = RAIDFRAME_CONFIGURE; 205 1.6 oster force = 1; 206 1.1 oster num_options++; 207 1.1 oster break; 208 1.1 oster case 'f': 209 1.1 oster action = RAIDFRAME_FAIL_DISK; 210 1.72 mlelstv get_comp(component, optarg, sizeof(component)); 211 1.1 oster do_recon = 0; 212 1.1 oster num_options++; 213 1.1 oster break; 214 1.1 oster case 'F': 215 1.1 oster action = RAIDFRAME_FAIL_DISK; 216 1.72 mlelstv get_comp(component, optarg, sizeof(component)); 217 1.1 oster do_recon = 1; 218 1.5 oster num_options++; 219 1.5 oster break; 220 1.5 oster case 'g': 221 1.5 oster action = RAIDFRAME_GET_COMPONENT_LABEL; 222 1.72 mlelstv get_comp(component, optarg, sizeof(component)); 223 1.30 simonb openmode = O_RDONLY; 224 1.5 oster num_options++; 225 1.5 oster break; 226 1.27 lukem case 'G': 227 1.27 lukem action = RAIDFRAME_GET_INFO; 228 1.30 simonb openmode = O_RDONLY; 229 1.27 lukem do_output = 1; 230 1.27 lukem num_options++; 231 1.27 lukem break; 232 1.6 oster case 'i': 233 1.6 oster action = RAIDFRAME_REWRITEPARITY; 234 1.6 oster num_options++; 235 1.6 oster break; 236 1.5 oster case 'I': 237 1.5 oster action = RAIDFRAME_INIT_LABELS; 238 1.59 christos serial_number = xstrtouint(optarg); 239 1.5 oster num_options++; 240 1.5 oster break; 241 1.85 rillig case 'l': 242 1.73 oster action = RAIDFRAME_SET_COMPONENT_LABEL; 243 1.73 oster get_comp(component, optarg, sizeof(component)); 244 1.73 oster num_options++; 245 1.73 oster break; 246 1.74 oster case 'L': 247 1.74 oster action = RAIDFRAME_RESCAN; 248 1.74 oster num_options++; 249 1.74 oster break; 250 1.42 jld case 'm': 251 1.42 jld action = RAIDFRAME_PARITYMAP_STATUS; 252 1.42 jld openmode = O_RDONLY; 253 1.42 jld num_options++; 254 1.42 jld break; 255 1.42 jld case 'M': 256 1.42 jld action = RAIDFRAME_PARITYMAP_SET_DISABLE; 257 1.42 jld parityconf = strdup(optarg); 258 1.42 jld num_options++; 259 1.58 manu /* XXXjld: should rf_pm_configure do the strtol()s? */ 260 1.42 jld i = 0; 261 1.42 jld while (i < 3 && optind < argc && 262 1.86 rillig isdigit((unsigned char)argv[optind][0])) 263 1.59 christos parityparams[i++] = xstrtouint(argv[optind++]); 264 1.42 jld while (i < 3) 265 1.42 jld parityparams[i++] = 0; 266 1.42 jld break; 267 1.76 kre case 'p': 268 1.76 kre action = RAIDFRAME_CHECK_PARITY; 269 1.76 kre openmode = O_RDONLY; 270 1.76 kre num_options++; 271 1.76 kre break; 272 1.76 kre case 'P': 273 1.76 kre action = RAIDFRAME_CHECK_PARITY; 274 1.76 kre do_rewrite = 1; 275 1.76 kre num_options++; 276 1.76 kre break; 277 1.1 oster case 'r': 278 1.79 oster action = RAIDFRAME_REMOVE_COMPONENT; 279 1.72 mlelstv get_comp(component, optarg, sizeof(component)); 280 1.1 oster num_options++; 281 1.1 oster break; 282 1.1 oster case 'R': 283 1.72 mlelstv get_comp(component, optarg, sizeof(component)); 284 1.6 oster action = RAIDFRAME_REBUILD_IN_PLACE; 285 1.1 oster num_options++; 286 1.1 oster break; 287 1.1 oster case 's': 288 1.1 oster action = RAIDFRAME_GET_INFO; 289 1.30 simonb openmode = O_RDONLY; 290 1.1 oster num_options++; 291 1.1 oster break; 292 1.6 oster case 'S': 293 1.19 oster action = RAIDFRAME_CHECK_RECON_STATUS_EXT; 294 1.30 simonb openmode = O_RDONLY; 295 1.6 oster num_options++; 296 1.6 oster break; 297 1.78 kre case 't': 298 1.78 kre action = CONFIGURE_TEST; 299 1.78 kre strlcpy(config_filename, optarg, 300 1.78 kre sizeof(config_filename)); 301 1.78 kre num_options++; 302 1.78 kre break; 303 1.1 oster case 'u': 304 1.1 oster action = RAIDFRAME_SHUTDOWN; 305 1.1 oster num_options++; 306 1.1 oster break; 307 1.64 christos case 'U': 308 1.64 christos action = RAIDFRAME_SET_LAST_UNIT; 309 1.64 christos num_options++; 310 1.64 christos last_unit = atoi(optarg); 311 1.64 christos if (last_unit < 0) 312 1.64 christos errx(1, "Bad last unit %s", optarg); 313 1.64 christos break; 314 1.10 oster case 'v': 315 1.10 oster verbose = 1; 316 1.85 rillig /* Don't bump num_options, as '-v' is not 317 1.10 oster an option like the others */ 318 1.10 oster /* num_options++; */ 319 1.10 oster break; 320 1.1 oster default: 321 1.1 oster usage(); 322 1.1 oster } 323 1.1 oster argc -= optind; 324 1.1 oster argv += optind; 325 1.1 oster 326 1.78 kre if (num_options > 1) 327 1.78 kre usage(); 328 1.78 kre 329 1.78 kre if (action == CONFIGURE_TEST) { 330 1.78 kre RF_Config_t cfg; 331 1.78 kre 332 1.78 kre if (argc != 0) 333 1.78 kre usage(); 334 1.78 kre if (rf_MakeConfig(config_filename, &cfg) != 0) 335 1.78 kre exit(1); 336 1.78 kre exit(0);; 337 1.78 kre } 338 1.78 kre 339 1.78 kre if (argc != 1) 340 1.1 oster usage(); 341 1.1 oster 342 1.50 pooka if (prog_init && prog_init() == -1) 343 1.50 pooka err(1, "init failed"); 344 1.50 pooka 345 1.32 itojun strlcpy(name, argv[0], sizeof(name)); 346 1.41 pooka fd = opendisk1(name, openmode, dev_name, sizeof(dev_name), 0, 347 1.50 pooka prog_open); 348 1.45 christos if (fd == -1) 349 1.45 christos err(1, "Unable to open device file: %s", name); 350 1.51 pooka if (prog_fstat(fd, &st) == -1) 351 1.45 christos err(1, "stat failure on: %s", dev_name); 352 1.45 christos if (!S_ISBLK(st.st_mode) && !S_ISCHR(st.st_mode)) 353 1.45 christos err(1, "invalid device: %s", dev_name); 354 1.1 oster 355 1.28 oster raidID = DISKUNIT(st.st_rdev); 356 1.1 oster 357 1.75 kre switch (action) { 358 1.5 oster case RAIDFRAME_ADD_HOT_SPARE: 359 1.12 oster add_hot_spare(fd, component); 360 1.5 oster break; 361 1.79 oster case RAIDFRAME_REMOVE_COMPONENT: 362 1.79 oster remove_component(fd, component); 363 1.6 oster break; 364 1.1 oster case RAIDFRAME_CONFIGURE: 365 1.12 oster rf_configure(fd, config_filename, force); 366 1.12 oster break; 367 1.12 oster case RAIDFRAME_SET_AUTOCONFIG: 368 1.12 oster set_autoconfig(fd, raidID, autoconf); 369 1.1 oster break; 370 1.1 oster case RAIDFRAME_FAIL_DISK: 371 1.12 oster rf_fail_disk(fd, component, do_recon); 372 1.5 oster break; 373 1.5 oster case RAIDFRAME_SET_COMPONENT_LABEL: 374 1.12 oster set_component_label(fd, component); 375 1.5 oster break; 376 1.5 oster case RAIDFRAME_GET_COMPONENT_LABEL: 377 1.12 oster get_component_label(fd, component); 378 1.5 oster break; 379 1.5 oster case RAIDFRAME_INIT_LABELS: 380 1.12 oster init_component_labels(fd, serial_number); 381 1.1 oster break; 382 1.1 oster case RAIDFRAME_REWRITEPARITY: 383 1.1 oster printf("Initiating re-write of parity\n"); 384 1.85 rillig do_ioctl(fd, RAIDFRAME_REWRITEPARITY, NULL, 385 1.1 oster "RAIDFRAME_REWRITEPARITY"); 386 1.10 oster if (verbose) { 387 1.10 oster sleep(3); /* XXX give it time to get started */ 388 1.10 oster printf("Parity Re-write status:\n"); 389 1.19 oster do_meter(fd, RAIDFRAME_CHECK_PARITYREWRITE_STATUS_EXT); 390 1.10 oster } 391 1.1 oster break; 392 1.19 oster case RAIDFRAME_CHECK_RECON_STATUS_EXT: 393 1.14 oster check_status(fd,1); 394 1.1 oster break; 395 1.1 oster case RAIDFRAME_GET_INFO: 396 1.27 lukem if (do_output) 397 1.27 lukem rf_output_configuration(fd, dev_name); 398 1.27 lukem else 399 1.27 lukem rf_get_device_status(fd); 400 1.1 oster break; 401 1.42 jld case RAIDFRAME_PARITYMAP_STATUS: 402 1.42 jld rf_output_pmstat(fd, raidID); 403 1.42 jld break; 404 1.42 jld case RAIDFRAME_PARITYMAP_SET_DISABLE: 405 1.42 jld rf_pm_configure(fd, raidID, parityconf, parityparams); 406 1.42 jld break; 407 1.6 oster case RAIDFRAME_REBUILD_IN_PLACE: 408 1.12 oster rebuild_in_place(fd, component); 409 1.6 oster break; 410 1.7 oster case RAIDFRAME_CHECK_PARITY: 411 1.12 oster check_parity(fd, do_rewrite, dev_name); 412 1.7 oster break; 413 1.1 oster case RAIDFRAME_SHUTDOWN: 414 1.1 oster do_ioctl(fd, RAIDFRAME_SHUTDOWN, NULL, "RAIDFRAME_SHUTDOWN"); 415 1.1 oster break; 416 1.64 christos case RAIDFRAME_SET_LAST_UNIT: 417 1.64 christos do_ioctl(fd, RAIDFRAME_SET_LAST_UNIT, &last_unit, 418 1.64 christos "RAIDFRAME_SET_LAST_UNIT"); 419 1.64 christos break; 420 1.74 oster case RAIDFRAME_RESCAN: 421 1.74 oster do_ioctl(fd, RAIDFRAME_RESCAN, NULL, "RAIDFRAME_RESCAN"); 422 1.74 oster break; 423 1.1 oster default: 424 1.1 oster break; 425 1.1 oster } 426 1.1 oster 427 1.50 pooka prog_close(fd); 428 1.1 oster exit(0); 429 1.1 oster } 430 1.1 oster 431 1.10 oster void 432 1.36 xtraeme do_ioctl(int fd, unsigned long command, void *arg, const char *ioctl_name) 433 1.1 oster { 434 1.50 pooka if (prog_ioctl(fd, command, arg) == -1) 435 1.45 christos err(1, "ioctl (%s) failed", ioctl_name); 436 1.1 oster } 437 1.1 oster 438 1.1 oster 439 1.1 oster static void 440 1.36 xtraeme rf_configure(int fd, char *config_file, int force) 441 1.1 oster { 442 1.1 oster void *generic; 443 1.1 oster RF_Config_t cfg; 444 1.1 oster 445 1.45 christos if (rf_MakeConfig( config_file, &cfg ) != 0) 446 1.45 christos err(1, "Unable to create RAIDframe configuration structure"); 447 1.85 rillig 448 1.6 oster cfg.force = force; 449 1.6 oster 450 1.85 rillig /* 451 1.18 thorpej * Note the extra level of redirection needed here, since 452 1.85 rillig * what we really want to pass in is a pointer to the pointer to 453 1.85 rillig * the configuration structure. 454 1.1 oster */ 455 1.1 oster 456 1.45 christos generic = &cfg; 457 1.6 oster do_ioctl(fd, RAIDFRAME_CONFIGURE, &generic, "RAIDFRAME_CONFIGURE"); 458 1.1 oster } 459 1.1 oster 460 1.18 thorpej static const char * 461 1.36 xtraeme device_status(RF_DiskStatus_t status) 462 1.1 oster { 463 1.1 oster 464 1.1 oster switch (status) { 465 1.1 oster case rf_ds_optimal: 466 1.18 thorpej return ("optimal"); 467 1.1 oster case rf_ds_failed: 468 1.18 thorpej return ("failed"); 469 1.1 oster case rf_ds_reconstructing: 470 1.18 thorpej return ("reconstructing"); 471 1.1 oster case rf_ds_dist_spared: 472 1.18 thorpej return ("dist_spared"); 473 1.1 oster case rf_ds_spared: 474 1.18 thorpej return ("spared"); 475 1.1 oster case rf_ds_spare: 476 1.18 thorpej return ("spare"); 477 1.1 oster case rf_ds_used_spare: 478 1.18 thorpej return ("used_spare"); 479 1.1 oster default: 480 1.18 thorpej return ("UNKNOWN"); 481 1.1 oster } 482 1.18 thorpej /* NOTREACHED */ 483 1.1 oster } 484 1.1 oster 485 1.1 oster static void 486 1.36 xtraeme rf_get_device_status(int fd) 487 1.1 oster { 488 1.1 oster RF_DeviceConfig_t device_config; 489 1.1 oster void *cfg_ptr; 490 1.8 oster int is_clean; 491 1.70 mlelstv int i, nspares; 492 1.1 oster 493 1.1 oster cfg_ptr = &device_config; 494 1.1 oster 495 1.1 oster do_ioctl(fd, RAIDFRAME_GET_INFO, &cfg_ptr, "RAIDFRAME_GET_INFO"); 496 1.1 oster 497 1.1 oster printf("Components:\n"); 498 1.1 oster for(i=0; i < device_config.ndevs; i++) { 499 1.85 rillig printf("%20s: %s\n", device_config.devs[i].devname, 500 1.1 oster device_status(device_config.devs[i].status)); 501 1.1 oster } 502 1.70 mlelstv 503 1.71 mlelstv nspares = MIN(device_config.nspares, 504 1.70 mlelstv __arraycount(device_config.spares)); 505 1.70 mlelstv 506 1.70 mlelstv if (nspares > 0) { 507 1.1 oster printf("Spares:\n"); 508 1.70 mlelstv for(i=0; i < nspares; i++) { 509 1.5 oster printf("%20s: %s\n", 510 1.85 rillig device_config.spares[i].devname, 511 1.1 oster device_status(device_config.spares[i].status)); 512 1.1 oster } 513 1.1 oster } else { 514 1.1 oster printf("No spares.\n"); 515 1.8 oster } 516 1.14 oster for(i=0; i < device_config.ndevs; i++) { 517 1.14 oster if (device_config.devs[i].status == rf_ds_optimal) { 518 1.14 oster get_component_label(fd, device_config.devs[i].devname); 519 1.14 oster } else { 520 1.14 oster printf("%s status is: %s. Skipping label.\n", 521 1.14 oster device_config.devs[i].devname, 522 1.14 oster device_status(device_config.devs[i].status)); 523 1.14 oster } 524 1.14 oster } 525 1.23 oster 526 1.70 mlelstv if (nspares > 0) { 527 1.70 mlelstv for(i=0; i < nspares; i++) { 528 1.85 rillig if ((device_config.spares[i].status == 529 1.21 oster rf_ds_optimal) || 530 1.85 rillig (device_config.spares[i].status == 531 1.21 oster rf_ds_used_spare)) { 532 1.85 rillig get_component_label(fd, 533 1.21 oster device_config.spares[i].devname); 534 1.21 oster } else { 535 1.21 oster printf("%s status is: %s. Skipping label.\n", 536 1.21 oster device_config.spares[i].devname, 537 1.75 kre device_status( 538 1.75 kre device_config.spares[i].status)); 539 1.85 rillig } 540 1.21 oster } 541 1.21 oster } 542 1.23 oster 543 1.8 oster do_ioctl(fd, RAIDFRAME_CHECK_PARITY, &is_clean, 544 1.8 oster "RAIDFRAME_CHECK_PARITY"); 545 1.8 oster if (is_clean) { 546 1.8 oster printf("Parity status: clean\n"); 547 1.8 oster } else { 548 1.8 oster printf("Parity status: DIRTY\n"); 549 1.1 oster } 550 1.14 oster check_status(fd,0); 551 1.1 oster } 552 1.1 oster 553 1.1 oster static void 554 1.42 jld rf_output_pmstat(int fd, int raidID) 555 1.42 jld { 556 1.42 jld char srs[7]; 557 1.47 plunky unsigned int i, j; 558 1.47 plunky int dis, dr; 559 1.42 jld struct rf_pmstat st; 560 1.42 jld 561 1.50 pooka if (prog_ioctl(fd, RAIDFRAME_PARITYMAP_STATUS, &st) == -1) { 562 1.48 jld if (errno == EINVAL) { 563 1.48 jld printf("raid%d: has no parity; parity map disabled\n", 564 1.48 jld raidID); 565 1.48 jld return; 566 1.48 jld } 567 1.48 jld err(1, "ioctl (%s) failed", "RAIDFRAME_PARITYMAP_STATUS"); 568 1.48 jld } 569 1.48 jld 570 1.42 jld if (st.enabled) { 571 1.85 rillig if (0 > humanize_number(srs, 7, st.region_size * DEV_BSIZE, 572 1.42 jld "B", HN_AUTOSCALE, HN_NOSPACE)) 573 1.42 jld strlcpy(srs, "???", 7); 574 1.42 jld 575 1.42 jld printf("raid%d: parity map enabled with %u regions of %s\n", 576 1.42 jld raidID, st.params.regions, srs); 577 1.43 jld printf("raid%d: regions marked clean after %d intervals of" 578 1.42 jld " %d.%03ds\n", raidID, st.params.cooldown, 579 1.42 jld st.params.tickms / 1000, st.params.tickms % 1000); 580 1.42 jld printf("raid%d: write/sync/clean counters " 581 1.42 jld "%"PRIu64"/%"PRIu64"/%"PRIu64"\n", raidID, 582 1.42 jld st.ctrs.nwrite, st.ctrs.ncachesync, st.ctrs.nclearing); 583 1.42 jld 584 1.42 jld dr = 0; 585 1.46 jld for (i = 0; i < st.params.regions; i++) 586 1.42 jld if (isset(st.dirty, i)) 587 1.42 jld dr++; 588 1.42 jld printf("raid%d: %d dirty region%s\n", raidID, dr, 589 1.42 jld dr == 1 ? "" : "s"); 590 1.42 jld 591 1.42 jld if (verbose > 0) { 592 1.42 jld for (i = 0; i < RF_PARITYMAP_NBYTE; i += 32) { 593 1.42 jld printf(" "); 594 1.42 jld for (j = i; j < RF_PARITYMAP_NBYTE 595 1.42 jld && j < i + 32; j++) 596 1.85 rillig printf("%x%x", st.dirty[j] & 15, 597 1.42 jld (st.dirty[j] >> 4) & 15); 598 1.42 jld printf("\n"); 599 1.42 jld } 600 1.42 jld } 601 1.42 jld } else { 602 1.42 jld printf("raid%d: parity map disabled\n", raidID); 603 1.42 jld } 604 1.42 jld 605 1.42 jld do_ioctl(fd, RAIDFRAME_PARITYMAP_GET_DISABLE, &dis, 606 1.42 jld "RAIDFRAME_PARITYMAP_GET_DISABLE"); 607 1.85 rillig printf("raid%d: parity map will %s %sabled on next configure\n", 608 1.42 jld raidID, dis == st.enabled ? "be" : "remain", dis ? "dis" : "en"); 609 1.42 jld } 610 1.42 jld 611 1.42 jld static void 612 1.42 jld rf_pm_configure(int fd, int raidID, char *parityconf, int parityparams[]) 613 1.42 jld { 614 1.42 jld int dis; 615 1.42 jld struct rf_pmparams params; 616 1.42 jld 617 1.42 jld if (strcasecmp(parityconf, "yes") == 0) 618 1.42 jld dis = 0; 619 1.42 jld else if (strcasecmp(parityconf, "no") == 0) 620 1.42 jld dis = 1; 621 1.42 jld else if (strcasecmp(parityconf, "set") == 0) { 622 1.42 jld params.cooldown = parityparams[0]; 623 1.42 jld params.tickms = parityparams[1]; 624 1.42 jld params.regions = parityparams[2]; 625 1.85 rillig 626 1.42 jld do_ioctl(fd, RAIDFRAME_PARITYMAP_SET_PARAMS, ¶ms, 627 1.42 jld "RAIDFRAME_PARITYMAP_SET_PARAMS"); 628 1.42 jld 629 1.42 jld if (params.cooldown != 0 || params.tickms != 0) { 630 1.42 jld printf("raid%d: parity cleaned after", raidID); 631 1.42 jld if (params.cooldown != 0) 632 1.42 jld printf(" %d", params.cooldown); 633 1.42 jld printf(" intervals"); 634 1.42 jld if (params.tickms != 0) { 635 1.42 jld printf(" of %d.%03ds", params.tickms / 1000, 636 1.42 jld params.tickms % 1000); 637 1.42 jld } 638 1.42 jld printf("\n"); 639 1.42 jld } 640 1.42 jld if (params.regions != 0) 641 1.42 jld printf("raid%d: will use %d regions on next" 642 1.42 jld " configuration\n", raidID, params.regions); 643 1.42 jld 644 1.42 jld return; 645 1.42 jld /* XXX the control flow here could be prettier. */ 646 1.45 christos } else 647 1.45 christos err(1, "`%s' is not a valid parity map command", parityconf); 648 1.42 jld 649 1.42 jld do_ioctl(fd, RAIDFRAME_PARITYMAP_SET_DISABLE, &dis, 650 1.42 jld "RAIDFRAME_PARITYMAP_SET_DISABLE"); 651 1.85 rillig printf("raid%d: parity map will be %sabled on next configure\n", 652 1.42 jld raidID, dis ? "dis" : "en"); 653 1.42 jld } 654 1.42 jld 655 1.62 mrg /* convert "component0" into "absent" */ 656 1.63 bad static const char *rf_output_devname(const char *name) 657 1.62 mrg { 658 1.62 mrg 659 1.63 bad if (strncmp(name, "component", 9) == 0) 660 1.62 mrg return "absent"; 661 1.63 bad return name; 662 1.62 mrg } 663 1.42 jld 664 1.42 jld static void 665 1.36 xtraeme rf_output_configuration(int fd, const char *name) 666 1.27 lukem { 667 1.27 lukem RF_DeviceConfig_t device_config; 668 1.27 lukem void *cfg_ptr; 669 1.71 mlelstv int i, nspares; 670 1.27 lukem RF_ComponentLabel_t component_label; 671 1.27 lukem void *label_ptr; 672 1.27 lukem int component_num; 673 1.27 lukem int num_cols; 674 1.27 lukem 675 1.27 lukem cfg_ptr = &device_config; 676 1.27 lukem 677 1.27 lukem printf("# raidctl config file for %s\n", name); 678 1.27 lukem printf("\n"); 679 1.27 lukem do_ioctl(fd, RAIDFRAME_GET_INFO, &cfg_ptr, "RAIDFRAME_GET_INFO"); 680 1.27 lukem 681 1.71 mlelstv nspares = MIN(device_config.nspares, 682 1.71 mlelstv __arraycount(device_config.spares)); 683 1.85 rillig 684 1.27 lukem printf("START array\n"); 685 1.77 kre printf("# numCol numSpare\n"); 686 1.77 kre printf("%d %d\n", device_config.cols, device_config.nspares); 687 1.27 lukem printf("\n"); 688 1.27 lukem 689 1.27 lukem printf("START disks\n"); 690 1.27 lukem for(i=0; i < device_config.ndevs; i++) 691 1.62 mrg printf("%s\n", 692 1.62 mrg rf_output_devname(device_config.devs[i].devname)); 693 1.27 lukem printf("\n"); 694 1.27 lukem 695 1.71 mlelstv if (nspares > 0) { 696 1.27 lukem printf("START spare\n"); 697 1.71 mlelstv for(i=0; i < nspares; i++) 698 1.27 lukem printf("%s\n", device_config.spares[i].devname); 699 1.27 lukem printf("\n"); 700 1.27 lukem } 701 1.27 lukem 702 1.27 lukem for(i=0; i < device_config.ndevs; i++) { 703 1.27 lukem if (device_config.devs[i].status == rf_ds_optimal) 704 1.27 lukem break; 705 1.27 lukem } 706 1.27 lukem if (i == device_config.ndevs) { 707 1.27 lukem printf("# WARNING: no optimal components; using %s\n", 708 1.27 lukem device_config.devs[0].devname); 709 1.27 lukem i = 0; 710 1.27 lukem } 711 1.27 lukem get_component_number(fd, device_config.devs[i].devname, 712 1.27 lukem &component_num, &num_cols); 713 1.27 lukem memset(&component_label, 0, sizeof(RF_ComponentLabel_t)); 714 1.27 lukem component_label.row = component_num / num_cols; 715 1.27 lukem component_label.column = component_num % num_cols; 716 1.27 lukem label_ptr = &component_label; 717 1.67 nakayama do_ioctl(fd, RAIDFRAME_GET_COMPONENT_LABEL, label_ptr, 718 1.27 lukem "RAIDFRAME_GET_COMPONENT_LABEL"); 719 1.27 lukem 720 1.27 lukem printf("START layout\n"); 721 1.27 lukem printf( 722 1.27 lukem "# sectPerSU SUsPerParityUnit SUsPerReconUnit RAID_level_%c\n", 723 1.27 lukem (char) component_label.parityConfig); 724 1.85 rillig printf("%d %d %d %c\n", 725 1.85 rillig component_label.sectPerSU, component_label.SUsPerPU, 726 1.27 lukem component_label.SUsPerRU, (char) component_label.parityConfig); 727 1.27 lukem printf("\n"); 728 1.27 lukem 729 1.27 lukem printf("START queue\n"); 730 1.27 lukem printf("fifo %d\n", device_config.maxqdepth); 731 1.27 lukem } 732 1.27 lukem 733 1.27 lukem static void 734 1.36 xtraeme get_component_number(int fd, char *component_name, int *component_number, 735 1.36 xtraeme int *num_columns) 736 1.1 oster { 737 1.1 oster RF_DeviceConfig_t device_config; 738 1.1 oster void *cfg_ptr; 739 1.71 mlelstv int i, nspares; 740 1.1 oster int found; 741 1.1 oster 742 1.6 oster *component_number = -1; 743 1.85 rillig 744 1.1 oster /* Assuming a full path spec... */ 745 1.1 oster cfg_ptr = &device_config; 746 1.85 rillig do_ioctl(fd, RAIDFRAME_GET_INFO, &cfg_ptr, 747 1.1 oster "RAIDFRAME_GET_INFO"); 748 1.6 oster 749 1.6 oster *num_columns = device_config.cols; 750 1.71 mlelstv 751 1.71 mlelstv nspares = MIN(device_config.nspares, 752 1.71 mlelstv __arraycount(device_config.spares)); 753 1.85 rillig 754 1.1 oster found = 0; 755 1.1 oster for(i=0; i < device_config.ndevs; i++) { 756 1.6 oster if (strncmp(component_name, device_config.devs[i].devname, 757 1.1 oster PATH_MAX)==0) { 758 1.1 oster found = 1; 759 1.6 oster *component_number = i; 760 1.1 oster } 761 1.1 oster } 762 1.21 oster if (!found) { /* maybe it's a spare? */ 763 1.71 mlelstv for(i=0; i < nspares; i++) { 764 1.85 rillig if (strncmp(component_name, 765 1.21 oster device_config.spares[i].devname, 766 1.21 oster PATH_MAX)==0) { 767 1.21 oster found = 1; 768 1.21 oster *component_number = i + device_config.ndevs; 769 1.23 oster /* the way spares are done should 770 1.23 oster really change... */ 771 1.85 rillig *num_columns = device_config.cols + 772 1.23 oster device_config.nspares; 773 1.21 oster } 774 1.21 oster } 775 1.21 oster } 776 1.21 oster 777 1.45 christos if (!found) 778 1.45 christos err(1,"%s is not a component of this device", component_name); 779 1.6 oster } 780 1.6 oster 781 1.6 oster static void 782 1.36 xtraeme rf_fail_disk(int fd, char *component_to_fail, int do_recon) 783 1.6 oster { 784 1.6 oster struct rf_recon_req recon_request; 785 1.6 oster int component_num; 786 1.6 oster int num_cols; 787 1.6 oster 788 1.6 oster get_component_number(fd, component_to_fail, &component_num, &num_cols); 789 1.1 oster 790 1.6 oster recon_request.col = component_num % num_cols; 791 1.1 oster if (do_recon) { 792 1.1 oster recon_request.flags = RF_FDFLAGS_RECON; 793 1.1 oster } else { 794 1.1 oster recon_request.flags = RF_FDFLAGS_NONE; 795 1.1 oster } 796 1.85 rillig do_ioctl(fd, RAIDFRAME_FAIL_DISK, &recon_request, 797 1.1 oster "RAIDFRAME_FAIL_DISK"); 798 1.10 oster if (do_recon && verbose) { 799 1.10 oster printf("Reconstruction status:\n"); 800 1.10 oster sleep(3); /* XXX give reconstruction a chance to start */ 801 1.19 oster do_meter(fd,RAIDFRAME_CHECK_RECON_STATUS_EXT); 802 1.10 oster } 803 1.1 oster } 804 1.1 oster 805 1.1 oster static void 806 1.36 xtraeme get_component_label(int fd, char *component) 807 1.5 oster { 808 1.5 oster RF_ComponentLabel_t component_label; 809 1.5 oster void *label_ptr; 810 1.5 oster int component_num; 811 1.6 oster int num_cols; 812 1.5 oster 813 1.6 oster get_component_number(fd, component, &component_num, &num_cols); 814 1.5 oster 815 1.5 oster memset( &component_label, 0, sizeof(RF_ComponentLabel_t)); 816 1.6 oster component_label.row = component_num / num_cols; 817 1.6 oster component_label.column = component_num % num_cols; 818 1.5 oster 819 1.5 oster label_ptr = &component_label; 820 1.67 nakayama do_ioctl( fd, RAIDFRAME_GET_COMPONENT_LABEL, label_ptr, 821 1.5 oster "RAIDFRAME_GET_COMPONENT_LABEL"); 822 1.5 oster 823 1.5 oster printf("Component label for %s:\n",component); 824 1.12 oster 825 1.25 lukem printf(" Row: %d, Column: %d, Num Rows: %d, Num Columns: %d\n", 826 1.85 rillig component_label.row, component_label.column, 827 1.12 oster component_label.num_rows, component_label.num_columns); 828 1.54 mrg printf(" Version: %d, Serial Number: %u, Mod Counter: %d\n", 829 1.12 oster component_label.version, component_label.serial_number, 830 1.12 oster component_label.mod_counter); 831 1.25 lukem printf(" Clean: %s, Status: %d\n", 832 1.85 rillig component_label.clean ? "Yes" : "No", 833 1.12 oster component_label.status ); 834 1.25 lukem printf(" sectPerSU: %d, SUsPerPU: %d, SUsPerRU: %d\n", 835 1.85 rillig component_label.sectPerSU, component_label.SUsPerPU, 836 1.12 oster component_label.SUsPerRU); 837 1.52 enami printf(" Queue size: %d, blocksize: %d, numBlocks: %"PRIu64"\n", 838 1.25 lukem component_label.maxOutstanding, component_label.blockSize, 839 1.52 enami rf_component_label_numblocks(&component_label)); 840 1.25 lukem printf(" RAID Level: %c\n", (char) component_label.parityConfig); 841 1.85 rillig printf(" Autoconfig: %s\n", 842 1.12 oster component_label.autoconfigure ? "Yes" : "No" ); 843 1.15 oster printf(" Root partition: %s\n", 844 1.57 christos rootpart[component_label.root_partition & 3]); 845 1.12 oster printf(" Last configured as: raid%d\n", component_label.last_unit ); 846 1.5 oster } 847 1.5 oster 848 1.5 oster static void 849 1.36 xtraeme set_component_label(int fd, char *component) 850 1.5 oster { 851 1.5 oster RF_ComponentLabel_t component_label; 852 1.5 oster int component_num; 853 1.6 oster int num_cols; 854 1.5 oster 855 1.6 oster get_component_number(fd, component, &component_num, &num_cols); 856 1.6 oster 857 1.6 oster /* XXX This is currently here for testing, and future expandability */ 858 1.5 oster 859 1.5 oster component_label.version = 1; 860 1.5 oster component_label.serial_number = 123456; 861 1.5 oster component_label.mod_counter = 0; 862 1.6 oster component_label.row = component_num / num_cols; 863 1.6 oster component_label.column = component_num % num_cols; 864 1.5 oster component_label.num_rows = 0; 865 1.5 oster component_label.num_columns = 5; 866 1.5 oster component_label.clean = 0; 867 1.5 oster component_label.status = 1; 868 1.85 rillig 869 1.5 oster do_ioctl( fd, RAIDFRAME_SET_COMPONENT_LABEL, &component_label, 870 1.5 oster "RAIDFRAME_SET_COMPONENT_LABEL"); 871 1.5 oster } 872 1.5 oster 873 1.5 oster 874 1.5 oster static void 875 1.36 xtraeme init_component_labels(int fd, int serial_number) 876 1.5 oster { 877 1.5 oster RF_ComponentLabel_t component_label; 878 1.5 oster 879 1.5 oster component_label.version = 0; 880 1.5 oster component_label.serial_number = serial_number; 881 1.5 oster component_label.mod_counter = 0; 882 1.5 oster component_label.row = 0; 883 1.5 oster component_label.column = 0; 884 1.5 oster component_label.num_rows = 0; 885 1.5 oster component_label.num_columns = 0; 886 1.5 oster component_label.clean = 0; 887 1.5 oster component_label.status = 0; 888 1.85 rillig 889 1.5 oster do_ioctl( fd, RAIDFRAME_INIT_LABELS, &component_label, 890 1.69 oster "RAIDFRAME_INIT_LABELS"); 891 1.12 oster } 892 1.12 oster 893 1.12 oster static void 894 1.36 xtraeme set_autoconfig(int fd, int raidID, char *autoconf) 895 1.12 oster { 896 1.12 oster int auto_config; 897 1.12 oster int root_config; 898 1.12 oster 899 1.12 oster auto_config = 0; 900 1.12 oster root_config = 0; 901 1.12 oster 902 1.57 christos if (strncasecmp(autoconf, "root", 4) == 0 || 903 1.57 christos strncasecmp(autoconf, "hard", 4) == 0 || 904 1.61 sborrill strncasecmp(autoconf, "force", 5) == 0) { 905 1.12 oster root_config = 1; 906 1.57 christos } else if (strncasecmp(autoconf, "soft", 4) == 0) { 907 1.57 christos root_config = 2; 908 1.12 oster } 909 1.12 oster 910 1.12 oster if ((strncasecmp(autoconf,"yes", 3) == 0) || 911 1.57 christos root_config > 0) { 912 1.12 oster auto_config = 1; 913 1.12 oster } 914 1.12 oster 915 1.12 oster do_ioctl(fd, RAIDFRAME_SET_AUTOCONFIG, &auto_config, 916 1.12 oster "RAIDFRAME_SET_AUTOCONFIG"); 917 1.12 oster 918 1.12 oster do_ioctl(fd, RAIDFRAME_SET_ROOT, &root_config, 919 1.12 oster "RAIDFRAME_SET_ROOT"); 920 1.12 oster 921 1.80 oster if (verbose) { 922 1.80 oster printf("raid%d: Autoconfigure: %s\n", raidID, 923 1.80 oster auto_config ? "Yes" : "No"); 924 1.80 oster if (auto_config == 1) { 925 1.80 oster printf("raid%d: Root: %s\n", raidID, rootpart[root_config]); 926 1.80 oster } 927 1.12 oster } 928 1.5 oster } 929 1.5 oster 930 1.5 oster static void 931 1.36 xtraeme add_hot_spare(int fd, char *component) 932 1.5 oster { 933 1.6 oster RF_SingleComponent_t hot_spare; 934 1.5 oster 935 1.6 oster hot_spare.row = 0; 936 1.6 oster hot_spare.column = 0; 937 1.85 rillig strncpy(hot_spare.component_name, component, 938 1.6 oster sizeof(hot_spare.component_name)); 939 1.85 rillig 940 1.5 oster do_ioctl( fd, RAIDFRAME_ADD_HOT_SPARE, &hot_spare, 941 1.5 oster "RAIDFRAME_ADD_HOT_SPARE"); 942 1.6 oster } 943 1.6 oster 944 1.6 oster static void 945 1.79 oster remove_component(int fd, char *component) 946 1.6 oster { 947 1.79 oster RF_SingleComponent_t comp; 948 1.6 oster int component_num; 949 1.6 oster int num_cols; 950 1.5 oster 951 1.6 oster get_component_number(fd, component, &component_num, &num_cols); 952 1.6 oster 953 1.79 oster comp.row = component_num / num_cols; 954 1.79 oster comp.column = component_num % num_cols; 955 1.6 oster 956 1.85 rillig strncpy(comp.component_name, component, 957 1.79 oster sizeof(comp.component_name)); 958 1.85 rillig 959 1.79 oster do_ioctl( fd, RAIDFRAME_REMOVE_COMPONENT, &comp, 960 1.79 oster "RAIDFRAME_REMOVE_COMPONENT"); 961 1.6 oster } 962 1.6 oster 963 1.6 oster static void 964 1.36 xtraeme rebuild_in_place(int fd, char *component) 965 1.6 oster { 966 1.6 oster RF_SingleComponent_t comp; 967 1.6 oster int component_num; 968 1.6 oster int num_cols; 969 1.6 oster 970 1.6 oster get_component_number(fd, component, &component_num, &num_cols); 971 1.6 oster 972 1.6 oster comp.row = 0; 973 1.6 oster comp.column = component_num; 974 1.6 oster strncpy(comp.component_name, component, sizeof(comp.component_name)); 975 1.85 rillig 976 1.6 oster do_ioctl( fd, RAIDFRAME_REBUILD_IN_PLACE, &comp, 977 1.6 oster "RAIDFRAME_REBUILD_IN_PLACE"); 978 1.10 oster 979 1.10 oster if (verbose) { 980 1.10 oster printf("Reconstruction status:\n"); 981 1.10 oster sleep(3); /* XXX give reconstruction a chance to start */ 982 1.19 oster do_meter(fd,RAIDFRAME_CHECK_RECON_STATUS_EXT); 983 1.10 oster } 984 1.10 oster 985 1.10 oster } 986 1.10 oster 987 1.10 oster static void 988 1.36 xtraeme check_parity(int fd, int do_rewrite, char *dev_name) 989 1.10 oster { 990 1.10 oster int is_clean; 991 1.10 oster int percent_done; 992 1.10 oster 993 1.10 oster is_clean = 0; 994 1.10 oster percent_done = 0; 995 1.10 oster do_ioctl(fd, RAIDFRAME_CHECK_PARITY, &is_clean, 996 1.10 oster "RAIDFRAME_CHECK_PARITY"); 997 1.10 oster if (is_clean) { 998 1.10 oster printf("%s: Parity status: clean\n",dev_name); 999 1.10 oster } else { 1000 1.10 oster printf("%s: Parity status: DIRTY\n",dev_name); 1001 1.10 oster if (do_rewrite) { 1002 1.10 oster printf("%s: Initiating re-write of parity\n", 1003 1.10 oster dev_name); 1004 1.85 rillig do_ioctl(fd, RAIDFRAME_REWRITEPARITY, NULL, 1005 1.10 oster "RAIDFRAME_REWRITEPARITY"); 1006 1.10 oster sleep(3); /* XXX give it time to 1007 1.10 oster get started. */ 1008 1.10 oster if (verbose) { 1009 1.10 oster printf("Parity Re-write status:\n"); 1010 1.75 kre do_meter(fd, 1011 1.75 kre RAIDFRAME_CHECK_PARITYREWRITE_STATUS_EXT); 1012 1.10 oster } else { 1013 1.85 rillig do_ioctl(fd, 1014 1.85 rillig RAIDFRAME_CHECK_PARITYREWRITE_STATUS, 1015 1.85 rillig &percent_done, 1016 1.10 oster "RAIDFRAME_CHECK_PARITYREWRITE_STATUS" 1017 1.10 oster ); 1018 1.10 oster while( percent_done < 100 ) { 1019 1.24 oster sleep(3); /* wait a bit... */ 1020 1.75 kre do_ioctl(fd, 1021 1.75 kre RAIDFRAME_CHECK_PARITYREWRITE_STATUS, 1022 1.75 kre &percent_done, 1023 1.75 kre "RAIDFRAME_CHECK_PARITYREWRITE_STATUS"); 1024 1.10 oster } 1025 1.10 oster 1026 1.10 oster } 1027 1.75 kre printf("%s: Parity Re-write complete\n", dev_name); 1028 1.10 oster } else { 1029 1.10 oster /* parity is wrong, and is not being fixed. 1030 1.10 oster Exit w/ an error. */ 1031 1.10 oster exit(1); 1032 1.10 oster } 1033 1.10 oster } 1034 1.10 oster } 1035 1.10 oster 1036 1.10 oster 1037 1.10 oster static void 1038 1.36 xtraeme check_status(int fd, int meter) 1039 1.10 oster { 1040 1.10 oster int recon_percent_done = 0; 1041 1.10 oster int parity_percent_done = 0; 1042 1.10 oster 1043 1.85 rillig do_ioctl(fd, RAIDFRAME_CHECK_RECON_STATUS, &recon_percent_done, 1044 1.10 oster "RAIDFRAME_CHECK_RECON_STATUS"); 1045 1.10 oster printf("Reconstruction is %d%% complete.\n", recon_percent_done); 1046 1.85 rillig do_ioctl(fd, RAIDFRAME_CHECK_PARITYREWRITE_STATUS, 1047 1.85 rillig &parity_percent_done, 1048 1.10 oster "RAIDFRAME_CHECK_PARITYREWRITE_STATUS"); 1049 1.10 oster printf("Parity Re-write is %d%% complete.\n", parity_percent_done); 1050 1.10 oster 1051 1.14 oster if (meter) { 1052 1.14 oster /* These 3 should be mutually exclusive at this point */ 1053 1.14 oster if (recon_percent_done < 100) { 1054 1.14 oster printf("Reconstruction status:\n"); 1055 1.19 oster do_meter(fd,RAIDFRAME_CHECK_RECON_STATUS_EXT); 1056 1.14 oster } else if (parity_percent_done < 100) { 1057 1.14 oster printf("Parity Re-write status:\n"); 1058 1.19 oster do_meter(fd,RAIDFRAME_CHECK_PARITYREWRITE_STATUS_EXT); 1059 1.14 oster } 1060 1.10 oster } 1061 1.10 oster } 1062 1.10 oster 1063 1.18 thorpej const char *tbits = "|/-\\"; 1064 1.10 oster 1065 1.10 oster static void 1066 1.36 xtraeme do_meter(int fd, u_long option) 1067 1.10 oster { 1068 1.10 oster int percent_done; 1069 1.37 oster RF_uint64 start_value; 1070 1.19 oster RF_ProgressInfo_t progressInfo; 1071 1.19 oster void *pInfoPtr; 1072 1.10 oster struct timeval start_time; 1073 1.10 oster struct timeval current_time; 1074 1.10 oster double elapsed; 1075 1.10 oster int elapsed_sec; 1076 1.10 oster int elapsed_usec; 1077 1.20 oster int simple_eta,last_eta; 1078 1.10 oster double rate; 1079 1.37 oster RF_uint64 amount; 1080 1.10 oster int tbit_value; 1081 1.10 oster char bar_buffer[1024]; 1082 1.10 oster char eta_buffer[1024]; 1083 1.10 oster 1084 1.45 christos if (gettimeofday(&start_time,NULL) == -1) 1085 1.45 christos err(1, "gettimeofday failed!?!?"); 1086 1.19 oster memset(&progressInfo, 0, sizeof(RF_ProgressInfo_t)); 1087 1.19 oster pInfoPtr=&progressInfo; 1088 1.19 oster 1089 1.10 oster percent_done = 0; 1090 1.67 nakayama do_ioctl(fd, option, pInfoPtr, ""); 1091 1.37 oster start_value = progressInfo.completed; 1092 1.10 oster current_time = start_time; 1093 1.38 lukem simple_eta = 0; 1094 1.38 lukem last_eta = 0; 1095 1.85 rillig 1096 1.10 oster tbit_value = 0; 1097 1.19 oster while(progressInfo.completed < progressInfo.total) { 1098 1.10 oster 1099 1.85 rillig percent_done = (progressInfo.completed * 100) / 1100 1.19 oster progressInfo.total; 1101 1.10 oster 1102 1.10 oster get_bar(bar_buffer, percent_done, 40); 1103 1.85 rillig 1104 1.20 oster elapsed_sec = current_time.tv_sec - start_time.tv_sec; 1105 1.20 oster elapsed_usec = current_time.tv_usec - start_time.tv_usec; 1106 1.10 oster if (elapsed_usec < 0) { 1107 1.10 oster elapsed_usec-=1000000; 1108 1.10 oster elapsed_sec++; 1109 1.10 oster } 1110 1.20 oster 1111 1.85 rillig elapsed = (double) elapsed_sec + 1112 1.10 oster (double) elapsed_usec / 1000000.0; 1113 1.19 oster 1114 1.20 oster amount = progressInfo.completed - start_value; 1115 1.19 oster 1116 1.10 oster if (amount <= 0) { /* we don't do negatives (yet?) */ 1117 1.10 oster amount = 0; 1118 1.10 oster } 1119 1.20 oster 1120 1.22 oster if (elapsed == 0) 1121 1.22 oster rate = 0.0; 1122 1.22 oster else 1123 1.22 oster rate = amount / elapsed; 1124 1.10 oster 1125 1.10 oster if (rate > 0.0) { 1126 1.85 rillig simple_eta = (int) (((double)progressInfo.total - 1127 1.85 rillig (double) progressInfo.completed) 1128 1.19 oster / rate); 1129 1.10 oster } else { 1130 1.10 oster simple_eta = -1; 1131 1.10 oster } 1132 1.19 oster 1133 1.85 rillig if (simple_eta <=0) { 1134 1.10 oster simple_eta = last_eta; 1135 1.10 oster } else { 1136 1.10 oster last_eta = simple_eta; 1137 1.10 oster } 1138 1.10 oster 1139 1.68 mrg get_time_string(eta_buffer, sizeof eta_buffer, simple_eta); 1140 1.10 oster 1141 1.55 christos fprintf(stdout,"\r%3d%% |%s| ETA: %s %c", 1142 1.55 christos percent_done,bar_buffer,eta_buffer,tbits[tbit_value]); 1143 1.10 oster fflush(stdout); 1144 1.10 oster 1145 1.85 rillig if (++tbit_value>3) 1146 1.10 oster tbit_value = 0; 1147 1.10 oster 1148 1.10 oster sleep(2); 1149 1.10 oster 1150 1.45 christos if (gettimeofday(¤t_time,NULL) == -1) 1151 1.45 christos err(1, "gettimeofday failed!?!?"); 1152 1.10 oster 1153 1.67 nakayama do_ioctl( fd, option, pInfoPtr, ""); 1154 1.85 rillig 1155 1.10 oster 1156 1.10 oster } 1157 1.10 oster printf("\n"); 1158 1.10 oster } 1159 1.10 oster /* 40 '*''s per line, then 40 ' ''s line. */ 1160 1.10 oster /* If you've got a screen wider than 160 characters, "tough" */ 1161 1.10 oster 1162 1.10 oster #define STAR_MIDPOINT 4*40 1163 1.10 oster const char stars[] = "****************************************" 1164 1.10 oster "****************************************" 1165 1.10 oster "****************************************" 1166 1.10 oster "****************************************" 1167 1.10 oster " " 1168 1.10 oster " " 1169 1.10 oster " " 1170 1.10 oster " " 1171 1.10 oster " "; 1172 1.10 oster 1173 1.10 oster static void 1174 1.36 xtraeme get_bar(char *string, double percent, int max_strlen) 1175 1.10 oster { 1176 1.10 oster int offset; 1177 1.10 oster 1178 1.10 oster if (max_strlen > STAR_MIDPOINT) { 1179 1.10 oster max_strlen = STAR_MIDPOINT; 1180 1.10 oster } 1181 1.85 rillig offset = STAR_MIDPOINT - 1182 1.10 oster (int)((percent * max_strlen)/ 100); 1183 1.10 oster if (offset < 0) 1184 1.10 oster offset = 0; 1185 1.60 pooka snprintf(string,max_strlen,"%s",stars+offset); 1186 1.10 oster } 1187 1.10 oster 1188 1.10 oster static void 1189 1.68 mrg get_time_string(char *string, size_t len, int simple_time) 1190 1.10 oster { 1191 1.10 oster int minutes, seconds, hours; 1192 1.68 mrg char hours_buffer[8]; 1193 1.10 oster char minutes_buffer[5]; 1194 1.10 oster char seconds_buffer[5]; 1195 1.10 oster 1196 1.10 oster if (simple_time >= 0) { 1197 1.10 oster 1198 1.68 mrg minutes = simple_time / 60; 1199 1.68 mrg seconds = simple_time - 60*minutes; 1200 1.10 oster hours = minutes / 60; 1201 1.10 oster minutes = minutes - 60*hours; 1202 1.68 mrg #if defined(__GNUC__) 1203 1.68 mrg /* 1204 1.68 mrg * snprintf() truncation checker fails to detect that seconds 1205 1.68 mrg * and minutes will be 0-59 range. 1206 1.68 mrg */ 1207 1.68 mrg if (minutes < 0 || minutes > 60) 1208 1.68 mrg minutes = 60; 1209 1.68 mrg if (seconds < 0 || seconds > 60) 1210 1.68 mrg seconds = 60; 1211 1.68 mrg #endif 1212 1.85 rillig 1213 1.10 oster if (hours > 0) { 1214 1.75 kre snprintf(hours_buffer,sizeof hours_buffer, 1215 1.75 kre "%02d:",hours); 1216 1.10 oster } else { 1217 1.68 mrg snprintf(hours_buffer,sizeof hours_buffer," "); 1218 1.10 oster } 1219 1.85 rillig 1220 1.68 mrg snprintf(minutes_buffer,sizeof minutes_buffer,"%02d:",minutes); 1221 1.68 mrg snprintf(seconds_buffer,sizeof seconds_buffer,"%02d",seconds); 1222 1.68 mrg snprintf(string,len,"%s%s%s", 1223 1.10 oster hours_buffer, minutes_buffer, seconds_buffer); 1224 1.10 oster } else { 1225 1.68 mrg snprintf(string,len," --:--"); 1226 1.10 oster } 1227 1.85 rillig 1228 1.5 oster } 1229 1.5 oster 1230 1.80 oster /* Simplified RAID creation with a single command line... */ 1231 1.80 oster static void 1232 1.80 oster rf_simple_create(int fd, int argc, char *argv[]) 1233 1.80 oster { 1234 1.80 oster int i; 1235 1.80 oster int level; 1236 1.80 oster int num_components; 1237 1.80 oster char *components[RF_MAXCOL]; 1238 1.80 oster void *generic; 1239 1.80 oster RF_Config_t cfg; 1240 1.80 oster 1241 1.85 rillig /* 1242 1.80 oster * Note the extra level of redirection needed here, since 1243 1.85 rillig * what we really want to pass in is a pointer to the pointer to 1244 1.85 rillig * the configuration structure. 1245 1.80 oster */ 1246 1.80 oster 1247 1.85 rillig 1248 1.80 oster if (strcmp(argv[0],"mirror")==0) { 1249 1.80 oster level = 1; 1250 1.80 oster } else 1251 1.80 oster level = atoi(argv[0]); 1252 1.80 oster 1253 1.80 oster if (level != 0 && level != 1 && level !=5) 1254 1.80 oster usage(); 1255 1.80 oster 1256 1.80 oster /* remaining args must be components */ 1257 1.80 oster num_components = 0; 1258 1.80 oster for (i=1 ; i<argc ; i++) { 1259 1.80 oster components[i-1] = argv[i]; 1260 1.80 oster num_components++; 1261 1.80 oster } 1262 1.80 oster 1263 1.80 oster /* Level 0 must have at least two components. 1264 1.80 oster Level 1 must have exactly two components. 1265 1.80 oster Level 5 must have at least three components. */ 1266 1.80 oster if ((level == 0 && num_components < 2) || 1267 1.80 oster (level == 1 && num_components != 2) || 1268 1.80 oster (level == 5 && num_components < 3)) 1269 1.80 oster usage(); 1270 1.80 oster 1271 1.80 oster /* build a config... */ 1272 1.80 oster 1273 1.80 oster memset(&cfg, 0, sizeof(cfg)); 1274 1.80 oster 1275 1.80 oster cfg.numCol = num_components; 1276 1.80 oster cfg.numSpare = 0; 1277 1.80 oster 1278 1.80 oster for (i=0 ; i<num_components; i++) { 1279 1.80 oster strlcpy(cfg.devnames[0][i], components[i], 1280 1.80 oster sizeof(cfg.devnames[0][i])); 1281 1.80 oster } 1282 1.80 oster 1283 1.80 oster /* pick some reasonable values for sectPerSU, etc. */ 1284 1.80 oster if (level == 0) { 1285 1.80 oster if (num_components == 2) { 1286 1.80 oster /* 64 blocks (32K) per component - 64K data per stripe */ 1287 1.80 oster cfg.sectPerSU = 64; 1288 1.80 oster } else if (num_components == 3 || num_components == 4) { 1289 1.80 oster /* 32 blocks (16K) per component - 64K data per strip for 1290 1.80 oster the 4-component case. */ 1291 1.80 oster cfg.sectPerSU = 32; 1292 1.80 oster } else { 1293 1.80 oster /* 16 blocks (8K) per component */ 1294 1.80 oster cfg.sectPerSU = 16; 1295 1.80 oster } 1296 1.80 oster } else if (level == 1) { 1297 1.83 andvar /* 128 blocks (64K per component) - 64K per stripe */ 1298 1.80 oster cfg.sectPerSU = 128; 1299 1.80 oster } else if (level == 5) { 1300 1.80 oster if (num_components == 3) { 1301 1.80 oster /* 64 blocks (32K) per disk - 64K data per stripe */ 1302 1.80 oster cfg.sectPerSU = 64; 1303 1.80 oster } else if (num_components >= 4 && num_components < 9) { 1304 1.85 rillig /* 4 components makes 3 data components. No power of 2 is 1305 1.80 oster evenly divisible by 3 so performance will be lousy 1306 1.80 oster regardless of what number we choose here. 5 components is 1307 1.80 oster what we are really hoping for here, as 5 components with 4 1308 1.80 oster data components on RAID 5 means 32 blocks (16K) per data 1309 1.80 oster component, or 64K per stripe */ 1310 1.80 oster cfg.sectPerSU = 32; 1311 1.80 oster } else { 1312 1.80 oster /* 9 components here is optimal for 16 blocks (8K) per data 1313 1.80 oster component */ 1314 1.80 oster cfg.sectPerSU = 16; 1315 1.80 oster } 1316 1.80 oster } else 1317 1.85 rillig usage(); 1318 1.85 rillig 1319 1.80 oster cfg.SUsPerPU = 1; 1320 1.80 oster cfg.SUsPerRU = 1; 1321 1.80 oster cfg.parityConfig = '0' + level; 1322 1.80 oster strlcpy(cfg.diskQueueType, "fifo", sizeof(cfg.diskQueueType)); 1323 1.80 oster cfg.maxOutstandingDiskReqs = 1; 1324 1.80 oster cfg.force = 1; 1325 1.80 oster 1326 1.80 oster /* configure... */ 1327 1.80 oster 1328 1.80 oster generic = &cfg; 1329 1.80 oster do_ioctl(fd, RAIDFRAME_CONFIGURE, &generic, "RAIDFRAME_CONFIGURE"); 1330 1.80 oster 1331 1.85 rillig if (level == 1 || level == 5) 1332 1.85 rillig do_ioctl(fd, RAIDFRAME_REWRITEPARITY, NULL, 1333 1.80 oster "RAIDFRAME_REWRITEPARITY"); 1334 1.80 oster } 1335 1.85 rillig 1336 1.80 oster 1337 1.5 oster static void 1338 1.36 xtraeme usage(void) 1339 1.1 oster { 1340 1.26 cgd const char *progname = getprogname(); 1341 1.26 cgd 1342 1.75 kre fprintf(stderr, 1343 1.80 oster "usage: %s dev create [0 | 1 | mirror | 5] component component ...\n", 1344 1.80 oster progname); 1345 1.80 oster fprintf(stderr, " %s [-v] -A [yes | no | softroot | hardroot] dev\n", 1346 1.80 oster progname); 1347 1.65 wiz fprintf(stderr, " %s [-v] -a component dev\n", progname); 1348 1.26 cgd fprintf(stderr, " %s [-v] -B dev\n", progname); 1349 1.65 wiz fprintf(stderr, " %s [-v] -C config_file dev\n", progname); 1350 1.26 cgd fprintf(stderr, " %s [-v] -c config_file dev\n", progname); 1351 1.65 wiz fprintf(stderr, " %s [-v] -F component dev\n", progname); 1352 1.26 cgd fprintf(stderr, " %s [-v] -f component dev\n", progname); 1353 1.65 wiz fprintf(stderr, " %s [-v] -G dev\n", progname); 1354 1.26 cgd fprintf(stderr, " %s [-v] -g component dev\n", progname); 1355 1.65 wiz fprintf(stderr, " %s [-v] -I serial_number dev\n", progname); 1356 1.26 cgd fprintf(stderr, " %s [-v] -i dev\n", progname); 1357 1.42 jld fprintf(stderr, " %s [-v] -M [yes | no | set params] dev\n", 1358 1.42 jld progname); 1359 1.65 wiz fprintf(stderr, " %s [-v] -m dev\n", progname); 1360 1.65 wiz fprintf(stderr, " %s [-v] -P dev\n", progname); 1361 1.31 oster fprintf(stderr, " %s [-v] -p dev\n", progname); 1362 1.65 wiz fprintf(stderr, " %s [-v] -R component dev\n", progname); 1363 1.85 rillig fprintf(stderr, " %s [-v] -r component dev\n", progname); 1364 1.65 wiz fprintf(stderr, " %s [-v] -S dev\n", progname); 1365 1.26 cgd fprintf(stderr, " %s [-v] -s dev\n", progname); 1366 1.78 kre fprintf(stderr, " %s [-v] -t config_file\n", progname); 1367 1.65 wiz fprintf(stderr, " %s [-v] -U unit dev\n", progname); 1368 1.26 cgd fprintf(stderr, " %s [-v] -u dev\n", progname); 1369 1.1 oster exit(1); 1370 1.1 oster /* NOTREACHED */ 1371 1.1 oster } 1372 1.58 manu 1373 1.58 manu static unsigned int 1374 1.59 christos xstrtouint(const char *str) 1375 1.58 manu { 1376 1.59 christos int e; 1377 1.59 christos unsigned int num = (unsigned int)strtou(str, NULL, 10, 0, INT_MAX, &e); 1378 1.59 christos if (e) 1379 1.59 christos errc(EXIT_FAILURE, e, "Bad number `%s'", str); 1380 1.59 christos return num; 1381 1.58 manu } 1382