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