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 #include <sys/types.h> 27 #include <sys/stat.h> 28 #include <sys/mkdev.h> 29 #include <sys/param.h> 30 #include <sys/wait.h> 31 #include <fcntl.h> 32 #include <stdarg.h> 33 #include <stdlib.h> 34 #include <strings.h> 35 #include <errno.h> 36 #include <stdio.h> 37 #include <locale.h> 38 #include <unistd.h> 39 #include <search.h> 40 #include <libgen.h> 41 #include <nsctl.h> 42 43 #include <sys/unistat/spcs_s.h> 44 #include <sys/unistat/spcs_s_u.h> 45 #include <sys/unistat/spcs_errors.h> 46 47 #include <sys/nsctl/sv.h> 48 #include <sys/nsctl/sv_impl.h> 49 50 #include <sys/nsctl/cfg.h> 51 #include <sys/nsctl/nsc_hash.h> 52 53 #include "../sv/svadm.h" 54 55 56 static int sv_max_devices; 57 58 59 /* 60 * support for the special cluster tag "local" to be used with -C in a 61 * cluster for local volumes. 62 */ 63 64 #define SV_LOCAL_TAG "local" 65 66 static int sv_islocal; 67 68 /* 69 * libcfg access. 70 */ 71 72 static CFGFILE *cfg; /* libcfg file pointer */ 73 static int cfg_changed; /* set to 1 if we need to commit changes */ 74 75 static char *cfg_cluster_tag; /* local cluster tag */ 76 77 static char *implicit_tag; /* implicit cluster tag */ 78 79 80 /* 81 * Print width for print_sv() output. 82 */ 83 84 #define STATWIDTH (SV_MAXPATH / 2) 85 86 /* 87 * Pathnames. 88 */ 89 90 static const caddr_t sv_rpath = SV_DEVICE; 91 92 /* 93 * Functions. 94 */ 95 96 static int read_config_file(const caddr_t, sv_name_t []); 97 static int enable_dev(sv_name_t *); 98 static int disable_dev(const caddr_t); 99 static void error(spcs_s_info_t *, caddr_t, ...); 100 static void create_cfg_hash(); 101 static int find_in_hash(char *path); 102 static void destroy_hashtable(); 103 static void remove_from_cfgfile(char *path, int setnumber); 104 105 static caddr_t program; 106 107 static void 108 sv_cfg_open(CFGLOCK mode) 109 { 110 if (cfg != NULL) 111 return; 112 113 cfg = cfg_open(NULL); 114 if (cfg == NULL) { 115 error(NULL, gettext("unable to access the configuration")); 116 /* NOTREACHED */ 117 } 118 119 if (cfg_cluster_tag && *cfg_cluster_tag) { 120 cfg_resource(cfg, cfg_cluster_tag); 121 } else { 122 cfg_resource(cfg, NULL); 123 } 124 if (!cfg_lock(cfg, mode)) { 125 error(NULL, gettext("unable to lock the configuration")); 126 /* NOTREACHED */ 127 } 128 } 129 130 131 static void 132 sv_cfg_close(void) 133 { 134 if (cfg == NULL) 135 return; 136 137 if (cfg_changed) { 138 (void) cfg_commit(cfg); 139 cfg_changed = 0; 140 } 141 142 cfg_close(cfg); 143 cfg = NULL; 144 } 145 146 147 148 static void 149 usage(void) 150 { 151 (void) fprintf(stderr, gettext("usage:\n")); 152 153 (void) fprintf(stderr, gettext( 154 "\t%s -h help\n"), program); 155 156 (void) fprintf(stderr, gettext( 157 "\t%s [-C tag] display status\n"), 158 program); 159 160 (void) fprintf(stderr, gettext( 161 "\t%s [-C tag] -i display " 162 "extended status\n"), program); 163 164 (void) fprintf(stderr, gettext( 165 "\t%s [-C tag] -v display " 166 "version number\n"), program); 167 168 (void) fprintf(stderr, gettext( 169 "\t%s [-C tag] -e { -f file | volume } enable\n"), program); 170 171 (void) fprintf(stderr, gettext( 172 "\t%s [-C tag] -d { -f file | volume } disable\n"), program); 173 174 (void) fprintf(stderr, gettext( 175 "\t%s [-C tag] -r { -f file | volume } reconfigure\n"), program); 176 177 sv_cfg_close(); 178 } 179 180 static void 181 message(caddr_t prefix, spcs_s_info_t *status, caddr_t string, va_list ap) 182 { 183 (void) fprintf(stderr, "%s: %s: ", program, prefix); 184 (void) vfprintf(stderr, string, ap); 185 (void) fprintf(stderr, "\n"); 186 187 if (status) { 188 spcs_s_report(*status, stderr); 189 spcs_s_ufree(status); 190 } 191 } 192 193 194 static void 195 error(spcs_s_info_t *status, caddr_t string, ...) 196 { 197 va_list ap; 198 va_start(ap, string); 199 200 message(gettext("error"), status, string, ap); 201 202 va_end(ap); 203 204 sv_cfg_close(); 205 exit(1); 206 } 207 208 209 static void 210 warn(spcs_s_info_t *status, caddr_t string, ...) 211 { 212 va_list ap; 213 va_start(ap, string); 214 215 message(gettext("warning"), status, string, ap); 216 217 va_end(ap); 218 } 219 220 221 static void 222 sv_get_maxdevs(void) 223 { 224 sv_name_t svn[1]; 225 sv_list_t svl; 226 int fd; 227 228 if (sv_max_devices > 0) 229 return; 230 231 fd = open(sv_rpath, O_RDONLY); 232 if (fd < 0) 233 error(NULL, gettext("unable to open %s: %s"), 234 sv_rpath, strerror(errno)); 235 236 bzero(&svl, sizeof (svl)); 237 bzero(&svn[0], sizeof (svn)); 238 239 svl.svl_names = &svn[0]; 240 svl.svl_error = spcs_s_ucreate(); 241 242 if (ioctl(fd, SVIOC_LIST, &svl) < 0) { 243 (void) close(fd); 244 error(&svl.svl_error, gettext("unable to get max devs")); 245 } 246 247 spcs_s_ufree(&svl.svl_error); 248 sv_max_devices = svl.svl_maxdevs; 249 250 (void) close(fd); 251 } 252 253 254 static sv_name_t * 255 sv_alloc_svnames(void) 256 { 257 sv_name_t *svn = NULL; 258 259 sv_get_maxdevs(); 260 261 svn = calloc(sv_max_devices, sizeof (*svn)); 262 if (svn == NULL) { 263 error(NULL, "unable to allocate %ld bytes of memory", 264 sv_max_devices * sizeof (*svn)); 265 } 266 267 return (svn); 268 } 269 270 271 static void 272 sv_check_dgislocal(char *dgname) 273 { 274 char *othernode; 275 int rc; 276 277 /* 278 * check where this disk service is mastered 279 */ 280 281 rc = cfg_dgname_islocal(dgname, &othernode); 282 if (rc < 0) { 283 error(NULL, gettext("unable to find " 284 "disk service, %s: %s"), dgname, strerror(errno)); 285 } 286 287 if (rc == 0) { 288 error(NULL, gettext("disk service, %s, is " 289 "active on node \"%s\"\nPlease re-issue " 290 "the command on that node"), dgname, othernode); 291 } 292 } 293 294 295 /* 296 * Carry out cluster based checks for a specified volume, or just 297 * global options. 298 */ 299 static void 300 sv_check_cluster(char *path) 301 { 302 char dgname[CFG_MAX_BUF]; 303 static int sv_iscluster = -1; /* set to 1 if running in a cluster */ 304 305 /* 306 * Find out if we are running in a cluster 307 */ 308 if (sv_iscluster == -1) { 309 if ((sv_iscluster = cfg_iscluster()) < 0) { 310 error(NULL, gettext("unable to ascertain environment")); 311 } 312 } 313 314 if (!sv_iscluster && cfg_cluster_tag != NULL) { 315 error(NULL, gettext("-C is not valid when not in a cluster")); 316 } 317 318 if (!sv_iscluster || sv_islocal || path == NULL) { 319 return; 320 } 321 322 323 /* 324 * Cluster-only checks on pathname 325 */ 326 if (cfg_dgname(path, dgname, sizeof (dgname)) == NULL) { 327 error(NULL, gettext("unable to determine " 328 "disk group name for %s"), path); 329 return; 330 } 331 332 if (cfg_cluster_tag != NULL) { 333 /* 334 * Do dgislocal check now in case path did not contain 335 * a dgname. 336 * 337 * E.g. adding a /dev/did/ device to a disk service. 338 */ 339 340 sv_check_dgislocal(cfg_cluster_tag); 341 } 342 343 if (strcmp(dgname, "") == 0) 344 return; /* NULL dgname is valid */ 345 346 if (cfg_cluster_tag == NULL) { 347 /* 348 * Implicitly set the cluster tag to dgname 349 */ 350 351 sv_check_dgislocal(dgname); 352 353 if (implicit_tag) { 354 free(implicit_tag); 355 implicit_tag = NULL; 356 } 357 358 implicit_tag = strdup(dgname); 359 if (implicit_tag == NULL) { 360 error(NULL, 361 gettext("unable to allocate memory " 362 "for cluster tag")); 363 } 364 } else { 365 /* 366 * Check dgname and cluster tag from -C are the same. 367 */ 368 369 if (strcmp(dgname, cfg_cluster_tag) != 0) { 370 error(NULL, 371 gettext("-C (%s) does not match disk group " 372 "name (%s) for %s"), cfg_cluster_tag, 373 dgname, path); 374 } 375 376 /* 377 * sv_check_dgislocal(cfg_cluster_tag) was called above. 378 */ 379 } 380 } 381 382 383 static void 384 print_version(void) 385 { 386 sv_version_t svv; 387 int fd; 388 389 bzero(&svv, sizeof (svv)); 390 svv.svv_error = spcs_s_ucreate(); 391 392 fd = open(sv_rpath, O_RDONLY); 393 if (fd < 0) { 394 warn(NULL, gettext("unable to open %s: %s"), 395 sv_rpath, strerror(errno)); 396 return; 397 } 398 399 if (ioctl(fd, SVIOC_VERSION, &svv) != 0) { 400 error(&svv.svv_error, 401 gettext("unable to read the version number")); 402 /* NOTREACHED */ 403 } 404 405 spcs_s_ufree(&svv.svv_error); 406 #ifdef DEBUG 407 (void) printf(gettext("Storage Volume version %d.%d.%d.%d\n"), 408 svv.svv_major_rev, svv.svv_minor_rev, 409 svv.svv_micro_rev, svv.svv_baseline_rev); 410 #else 411 if (svv.svv_micro_rev) { 412 (void) printf(gettext("Storage Volume version %d.%d.%d\n"), 413 svv.svv_major_rev, svv.svv_minor_rev, svv.svv_micro_rev); 414 } else { 415 (void) printf(gettext("Storage Volume version %d.%d\n"), 416 svv.svv_major_rev, svv.svv_minor_rev); 417 } 418 #endif 419 420 (void) close(fd); 421 } 422 423 int 424 main(int argc, char *argv[]) 425 { 426 extern int optind; 427 extern char *optarg; 428 char *conf_file = NULL; 429 int enable, disable, compare, print, version; 430 int opt, Cflag, fflag, iflag; 431 int rc; 432 433 (void) setlocale(LC_ALL, ""); 434 (void) textdomain("svadm"); 435 436 program = strdup(basename(argv[0])); 437 438 Cflag = fflag = iflag = 0; 439 compare = enable = disable = version = 0; 440 441 print = 1; 442 443 while ((opt = getopt(argc, argv, "C:def:hirv")) != EOF) { 444 switch (opt) { 445 446 case 'C': 447 if (Cflag) { 448 warn(NULL, 449 gettext("-C specified multiple times")); 450 usage(); 451 exit(2); 452 /* NOTREACHED */ 453 } 454 455 Cflag++; 456 cfg_cluster_tag = optarg; 457 break; 458 459 case 'e': 460 print = 0; 461 enable++; 462 break; 463 464 case 'd': 465 print = 0; 466 disable++; 467 break; 468 469 case 'f': 470 fflag++; 471 conf_file = optarg; 472 break; 473 474 case 'i': 475 iflag++; 476 break; 477 478 case 'r': 479 /* Compare running system with sv.cf */ 480 print = 0; 481 compare++; 482 break; 483 484 case 'v': 485 print = 0; 486 version++; 487 break; 488 489 case 'h': 490 usage(); 491 exit(0); 492 493 default: 494 usage(); 495 exit(2); 496 /* NOTREACHED */ 497 } 498 } 499 500 501 /* 502 * Usage checks 503 */ 504 505 if ((enable + disable + compare) > 1) { 506 warn(NULL, gettext("-d, -e and -r are mutually exclusive")); 507 usage(); 508 exit(2); 509 } 510 511 if (fflag && (print || version)) { 512 warn(NULL, gettext("-f is only valid with -d, -e or -r")); 513 usage(); 514 exit(2); 515 } 516 517 if (fflag && optind != argc) { 518 usage(); 519 exit(2); 520 } 521 522 if (print || version) { 523 /* check for no more args */ 524 525 if (optind != argc) { 526 usage(); 527 exit(2); 528 } 529 } else { 530 /* check for inline args */ 531 532 if (!fflag && (argc - optind) != 1) { 533 usage(); 534 exit(2); 535 } 536 } 537 538 if (!print && iflag) { 539 usage(); 540 exit(2); 541 } 542 543 544 /* 545 * Check for the special cluster tag and convert into the 546 * internal representation. 547 */ 548 549 if (cfg_cluster_tag != NULL && 550 strcmp(cfg_cluster_tag, SV_LOCAL_TAG) == 0) { 551 cfg_cluster_tag = "-"; 552 sv_islocal = 1; 553 } 554 555 556 /* 557 * Process commands 558 */ 559 560 if (optind != argc) { 561 /* deal with inline volume argument */ 562 563 rc = 0; 564 if (enable) 565 rc = enable_one_sv(argv[optind]); 566 else if (disable) 567 rc = disable_one_sv(argv[optind]); 568 else /* if (compare) */ 569 compare_one_sv(argv[optind]); 570 571 if (rc != 0) 572 return (1); 573 574 return (0); 575 } 576 577 rc = 0; 578 if (enable) 579 rc = enable_sv(conf_file); 580 else if (disable) 581 rc = disable_sv(conf_file); 582 else if (compare) 583 compare_sv(conf_file); 584 else if (print) 585 print_sv(iflag); 586 else /* if (version) */ 587 print_version(); 588 589 if (rc != 0) 590 return (1); 591 592 return (0); 593 } 594 595 596 597 /* LINT - not static as fwcadm uses it */ 598 static int 599 enable_sv(char *conf_file) 600 { 601 int index; 602 sv_name_t *svn; 603 int cnt; 604 int rc, ret; 605 606 svn = sv_alloc_svnames(); 607 608 index = read_config_file(conf_file, svn); 609 610 rc = ret = 0; 611 612 for (cnt = 0; cnt < index; cnt++) { 613 614 /* 615 * Check for more data. 616 */ 617 if (svn[cnt].svn_path[0] == '\0') { 618 /* 619 * This was set when reading sv.conf. After the last 620 * line svn_path was set to \0, so we are finished. 621 * We shouldn't get here, but put this in just in 622 * case. 623 */ 624 break; 625 } 626 rc = enable_dev(&svn[cnt]); 627 if (rc && !ret) 628 ret = rc; 629 } 630 631 sv_cfg_close(); 632 633 return (ret); 634 } 635 636 637 /* LINT - not static as fwcadm uses it */ 638 static int 639 enable_one_sv(caddr_t path) 640 { 641 sv_name_t svn; 642 int rc; 643 644 sv_get_maxdevs(); 645 646 bzero(&svn, sizeof (svn)); 647 (void) strncpy(svn.svn_path, path, sizeof (svn.svn_path)); 648 svn.svn_mode = (NSC_DEVICE | NSC_CACHE); 649 650 /* force NULL termination */ 651 svn.svn_path[sizeof (svn.svn_path) - 1] = '\0'; 652 653 rc = enable_dev(&svn); 654 sv_cfg_close(); 655 656 return (rc); 657 } 658 659 660 static int 661 enable_dev(sv_name_t *svn) 662 { 663 char buf[CFG_MAX_BUF]; 664 struct stat stb; 665 sv_conf_t svc; 666 int fd; 667 int sev; 668 int rc; 669 char *lcltag; 670 char *altname; 671 672 sv_check_cluster(svn->svn_path); 673 sv_cfg_open(CFG_WRLOCK); 674 675 bzero(&svc, sizeof (svc)); 676 677 if (stat(svn->svn_path, &stb) != 0) { 678 warn(NULL, gettext("unable to access %s: %s"), 679 svn->svn_path, strerror(errno)); 680 return (1); 681 } 682 683 if (!S_ISCHR(stb.st_mode)) { 684 warn(NULL, gettext("%s is not a character device - ignored"), 685 svn->svn_path); 686 return (1); 687 } 688 689 svc.svc_major = major(stb.st_rdev); 690 svc.svc_minor = minor(stb.st_rdev); 691 (void) strncpy(svc.svc_path, svn->svn_path, sizeof (svc.svc_path)); 692 693 fd = open(sv_rpath, O_RDONLY); 694 if (fd < 0) { 695 warn(NULL, gettext("unable to open %s: %s"), 696 svn->svn_path, strerror(errno)); 697 return (1); 698 } 699 700 svc.svc_flag = svn->svn_mode; 701 svc.svc_error = spcs_s_ucreate(); 702 703 /* first, check for duplicates */ 704 rc = cfg_get_canonical_name(cfg, svn->svn_path, &altname); 705 if (rc < 0) { 706 spcs_log("sv", NULL, gettext("Unable to parse config file")); 707 warn(NULL, gettext("Unable to parse config file")); 708 (void) close(fd); 709 return (1); 710 } 711 if (rc) { 712 error(NULL, gettext("'%s' has already been configured as " 713 "'%s'. Re-enter command with the latter name."), 714 svn->svn_path, altname); 715 } 716 717 /* secondly, try to insert it into the dsvol config */ 718 if (implicit_tag && *implicit_tag) { 719 lcltag = implicit_tag; 720 } else if (cfg_cluster_tag && *cfg_cluster_tag) { 721 lcltag = cfg_cluster_tag; 722 } else { 723 lcltag = "-"; 724 } 725 rc = cfg_add_user(cfg, svn->svn_path, lcltag, "sv"); 726 if (CFG_USER_ERR == rc) { 727 spcs_log("sv", NULL, 728 gettext("%s: unable to put %s into dsvol cfg"), 729 program, svn->svn_path); 730 warn(NULL, gettext("unable to put %s into dsvol cfg"), 731 svn->svn_path); 732 (void) close(fd); 733 return (1); 734 } 735 cfg_changed = 1; 736 737 if (CFG_USER_OK == rc) { 738 /* success */ 739 (void) close(fd); 740 return (0); 741 } 742 743 if (ioctl(fd, SVIOC_ENABLE, &svc) < 0) { 744 if ((CFG_USER_REPEAT == rc) && (SV_EENABLED == errno)) { 745 /* it's ok -- we were just double-checking */ 746 (void) close(fd); 747 return (0); 748 } 749 750 spcs_log("sv", &svc.svc_error, 751 gettext("%s: unable to enable %s"), 752 program, svn->svn_path); 753 754 warn(&svc.svc_error, gettext("unable to enable %s"), 755 svn->svn_path); 756 757 /* remove it from dsvol, if we're the ones who put it in */ 758 if (CFG_USER_FIRST == rc) { 759 (void) cfg_rem_user(cfg, svn->svn_path, lcltag, "sv"); 760 } 761 (void) close(fd); 762 return (1); 763 } 764 765 spcs_log("sv", NULL, gettext("%s: enabled %s"), 766 program, svn->svn_path); 767 768 if (implicit_tag != NULL) { 769 #ifdef DEBUG 770 if (cfg_cluster_tag != NULL) { 771 error(NULL, 772 gettext("enable_dev: -C %s AND implicit_tag %s!"), 773 cfg_cluster_tag, implicit_tag); 774 } 775 #endif 776 777 (void) snprintf(buf, sizeof (buf), "%s - %s", 778 svc.svc_path, implicit_tag); 779 } else { 780 (void) strcpy(buf, svc.svc_path); 781 } 782 783 rc = 0; 784 if (cfg_put_cstring(cfg, "sv", buf, sizeof (buf)) < 0) { 785 warn(NULL, 786 gettext("unable to add %s to configuration storage: %s"), 787 svc.svc_path, cfg_error(&sev)); 788 rc = 1; 789 } 790 791 cfg_changed = 1; 792 spcs_s_ufree(&svc.svc_error); 793 (void) close(fd); 794 795 return (rc); 796 } 797 798 799 /* 800 * This routine parses the config file passed in via conf_file and 801 * stores the data in the svn array. The return value is the number 802 * of entries read from conf_file. If an error occurs the error() 803 * routine is called (which exits the program). 804 */ 805 static int 806 read_config_file(const caddr_t conf_file, sv_name_t svn[]) 807 { 808 char line[1024], rdev[1024], junk[1024]; 809 struct stat stb; 810 int lineno; 811 int cnt, i; 812 int index = 0; /* Current location in svn array */ 813 sv_name_t *cur_svn; /* Pointer to svn[index] */ 814 FILE *fp; 815 816 if (access(conf_file, R_OK) != 0 || 817 stat(conf_file, &stb) != 0 || 818 !S_ISREG(stb.st_mode)) { 819 error(NULL, gettext("cannot read config file %s"), conf_file); 820 } 821 822 if ((fp = fopen(conf_file, "r")) == NULL) { 823 error(NULL, gettext("unable to open config file %s: %s"), 824 conf_file, strerror(errno)); 825 } 826 827 lineno = 0; 828 829 while (fgets(line, sizeof (line), fp) != NULL) { 830 lineno++; 831 832 i = strlen(line); 833 834 if (i < 1) 835 continue; 836 837 if (line[i-1] == '\n') 838 line[i-1] = '\0'; 839 else if (i == (sizeof (line) - 1)) { 840 warn(NULL, gettext( 841 "line %d: line too long -- should be less than %d characters"), 842 lineno, (sizeof (line) - 1)); 843 warn(NULL, gettext("line %d: ignored"), lineno); 844 } 845 846 /* 847 * check for comment line. 848 */ 849 if (line[0] == '#') 850 continue; 851 852 cnt = sscanf(line, "%s %s", rdev, junk); 853 854 if (cnt != 1 && cnt != 2) { 855 if (cnt > 0) { 856 warn(NULL, gettext("line %d: invalid format"), 857 lineno); 858 warn(NULL, gettext("line %d: ignored"), lineno); 859 } 860 continue; 861 } 862 863 rdev[sizeof (rdev) - 1] = '\0'; 864 865 cur_svn = &svn[index]; /* For easier reading below */ 866 867 if (strlen(rdev) >= sizeof (cur_svn->svn_path)) { 868 warn(NULL, gettext( 869 "line %d: raw device name (%s) longer than %d characters"), 870 lineno, rdev, 871 (sizeof (cur_svn->svn_path) - 1)); 872 warn(NULL, gettext("line %d: ignored"), lineno); 873 continue; 874 } 875 876 (void) strcpy(cur_svn->svn_path, rdev); 877 cur_svn->svn_mode = (NSC_DEVICE | NSC_CACHE); 878 879 index++; 880 } 881 882 /* Set the last path to NULL */ 883 svn[index].svn_path[0] = '\0'; 884 885 (void) fclose(fp); 886 887 return (index); 888 } 889 890 891 /* 892 * Disable the device from the kernel configuration. 893 * 894 * RETURN: 895 * 0 on success 896 * non-zero on failure. 897 * 898 * Failures are reported to the user. 899 */ 900 static int 901 disable_dev(const caddr_t path) 902 { 903 struct stat stb; 904 sv_conf_t svc; 905 int fd; 906 907 sv_check_cluster(path); 908 909 if (stat(path, &stb) < 0) { 910 svc.svc_major = (major_t)-1; 911 svc.svc_minor = (minor_t)-1; 912 } else { 913 svc.svc_major = major(stb.st_rdev); 914 svc.svc_minor = minor(stb.st_rdev); 915 } 916 917 if ((fd = open(sv_rpath, O_RDONLY)) < 0) { 918 warn(NULL, gettext("unable to open %s: %s"), 919 sv_rpath, strerror(errno)); 920 return (-1); 921 } 922 923 (void) strcpy(svc.svc_path, path); 924 svc.svc_error = spcs_s_ucreate(); 925 926 /* 927 * Issue the ioctl to attempt to disable this device. Note that all 928 * the libdscfg details are handled elsewhere. 929 */ 930 if (ioctl(fd, SVIOC_DISABLE, &svc) < 0) { 931 if (errno != SV_EDISABLED) { 932 spcs_log("sv", &svc.svc_error, 933 gettext("%s: unable to disable %s"), 934 program, path); 935 936 warn(&svc.svc_error, 937 gettext("unable to disable %s"), path); 938 (void) close(fd); 939 return (-1); 940 } 941 } 942 943 spcs_log("sv", NULL, gettext("%s: disabled %s"), program, path); 944 945 spcs_s_ufree(&svc.svc_error); 946 (void) close(fd); 947 948 return (0); 949 } 950 951 952 static void 953 print_cluster_tag(const int setnumber) 954 { 955 char buf[CFG_MAX_BUF]; 956 char key[CFG_MAX_KEY]; 957 958 bzero(buf, sizeof (buf)); 959 (void) snprintf(key, sizeof (key), "sv.set%d.cnode", setnumber); 960 961 (void) cfg_get_cstring(cfg, key, buf, sizeof (buf)); 962 963 if (*buf != '\0') { 964 if (strcmp(buf, "-") == 0) { 965 (void) printf(" [%s]", gettext("local to node")); 966 } else { 967 (void) printf(" [%s: %s]", gettext("cluster"), buf); 968 } 969 } 970 } 971 972 973 /* LINT - not static as fwcadm uses it */ 974 static void 975 print_sv(int verbose) 976 { 977 sv_name_t *svn, *svn_system; /* Devices in system */ 978 sv_list_t svl_system; 979 int fd, i; 980 int setnumber; 981 982 sv_check_cluster(NULL); 983 sv_cfg_open(CFG_RDLOCK); 984 985 svn_system = sv_alloc_svnames(); 986 987 if ((fd = open(sv_rpath, O_RDONLY)) < 0) { 988 (void) printf(gettext("unable to open %s: %s"), 989 sv_rpath, strerror(errno)); 990 return; 991 } 992 993 /* Grab the system list from the driver */ 994 svl_system.svl_count = sv_max_devices; 995 svl_system.svl_names = &svn_system[0]; 996 svl_system.svl_error = spcs_s_ucreate(); 997 998 if (ioctl(fd, SVIOC_LIST, &svl_system) < 0) { 999 error(&svl_system.svl_error, gettext("unable to get list")); 1000 } 1001 1002 spcs_s_ufree(&svl_system.svl_error); 1003 (void) close(fd); 1004 1005 /* 1006 * We build a hashmap out of the entries from the config file to make 1007 * searching faster. We end up taking a performance hit when the # of 1008 * volumes is small, but for larger configurations it's a 1009 * HUGE improvement. 1010 */ 1011 1012 /* build the hashtable */ 1013 cfg_rewind(cfg, CFG_SEC_CONF); 1014 create_cfg_hash(); 1015 1016 /* 1017 * For each volume found from the kernel, print out 1018 * info about it from the kernel. 1019 */ 1020 for (i = 0; i < svl_system.svl_count; i++) { 1021 if (*svn_system[i].svn_path == '\0') { 1022 break; 1023 } 1024 1025 svn = &svn_system[i]; 1026 if (svn->svn_mode == 0) { 1027 #ifdef DEBUG 1028 (void) printf(gettext("%s [kernel guard]\n"), 1029 svn->svn_path); 1030 #endif 1031 continue; 1032 } 1033 /* get sv entry from the hashtable */ 1034 if ((setnumber = find_in_hash(svn->svn_path)) != -1) { 1035 (void) printf("%-*s", STATWIDTH, svn->svn_path); 1036 1037 if (verbose) { 1038 print_cluster_tag(setnumber); 1039 } 1040 1041 (void) printf("\n"); 1042 1043 } else { 1044 /* 1045 * We didn't find the entry in the hashtable. Let 1046 * the user know that the persistent storage is 1047 * inconsistent with the kernel configuration. 1048 */ 1049 if (cfg_cluster_tag == NULL) 1050 warn(NULL, gettext( 1051 "%s is configured, but not in the " 1052 "config storage"), svn->svn_path); 1053 } 1054 } 1055 1056 /* free up the hashtable */ 1057 destroy_hashtable(); 1058 1059 sv_cfg_close(); 1060 } 1061 1062 1063 /* LINT - not static as fwcadm uses it */ 1064 static int 1065 disable_sv(char *conf_file) 1066 { 1067 sv_name_t *svn, *svn_system; /* Devices in system */ 1068 sv_list_t svl_system; 1069 int fd, i, setnumber; 1070 int rc, ret; 1071 1072 svn_system = sv_alloc_svnames(); 1073 1074 rc = ret = 0; 1075 1076 if (conf_file == NULL) { 1077 if ((fd = open(sv_rpath, O_RDONLY)) < 0) { 1078 (void) printf(gettext("unable to open %s: %s"), 1079 sv_rpath, strerror(errno)); 1080 return (1); 1081 } 1082 1083 /* Grab the system list from the driver */ 1084 svl_system.svl_count = sv_max_devices; 1085 svl_system.svl_names = &svn_system[0]; 1086 svl_system.svl_error = spcs_s_ucreate(); 1087 1088 if (ioctl(fd, SVIOC_LIST, &svl_system) < 0) { 1089 error(&(svl_system.svl_error), 1090 gettext("unable to get list")); 1091 } 1092 1093 spcs_s_ufree(&(svl_system.svl_error)); 1094 (void) close(fd); 1095 } else { 1096 svl_system.svl_count = read_config_file(conf_file, svn_system); 1097 } 1098 1099 1100 for (i = 0; i < svl_system.svl_count; i++) { 1101 if (*svn_system[i].svn_path == '\0') 1102 break; 1103 1104 svn = &svn_system[i]; 1105 1106 sv_check_cluster(svn->svn_path); 1107 sv_cfg_open(CFG_WRLOCK); 1108 create_cfg_hash(); 1109 rc = 0; 1110 if ((setnumber = find_in_hash(svn->svn_path)) != -1) { 1111 if ((rc = disable_dev(svn->svn_path)) != -1) { 1112 remove_from_cfgfile(svn->svn_path, setnumber); 1113 } else if (errno == SV_ENODEV) { 1114 remove_from_cfgfile(svn->svn_path, setnumber); 1115 } 1116 } else { 1117 /* warn the user that we didn't find it in cfg file */ 1118 warn(NULL, gettext( 1119 "%s was not found in the config storage"), 1120 svn->svn_path); 1121 /* try to disable anyway */ 1122 (void) disable_dev(svn->svn_path); 1123 rc = 1; 1124 } 1125 1126 sv_cfg_close(); 1127 destroy_hashtable(); 1128 1129 if (rc && !ret) 1130 ret = rc; 1131 } 1132 1133 return (ret); 1134 } 1135 1136 1137 /* LINT - not static as fwcadm uses it */ 1138 static int 1139 disable_one_sv(char *path) 1140 { 1141 int setnumber; 1142 int rc; 1143 1144 sv_get_maxdevs(); 1145 sv_check_cluster(path); 1146 sv_cfg_open(CFG_WRLOCK); 1147 1148 create_cfg_hash(); 1149 if ((setnumber = find_in_hash(path)) != -1) { 1150 /* remove from kernel */ 1151 if ((rc = disable_dev(path)) == 0) { 1152 /* remove the cfgline */ 1153 remove_from_cfgfile(path, setnumber); 1154 } else if (errno == SV_ENODEV) { 1155 remove_from_cfgfile(path, setnumber); 1156 } 1157 } else { 1158 /* warn the user that we didn't find it in cfg file */ 1159 warn(NULL, 1160 gettext("%s was not found in the config storage"), path); 1161 /* still attempt to remove */ 1162 (void) disable_dev(path); 1163 rc = 1; 1164 } 1165 destroy_hashtable(); 1166 1167 sv_cfg_close(); 1168 return (rc); 1169 } 1170 1171 1172 static void 1173 compare_tag(char *path) 1174 { 1175 char buf[CFG_MAX_BUF], vol[CFG_MAX_BUF], cnode[CFG_MAX_BUF]; 1176 char key[CFG_MAX_KEY]; 1177 int found, setnumber, i; 1178 char *tag; 1179 1180 sv_check_cluster(path); 1181 cfg_resource(cfg, (char *)NULL); /* reset */ 1182 cfg_rewind(cfg, CFG_SEC_CONF); 1183 1184 #ifdef DEBUG 1185 if (cfg_cluster_tag != NULL && implicit_tag != NULL) { 1186 error(NULL, gettext("compare_tag: -C %s AND implicit_tag %s!"), 1187 cfg_cluster_tag, implicit_tag); 1188 } 1189 #endif 1190 1191 if (cfg_cluster_tag != NULL) 1192 tag = cfg_cluster_tag; 1193 else if (implicit_tag != NULL) 1194 tag = implicit_tag; 1195 else 1196 tag = "-"; 1197 1198 found = 0; 1199 for (i = 0; i < sv_max_devices; i++) { 1200 setnumber = i + 1; 1201 (void) snprintf(key, sizeof (key), "sv.set%d", setnumber); 1202 if (cfg_get_cstring(cfg, key, buf, sizeof (buf)) < 0) { 1203 break; 1204 } 1205 1206 if (sscanf(buf, "%s - %s", vol, cnode) != 2) { 1207 continue; 1208 } 1209 1210 if (strcmp(path, vol) == 0) { 1211 found = 1; 1212 break; 1213 } 1214 } 1215 1216 if (!found) { 1217 warn(NULL, gettext("unable to find %s in the configuration"), 1218 path); 1219 return; 1220 } 1221 1222 /* have name match, compare cnode to new tag */ 1223 1224 if (strcmp(tag, cnode) == 0) { 1225 /* cluster tags match */ 1226 return; 1227 } 1228 1229 /* need to change the cluster tag */ 1230 1231 (void) snprintf(key, sizeof (key), "sv.set%d.cnode", setnumber); 1232 if (cfg_put_cstring(cfg, key, tag, strlen(tag)) < 0) { 1233 warn(NULL, 1234 gettext("unable to change cluster tag for %s"), path); 1235 return; 1236 } 1237 1238 cfg_changed = 1; 1239 1240 /* change "-" tags to "" for display purposes */ 1241 1242 if (strcmp(tag, "-") == 0) 1243 tag = ""; 1244 1245 if (strcmp(cnode, "-") == 0) 1246 (void) strcpy(cnode, ""); 1247 1248 (void) printf( 1249 gettext("%s: changed cluster tag for %s from \"%s\" to \"%s\"\n"), 1250 program, path, cnode, tag); 1251 1252 spcs_log("sv", NULL, 1253 gettext("%s: changed cluster tag for %s from \"%s\" to \"%s\""), 1254 program, path, cnode, tag); 1255 } 1256 1257 1258 /* LINT - not static as fwcadm uses it */ 1259 static void 1260 compare_sv(char *conf_file) 1261 { 1262 sv_name_t *svn_config; /* Devices in config file */ 1263 sv_name_t *svn_system; /* Devices in system */ 1264 sv_name_t *enable; /* Devices that need enabled */ 1265 sv_list_t svl_system; 1266 int config_cnt; 1267 int sys_cnt = 0; 1268 int setnumber, i, j; 1269 int index = 0; /* Index in enable[] */ 1270 int found; 1271 int fd0; 1272 1273 svn_config = sv_alloc_svnames(); 1274 svn_system = sv_alloc_svnames(); 1275 enable = sv_alloc_svnames(); 1276 1277 bzero(svn_system, sizeof (svn_system)); 1278 bzero(&svl_system, sizeof (svl_system)); 1279 bzero(enable, sizeof (enable)); 1280 1281 /* 1282 * Read the configuration file 1283 * The return value is the number of entries 1284 */ 1285 config_cnt = read_config_file(conf_file, svn_config); 1286 1287 if ((fd0 = open(sv_rpath, O_RDONLY)) < 0) 1288 error(NULL, gettext("unable to open %s: %s"), 1289 sv_rpath, strerror(errno)); 1290 1291 /* Grab the system list from the driver */ 1292 svl_system.svl_count = sv_max_devices; 1293 svl_system.svl_names = &svn_system[0]; 1294 svl_system.svl_error = spcs_s_ucreate(); 1295 1296 if (ioctl(fd0, SVIOC_LIST, &svl_system) < 0) { 1297 error(&svl_system.svl_error, gettext("unable to get list")); 1298 } 1299 1300 spcs_s_ufree(&svl_system.svl_error); 1301 (void) close(fd0); 1302 1303 /* 1304 * Count the number of devices in the system. 1305 * The last entry in the array has '\0' for a path name. 1306 */ 1307 for (j = 0; j < sv_max_devices; j++) { 1308 if (svn_system[j].svn_path[0] != '\0') { 1309 sys_cnt++; 1310 } else { 1311 break; 1312 } 1313 } 1314 /* 1315 * Compare the configuration array with the system array. 1316 * Mark any differences and disable conflicting devices. 1317 */ 1318 for (i = 0; i < config_cnt; i++) { 1319 found = 0; 1320 for (j = 0; j < sys_cnt; j++) { 1321 if (svn_system[j].svn_path[0] == '\0' || 1322 svn_system[j].svn_mode == 0) 1323 continue; 1324 1325 /* Check to see if path matches */ 1326 if (strcmp(svn_system[j].svn_path, 1327 svn_config[i].svn_path) == 0) { 1328 /* Found a match */ 1329 svn_system[j].svn_path[0] = '\0'; 1330 found++; 1331 break; 1332 } 1333 } 1334 1335 if (!found) { 1336 /* Minor number not in system = > enable device */ 1337 enable[index].svn_mode = svn_config[i].svn_mode; 1338 (void) strcpy(enable[index].svn_path, 1339 svn_config[i].svn_path); 1340 index++; 1341 } 1342 } 1343 1344 /* Disable any devices that weren't in the config file */ 1345 for (j = 0; j < sys_cnt; j++) { 1346 sv_check_cluster(NULL); 1347 sv_cfg_open(CFG_WRLOCK); 1348 create_cfg_hash(); 1349 if (svn_system[j].svn_path[0] != '\0' && 1350 svn_system[j].svn_mode != 0) { 1351 (void) printf(gettext("%s: disabling sv: %s\n"), 1352 program, svn_system[j].svn_path); 1353 if (disable_dev(svn_system[j].svn_path) == 0) { 1354 setnumber = 1355 find_in_hash(svn_system[j].svn_path); 1356 if (setnumber != -1) { 1357 /* the volume was found in cfg store */ 1358 remove_from_cfgfile( 1359 svn_system[j].svn_path, setnumber); 1360 } 1361 } 1362 } 1363 sv_cfg_close(); 1364 destroy_hashtable(); 1365 } 1366 1367 while (index) { 1368 /* 1369 * Config file doesn't match system => enable the devices 1370 * in enable[] 1371 */ 1372 index--; 1373 (void) printf(gettext("%s: enabling new sv: %s\n"), 1374 program, enable[index].svn_path); 1375 (void) enable_dev(&enable[index]); 1376 } 1377 1378 /* 1379 * Search for entries where the cluster tag has changed. 1380 */ 1381 sv_check_cluster(NULL); 1382 sv_cfg_open(CFG_WRLOCK); 1383 1384 for (i = 0; i < sv_max_devices; i++) { 1385 if (svn_config[i].svn_path[0] == '\0') 1386 break; 1387 1388 compare_tag(svn_config[i].svn_path); 1389 } 1390 1391 sv_cfg_close(); 1392 } 1393 1394 1395 /* 1396 * We assume that the volume is already enabled and we can only 1397 * be changing the cluster tag. Anything else is an error. 1398 */ 1399 /* LINT - not static as fwcadm uses it */ 1400 static void 1401 compare_one_sv(char *path) 1402 { 1403 sv_get_maxdevs(); 1404 sv_check_cluster(NULL); 1405 sv_cfg_open(CFG_WRLOCK); 1406 1407 compare_tag(path); 1408 1409 sv_cfg_close(); 1410 } 1411 1412 /* 1413 * Read all sets from the libdscfg configuration file, and store everything in 1414 * the hashfile. 1415 * 1416 * We assume that the config file has been opened & rewound for us. We store 1417 * the volume name as the key, and the setnumber where we found it as the data. 1418 * 1419 * The caller can pass in a pointer to the maximum number of volumes, or 1420 * a pointer to NULL, specifying we want 'all' the volumes. The table is 1421 * searched using find_in_hash. 1422 */ 1423 static void 1424 create_cfg_hash() 1425 { 1426 char key[CFG_MAX_KEY], buf[CFG_MAX_BUF]; 1427 char vol[CFG_MAX_BUF], cnode[CFG_MAX_BUF]; 1428 int setnumber; 1429 ENTRY item; 1430 1431 if (hcreate((size_t)sv_max_devices) == 0) 1432 error(NULL, gettext("unable to create hash table")); 1433 1434 for (setnumber = 1; /* CSTYLED */; setnumber++) { 1435 (void) snprintf(key, sizeof (key), "sv.set%d", setnumber); 1436 if (cfg_get_cstring(cfg, key, buf, sizeof (buf)) < 0) 1437 break; 1438 1439 if (sscanf(buf, "%s - %s", vol, cnode) != 2) { 1440 continue; 1441 } 1442 1443 item.key = strdup(vol); 1444 item.data = (void *)setnumber; 1445 if (hsearch(item, ENTER) == NULL) { 1446 error(NULL, 1447 gettext("unable to add entry to hash table")); 1448 } 1449 } 1450 } 1451 1452 /* 1453 * Function to search the hash for a specific volume. If it is found, 1454 * we return the set number. If it isn't found, we return -1 1455 */ 1456 static int 1457 find_in_hash(char *path) 1458 { 1459 ENTRY *found_entry, item; 1460 int retval = -1; 1461 1462 item.key = path; 1463 1464 if ((found_entry = hsearch(item, FIND)) != NULL) { 1465 retval = (int)found_entry->data; 1466 } 1467 1468 return (retval); 1469 } 1470 1471 /* 1472 * Just a wrapper to destory the hashtable. At some point in the future we 1473 * might want to do something more.... For instance, verify that the cfg 1474 * database and the kernel configuration match (?) Just an idea. 1475 */ 1476 static void 1477 destroy_hashtable() 1478 { 1479 hdestroy(); 1480 } 1481 1482 /* 1483 * This function will remove a particular set from the config file. 1484 * 1485 * We make a whole host of assumptions: 1486 * o the hashfile is up to date; 1487 * o The config file has been opened with a WRLOCK for us. 1488 */ 1489 static void 1490 remove_from_cfgfile(char *path, int setnumber) 1491 { 1492 char key[CFG_MAX_KEY]; 1493 int sev; 1494 char *lcltag; 1495 1496 /* attempt to remove the volume from config storage */ 1497 (void) snprintf(key, sizeof (key), "sv.set%d", setnumber); 1498 if (cfg_put_cstring(cfg, key, NULL, 0) < 0) { 1499 warn(NULL, gettext("unable to remove %s from " 1500 "config storage: %s"), path, cfg_error(&sev)); 1501 } else { 1502 if (implicit_tag && *implicit_tag) { 1503 lcltag = implicit_tag; 1504 } else if (cfg_cluster_tag && *cfg_cluster_tag) { 1505 lcltag = cfg_cluster_tag; 1506 } else { 1507 lcltag = "-"; 1508 } 1509 if (cfg_rem_user(cfg, path, lcltag, "sv") != CFG_USER_LAST) { 1510 warn(NULL, gettext("unable to remove %s from dsvol"), 1511 path); 1512 } 1513 cfg_changed = 1; 1514 } 1515 } 1516