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 2009 Sun Microsystems, Inc. All rights reserved. 23 * Use is subject to license terms. 24 */ 25 26 /* 27 * Copyright 2015, Joyent, Inc. All rights reserved. 28 */ 29 30 /* 31 * svcadm - request adminstrative actions for service instances 32 */ 33 34 #include <locale.h> 35 #include <libintl.h> 36 #include <libscf.h> 37 #include <libscf_priv.h> 38 #include <libcontract.h> 39 #include <libcontract_priv.h> 40 #include <sys/contract/process.h> 41 #include <libuutil.h> 42 #include <stddef.h> 43 #include <stdio.h> 44 #include <stdlib.h> 45 #include <string.h> 46 #include <unistd.h> 47 #include <fcntl.h> 48 #include <procfs.h> 49 #include <assert.h> 50 #include <errno.h> 51 #include <zone.h> 52 53 #ifndef TEXT_DOMAIN 54 #define TEXT_DOMAIN "SUNW_OST_OSCMD" 55 #endif /* TEXT_DOMAIN */ 56 57 /* Must be a power of two */ 58 #define HT_BUCKETS 64 59 60 /* 61 * Exit codes for enable and disable -s. 62 */ 63 #define EXIT_SVC_FAILURE 3 64 #define EXIT_DEP_FAILURE 4 65 66 #define WALK_FLAGS (SCF_WALK_UNIPARTIAL | SCF_WALK_MULTIPLE) 67 68 /* 69 * How long we will wait (in seconds) for a service to change state 70 * before re-checking its dependencies. 71 */ 72 #define WAIT_INTERVAL 3 73 74 #define bad_error(func, err) \ 75 uu_panic("%s:%d: %s() failed with unexpected error %d.\n", \ 76 __FILE__, __LINE__, (func), (err)); 77 78 struct ht_elt { 79 struct ht_elt *next; 80 boolean_t active; 81 char str[1]; 82 }; 83 84 85 scf_handle_t *h; 86 ssize_t max_scf_fmri_sz; 87 static const char *emsg_permission_denied; 88 static const char *emsg_nomem; 89 static const char *emsg_create_pg_perm_denied; 90 static const char *emsg_pg_perm_denied; 91 static const char *emsg_prop_perm_denied; 92 static const char *emsg_no_service; 93 94 static int exit_status = 0; 95 static int verbose = 0; 96 static char *scratch_fmri; 97 static char *g_zonename = NULL; 98 static char svcstate[80]; 99 static boolean_t svcsearch = B_FALSE; 100 101 static struct ht_elt **visited; 102 103 void do_scfdie(int lineno) __NORETURN; 104 static void usage_milestone(void) __NORETURN; 105 static void set_astring_prop(const char *, const char *, const char *, 106 uint32_t, const char *, const char *); 107 static void pr_warn(const char *format, ...); 108 109 /* 110 * Visitors from synch.c, needed for enable -s and disable -s. 111 */ 112 extern int is_enabled(scf_instance_t *); 113 extern int has_potential(scf_instance_t *, int); 114 115 void 116 do_scfdie(int lineno) 117 { 118 scf_error_t err; 119 120 switch (err = scf_error()) { 121 case SCF_ERROR_CONNECTION_BROKEN: 122 uu_die(gettext("Connection to repository server broken. " 123 "Exiting.\n")); 124 /* NOTREACHED */ 125 126 case SCF_ERROR_BACKEND_READONLY: 127 uu_die(gettext("Repository is read-only. Exiting.\n")); 128 /* NOTREACHED */ 129 130 default: 131 #ifdef NDEBUG 132 uu_die(gettext("Unexpected libscf error: %s. Exiting.\n"), 133 scf_strerror(err)); 134 #else 135 uu_die("Unexpected libscf error on line %d: %s.\n", lineno, 136 scf_strerror(err)); 137 #endif 138 } 139 } 140 141 #define scfdie() do_scfdie(__LINE__) 142 143 static void 144 usage() 145 { 146 (void) fprintf(stderr, gettext( 147 "Usage: %1$s [-S <state>] [-v] [-Z | -z zone] [cmd [args ... ]]\n\n" 148 "\t%1$s enable [-rst] [<service> ...]\t- enable and online service(s)\n" 149 "\t%1$s disable [-st] [<service> ...]\t- disable and offline " 150 "service(s)\n" 151 "\t%1$s restart [-d] [<service> ...]\t- restart specified service(s)\n" 152 "\t%1$s refresh [<service> ...]\t\t- re-read service configuration\n" 153 "\t%1$s mark [-It] <state> [<service> ...] - set maintenance state\n" 154 "\t%1$s clear [<service> ...]\t\t- clear maintenance state\n" 155 "\t%1$s milestone [-d] <milestone>\t- advance to a service milestone\n" 156 "\n\t" 157 "Services can be specified using an FMRI, abbreviation, or fnmatch(5)\n" 158 "\tpattern, as shown in these examples for svc:/network/smtp:sendmail\n" 159 "\n" 160 "\t%1$s <cmd> svc:/network/smtp:sendmail\n" 161 "\t%1$s <cmd> network/smtp:sendmail\n" 162 "\t%1$s <cmd> network/*mail\n" 163 "\t%1$s <cmd> network/smtp\n" 164 "\t%1$s <cmd> smtp:sendmail\n" 165 "\t%1$s <cmd> smtp\n" 166 "\t%1$s <cmd> sendmail\n"), uu_getpname()); 167 168 exit(UU_EXIT_USAGE); 169 } 170 171 172 /* 173 * FMRI hash table for recursive enable. 174 */ 175 176 static uint32_t 177 hash_fmri(const char *str) 178 { 179 uint32_t h = 0, g; 180 const char *p; 181 182 /* Generic hash function from uts/common/os/modhash.c . */ 183 for (p = str; *p != '\0'; ++p) { 184 h = (h << 4) + *p; 185 if ((g = (h & 0xf0000000)) != 0) { 186 h ^= (g >> 24); 187 h ^= g; 188 } 189 } 190 191 return (h); 192 } 193 194 /* 195 * Return 1 if str has been visited, 0 if it has not, and -1 if memory could not 196 * be allocated. 197 */ 198 static int 199 visited_find_or_add(const char *str, struct ht_elt **hep) 200 { 201 uint32_t h; 202 uint_t i; 203 struct ht_elt *he; 204 205 h = hash_fmri(str); 206 i = h & (HT_BUCKETS - 1); 207 208 for (he = visited[i]; he != NULL; he = he->next) { 209 if (strcmp(he->str, str) == 0) { 210 if (hep) 211 *hep = he; 212 return (1); 213 } 214 } 215 216 he = malloc(offsetof(struct ht_elt, str) + strlen(str) + 1); 217 if (he == NULL) 218 return (-1); 219 220 (void) strcpy(he->str, str); 221 222 he->next = visited[i]; 223 visited[i] = he; 224 225 if (hep) 226 *hep = he; 227 return (0); 228 } 229 230 231 /* 232 * Returns 0, ECANCELED if pg is deleted, ENOENT if propname doesn't exist, 233 * EINVAL if the property is not of boolean type or has no values, and E2BIG 234 * if it has more than one value. *bp is set if 0 or E2BIG is returned. 235 */ 236 int 237 get_bool_prop(scf_propertygroup_t *pg, const char *propname, uint8_t *bp) 238 { 239 scf_property_t *prop; 240 scf_value_t *val; 241 int ret; 242 243 if ((prop = scf_property_create(h)) == NULL || 244 (val = scf_value_create(h)) == NULL) 245 scfdie(); 246 247 if (scf_pg_get_property(pg, propname, prop) != 0) { 248 switch (scf_error()) { 249 case SCF_ERROR_DELETED: 250 ret = ECANCELED; 251 goto out; 252 253 case SCF_ERROR_NOT_FOUND: 254 ret = ENOENT; 255 goto out; 256 257 case SCF_ERROR_NOT_SET: 258 assert(0); 259 abort(); 260 /* NOTREACHED */ 261 262 default: 263 scfdie(); 264 } 265 } 266 267 if (scf_property_get_value(prop, val) == 0) { 268 ret = 0; 269 } else { 270 switch (scf_error()) { 271 case SCF_ERROR_DELETED: 272 ret = ENOENT; 273 goto out; 274 275 case SCF_ERROR_NOT_FOUND: 276 ret = EINVAL; 277 goto out; 278 279 case SCF_ERROR_CONSTRAINT_VIOLATED: 280 ret = E2BIG; 281 break; 282 283 case SCF_ERROR_NOT_SET: 284 assert(0); 285 abort(); 286 /* NOTREACHED */ 287 288 default: 289 scfdie(); 290 } 291 } 292 293 if (scf_value_get_boolean(val, bp) != 0) { 294 if (scf_error() != SCF_ERROR_TYPE_MISMATCH) 295 scfdie(); 296 297 ret = EINVAL; 298 goto out; 299 } 300 301 out: 302 scf_value_destroy(val); 303 scf_property_destroy(prop); 304 return (ret); 305 } 306 307 /* 308 * Returns 0, EPERM, or EROFS. 309 */ 310 static int 311 set_bool_prop(scf_propertygroup_t *pg, const char *propname, boolean_t b) 312 { 313 scf_value_t *v; 314 scf_transaction_t *tx; 315 scf_transaction_entry_t *ent; 316 int ret = 0, r; 317 318 if ((tx = scf_transaction_create(h)) == NULL || 319 (ent = scf_entry_create(h)) == NULL || 320 (v = scf_value_create(h)) == NULL) 321 scfdie(); 322 323 scf_value_set_boolean(v, b); 324 325 for (;;) { 326 if (scf_transaction_start(tx, pg) == -1) { 327 switch (scf_error()) { 328 case SCF_ERROR_PERMISSION_DENIED: 329 ret = EPERM; 330 goto out; 331 332 case SCF_ERROR_BACKEND_READONLY: 333 ret = EROFS; 334 goto out; 335 336 default: 337 scfdie(); 338 } 339 } 340 341 if (scf_transaction_property_change_type(tx, ent, propname, 342 SCF_TYPE_BOOLEAN) != 0) { 343 if (scf_error() != SCF_ERROR_NOT_FOUND) 344 scfdie(); 345 346 if (scf_transaction_property_new(tx, ent, propname, 347 SCF_TYPE_BOOLEAN) != 0) 348 scfdie(); 349 } 350 351 r = scf_entry_add_value(ent, v); 352 assert(r == 0); 353 354 r = scf_transaction_commit(tx); 355 if (r == 1) 356 break; 357 358 scf_transaction_reset(tx); 359 360 if (r != 0) { 361 switch (scf_error()) { 362 case SCF_ERROR_PERMISSION_DENIED: 363 ret = EPERM; 364 goto out; 365 366 case SCF_ERROR_BACKEND_READONLY: 367 ret = EROFS; 368 goto out; 369 370 default: 371 scfdie(); 372 } 373 } 374 375 if (scf_pg_update(pg) == -1) 376 scfdie(); 377 } 378 379 out: 380 scf_transaction_destroy(tx); 381 scf_entry_destroy(ent); 382 scf_value_destroy(v); 383 return (ret); 384 } 385 386 /* 387 * Gets the single astring value of the propname property of pg. prop & v are 388 * scratch space. Returns the length of the string on success or 389 * -ENOENT - pg has no property named propname 390 * -E2BIG - property has no values or multiple values 391 * -EINVAL - property type is not compatible with astring 392 */ 393 ssize_t 394 get_astring_prop(const scf_propertygroup_t *pg, const char *propname, 395 scf_property_t *prop, scf_value_t *v, char *buf, size_t bufsz) 396 { 397 ssize_t sz; 398 399 if (scf_pg_get_property(pg, propname, prop) != 0) { 400 if (scf_error() != SCF_ERROR_NOT_FOUND) 401 scfdie(); 402 403 return (-ENOENT); 404 } 405 406 if (scf_property_get_value(prop, v) != 0) { 407 switch (scf_error()) { 408 case SCF_ERROR_NOT_FOUND: 409 case SCF_ERROR_CONSTRAINT_VIOLATED: 410 return (-E2BIG); 411 412 default: 413 scfdie(); 414 } 415 } 416 417 sz = scf_value_get_astring(v, buf, bufsz); 418 if (sz < 0) { 419 if (scf_error() != SCF_ERROR_TYPE_MISMATCH) 420 scfdie(); 421 422 return (-EINVAL); 423 } 424 425 return (sz); 426 } 427 428 /* 429 * Returns 0 or EPERM. 430 */ 431 static int 432 pg_get_or_add(const scf_instance_t *inst, const char *pgname, 433 const char *pgtype, uint32_t pgflags, scf_propertygroup_t *pg) 434 { 435 again: 436 if (scf_instance_get_pg(inst, pgname, pg) == 0) 437 return (0); 438 439 if (scf_error() != SCF_ERROR_NOT_FOUND) 440 scfdie(); 441 442 if (scf_instance_add_pg(inst, pgname, pgtype, pgflags, pg) == 0) 443 return (0); 444 445 switch (scf_error()) { 446 case SCF_ERROR_EXISTS: 447 goto again; 448 449 case SCF_ERROR_PERMISSION_DENIED: 450 return (EPERM); 451 452 default: 453 scfdie(); 454 /* NOTREACHED */ 455 } 456 } 457 458 static int 459 my_ct_name(char *out, size_t len) 460 { 461 ct_stathdl_t st; 462 char *ct_fmri; 463 ctid_t ct; 464 int fd, errno, ret; 465 466 if ((ct = getctid()) == -1) 467 uu_die(gettext("Could not get contract id for process")); 468 469 fd = contract_open(ct, "process", "status", O_RDONLY); 470 471 if ((errno = ct_status_read(fd, CTD_ALL, &st)) != 0) 472 uu_warn(gettext("Could not read status of contract " 473 "%ld: %s.\n"), ct, strerror(errno)); 474 475 if ((errno = ct_pr_status_get_svc_fmri(st, &ct_fmri)) != 0) 476 uu_warn(gettext("Could not get svc_fmri for contract " 477 "%ld: %s.\n"), ct, strerror(errno)); 478 479 ret = strlcpy(out, ct_fmri, len); 480 481 ct_status_free(st); 482 (void) close(fd); 483 484 return (ret); 485 } 486 487 /* 488 * Set auxiliary_tty and auxiliary_fmri properties in restarter_actions pg to 489 * communicate whether the action is requested from a tty and the fmri of the 490 * responsible process. 491 * 492 * Returns 0, EPERM, or EROFS 493 */ 494 static int 495 restarter_setup(const char *fmri, const scf_instance_t *inst) 496 { 497 boolean_t b = B_FALSE; 498 scf_propertygroup_t *pg = NULL; 499 int ret = 0; 500 501 if ((pg = scf_pg_create(h)) == NULL) 502 scfdie(); 503 504 if (pg_get_or_add(inst, SCF_PG_RESTARTER_ACTIONS, 505 SCF_PG_RESTARTER_ACTIONS_TYPE, SCF_PG_RESTARTER_ACTIONS_FLAGS, 506 pg) == EPERM) { 507 if (!verbose) 508 uu_warn(emsg_permission_denied, fmri); 509 else 510 uu_warn(emsg_create_pg_perm_denied, fmri, 511 SCF_PG_RESTARTER_ACTIONS); 512 513 ret = EPERM; 514 goto out; 515 } 516 517 /* Set auxiliary_tty property */ 518 if (isatty(STDIN_FILENO)) 519 b = B_TRUE; 520 521 /* Create and set state to disabled */ 522 switch (set_bool_prop(pg, SCF_PROPERTY_AUX_TTY, b) != 0) { 523 case 0: 524 break; 525 526 case EPERM: 527 if (!verbose) 528 uu_warn(emsg_permission_denied, fmri); 529 else 530 uu_warn(emsg_prop_perm_denied, fmri, 531 SCF_PG_RESTARTER_ACTIONS, SCF_PROPERTY_AUX_TTY); 532 533 ret = EPERM; 534 goto out; 535 /* NOTREACHED */ 536 537 case EROFS: 538 /* Shouldn't happen, but it can. */ 539 if (!verbose) 540 uu_warn(gettext("%s: Repository read-only.\n"), fmri); 541 else 542 uu_warn(gettext("%s: Could not set %s/%s " 543 "(repository read-only).\n"), fmri, 544 SCF_PG_RESTARTER_ACTIONS, SCF_PROPERTY_AUX_TTY); 545 546 ret = EROFS; 547 goto out; 548 /* NOTREACHED */ 549 550 default: 551 scfdie(); 552 } 553 554 if (my_ct_name(scratch_fmri, max_scf_fmri_sz) > 0) { 555 set_astring_prop(fmri, SCF_PG_RESTARTER_ACTIONS, 556 SCF_PG_RESTARTER_ACTIONS_TYPE, 557 SCF_PG_RESTARTER_ACTIONS_FLAGS, 558 SCF_PROPERTY_AUX_FMRI, scratch_fmri); 559 } else { 560 uu_warn(gettext("%s: Could not set %s/%s: " 561 "my_ct_name failed.\n"), fmri, 562 SCF_PG_RESTARTER_ACTIONS, SCF_PROPERTY_AUX_FMRI); 563 } 564 565 out: 566 scf_pg_destroy(pg); 567 return (ret); 568 } 569 570 /* 571 * Enable or disable inst, per enable. If temp is true, set 572 * general_ovr/enabled. Otherwise set general/enabled and delete 573 * general_ovr/enabled if it exists (order is important here: we don't want the 574 * enabled status to glitch). 575 */ 576 static void 577 set_inst_enabled(const char *fmri, scf_instance_t *inst, boolean_t temp, 578 boolean_t enable) 579 { 580 scf_propertygroup_t *pg; 581 uint8_t b; 582 const char *pgname = NULL; /* For emsg_pg_perm_denied */ 583 int r; 584 585 pg = scf_pg_create(h); 586 if (pg == NULL) 587 scfdie(); 588 589 if (restarter_setup(fmri, inst)) 590 goto out; 591 592 /* 593 * An instance's configuration is incomplete if general/enabled 594 * doesn't exist. Create both the property group and property 595 * here if they don't exist. 596 */ 597 pgname = SCF_PG_GENERAL; 598 if (pg_get_or_add(inst, pgname, SCF_PG_GENERAL_TYPE, 599 SCF_PG_GENERAL_FLAGS, pg) != 0) 600 goto eperm; 601 602 if (get_bool_prop(pg, SCF_PROPERTY_ENABLED, &b) != 0) { 603 /* Create and set state to disabled */ 604 switch (set_bool_prop(pg, SCF_PROPERTY_ENABLED, B_FALSE) != 0) { 605 case 0: 606 break; 607 608 case EPERM: 609 goto eperm; 610 611 case EROFS: 612 /* Shouldn't happen, but it can. */ 613 if (!verbose) 614 uu_warn(gettext("%s: Repository read-only.\n"), 615 fmri); 616 else 617 uu_warn(gettext("%s: Could not set %s/%s " 618 "(repository read-only).\n"), fmri, 619 SCF_PG_GENERAL, SCF_PROPERTY_ENABLED); 620 goto out; 621 622 default: 623 assert(0); 624 abort(); 625 } 626 } 627 628 if (temp) { 629 /* Set general_ovr/enabled */ 630 pgname = SCF_PG_GENERAL_OVR; 631 if (pg_get_or_add(inst, pgname, SCF_PG_GENERAL_OVR_TYPE, 632 SCF_PG_GENERAL_OVR_FLAGS, pg) != 0) 633 goto eperm; 634 635 switch (set_bool_prop(pg, SCF_PROPERTY_ENABLED, enable) != 0) { 636 case 0: 637 break; 638 639 case EPERM: 640 goto eperm; 641 642 case EROFS: 643 /* Shouldn't happen, but it can. */ 644 if (!verbose) 645 uu_warn(gettext("%s: Repository read-only.\n"), 646 fmri); 647 else 648 uu_warn(gettext("%s: Could not set %s/%s " 649 "(repository read-only).\n"), fmri, 650 SCF_PG_GENERAL_OVR, SCF_PROPERTY_ENABLED); 651 goto out; 652 653 default: 654 assert(0); 655 abort(); 656 } 657 658 if (verbose) 659 (void) printf(enable ? 660 gettext("%s temporarily enabled.\n") : 661 gettext("%s temporarily disabled.\n"), fmri); 662 } else { 663 again: 664 /* 665 * Both pg and property should exist since we created 666 * them earlier. However, there's still a chance that 667 * someone may have deleted the property out from under 668 * us. 669 */ 670 if (pg_get_or_add(inst, pgname, SCF_PG_GENERAL_TYPE, 671 SCF_PG_GENERAL_FLAGS, pg) != 0) 672 goto eperm; 673 674 switch (set_bool_prop(pg, SCF_PROPERTY_ENABLED, enable)) { 675 case 0: 676 break; 677 678 case EPERM: 679 goto eperm; 680 681 case EROFS: 682 /* 683 * If general/enabled is already set the way we want, 684 * proceed. 685 */ 686 switch (get_bool_prop(pg, SCF_PROPERTY_ENABLED, &b)) { 687 case 0: 688 if ((b != 0) == (enable != B_FALSE)) 689 break; 690 /* FALLTHROUGH */ 691 692 case ENOENT: 693 case EINVAL: 694 case E2BIG: 695 if (!verbose) 696 uu_warn(gettext("%s: Repository " 697 "read-only.\n"), fmri); 698 else 699 uu_warn(gettext("%s: Could not set " 700 "%s/%s (repository read-only).\n"), 701 fmri, SCF_PG_GENERAL, 702 SCF_PROPERTY_ENABLED); 703 goto out; 704 705 case ECANCELED: 706 goto again; 707 708 default: 709 assert(0); 710 abort(); 711 } 712 break; 713 714 default: 715 assert(0); 716 abort(); 717 } 718 719 pgname = SCF_PG_GENERAL_OVR; 720 r = scf_instance_delete_prop(inst, pgname, 721 SCF_PROPERTY_ENABLED); 722 switch (r) { 723 case 0: 724 break; 725 726 case ECANCELED: 727 uu_warn(emsg_no_service, fmri); 728 goto out; 729 730 case EPERM: 731 goto eperm; 732 733 case EACCES: 734 uu_warn(gettext("Could not delete %s/%s " 735 "property of %s: backend access denied.\n"), 736 pgname, SCF_PROPERTY_ENABLED, fmri); 737 goto out; 738 739 case EROFS: 740 uu_warn(gettext("Could not delete %s/%s " 741 "property of %s: backend is read-only.\n"), 742 pgname, SCF_PROPERTY_ENABLED, fmri); 743 goto out; 744 745 default: 746 bad_error("scf_instance_delete_prop", r); 747 } 748 749 if (verbose) 750 (void) printf(enable ? gettext("%s enabled.\n") : 751 gettext("%s disabled.\n"), fmri); 752 } 753 754 scf_pg_destroy(pg); 755 return; 756 757 eperm: 758 assert(pgname != NULL); 759 if (!verbose) 760 uu_warn(emsg_permission_denied, fmri); 761 else 762 uu_warn(emsg_pg_perm_denied, fmri, pgname); 763 764 out: 765 scf_pg_destroy(pg); 766 exit_status = 1; 767 } 768 769 /* 770 * Set inst to the instance which corresponds to fmri. If fmri identifies 771 * a service with a single instance, get that instance. 772 * 773 * Fails with 774 * ENOTSUP - fmri has an unsupported scheme 775 * EINVAL - fmri is invalid 776 * ENOTDIR - fmri does not identify a service or instance 777 * ENOENT - could not locate instance 778 * E2BIG - fmri is a service with multiple instances (warning not printed) 779 */ 780 static int 781 get_inst_mult(const char *fmri, scf_instance_t *inst) 782 { 783 char *cfmri; 784 const char *svc_name, *inst_name, *pg_name; 785 scf_service_t *svc; 786 scf_instance_t *inst2; 787 scf_iter_t *iter; 788 int ret; 789 790 if (strncmp(fmri, "lrc:", sizeof ("lrc:") - 1) == 0) { 791 uu_warn(gettext("FMRI \"%s\" is a legacy service.\n"), fmri); 792 exit_status = 1; 793 return (ENOTSUP); 794 } 795 796 cfmri = strdup(fmri); 797 if (cfmri == NULL) 798 uu_die(emsg_nomem); 799 800 if (scf_parse_svc_fmri(cfmri, NULL, &svc_name, &inst_name, &pg_name, 801 NULL) != SCF_SUCCESS) { 802 free(cfmri); 803 uu_warn(gettext("FMRI \"%s\" is invalid.\n"), fmri); 804 exit_status = 1; 805 return (EINVAL); 806 } 807 808 free(cfmri); 809 810 if (svc_name == NULL || pg_name != NULL) { 811 uu_warn(gettext( 812 "FMRI \"%s\" does not designate a service or instance.\n"), 813 fmri); 814 exit_status = 1; 815 return (ENOTDIR); 816 } 817 818 if (inst_name != NULL) { 819 if (scf_handle_decode_fmri(h, fmri, NULL, NULL, inst, NULL, 820 NULL, SCF_DECODE_FMRI_EXACT) == 0) 821 return (0); 822 823 if (scf_error() != SCF_ERROR_NOT_FOUND) 824 scfdie(); 825 826 uu_warn(gettext("No such instance \"%s\".\n"), fmri); 827 exit_status = 1; 828 829 return (ENOENT); 830 } 831 832 if ((svc = scf_service_create(h)) == NULL || 833 (inst2 = scf_instance_create(h)) == NULL || 834 (iter = scf_iter_create(h)) == NULL) 835 scfdie(); 836 837 if (scf_handle_decode_fmri(h, fmri, NULL, svc, NULL, NULL, NULL, 838 SCF_DECODE_FMRI_EXACT) != SCF_SUCCESS) { 839 if (scf_error() != SCF_ERROR_NOT_FOUND) 840 scfdie(); 841 842 uu_warn(emsg_no_service, fmri); 843 exit_status = 1; 844 845 ret = ENOENT; 846 goto out; 847 } 848 849 /* If the service has only one child, use it. */ 850 if (scf_iter_service_instances(iter, svc) != SCF_SUCCESS) 851 scfdie(); 852 853 ret = scf_iter_next_instance(iter, inst); 854 if (ret < 0) 855 scfdie(); 856 if (ret != 1) { 857 uu_warn(gettext("Service \"%s\" has no instances.\n"), 858 fmri); 859 exit_status = 1; 860 ret = ENOENT; 861 goto out; 862 } 863 864 ret = scf_iter_next_instance(iter, inst2); 865 if (ret < 0) 866 scfdie(); 867 868 if (ret != 0) { 869 ret = E2BIG; 870 goto out; 871 } 872 873 ret = 0; 874 875 out: 876 scf_iter_destroy(iter); 877 scf_instance_destroy(inst2); 878 scf_service_destroy(svc); 879 return (ret); 880 } 881 882 /* 883 * Same as get_inst_mult(), but on E2BIG prints a warning and returns ENOENT. 884 */ 885 static int 886 get_inst(const char *fmri, scf_instance_t *inst) 887 { 888 int r; 889 890 r = get_inst_mult(fmri, inst); 891 if (r != E2BIG) 892 return (r); 893 894 uu_warn(gettext("operation on service %s is ambiguous; " 895 "instance specification needed.\n"), fmri); 896 return (ENOENT); 897 } 898 899 static char * 900 inst_get_fmri(const scf_instance_t *inst) 901 { 902 ssize_t sz; 903 904 sz = scf_instance_to_fmri(inst, scratch_fmri, max_scf_fmri_sz); 905 if (sz < 0) 906 scfdie(); 907 if (sz >= max_scf_fmri_sz) 908 uu_die(gettext("scf_instance_to_fmri() returned unexpectedly " 909 "long value.\n")); 910 911 return (scratch_fmri); 912 } 913 914 static ssize_t 915 dep_get_astring(const char *fmri, const char *pgname, 916 const scf_propertygroup_t *pg, const char *propname, scf_property_t *prop, 917 scf_value_t *v, char *buf, size_t bufsz) 918 { 919 ssize_t sz; 920 921 sz = get_astring_prop(pg, propname, prop, v, buf, bufsz); 922 if (sz >= 0) 923 return (sz); 924 925 switch (-sz) { 926 case ENOENT: 927 uu_warn(gettext("\"%s\" is misconfigured (\"%s\" dependency " 928 "lacks \"%s\" property.)\n"), fmri, pgname, propname); 929 return (-1); 930 931 case E2BIG: 932 uu_warn(gettext("\"%s\" is misconfigured (\"%s/%s\" property " 933 "is not single-valued.)\n"), fmri, pgname, propname); 934 return (-1); 935 936 case EINVAL: 937 uu_warn(gettext("\"%s\" is misconfigured (\"%s/%s\" property " 938 "is not of astring type.)\n"), fmri, pgname, propname); 939 return (-1); 940 941 default: 942 assert(0); 943 abort(); 944 /* NOTREACHED */ 945 } 946 } 947 948 static boolean_t 949 multiple_instances(scf_iter_t *iter, scf_value_t *v, char *buf) 950 { 951 int count = 0, r; 952 boolean_t ret; 953 scf_instance_t *inst; 954 955 inst = scf_instance_create(h); 956 if (inst == NULL) 957 scfdie(); 958 959 for (;;) { 960 r = scf_iter_next_value(iter, v); 961 if (r == 0) { 962 ret = B_FALSE; 963 goto out; 964 } 965 if (r != 1) 966 scfdie(); 967 968 if (scf_value_get_astring(v, buf, max_scf_fmri_sz) < 0) 969 scfdie(); 970 971 switch (get_inst_mult(buf, inst)) { 972 case 0: 973 ++count; 974 if (count > 1) { 975 ret = B_TRUE; 976 goto out; 977 } 978 break; 979 980 case ENOTSUP: 981 case EINVAL: 982 case ENOTDIR: 983 case ENOENT: 984 continue; 985 986 case E2BIG: 987 ret = B_TRUE; 988 goto out; 989 990 default: 991 assert(0); 992 abort(); 993 } 994 } 995 996 out: 997 scf_instance_destroy(inst); 998 return (ret); 999 } 1000 1001 /* 1002 * Enable the service or instance identified by fmri and its dependencies, 1003 * recursively. Specifically, call get_inst(fmri), enable the result, and 1004 * recurse on its restarter and the dependencies. To avoid duplication of 1005 * effort or looping around a dependency cycle, each FMRI is entered into the 1006 * "visited" hash table. While recursing, the hash table entry is marked 1007 * "active", so that if we come upon it again, we know we've hit a cycle. 1008 * exclude_all and optional_all dependencies are ignored. require_any 1009 * dependencies are followed only if they comprise a single service; otherwise 1010 * the user is warned. 1011 * 1012 * fmri must point to a writable max_scf_fmri_sz buffer. Returns EINVAL if fmri 1013 * is invalid, E2BIG if fmri identifies a service with multiple instances, ELOOP 1014 * on cycle detection, or 0 on success. 1015 */ 1016 static int 1017 enable_fmri_rec(char *fmri, boolean_t temp) 1018 { 1019 scf_instance_t *inst; 1020 scf_snapshot_t *snap; 1021 scf_propertygroup_t *pg; 1022 scf_property_t *prop; 1023 scf_value_t *v; 1024 scf_iter_t *pg_iter, *val_iter; 1025 scf_type_t ty; 1026 char *buf, *pgname; 1027 ssize_t name_sz, len, sz; 1028 int ret; 1029 struct ht_elt *he; 1030 1031 len = scf_canonify_fmri(fmri, fmri, max_scf_fmri_sz); 1032 if (len < 0) { 1033 assert(scf_error() == SCF_ERROR_INVALID_ARGUMENT); 1034 return (EINVAL); 1035 } 1036 assert(len < max_scf_fmri_sz); 1037 1038 switch (visited_find_or_add(fmri, &he)) { 1039 case 0: 1040 he->active = B_TRUE; 1041 break; 1042 1043 case 1: 1044 return (he->active ? ELOOP : 0); 1045 1046 case -1: 1047 uu_die(emsg_nomem); 1048 1049 default: 1050 assert(0); 1051 abort(); 1052 } 1053 1054 inst = scf_instance_create(h); 1055 if (inst == NULL) 1056 scfdie(); 1057 1058 switch (get_inst_mult(fmri, inst)) { 1059 case 0: 1060 break; 1061 1062 case E2BIG: 1063 he->active = B_FALSE; 1064 return (E2BIG); 1065 1066 default: 1067 he->active = B_FALSE; 1068 return (0); 1069 } 1070 1071 set_inst_enabled(fmri, inst, temp, B_TRUE); 1072 1073 if ((snap = scf_snapshot_create(h)) == NULL || 1074 (pg = scf_pg_create(h)) == NULL || 1075 (prop = scf_property_create(h)) == NULL || 1076 (v = scf_value_create(h)) == NULL || 1077 (pg_iter = scf_iter_create(h)) == NULL || 1078 (val_iter = scf_iter_create(h)) == NULL) 1079 scfdie(); 1080 1081 buf = malloc(max_scf_fmri_sz); 1082 if (buf == NULL) 1083 uu_die(emsg_nomem); 1084 1085 name_sz = scf_limit(SCF_LIMIT_MAX_NAME_LENGTH); 1086 if (name_sz < 0) 1087 scfdie(); 1088 ++name_sz; 1089 pgname = malloc(name_sz); 1090 if (pgname == NULL) 1091 uu_die(emsg_nomem); 1092 1093 if (scf_instance_get_snapshot(inst, "running", snap) != 0) { 1094 if (scf_error() != SCF_ERROR_NOT_FOUND) 1095 scfdie(); 1096 1097 scf_snapshot_destroy(snap); 1098 snap = NULL; 1099 } 1100 1101 /* Enable restarter */ 1102 if (scf_instance_get_pg_composed(inst, snap, SCF_PG_GENERAL, pg) != 0) { 1103 if (scf_error() != SCF_ERROR_NOT_FOUND) 1104 scfdie(); 1105 1106 uu_warn(gettext("\"%s\" is misconfigured (lacks \"%s\" " 1107 "property group).\n"), fmri, SCF_PG_GENERAL); 1108 ret = 0; 1109 goto out; 1110 } 1111 1112 sz = get_astring_prop(pg, SCF_PROPERTY_RESTARTER, prop, v, buf, 1113 max_scf_fmri_sz); 1114 if (sz > max_scf_fmri_sz) { 1115 uu_warn(gettext("\"%s\" is misconfigured (the value of " 1116 "\"%s/%s\" is too long).\n"), fmri, SCF_PG_GENERAL, 1117 SCF_PROPERTY_RESTARTER); 1118 ret = 0; 1119 goto out; 1120 } else if (sz >= 0) { 1121 switch (enable_fmri_rec(buf, temp)) { 1122 case 0: 1123 break; 1124 1125 case EINVAL: 1126 uu_warn(gettext("Restarter FMRI for \"%s\" is " 1127 "invalid.\n"), fmri); 1128 break; 1129 1130 case E2BIG: 1131 uu_warn(gettext("Restarter FMRI for \"%s\" identifies " 1132 "a service with multiple instances.\n"), fmri); 1133 break; 1134 1135 case ELOOP: 1136 ret = ELOOP; 1137 goto out; 1138 1139 default: 1140 assert(0); 1141 abort(); 1142 } 1143 } else if (sz < 0) { 1144 switch (-sz) { 1145 case ENOENT: 1146 break; 1147 1148 case E2BIG: 1149 uu_warn(gettext("\"%s\" is misconfigured (\"%s/%s\" " 1150 "property is not single-valued).\n"), fmri, 1151 SCF_PG_GENERAL, SCF_PROPERTY_RESTARTER); 1152 ret = 0; 1153 goto out; 1154 1155 case EINVAL: 1156 uu_warn(gettext("\"%s\" is misconfigured (\"%s/%s\" " 1157 "property is not of astring type).\n"), fmri, 1158 SCF_PG_GENERAL, SCF_PROPERTY_RESTARTER); 1159 ret = 0; 1160 goto out; 1161 1162 default: 1163 assert(0); 1164 abort(); 1165 } 1166 } 1167 1168 if (scf_iter_instance_pgs_typed_composed(pg_iter, inst, snap, 1169 SCF_GROUP_DEPENDENCY) == -1) 1170 scfdie(); 1171 1172 while (scf_iter_next_pg(pg_iter, pg) > 0) { 1173 len = scf_pg_get_name(pg, pgname, name_sz); 1174 if (len < 0) 1175 scfdie(); 1176 assert(len < name_sz); 1177 1178 if (dep_get_astring(fmri, pgname, pg, SCF_PROPERTY_TYPE, prop, 1179 v, buf, max_scf_fmri_sz) < 0) 1180 continue; 1181 1182 if (strcmp(buf, "service") != 0) 1183 continue; 1184 1185 if (dep_get_astring(fmri, pgname, pg, SCF_PROPERTY_GROUPING, 1186 prop, v, buf, max_scf_fmri_sz) < 0) 1187 continue; 1188 1189 if (strcmp(buf, SCF_DEP_EXCLUDE_ALL) == 0 || 1190 strcmp(buf, SCF_DEP_OPTIONAL_ALL) == 0) 1191 continue; 1192 1193 if (strcmp(buf, SCF_DEP_REQUIRE_ALL) != 0 && 1194 strcmp(buf, SCF_DEP_REQUIRE_ANY) != 0) { 1195 uu_warn(gettext("Dependency \"%s\" of \"%s\" has " 1196 "unknown type \"%s\".\n"), pgname, fmri, buf); 1197 continue; 1198 } 1199 1200 if (scf_pg_get_property(pg, SCF_PROPERTY_ENTITIES, prop) == 1201 -1) { 1202 if (scf_error() != SCF_ERROR_NOT_FOUND) 1203 scfdie(); 1204 1205 uu_warn(gettext("\"%s\" is misconfigured (\"%s\" " 1206 "dependency lacks \"%s\" property.)\n"), fmri, 1207 pgname, SCF_PROPERTY_ENTITIES); 1208 continue; 1209 } 1210 1211 if (scf_property_type(prop, &ty) != SCF_SUCCESS) 1212 scfdie(); 1213 1214 if (ty != SCF_TYPE_FMRI) { 1215 uu_warn(gettext("\"%s\" is misconfigured (property " 1216 "\"%s/%s\" is not of fmri type).\n"), fmri, pgname, 1217 SCF_PROPERTY_ENTITIES); 1218 continue; 1219 } 1220 1221 if (scf_iter_property_values(val_iter, prop) == -1) 1222 scfdie(); 1223 1224 if (strcmp(buf, SCF_DEP_REQUIRE_ANY) == 0) { 1225 if (multiple_instances(val_iter, v, buf)) { 1226 (void) printf(gettext("%s requires one of:\n"), 1227 fmri); 1228 1229 if (scf_iter_property_values(val_iter, prop) != 1230 0) 1231 scfdie(); 1232 1233 for (;;) { 1234 int r; 1235 1236 r = scf_iter_next_value(val_iter, v); 1237 if (r == 0) 1238 break; 1239 if (r != 1) 1240 scfdie(); 1241 1242 if (scf_value_get_astring(v, buf, 1243 max_scf_fmri_sz) < 0) 1244 scfdie(); 1245 1246 (void) fputs(" ", stdout); 1247 (void) puts(buf); 1248 } 1249 1250 continue; 1251 } 1252 1253 /* 1254 * Since there's only one instance, we can enable it. 1255 * Reset val_iter and continue. 1256 */ 1257 if (scf_iter_property_values(val_iter, prop) != 0) 1258 scfdie(); 1259 } 1260 1261 for (;;) { 1262 ret = scf_iter_next_value(val_iter, v); 1263 if (ret == 0) 1264 break; 1265 if (ret != 1) 1266 scfdie(); 1267 1268 if (scf_value_get_astring(v, buf, max_scf_fmri_sz) == 1269 -1) 1270 scfdie(); 1271 1272 switch (enable_fmri_rec(buf, temp)) { 1273 case 0: 1274 break; 1275 1276 case EINVAL: 1277 uu_warn(gettext("\"%s\" dependency of \"%s\" " 1278 "has invalid FMRI \"%s\".\n"), pgname, 1279 fmri, buf); 1280 break; 1281 1282 case E2BIG: 1283 uu_warn(gettext("%s depends on %s, which has " 1284 "multiple instances.\n"), fmri, buf); 1285 break; 1286 1287 case ELOOP: 1288 ret = ELOOP; 1289 goto out; 1290 1291 default: 1292 assert(0); 1293 abort(); 1294 } 1295 } 1296 } 1297 1298 ret = 0; 1299 1300 out: 1301 he->active = B_FALSE; 1302 1303 free(buf); 1304 free(pgname); 1305 1306 (void) scf_value_destroy(v); 1307 scf_property_destroy(prop); 1308 scf_pg_destroy(pg); 1309 scf_snapshot_destroy(snap); 1310 scf_iter_destroy(pg_iter); 1311 scf_iter_destroy(val_iter); 1312 1313 return (ret); 1314 } 1315 1316 /* 1317 * fmri here is only used for verbose messages. 1318 */ 1319 static void 1320 set_inst_action(const char *fmri, const scf_instance_t *inst, 1321 const char *action) 1322 { 1323 scf_transaction_t *tx; 1324 scf_transaction_entry_t *ent; 1325 scf_propertygroup_t *pg; 1326 scf_property_t *prop; 1327 scf_value_t *v; 1328 int ret; 1329 int64_t t; 1330 hrtime_t timestamp; 1331 1332 const char * const scf_pg_restarter_actions = SCF_PG_RESTARTER_ACTIONS; 1333 1334 if ((pg = scf_pg_create(h)) == NULL || 1335 (prop = scf_property_create(h)) == NULL || 1336 (v = scf_value_create(h)) == NULL || 1337 (tx = scf_transaction_create(h)) == NULL || 1338 (ent = scf_entry_create(h)) == NULL) 1339 scfdie(); 1340 1341 if (restarter_setup(fmri, inst)) { 1342 exit_status = 1; 1343 goto out; 1344 } 1345 1346 if (scf_instance_get_pg(inst, scf_pg_restarter_actions, pg) == -1) { 1347 if (scf_error() != SCF_ERROR_NOT_FOUND) 1348 scfdie(); 1349 1350 /* Try creating the restarter_actions property group. */ 1351 if (scf_instance_add_pg(inst, scf_pg_restarter_actions, 1352 SCF_PG_RESTARTER_ACTIONS_TYPE, 1353 SCF_PG_RESTARTER_ACTIONS_FLAGS, pg) == -1) { 1354 switch (scf_error()) { 1355 case SCF_ERROR_EXISTS: 1356 /* Someone must have added it. */ 1357 break; 1358 1359 case SCF_ERROR_PERMISSION_DENIED: 1360 if (!verbose) 1361 uu_warn(emsg_permission_denied, fmri); 1362 else 1363 uu_warn(emsg_create_pg_perm_denied, 1364 fmri, scf_pg_restarter_actions); 1365 goto out; 1366 1367 default: 1368 scfdie(); 1369 } 1370 } 1371 } 1372 1373 /* 1374 * If we lose the transaction race and need to retry, there are 2 1375 * potential other winners: 1376 * - another process setting actions 1377 * - the restarter marking the action complete 1378 * Therefore, re-read the property every time through the loop before 1379 * making any decisions based on their values. 1380 */ 1381 do { 1382 timestamp = gethrtime(); 1383 1384 if (scf_transaction_start(tx, pg) == -1) { 1385 if (scf_error() != SCF_ERROR_PERMISSION_DENIED) 1386 scfdie(); 1387 1388 if (!verbose) 1389 uu_warn(emsg_permission_denied, fmri); 1390 else 1391 uu_warn(emsg_pg_perm_denied, fmri, 1392 scf_pg_restarter_actions); 1393 goto out; 1394 } 1395 1396 if (scf_pg_get_property(pg, action, prop) == -1) { 1397 if (scf_error() != SCF_ERROR_NOT_FOUND) 1398 scfdie(); 1399 if (scf_transaction_property_new(tx, ent, 1400 action, SCF_TYPE_INTEGER) == -1) 1401 scfdie(); 1402 goto action_set; 1403 } else { 1404 if (scf_transaction_property_change_type(tx, ent, 1405 action, SCF_TYPE_INTEGER) == -1) 1406 scfdie(); 1407 } 1408 1409 if (scf_property_get_value(prop, v) == -1) { 1410 switch (scf_error()) { 1411 case SCF_ERROR_CONSTRAINT_VIOLATED: 1412 case SCF_ERROR_NOT_FOUND: 1413 /* Misconfigured, so set anyway. */ 1414 goto action_set; 1415 1416 default: 1417 scfdie(); 1418 } 1419 } else { 1420 if (scf_value_get_integer(v, &t) == -1) { 1421 assert(scf_error() == SCF_ERROR_TYPE_MISMATCH); 1422 goto action_set; 1423 } 1424 if (t > timestamp) 1425 break; 1426 } 1427 1428 action_set: 1429 scf_value_set_integer(v, timestamp); 1430 if (scf_entry_add_value(ent, v) == -1) 1431 scfdie(); 1432 1433 ret = scf_transaction_commit(tx); 1434 if (ret == -1) { 1435 if (scf_error() != SCF_ERROR_PERMISSION_DENIED) 1436 scfdie(); 1437 1438 if (!verbose) 1439 uu_warn(emsg_permission_denied, fmri); 1440 else 1441 uu_warn(emsg_prop_perm_denied, fmri, 1442 scf_pg_restarter_actions, action); 1443 scf_transaction_reset(tx); 1444 goto out; 1445 } 1446 1447 scf_transaction_reset(tx); 1448 1449 if (ret == 0) { 1450 if (scf_pg_update(pg) == -1) 1451 scfdie(); 1452 } 1453 } while (ret == 0); 1454 1455 if (verbose) 1456 (void) printf(gettext("Action %s set for %s.\n"), action, fmri); 1457 1458 out: 1459 scf_value_destroy(v); 1460 scf_entry_destroy(ent); 1461 scf_transaction_destroy(tx); 1462 scf_property_destroy(prop); 1463 scf_pg_destroy(pg); 1464 } 1465 1466 /* 1467 * Get the state of inst. state should point to a buffer of 1468 * MAX_SCF_STATE_STRING_SZ bytes. Returns 0 on success or -1 if 1469 * no restarter property group 1470 * no state property 1471 * state property is misconfigured (wrong type, not single-valued) 1472 * state value is too long 1473 * In these cases, fmri is used to print a warning. 1474 * 1475 * If pgp is non-NULL, a successful call to inst_get_state will store 1476 * the SCF_PG_RESTARTER property group in *pgp, and the caller will be 1477 * responsible for calling scf_pg_destroy on the property group. 1478 */ 1479 int 1480 inst_get_state(scf_instance_t *inst, char *state, const char *fmri, 1481 scf_propertygroup_t **pgp) 1482 { 1483 scf_propertygroup_t *pg; 1484 scf_property_t *prop; 1485 scf_value_t *val; 1486 int ret = -1; 1487 ssize_t szret; 1488 1489 if ((pg = scf_pg_create(h)) == NULL || 1490 (prop = scf_property_create(h)) == NULL || 1491 (val = scf_value_create(h)) == NULL) 1492 scfdie(); 1493 1494 if (scf_instance_get_pg(inst, SCF_PG_RESTARTER, pg) != SCF_SUCCESS) { 1495 if (scf_error() != SCF_ERROR_NOT_FOUND) 1496 scfdie(); 1497 1498 uu_warn(gettext("%s is misconfigured (lacks \"%s\" property " 1499 "group).\n"), fmri ? fmri : inst_get_fmri(inst), 1500 SCF_PG_RESTARTER); 1501 goto out; 1502 } 1503 1504 szret = get_astring_prop(pg, SCF_PROPERTY_STATE, prop, val, state, 1505 MAX_SCF_STATE_STRING_SZ); 1506 if (szret < 0) { 1507 switch (-szret) { 1508 case ENOENT: 1509 uu_warn(gettext("%s is misconfigured (\"%s\" property " 1510 "group lacks \"%s\" property).\n"), 1511 fmri ? fmri : inst_get_fmri(inst), SCF_PG_RESTARTER, 1512 SCF_PROPERTY_STATE); 1513 goto out; 1514 1515 case E2BIG: 1516 uu_warn(gettext("%s is misconfigured (\"%s/%s\" " 1517 "property is not single-valued).\n"), 1518 fmri ? fmri : inst_get_fmri(inst), SCF_PG_RESTARTER, 1519 SCF_PROPERTY_STATE); 1520 goto out; 1521 1522 case EINVAL: 1523 uu_warn(gettext("%s is misconfigured (\"%s/%s\" " 1524 "property is not of type astring).\n"), 1525 fmri ? fmri : inst_get_fmri(inst), SCF_PG_RESTARTER, 1526 SCF_PROPERTY_STATE); 1527 goto out; 1528 1529 default: 1530 assert(0); 1531 abort(); 1532 } 1533 } 1534 if (szret >= MAX_SCF_STATE_STRING_SZ) { 1535 uu_warn(gettext("%s is misconfigured (\"%s/%s\" property value " 1536 "is too long).\n"), fmri ? fmri : inst_get_fmri(inst), 1537 SCF_PG_RESTARTER, SCF_PROPERTY_STATE); 1538 goto out; 1539 } 1540 1541 ret = 0; 1542 if (pgp) 1543 *pgp = pg; 1544 1545 out: 1546 (void) scf_value_destroy(val); 1547 scf_property_destroy(prop); 1548 if (ret || pgp == NULL) 1549 scf_pg_destroy(pg); 1550 return (ret); 1551 } 1552 1553 static void 1554 set_astring_prop(const char *fmri, const char *pgname, const char *pgtype, 1555 uint32_t pgflags, const char *propname, const char *str) 1556 { 1557 scf_instance_t *inst; 1558 scf_propertygroup_t *pg; 1559 scf_property_t *prop; 1560 scf_value_t *val; 1561 scf_transaction_t *tx; 1562 scf_transaction_entry_t *txent; 1563 int ret; 1564 1565 inst = scf_instance_create(h); 1566 if (inst == NULL) 1567 scfdie(); 1568 1569 if (get_inst(fmri, inst) != 0) 1570 return; 1571 1572 if ((pg = scf_pg_create(h)) == NULL || 1573 (prop = scf_property_create(h)) == NULL || 1574 (val = scf_value_create(h)) == NULL || 1575 (tx = scf_transaction_create(h)) == NULL || 1576 (txent = scf_entry_create(h)) == NULL) 1577 scfdie(); 1578 1579 if (scf_instance_get_pg(inst, pgname, pg) != SCF_SUCCESS) { 1580 if (scf_error() != SCF_ERROR_NOT_FOUND) 1581 scfdie(); 1582 1583 if (scf_instance_add_pg(inst, pgname, pgtype, pgflags, pg) != 1584 SCF_SUCCESS) { 1585 switch (scf_error()) { 1586 case SCF_ERROR_EXISTS: 1587 if (scf_instance_get_pg(inst, pgname, pg) != 1588 SCF_SUCCESS) { 1589 if (scf_error() != SCF_ERROR_NOT_FOUND) 1590 scfdie(); 1591 1592 uu_warn(gettext("Repository write " 1593 "contention.\n")); 1594 goto out; 1595 } 1596 break; 1597 1598 case SCF_ERROR_PERMISSION_DENIED: 1599 if (!verbose) 1600 uu_warn(emsg_permission_denied, fmri); 1601 else 1602 uu_warn(emsg_create_pg_perm_denied, 1603 fmri, pgname); 1604 goto out; 1605 1606 default: 1607 scfdie(); 1608 } 1609 } 1610 } 1611 1612 do { 1613 if (scf_transaction_start(tx, pg) != SCF_SUCCESS) { 1614 if (scf_error() != SCF_ERROR_PERMISSION_DENIED) 1615 scfdie(); 1616 1617 if (!verbose) 1618 uu_warn(emsg_permission_denied, fmri); 1619 else 1620 uu_warn(emsg_pg_perm_denied, fmri, pgname); 1621 goto out; 1622 } 1623 1624 if (scf_transaction_property_change_type(tx, txent, propname, 1625 SCF_TYPE_ASTRING) != 0) { 1626 if (scf_error() != SCF_ERROR_NOT_FOUND) 1627 scfdie(); 1628 1629 if (scf_transaction_property_new(tx, txent, propname, 1630 SCF_TYPE_ASTRING) != 0) 1631 scfdie(); 1632 } 1633 1634 if (scf_value_set_astring(val, str) != SCF_SUCCESS) 1635 scfdie(); 1636 1637 if (scf_entry_add_value(txent, val) != SCF_SUCCESS) 1638 scfdie(); 1639 1640 ret = scf_transaction_commit(tx); 1641 if (ret == -1) { 1642 if (scf_error() != SCF_ERROR_PERMISSION_DENIED) 1643 scfdie(); 1644 1645 if (!verbose) 1646 uu_warn(emsg_permission_denied, fmri); 1647 else 1648 uu_warn(emsg_prop_perm_denied, fmri, pgname, 1649 propname); 1650 goto out; 1651 } 1652 1653 if (ret == 0) { 1654 scf_transaction_reset(tx); 1655 1656 if (scf_pg_update(pg) == -1) 1657 scfdie(); 1658 } 1659 } while (ret == 0); 1660 1661 out: 1662 scf_transaction_destroy(tx); 1663 scf_entry_destroy(txent); 1664 scf_value_destroy(val); 1665 scf_property_destroy(prop); 1666 scf_pg_destroy(pg); 1667 scf_instance_destroy(inst); 1668 } 1669 1670 1671 /* 1672 * Flags to control enable and disable actions. 1673 */ 1674 #define SET_ENABLED 0x1 1675 #define SET_TEMPORARY 0x2 1676 #define SET_RECURSIVE 0x4 1677 1678 static int 1679 set_fmri_enabled(void *data, scf_walkinfo_t *wip) 1680 { 1681 int flags = (int)data; 1682 1683 assert(wip->inst != NULL); 1684 assert(wip->pg == NULL); 1685 1686 if (svcsearch) { 1687 char state[MAX_SCF_STATE_STRING_SZ]; 1688 1689 if (inst_get_state(wip->inst, state, wip->fmri, NULL) != 0) 1690 return (0); 1691 if (strcmp(state, svcstate) != 0) 1692 return (0); 1693 } 1694 1695 if (flags & SET_RECURSIVE) { 1696 char *fmri_buf = malloc(max_scf_fmri_sz); 1697 if (fmri_buf == NULL) 1698 uu_die(emsg_nomem); 1699 1700 visited = calloc(HT_BUCKETS, sizeof (*visited)); 1701 if (visited == NULL) 1702 uu_die(emsg_nomem); 1703 1704 /* scf_walk_fmri() guarantees that fmri isn't too long */ 1705 assert(strlen(wip->fmri) <= max_scf_fmri_sz); 1706 (void) strlcpy(fmri_buf, wip->fmri, max_scf_fmri_sz); 1707 1708 switch (enable_fmri_rec(fmri_buf, (flags & SET_TEMPORARY))) { 1709 case E2BIG: 1710 uu_warn(gettext("operation on service %s is ambiguous; " 1711 "instance specification needed.\n"), fmri_buf); 1712 break; 1713 1714 case ELOOP: 1715 uu_warn(gettext("%s: Dependency cycle detected.\n"), 1716 fmri_buf); 1717 } 1718 1719 free(visited); 1720 free(fmri_buf); 1721 1722 } else { 1723 set_inst_enabled(wip->fmri, wip->inst, 1724 (flags & SET_TEMPORARY) != 0, (flags & SET_ENABLED) != 0); 1725 } 1726 1727 return (0); 1728 } 1729 1730 /* ARGSUSED */ 1731 static int 1732 wait_fmri_enabled(void *data, scf_walkinfo_t *wip) 1733 { 1734 scf_propertygroup_t *pg = NULL; 1735 char state[MAX_SCF_STATE_STRING_SZ]; 1736 1737 assert(wip->inst != NULL); 1738 assert(wip->pg == NULL); 1739 1740 do { 1741 if (pg) 1742 scf_pg_destroy(pg); 1743 if (inst_get_state(wip->inst, state, wip->fmri, &pg) != 0) { 1744 exit_status = EXIT_SVC_FAILURE; 1745 return (0); 1746 } 1747 1748 if (strcmp(state, SCF_STATE_STRING_ONLINE) == 0 || 1749 strcmp(state, SCF_STATE_STRING_DEGRADED) == 0) { 1750 /* 1751 * We're done. 1752 */ 1753 goto out; 1754 } 1755 1756 if (strcmp(state, SCF_STATE_STRING_MAINT) == 0) { 1757 /* 1758 * The service is ill. 1759 */ 1760 uu_warn(gettext("Instance \"%s\" is in maintenance" 1761 " state.\n"), wip->fmri); 1762 exit_status = EXIT_SVC_FAILURE; 1763 goto out; 1764 } 1765 1766 if (!is_enabled(wip->inst)) { 1767 /* 1768 * Someone stepped in and disabled the service. 1769 */ 1770 uu_warn(gettext("Instance \"%s\" has been disabled" 1771 " by another entity.\n"), wip->fmri); 1772 exit_status = EXIT_SVC_FAILURE; 1773 goto out; 1774 } 1775 1776 if (!has_potential(wip->inst, B_FALSE)) { 1777 /* 1778 * Our dependencies aren't met. We'll never 1779 * amount to anything. 1780 */ 1781 uu_warn(gettext("Instance \"%s\" has unsatisfied" 1782 " dependencies.\n"), wip->fmri); 1783 /* 1784 * EXIT_SVC_FAILURE takes precedence over 1785 * EXIT_DEP_FAILURE 1786 */ 1787 if (exit_status == 0) 1788 exit_status = EXIT_DEP_FAILURE; 1789 goto out; 1790 } 1791 } while (_scf_pg_wait(pg, WAIT_INTERVAL) >= 0); 1792 scfdie(); 1793 /* NOTREACHED */ 1794 1795 out: 1796 scf_pg_destroy(pg); 1797 return (0); 1798 } 1799 1800 /* ARGSUSED */ 1801 static int 1802 wait_fmri_disabled(void *data, scf_walkinfo_t *wip) 1803 { 1804 scf_propertygroup_t *pg = NULL; 1805 char state[MAX_SCF_STATE_STRING_SZ]; 1806 1807 assert(wip->inst != NULL); 1808 assert(wip->pg == NULL); 1809 1810 do { 1811 if (pg) 1812 scf_pg_destroy(pg); 1813 if (inst_get_state(wip->inst, state, wip->fmri, &pg) != 0) { 1814 exit_status = EXIT_SVC_FAILURE; 1815 return (0); 1816 } 1817 1818 if (strcmp(state, SCF_STATE_STRING_DISABLED) == 0) { 1819 /* 1820 * We're done. 1821 */ 1822 goto out; 1823 } 1824 1825 if (is_enabled(wip->inst)) { 1826 /* 1827 * Someone stepped in and enabled the service. 1828 */ 1829 uu_warn(gettext("Instance \"%s\" has been enabled" 1830 " by another entity.\n"), wip->fmri); 1831 exit_status = EXIT_SVC_FAILURE; 1832 goto out; 1833 } 1834 1835 if (!has_potential(wip->inst, B_TRUE)) { 1836 /* 1837 * Our restarter is hopeless. 1838 */ 1839 uu_warn(gettext("Restarter for instance \"%s\" is" 1840 " unavailable.\n"), wip->fmri); 1841 /* 1842 * EXIT_SVC_FAILURE takes precedence over 1843 * EXIT_DEP_FAILURE 1844 */ 1845 if (exit_status == 0) 1846 exit_status = EXIT_DEP_FAILURE; 1847 goto out; 1848 } 1849 1850 } while (_scf_pg_wait(pg, WAIT_INTERVAL) >= 0); 1851 scfdie(); 1852 /* NOTREACHED */ 1853 1854 out: 1855 scf_pg_destroy(pg); 1856 return (0); 1857 } 1858 1859 /* ARGSUSED */ 1860 static int 1861 clear_instance(void *data, scf_walkinfo_t *wip) 1862 { 1863 char state[MAX_SCF_STATE_STRING_SZ]; 1864 1865 assert(wip->inst != NULL); 1866 assert(wip->pg == NULL); 1867 1868 if (inst_get_state(wip->inst, state, wip->fmri, NULL) != 0) 1869 return (0); 1870 1871 if (svcsearch && strcmp(state, svcstate) != 0) 1872 return (0); 1873 1874 if (strcmp(state, SCF_STATE_STRING_MAINT) == 0) { 1875 set_inst_action(wip->fmri, wip->inst, SCF_PROPERTY_MAINT_OFF); 1876 } else if (strcmp(state, SCF_STATE_STRING_DEGRADED) == 1877 0) { 1878 set_inst_action(wip->fmri, wip->inst, SCF_PROPERTY_RESTORE); 1879 } else { 1880 uu_warn(gettext("Instance \"%s\" is not in a " 1881 "maintenance or degraded state.\n"), wip->fmri); 1882 1883 exit_status = 1; 1884 } 1885 1886 return (0); 1887 } 1888 1889 static int 1890 set_fmri_action(void *action, scf_walkinfo_t *wip) 1891 { 1892 assert(wip->inst != NULL && wip->pg == NULL); 1893 1894 if (svcsearch) { 1895 char state[MAX_SCF_STATE_STRING_SZ]; 1896 1897 if (inst_get_state(wip->inst, state, wip->fmri, NULL) != 0) 1898 return (0); 1899 if (strcmp(state, svcstate) != 0) 1900 return (0); 1901 } 1902 1903 set_inst_action(wip->fmri, wip->inst, action); 1904 1905 return (0); 1906 } 1907 1908 /* 1909 * Flags to control 'mark' action. 1910 */ 1911 #define MARK_IMMEDIATE 0x1 1912 #define MARK_TEMPORARY 0x2 1913 1914 static int 1915 force_degraded(void *data, scf_walkinfo_t *wip) 1916 { 1917 int flags = (int)data; 1918 char state[MAX_SCF_STATE_STRING_SZ]; 1919 1920 if (inst_get_state(wip->inst, state, wip->fmri, NULL) != 0) { 1921 exit_status = 1; 1922 return (0); 1923 } 1924 1925 if (svcsearch && strcmp(state, svcstate) != 0) 1926 return (0); 1927 1928 if (strcmp(state, SCF_STATE_STRING_ONLINE) != 0) { 1929 uu_warn(gettext("Instance \"%s\" is not online.\n"), wip->fmri); 1930 exit_status = 1; 1931 return (0); 1932 } 1933 1934 set_inst_action(wip->fmri, wip->inst, (flags & MARK_IMMEDIATE) ? 1935 SCF_PROPERTY_DEGRADE_IMMEDIATE : SCF_PROPERTY_DEGRADED); 1936 1937 return (0); 1938 } 1939 1940 static int 1941 force_maintenance(void *data, scf_walkinfo_t *wip) 1942 { 1943 int flags = (int)data; 1944 const char *prop; 1945 1946 if (svcsearch) { 1947 char state[MAX_SCF_STATE_STRING_SZ]; 1948 1949 if (inst_get_state(wip->inst, state, wip->fmri, NULL) != 0) 1950 return (0); 1951 if (strcmp(state, svcstate) != 0) 1952 return (0); 1953 } 1954 1955 if (flags & MARK_IMMEDIATE) { 1956 prop = (flags & MARK_TEMPORARY) ? 1957 SCF_PROPERTY_MAINT_ON_IMMTEMP : 1958 SCF_PROPERTY_MAINT_ON_IMMEDIATE; 1959 } else { 1960 prop = (flags & MARK_TEMPORARY) ? 1961 SCF_PROPERTY_MAINT_ON_TEMPORARY : 1962 SCF_PROPERTY_MAINT_ON; 1963 } 1964 1965 set_inst_action(wip->fmri, wip->inst, prop); 1966 1967 return (0); 1968 } 1969 1970 static void 1971 set_milestone(const char *fmri, boolean_t temporary) 1972 { 1973 scf_instance_t *inst; 1974 scf_propertygroup_t *pg; 1975 int r; 1976 1977 if (temporary) { 1978 set_astring_prop(SCF_SERVICE_STARTD, SCF_PG_OPTIONS_OVR, 1979 SCF_PG_OPTIONS_OVR_TYPE, SCF_PG_OPTIONS_OVR_FLAGS, 1980 SCF_PROPERTY_MILESTONE, fmri); 1981 return; 1982 } 1983 1984 if ((inst = scf_instance_create(h)) == NULL || 1985 (pg = scf_pg_create(h)) == NULL) 1986 scfdie(); 1987 1988 if (get_inst(SCF_SERVICE_STARTD, inst) != 0) { 1989 scf_instance_destroy(inst); 1990 return; 1991 } 1992 1993 /* 1994 * Set the persistent milestone before deleting the override so we don't 1995 * glitch. 1996 */ 1997 set_astring_prop(SCF_SERVICE_STARTD, SCF_PG_OPTIONS, 1998 SCF_PG_OPTIONS_TYPE, SCF_PG_OPTIONS_FLAGS, SCF_PROPERTY_MILESTONE, 1999 fmri); 2000 2001 r = scf_instance_delete_prop(inst, SCF_PG_OPTIONS_OVR, 2002 SCF_PROPERTY_MILESTONE); 2003 switch (r) { 2004 case 0: 2005 break; 2006 2007 case ECANCELED: 2008 uu_warn(emsg_no_service, fmri); 2009 exit_status = 1; 2010 goto out; 2011 2012 case EPERM: 2013 uu_warn(gettext("Could not delete %s/%s property of " 2014 "%s: permission denied.\n"), SCF_PG_OPTIONS_OVR, 2015 SCF_PROPERTY_MILESTONE, SCF_SERVICE_STARTD); 2016 exit_status = 1; 2017 goto out; 2018 2019 case EACCES: 2020 uu_warn(gettext("Could not delete %s/%s property of " 2021 "%s: access denied.\n"), SCF_PG_OPTIONS_OVR, 2022 SCF_PROPERTY_MILESTONE, SCF_SERVICE_STARTD); 2023 exit_status = 1; 2024 goto out; 2025 2026 case EROFS: 2027 uu_warn(gettext("Could not delete %s/%s property of " 2028 "%s: backend read-only.\n"), SCF_PG_OPTIONS_OVR, 2029 SCF_PROPERTY_MILESTONE, SCF_SERVICE_STARTD); 2030 exit_status = 1; 2031 goto out; 2032 2033 default: 2034 bad_error("scf_instance_delete_prop", r); 2035 } 2036 2037 out: 2038 scf_pg_destroy(pg); 2039 scf_instance_destroy(inst); 2040 } 2041 2042 static char const *milestones[] = { 2043 SCF_MILESTONE_SINGLE_USER, 2044 SCF_MILESTONE_MULTI_USER, 2045 SCF_MILESTONE_MULTI_USER_SERVER, 2046 NULL 2047 }; 2048 2049 static void 2050 usage_milestone(void) 2051 { 2052 const char **ms; 2053 2054 (void) fprintf(stderr, gettext( 2055 "Usage: svcadm milestone [-d] <milestone>\n\n" 2056 "\t-d\tmake the specified milestone the default for system boot\n\n" 2057 "\tMilestones can be specified using an FMRI or abbreviation.\n" 2058 "\tThe major milestones are as follows:\n\n" 2059 "\tall\n" 2060 "\tnone\n")); 2061 2062 for (ms = milestones; *ms != NULL; ms++) 2063 (void) fprintf(stderr, "\t%s\n", *ms); 2064 2065 exit(UU_EXIT_USAGE); 2066 } 2067 2068 static const char * 2069 validate_milestone(const char *milestone) 2070 { 2071 const char **ms; 2072 const char *tmp; 2073 size_t len; 2074 2075 if (strcmp(milestone, "all") == 0) 2076 return (milestone); 2077 2078 if (strcmp(milestone, "none") == 0) 2079 return (milestone); 2080 2081 /* 2082 * Determine if this is a full or partial milestone 2083 */ 2084 for (ms = milestones; *ms != NULL; ms++) { 2085 if ((tmp = strstr(*ms, milestone)) != NULL) { 2086 len = strlen(milestone); 2087 2088 /* 2089 * The beginning of the string must align with the start 2090 * of a milestone fmri, or on the boundary between 2091 * elements. The end of the string must align with the 2092 * end of the milestone, or at the instance boundary. 2093 */ 2094 if ((tmp == *ms || tmp[-1] == '/') && 2095 (tmp[len] == '\0' || tmp[len] == ':')) 2096 return (*ms); 2097 } 2098 } 2099 2100 (void) fprintf(stderr, 2101 gettext("\"%s\" is not a valid major milestone.\n"), milestone); 2102 2103 usage_milestone(); 2104 /* NOTREACHED */ 2105 } 2106 2107 /*PRINTFLIKE1*/ 2108 static void 2109 pr_warn(const char *format, ...) 2110 { 2111 const char *pname = uu_getpname(); 2112 va_list alist; 2113 2114 va_start(alist, format); 2115 2116 if (pname != NULL) 2117 (void) fprintf(stderr, "%s", pname); 2118 2119 if (g_zonename != NULL) 2120 (void) fprintf(stderr, " (%s)", g_zonename); 2121 2122 (void) fprintf(stderr, ": "); 2123 2124 (void) vfprintf(stderr, format, alist); 2125 2126 if (strrchr(format, '\n') == NULL) 2127 (void) fprintf(stderr, ": %s\n", strerror(errno)); 2128 2129 va_end(alist); 2130 } 2131 2132 /*ARGSUSED*/ 2133 static void 2134 quiet(const char *fmt, ...) 2135 { 2136 /* Do nothing */ 2137 } 2138 2139 int 2140 main(int argc, char *argv[]) 2141 { 2142 int o; 2143 int err; 2144 int sw_back; 2145 boolean_t do_zones = B_FALSE; 2146 boolean_t do_a_zone = B_FALSE; 2147 char zonename[ZONENAME_MAX]; 2148 uint_t nzents = 0, zent = 0; 2149 zoneid_t *zids = NULL; 2150 int orig_optind, orig_argc; 2151 char **orig_argv; 2152 2153 (void) setlocale(LC_ALL, ""); 2154 (void) textdomain(TEXT_DOMAIN); 2155 2156 (void) uu_setpname(argv[0]); 2157 2158 if (argc < 2) 2159 usage(); 2160 2161 max_scf_fmri_sz = scf_limit(SCF_LIMIT_MAX_FMRI_LENGTH); 2162 if (max_scf_fmri_sz < 0) 2163 scfdie(); 2164 ++max_scf_fmri_sz; 2165 2166 scratch_fmri = malloc(max_scf_fmri_sz); 2167 if (scratch_fmri == NULL) 2168 uu_die(emsg_nomem); 2169 2170 while ((o = getopt(argc, argv, "S:vZz:")) != -1) { 2171 switch (o) { 2172 case 'S': 2173 (void) strlcpy(svcstate, optarg, sizeof (svcstate)); 2174 svcsearch = B_TRUE; 2175 break; 2176 2177 case 'v': 2178 verbose = 1; 2179 break; 2180 2181 case 'z': 2182 if (getzoneid() != GLOBAL_ZONEID) 2183 uu_die(gettext("svcadm -z may only be used " 2184 "from the global zone\n")); 2185 if (do_zones) 2186 usage(); 2187 2188 (void) strlcpy(zonename, optarg, sizeof (zonename)); 2189 do_a_zone = B_TRUE; 2190 break; 2191 2192 case 'Z': 2193 if (getzoneid() != GLOBAL_ZONEID) 2194 uu_die(gettext("svcadm -Z may only be used " 2195 "from the global zone\n")); 2196 if (do_a_zone) 2197 usage(); 2198 2199 do_zones = B_TRUE; 2200 break; 2201 2202 default: 2203 usage(); 2204 } 2205 } 2206 2207 while (do_zones) { 2208 uint_t found; 2209 2210 if (zone_list(NULL, &nzents) != 0) 2211 uu_die(gettext("could not get number of zones")); 2212 2213 if ((zids = malloc(nzents * sizeof (zoneid_t))) == NULL) { 2214 uu_die(gettext("could not allocate array for " 2215 "%d zone IDs"), nzents); 2216 } 2217 2218 found = nzents; 2219 2220 if (zone_list(zids, &found) != 0) 2221 uu_die(gettext("could not get zone list")); 2222 2223 /* 2224 * If the number of zones has not changed between our calls to 2225 * zone_list(), we're done -- otherwise, we must free our array 2226 * of zone IDs and take another lap. 2227 */ 2228 if (found == nzents) 2229 break; 2230 2231 free(zids); 2232 } 2233 2234 emsg_permission_denied = gettext("%s: Permission denied.\n"); 2235 emsg_nomem = gettext("Out of memory.\n"); 2236 emsg_create_pg_perm_denied = gettext("%s: Couldn't create \"%s\" " 2237 "property group (permission denied).\n"); 2238 emsg_pg_perm_denied = gettext("%s: Couldn't modify \"%s\" property " 2239 "group (permission denied).\n"); 2240 emsg_prop_perm_denied = gettext("%s: Couldn't modify \"%s/%s\" " 2241 "property (permission denied).\n"); 2242 emsg_no_service = gettext("No such service \"%s\".\n"); 2243 2244 orig_optind = optind; 2245 orig_argc = argc; 2246 orig_argv = argv; 2247 2248 again: 2249 h = scf_handle_create(SCF_VERSION); 2250 if (h == NULL) 2251 scfdie(); 2252 2253 if (do_zones) { 2254 zone_status_t status; 2255 2256 if (zone_getattr(zids[zent], ZONE_ATTR_STATUS, &status, 2257 sizeof (status)) < 0 || status != ZONE_IS_RUNNING) { 2258 /* 2259 * If this zone is not running or we cannot 2260 * get its status, we do not want to attempt 2261 * to bind an SCF handle to it, lest we 2262 * accidentally interfere with a zone that 2263 * is not yet running by looking up a door 2264 * to its svc.configd (which could potentially 2265 * block a mount with an EBUSY). 2266 */ 2267 zent++; 2268 goto nextzone; 2269 } 2270 2271 if (getzonenamebyid(zids[zent++], zonename, 2272 sizeof (zonename)) < 0) { 2273 uu_warn(gettext("could not get name for " 2274 "zone %d; ignoring"), zids[zent - 1]); 2275 goto nextzone; 2276 } 2277 2278 g_zonename = zonename; 2279 } 2280 2281 if (do_a_zone || do_zones) { 2282 scf_value_t *zone; 2283 2284 if ((zone = scf_value_create(h)) == NULL) 2285 scfdie(); 2286 2287 if (scf_value_set_astring(zone, zonename) != SCF_SUCCESS) 2288 scfdie(); 2289 2290 if (scf_handle_decorate(h, "zone", zone) != SCF_SUCCESS) { 2291 if (do_a_zone) { 2292 uu_die(gettext("invalid zone '%s'\n"), optarg); 2293 } else { 2294 scf_value_destroy(zone); 2295 goto nextzone; 2296 } 2297 } 2298 2299 scf_value_destroy(zone); 2300 } 2301 2302 if (scf_handle_bind(h) == -1) { 2303 if (do_zones) 2304 goto nextzone; 2305 2306 uu_die(gettext("Couldn't bind to configuration repository: " 2307 "%s.\n"), scf_strerror(scf_error())); 2308 } 2309 2310 optind = orig_optind; 2311 argc = orig_argc; 2312 argv = orig_argv; 2313 2314 if (optind >= argc) 2315 usage(); 2316 2317 if (strcmp(argv[optind], "enable") == 0) { 2318 int flags = SET_ENABLED; 2319 int wait = 0; 2320 int error = 0; 2321 2322 ++optind; 2323 2324 while ((o = getopt(argc, argv, "rst")) != -1) { 2325 if (o == 'r') 2326 flags |= SET_RECURSIVE; 2327 else if (o == 't') 2328 flags |= SET_TEMPORARY; 2329 else if (o == 's') 2330 wait = 1; 2331 else if (o == '?') 2332 usage(); 2333 else { 2334 assert(0); 2335 abort(); 2336 } 2337 } 2338 argc -= optind; 2339 argv += optind; 2340 2341 if (argc == 0 && !svcsearch) 2342 usage(); 2343 2344 if (argc > 0 && svcsearch) 2345 usage(); 2346 2347 /* 2348 * We want to continue with -s processing if we had 2349 * invalid options, but not if an enable failed. We 2350 * squelch output the second time we walk fmris; we saw 2351 * the errors the first time. 2352 */ 2353 if ((err = scf_walk_fmri(h, argc, argv, WALK_FLAGS, 2354 set_fmri_enabled, (void *)flags, &error, pr_warn)) != 0) { 2355 2356 pr_warn(gettext("failed to iterate over " 2357 "instances: %s\n"), scf_strerror(err)); 2358 exit_status = UU_EXIT_FATAL; 2359 2360 } else if (wait && exit_status == 0 && 2361 (err = scf_walk_fmri(h, argc, argv, WALK_FLAGS, 2362 wait_fmri_enabled, (void *)flags, &error, quiet)) != 0) { 2363 2364 pr_warn(gettext("failed to iterate over " 2365 "instances: %s\n"), scf_strerror(err)); 2366 exit_status = UU_EXIT_FATAL; 2367 } 2368 2369 if (error > 0) 2370 exit_status = error; 2371 2372 } else if (strcmp(argv[optind], "disable") == 0) { 2373 int flags = 0; 2374 int wait = 0; 2375 int error = 0; 2376 2377 ++optind; 2378 2379 while ((o = getopt(argc, argv, "st")) != -1) { 2380 if (o == 't') 2381 flags |= SET_TEMPORARY; 2382 else if (o == 's') 2383 wait = 1; 2384 else if (o == '?') 2385 usage(); 2386 else { 2387 assert(0); 2388 abort(); 2389 } 2390 } 2391 argc -= optind; 2392 argv += optind; 2393 2394 if (argc == 0 && !svcsearch) 2395 usage(); 2396 2397 if (argc > 0 && svcsearch) 2398 usage(); 2399 2400 /* 2401 * We want to continue with -s processing if we had 2402 * invalid options, but not if a disable failed. We 2403 * squelch output the second time we walk fmris; we saw 2404 * the errors the first time. 2405 */ 2406 if ((err = scf_walk_fmri(h, argc, argv, WALK_FLAGS, 2407 set_fmri_enabled, (void *)flags, &exit_status, 2408 pr_warn)) != 0) { 2409 2410 pr_warn(gettext("failed to iterate over " 2411 "instances: %s\n"), scf_strerror(err)); 2412 exit_status = UU_EXIT_FATAL; 2413 2414 } else if (wait && exit_status == 0 && 2415 (err = scf_walk_fmri(h, argc, argv, WALK_FLAGS, 2416 wait_fmri_disabled, (void *)flags, &error, quiet)) != 0) { 2417 2418 pr_warn(gettext("failed to iterate over " 2419 "instances: %s\n"), scf_strerror(err)); 2420 exit_status = UU_EXIT_FATAL; 2421 } 2422 2423 if (error > 0) 2424 exit_status = error; 2425 2426 } else if (strcmp(argv[optind], "restart") == 0) { 2427 boolean_t do_dump = B_FALSE; 2428 2429 ++optind; 2430 2431 while ((o = getopt(argc, argv, "d")) != -1) { 2432 if (o == 'd') 2433 do_dump = B_TRUE; 2434 else if (o == '?') 2435 usage(); 2436 else { 2437 assert(0); 2438 abort(); 2439 } 2440 } 2441 argc -= optind; 2442 argv += optind; 2443 2444 if (argc == 0 && !svcsearch) 2445 usage(); 2446 2447 if (argc > 0 && svcsearch) 2448 usage(); 2449 2450 if (do_dump) { 2451 if ((err = scf_walk_fmri(h, argc, argv, WALK_FLAGS, 2452 set_fmri_action, (void *)SCF_PROPERTY_DODUMP, 2453 &exit_status, pr_warn)) != 0) { 2454 pr_warn(gettext("failed to iterate over " 2455 "instances: %s\n"), scf_strerror(err)); 2456 exit_status = UU_EXIT_FATAL; 2457 } 2458 } 2459 2460 if ((err = scf_walk_fmri(h, argc, argv, WALK_FLAGS, 2461 set_fmri_action, (void *)SCF_PROPERTY_RESTART, &exit_status, 2462 pr_warn)) != 0) { 2463 pr_warn(gettext("failed to iterate over " 2464 "instances: %s\n"), scf_strerror(err)); 2465 exit_status = UU_EXIT_FATAL; 2466 } 2467 2468 } else if (strcmp(argv[optind], "refresh") == 0) { 2469 ++optind; 2470 argc -= optind; 2471 argv += optind; 2472 2473 if (argc == 0 && !svcsearch) 2474 usage(); 2475 2476 if (argc > 0 && svcsearch) 2477 usage(); 2478 2479 if ((err = scf_walk_fmri(h, argc, argv, WALK_FLAGS, 2480 set_fmri_action, (void *)SCF_PROPERTY_REFRESH, &exit_status, 2481 pr_warn)) != 0) { 2482 pr_warn(gettext("failed to iterate over " 2483 "instances: %s\n"), scf_strerror(scf_error())); 2484 exit_status = UU_EXIT_FATAL; 2485 } 2486 2487 } else if (strcmp(argv[optind], "mark") == 0) { 2488 int flags = 0; 2489 scf_walk_callback callback; 2490 2491 ++optind; 2492 2493 while ((o = getopt(argc, argv, "It")) != -1) { 2494 if (o == 'I') 2495 flags |= MARK_IMMEDIATE; 2496 else if (o == 't') 2497 flags |= MARK_TEMPORARY; 2498 else if (o == '?') 2499 usage(); 2500 else { 2501 assert(0); 2502 abort(); 2503 } 2504 } 2505 2506 if (argc - optind < 2) 2507 usage(); 2508 2509 if (strcmp(argv[optind], "degraded") == 0) { 2510 if (flags & MARK_TEMPORARY) 2511 uu_xdie(UU_EXIT_USAGE, gettext("-t may not be " 2512 "used with degraded.\n")); 2513 callback = force_degraded; 2514 2515 } else if (strcmp(argv[optind], "maintenance") == 0) { 2516 callback = force_maintenance; 2517 } else { 2518 usage(); 2519 } 2520 2521 optind++; 2522 argc -= optind; 2523 argv += optind; 2524 2525 if (argc == 0 && !svcsearch) 2526 usage(); 2527 2528 if (argc > 0 && svcsearch) 2529 usage(); 2530 2531 if ((err = scf_walk_fmri(h, argc, argv, WALK_FLAGS, callback, 2532 NULL, &exit_status, pr_warn)) != 0) { 2533 pr_warn(gettext("failed to iterate over " 2534 "instances: %s\n"), 2535 scf_strerror(err)); 2536 exit_status = UU_EXIT_FATAL; 2537 } 2538 2539 } else if (strcmp(argv[optind], "clear") == 0) { 2540 ++optind; 2541 argc -= optind; 2542 argv += optind; 2543 2544 if (argc == 0 && !svcsearch) 2545 usage(); 2546 2547 if (svcsearch) { 2548 if (argc > 0) 2549 usage(); 2550 if (strcmp(svcstate, SCF_STATE_STRING_MAINT) != 0 && 2551 strcmp(svcstate, SCF_STATE_STRING_DEGRADED) != 0) 2552 uu_die(gettext("State must be '%s' or '%s'\n"), 2553 SCF_STATE_STRING_MAINT, 2554 SCF_STATE_STRING_DEGRADED); 2555 } 2556 2557 if ((err = scf_walk_fmri(h, argc, argv, WALK_FLAGS, 2558 clear_instance, NULL, &exit_status, pr_warn)) != 0) { 2559 pr_warn(gettext("failed to iterate over " 2560 "instances: %s\n"), scf_strerror(err)); 2561 exit_status = UU_EXIT_FATAL; 2562 } 2563 2564 } else if (strcmp(argv[optind], "milestone") == 0) { 2565 boolean_t temporary = B_TRUE; 2566 const char *milestone; 2567 2568 ++optind; 2569 2570 while ((o = getopt(argc, argv, "d")) != -1) { 2571 if (o == 'd') 2572 temporary = B_FALSE; 2573 else if (o == '?') 2574 usage_milestone(); 2575 else { 2576 assert(0); 2577 abort(); 2578 } 2579 } 2580 2581 if (optind >= argc) 2582 usage_milestone(); 2583 2584 milestone = validate_milestone(argv[optind]); 2585 2586 set_milestone(milestone, temporary); 2587 } else if (strcmp(argv[optind], "_smf_backup") == 0) { 2588 const char *reason = NULL; 2589 2590 ++optind; 2591 2592 if (optind != argc - 1) 2593 usage(); 2594 2595 if ((err = _scf_request_backup(h, argv[optind])) != 2596 SCF_SUCCESS) { 2597 switch (scf_error()) { 2598 case SCF_ERROR_NOT_BOUND: 2599 case SCF_ERROR_CONNECTION_BROKEN: 2600 case SCF_ERROR_BACKEND_READONLY: 2601 scfdie(); 2602 break; 2603 2604 case SCF_ERROR_PERMISSION_DENIED: 2605 case SCF_ERROR_INVALID_ARGUMENT: 2606 reason = scf_strerror(scf_error()); 2607 break; 2608 2609 case SCF_ERROR_INTERNAL: 2610 reason = 2611 "unknown error (see console for details)"; 2612 break; 2613 } 2614 2615 pr_warn("failed to backup repository: %s\n", reason); 2616 exit_status = UU_EXIT_FATAL; 2617 } 2618 } else if (strcmp(argv[optind], "_smf_repository_switch") == 0) { 2619 const char *reason = NULL; 2620 2621 ++optind; 2622 2623 /* 2624 * Check argument and setup scf_switch structure 2625 */ 2626 if (optind != argc - 1) 2627 exit(1); 2628 2629 if (strcmp(argv[optind], "fast") == 0) { 2630 sw_back = 0; 2631 } else if (strcmp(argv[optind], "perm") == 0) { 2632 sw_back = 1; 2633 } else { 2634 exit(UU_EXIT_USAGE); 2635 } 2636 2637 /* 2638 * Call into switch primitive 2639 */ 2640 if ((err = _scf_repository_switch(h, sw_back)) != 2641 SCF_SUCCESS) { 2642 /* 2643 * Retrieve per thread SCF error code 2644 */ 2645 switch (scf_error()) { 2646 case SCF_ERROR_NOT_BOUND: 2647 abort(); 2648 /* NOTREACHED */ 2649 2650 case SCF_ERROR_CONNECTION_BROKEN: 2651 case SCF_ERROR_BACKEND_READONLY: 2652 scfdie(); 2653 /* NOTREACHED */ 2654 2655 case SCF_ERROR_PERMISSION_DENIED: 2656 case SCF_ERROR_INVALID_ARGUMENT: 2657 reason = scf_strerror(scf_error()); 2658 break; 2659 2660 case SCF_ERROR_INTERNAL: 2661 reason = "File operation error: (see console)"; 2662 break; 2663 2664 default: 2665 abort(); 2666 /* NOTREACHED */ 2667 } 2668 2669 pr_warn("failed to switch repository: %s\n", reason); 2670 exit_status = UU_EXIT_FATAL; 2671 } 2672 } else { 2673 usage(); 2674 } 2675 2676 if (scf_handle_unbind(h) == -1) 2677 scfdie(); 2678 nextzone: 2679 scf_handle_destroy(h); 2680 if (do_zones && zent < nzents) 2681 goto again; 2682 2683 return (exit_status); 2684 } 2685