1 /* 2 * CDDL HEADER START 3 * 4 * The contents of this file are subject to the terms of the 5 * Common Development and Distribution License (the "License"). 6 * You may not use this file except in compliance with the License. 7 * 8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 9 * or http://www.opensolaris.org/os/licensing. 10 * See the License for the specific language governing permissions 11 * and limitations under the License. 12 * 13 * When distributing Covered Code, include this CDDL HEADER in each 14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 15 * If applicable, add the following below this CDDL HEADER, with the 16 * fields enclosed by brackets "[]" replaced with your own identifying 17 * information: Portions Copyright [yyyy] [name of copyright owner] 18 * 19 * CDDL HEADER END 20 */ 21 /* 22 * Copyright 2006 Sun Microsystems, Inc. All rights reserved. 23 * Use is subject to license terms. 24 */ 25 26 #pragma ident "%Z%%M% %I% %E% SMI" 27 28 /* 29 * Just in case we're not in a build environment, make sure that 30 * TEXT_DOMAIN gets set to something. 31 */ 32 #if !defined(TEXT_DOMAIN) 33 #define TEXT_DOMAIN "SYS_TEST" 34 #endif 35 36 /* 37 * stripe operations 38 */ 39 40 #include <limits.h> 41 #include <stdlib.h> 42 #include <meta.h> 43 #include <sys/lvm/md_stripe.h> 44 #include <sys/lvm/md_convert.h> 45 46 #define QUOTE(x) #x 47 #define VAL2STR(x) QUOTE(x) 48 49 /* 50 * replace stripe/concat 51 */ 52 int 53 meta_stripe_replace( 54 mdsetname_t *sp, 55 mdname_t *stripenp, 56 mdname_t *oldnp, 57 mdname_t *newnp, 58 mdcmdopts_t options, 59 md_error_t *ep 60 ) 61 { 62 replace_params_t params; 63 md_dev64_t old_dev, 64 new_dev; 65 diskaddr_t new_start_blk, 66 new_end_blk, 67 label, 68 size, 69 start_blk; 70 71 /* should have same set */ 72 assert(sp != NULL); 73 assert(sp->setno == MD_MIN2SET(meta_getminor(stripenp->dev))); 74 75 new_dev = newnp->dev; 76 new_start_blk = newnp->start_blk; 77 new_end_blk = newnp->end_blk; 78 79 meta_invalidate_name(stripenp); 80 81 /* the old device binding is now established */ 82 if ((old_dev = oldnp->dev) == NODEV64) 83 return (mdsyserror(ep, ENODEV, oldnp->cname)); 84 85 if (((strcmp(oldnp->rname, newnp->rname) == 0) && 86 (old_dev != new_dev))) { 87 newnp->dev = new_dev; 88 newnp->start_blk = new_start_blk; 89 newnp->end_blk = new_end_blk; 90 } 91 92 if ((size = metagetsize(newnp, ep)) == MD_DISKADDR_ERROR) 93 return (-1); 94 if ((label = metagetlabel(newnp, ep)) == MD_DISKADDR_ERROR) 95 return (-1); 96 if ((start_blk = metagetstart(sp, newnp, ep)) == MD_DISKADDR_ERROR) 97 return (-1); 98 if (start_blk >= size) { 99 (void) mdsyserror(ep, ENOSPC, newnp->cname); 100 return (-1); 101 } 102 103 /* In dryrun mode (DOIT not set) we must not alter the mddb */ 104 if (options & MDCMD_DOIT) { 105 if (add_key_name(sp, newnp, NULL, ep) != 0) 106 return (-1); 107 } 108 109 /* 110 * There is no need to call meta_fixdevid() here as this function is 111 * only called by the metareplace -c command which actually does 112 * nothing (in terms of a resync) and thus does nothing with the devid. 113 */ 114 115 (void) memset(¶ms, 0, sizeof (params)); 116 params.mnum = meta_getminor(stripenp->dev); 117 MD_SETDRIVERNAME(¶ms, MD_STRIPE, sp->setno); 118 119 params.cmd = REPLACE_COMP; 120 params.old_dev = old_dev; 121 params.new_dev = new_dev; 122 params.new_key = newnp->key; 123 params.start_blk = newnp->start_blk; 124 params.number_blks = size; 125 /* Is this just a dryrun ? */ 126 if ((options & MDCMD_DOIT) == 0) { 127 params.options |= MDIOCTL_DRYRUN; 128 } 129 if (label == 0) 130 params.has_label = 0; 131 else 132 params.has_label = 1; 133 if (metaioctl(MD_IOCREPLACE, ¶ms, ¶ms.mde, NULL) != 0) { 134 if (options & MDCMD_DOIT) 135 (void) del_key_name(sp, newnp, ep); 136 return (mdstealerror(ep, ¶ms.mde)); 137 } 138 meta_invalidate_name(oldnp); 139 meta_invalidate_name(newnp); 140 meta_invalidate_name(stripenp); 141 142 if (options & MDCMD_PRINT) { 143 (void) printf(dgettext(TEXT_DOMAIN, 144 "%s: device %s is replaced with %s\n"), 145 stripenp->cname, oldnp->cname, newnp->cname); 146 147 } 148 return (0); 149 } 150 151 152 /* 153 * FUNCTION: meta_get_stripe_names() 154 * INPUT: sp - the set name to get stripes from 155 * options - options from the command line 156 * OUTPUT: nlpp - list of all stripe names 157 * ep - return error pointer 158 * RETURNS: int - -1 if error, 0 success 159 * PURPOSE: returns a list of all stripes in the metadb 160 * for all devices in the specified set 161 */ 162 int 163 meta_get_stripe_names( 164 mdsetname_t *sp, 165 mdnamelist_t **nlpp, 166 int options, 167 md_error_t *ep 168 ) 169 { 170 return (meta_get_names(MD_STRIPE, sp, nlpp, options, ep)); 171 } 172 173 /* 174 * free stripe 175 */ 176 void 177 meta_free_stripe( 178 md_stripe_t *stripep 179 ) 180 { 181 uint_t row; 182 183 for (row = 0; (row < stripep->rows.rows_len); ++row) { 184 md_row_t *rp = &stripep->rows.rows_val[row]; 185 186 if (rp->comps.comps_val != NULL) { 187 assert(rp->comps.comps_len > 0); 188 Free(rp->comps.comps_val); 189 } 190 } 191 if (stripep->rows.rows_val != NULL) { 192 assert(stripep->rows.rows_len > 0); 193 Free(stripep->rows.rows_val); 194 } 195 Free(stripep); 196 } 197 198 199 /* 200 * get stripe (common) 201 */ 202 md_stripe_t * 203 meta_get_stripe_common( 204 mdsetname_t *sp, 205 mdname_t *stripenp, 206 int fast, 207 md_error_t *ep 208 ) 209 { 210 mddrivename_t *dnp = stripenp->drivenamep; 211 char *miscname; 212 ms_unit_t *ms; 213 md_stripe_t *stripep; 214 uint_t row; 215 216 /* must have set */ 217 assert(sp != NULL); 218 assert(sp->setno == MD_MIN2SET(meta_getminor(stripenp->dev))); 219 220 /* short circuit */ 221 if (dnp->unitp != NULL) { 222 assert(dnp->unitp->type == MD_DEVICE); 223 return ((md_stripe_t *)dnp->unitp); 224 } 225 226 /* get miscname and unit */ 227 if ((miscname = metagetmiscname(stripenp, ep)) == NULL) 228 return (NULL); 229 if (strcmp(miscname, MD_STRIPE) != 0) { 230 (void) mdmderror(ep, MDE_NOT_STRIPE, 231 meta_getminor(stripenp->dev), stripenp->cname); 232 return (NULL); 233 } 234 if ((ms = (ms_unit_t *)meta_get_mdunit(sp, stripenp, ep)) == NULL) 235 return (NULL); 236 assert(ms->c.un_type == MD_DEVICE); 237 238 /* allocate stripe */ 239 stripep = Zalloc(sizeof (*stripep)); 240 241 /* allocate rows */ 242 assert(ms->un_nrows > 0); 243 stripep->rows.rows_len = ms->un_nrows; 244 stripep->rows.rows_val = Zalloc(stripep->rows.rows_len * 245 sizeof (*stripep->rows.rows_val)); 246 247 /* get common info */ 248 stripep->common.namep = stripenp; 249 stripep->common.type = ms->c.un_type; 250 stripep->common.state = ms->c.un_status; 251 stripep->common.capabilities = ms->c.un_capabilities; 252 stripep->common.parent = ms->c.un_parent; 253 stripep->common.size = ms->c.un_total_blocks; 254 stripep->common.user_flags = ms->c.un_user_flags; 255 stripep->common.revision = ms->c.un_revision; 256 257 /* get options */ 258 if ((ms->un_hsp_id != MD_HSP_NONE) && 259 ((stripep->hspnamep = metahsphspname(&sp, ms->un_hsp_id, 260 ep)) == NULL)) { 261 goto out; 262 } 263 264 /* get rows */ 265 for (row = 0; (row < ms->un_nrows); ++row) { 266 struct ms_row *mdr = &ms->un_row[row]; 267 struct ms_comp *mdcomp = (void *)&((char *)ms)[ms->un_ocomp]; 268 md_row_t *rp = &stripep->rows.rows_val[row]; 269 uint_t comp, c; 270 271 /* get interlace */ 272 rp->interlace = mdr->un_interlace; 273 274 /* allocate comps */ 275 assert(mdr->un_ncomp > 0); 276 rp->comps.comps_len = mdr->un_ncomp; 277 rp->comps.comps_val = Zalloc(rp->comps.comps_len * 278 sizeof (*rp->comps.comps_val)); 279 280 /* get components */ 281 for (comp = 0, c = mdr->un_icomp; (comp < mdr->un_ncomp); 282 ++comp, ++c) { 283 struct ms_comp *mdc = &mdcomp[c]; 284 diskaddr_t comp_start_blk = mdc->un_start_block; 285 md_comp_t *cp = &rp->comps.comps_val[comp]; 286 287 /* get the component name */ 288 cp->compnamep = metakeyname(&sp, mdc->un_key, fast, ep); 289 if (cp->compnamep == NULL) 290 goto out; 291 292 /* if hotspared */ 293 if (mdc->un_mirror.ms_hs_id != 0) { 294 diskaddr_t hs_start_blk = mdc->un_start_block; 295 296 /* get the hotspare name */ 297 cp->hsnamep = metakeyname(&sp, 298 mdc->un_mirror.ms_hs_key, fast, ep); 299 if (cp->hsnamep == NULL) 300 goto out; 301 302 if (getenv("META_DEBUG_START_BLK") != NULL) { 303 if (metagetstart(sp, cp->hsnamep, 304 ep) == MD_DISKADDR_ERROR) 305 mdclrerror(ep); 306 307 if ((cp->hsnamep->start_blk == 0) && 308 (hs_start_blk != 0)) 309 md_eprintf(dgettext(TEXT_DOMAIN, 310 "%s: suspected bad start block," 311 " seems labelled [stripe/hs]\n"), 312 cp->hsnamep->cname); 313 314 if ((cp->hsnamep->start_blk > 0) && 315 (hs_start_blk == 0) && 316 ! ((row == 0) && (comp == 0))) 317 md_eprintf(dgettext(TEXT_DOMAIN, 318 "%s: suspected bad start block, " 319 "seems unlabelled [stripe/hs]\n"), 320 cp->hsnamep->cname); 321 } 322 /* override any start_blk */ 323 cp->hsnamep->start_blk = hs_start_blk; 324 325 /* get the right component start_blk */ 326 comp_start_blk = mdc->un_mirror.ms_orig_blk; 327 } else { 328 if (getenv("META_DEBUG_START_BLK") != NULL) { 329 if (metagetstart(sp, cp->compnamep, 330 ep) == MD_DISKADDR_ERROR) 331 mdclrerror(ep); 332 333 if ((cp->compnamep->start_blk == 0) && 334 (comp_start_blk != 0)) 335 md_eprintf(dgettext(TEXT_DOMAIN, 336 "%s: suspected bad start block," 337 " seems labelled [stripe]"), 338 cp->compnamep->cname); 339 340 if ((cp->compnamep->start_blk > 0) && 341 (comp_start_blk == 0) && 342 ! ((row == 0) && (comp == 0))) 343 md_eprintf(dgettext(TEXT_DOMAIN, 344 "%s: suspected bad start block, " 345 "seems unlabelled [stripe]"), 346 cp->compnamep->cname); 347 } 348 } 349 350 /* override any start_blk */ 351 cp->compnamep->start_blk = comp_start_blk; 352 353 /* get state */ 354 cp->state = mdc->un_mirror.ms_state; 355 356 /* get time of last state change */ 357 cp->timestamp = mdc->un_mirror.ms_timestamp; 358 359 /* get lasterr count */ 360 cp->lasterrcnt = mdc->un_mirror.ms_lasterrcnt; 361 } 362 } 363 364 /* cleanup, return success */ 365 Free(ms); 366 dnp->unitp = (md_common_t *)stripep; 367 return (stripep); 368 369 /* cleanup, return error */ 370 out: 371 Free(ms); 372 meta_free_stripe(stripep); 373 return (NULL); 374 } 375 376 /* 377 * get stripe 378 */ 379 md_stripe_t * 380 meta_get_stripe( 381 mdsetname_t *sp, 382 mdname_t *stripenp, 383 md_error_t *ep 384 ) 385 { 386 return (meta_get_stripe_common(sp, stripenp, 0, ep)); 387 } 388 389 /* 390 * check stripe for dev 391 */ 392 static int 393 in_stripe( 394 mdsetname_t *sp, 395 mdname_t *stripenp, 396 mdname_t *np, 397 diskaddr_t slblk, 398 diskaddr_t nblks, 399 md_error_t *ep 400 ) 401 { 402 md_stripe_t *stripep; 403 uint_t row; 404 405 /* should be in the same set */ 406 assert(sp != NULL); 407 408 /* get unit */ 409 if ((stripep = meta_get_stripe(sp, stripenp, ep)) == NULL) 410 return (-1); 411 412 /* look in rows */ 413 for (row = 0; (row < stripep->rows.rows_len); ++row) { 414 md_row_t *rp = &stripep->rows.rows_val[row]; 415 uint_t comp; 416 417 /* look in columns */ 418 for (comp = 0; (comp < rp->comps.comps_len); ++comp) { 419 md_comp_t *cp = &rp->comps.comps_val[comp]; 420 mdname_t *compnp = cp->compnamep; 421 diskaddr_t comp_sblk; 422 int err; 423 424 /* check same drive since metagetstart() can fail */ 425 if ((err = meta_check_samedrive(np, compnp, ep)) < 0) 426 return (-1); 427 else if (err == 0) 428 continue; 429 430 /* check overlap */ 431 if ((comp_sblk = metagetstart(sp, compnp, ep)) == 432 MD_DISKADDR_ERROR) 433 return (-1); 434 if (meta_check_overlap(stripenp->cname, np, 435 slblk, nblks, compnp, comp_sblk, -1, 436 ep) != 0) { 437 return (-1); 438 } 439 } 440 } 441 442 /* return success */ 443 return (0); 444 } 445 446 /* 447 * check to see if we're in a stripe 448 */ 449 int 450 meta_check_instripe( 451 mdsetname_t *sp, 452 mdname_t *np, 453 diskaddr_t slblk, 454 diskaddr_t nblks, 455 md_error_t *ep 456 ) 457 { 458 mdnamelist_t *stripenlp = NULL; 459 mdnamelist_t *p; 460 int rval = 0; 461 462 /* should have a set */ 463 assert(sp != NULL); 464 465 /* for each stripe */ 466 if (meta_get_stripe_names(sp, &stripenlp, 0, ep) < 0) 467 return (-1); 468 for (p = stripenlp; (p != NULL); p = p->next) { 469 mdname_t *stripenp = p->namep; 470 471 /* check stripe */ 472 if (in_stripe(sp, stripenp, np, slblk, nblks, ep) != 0) { 473 rval = -1; 474 break; 475 } 476 } 477 478 /* cleanup, return success */ 479 metafreenamelist(stripenlp); 480 return (rval); 481 } 482 483 /* 484 * check component 485 */ 486 int 487 meta_check_component( 488 mdsetname_t *sp, 489 mdname_t *np, 490 int force, 491 md_error_t *ep 492 ) 493 { 494 mdchkopts_t options = (MDCHK_ALLOW_MDDB); 495 md_common_t *mdp; 496 497 /* 498 * See if we are a soft partition: meta_sp_issp() returns 0 if 499 * np points to a soft partition, so the if and else clauses 500 * here represent "not a soft partition" and "soft partition," 501 * respectively. 502 */ 503 if (meta_sp_issp(sp, np, ep) != 0) { 504 /* make sure we have a disk */ 505 if (metachkcomp(np, ep) != 0) 506 return (-1); 507 } else { 508 /* make sure soft partition can parent & doesn't have parent */ 509 if ((mdp = meta_get_unit(sp, np, ep)) == NULL) 510 return (mdmderror(ep, MDE_INVAL_UNIT, NULL, 511 np->cname)); 512 if (mdp->capabilities == MD_CANT_PARENT) 513 return (mdmderror(ep, MDE_INVAL_UNIT, NULL, 514 np->cname)); 515 if (MD_HAS_PARENT(mdp->parent)) { 516 mdname_t *pnp; 517 518 pnp = metamnumname(&sp, mdp->parent, 0, ep); 519 if (pnp == NULL) { 520 return (-1); 521 } 522 523 return (mduseerror(ep, MDE_ALREADY, np->dev, 524 pnp->cname, np->cname)); 525 } 526 } 527 528 /* check to ensure that it is not already in use */ 529 if ((! force) && 530 (meta_check_inuse(sp, np, MDCHK_INUSE, ep) != 0)) { 531 return (-1); 532 } 533 534 /* make sure it is in the set */ 535 if (meta_check_inset(sp, np, ep) != 0) 536 return (-1); 537 538 /* make sure its not in a metadevice */ 539 if (meta_check_inmeta(sp, np, options, 0, -1, ep) != 0) 540 return (-1); 541 542 /* return success */ 543 return (0); 544 } 545 546 /* 547 * print stripe 548 */ 549 static int 550 stripe_print( 551 md_stripe_t *stripep, 552 char *fname, 553 FILE *fp, 554 mdprtopts_t options, 555 md_error_t *ep 556 ) 557 { 558 uint_t row; 559 int rval = -1; 560 561 if (options & PRINT_LARGEDEVICES) { 562 if (stripep->common.revision != MD_64BIT_META_DEV) { 563 rval = 0; 564 goto out; 565 } 566 } 567 568 if (options & PRINT_FN) { 569 if (stripep->common.revision != MD_FN_META_DEV) { 570 rval = 0; 571 goto out; 572 } 573 } 574 575 /* print name and num rows */ 576 if (fprintf(fp, "%s %u", 577 stripep->common.namep->cname, stripep->rows.rows_len) == EOF) 578 goto out; 579 580 /* print rows */ 581 for (row = 0; (row < stripep->rows.rows_len); ++row) { 582 md_row_t *rp = &stripep->rows.rows_val[row]; 583 uint_t comp; 584 585 /* print num components */ 586 if (fprintf(fp, " %u", rp->comps.comps_len) == EOF) 587 goto out; 588 589 /* 590 * Print components. Always print the full path name. 591 */ 592 for (comp = 0; (comp < rp->comps.comps_len); ++comp) { 593 md_comp_t *cp = &rp->comps.comps_val[comp]; 594 595 if (fprintf(fp, " %s", cp->compnamep->rname) == EOF) 596 goto out; 597 } 598 599 /* print interlace */ 600 if (rp->comps.comps_len > 1) 601 if (fprintf(fp, " -i %lldb", rp->interlace) == EOF) 602 goto out; 603 604 /* print continuation */ 605 if (row != (stripep->rows.rows_len - 1)) 606 if (fprintf(fp, " \\\n\t") == EOF) 607 goto out; 608 } 609 610 /* print hotspare name */ 611 if (stripep->hspnamep != NULL) 612 if (fprintf(fp, " -h %s", stripep->hspnamep->hspname) == EOF) 613 goto out; 614 615 /* terminate last line */ 616 if (fprintf(fp, "\n") == EOF) 617 goto out; 618 619 /* success */ 620 rval = 0; 621 622 /* cleanup, return error */ 623 out: 624 if (rval != 0) 625 (void) mdsyserror(ep, errno, fname); 626 return (rval); 627 } 628 629 /* 630 * convert component state to name 631 */ 632 char * 633 comp_state_to_name( 634 md_comp_t *mdcp, 635 md_timeval32_t *tvp, 636 uint_t tstate /* Errored tstate flags */ 637 ) 638 { 639 comp_state_t state = mdcp->state; 640 641 /* grab time */ 642 if (tvp != NULL) 643 *tvp = mdcp->timestamp; 644 645 if (tstate != 0) { 646 return (dgettext(TEXT_DOMAIN, "Unavailable")); 647 } 648 649 /* return state */ 650 switch (state) { 651 case CS_OKAY: 652 return (dgettext(TEXT_DOMAIN, "Okay")); 653 case CS_ERRED: 654 return (dgettext(TEXT_DOMAIN, "Maintenance")); 655 case CS_LAST_ERRED: 656 return (dgettext(TEXT_DOMAIN, "Last Erred")); 657 case CS_RESYNC: 658 return (dgettext(TEXT_DOMAIN, "Resyncing")); 659 default: 660 return (dgettext(TEXT_DOMAIN, "invalid")); 661 } 662 } 663 664 /* 665 * print subdevice stripe row 666 */ 667 static int 668 subdev_row_report( 669 mdsetname_t *sp, 670 md_row_t *rp, 671 char *fname, 672 FILE *fp, 673 mdprtopts_t options, 674 uint_t top_tstate, /* Errored tstate flags */ 675 md_error_t *ep 676 ) 677 { 678 uint_t comp; 679 int rval = -1; 680 ddi_devid_t dtp; 681 int len = 0; 682 683 684 /* 685 * building a format string on the fly that will be used 686 * in fprintf. This is to allow really really long ctd names 687 */ 688 for (comp = 0; (comp < rp->comps.comps_len); ++comp) { 689 md_comp_t *cp = &rp->comps.comps_val[comp]; 690 char *cname = cp->compnamep->cname; 691 692 len = max(len, strlen(cname)); 693 } 694 695 len = max(len, strlen(dgettext(TEXT_DOMAIN, "Device"))); 696 len += 2; 697 /* print header */ 698 if (! (options & PRINT_TIMES)) { 699 if (fprintf(fp, 700 "\t%-*.*s %-12.12s %5.5s %12.12s %5.5s %s\n", 701 len, len, 702 dgettext(TEXT_DOMAIN, "Device"), 703 dgettext(TEXT_DOMAIN, "Start Block"), 704 dgettext(TEXT_DOMAIN, "Dbase"), 705 dgettext(TEXT_DOMAIN, "State"), 706 dgettext(TEXT_DOMAIN, "Reloc"), 707 dgettext(TEXT_DOMAIN, "Hot Spare")) == EOF) { 708 goto out; 709 } 710 } else { 711 if (fprintf(fp, 712 "\t%-*s %5s %5s %-11s %-5s %-9s %s\n", 713 len, 714 dgettext(TEXT_DOMAIN, "Device"), 715 dgettext(TEXT_DOMAIN, "Start"), 716 dgettext(TEXT_DOMAIN, "Dbase"), 717 dgettext(TEXT_DOMAIN, "State"), 718 dgettext(TEXT_DOMAIN, "Reloc"), 719 dgettext(TEXT_DOMAIN, "Hot Spare"), 720 dgettext(TEXT_DOMAIN, "Time")) == EOF) { 721 goto out; 722 } 723 } 724 725 726 /* print components */ 727 for (comp = 0; (comp < rp->comps.comps_len); ++comp) { 728 md_comp_t *cp = &rp->comps.comps_val[comp]; 729 mdname_t *namep = cp->compnamep; 730 char *cname = namep->cname; 731 diskaddr_t start_blk; 732 int has_mddb; 733 char *has_mddb_str; 734 char *comp_state; 735 md_timeval32_t tv; 736 char *hsname = ((cp->hsnamep != NULL) ? 737 cp->hsnamep->cname : ""); 738 char *devid = " "; 739 mdname_t *didnp = NULL; 740 uint_t tstate = 0; 741 742 /* get info */ 743 if ((start_blk = metagetstart(sp, namep, ep)) == 744 MD_DISKADDR_ERROR) { 745 return (-1); 746 } 747 if ((has_mddb = metahasmddb(sp, namep, ep)) < 0) { 748 return (-1); 749 } 750 if (has_mddb) 751 has_mddb_str = dgettext(TEXT_DOMAIN, "Yes"); 752 else 753 has_mddb_str = dgettext(TEXT_DOMAIN, "No"); 754 755 /* 756 * If the component is a metadevice, print out either 757 * unavailable or the state of the metadevice, if not 758 * a metadevice, print nothing if the state of the 759 * stripe is unavailable 760 */ 761 if (metaismeta(namep)) { 762 if (meta_get_tstate(namep->dev, &tstate, ep) != 0) 763 return (-1); 764 comp_state = comp_state_to_name(cp, &tv, tstate & 765 MD_DEV_ERRORED); 766 } else { 767 /* 768 * if top_tstate is set, that implies that you have 769 * a ctd type device with an unavailable metadevice 770 * on top of it. If so, print a - for it's state 771 */ 772 if (top_tstate != 0) 773 comp_state = "-"; 774 else 775 comp_state = comp_state_to_name(cp, &tv, 776 tstate & MD_DEV_ERRORED); 777 } 778 779 /* populate the key in the name_p structure */ 780 if ((didnp = metadevname(&sp, namep->dev, ep)) 781 == NULL) { 782 return (-1); 783 } 784 785 /* determine if devid does NOT exist */ 786 if (options & PRINT_DEVID) { 787 if ((dtp = meta_getdidbykey(sp->setno, getmyside(sp, ep), 788 didnp->key, ep)) == NULL) 789 devid = dgettext(TEXT_DOMAIN, "No "); 790 else { 791 devid = dgettext(TEXT_DOMAIN, "Yes"); 792 free(dtp); 793 } 794 } 795 /* print info */ 796 /* 797 * building a format string on the fly that will be used 798 * in fprintf. This is to allow really really long ctd names 799 */ 800 if (! (options & PRINT_TIMES)) { 801 if (fprintf(fp, 802 "\t%-*s %8lld %-5.5s %12.12s %5.5s %s\n", 803 len, cname, start_blk, 804 has_mddb_str, comp_state, devid, hsname) == EOF) { 805 goto out; 806 } 807 } else { 808 char *timep = meta_print_time(&tv); 809 810 if (fprintf(fp, 811 "\t%-*s %5lld %-5s %-11s %-5s %-9s %s\n", 812 len, cname, start_blk, 813 has_mddb_str, comp_state, devid, hsname, 814 timep) == EOF) { 815 goto out; 816 } 817 } 818 } 819 820 /* success */ 821 rval = 0; 822 823 /* cleanup, return error */ 824 out: 825 if (rval != 0) 826 (void) mdsyserror(ep, errno, fname); 827 return (rval); 828 } 829 830 /* 831 * print toplevel stripe row 832 */ 833 /*ARGSUSED4*/ 834 static int 835 toplev_row_report( 836 mdsetname_t *sp, 837 md_row_t *rp, 838 char *fname, 839 FILE *fp, 840 mdprtopts_t options, 841 md_error_t *ep 842 ) 843 { 844 uint_t comp; 845 int rval = -1; 846 char *devid = " "; 847 mdname_t *didnp = NULL; 848 int len = 0; 849 850 /* 851 * building a format string on the fly that will be used 852 * in fprintf. This is to allow really really long ctd names 853 */ 854 for (comp = 0; (comp < rp->comps.comps_len); ++comp) { 855 len = max(len, 856 strlen(rp->comps.comps_val[comp].compnamep->cname)); 857 } 858 859 len = max(len, strlen(dgettext(TEXT_DOMAIN, "Device"))); 860 len += 2; 861 /* print header */ 862 if (fprintf(fp, 863 "\t%-*.*s %-12.12s %-5.5s\t%s\n", 864 len, len, 865 dgettext(TEXT_DOMAIN, "Device"), 866 dgettext(TEXT_DOMAIN, "Start Block"), 867 dgettext(TEXT_DOMAIN, "Dbase"), 868 dgettext(TEXT_DOMAIN, "Reloc")) == EOF) { 869 goto out; 870 } 871 872 /* print components */ 873 for (comp = 0; (comp < rp->comps.comps_len); ++comp) { 874 md_comp_t *cp = &rp->comps.comps_val[comp]; 875 mdname_t *namep = cp->compnamep; 876 char *cname = namep->cname; 877 diskaddr_t start_blk; 878 int has_mddb; 879 char *has_mddb_str; 880 ddi_devid_t dtp; 881 882 /* get info */ 883 if ((start_blk = metagetstart(sp, namep, ep)) == 884 MD_DISKADDR_ERROR) { 885 return (-1); 886 } 887 if ((has_mddb = metahasmddb(sp, namep, ep)) < 0) { 888 return (-1); 889 } 890 if (has_mddb) 891 has_mddb_str = dgettext(TEXT_DOMAIN, "Yes"); 892 else 893 has_mddb_str = dgettext(TEXT_DOMAIN, "No"); 894 895 /* populate the key in the name_p structure */ 896 if ((didnp = metadevname(&sp, namep->dev, ep)) 897 == NULL) { 898 return (-1); 899 } 900 901 /* determine if devid does NOT exist */ 902 if (options & PRINT_DEVID) { 903 if ((dtp = meta_getdidbykey(sp->setno, getmyside(sp, ep), 904 didnp->key, ep)) == NULL) { 905 devid = dgettext(TEXT_DOMAIN, "No "); 906 } else { 907 devid = dgettext(TEXT_DOMAIN, "Yes"); 908 free(dtp); 909 } 910 } 911 /* print info */ 912 /* 913 * building a format string on the fly that will be used 914 * in fprintf. This is to allow really really long ctd names 915 */ 916 if (fprintf(fp, 917 "\t%-*s %8lld %-5.5s\t%s\n", len, 918 cname, start_blk, has_mddb_str, devid) == EOF) { 919 goto out; 920 } 921 } 922 923 /* success */ 924 rval = 0; 925 926 /* cleanup, return error */ 927 out: 928 if (rval != 0) 929 (void) mdsyserror(ep, errno, fname); 930 return (rval); 931 } 932 933 /* 934 * print stripe options 935 */ 936 int 937 meta_print_stripe_options( 938 mdhspname_t *hspnamep, 939 char *fname, 940 FILE *fp, 941 md_error_t *ep 942 ) 943 { 944 char *hspname = ((hspnamep != NULL) ? hspnamep->hspname : 945 dgettext(TEXT_DOMAIN, "none")); 946 int rval = -1; 947 948 /* print options */ 949 if (fprintf(fp, dgettext(TEXT_DOMAIN, 950 " Hot spare pool: %s\n"), hspname) == EOF) { 951 goto out; 952 } 953 954 /* success */ 955 rval = 0; 956 957 /* cleanup, return error */ 958 out: 959 if (rval != 0) 960 (void) mdsyserror(ep, errno, fname); 961 return (rval); 962 } 963 964 /* 965 * report stripe 966 */ 967 static int 968 stripe_report( 969 mdsetname_t *sp, 970 md_stripe_t *stripep, 971 mdnamelist_t **nlpp, 972 char *fname, 973 FILE *fp, 974 mdprtopts_t options, 975 md_error_t *ep 976 ) 977 { 978 uint_t row; 979 int rval = -1; 980 uint_t tstate = 0; 981 982 /* 983 * if the -B option has been specified check to see if the 984 * metadevice is s "big" one and print if so, also if a 985 * big device we need to store the ctd involved for use in 986 * printing out the relocation information. 987 */ 988 if (options & PRINT_LARGEDEVICES) { 989 if ((stripep->common.revision & MD_64BIT_META_DEV) == 0) { 990 rval = 0; 991 goto out; 992 } else { 993 if (meta_getdevs(sp, stripep->common.namep, 994 nlpp, ep) != 0) 995 goto out; 996 } 997 } 998 999 /* 1000 * if the -D option has been specified check to see if the 1001 * metadevice has a descriptive name and print if so, also if a 1002 * descriptive device name we need to store the ctd involved 1003 * for use in printing out the relocation information. 1004 */ 1005 if (options & PRINT_FN) { 1006 if ((stripep->common.revision & MD_FN_META_DEV) == 0) { 1007 rval = 0; 1008 goto out; 1009 } else { 1010 if (meta_getdevs(sp, stripep->common.namep, 1011 nlpp, ep) != 0) 1012 goto out; 1013 } 1014 } 1015 1016 /* print header */ 1017 if (options & PRINT_HEADER) { 1018 if (fprintf(fp, "%s: Concat/Stripe\n", 1019 stripep->common.namep->cname) == EOF) { 1020 goto out; 1021 } 1022 1023 } 1024 1025 /* print hotspare pool */ 1026 if (stripep->hspnamep != NULL) { 1027 if (meta_print_stripe_options(stripep->hspnamep, 1028 fname, fp, ep) != 0) { 1029 return (-1); 1030 } 1031 } 1032 1033 if (metaismeta(stripep->common.namep)) { 1034 if (meta_get_tstate(stripep->common.namep->dev, &tstate, ep) 1035 != 0) 1036 return (-1); 1037 } 1038 if ((tstate & MD_DEV_ERRORED) != 0) { 1039 if (fprintf(fp, dgettext(TEXT_DOMAIN, 1040 " State: Unavailable\n" 1041 " Reconnect disk and invoke: metastat -i\n")) == EOF) { 1042 goto out; 1043 } 1044 } 1045 1046 /* print size */ 1047 if (fprintf(fp, dgettext(TEXT_DOMAIN, " Size: %lld blocks (%s)\n"), 1048 stripep->common.size, 1049 meta_number_to_string(stripep->common.size, DEV_BSIZE)) 1050 == EOF) { 1051 goto out; 1052 } 1053 1054 /* print rows */ 1055 for (row = 0; (row < stripep->rows.rows_len); ++row) { 1056 md_row_t *rp = &stripep->rows.rows_val[row]; 1057 1058 /* print stripe and interlace */ 1059 if (rp->comps.comps_len > 1) { 1060 if (fprintf(fp, dgettext(TEXT_DOMAIN, 1061 " Stripe %u: (interlace: %lld blocks)\n"), 1062 row, rp->interlace) == EOF) { 1063 goto out; 1064 } 1065 } else { 1066 if (fprintf(fp, dgettext(TEXT_DOMAIN, 1067 " Stripe %u:\n"), 1068 row) == EOF) { 1069 goto out; 1070 } 1071 } 1072 1073 /* print components appropriately */ 1074 if (MD_HAS_PARENT(stripep->common.parent)) { 1075 if (subdev_row_report(sp, rp, fname, fp, options, 1076 tstate & MD_DEV_ERRORED, ep) != 0) { 1077 return (-1); 1078 } 1079 } else { 1080 if (toplev_row_report(sp, rp, fname, fp, options, 1081 ep) != 0) { 1082 return (-1); 1083 } 1084 } 1085 } 1086 1087 /* add extra line */ 1088 if (fprintf(fp, "\n") == EOF) 1089 goto out; 1090 1091 /* success */ 1092 rval = 0; 1093 1094 /* cleanup, return error */ 1095 out: 1096 if (rval != 0) 1097 (void) mdsyserror(ep, errno, fname); 1098 return (rval); 1099 } 1100 1101 /* 1102 * print/report stripe 1103 */ 1104 int 1105 meta_stripe_print( 1106 mdsetname_t *sp, 1107 mdname_t *stripenp, 1108 mdnamelist_t **nlpp, 1109 char *fname, 1110 FILE *fp, 1111 mdprtopts_t options, 1112 md_error_t *ep 1113 ) 1114 { 1115 md_stripe_t *stripep; 1116 int row, comp; 1117 1118 /* should have same set */ 1119 assert(sp != NULL); 1120 assert((stripenp == NULL) || 1121 (sp->setno == MD_MIN2SET(meta_getminor(stripenp->dev)))); 1122 1123 /* print all stripes */ 1124 if (stripenp == NULL) { 1125 mdnamelist_t *nlp = NULL; 1126 mdnamelist_t *p; 1127 int cnt; 1128 int rval = 0; 1129 1130 /* get list */ 1131 if ((cnt = meta_get_stripe_names(sp, &nlp, options, ep)) < 0) 1132 return (-1); 1133 else if (cnt == 0) 1134 return (0); 1135 1136 /* recurse */ 1137 for (p = nlp; (p != NULL); p = p->next) { 1138 mdname_t *np = p->namep; 1139 1140 if (meta_stripe_print(sp, np, nlpp, fname, fp, 1141 options, ep) != 0) 1142 rval = -1; 1143 } 1144 1145 /* cleanup, return success */ 1146 metafreenamelist(nlp); 1147 return (rval); 1148 } 1149 1150 /* get unit structure */ 1151 if ((stripep = meta_get_stripe_common(sp, stripenp, 1152 ((options & PRINT_FAST) ? 1 : 0), ep)) == NULL) 1153 return (-1); 1154 1155 /* check for parented */ 1156 if ((! (options & PRINT_SUBDEVS)) && 1157 (MD_HAS_PARENT(stripep->common.parent))) { 1158 return (0); 1159 } 1160 1161 /* print appropriate detail */ 1162 if (options & PRINT_SHORT) { 1163 if (stripe_print(stripep, fname, fp, options, ep) != 0) 1164 return (-1); 1165 } else { 1166 if (stripe_report(sp, stripep, nlpp, fname, fp, options, 1167 ep) != 0) 1168 return (-1); 1169 } 1170 1171 /* Recurse on components that are metadevices */ 1172 for (row = 0; (row < stripep->rows.rows_len); ++row) { 1173 md_row_t *rp = &stripep->rows.rows_val[row]; 1174 1175 /* look for components that are metadevices */ 1176 for (comp = 0; (comp < rp->comps.comps_len); ++comp) { 1177 md_comp_t *cp = &rp->comps.comps_val[comp]; 1178 mdname_t *namep = cp->compnamep; 1179 1180 if ((metaismeta(namep)) && 1181 (meta_print_name(sp, namep, nlpp, fname, fp, 1182 (options | PRINT_HEADER | PRINT_SUBDEVS), 1183 NULL, ep) != 0)) { 1184 return (-1); 1185 } 1186 } 1187 } 1188 return (0); 1189 } 1190 1191 /* 1192 * find stripe component to replace 1193 */ 1194 int 1195 meta_find_erred_comp( 1196 mdsetname_t *sp, 1197 mdname_t *stripenp, 1198 mdname_t **compnpp, 1199 comp_state_t *compstate, 1200 md_error_t *ep 1201 ) 1202 { 1203 md_stripe_t *stripep; 1204 md_comp_t *compp = NULL; 1205 uint_t lasterrcnt = 0; 1206 uint_t row; 1207 1208 /* get stripe */ 1209 *compnpp = NULL; 1210 if ((stripep = meta_get_stripe_common(sp, stripenp, 1, ep)) == NULL) 1211 return (-1); 1212 1213 /* 1214 * Try to find the first erred component. 1215 * If there is not one, then look for the 1216 * first last_erred component. 1217 */ 1218 for (row = 0; (row < stripep->rows.rows_len); ++row) { 1219 md_row_t *rp = &stripep->rows.rows_val[row]; 1220 uint_t comp; 1221 1222 for (comp = 0; (comp < rp->comps.comps_len); ++comp) { 1223 md_comp_t *cp = &rp->comps.comps_val[comp]; 1224 1225 if ((cp->state == CS_ERRED) && ((compp == NULL) || 1226 (cp->lasterrcnt < lasterrcnt))) { 1227 compp = cp; 1228 lasterrcnt = cp->lasterrcnt; 1229 } 1230 } 1231 } 1232 for (row = 0; (row < stripep->rows.rows_len); ++row) { 1233 md_row_t *rp = &stripep->rows.rows_val[row]; 1234 uint_t comp; 1235 1236 for (comp = 0; (comp < rp->comps.comps_len); ++comp) { 1237 md_comp_t *cp = &rp->comps.comps_val[comp]; 1238 1239 if ((cp->state == CS_LAST_ERRED) && ((compp == NULL) || 1240 (cp->lasterrcnt < lasterrcnt))) { 1241 compp = cp; 1242 lasterrcnt = cp->lasterrcnt; 1243 } 1244 } 1245 } 1246 1247 /* return component */ 1248 if (compp != NULL) { 1249 *compnpp = compp->compnamep; 1250 *compstate = compp->state; 1251 } 1252 1253 /* return success */ 1254 return (0); 1255 } 1256 1257 /* 1258 * invalidate component names 1259 */ 1260 static int 1261 invalidate_components( 1262 mdsetname_t *sp, 1263 mdname_t *stripenp, 1264 md_error_t *ep 1265 ) 1266 { 1267 md_stripe_t *stripep; 1268 uint_t row; 1269 1270 if ((stripep = meta_get_stripe(sp, stripenp, ep)) == NULL) 1271 return (-1); 1272 for (row = 0; (row < stripep->rows.rows_len); ++row) { 1273 md_row_t *rp = &stripep->rows.rows_val[row]; 1274 uint_t comp; 1275 1276 for (comp = 0; (comp < rp->comps.comps_len); ++comp) { 1277 md_comp_t *cp = &rp->comps.comps_val[comp]; 1278 mdname_t *compnp = cp->compnamep; 1279 1280 meta_invalidate_name(compnp); 1281 } 1282 } 1283 return (0); 1284 } 1285 1286 /* 1287 * attach components to stripe 1288 */ 1289 int 1290 meta_stripe_attach( 1291 mdsetname_t *sp, 1292 mdname_t *stripenp, 1293 mdnamelist_t *nlp, 1294 diskaddr_t interlace, 1295 mdcmdopts_t options, 1296 md_error_t *ep 1297 ) 1298 { 1299 mdnamelist_t *lp; 1300 ms_unit_t *old_un, *new_un; 1301 struct ms_row *mdr, *new_mdr; 1302 uint_t newcomps, ncomps, icomp; 1303 uint_t row; 1304 size_t mdsize, first_comp; 1305 diskaddr_t new_blks; 1306 diskaddr_t limit; 1307 diskaddr_t disk_size = 0; 1308 ms_comp_t *mdcomp, *new_comp; 1309 uint_t write_reinstruct = 0; 1310 uint_t read_reinstruct = 0; 1311 mdnamelist_t *keynlp = NULL; 1312 uint_t round_cyl = 1; 1313 minor_t parent; 1314 md_grow_params_t mgp; 1315 int rval = -1; 1316 md_timeval32_t creation_time; 1317 int create_flag = MD_CRO_32BIT; 1318 1319 /* should have a set */ 1320 assert(sp != NULL); 1321 assert(sp->setno == MD_MIN2SET(meta_getminor(stripenp->dev))); 1322 1323 /* check type */ 1324 if (metachkmeta(stripenp, ep) != 0) 1325 return (-1); 1326 1327 /* check and count components */ 1328 assert(nlp != NULL); 1329 newcomps = 0; 1330 for (lp = nlp; (lp != NULL); lp = lp->next) { 1331 mdname_t *np = lp->namep; 1332 mdnamelist_t *p; 1333 1334 /* check against existing devices */ 1335 if (meta_check_component(sp, np, 0, ep) != 0) 1336 return (-1); 1337 1338 /* check against ourselves */ 1339 for (p = lp->next; (p != NULL); p = p->next) { 1340 if (meta_check_overlap(np->cname, np, 0, -1, 1341 p->namep, 0, -1, ep) != 0) { 1342 return (-1); 1343 } 1344 } 1345 1346 /* count */ 1347 ++newcomps; 1348 } 1349 1350 /* get old unit */ 1351 if ((old_un = (ms_unit_t *)meta_get_mdunit(sp, stripenp, ep)) == NULL) 1352 return (-1); 1353 1354 /* if zero, inherit the last rows interlace value */ 1355 if (interlace == 0) { 1356 mdr = &old_un->un_row[old_un->un_nrows - 1]; 1357 interlace = mdr->un_interlace; 1358 } 1359 1360 /* 1361 * calculate size of new unit structure 1362 */ 1363 1364 /* unit + rows */ 1365 mdsize = sizeof (ms_unit_t) - sizeof (struct ms_row); 1366 mdsize += sizeof (struct ms_row) * (old_un->un_nrows + 1); 1367 1368 /* number of new components being added */ 1369 ncomps = newcomps; 1370 1371 /* count the # of components in the old unit */ 1372 mdr = &old_un->un_row[0]; 1373 for (row = 0; (row < old_un->un_nrows); row++) 1374 ncomps += mdr[row].un_ncomp; 1375 first_comp = roundup(mdsize, sizeof (long long)); 1376 mdsize += sizeof (ms_comp_t) * ncomps + (first_comp - mdsize); 1377 1378 /* allocate new unit */ 1379 new_un = Zalloc(mdsize); 1380 new_un->un_ocomp = first_comp; 1381 1382 /* compute new data */ 1383 new_mdr = &new_un->un_row[old_un->un_nrows]; 1384 new_mdr->un_icomp = ncomps - newcomps; 1385 new_mdr->un_ncomp = newcomps; 1386 new_mdr->un_blocks = 0; 1387 new_mdr->un_cum_blocks = 1388 old_un->un_row[old_un->un_nrows - 1].un_cum_blocks; 1389 new_mdr->un_interlace = interlace; 1390 1391 /* for each new device */ 1392 mdcomp = (struct ms_comp *)(void *)&((char *)new_un)[new_un->un_ocomp]; 1393 icomp = new_mdr->un_icomp; 1394 if (meta_gettimeofday(&creation_time) == -1) 1395 return (mdsyserror(ep, errno, NULL)); 1396 for (lp = nlp; (lp != NULL); lp = lp->next) { 1397 mdname_t *np = lp->namep; 1398 diskaddr_t size, start_blk; 1399 mdgeom_t *geomp; 1400 1401 /* figure out how big */ 1402 if ((size = metagetsize(np, ep)) == MD_DISKADDR_ERROR) 1403 goto out; 1404 if ((start_blk = metagetstart(sp, np, ep)) == 1405 MD_DISKADDR_ERROR) 1406 goto out; 1407 if (start_blk >= size) { 1408 (void) mdsyserror(ep, ENOSPC, np->cname); 1409 goto out; 1410 } 1411 size -= start_blk; 1412 if (newcomps > 1) 1413 size = rounddown(size, interlace); 1414 1415 /* adjust for smallest disk */ 1416 if (disk_size == 0) { 1417 disk_size = size; 1418 } else if (size < disk_size) { 1419 disk_size = size; 1420 } 1421 1422 /* get worst reinstructs */ 1423 if ((geomp = metagetgeom(np, ep)) == NULL) 1424 goto out; 1425 if (geomp->write_reinstruct > write_reinstruct) 1426 write_reinstruct = geomp->write_reinstruct; 1427 if (geomp->read_reinstruct > read_reinstruct) 1428 read_reinstruct = geomp->read_reinstruct; 1429 1430 /* In dryrun mode (DOIT not set) we must not alter the mddb */ 1431 if (options & MDCMD_DOIT) { 1432 /* store name in namespace */ 1433 if (add_key_name(sp, np, &keynlp, ep) != 0) 1434 goto out; 1435 } 1436 1437 /* build new component */ 1438 new_comp = &mdcomp[icomp++]; 1439 new_comp->un_key = np->key; 1440 new_comp->un_dev = np->dev; 1441 new_comp->un_start_block = start_blk; 1442 new_comp->un_mirror.ms_state = CS_OKAY; 1443 new_comp->un_mirror.ms_timestamp = creation_time; 1444 } 1445 1446 limit = LLONG_MAX; 1447 1448 /* compute new size */ 1449 new_mdr->un_blocks = new_mdr->un_ncomp * disk_size; 1450 new_blks = new_mdr->un_cum_blocks + new_mdr->un_blocks; 1451 if (new_blks > limit) { 1452 new_mdr->un_cum_blocks = limit; 1453 new_blks = limit; 1454 md_eprintf(dgettext(TEXT_DOMAIN, 1455 "unit size overflow, limit is %lld blocks\n"), 1456 limit); 1457 } else { 1458 new_mdr->un_cum_blocks += new_mdr->un_blocks; 1459 } 1460 new_un->c.un_actual_tb = new_mdr->un_cum_blocks; 1461 new_un->un_nrows = old_un->un_nrows + 1; 1462 1463 /* adjust geometry */ 1464 new_un->c.un_nhead = old_un->c.un_nhead; 1465 new_un->c.un_nsect = old_un->c.un_nsect; 1466 new_un->c.un_rpm = old_un->c.un_rpm; 1467 new_un->c.un_wr_reinstruct = old_un->c.un_wr_reinstruct; 1468 new_un->c.un_rd_reinstruct = old_un->c.un_rd_reinstruct; 1469 if (meta_adjust_geom((md_unit_t *)new_un, stripenp, 1470 write_reinstruct, read_reinstruct, round_cyl, ep) != 0) 1471 goto out; 1472 1473 /* if in dryrun mode, we are done here. */ 1474 if ((options & MDCMD_DOIT) == 0) { 1475 if (options & MDCMD_PRINT) { 1476 if (newcomps == 1) { 1477 (void) printf(dgettext(TEXT_DOMAIN, 1478 "%s: attaching component would suceed\n"), 1479 stripenp->cname); 1480 } else { 1481 (void) printf(dgettext(TEXT_DOMAIN, 1482 "%s: attaching components would suceed\n"), 1483 stripenp->cname); 1484 } 1485 } 1486 rval = 0; /* success */ 1487 goto out; 1488 } 1489 1490 create_flag = meta_check_devicesize(new_un->c.un_total_blocks); 1491 1492 /* grow stripe */ 1493 (void) memset(&mgp, 0, sizeof (mgp)); 1494 mgp.mnum = MD_SID(old_un); 1495 MD_SETDRIVERNAME(&mgp, MD_STRIPE, sp->setno); 1496 mgp.size = mdsize; 1497 mgp.mdp = (uintptr_t)new_un; 1498 mgp.nrows = old_un->un_nrows; 1499 if (create_flag == MD_CRO_32BIT) { 1500 mgp.options = MD_CRO_32BIT; 1501 new_un->c.un_revision &= ~MD_64BIT_META_DEV; 1502 } else { 1503 mgp.options = MD_CRO_64BIT; 1504 new_un->c.un_revision |= MD_64BIT_META_DEV; 1505 } 1506 1507 if ((MD_HAS_PARENT(old_un->c.un_parent)) && 1508 (old_un->c.un_parent != MD_MULTI_PARENT)) { 1509 mgp.npar = 1; 1510 parent = old_un->c.un_parent; 1511 mgp.par = (uintptr_t)(&parent); 1512 } 1513 1514 if (metaioctl(MD_IOCGROW, &mgp, &mgp.mde, NULL) != 0) { 1515 (void) mdstealerror(ep, &mgp.mde); 1516 goto out; 1517 } 1518 1519 /* clear cache */ 1520 if (invalidate_components(sp, stripenp, ep) != 0) 1521 goto out; 1522 meta_invalidate_name(stripenp); 1523 1524 /* let em know */ 1525 if (options & MDCMD_PRINT) { 1526 if (newcomps == 1) { 1527 (void) printf(dgettext(TEXT_DOMAIN, 1528 "%s: component is attached\n"), stripenp->cname); 1529 } else { 1530 (void) printf(dgettext(TEXT_DOMAIN, 1531 "%s: components are attached\n"), stripenp->cname); 1532 } 1533 (void) fflush(stdout); 1534 } 1535 1536 /* grow any parents */ 1537 if (meta_concat_parent(sp, stripenp, ep) != 0) 1538 return (-1); 1539 1540 rval = 0; /* success */ 1541 1542 /* cleanup, return error */ 1543 out: 1544 Free(old_un); 1545 Free(new_un); 1546 if (options & MDCMD_DOIT) { 1547 if (rval != 0) 1548 (void) del_key_names(sp, keynlp, NULL); 1549 metafreenamelist(keynlp); 1550 } 1551 return (rval); 1552 } 1553 1554 /* 1555 * get stripe parameters 1556 */ 1557 int 1558 meta_stripe_get_params( 1559 mdsetname_t *sp, 1560 mdname_t *stripenp, 1561 ms_params_t *paramsp, 1562 md_error_t *ep 1563 ) 1564 { 1565 md_stripe_t *stripep; 1566 1567 /* should have a set */ 1568 assert(sp != NULL); 1569 assert(sp->setno == MD_MIN2SET(meta_getminor(stripenp->dev))); 1570 1571 /* check name */ 1572 if (metachkmeta(stripenp, ep) != 0) 1573 return (-1); 1574 1575 /* get unit */ 1576 if ((stripep = meta_get_stripe(sp, stripenp, ep)) == NULL) 1577 return (-1); 1578 1579 /* return parameters */ 1580 (void) memset(paramsp, 0, sizeof (*paramsp)); 1581 if (stripep->hspnamep == NULL) 1582 paramsp->hsp_id = MD_HSP_NONE; 1583 else 1584 paramsp->hsp_id = stripep->hspnamep->hsp; 1585 return (0); 1586 } 1587 1588 /* 1589 * set stripe parameters 1590 */ 1591 int 1592 meta_stripe_set_params( 1593 mdsetname_t *sp, 1594 mdname_t *stripenp, 1595 ms_params_t *paramsp, 1596 md_error_t *ep 1597 ) 1598 { 1599 md_stripe_params_t msp; 1600 1601 /* should have a set */ 1602 assert(sp != NULL); 1603 assert(sp->setno == MD_MIN2SET(meta_getminor(stripenp->dev))); 1604 1605 /* check name */ 1606 if (metachkmeta(stripenp, ep) != 0) 1607 return (-1); 1608 1609 /* set parameters */ 1610 (void) memset(&msp, 0, sizeof (msp)); 1611 MD_SETDRIVERNAME(&msp, MD_STRIPE, sp->setno); 1612 msp.mnum = meta_getminor(stripenp->dev); 1613 msp.params = *paramsp; 1614 if (metaioctl(MD_IOCCHANGE, &msp, &msp.mde, stripenp->cname) != 0) 1615 return (mdstealerror(ep, &msp.mde)); 1616 1617 /* clear cache */ 1618 meta_invalidate_name(stripenp); 1619 1620 /* return success */ 1621 return (0); 1622 } 1623 1624 /* 1625 * check for dups in the stripe itself 1626 */ 1627 static int 1628 check_twice( 1629 md_stripe_t *stripep, 1630 uint_t row, 1631 uint_t comp, 1632 md_error_t *ep 1633 ) 1634 { 1635 mdname_t *stripenp = stripep->common.namep; 1636 mdname_t *thisnp; 1637 uint_t r; 1638 1639 thisnp = stripep->rows.rows_val[row].comps.comps_val[comp].compnamep; 1640 for (r = 0; (r <= row); ++r) { 1641 md_row_t *rp = &stripep->rows.rows_val[r]; 1642 uint_t e = ((r == row) ? comp : rp->comps.comps_len); 1643 uint_t c; 1644 1645 for (c = 0; (c < e); ++c) { 1646 md_comp_t *cp = &rp->comps.comps_val[c]; 1647 mdname_t *compnp = cp->compnamep; 1648 1649 if (meta_check_overlap(stripenp->cname, thisnp, 0, -1, 1650 compnp, 0, -1, ep) != 0) { 1651 return (-1); 1652 } 1653 } 1654 } 1655 return (0); 1656 } 1657 1658 /* 1659 * default stripe interlace 1660 */ 1661 diskaddr_t 1662 meta_default_stripe_interlace(void) 1663 { 1664 diskaddr_t interlace; 1665 1666 /* default to 512k, round up if necessary */ 1667 interlace = btodb(512 * 1024); 1668 if (interlace < btodb(MININTERLACE)) 1669 interlace = roundup(MININTERLACE, interlace); 1670 return (interlace); 1671 } 1672 1673 /* 1674 * convert interlaces 1675 */ 1676 int 1677 meta_stripe_check_interlace( 1678 diskaddr_t interlace, 1679 char *uname, 1680 md_error_t *ep 1681 ) 1682 { 1683 if ((interlace < btodb(MININTERLACE)) || 1684 (interlace > btodb(MAXINTERLACE))) { 1685 return (mderror(ep, MDE_BAD_INTERLACE, uname)); 1686 } 1687 return (0); 1688 } 1689 1690 1691 /* 1692 * check stripe 1693 */ 1694 int 1695 meta_check_stripe( 1696 mdsetname_t *sp, 1697 md_stripe_t *stripep, 1698 mdcmdopts_t options, 1699 md_error_t *ep 1700 ) 1701 { 1702 mdname_t *stripenp = stripep->common.namep; 1703 int force = ((options & MDCMD_FORCE) ? 1 : 0); 1704 int doit = ((options & MDCMD_DOIT) ? 1 : 0); 1705 int updateit = ((options & MDCMD_UPDATE) ? 1 : 0); 1706 uint_t row; 1707 1708 /* check rows */ 1709 if (stripep->rows.rows_len < 1) { 1710 return (mdmderror(ep, MDE_BAD_STRIPE, 1711 meta_getminor(stripenp->dev), stripenp->cname)); 1712 } 1713 for (row = 0; (row < stripep->rows.rows_len); ++row) { 1714 md_row_t *rp = &stripep->rows.rows_val[row]; 1715 uint_t comp; 1716 1717 /* check number */ 1718 if (rp->comps.comps_len < 1) { 1719 return (mdmderror(ep, MDE_BAD_STRIPE, 1720 meta_getminor(stripenp->dev), stripenp->cname)); 1721 } 1722 1723 /* compute default interlace */ 1724 if (rp->interlace == 0) { 1725 rp->interlace = meta_default_stripe_interlace(); 1726 } 1727 1728 /* check interlace */ 1729 if (meta_stripe_check_interlace(rp->interlace, stripenp->cname, 1730 ep) != 0) { 1731 return (-1); 1732 } 1733 1734 /* check components */ 1735 for (comp = 0; (comp < rp->comps.comps_len); ++comp) { 1736 md_comp_t *cp = &rp->comps.comps_val[comp]; 1737 mdname_t *compnp = cp->compnamep; 1738 diskaddr_t start_blk, size; 1739 1740 /* check component */ 1741 if (!updateit) { 1742 if (meta_check_component(sp, compnp, 1743 force, ep) != 0) 1744 return (-1); 1745 if (((start_blk = metagetstart(sp, compnp, 1746 ep)) == MD_DISKADDR_ERROR) || 1747 ((size = metagetsize(compnp, ep)) == 1748 MD_DISKADDR_ERROR)) { 1749 return (-1); 1750 } 1751 if (start_blk >= size) 1752 return (mdsyserror(ep, ENOSPC, 1753 compnp->cname)); 1754 size -= start_blk; 1755 size = rounddown(size, rp->interlace); 1756 if (size == 0) 1757 return (mdsyserror(ep, ENOSPC, 1758 compnp->cname)); 1759 } 1760 1761 /* check this stripe too */ 1762 if (check_twice(stripep, row, comp, ep) != 0) 1763 return (-1); 1764 } 1765 } 1766 1767 /* check hotspare pool name */ 1768 if (doit) { 1769 if ((stripep->hspnamep != NULL) && 1770 (metachkhsp(sp, stripep->hspnamep, ep) != 0)) { 1771 return (-1); 1772 } 1773 } 1774 1775 /* return success */ 1776 return (0); 1777 } 1778 1779 /* 1780 * setup stripe geometry 1781 */ 1782 static int 1783 stripe_geom( 1784 md_stripe_t *stripep, 1785 ms_unit_t *ms, 1786 md_error_t *ep 1787 ) 1788 { 1789 uint_t nrow = stripep->rows.rows_len; 1790 uint_t write_reinstruct = 0; 1791 uint_t read_reinstruct = 0; 1792 uint_t round_cyl = 1; 1793 uint_t row; 1794 mdgeom_t *geomp; 1795 diskaddr_t first_row_size = 0; 1796 char *miscname; 1797 int is_sp = 0; 1798 1799 /* get worst reinstructs */ 1800 for (row = 0; (row < nrow); ++row) { 1801 md_row_t *rp = &stripep->rows.rows_val[row]; 1802 uint_t ncomp = rp->comps.comps_len; 1803 uint_t comp; 1804 1805 for (comp = 0; (comp < ncomp); ++comp) { 1806 md_comp_t *cp = &rp->comps.comps_val[comp]; 1807 mdname_t *compnp = cp->compnamep; 1808 1809 if ((geomp = metagetgeom(compnp, ep)) == NULL) 1810 return (-1); 1811 if (geomp->write_reinstruct > write_reinstruct) 1812 write_reinstruct = geomp->write_reinstruct; 1813 if (geomp->read_reinstruct > read_reinstruct) 1814 read_reinstruct = geomp->read_reinstruct; 1815 } 1816 } 1817 1818 if ((geomp = metagetgeom( 1819 stripep->rows.rows_val[0].comps.comps_val[0].compnamep, 1820 ep)) == NULL) { 1821 return (-1); 1822 } 1823 /* 1824 * Figure out if the first component is a softpartition as the 1825 * truncation check only occurs on them. 1826 */ 1827 if ((miscname = metagetmiscname( 1828 stripep->rows.rows_val[0].comps.comps_val[0].compnamep, 1829 ep)) == NULL) { 1830 if (!mdisdeverror(ep, MDE_NOT_META)) 1831 return (-1); 1832 } else if (strcmp(miscname, MD_SP) == 0) { 1833 is_sp = 1; 1834 } 1835 1836 1837 /* setup geometry from first device */ 1838 if (meta_setup_geom((md_unit_t *)ms, stripep->common.namep, geomp, 1839 write_reinstruct, read_reinstruct, round_cyl, ep) != 0) 1840 return (-1); 1841 1842 /* 1843 * Here we want to make sure that any truncation did not 1844 * result in lost data (or, more appropriately, inaccessible 1845 * data). 1846 * 1847 * This is mainly a danger for (1, 1) concats, but it is 1848 * mathematically possible for other somewhat contrived 1849 * arrangements where in the sum of the lengths of each row 1850 * beyond the first is smaller than the cylinder size of the 1851 * only component in the first row. 1852 * 1853 * It is tempting to simply test for truncation here, by 1854 * (md->c.un_total_blocks < md->c.un_actual_tb). That does 1855 * not tell us, however, if rounding resulted in data loss, 1856 * rather only that it occurred. The somewhat less obvious 1857 * test below covers both the obvious (1, 1) case and the 1858 * aforementioned corner case. 1859 */ 1860 first_row_size = ms->un_row[0].un_blocks; 1861 if (is_sp == 1) { 1862 md_unit_t *md = (md_unit_t *)ms; 1863 1864 if (md->c.un_total_blocks < first_row_size) { 1865 char buf[] = VAL2STR(ULLONG_MAX); 1866 1867 /* 1868 * The only difference here is the text of the error 1869 * message, since the remediation is slightly 1870 * different in the one-component versus 1871 * multiple-component cases. 1872 */ 1873 if (nrow == 1) { 1874 (void) mderror(ep, MDE_STRIPE_TRUNC_SINGLE, 1875 stripep->common.namep->cname); 1876 } else { 1877 (void) mderror(ep, MDE_STRIPE_TRUNC_MULTIPLE, 1878 stripep->common.namep->cname); 1879 } 1880 1881 /* 1882 * By the size comparison above and the initialization 1883 * of buf[] in terms of ULLONG_MAX, we guarantee that 1884 * the value arg is non-negative and that we won't 1885 * overflow the container. 1886 */ 1887 mderrorextra(ep, ulltostr((md->c.un_total_blocks + 1888 (geomp->nhead * geomp->nsect)) 1889 - first_row_size, &buf[sizeof (buf) - 1])); 1890 1891 return (-1); 1892 } 1893 } 1894 1895 /* return success */ 1896 return (0); 1897 } 1898 1899 /* 1900 * create stripe 1901 */ 1902 int 1903 meta_create_stripe( 1904 mdsetname_t *sp, 1905 md_stripe_t *stripep, 1906 mdcmdopts_t options, 1907 md_error_t *ep 1908 ) 1909 { 1910 mdname_t *stripenp = stripep->common.namep; 1911 int force = ((options & MDCMD_FORCE) ? 1 : 0); 1912 int doall = ((options & MDCMD_ALLOPTION) ? 1 : 0); 1913 uint_t nrow = stripep->rows.rows_len; 1914 uint_t ncomp = 0; 1915 uint_t icomp = 0; 1916 diskaddr_t cum_blocks = 0; 1917 diskaddr_t limit; 1918 size_t mdsize, first_comp; 1919 uint_t row; 1920 ms_unit_t *ms; 1921 ms_comp_t *mdcomp; 1922 mdnamelist_t *keynlp = NULL; 1923 md_set_params_t set_params; 1924 int rval = -1; 1925 md_timeval32_t creation_time; 1926 int create_flag = MD_CRO_32BIT; 1927 1928 /* validate stripe */ 1929 if (meta_check_stripe(sp, stripep, options, ep) != 0) 1930 return (-1); 1931 1932 /* allocate stripe unit */ 1933 mdsize = sizeof (*ms) - sizeof (ms->un_row[0]); 1934 mdsize += sizeof (ms->un_row) * nrow; 1935 for (row = 0; (row < nrow); ++row) { 1936 md_row_t *rp = &stripep->rows.rows_val[row]; 1937 1938 ncomp += rp->comps.comps_len; 1939 } 1940 first_comp = roundup(mdsize, sizeof (long long)); 1941 mdsize += (first_comp - mdsize) + (ncomp * sizeof (ms_comp_t)); 1942 ms = Zalloc(mdsize); 1943 ms->un_ocomp = first_comp; 1944 if (meta_gettimeofday(&creation_time) == -1) 1945 return (mdsyserror(ep, errno, NULL)); 1946 1947 /* do rows */ 1948 mdcomp = (ms_comp_t *)(void *)&((char *)ms)[ms->un_ocomp]; 1949 for (row = 0; (row < nrow); ++row) { 1950 md_row_t *rp = &stripep->rows.rows_val[row]; 1951 uint_t ncomp = rp->comps.comps_len; 1952 struct ms_row *mdr = &ms->un_row[row]; 1953 diskaddr_t disk_size = 0; 1954 uint_t comp; 1955 1956 /* setup component count and offfset */ 1957 mdr->un_icomp = icomp; 1958 mdr->un_ncomp = ncomp; 1959 1960 /* do components */ 1961 for (comp = 0; (comp < ncomp); ++comp) { 1962 md_comp_t *cp = &rp->comps.comps_val[comp]; 1963 mdname_t *compnp = cp->compnamep; 1964 ms_comp_t *mdc = &mdcomp[icomp++]; 1965 diskaddr_t size, start_blk; 1966 1967 /* 1968 * get start and size 1969 * if first component is labelled, include label 1970 */ 1971 if ((size = metagetsize(compnp, ep)) == 1972 MD_DISKADDR_ERROR) 1973 goto out; 1974 if ((start_blk = metagetstart(sp, compnp, ep)) == 1975 MD_DISKADDR_ERROR) 1976 goto out; 1977 if ((row == 0) && (comp == 0)) { 1978 diskaddr_t label; 1979 int has_db; 1980 1981 if ((has_db = metahasmddb(sp, compnp, ep)) < 0) 1982 goto out; 1983 if ((label = metagetlabel(compnp, ep)) == 1984 MD_DISKADDR_ERROR) 1985 goto out; 1986 if ((has_db == 0) && (label != 0)) { 1987 ms->c.un_flag |= MD_LABELED; 1988 start_blk = compnp->start_blk = 0; 1989 } 1990 } 1991 /* make sure we still have something left */ 1992 if (start_blk >= size) { 1993 (void) mdsyserror(ep, ENOSPC, compnp->cname); 1994 goto out; 1995 } 1996 size -= start_blk; 1997 1998 /* 1999 * round down by interlace: this only applies 2000 * if this row is a stripe, as indicated by 2001 * (ncomp > 1) 2002 */ 2003 if (ncomp > 1) 2004 size = rounddown(size, rp->interlace); 2005 2006 if (size == 0) { 2007 (void) mdsyserror(ep, ENOSPC, compnp->cname); 2008 goto out; 2009 } 2010 2011 /* 2012 * adjust for smallest disk: for a concat (any 2013 * row with only one component), this will 2014 * never hit the second conditional. 2015 */ 2016 if (disk_size == 0) { 2017 disk_size = size; 2018 } else if (size < disk_size) { 2019 disk_size = size; 2020 } 2021 2022 if (options & MDCMD_DOIT) { 2023 /* store name in namespace */ 2024 if (add_key_name(sp, compnp, &keynlp, ep) != 0) 2025 goto out; 2026 } 2027 2028 /* setup component */ 2029 mdc->un_key = compnp->key; 2030 mdc->un_dev = compnp->dev; 2031 mdc->un_start_block = start_blk; 2032 mdc->un_mirror.ms_state = CS_OKAY; 2033 mdc->un_mirror.ms_timestamp = creation_time; 2034 } 2035 limit = LLONG_MAX; 2036 2037 /* setup row */ 2038 mdr->un_blocks = mdr->un_ncomp * disk_size; 2039 cum_blocks += mdr->un_blocks; 2040 if (cum_blocks > limit) { 2041 cum_blocks = limit; 2042 md_eprintf(dgettext(TEXT_DOMAIN, 2043 "unit size overflow, limit is %lld blocks\n"), 2044 limit); 2045 } 2046 mdr->un_cum_blocks = cum_blocks; 2047 mdr->un_interlace = rp->interlace; 2048 } 2049 2050 /* setup unit */ 2051 ms->c.un_type = MD_DEVICE; 2052 MD_SID(ms) = meta_getminor(stripenp->dev); 2053 ms->c.un_actual_tb = cum_blocks; 2054 ms->c.un_size = mdsize; 2055 if (stripep->hspnamep != NULL) 2056 ms->un_hsp_id = stripep->hspnamep->hsp; 2057 else 2058 ms->un_hsp_id = MD_HSP_NONE; 2059 ms->un_nrows = nrow; 2060 2061 /* fill in the size of the stripe */ 2062 if (options & MDCMD_UPDATE) { 2063 stripep->common.size = ms->c.un_total_blocks; 2064 for (row = 0; (row < nrow); ++row) { 2065 stripep->rows.rows_val[row].row_size = 2066 ms->un_row[row].un_blocks; 2067 } 2068 } 2069 2070 if (stripe_geom(stripep, ms, ep) != 0) { 2071 /* 2072 * If the device is being truncated then only allow this 2073 * if the user is aware (using the -f option) or they 2074 * are in a recovery/complete build situation (using the -a 2075 * option). 2076 */ 2077 if ((mdiserror(ep, MDE_STRIPE_TRUNC_SINGLE) || 2078 mdiserror(ep, MDE_STRIPE_TRUNC_MULTIPLE)) && 2079 (force || doall)) { 2080 md_eprintf(dgettext(TEXT_DOMAIN, 2081 "%s: WARNING: This form of metainit is not recommended.\n" 2082 "The stripe is truncating the size of the underlying device.\n" 2083 "Please see ERRORS in metainit(1M) for additional information.\n"), 2084 stripenp->cname); 2085 mdclrerror(ep); 2086 } else { 2087 goto out; 2088 } 2089 } 2090 2091 create_flag = meta_check_devicesize(ms->c.un_total_blocks); 2092 2093 /* if we're not doing anything, return success */ 2094 if (! (options & MDCMD_DOIT)) { 2095 rval = 0; /* success */ 2096 goto out; 2097 } 2098 2099 /* create stripe */ 2100 (void) memset(&set_params, 0, sizeof (set_params)); 2101 2102 /* did the user tell us to generate a large device? */ 2103 if (create_flag == MD_CRO_64BIT) { 2104 ms->c.un_revision |= MD_64BIT_META_DEV; 2105 set_params.options = MD_CRO_64BIT; 2106 } else { 2107 ms->c.un_revision &= ~MD_64BIT_META_DEV; 2108 set_params.options = MD_CRO_32BIT; 2109 } 2110 2111 set_params.mnum = MD_SID(ms); 2112 set_params.size = ms->c.un_size; 2113 set_params.mdp = (uintptr_t)ms; 2114 MD_SETDRIVERNAME(&set_params, MD_STRIPE, MD_MIN2SET(set_params.mnum)); 2115 if (metaioctl(MD_IOCSET, &set_params, &set_params.mde, 2116 stripenp->cname) != 0) { 2117 (void) mdstealerror(ep, &set_params.mde); 2118 goto out; 2119 } 2120 rval = 0; /* success */ 2121 2122 /* cleanup, return success */ 2123 out: 2124 Free(ms); 2125 if (rval != 0) { 2126 (void) del_key_names(sp, keynlp, NULL); 2127 } 2128 2129 metafreenamelist(keynlp); 2130 if ((rval == 0) && (options & MDCMD_DOIT)) { 2131 if (invalidate_components(sp, stripenp, ep) != 0) 2132 rval = -1; 2133 meta_invalidate_name(stripenp); 2134 } 2135 return (rval); 2136 } 2137 2138 /* 2139 * initialize stripe 2140 * NOTE: this functions is metainit(1m)'s command line parser! 2141 */ 2142 int 2143 meta_init_stripe( 2144 mdsetname_t **spp, 2145 int argc, 2146 char *argv[], 2147 mdcmdopts_t options, 2148 md_error_t *ep 2149 ) 2150 { 2151 char *uname = argv[0]; 2152 mdname_t *stripenp = NULL; 2153 int old_optind; 2154 int c; 2155 md_stripe_t *stripep = NULL; 2156 uint_t nrow, row; 2157 int rval = -1; 2158 2159 /* get stripe name */ 2160 assert(argc > 0); 2161 if (argc < 1) 2162 goto syntax; 2163 2164 if ((stripenp = metaname(spp, uname, META_DEVICE, ep)) == NULL) 2165 goto out; 2166 assert(*spp != NULL); 2167 uname = stripenp->cname; 2168 if (metachkmeta(stripenp, ep) != 0) 2169 goto out; 2170 2171 if (!(options & MDCMD_NOLOCK)) { 2172 /* grab set lock */ 2173 if (meta_lock(*spp, TRUE, ep)) 2174 goto out; 2175 2176 if (meta_check_ownership(*spp, ep) != 0) 2177 goto out; 2178 } 2179 2180 /* see if it exists already */ 2181 if (metagetmiscname(stripenp, ep) != NULL) { 2182 (void) mdmderror(ep, MDE_UNIT_ALREADY_SETUP, 2183 meta_getminor(stripenp->dev), uname); 2184 goto out; 2185 } else if (! mdismderror(ep, MDE_UNIT_NOT_SETUP)) { 2186 goto out; 2187 } else { 2188 mdclrerror(ep); 2189 } 2190 --argc, ++argv; 2191 2192 /* parse general options */ 2193 optind = 0; 2194 opterr = 0; 2195 if (getopt(argc, argv, "") != -1) 2196 goto options; 2197 2198 /* allocate stripe */ 2199 stripep = Zalloc(sizeof (*stripep)); 2200 2201 /* setup common */ 2202 stripep->common.namep = stripenp; 2203 stripep->common.type = MD_DEVICE; 2204 2205 /* allocate and parse rows */ 2206 if (argc < 1) { 2207 (void) mdmderror(ep, MDE_NROWS, meta_getminor(stripenp->dev), 2208 uname); 2209 goto out; 2210 } else if ((sscanf(argv[0], "%u", &nrow) != 1) || ((int)nrow < 0)) { 2211 goto syntax; 2212 } else if (nrow < 1) { 2213 (void) mdmderror(ep, MDE_NROWS, meta_getminor(stripenp->dev), 2214 uname); 2215 goto out; 2216 } 2217 --argc, ++argv; 2218 stripep->rows.rows_len = nrow; 2219 stripep->rows.rows_val = 2220 Zalloc(nrow * sizeof (*stripep->rows.rows_val)); 2221 for (row = 0; (row < nrow); ++row) { 2222 md_row_t *mdr = &stripep->rows.rows_val[row]; 2223 uint_t ncomp, comp; 2224 2225 /* allocate and parse components */ 2226 if (argc < 1) { 2227 (void) mdmderror(ep, MDE_NROWS, 2228 meta_getminor(stripenp->dev), uname); 2229 goto out; 2230 } else if ((sscanf(argv[0], "%u", &ncomp) != 1) || 2231 ((int)ncomp < 0)) { 2232 goto syntax; 2233 } else if (ncomp < 1) { 2234 (void) mdmderror(ep, MDE_NCOMPS, 2235 meta_getminor(stripenp->dev), uname); 2236 goto out; 2237 } 2238 --argc, ++argv; 2239 mdr->comps.comps_len = ncomp; 2240 mdr->comps.comps_val = 2241 Zalloc(ncomp * sizeof (*mdr->comps.comps_val)); 2242 for (comp = 0; (comp < ncomp); ++comp) { 2243 md_comp_t *mdc = &mdr->comps.comps_val[comp]; 2244 mdname_t *compnp; 2245 2246 /* parse component name */ 2247 if (argc < 1) { 2248 (void) mdmderror(ep, MDE_NCOMPS, 2249 meta_getminor(stripenp->dev), uname); 2250 goto out; 2251 } 2252 if ((compnp = metaname(spp, argv[0], UNKNOWN, 2253 ep)) == NULL) { 2254 goto out; 2255 } 2256 /* check for soft partition */ 2257 if (meta_sp_issp(*spp, compnp, ep) != 0) { 2258 /* check disk */ 2259 if (metachkcomp(compnp, ep) != 0) { 2260 goto out; 2261 } 2262 } 2263 mdc->compnamep = compnp; 2264 --argc, ++argv; 2265 } 2266 2267 /* parse row options */ 2268 old_optind = optind = 0; 2269 opterr = 0; 2270 while ((c = getopt(argc, argv, "i:")) != -1) { 2271 switch (c) { 2272 case 'i': 2273 if (parse_interlace(uname, optarg, 2274 &mdr->interlace, ep) != 0) { 2275 goto out; 2276 } 2277 if (meta_stripe_check_interlace(mdr->interlace, 2278 uname, ep)) 2279 goto out; 2280 break; 2281 2282 default: 2283 optind = old_optind; /* bomb out later */ 2284 goto done_row_opts; 2285 } 2286 old_optind = optind; 2287 } 2288 done_row_opts: 2289 argc -= optind; 2290 argv += optind; 2291 } 2292 2293 /* parse stripe options */ 2294 old_optind = optind = 0; 2295 opterr = 0; 2296 while ((c = getopt(argc, argv, "h:")) != -1) { 2297 switch (c) { 2298 case 'h': 2299 if ((stripep->hspnamep = metahspname(spp, optarg, 2300 ep)) == NULL) { 2301 goto out; 2302 } 2303 2304 /* 2305 * Get out if the specified hotspare pool really 2306 * doesn't exist. 2307 */ 2308 if (stripep->hspnamep->hsp == MD_HSP_NONE) { 2309 (void) mdhsperror(ep, MDE_INVAL_HSP, 2310 stripep->hspnamep->hsp, optarg); 2311 goto out; 2312 } 2313 break; 2314 2315 default: 2316 argc += old_optind; 2317 argv += old_optind; 2318 goto options; 2319 } 2320 old_optind = optind; 2321 } 2322 argc -= optind; 2323 argv += optind; 2324 2325 /* we should be at the end */ 2326 if (argc != 0) 2327 goto syntax; 2328 2329 /* create stripe */ 2330 if (meta_create_stripe(*spp, stripep, options, ep) != 0) 2331 goto out; 2332 rval = 0; /* success */ 2333 2334 /* let em know */ 2335 if (options & MDCMD_PRINT) { 2336 (void) printf(dgettext(TEXT_DOMAIN, 2337 "%s: Concat/Stripe is setup\n"), 2338 uname); 2339 (void) fflush(stdout); 2340 } 2341 goto out; 2342 2343 /* syntax error */ 2344 syntax: 2345 rval = meta_cook_syntax(ep, MDE_SYNTAX, uname, argc, argv); 2346 goto out; 2347 2348 /* options error */ 2349 options: 2350 rval = meta_cook_syntax(ep, MDE_OPTION, uname, argc, argv); 2351 goto out; 2352 2353 /* cleanup, return error */ 2354 out: 2355 if (stripep != NULL) 2356 meta_free_stripe(stripep); 2357 return (rval); 2358 } 2359 2360 /* 2361 * reset stripes 2362 */ 2363 int 2364 meta_stripe_reset( 2365 mdsetname_t *sp, 2366 mdname_t *stripenp, 2367 mdcmdopts_t options, 2368 md_error_t *ep 2369 ) 2370 { 2371 md_stripe_t *stripep; 2372 int rval = -1; 2373 int row, comp; 2374 2375 /* should have same set */ 2376 assert(sp != NULL); 2377 assert((stripenp == NULL) || 2378 (sp->setno == MD_MIN2SET(meta_getminor(stripenp->dev)))); 2379 2380 /* reset all stripes */ 2381 if (stripenp == NULL) { 2382 mdnamelist_t *stripenlp = NULL; 2383 mdnamelist_t *p; 2384 2385 /* for each stripe */ 2386 rval = 0; 2387 if (meta_get_stripe_names(sp, &stripenlp, 0, ep) < 0) 2388 return (-1); 2389 for (p = stripenlp; (p != NULL); p = p->next) { 2390 /* reset stripe */ 2391 stripenp = p->namep; 2392 2393 /* 2394 * If this is a multi-node set, we send a series 2395 * of individual metaclear commands. 2396 */ 2397 if (meta_is_mn_set(sp, ep)) { 2398 if (meta_mn_send_metaclear_command(sp, 2399 stripenp->cname, options, 0, ep) != 0) { 2400 rval = -1; 2401 break; 2402 } 2403 } else { 2404 if (meta_stripe_reset(sp, stripenp, 2405 options, ep) != 0) { 2406 rval = -1; 2407 break; 2408 } 2409 } 2410 } 2411 2412 /* cleanup, return success */ 2413 metafreenamelist(stripenlp); 2414 return (rval); 2415 } 2416 2417 /* check name */ 2418 if (metachkmeta(stripenp, ep) != 0) 2419 return (-1); 2420 2421 /* get unit structure */ 2422 if ((stripep = meta_get_stripe(sp, stripenp, ep)) == NULL) 2423 return (-1); 2424 2425 /* make sure nobody owns us */ 2426 if (MD_HAS_PARENT(stripep->common.parent)) { 2427 return (mdmderror(ep, MDE_IN_USE, meta_getminor(stripenp->dev), 2428 stripenp->cname)); 2429 } 2430 2431 /* clear subdevices cache */ 2432 if (invalidate_components(sp, stripenp, ep) != 0) 2433 return (-1); 2434 2435 /* clear metadevice */ 2436 if (meta_reset(sp, stripenp, options, ep) != 0) 2437 goto out; 2438 rval = 0; /* success */ 2439 2440 /* let em know */ 2441 if (options & MDCMD_PRINT) { 2442 (void) printf(dgettext(TEXT_DOMAIN, 2443 "%s: Concat/Stripe is cleared\n"), 2444 stripenp->cname); 2445 (void) fflush(stdout); 2446 } 2447 2448 /* clear subdevices */ 2449 if (! (options & MDCMD_RECURSE)) 2450 goto out; 2451 2452 for (row = 0; (row < stripep->rows.rows_len); ++row) { 2453 md_row_t *rp = &stripep->rows.rows_val[row]; 2454 for (comp = 0; (comp < rp->comps.comps_len); ++comp) { 2455 md_comp_t *cp = &rp->comps.comps_val[comp]; 2456 mdname_t *compnp = cp->compnamep; 2457 2458 /* only recurse on metadevices */ 2459 if (! metaismeta(compnp)) 2460 continue; 2461 2462 if (meta_reset_by_name(sp, compnp, options, ep) != 0) 2463 rval = -1; 2464 } 2465 } 2466 2467 /* cleanup, return success */ 2468 out: 2469 meta_invalidate_name(stripenp); 2470 return (rval); 2471 } 2472 2473 /* 2474 * reports TRUE if any stripe component is in error 2475 */ 2476 int 2477 meta_stripe_anycomp_is_err(mdsetname_t *sp, mdnamelist_t *stripe_names) 2478 { 2479 mdnamelist_t *nlp; 2480 md_error_t status = mdnullerror; 2481 md_error_t *ep = &status; 2482 int any_errs = FALSE; 2483 2484 for (nlp = stripe_names; nlp; nlp = nlp->next) { 2485 md_stripe_t *stripep; 2486 int row; 2487 2488 if ((stripep = meta_get_stripe(sp, nlp->namep, ep)) == NULL) { 2489 any_errs |= TRUE; 2490 goto out; 2491 } 2492 2493 for (row = 0; row < stripep->rows.rows_len; ++row) { 2494 md_row_t *rp = &stripep->rows.rows_val[row]; 2495 uint_t comp; 2496 2497 for (comp = 0; comp < rp->comps.comps_len; ++comp) { 2498 md_comp_t *cp = &rp->comps.comps_val[comp]; 2499 2500 if (cp->state != CS_OKAY) { 2501 any_errs |= TRUE; 2502 goto out; 2503 } 2504 } 2505 } 2506 } 2507 out: 2508 if (!mdisok(ep)) 2509 mdclrerror(ep); 2510 2511 return (any_errs); 2512 } 2513