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 /* 23 * Copyright (c) 2004, 2010, Oracle and/or its affiliates. All rights reserved. 24 * Copyright 2020 Joyent, Inc. 25 * Copyright (c) 2015, 2016 by Delphix. All rights reserved. 26 */ 27 28 /* 29 * svcs - display attributes of service instances 30 * 31 * We have two output formats and six instance selection mechanisms. The 32 * primary output format is a line of attributes (selected by -o), possibly 33 * followed by process description lines (if -p is specified), for each 34 * instance selected. The columns available to display are described by the 35 * struct column columns array. The columns to actually display are kept in 36 * the opt_columns array as indicies into the columns array. The selection 37 * mechanisms available for this format are service FMRIs (selects all child 38 * instances), instance FMRIs, instance FMRI glob patterns, instances with 39 * a certain restarter (-R), dependencies of instances (-d), and dependents of 40 * instances (-D). Since the lines must be sorted (per -sS), we'll just stick 41 * each into a data structure and print them in order when we're done. To 42 * avoid listing the same instance twice (when -d and -D aren't given), we'll 43 * use a hash table of FMRIs to record that we've listed (added to the tree) 44 * an instance. 45 * 46 * The secondary output format (-l "long") is a paragraph of text for the 47 * services or instances selected. Not needing to be sorted, it's implemented 48 * by just calling print_detailed() for each FMRI given. 49 */ 50 51 #include "svcs.h" 52 #include "notify_params.h" 53 54 /* Get the byteorder macros to ease sorting. */ 55 #include <sys/types.h> 56 #include <netinet/in.h> 57 #include <inttypes.h> 58 59 #include <sys/contract.h> 60 #include <sys/ctfs.h> 61 #include <sys/stat.h> 62 63 #include <assert.h> 64 #include <errno.h> 65 #include <fcntl.h> 66 #include <fnmatch.h> 67 #include <libcontract.h> 68 #include <libcontract_priv.h> 69 #include <libintl.h> 70 #include <libscf.h> 71 #include <libscf_priv.h> 72 #include <libuutil.h> 73 #include <libnvpair.h> 74 #include <libproc.h> 75 #include <locale.h> 76 #include <stdarg.h> 77 #include <stdio.h> 78 #include <stdlib.h> 79 #include <strings.h> 80 #include <time.h> 81 #include <libzonecfg.h> 82 #include <zone.h> 83 84 #ifndef TEXT_DOMAIN 85 #define TEXT_DOMAIN "SUNW_OST_OSCMD" 86 #endif /* TEXT_DOMAIN */ 87 88 #define LEGACY_UNKNOWN "unknown" 89 90 /* 91 * Per proc(5) when pr_nlwp, pr_nzomb, and pr_lwp.pr_lwpid are all 0, 92 * the process is a zombie. 93 */ 94 #define IS_ZOMBIE(_psip) \ 95 ((_psip)->pr_nlwp == 0 && (_psip)->pr_nzomb == 0 && \ 96 (_psip)->pr_lwp.pr_lwpid == 0) 97 98 /* 99 * An AVL-storable node for output lines and the keys to sort them by. 100 */ 101 struct avl_string { 102 uu_avl_node_t node; 103 char *key; 104 char *str; 105 }; 106 107 /* 108 * For lists of parsed restarter FMRIs. 109 */ 110 struct pfmri_list { 111 const char *scope; 112 const char *service; 113 const char *instance; 114 struct pfmri_list *next; 115 }; 116 117 118 /* 119 * Globals 120 */ 121 scf_handle_t *h; 122 static scf_propertygroup_t *g_pg; 123 static scf_property_t *g_prop; 124 static scf_value_t *g_val; 125 126 static size_t line_sz; /* Bytes in the header line. */ 127 static size_t sortkey_sz; /* Bytes in sort keys. */ 128 static uu_avl_pool_t *lines_pool; 129 static uu_avl_t *lines; /* Output lines. */ 130 int exit_status; 131 ssize_t max_scf_name_length; 132 ssize_t max_scf_value_length; 133 ssize_t max_scf_fmri_length; 134 static ssize_t max_scf_type_length; 135 static time_t now; 136 static struct pfmri_list *restarters = NULL; 137 static int first_paragraph = 1; /* For -l mode. */ 138 static char *common_name_buf; /* Sized for maximal length value. */ 139 char *locale; /* Current locale. */ 140 char *g_zonename; /* zone being operated upon */ 141 142 /* 143 * Pathname storage for path generated from the fmri. 144 * Used for reading the ctid and (start) pid files for an inetd service. 145 */ 146 static char genfmri_filename[MAXPATHLEN] = ""; 147 148 /* Options */ 149 static int *opt_columns = NULL; /* Indices into columns to display. */ 150 static int opt_cnum = 0; 151 static int opt_processes = 0; /* Print processes? */ 152 static int *opt_sort = NULL; /* Indices into columns to sort. */ 153 static int opt_snum = 0; 154 static int opt_nstate_shown = 0; /* Will nstate be shown? */ 155 static int opt_verbose = 0; 156 static char *opt_zone; /* zone selected, if any */ 157 158 /* Minimize string constants. */ 159 static const char * const scf_property_state = SCF_PROPERTY_STATE; 160 static const char * const scf_property_next_state = SCF_PROPERTY_NEXT_STATE; 161 static const char * const scf_property_contract = SCF_PROPERTY_CONTRACT; 162 163 164 /* 165 * Utility functions 166 */ 167 168 /* 169 * For unexpected libscf errors. The ending newline is necessary to keep 170 * uu_die() from appending the errno error. 171 */ 172 #ifndef NDEBUG 173 void 174 do_scfdie(const char *file, int line) 175 { 176 uu_die(gettext("%s:%d: Unexpected libscf error: %s. Exiting.\n"), 177 file, line, scf_strerror(scf_error())); 178 } 179 #else 180 void 181 scfdie(void) 182 { 183 uu_die(gettext("Unexpected libscf error: %s. Exiting.\n"), 184 scf_strerror(scf_error())); 185 } 186 #endif 187 188 void * 189 safe_malloc(size_t sz) 190 { 191 void *ptr; 192 193 ptr = malloc(sz); 194 if (ptr == NULL) 195 uu_die(gettext("Out of memory")); 196 197 return (ptr); 198 } 199 200 char * 201 safe_strdup(const char *str) 202 { 203 char *cp; 204 205 cp = strdup(str); 206 if (cp == NULL) 207 uu_die(gettext("Out of memory.\n")); 208 209 return (cp); 210 } 211 212 /* 213 * FMRI hashtable. For uniquifing listings. 214 */ 215 216 struct ht_elem { 217 const char *fmri; 218 struct ht_elem *next; 219 }; 220 221 static struct ht_elem **ht_buckets = NULL; 222 static uint_t ht_buckets_num = 0; 223 static uint_t ht_num; 224 225 static void 226 ht_free(void) 227 { 228 struct ht_elem *elem, *next; 229 int i; 230 231 for (i = 0; i < ht_buckets_num; i++) { 232 for (elem = ht_buckets[i]; elem != NULL; elem = next) { 233 next = elem->next; 234 free((char *)elem->fmri); 235 free(elem); 236 } 237 } 238 239 free(ht_buckets); 240 ht_buckets_num = 0; 241 ht_buckets = NULL; 242 } 243 244 static void 245 ht_init(void) 246 { 247 assert(ht_buckets == NULL); 248 249 ht_buckets_num = 8; 250 ht_buckets = safe_malloc(sizeof (*ht_buckets) * ht_buckets_num); 251 bzero(ht_buckets, sizeof (*ht_buckets) * ht_buckets_num); 252 ht_num = 0; 253 } 254 255 static uint_t 256 ht_hash_fmri(const char *fmri) 257 { 258 uint_t h = 0, g; 259 const char *p, *k; 260 261 /* All FMRIs begin with svc:/, so skip that part. */ 262 assert(strncmp(fmri, "svc:/", sizeof ("svc:/") - 1) == 0); 263 k = fmri + sizeof ("svc:/") - 1; 264 265 /* 266 * Generic hash function from uts/common/os/modhash.c. 267 */ 268 for (p = k; *p != '\0'; ++p) { 269 h = (h << 4) + *p; 270 if ((g = (h & 0xf0000000)) != 0) { 271 h ^= (g >> 24); 272 h ^= g; 273 } 274 } 275 276 return (h); 277 } 278 279 static void 280 ht_grow() 281 { 282 uint_t new_ht_buckets_num; 283 struct ht_elem **new_ht_buckets; 284 int i; 285 286 new_ht_buckets_num = ht_buckets_num * 2; 287 assert(new_ht_buckets_num > ht_buckets_num); 288 new_ht_buckets = 289 safe_malloc(sizeof (*new_ht_buckets) * new_ht_buckets_num); 290 bzero(new_ht_buckets, sizeof (*new_ht_buckets) * new_ht_buckets_num); 291 292 for (i = 0; i < ht_buckets_num; ++i) { 293 struct ht_elem *elem, *next; 294 295 for (elem = ht_buckets[i]; elem != NULL; elem = next) { 296 uint_t h; 297 298 next = elem->next; 299 300 h = ht_hash_fmri(elem->fmri); 301 302 elem->next = 303 new_ht_buckets[h & (new_ht_buckets_num - 1)]; 304 new_ht_buckets[h & (new_ht_buckets_num - 1)] = elem; 305 } 306 } 307 308 free(ht_buckets); 309 310 ht_buckets = new_ht_buckets; 311 ht_buckets_num = new_ht_buckets_num; 312 } 313 314 /* 315 * Add an FMRI to the hash table. Returns 1 if it was already there, 316 * 0 otherwise. 317 */ 318 static int 319 ht_add(const char *fmri) 320 { 321 uint_t h; 322 struct ht_elem *elem; 323 324 h = ht_hash_fmri(fmri); 325 326 elem = ht_buckets[h & (ht_buckets_num - 1)]; 327 328 for (; elem != NULL; elem = elem->next) { 329 if (strcmp(elem->fmri, fmri) == 0) 330 return (1); 331 } 332 333 /* Grow when average chain length is over 3. */ 334 if (ht_num > 3 * ht_buckets_num) 335 ht_grow(); 336 337 ++ht_num; 338 339 elem = safe_malloc(sizeof (*elem)); 340 elem->fmri = strdup(fmri); 341 elem->next = ht_buckets[h & (ht_buckets_num - 1)]; 342 ht_buckets[h & (ht_buckets_num - 1)] = elem; 343 344 return (0); 345 } 346 347 348 349 /* 350 * Convenience libscf wrapper functions. 351 */ 352 353 /* 354 * Get the single value of the named property in the given property group, 355 * which must have type ty, and put it in *vp. If ty is SCF_TYPE_ASTRING, vp 356 * is taken to be a char **, and sz is the size of the buffer. sz is unused 357 * otherwise. Return 0 on success, -1 if the property doesn't exist, has the 358 * wrong type, or doesn't have a single value. If flags has EMPTY_OK, don't 359 * complain if the property has no values (but return nonzero). If flags has 360 * MULTI_OK and the property has multiple values, succeed with E2BIG. 361 */ 362 int 363 pg_get_single_val(scf_propertygroup_t *pg, const char *propname, scf_type_t ty, 364 void *vp, size_t sz, uint_t flags) 365 { 366 char *buf, root[MAXPATHLEN]; 367 size_t buf_sz; 368 int ret = -1, r; 369 boolean_t multi = B_FALSE; 370 371 assert((flags & ~(EMPTY_OK | MULTI_OK)) == 0); 372 373 if (scf_pg_get_property(pg, propname, g_prop) == -1) { 374 if (scf_error() != SCF_ERROR_NOT_FOUND) 375 scfdie(); 376 377 goto out; 378 } 379 380 if (scf_property_is_type(g_prop, ty) != SCF_SUCCESS) { 381 if (scf_error() == SCF_ERROR_TYPE_MISMATCH) 382 goto misconfigured; 383 scfdie(); 384 } 385 386 if (scf_property_get_value(g_prop, g_val) != SCF_SUCCESS) { 387 switch (scf_error()) { 388 case SCF_ERROR_NOT_FOUND: 389 if (flags & EMPTY_OK) 390 goto out; 391 goto misconfigured; 392 393 case SCF_ERROR_CONSTRAINT_VIOLATED: 394 if (flags & MULTI_OK) { 395 multi = B_TRUE; 396 break; 397 } 398 goto misconfigured; 399 400 case SCF_ERROR_PERMISSION_DENIED: 401 default: 402 scfdie(); 403 } 404 } 405 406 switch (ty) { 407 case SCF_TYPE_ASTRING: 408 r = scf_value_get_astring(g_val, vp, sz); 409 if (r == 0 && !(flags & EMPTY_OK)) { 410 uu_die(gettext("Unexpected empty string for property " 411 "%s. Exiting.\n"), propname); 412 } 413 if (r >= 0) 414 r = SCF_SUCCESS; 415 break; 416 417 case SCF_TYPE_BOOLEAN: 418 r = scf_value_get_boolean(g_val, (uint8_t *)vp); 419 break; 420 421 case SCF_TYPE_COUNT: 422 r = scf_value_get_count(g_val, (uint64_t *)vp); 423 break; 424 425 case SCF_TYPE_INTEGER: 426 r = scf_value_get_integer(g_val, (int64_t *)vp); 427 break; 428 429 case SCF_TYPE_TIME: { 430 int64_t sec; 431 int32_t ns; 432 r = scf_value_get_time(g_val, &sec, &ns); 433 ((struct timeval *)vp)->tv_sec = sec; 434 ((struct timeval *)vp)->tv_usec = ns / 1000; 435 break; 436 } 437 438 case SCF_TYPE_USTRING: 439 r = scf_value_get_ustring(g_val, vp, sz) > 0 ? SCF_SUCCESS : -1; 440 break; 441 442 default: 443 #ifndef NDEBUG 444 uu_warn("%s:%d: Unknown type %d.\n", __FILE__, __LINE__, ty); 445 #endif 446 abort(); 447 } 448 if (r != SCF_SUCCESS) 449 scfdie(); 450 451 ret = multi ? E2BIG : 0; 452 goto out; 453 454 misconfigured: 455 buf_sz = max_scf_fmri_length + 1; 456 buf = safe_malloc(buf_sz); 457 if (scf_property_to_fmri(g_prop, buf, buf_sz) == -1) 458 scfdie(); 459 460 uu_warn(gettext("Property \"%s\" is misconfigured.\n"), buf); 461 462 free(buf); 463 464 out: 465 if (ret != 0 || g_zonename == NULL || 466 (strcmp(propname, SCF_PROPERTY_LOGFILE) != 0 && 467 strcmp(propname, SCF_PROPERTY_ALT_LOGFILE) != 0)) 468 return (ret); 469 470 /* 471 * If we're here, we have a log file and we have specified a zone. 472 * As a convenience, we're going to prepend the zone path to the 473 * name of the log file. 474 */ 475 root[0] = '\0'; 476 (void) zone_get_rootpath(g_zonename, root, sizeof (root)); 477 (void) strlcat(root, vp, sizeof (root)); 478 (void) snprintf(vp, sz, "%s", root); 479 480 return (ret); 481 } 482 483 static scf_snapshot_t * 484 get_running_snapshot(scf_instance_t *inst) 485 { 486 scf_snapshot_t *snap; 487 488 snap = scf_snapshot_create(h); 489 if (snap == NULL) 490 scfdie(); 491 492 if (scf_instance_get_snapshot(inst, "running", snap) == 0) 493 return (snap); 494 495 if (scf_error() != SCF_ERROR_NOT_FOUND) 496 scfdie(); 497 498 scf_snapshot_destroy(snap); 499 return (NULL); 500 } 501 502 /* 503 * As pg_get_single_val(), except look the property group up in an 504 * instance. If "use_running" is set, and the running snapshot exists, 505 * do a composed lookup there. Otherwise, do an (optionally composed) 506 * lookup on the current values. Note that lookups using snapshots are 507 * always composed. 508 */ 509 int 510 inst_get_single_val(scf_instance_t *inst, const char *pgname, 511 const char *propname, scf_type_t ty, void *vp, size_t sz, uint_t flags, 512 int use_running, int composed) 513 { 514 scf_snapshot_t *snap = NULL; 515 int r; 516 517 if (use_running) 518 snap = get_running_snapshot(inst); 519 if (composed || use_running) 520 r = scf_instance_get_pg_composed(inst, snap, pgname, g_pg); 521 else 522 r = scf_instance_get_pg(inst, pgname, g_pg); 523 if (snap) 524 scf_snapshot_destroy(snap); 525 if (r == -1) 526 return (-1); 527 528 r = pg_get_single_val(g_pg, propname, ty, vp, sz, flags); 529 530 return (r); 531 } 532 533 static int 534 instance_enabled(scf_instance_t *inst, boolean_t temp) 535 { 536 uint8_t b; 537 538 if (inst_get_single_val(inst, 539 temp ? SCF_PG_GENERAL_OVR : SCF_PG_GENERAL, SCF_PROPERTY_ENABLED, 540 SCF_TYPE_BOOLEAN, &b, 0, 0, 0, 0) != 0) 541 return (-1); 542 543 return (b ? 1 : 0); 544 } 545 546 /* 547 * Get a string property from the restarter property group of the given 548 * instance. Return an empty string on normal problems. 549 */ 550 static void 551 get_restarter_string_prop(scf_instance_t *inst, const char *pname, 552 char *buf, size_t buf_sz) 553 { 554 if (inst_get_single_val(inst, SCF_PG_RESTARTER, pname, 555 SCF_TYPE_ASTRING, buf, buf_sz, 0, 0, 1) != 0) 556 *buf = '\0'; 557 } 558 559 static int 560 get_restarter_time_prop(scf_instance_t *inst, const char *pname, 561 struct timeval *tvp, int ok_if_empty) 562 { 563 int r; 564 565 r = inst_get_single_val(inst, SCF_PG_RESTARTER, pname, SCF_TYPE_TIME, 566 tvp, 0, ok_if_empty ? EMPTY_OK : 0, 0, 1); 567 568 return (r == 0 ? 0 : -1); 569 } 570 571 static int 572 get_restarter_count_prop(scf_instance_t *inst, const char *pname, uint64_t *cp, 573 uint_t flags) 574 { 575 return (inst_get_single_val(inst, SCF_PG_RESTARTER, pname, 576 SCF_TYPE_COUNT, cp, 0, flags, 0, 1)); 577 } 578 579 580 /* 581 * Generic functions 582 */ 583 584 /* 585 * Return an array of pids associated with the given contract id. 586 * Returned pids are added to the end of the pidsp array. 587 */ 588 static void 589 ctid_to_pids(uint64_t c, pid_t **pidsp, uint_t *np) 590 { 591 ct_stathdl_t ctst; 592 uint_t m; 593 int fd; 594 int r, err; 595 pid_t *pids; 596 597 fd = contract_open(c, NULL, "status", O_RDONLY); 598 if (fd < 0) 599 return; 600 601 err = ct_status_read(fd, CTD_ALL, &ctst); 602 if (err != 0) { 603 uu_warn(gettext("Could not read status of contract " 604 "%ld: %s.\n"), c, strerror(err)); 605 (void) close(fd); 606 return; 607 } 608 609 (void) close(fd); 610 611 r = ct_pr_status_get_members(ctst, &pids, &m); 612 assert(r == 0); 613 614 if (m == 0) { 615 ct_status_free(ctst); 616 return; 617 } 618 619 *pidsp = realloc(*pidsp, (*np + m) * sizeof (*pidsp)); 620 if (*pidsp == NULL) 621 uu_die(gettext("Out of memory")); 622 623 bcopy(pids, *pidsp + *np, m * sizeof (*pids)); 624 *np += m; 625 626 ct_status_free(ctst); 627 } 628 629 static int 630 propvals_to_pids(scf_propertygroup_t *pg, const char *pname, pid_t **pidsp, 631 uint_t *np, scf_property_t *prop, scf_value_t *val, scf_iter_t *iter) 632 { 633 scf_type_t ty; 634 uint64_t c; 635 int r; 636 637 if (scf_pg_get_property(pg, pname, prop) != 0) { 638 if (scf_error() != SCF_ERROR_NOT_FOUND) 639 scfdie(); 640 641 return (ENOENT); 642 } 643 644 if (scf_property_type(prop, &ty) != 0) 645 scfdie(); 646 647 if (ty != SCF_TYPE_COUNT) 648 return (EINVAL); 649 650 if (scf_iter_property_values(iter, prop) != 0) 651 scfdie(); 652 653 for (;;) { 654 r = scf_iter_next_value(iter, val); 655 if (r == -1) 656 scfdie(); 657 if (r == 0) 658 break; 659 660 if (scf_value_get_count(val, &c) != 0) 661 scfdie(); 662 663 ctid_to_pids(c, pidsp, np); 664 } 665 666 return (0); 667 } 668 669 /* 670 * Check if instance has general/restarter property that matches 671 * given string. Restarter string must be in canonified form. 672 * Returns 0 for success; -1 otherwise. 673 */ 674 static int 675 check_for_restarter(scf_instance_t *inst, const char *restarter) 676 { 677 char *fmri_buf; 678 char *fmri_buf_canonified = NULL; 679 int ret = -1; 680 681 if (inst == NULL) 682 return (-1); 683 684 /* Get restarter */ 685 fmri_buf = safe_malloc(max_scf_fmri_length + 1); 686 if (inst_get_single_val(inst, SCF_PG_GENERAL, 687 SCF_PROPERTY_RESTARTER, SCF_TYPE_ASTRING, fmri_buf, 688 max_scf_fmri_length + 1, 0, 0, 1) != 0) 689 goto out; 690 691 fmri_buf_canonified = safe_malloc(max_scf_fmri_length + 1); 692 if (scf_canonify_fmri(fmri_buf, fmri_buf_canonified, 693 (max_scf_fmri_length + 1)) < 0) 694 goto out; 695 696 if (strcmp(fmri_buf, restarter) == 0) 697 ret = 0; 698 699 out: 700 free(fmri_buf); 701 if (fmri_buf_canonified) 702 free(fmri_buf_canonified); 703 return (ret); 704 } 705 706 /* 707 * Common code that is used by ctids_by_restarter and pids_by_restarter. 708 * Checks for a common restarter and if one is available, it generates 709 * the appropriate filename using wip->fmri and stores that in the 710 * global genfmri_filename. 711 * 712 * Restarters currently supported are: svc:/network/inetd:default 713 * If a restarter specific action is available, then restarter_spec 714 * is set to 1. If a restarter specific action is not available, then 715 * restarter_spec is set to 0 and a -1 is returned. 716 * 717 * Returns: 718 * 0 if success: restarter specific action found and filename generated 719 * -1 if restarter specific action not found, 720 * if restarter specific action found but an error was encountered 721 * during the generation of the wip->fmri based filename 722 */ 723 static int 724 common_by_restarter(scf_instance_t *inst, const char *fmri, 725 int *restarter_specp) 726 { 727 int ret = -1; 728 int r; 729 730 /* Check for inetd specific restarter */ 731 if (check_for_restarter(inst, "svc:/network/inetd:default") != 0) { 732 *restarter_specp = 0; 733 return (ret); 734 } 735 736 *restarter_specp = 1; 737 738 /* Get the ctid filename associated with this instance */ 739 r = gen_filenms_from_fmri(fmri, "ctid", genfmri_filename, NULL); 740 741 switch (r) { 742 case 0: 743 break; 744 745 case -1: 746 /* 747 * Unable to get filename from fmri. Print warning 748 * and return failure with no ctids. 749 */ 750 uu_warn(gettext("Unable to read contract ids for %s -- " 751 "FMRI is too long\n"), fmri); 752 return (ret); 753 754 case -2: 755 /* 756 * The directory didn't exist, so no contracts. 757 * Return failure with no ctids. 758 */ 759 return (ret); 760 761 default: 762 uu_warn(gettext("%s:%d: gen_filenms_from_fmri() failed with " 763 "unknown error %d\n"), __FILE__, __LINE__, r); 764 abort(); 765 } 766 767 return (0); 768 769 } 770 771 /* 772 * Get or print a contract id using a restarter specific action. 773 * 774 * If the print_flag is not set, this routine gets the single contract 775 * id associated with this instance. 776 * If the print flag is set, then print each contract id found. 777 * 778 * Returns: 779 * 0 if success: restarter specific action found and used with no error 780 * -1 if restarter specific action not found 781 * -1 if restarter specific action found, but there was a failure 782 * -1 if print flag is not set and no contract id is found or multiple 783 * contract ids were found 784 * E2BIG if print flag is not set, MULTI_OK bit in flag is set and multiple 785 * contract ids were found 786 */ 787 static int 788 ctids_by_restarter(scf_walkinfo_t *wip, uint64_t *cp, int print_flag, 789 uint_t flags, int *restarter_specp, void (*callback_header)(), 790 void (*callback_ctid)(uint64_t)) 791 { 792 FILE *fp; 793 int ret = -1; 794 int fscanf_ret; 795 uint64_t cp2; 796 int rest_ret; 797 798 /* Check if callbacks are needed and were passed in */ 799 if (print_flag) { 800 if ((callback_header == NULL) || (callback_ctid == NULL)) 801 return (ret); 802 } 803 804 /* Check for restarter specific action and generation of filename */ 805 rest_ret = common_by_restarter(wip->inst, wip->fmri, restarter_specp); 806 if (rest_ret != 0) 807 return (rest_ret); 808 809 /* 810 * If fopen fails, then ctid file hasn't been created yet. 811 * If print_flag is set, this is ok; otherwise fail. 812 */ 813 if ((fp = fopen(genfmri_filename, "r")) == NULL) { 814 if (print_flag) 815 return (0); 816 goto out; 817 } 818 819 if (print_flag) { 820 /* 821 * Print all contract ids that are found. 822 * First callback to print ctid header. 823 */ 824 callback_header(); 825 826 /* fscanf may not set errno, so be sure to clear it first */ 827 errno = 0; 828 while ((fscanf_ret = fscanf(fp, "%llu", cp)) == 1) { 829 /* Callback to print contract id */ 830 callback_ctid(*cp); 831 errno = 0; 832 } 833 /* EOF is not a failure when no errno. */ 834 if ((fscanf_ret != EOF) || (errno != 0)) { 835 uu_die(gettext("Unable to read ctid file for %s"), 836 wip->fmri); 837 } 838 (void) putchar('\n'); 839 ret = 0; 840 } else { 841 /* Must find 1 ctid or fail */ 842 if (fscanf(fp, "%llu", cp) == 1) { 843 /* If 2nd ctid found - fail */ 844 if (fscanf(fp, "%llu", &cp2) == 1) { 845 if (flags & MULTI_OK) 846 ret = E2BIG; 847 } else { 848 /* Success - found only 1 ctid */ 849 ret = 0; 850 } 851 } 852 } 853 (void) fclose(fp); 854 855 out: 856 return (ret); 857 } 858 859 /* 860 * Get the process ids associated with an instance using a restarter 861 * specific action. 862 * 863 * Returns: 864 * 0 if success: restarter specific action found and used with no error 865 * -1 restarter specific action not found or if failure 866 */ 867 static int 868 pids_by_restarter(scf_instance_t *inst, const char *fmri, 869 pid_t **pids, uint_t *np, int *restarter_specp) 870 { 871 uint64_t c; 872 FILE *fp; 873 int fscanf_ret; 874 int rest_ret; 875 876 /* Check for restarter specific action and generation of filename */ 877 rest_ret = common_by_restarter(inst, fmri, restarter_specp); 878 if (rest_ret != 0) 879 return (rest_ret); 880 881 /* 882 * If fopen fails with ENOENT then the ctid file hasn't been 883 * created yet so return success. 884 * For all other errors - fail with uu_die. 885 */ 886 if ((fp = fopen(genfmri_filename, "r")) == NULL) { 887 if (errno == ENOENT) 888 return (0); 889 uu_die(gettext("Unable to open ctid file for %s"), fmri); 890 } 891 892 /* fscanf may not set errno, so be sure to clear it first */ 893 errno = 0; 894 while ((fscanf_ret = fscanf(fp, "%llu", &c)) == 1) { 895 if (c == 0) { 896 (void) fclose(fp); 897 uu_die(gettext("ctid file for %s has corrupt data"), 898 fmri); 899 } 900 ctid_to_pids(c, pids, np); 901 errno = 0; 902 } 903 /* EOF is not a failure when no errno. */ 904 if ((fscanf_ret != EOF) || (errno != 0)) { 905 uu_die(gettext("Unable to read ctid file for %s"), fmri); 906 } 907 908 (void) fclose(fp); 909 return (0); 910 } 911 912 static int 913 instance_processes(scf_instance_t *inst, const char *fmri, 914 pid_t **pids, uint_t *np) 915 { 916 scf_iter_t *iter; 917 int ret; 918 int restarter_spec; 919 920 /* Use the restarter specific get pids routine, if available. */ 921 ret = pids_by_restarter(inst, fmri, pids, np, &restarter_spec); 922 if (restarter_spec == 1) 923 return (ret); 924 925 if ((iter = scf_iter_create(h)) == NULL) 926 scfdie(); 927 928 if (scf_instance_get_pg(inst, SCF_PG_RESTARTER, g_pg) == 0) { 929 *pids = NULL; 930 *np = 0; 931 932 (void) propvals_to_pids(g_pg, scf_property_contract, pids, np, 933 g_prop, g_val, iter); 934 935 (void) propvals_to_pids(g_pg, SCF_PROPERTY_TRANSIENT_CONTRACT, 936 pids, np, g_prop, g_val, iter); 937 938 ret = 0; 939 } else { 940 if (scf_error() != SCF_ERROR_NOT_FOUND) 941 scfdie(); 942 943 ret = -1; 944 } 945 946 scf_iter_destroy(iter); 947 948 return (ret); 949 } 950 951 /* 952 * Column sprint and sortkey functions 953 */ 954 955 struct column { 956 const char *name; 957 int width; 958 959 /* 960 * This function should write the value for the column into buf, and 961 * grow or allocate buf accordingly. It should always write at least 962 * width bytes, blanking unused bytes with spaces. If the field is 963 * greater than the column width we allow it to overlap other columns. 964 * In particular, it shouldn't write any null bytes. (Though an extra 965 * null byte past the end is currently tolerated.) If the property 966 * group is non-NULL, then we are dealing with a legacy service. 967 */ 968 void (*sprint)(char **, scf_walkinfo_t *); 969 970 int sortkey_width; 971 972 /* 973 * This function should write sortkey_width bytes into buf which will 974 * cause memcmp() to sort it properly. (Unlike sprint() above, 975 * however, an extra null byte may overrun the buffer.) The second 976 * argument controls whether the results are sorted in forward or 977 * reverse order. 978 */ 979 void (*get_sortkey)(char *, int, scf_walkinfo_t *); 980 }; 981 982 static void 983 reverse_bytes(char *buf, size_t len) 984 { 985 int i; 986 987 for (i = 0; i < len; ++i) 988 buf[i] = ~buf[i]; 989 } 990 991 /* CTID */ 992 #define CTID_COLUMN_WIDTH 6 993 #define CTID_COLUMN_BUFSIZE 20 /* max ctid_t + space + \0 */ 994 995 static void 996 sprint_ctid(char **buf, scf_walkinfo_t *wip) 997 { 998 int r; 999 uint64_t c; 1000 size_t newsize = (*buf ? strlen(*buf) : 0) + CTID_COLUMN_BUFSIZE; 1001 char *newbuf = safe_malloc(newsize); 1002 int restarter_spec; 1003 1004 /* 1005 * Use the restarter specific get pids routine, if available. 1006 * Only check for non-legacy services (wip->pg == 0). 1007 */ 1008 if (wip->pg != NULL) { 1009 r = pg_get_single_val(wip->pg, scf_property_contract, 1010 SCF_TYPE_COUNT, &c, 0, EMPTY_OK | MULTI_OK); 1011 } else { 1012 r = ctids_by_restarter(wip, &c, 0, MULTI_OK, &restarter_spec, 1013 NULL, NULL); 1014 if (restarter_spec == 0) { 1015 /* No restarter specific routine */ 1016 r = get_restarter_count_prop(wip->inst, 1017 scf_property_contract, &c, EMPTY_OK | MULTI_OK); 1018 } 1019 } 1020 1021 if (r == 0) 1022 (void) snprintf(newbuf, newsize, "%s%*lu ", 1023 *buf ? *buf : "", CTID_COLUMN_WIDTH, (ctid_t)c); 1024 else if (r == E2BIG) 1025 (void) snprintf(newbuf, newsize, "%s%*lu* ", 1026 *buf ? *buf : "", CTID_COLUMN_WIDTH - 1, (ctid_t)c); 1027 else 1028 (void) snprintf(newbuf, newsize, "%s%*s ", 1029 *buf ? *buf : "", CTID_COLUMN_WIDTH, "-"); 1030 if (*buf) 1031 free(*buf); 1032 *buf = newbuf; 1033 } 1034 1035 #define CTID_SORTKEY_WIDTH (sizeof (uint64_t)) 1036 1037 static void 1038 sortkey_ctid(char *buf, int reverse, scf_walkinfo_t *wip) 1039 { 1040 int r; 1041 uint64_t c; 1042 int restarter_spec; 1043 1044 /* 1045 * Use the restarter specific get pids routine, if available. 1046 * Only check for non-legacy services (wip->pg == 0). 1047 */ 1048 if (wip->pg != NULL) { 1049 r = pg_get_single_val(wip->pg, scf_property_contract, 1050 SCF_TYPE_COUNT, &c, 0, EMPTY_OK); 1051 } else { 1052 r = ctids_by_restarter(wip, &c, 0, MULTI_OK, &restarter_spec, 1053 NULL, NULL); 1054 if (restarter_spec == 0) { 1055 /* No restarter specific routine */ 1056 r = get_restarter_count_prop(wip->inst, 1057 scf_property_contract, &c, EMPTY_OK); 1058 } 1059 } 1060 1061 if (r == 0) { 1062 /* 1063 * Use the id itself, but it must be big-endian for this to 1064 * work. 1065 */ 1066 c = BE_64(c); 1067 1068 bcopy(&c, buf, CTID_SORTKEY_WIDTH); 1069 } else { 1070 bzero(buf, CTID_SORTKEY_WIDTH); 1071 } 1072 1073 if (reverse) 1074 reverse_bytes(buf, CTID_SORTKEY_WIDTH); 1075 } 1076 1077 /* DESC */ 1078 #define DESC_COLUMN_WIDTH 100 1079 1080 static void 1081 sprint_desc(char **buf, scf_walkinfo_t *wip) 1082 { 1083 char *x; 1084 size_t newsize; 1085 char *newbuf; 1086 1087 if (common_name_buf == NULL) 1088 common_name_buf = safe_malloc(max_scf_value_length + 1); 1089 1090 bzero(common_name_buf, max_scf_value_length + 1); 1091 1092 if (wip->pg != NULL) { 1093 common_name_buf[0] = '-'; 1094 } else if (inst_get_single_val(wip->inst, SCF_PG_TM_COMMON_NAME, locale, 1095 SCF_TYPE_USTRING, common_name_buf, max_scf_value_length, 0, 1096 1, 1) == -1 && 1097 inst_get_single_val(wip->inst, SCF_PG_TM_COMMON_NAME, "C", 1098 SCF_TYPE_USTRING, common_name_buf, max_scf_value_length, 0, 1099 1, 1) == -1) { 1100 common_name_buf[0] = '-'; 1101 } 1102 1103 /* 1104 * Collapse multi-line tm_common_name values into a single line. 1105 */ 1106 for (x = common_name_buf; *x != '\0'; x++) 1107 if (*x == '\n') 1108 *x = ' '; 1109 1110 if (strlen(common_name_buf) > DESC_COLUMN_WIDTH) 1111 newsize = (*buf ? strlen(*buf) : 0) + 1112 strlen(common_name_buf) + 1; 1113 else 1114 newsize = (*buf ? strlen(*buf) : 0) + DESC_COLUMN_WIDTH + 1; 1115 newbuf = safe_malloc(newsize); 1116 (void) snprintf(newbuf, newsize, "%s%-*s ", *buf ? *buf : "", 1117 DESC_COLUMN_WIDTH, common_name_buf); 1118 if (*buf) 1119 free(*buf); 1120 *buf = newbuf; 1121 } 1122 1123 /* ARGSUSED */ 1124 static void 1125 sortkey_desc(char *buf, int reverse, scf_walkinfo_t *wip) 1126 { 1127 bzero(buf, DESC_COLUMN_WIDTH); 1128 } 1129 1130 /* State columns (STATE, NSTATE, S, N, SN, STA, NSTA) */ 1131 1132 static char 1133 state_to_char(const char *state) 1134 { 1135 if (strcmp(state, SCF_STATE_STRING_UNINIT) == 0) 1136 return ('u'); 1137 1138 if (strcmp(state, SCF_STATE_STRING_OFFLINE) == 0) 1139 return ('0'); 1140 1141 if (strcmp(state, SCF_STATE_STRING_ONLINE) == 0) 1142 return ('1'); 1143 1144 if (strcmp(state, SCF_STATE_STRING_MAINT) == 0) 1145 return ('m'); 1146 1147 if (strcmp(state, SCF_STATE_STRING_DISABLED) == 0) 1148 return ('d'); 1149 1150 if (strcmp(state, SCF_STATE_STRING_DEGRADED) == 0) 1151 return ('D'); 1152 1153 if (strcmp(state, SCF_STATE_STRING_LEGACY) == 0) 1154 return ('L'); 1155 1156 return ('?'); 1157 } 1158 1159 /* Return true if inst is transitioning. */ 1160 static int 1161 transitioning(scf_instance_t *inst) 1162 { 1163 char nstate_name[MAX_SCF_STATE_STRING_SZ]; 1164 1165 get_restarter_string_prop(inst, scf_property_next_state, nstate_name, 1166 sizeof (nstate_name)); 1167 1168 return (state_to_char(nstate_name) != '?'); 1169 } 1170 1171 /* ARGSUSED */ 1172 static void 1173 sortkey_states(const char *pname, char *buf, int reverse, scf_walkinfo_t *wip) 1174 { 1175 char state_name[MAX_SCF_STATE_STRING_SZ]; 1176 1177 /* 1178 * Lower numbers are printed first, so these are arranged from least 1179 * interesting ("legacy run") to most interesting (unknown). 1180 */ 1181 if (wip->pg == NULL) { 1182 get_restarter_string_prop(wip->inst, pname, state_name, 1183 sizeof (state_name)); 1184 1185 if (strcmp(state_name, SCF_STATE_STRING_ONLINE) == 0) 1186 *buf = 2; 1187 else if (strcmp(state_name, SCF_STATE_STRING_DEGRADED) == 0) 1188 *buf = 3; 1189 else if (strcmp(state_name, SCF_STATE_STRING_OFFLINE) == 0) 1190 *buf = 4; 1191 else if (strcmp(state_name, SCF_STATE_STRING_MAINT) == 0) 1192 *buf = 5; 1193 else if (strcmp(state_name, SCF_STATE_STRING_DISABLED) == 0) 1194 *buf = 1; 1195 else if (strcmp(state_name, SCF_STATE_STRING_UNINIT) == 0) 1196 *buf = 6; 1197 else 1198 *buf = 7; 1199 } else 1200 *buf = 0; 1201 1202 if (reverse) 1203 *buf = 255 - *buf; 1204 } 1205 1206 static void 1207 sprint_state(char **buf, scf_walkinfo_t *wip) 1208 { 1209 char state_name[MAX_SCF_STATE_STRING_SZ + 1]; 1210 size_t newsize; 1211 char *newbuf; 1212 1213 if (wip->pg == NULL) { 1214 get_restarter_string_prop(wip->inst, scf_property_state, 1215 state_name, sizeof (state_name)); 1216 1217 /* Don't print blank fields, to ease parsing. */ 1218 if (state_name[0] == '\0') { 1219 state_name[0] = '-'; 1220 state_name[1] = '\0'; 1221 } 1222 1223 if (!opt_nstate_shown && transitioning(wip->inst)) { 1224 /* Append an asterisk if nstate is valid. */ 1225 (void) strcat(state_name, "*"); 1226 } 1227 } else 1228 (void) strcpy(state_name, SCF_STATE_STRING_LEGACY); 1229 1230 newsize = (*buf ? strlen(*buf) : 0) + MAX_SCF_STATE_STRING_SZ + 2; 1231 newbuf = safe_malloc(newsize); 1232 (void) snprintf(newbuf, newsize, "%s%-*s ", *buf ? *buf : "", 1233 MAX_SCF_STATE_STRING_SZ + 1, state_name); 1234 1235 if (*buf) 1236 free(*buf); 1237 *buf = newbuf; 1238 } 1239 1240 static void 1241 sortkey_state(char *buf, int reverse, scf_walkinfo_t *wip) 1242 { 1243 sortkey_states(scf_property_state, buf, reverse, wip); 1244 } 1245 1246 static void 1247 sprint_nstate(char **buf, scf_walkinfo_t *wip) 1248 { 1249 char next_state_name[MAX_SCF_STATE_STRING_SZ]; 1250 boolean_t blank = 0; 1251 size_t newsize; 1252 char *newbuf; 1253 1254 if (wip->pg == NULL) { 1255 get_restarter_string_prop(wip->inst, scf_property_next_state, 1256 next_state_name, sizeof (next_state_name)); 1257 1258 /* Don't print blank fields, to ease parsing. */ 1259 if (next_state_name[0] == '\0' || 1260 strcmp(next_state_name, SCF_STATE_STRING_NONE) == 0) 1261 blank = 1; 1262 } else 1263 blank = 1; 1264 1265 if (blank) { 1266 next_state_name[0] = '-'; 1267 next_state_name[1] = '\0'; 1268 } 1269 1270 newsize = (*buf ? strlen(*buf) : 0) + MAX_SCF_STATE_STRING_SZ + 1; 1271 newbuf = safe_malloc(newsize); 1272 (void) snprintf(newbuf, newsize, "%s%-*s ", *buf ? *buf : "", 1273 MAX_SCF_STATE_STRING_SZ - 1, next_state_name); 1274 if (*buf) 1275 free(*buf); 1276 *buf = newbuf; 1277 } 1278 1279 static void 1280 sortkey_nstate(char *buf, int reverse, scf_walkinfo_t *wip) 1281 { 1282 sortkey_states(scf_property_next_state, buf, reverse, wip); 1283 } 1284 1285 static void 1286 sprint_s(char **buf, scf_walkinfo_t *wip) 1287 { 1288 char tmp[3]; 1289 char state_name[MAX_SCF_STATE_STRING_SZ]; 1290 size_t newsize = (*buf ? strlen(*buf) : 0) + 4; 1291 char *newbuf = safe_malloc(newsize); 1292 1293 if (wip->pg == NULL) { 1294 get_restarter_string_prop(wip->inst, scf_property_state, 1295 state_name, sizeof (state_name)); 1296 tmp[0] = state_to_char(state_name); 1297 1298 if (!opt_nstate_shown && transitioning(wip->inst)) 1299 tmp[1] = '*'; 1300 else 1301 tmp[1] = ' '; 1302 } else { 1303 tmp[0] = 'L'; 1304 tmp[1] = ' '; 1305 } 1306 tmp[2] = ' '; 1307 (void) snprintf(newbuf, newsize, "%s%-*s", *buf ? *buf : "", 1308 3, tmp); 1309 if (*buf) 1310 free(*buf); 1311 *buf = newbuf; 1312 } 1313 1314 static void 1315 sprint_n(char **buf, scf_walkinfo_t *wip) 1316 { 1317 char tmp[2]; 1318 size_t newsize = (*buf ? strlen(*buf) : 0) + 3; 1319 char *newbuf = safe_malloc(newsize); 1320 char nstate_name[MAX_SCF_STATE_STRING_SZ]; 1321 1322 if (wip->pg == NULL) { 1323 get_restarter_string_prop(wip->inst, scf_property_next_state, 1324 nstate_name, sizeof (nstate_name)); 1325 1326 if (strcmp(nstate_name, SCF_STATE_STRING_NONE) == 0) 1327 tmp[0] = '-'; 1328 else 1329 tmp[0] = state_to_char(nstate_name); 1330 } else 1331 tmp[0] = '-'; 1332 1333 (void) snprintf(newbuf, newsize, "%s%-*s ", *buf ? *buf : "", 1334 2, tmp); 1335 if (*buf) 1336 free(*buf); 1337 *buf = newbuf; 1338 } 1339 1340 static void 1341 sprint_sn(char **buf, scf_walkinfo_t *wip) 1342 { 1343 char tmp[3]; 1344 size_t newsize = (*buf ? strlen(*buf) : 0) + 4; 1345 char *newbuf = safe_malloc(newsize); 1346 char nstate_name[MAX_SCF_STATE_STRING_SZ]; 1347 char state_name[MAX_SCF_STATE_STRING_SZ]; 1348 1349 if (wip->pg == NULL) { 1350 get_restarter_string_prop(wip->inst, scf_property_state, 1351 state_name, sizeof (state_name)); 1352 get_restarter_string_prop(wip->inst, scf_property_next_state, 1353 nstate_name, sizeof (nstate_name)); 1354 tmp[0] = state_to_char(state_name); 1355 1356 if (strcmp(nstate_name, SCF_STATE_STRING_NONE) == 0) 1357 tmp[1] = '-'; 1358 else 1359 tmp[1] = state_to_char(nstate_name); 1360 } else { 1361 tmp[0] = 'L'; 1362 tmp[1] = '-'; 1363 } 1364 1365 tmp[2] = ' '; 1366 (void) snprintf(newbuf, newsize, "%s%-*s ", *buf ? *buf : "", 1367 3, tmp); 1368 if (*buf) 1369 free(*buf); 1370 *buf = newbuf; 1371 } 1372 1373 /* ARGSUSED */ 1374 static void 1375 sortkey_sn(char *buf, int reverse, scf_walkinfo_t *wip) 1376 { 1377 sortkey_state(buf, reverse, wip); 1378 sortkey_nstate(buf + 1, reverse, wip); 1379 } 1380 1381 static const char * 1382 state_abbrev(const char *state) 1383 { 1384 if (strcmp(state, SCF_STATE_STRING_UNINIT) == 0) 1385 return ("UN"); 1386 if (strcmp(state, SCF_STATE_STRING_OFFLINE) == 0) 1387 return ("OFF"); 1388 if (strcmp(state, SCF_STATE_STRING_ONLINE) == 0) 1389 return ("ON"); 1390 if (strcmp(state, SCF_STATE_STRING_MAINT) == 0) 1391 return ("MNT"); 1392 if (strcmp(state, SCF_STATE_STRING_DISABLED) == 0) 1393 return ("DIS"); 1394 if (strcmp(state, SCF_STATE_STRING_DEGRADED) == 0) 1395 return ("DGD"); 1396 if (strcmp(state, SCF_STATE_STRING_LEGACY) == 0) 1397 return ("LRC"); 1398 1399 return ("?"); 1400 } 1401 1402 static void 1403 sprint_sta(char **buf, scf_walkinfo_t *wip) 1404 { 1405 char state_name[MAX_SCF_STATE_STRING_SZ]; 1406 char sta[5]; 1407 size_t newsize = (*buf ? strlen(*buf) : 0) + 6; 1408 char *newbuf = safe_malloc(newsize); 1409 1410 if (wip->pg == NULL) 1411 get_restarter_string_prop(wip->inst, scf_property_state, 1412 state_name, sizeof (state_name)); 1413 else 1414 (void) strcpy(state_name, SCF_STATE_STRING_LEGACY); 1415 1416 (void) strcpy(sta, state_abbrev(state_name)); 1417 1418 if (wip->pg == NULL && !opt_nstate_shown && transitioning(wip->inst)) 1419 (void) strcat(sta, "*"); 1420 1421 (void) snprintf(newbuf, newsize, "%s%-4s ", *buf ? *buf : "", sta); 1422 if (*buf) 1423 free(*buf); 1424 *buf = newbuf; 1425 } 1426 1427 static void 1428 sprint_nsta(char **buf, scf_walkinfo_t *wip) 1429 { 1430 char state_name[MAX_SCF_STATE_STRING_SZ]; 1431 size_t newsize = (*buf ? strlen(*buf) : 0) + 6; 1432 char *newbuf = safe_malloc(newsize); 1433 1434 if (wip->pg == NULL) 1435 get_restarter_string_prop(wip->inst, scf_property_next_state, 1436 state_name, sizeof (state_name)); 1437 else 1438 (void) strcpy(state_name, SCF_STATE_STRING_NONE); 1439 1440 if (strcmp(state_name, SCF_STATE_STRING_NONE) == 0) 1441 (void) snprintf(newbuf, newsize, "%s%-4s ", *buf ? *buf : "", 1442 "-"); 1443 else 1444 (void) snprintf(newbuf, newsize, "%s%-4s ", *buf ? *buf : "", 1445 state_abbrev(state_name)); 1446 if (*buf) 1447 free(*buf); 1448 *buf = newbuf; 1449 } 1450 1451 /* FMRI */ 1452 #define FMRI_COLUMN_WIDTH 50 1453 static void 1454 sprint_fmri(char **buf, scf_walkinfo_t *wip) 1455 { 1456 char *fmri_buf = safe_malloc(max_scf_fmri_length + 1); 1457 size_t newsize; 1458 char *newbuf; 1459 1460 if (wip->pg == NULL) { 1461 if (scf_instance_to_fmri(wip->inst, fmri_buf, 1462 max_scf_fmri_length + 1) == -1) 1463 scfdie(); 1464 } else { 1465 (void) strcpy(fmri_buf, SCF_FMRI_LEGACY_PREFIX); 1466 if (pg_get_single_val(wip->pg, SCF_LEGACY_PROPERTY_NAME, 1467 SCF_TYPE_ASTRING, fmri_buf + 1468 sizeof (SCF_FMRI_LEGACY_PREFIX) - 1, 1469 max_scf_fmri_length + 1 - 1470 (sizeof (SCF_FMRI_LEGACY_PREFIX) - 1), 0) != 0) 1471 (void) strcat(fmri_buf, LEGACY_UNKNOWN); 1472 } 1473 1474 if (strlen(fmri_buf) > FMRI_COLUMN_WIDTH) 1475 newsize = (*buf ? strlen(*buf) : 0) + strlen(fmri_buf) + 2; 1476 else 1477 newsize = (*buf ? strlen(*buf) : 0) + FMRI_COLUMN_WIDTH + 2; 1478 newbuf = safe_malloc(newsize); 1479 (void) snprintf(newbuf, newsize, "%s%-*s ", *buf ? *buf : "", 1480 FMRI_COLUMN_WIDTH, fmri_buf); 1481 free(fmri_buf); 1482 if (*buf) 1483 free(*buf); 1484 *buf = newbuf; 1485 } 1486 1487 static void 1488 sortkey_fmri(char *buf, int reverse, scf_walkinfo_t *wip) 1489 { 1490 char *tmp = NULL; 1491 1492 sprint_fmri(&tmp, wip); 1493 bcopy(tmp, buf, FMRI_COLUMN_WIDTH); 1494 free(tmp); 1495 if (reverse) 1496 reverse_bytes(buf, FMRI_COLUMN_WIDTH); 1497 } 1498 1499 /* Component columns */ 1500 #define COMPONENT_COLUMN_WIDTH 20 1501 static void 1502 sprint_scope(char **buf, scf_walkinfo_t *wip) 1503 { 1504 char *scope_buf = safe_malloc(max_scf_name_length + 1); 1505 size_t newsize = (*buf ? strlen(*buf) : 0) + COMPONENT_COLUMN_WIDTH + 2; 1506 char *newbuf = safe_malloc(newsize); 1507 1508 assert(wip->scope != NULL); 1509 1510 if (scf_scope_get_name(wip->scope, scope_buf, max_scf_name_length) < 0) 1511 scfdie(); 1512 1513 (void) snprintf(newbuf, newsize, "%s%-*s ", *buf ? *buf : "", 1514 COMPONENT_COLUMN_WIDTH, scope_buf); 1515 if (*buf) 1516 free(*buf); 1517 *buf = newbuf; 1518 free(scope_buf); 1519 } 1520 1521 static void 1522 sortkey_scope(char *buf, int reverse, scf_walkinfo_t *wip) 1523 { 1524 char *tmp = NULL; 1525 1526 sprint_scope(&tmp, wip); 1527 bcopy(tmp, buf, COMPONENT_COLUMN_WIDTH); 1528 free(tmp); 1529 if (reverse) 1530 reverse_bytes(buf, COMPONENT_COLUMN_WIDTH); 1531 } 1532 1533 static void 1534 sprint_service(char **buf, scf_walkinfo_t *wip) 1535 { 1536 char *svc_buf = safe_malloc(max_scf_name_length + 1); 1537 char *newbuf; 1538 size_t newsize; 1539 1540 if (wip->pg == NULL) { 1541 if (scf_service_get_name(wip->svc, svc_buf, 1542 max_scf_name_length + 1) < 0) 1543 scfdie(); 1544 } else { 1545 if (pg_get_single_val(wip->pg, "name", SCF_TYPE_ASTRING, 1546 svc_buf, max_scf_name_length + 1, EMPTY_OK) != 0) 1547 (void) strcpy(svc_buf, LEGACY_UNKNOWN); 1548 } 1549 1550 1551 if (strlen(svc_buf) > COMPONENT_COLUMN_WIDTH) 1552 newsize = (*buf ? strlen(*buf) : 0) + strlen(svc_buf) + 2; 1553 else 1554 newsize = (*buf ? strlen(*buf) : 0) + 1555 COMPONENT_COLUMN_WIDTH + 2; 1556 newbuf = safe_malloc(newsize); 1557 (void) snprintf(newbuf, newsize, "%s%-*s ", *buf ? *buf : "", 1558 COMPONENT_COLUMN_WIDTH, svc_buf); 1559 free(svc_buf); 1560 if (*buf) 1561 free(*buf); 1562 *buf = newbuf; 1563 } 1564 1565 static void 1566 sortkey_service(char *buf, int reverse, scf_walkinfo_t *wip) 1567 { 1568 char *tmp = NULL; 1569 1570 sprint_service(&tmp, wip); 1571 bcopy(tmp, buf, COMPONENT_COLUMN_WIDTH); 1572 free(tmp); 1573 if (reverse) 1574 reverse_bytes(buf, COMPONENT_COLUMN_WIDTH); 1575 } 1576 1577 /* INST */ 1578 static void 1579 sprint_instance(char **buf, scf_walkinfo_t *wip) 1580 { 1581 char *tmp = safe_malloc(max_scf_name_length + 1); 1582 size_t newsize = (*buf ? strlen(*buf) : 0) + COMPONENT_COLUMN_WIDTH + 2; 1583 char *newbuf = safe_malloc(newsize); 1584 1585 if (wip->pg == NULL) { 1586 if (scf_instance_get_name(wip->inst, tmp, 1587 max_scf_name_length + 1) < 0) 1588 scfdie(); 1589 } else { 1590 tmp[0] = '-'; 1591 tmp[1] = '\0'; 1592 } 1593 1594 (void) snprintf(newbuf, newsize, "%s%-*s ", *buf ? *buf : "", 1595 COMPONENT_COLUMN_WIDTH, tmp); 1596 if (*buf) 1597 free(*buf); 1598 *buf = newbuf; 1599 free(tmp); 1600 } 1601 1602 static void 1603 sortkey_instance(char *buf, int reverse, scf_walkinfo_t *wip) 1604 { 1605 char *tmp = NULL; 1606 1607 sprint_instance(&tmp, wip); 1608 bcopy(tmp, buf, COMPONENT_COLUMN_WIDTH); 1609 free(tmp); 1610 if (reverse) 1611 reverse_bytes(buf, COMPONENT_COLUMN_WIDTH); 1612 } 1613 1614 /* STIME */ 1615 #define STIME_COLUMN_WIDTH 8 1616 #define FORMAT_TIME "%k:%M:%S" 1617 #define FORMAT_DATE "%b_%d " 1618 #define FORMAT_YEAR "%Y " 1619 1620 /* 1621 * sprint_stime() will allocate a new buffer and snprintf the services's 1622 * state timestamp. If the timestamp is unavailable for some reason 1623 * a '-' is given instead. 1624 */ 1625 static void 1626 sprint_stime(char **buf, scf_walkinfo_t *wip) 1627 { 1628 int r; 1629 struct timeval tv; 1630 time_t then; 1631 struct tm *tm; 1632 char st_buf[STIME_COLUMN_WIDTH + 1]; 1633 size_t newsize = (*buf ? strlen(*buf) : 0) + STIME_COLUMN_WIDTH + 2; 1634 char *newbuf = safe_malloc(newsize); 1635 1636 if (wip->pg == NULL) { 1637 r = get_restarter_time_prop(wip->inst, 1638 SCF_PROPERTY_STATE_TIMESTAMP, &tv, 0); 1639 } else { 1640 r = pg_get_single_val(wip->pg, SCF_PROPERTY_STATE_TIMESTAMP, 1641 SCF_TYPE_TIME, &tv, 0, 0); 1642 } 1643 1644 if (r != 0) { 1645 /* 1646 * There's something amiss with our service 1647 * so we'll print a '-' for STIME. 1648 */ 1649 (void) snprintf(newbuf, newsize, "%s%-*s", *buf ? *buf : "", 1650 STIME_COLUMN_WIDTH + 1, "-"); 1651 } else { 1652 /* tv should be valid so we'll format it */ 1653 then = (time_t)tv.tv_sec; 1654 1655 tm = localtime(&then); 1656 /* 1657 * Print time if started within the past 24 hours, print date 1658 * if within the past 12 months or, finally, print year if 1659 * started greater than 12 months ago. 1660 */ 1661 if (now - then < 24 * 60 * 60) { 1662 (void) strftime(st_buf, sizeof (st_buf), 1663 gettext(FORMAT_TIME), tm); 1664 } else if (now - then < 12 * 30 * 24 * 60 * 60) { 1665 (void) strftime(st_buf, sizeof (st_buf), 1666 gettext(FORMAT_DATE), tm); 1667 } else { 1668 (void) strftime(st_buf, sizeof (st_buf), 1669 gettext(FORMAT_YEAR), tm); 1670 } 1671 (void) snprintf(newbuf, newsize, "%s%-*s ", *buf ? *buf : "", 1672 STIME_COLUMN_WIDTH + 1, st_buf); 1673 } 1674 if (*buf) 1675 free(*buf); 1676 *buf = newbuf; 1677 } 1678 1679 #define STIME_SORTKEY_WIDTH (sizeof (uint64_t) + sizeof (uint32_t)) 1680 1681 /* ARGSUSED */ 1682 static void 1683 sortkey_stime(char *buf, int reverse, scf_walkinfo_t *wip) 1684 { 1685 struct timeval tv; 1686 int r; 1687 1688 if (wip->pg == NULL) 1689 r = get_restarter_time_prop(wip->inst, 1690 SCF_PROPERTY_STATE_TIMESTAMP, &tv, 0); 1691 else 1692 r = pg_get_single_val(wip->pg, SCF_PROPERTY_STATE_TIMESTAMP, 1693 SCF_TYPE_TIME, &tv, 0, 0); 1694 1695 if (r == 0) { 1696 int64_t sec; 1697 int32_t us; 1698 1699 /* Stick it straight into the buffer. */ 1700 sec = tv.tv_sec; 1701 us = tv.tv_usec; 1702 1703 sec = BE_64(sec); 1704 us = BE_32(us); 1705 bcopy(&sec, buf, sizeof (sec)); 1706 bcopy(&us, buf + sizeof (sec), sizeof (us)); 1707 } else { 1708 bzero(buf, STIME_SORTKEY_WIDTH); 1709 } 1710 1711 if (reverse) 1712 reverse_bytes(buf, STIME_SORTKEY_WIDTH); 1713 } 1714 1715 /* ZONE */ 1716 #define ZONE_COLUMN_WIDTH 16 1717 /*ARGSUSED*/ 1718 static void 1719 sprint_zone(char **buf, scf_walkinfo_t *wip) 1720 { 1721 size_t newsize; 1722 char *newbuf, *zonename = g_zonename, b[ZONENAME_MAX]; 1723 1724 if (zonename == NULL) { 1725 zoneid_t zoneid = getzoneid(); 1726 1727 if (getzonenamebyid(zoneid, b, sizeof (b)) < 0) 1728 uu_die(gettext("could not determine zone name")); 1729 1730 zonename = b; 1731 } 1732 1733 if (strlen(zonename) > ZONE_COLUMN_WIDTH) 1734 newsize = (*buf ? strlen(*buf) : 0) + strlen(zonename) + 2; 1735 else 1736 newsize = (*buf ? strlen(*buf) : 0) + ZONE_COLUMN_WIDTH + 2; 1737 1738 newbuf = safe_malloc(newsize); 1739 (void) snprintf(newbuf, newsize, "%s%-*s ", *buf ? *buf : "", 1740 ZONE_COLUMN_WIDTH, zonename); 1741 1742 if (*buf) 1743 free(*buf); 1744 *buf = newbuf; 1745 } 1746 1747 static void 1748 sortkey_zone(char *buf, int reverse, scf_walkinfo_t *wip) 1749 { 1750 char *tmp = NULL; 1751 1752 sprint_zone(&tmp, wip); 1753 bcopy(tmp, buf, ZONE_COLUMN_WIDTH); 1754 free(tmp); 1755 if (reverse) 1756 reverse_bytes(buf, ZONE_COLUMN_WIDTH); 1757 } 1758 1759 /* 1760 * Information about columns which can be displayed. If you add something, 1761 * check MAX_COLUMN_NAME_LENGTH_STR & update description_of_column() below. 1762 */ 1763 static const struct column columns[] = { 1764 { "CTID", CTID_COLUMN_WIDTH, sprint_ctid, 1765 CTID_SORTKEY_WIDTH, sortkey_ctid }, 1766 { "DESC", DESC_COLUMN_WIDTH, sprint_desc, 1767 DESC_COLUMN_WIDTH, sortkey_desc }, 1768 { "FMRI", FMRI_COLUMN_WIDTH, sprint_fmri, 1769 FMRI_COLUMN_WIDTH, sortkey_fmri }, 1770 { "INST", COMPONENT_COLUMN_WIDTH, sprint_instance, 1771 COMPONENT_COLUMN_WIDTH, sortkey_instance }, 1772 { "N", 1, sprint_n, 1, sortkey_nstate }, 1773 { "NSTA", 4, sprint_nsta, 1, sortkey_nstate }, 1774 { "NSTATE", MAX_SCF_STATE_STRING_SZ - 1, sprint_nstate, 1775 1, sortkey_nstate }, 1776 { "S", 2, sprint_s, 1, sortkey_state }, 1777 { "SCOPE", COMPONENT_COLUMN_WIDTH, sprint_scope, 1778 COMPONENT_COLUMN_WIDTH, sortkey_scope }, 1779 { "SN", 2, sprint_sn, 2, sortkey_sn }, 1780 { "SVC", COMPONENT_COLUMN_WIDTH, sprint_service, 1781 COMPONENT_COLUMN_WIDTH, sortkey_service }, 1782 { "STA", 4, sprint_sta, 1, sortkey_state }, 1783 { "STATE", MAX_SCF_STATE_STRING_SZ - 1 + 1, sprint_state, 1784 1, sortkey_state }, 1785 { "STIME", STIME_COLUMN_WIDTH, sprint_stime, 1786 STIME_SORTKEY_WIDTH, sortkey_stime }, 1787 { "ZONE", ZONE_COLUMN_WIDTH, sprint_zone, 1788 ZONE_COLUMN_WIDTH, sortkey_zone }, 1789 }; 1790 1791 #define MAX_COLUMN_NAME_LENGTH_STR "6" 1792 1793 static const int ncolumns = sizeof (columns) / sizeof (columns[0]); 1794 1795 /* 1796 * Necessary thanks to gettext() & xgettext. 1797 */ 1798 static const char * 1799 description_of_column(int c) 1800 { 1801 const char *s = NULL; 1802 1803 switch (c) { 1804 case 0: 1805 s = gettext("contract ID for service (see contract(5))"); 1806 break; 1807 case 1: 1808 s = gettext("human-readable description of the service"); 1809 break; 1810 case 2: 1811 s = gettext("Fault Managed Resource Identifier for service"); 1812 break; 1813 case 3: 1814 s = gettext("portion of the FMRI indicating service instance"); 1815 break; 1816 case 4: 1817 s = gettext("abbreviation for next state (if in transition)"); 1818 break; 1819 case 5: 1820 s = gettext("abbreviation for next state (if in transition)"); 1821 break; 1822 case 6: 1823 s = gettext("name for next state (if in transition)"); 1824 break; 1825 case 7: 1826 s = gettext("abbreviation for current state"); 1827 break; 1828 case 8: 1829 s = gettext("name for scope associated with service"); 1830 break; 1831 case 9: 1832 s = gettext("abbreviation for current state and next state"); 1833 break; 1834 case 10: 1835 s = gettext("portion of the FMRI representing service name"); 1836 break; 1837 case 11: 1838 s = gettext("abbreviation for current state"); 1839 break; 1840 case 12: 1841 s = gettext("name for current state"); 1842 break; 1843 case 13: 1844 s = gettext("time of last state change"); 1845 break; 1846 case 14: 1847 s = gettext("name of zone"); 1848 break; 1849 } 1850 1851 assert(s != NULL); 1852 return (s); 1853 } 1854 1855 1856 static void 1857 print_usage(const char *progname, FILE *f) 1858 { 1859 (void) fprintf(f, gettext( 1860 "Usage: %1$s [-aHpv] [-o col[,col ... ]] [-R restarter] " 1861 "[-sS col] [-Z | -z zone ]\n [<service> ...]\n" 1862 " %1$s -d | -D [-Hpv] [-o col[,col ... ]] [-sS col] " 1863 "[-Z | -z zone ]\n [<service> ...]\n" 1864 " %1$s [-l | -L] [-Z | -z zone] <service> ...\n" 1865 " %1$s -x [-v] [-Z | -z zone] [<service> ...]\n" 1866 " %1$s -?\n"), progname); 1867 } 1868 1869 static __NORETURN void 1870 argserr(const char *progname) 1871 { 1872 print_usage(progname, stderr); 1873 exit(UU_EXIT_USAGE); 1874 } 1875 1876 static void 1877 print_help(const char *progname) 1878 { 1879 int i; 1880 1881 print_usage(progname, stdout); 1882 1883 (void) printf(gettext("\n" 1884 "\t-a list all service instances rather than " 1885 "only those that are enabled\n" 1886 "\t-d list dependencies of the specified service(s)\n" 1887 "\t-D list dependents of the specified service(s)\n" 1888 "\t-H omit header line from output\n" 1889 "\t-l list detailed information about the specified service(s)\n" 1890 "\t-L list the log file associated with the specified service(s)\n" 1891 "\t-o list only the specified columns in the output\n" 1892 "\t-p list process IDs and names associated with each service\n" 1893 "\t-R list only those services with the specified restarter\n" 1894 "\t-s sort output in ascending order by the specified column(s)\n" 1895 "\t-S sort output in descending order by the specified column(s)\n" 1896 "\t-v list verbose information appropriate to the type of output\n" 1897 "\t-x explain the status of services that might require maintenance,\n" 1898 "\t or explain the status of the specified service(s)\n" 1899 "\t-z from global zone, show services in a specified zone\n" 1900 "\t-Z from global zone, show services in all zones\n" 1901 "\n\t" 1902 "Services can be specified using an FMRI, abbreviation, or fnmatch(7)\n" 1903 "\tpattern, as shown in these examples for svc:/network/smtp:sendmail\n" 1904 "\n" 1905 "\t%1$s [opts] svc:/network/smtp:sendmail\n" 1906 "\t%1$s [opts] network/smtp:sendmail\n" 1907 "\t%1$s [opts] network/*mail\n" 1908 "\t%1$s [opts] network/smtp\n" 1909 "\t%1$s [opts] smtp:sendmail\n" 1910 "\t%1$s [opts] smtp\n" 1911 "\t%1$s [opts] sendmail\n" 1912 "\n\t" 1913 "Columns for output or sorting can be specified using these names:\n" 1914 "\n"), progname); 1915 1916 for (i = 0; i < ncolumns; i++) { 1917 (void) printf("\t%-" MAX_COLUMN_NAME_LENGTH_STR "s %s\n", 1918 columns[i].name, description_of_column(i)); 1919 } 1920 } 1921 1922 1923 /* 1924 * A getsubopt()-like function which returns an index into the columns table. 1925 * On success, *optionp is set to point to the next sub-option, or the 1926 * terminating null if there are none. 1927 */ 1928 static int 1929 getcolumnopt(char **optionp) 1930 { 1931 char *str = *optionp, *cp; 1932 int i; 1933 1934 assert(optionp != NULL); 1935 assert(*optionp != NULL); 1936 1937 cp = strchr(*optionp, ','); 1938 if (cp != NULL) 1939 *cp = '\0'; 1940 1941 for (i = 0; i < ncolumns; ++i) { 1942 if (strcasecmp(str, columns[i].name) == 0) { 1943 if (cp != NULL) 1944 *optionp = cp + 1; 1945 else 1946 *optionp = strchr(*optionp, '\0'); 1947 1948 return (i); 1949 } 1950 } 1951 1952 return (-1); 1953 } 1954 1955 static void 1956 print_header() 1957 { 1958 int i; 1959 char *line_buf, *cp; 1960 1961 line_buf = safe_malloc(line_sz); 1962 cp = line_buf; 1963 for (i = 0; i < opt_cnum; ++i) { 1964 const struct column * const colp = &columns[opt_columns[i]]; 1965 1966 (void) snprintf(cp, colp->width + 1, "%-*s", colp->width, 1967 colp->name); 1968 cp += colp->width; 1969 *cp++ = ' '; 1970 } 1971 1972 /* Trim the trailing whitespace */ 1973 --cp; 1974 while (*cp == ' ') 1975 --cp; 1976 *(cp+1) = '\0'; 1977 (void) puts(line_buf); 1978 1979 free(line_buf); 1980 } 1981 1982 1983 1984 /* 1985 * Long listing (-l) functions. 1986 */ 1987 1988 static int 1989 pidcmp(const void *l, const void *r) 1990 { 1991 pid_t lp = *(pid_t *)l, rp = *(pid_t *)r; 1992 1993 if (lp < rp) 1994 return (-1); 1995 if (lp > rp) 1996 return (1); 1997 return (0); 1998 } 1999 2000 /* 2001 * This is the strlen() of the longest label ("description"), plus intercolumn 2002 * space. 2003 */ 2004 #define DETAILED_WIDTH (11 + 2) 2005 2006 /* 2007 * Callback routine to print header for contract id. 2008 * Called by ctids_by_restarter and print_detailed. 2009 */ 2010 static void 2011 print_ctid_header() 2012 { 2013 (void) printf("%-*s", DETAILED_WIDTH, "contract_id"); 2014 } 2015 2016 /* 2017 * Callback routine to print a contract id. 2018 * Called by ctids_by_restarter and print_detailed. 2019 */ 2020 static void 2021 print_ctid_detailed(uint64_t c) 2022 { 2023 (void) printf("%lu ", (ctid_t)c); 2024 } 2025 2026 static void 2027 detailed_list_processes(scf_walkinfo_t *wip) 2028 { 2029 uint64_t c; 2030 pid_t *pids; 2031 uint_t i, n; 2032 psinfo_t psi; 2033 2034 if (get_restarter_count_prop(wip->inst, scf_property_contract, &c, 2035 EMPTY_OK) != 0) 2036 return; 2037 2038 if (instance_processes(wip->inst, wip->fmri, &pids, &n) != 0) 2039 return; 2040 2041 qsort(pids, n, sizeof (*pids), pidcmp); 2042 2043 for (i = 0; i < n; ++i) { 2044 (void) printf("%-*s%lu", DETAILED_WIDTH, gettext("process"), 2045 pids[i]); 2046 2047 if (proc_get_psinfo(pids[i], &psi) == 0 && !IS_ZOMBIE(&psi)) 2048 (void) printf(" %.*s", PRARGSZ, psi.pr_psargs); 2049 2050 (void) putchar('\n'); 2051 } 2052 2053 free(pids); 2054 } 2055 2056 /* 2057 * Determines the state of a dependency. If the FMRI specifies a file, then we 2058 * fake up a state based on whether we can access the file. 2059 */ 2060 static void 2061 get_fmri_state(char *fmri, char *state, size_t state_sz) 2062 { 2063 char *lfmri; 2064 const char *svc_name, *inst_name, *pg_name, *path; 2065 scf_service_t *svc; 2066 scf_instance_t *inst; 2067 scf_iter_t *iter; 2068 2069 lfmri = safe_strdup(fmri); 2070 2071 /* 2072 * Check for file:// dependencies 2073 */ 2074 if (scf_parse_file_fmri(lfmri, NULL, &path) == SCF_SUCCESS) { 2075 struct stat64 statbuf; 2076 const char *msg; 2077 2078 if (stat64(path, &statbuf) == 0) 2079 msg = "online"; 2080 else if (errno == ENOENT) 2081 msg = "absent"; 2082 else 2083 msg = "unknown"; 2084 2085 (void) strlcpy(state, msg, state_sz); 2086 return; 2087 } 2088 2089 /* 2090 * scf_parse_file_fmri() may have overwritten part of the string, so 2091 * copy it back. 2092 */ 2093 (void) strcpy(lfmri, fmri); 2094 2095 if (scf_parse_svc_fmri(lfmri, NULL, &svc_name, &inst_name, 2096 &pg_name, NULL) != SCF_SUCCESS) { 2097 free(lfmri); 2098 (void) strlcpy(state, "invalid", state_sz); 2099 return; 2100 } 2101 2102 free(lfmri); 2103 2104 if (svc_name == NULL || pg_name != NULL) { 2105 (void) strlcpy(state, "invalid", state_sz); 2106 return; 2107 } 2108 2109 if (inst_name != NULL) { 2110 /* instance: get state */ 2111 inst = scf_instance_create(h); 2112 if (inst == NULL) 2113 scfdie(); 2114 2115 if (scf_handle_decode_fmri(h, fmri, NULL, NULL, inst, NULL, 2116 NULL, SCF_DECODE_FMRI_EXACT) == SCF_SUCCESS) 2117 get_restarter_string_prop(inst, scf_property_state, 2118 state, state_sz); 2119 else { 2120 switch (scf_error()) { 2121 case SCF_ERROR_INVALID_ARGUMENT: 2122 (void) strlcpy(state, "invalid", state_sz); 2123 break; 2124 case SCF_ERROR_NOT_FOUND: 2125 (void) strlcpy(state, "absent", state_sz); 2126 break; 2127 2128 default: 2129 scfdie(); 2130 } 2131 } 2132 2133 scf_instance_destroy(inst); 2134 return; 2135 } 2136 2137 /* 2138 * service: If only one instance, use that state. Otherwise, say 2139 * "multiple". 2140 */ 2141 if ((svc = scf_service_create(h)) == NULL || 2142 (inst = scf_instance_create(h)) == NULL || 2143 (iter = scf_iter_create(h)) == NULL) 2144 scfdie(); 2145 2146 if (scf_handle_decode_fmri(h, fmri, NULL, svc, NULL, NULL, NULL, 2147 SCF_DECODE_FMRI_EXACT) != SCF_SUCCESS) { 2148 switch (scf_error()) { 2149 case SCF_ERROR_INVALID_ARGUMENT: 2150 (void) strlcpy(state, "invalid", state_sz); 2151 goto out; 2152 case SCF_ERROR_NOT_FOUND: 2153 (void) strlcpy(state, "absent", state_sz); 2154 goto out; 2155 2156 default: 2157 scfdie(); 2158 } 2159 } 2160 2161 if (scf_iter_service_instances(iter, svc) != SCF_SUCCESS) 2162 scfdie(); 2163 2164 switch (scf_iter_next_instance(iter, inst)) { 2165 case 0: 2166 (void) strlcpy(state, "absent", state_sz); 2167 goto out; 2168 2169 case 1: 2170 break; 2171 2172 default: 2173 scfdie(); 2174 } 2175 2176 /* Get the state in case this is the only instance. */ 2177 get_restarter_string_prop(inst, scf_property_state, state, state_sz); 2178 2179 switch (scf_iter_next_instance(iter, inst)) { 2180 case 0: 2181 break; 2182 2183 case 1: 2184 /* Nope, multiple instances. */ 2185 (void) strlcpy(state, "multiple", state_sz); 2186 goto out; 2187 2188 default: 2189 scfdie(); 2190 } 2191 2192 out: 2193 scf_iter_destroy(iter); 2194 scf_instance_destroy(inst); 2195 scf_service_destroy(svc); 2196 } 2197 2198 static void 2199 print_application_properties(scf_walkinfo_t *wip, scf_snapshot_t *snap) 2200 { 2201 scf_iter_t *pg_iter, *prop_iter, *val_iter; 2202 scf_propertygroup_t *pg; 2203 scf_property_t *prop; 2204 scf_value_t *val; 2205 scf_pg_tmpl_t *pt; 2206 scf_prop_tmpl_t *prt; 2207 char *pg_name_buf = safe_malloc(max_scf_name_length + 1); 2208 char *prop_name_buf = safe_malloc(max_scf_name_length + 1); 2209 char *snap_name = safe_malloc(max_scf_name_length + 1); 2210 char *val_buf = safe_malloc(max_scf_value_length + 1); 2211 char *desc, *cp; 2212 scf_type_t type; 2213 int i, j, k; 2214 uint8_t vis; 2215 2216 if ((pg_iter = scf_iter_create(h)) == NULL || 2217 (prop_iter = scf_iter_create(h)) == NULL || 2218 (val_iter = scf_iter_create(h)) == NULL || 2219 (val = scf_value_create(h)) == NULL || 2220 (prop = scf_property_create(h)) == NULL || 2221 (pt = scf_tmpl_pg_create(h)) == NULL || 2222 (prt = scf_tmpl_prop_create(h)) == NULL || 2223 (pg = scf_pg_create(h)) == NULL) 2224 scfdie(); 2225 2226 if (scf_iter_instance_pgs_typed_composed(pg_iter, wip->inst, snap, 2227 SCF_PG_APP_DEFAULT) == -1) 2228 scfdie(); 2229 2230 /* 2231 * Format for output: 2232 * pg (pgtype) 2233 * description 2234 * pg/prop (proptype) = <value> <value> 2235 * description 2236 */ 2237 while ((i = scf_iter_next_pg(pg_iter, pg)) == 1) { 2238 int tmpl = 0; 2239 2240 if (scf_pg_get_name(pg, pg_name_buf, max_scf_name_length) < 0) 2241 scfdie(); 2242 if (scf_snapshot_get_name(snap, snap_name, 2243 max_scf_name_length) < 0) 2244 scfdie(); 2245 2246 if (scf_tmpl_get_by_pg_name(wip->fmri, snap_name, pg_name_buf, 2247 SCF_PG_APP_DEFAULT, pt, 0) == 0) 2248 tmpl = 1; 2249 else 2250 tmpl = 0; 2251 2252 (void) printf("%s (%s)\n", pg_name_buf, SCF_PG_APP_DEFAULT); 2253 2254 if (tmpl == 1 && scf_tmpl_pg_description(pt, NULL, &desc) > 0) { 2255 (void) printf(" %s\n", desc); 2256 free(desc); 2257 } 2258 2259 if (scf_iter_pg_properties(prop_iter, pg) == -1) 2260 scfdie(); 2261 while ((j = scf_iter_next_property(prop_iter, prop)) == 1) { 2262 if (scf_property_get_name(prop, prop_name_buf, 2263 max_scf_name_length) < 0) 2264 scfdie(); 2265 if (scf_property_type(prop, &type) == -1) 2266 scfdie(); 2267 2268 if ((tmpl == 1) && 2269 (scf_tmpl_get_by_prop(pt, prop_name_buf, prt, 2270 0) != 0)) 2271 tmpl = 0; 2272 2273 if (tmpl == 1 && 2274 scf_tmpl_prop_visibility(prt, &vis) != -1 && 2275 vis == SCF_TMPL_VISIBILITY_HIDDEN) 2276 continue; 2277 2278 (void) printf("%s/%s (%s) = ", pg_name_buf, 2279 prop_name_buf, scf_type_to_string(type)); 2280 2281 if (scf_iter_property_values(val_iter, prop) == -1) 2282 scfdie(); 2283 2284 while ((k = scf_iter_next_value(val_iter, val)) == 1) { 2285 if (scf_value_get_as_string(val, val_buf, 2286 max_scf_value_length + 1) < 0) 2287 scfdie(); 2288 if (strpbrk(val_buf, " \t\n\"()") != NULL) { 2289 (void) printf("\""); 2290 for (cp = val_buf; *cp != '\0'; ++cp) { 2291 if (*cp == '"' || *cp == '\\') 2292 (void) putc('\\', 2293 stdout); 2294 2295 (void) putc(*cp, stdout); 2296 } 2297 (void) printf("\""); 2298 } else { 2299 (void) printf("%s ", val_buf); 2300 } 2301 } 2302 2303 (void) printf("\n"); 2304 2305 if (k == -1) 2306 scfdie(); 2307 2308 if (tmpl == 1 && scf_tmpl_prop_description(prt, NULL, 2309 &desc) > 0) { 2310 (void) printf(" %s\n", desc); 2311 free(desc); 2312 } 2313 } 2314 if (j == -1) 2315 scfdie(); 2316 } 2317 if (i == -1) 2318 scfdie(); 2319 2320 2321 scf_iter_destroy(pg_iter); 2322 scf_iter_destroy(prop_iter); 2323 scf_iter_destroy(val_iter); 2324 scf_value_destroy(val); 2325 scf_property_destroy(prop); 2326 scf_tmpl_pg_destroy(pt); 2327 scf_tmpl_prop_destroy(prt); 2328 scf_pg_destroy(pg); 2329 free(pg_name_buf); 2330 free(prop_name_buf); 2331 free(snap_name); 2332 free(val_buf); 2333 } 2334 2335 static void 2336 print_detailed_dependency(scf_propertygroup_t *pg) 2337 { 2338 scf_property_t *eprop; 2339 scf_iter_t *iter; 2340 scf_type_t ty; 2341 char *val_buf; 2342 int i; 2343 2344 if ((eprop = scf_property_create(h)) == NULL || 2345 (iter = scf_iter_create(h)) == NULL) 2346 scfdie(); 2347 2348 val_buf = safe_malloc(max_scf_value_length + 1); 2349 2350 if (scf_pg_get_property(pg, SCF_PROPERTY_ENTITIES, eprop) != 2351 SCF_SUCCESS || 2352 scf_property_type(eprop, &ty) != SCF_SUCCESS || 2353 ty != SCF_TYPE_FMRI) 2354 return; 2355 2356 (void) printf("%-*s", DETAILED_WIDTH, gettext("dependency")); 2357 2358 /* Print the grouping */ 2359 if (pg_get_single_val(pg, SCF_PROPERTY_GROUPING, SCF_TYPE_ASTRING, 2360 val_buf, max_scf_value_length + 1, 0) == 0) 2361 (void) fputs(val_buf, stdout); 2362 else 2363 (void) putchar('?'); 2364 2365 (void) putchar('/'); 2366 2367 if (pg_get_single_val(pg, SCF_PROPERTY_RESTART_ON, SCF_TYPE_ASTRING, 2368 val_buf, max_scf_value_length + 1, 0) == 0) 2369 (void) fputs(val_buf, stdout); 2370 else 2371 (void) putchar('?'); 2372 2373 /* Print the dependency entities. */ 2374 if (scf_iter_property_values(iter, eprop) == -1) 2375 scfdie(); 2376 2377 while ((i = scf_iter_next_value(iter, g_val)) == 1) { 2378 char state[MAX_SCF_STATE_STRING_SZ]; 2379 2380 if (scf_value_get_astring(g_val, val_buf, 2381 max_scf_value_length + 1) < 0) 2382 scfdie(); 2383 2384 (void) putchar(' '); 2385 (void) fputs(val_buf, stdout); 2386 2387 /* Print the state. */ 2388 state[0] = '-'; 2389 state[1] = '\0'; 2390 2391 get_fmri_state(val_buf, state, sizeof (state)); 2392 2393 (void) printf(" (%s)", state); 2394 } 2395 if (i == -1) 2396 scfdie(); 2397 2398 (void) putchar('\n'); 2399 2400 free(val_buf); 2401 scf_iter_destroy(iter); 2402 scf_property_destroy(eprop); 2403 } 2404 2405 /* ARGSUSED */ 2406 static int 2407 print_detailed(void *unused, scf_walkinfo_t *wip) 2408 { 2409 scf_snapshot_t *snap; 2410 scf_propertygroup_t *rpg; 2411 scf_iter_t *pg_iter; 2412 2413 char *buf; 2414 char *timebuf; 2415 size_t tbsz; 2416 int ret; 2417 uint64_t c; 2418 int temp, perm; 2419 struct timeval tv; 2420 time_t stime; 2421 struct tm *tmp; 2422 int restarter_spec; 2423 int restarter_ret; 2424 2425 const char * const fmt = "%-*s%s\n"; 2426 2427 assert(wip->pg == NULL); 2428 2429 rpg = scf_pg_create(h); 2430 if (rpg == NULL) 2431 scfdie(); 2432 2433 if (first_paragraph) 2434 first_paragraph = 0; 2435 else 2436 (void) putchar('\n'); 2437 2438 buf = safe_malloc(max_scf_fmri_length + 1); 2439 2440 if (scf_instance_to_fmri(wip->inst, buf, max_scf_fmri_length + 1) != -1) 2441 (void) printf(fmt, DETAILED_WIDTH, "fmri", buf); 2442 2443 if (common_name_buf == NULL) 2444 common_name_buf = safe_malloc(max_scf_value_length + 1); 2445 2446 if (inst_get_single_val(wip->inst, SCF_PG_TM_COMMON_NAME, locale, 2447 SCF_TYPE_USTRING, common_name_buf, max_scf_value_length, 0, 1, 1) 2448 == 0) 2449 (void) printf(fmt, DETAILED_WIDTH, gettext("name"), 2450 common_name_buf); 2451 else if (inst_get_single_val(wip->inst, SCF_PG_TM_COMMON_NAME, "C", 2452 SCF_TYPE_USTRING, common_name_buf, max_scf_value_length, 0, 1, 1) 2453 == 0) 2454 (void) printf(fmt, DETAILED_WIDTH, gettext("name"), 2455 common_name_buf); 2456 2457 if (g_zonename != NULL) 2458 (void) printf(fmt, DETAILED_WIDTH, gettext("zone"), g_zonename); 2459 2460 /* 2461 * Synthesize an 'enabled' property that hides the enabled_ovr 2462 * implementation from the user. If the service has been temporarily 2463 * set to a state other than its permanent value, alert the user with 2464 * a '(temporary)' message. 2465 */ 2466 perm = instance_enabled(wip->inst, B_FALSE); 2467 temp = instance_enabled(wip->inst, B_TRUE); 2468 if (temp != -1) { 2469 if (temp != perm) 2470 (void) printf(gettext("%-*s%s (temporary)\n"), 2471 DETAILED_WIDTH, gettext("enabled"), 2472 temp ? gettext("true") : gettext("false")); 2473 else 2474 (void) printf(fmt, DETAILED_WIDTH, 2475 gettext("enabled"), temp ? gettext("true") : 2476 gettext("false")); 2477 } else if (perm != -1) { 2478 (void) printf(fmt, DETAILED_WIDTH, gettext("enabled"), 2479 perm ? gettext("true") : gettext("false")); 2480 } 2481 2482 if (temp == 0 || (temp == -1 && perm == 0)) { 2483 char comment[SCF_COMMENT_MAX_LENGTH] = ""; 2484 const char *pg = (temp != -1 && temp != perm) ? 2485 SCF_PG_GENERAL_OVR : SCF_PG_GENERAL; 2486 2487 (void) inst_get_single_val(wip->inst, pg, SCF_PROPERTY_COMMENT, 2488 SCF_TYPE_ASTRING, &comment, sizeof (comment), 2489 EMPTY_OK, 0, 0); 2490 2491 if (comment[0] != '\0') { 2492 printf(fmt, DETAILED_WIDTH, gettext("comment"), 2493 comment); 2494 } 2495 } 2496 2497 /* 2498 * Property values may be longer than max_scf_fmri_length, but these 2499 * shouldn't be, so we'll just reuse buf. The user can use svcprop if 2500 * they suspect something fishy. 2501 */ 2502 if (scf_instance_get_pg(wip->inst, SCF_PG_RESTARTER, rpg) != 0) { 2503 if (scf_error() != SCF_ERROR_NOT_FOUND) 2504 scfdie(); 2505 2506 scf_pg_destroy(rpg); 2507 rpg = NULL; 2508 } 2509 2510 if (rpg) { 2511 if (pg_get_single_val(rpg, scf_property_state, SCF_TYPE_ASTRING, 2512 buf, max_scf_fmri_length + 1, 0) == 0) 2513 (void) printf(fmt, DETAILED_WIDTH, gettext("state"), 2514 buf); 2515 2516 if (pg_get_single_val(rpg, scf_property_next_state, 2517 SCF_TYPE_ASTRING, buf, max_scf_fmri_length + 1, 0) == 0) 2518 (void) printf(fmt, DETAILED_WIDTH, 2519 gettext("next_state"), buf); 2520 2521 if (pg_get_single_val(rpg, SCF_PROPERTY_STATE_TIMESTAMP, 2522 SCF_TYPE_TIME, &tv, 0, 0) == 0) { 2523 stime = tv.tv_sec; 2524 tmp = localtime(&stime); 2525 for (tbsz = 50; ; tbsz *= 2) { 2526 timebuf = safe_malloc(tbsz); 2527 if (strftime(timebuf, tbsz, NULL, tmp) != 0) 2528 break; 2529 free(timebuf); 2530 } 2531 (void) printf(fmt, DETAILED_WIDTH, 2532 gettext("state_time"), 2533 timebuf); 2534 free(timebuf); 2535 } 2536 2537 if (pg_get_single_val(rpg, SCF_PROPERTY_ALT_LOGFILE, 2538 SCF_TYPE_ASTRING, buf, max_scf_fmri_length + 1, 0) == 0) 2539 (void) printf(fmt, DETAILED_WIDTH, 2540 gettext("alt_logfile"), buf); 2541 2542 if (pg_get_single_val(rpg, SCF_PROPERTY_LOGFILE, 2543 SCF_TYPE_ASTRING, buf, max_scf_fmri_length + 1, 0) == 0) 2544 (void) printf(fmt, DETAILED_WIDTH, gettext("logfile"), 2545 buf); 2546 } 2547 2548 if (inst_get_single_val(wip->inst, SCF_PG_GENERAL, 2549 SCF_PROPERTY_RESTARTER, SCF_TYPE_ASTRING, buf, 2550 max_scf_fmri_length + 1, 0, 0, 1) == 0) 2551 (void) printf(fmt, DETAILED_WIDTH, gettext("restarter"), buf); 2552 else 2553 (void) printf(fmt, DETAILED_WIDTH, gettext("restarter"), 2554 SCF_SERVICE_STARTD); 2555 2556 free(buf); 2557 2558 /* 2559 * Use the restarter specific routine to print the ctids, if available. 2560 * If restarter specific action is available and it fails, then die. 2561 */ 2562 restarter_ret = ctids_by_restarter(wip, &c, 1, 0, 2563 &restarter_spec, print_ctid_header, print_ctid_detailed); 2564 if (restarter_spec == 1) { 2565 if (restarter_ret != 0) 2566 uu_die(gettext("Unable to get restarter for %s"), 2567 wip->fmri); 2568 goto restarter_common; 2569 } 2570 2571 if (rpg) { 2572 scf_iter_t *iter; 2573 2574 if ((iter = scf_iter_create(h)) == NULL) 2575 scfdie(); 2576 2577 if (scf_pg_get_property(rpg, scf_property_contract, g_prop) == 2578 0) { 2579 if (scf_property_is_type(g_prop, SCF_TYPE_COUNT) == 0) { 2580 2581 /* Callback to print ctid header */ 2582 print_ctid_header(); 2583 2584 if (scf_iter_property_values(iter, g_prop) != 0) 2585 scfdie(); 2586 2587 for (;;) { 2588 ret = scf_iter_next_value(iter, g_val); 2589 if (ret == -1) 2590 scfdie(); 2591 if (ret == 0) 2592 break; 2593 2594 if (scf_value_get_count(g_val, &c) != 0) 2595 scfdie(); 2596 2597 /* Callback to print contract id. */ 2598 print_ctid_detailed(c); 2599 } 2600 2601 (void) putchar('\n'); 2602 } else { 2603 if (scf_error() != SCF_ERROR_TYPE_MISMATCH) 2604 scfdie(); 2605 } 2606 } else { 2607 if (scf_error() != SCF_ERROR_NOT_FOUND) 2608 scfdie(); 2609 } 2610 2611 scf_iter_destroy(iter); 2612 } else { 2613 if (scf_error() != SCF_ERROR_NOT_FOUND) 2614 scfdie(); 2615 } 2616 2617 restarter_common: 2618 scf_pg_destroy(rpg); 2619 2620 /* Dependencies. */ 2621 if ((pg_iter = scf_iter_create(h)) == NULL) 2622 scfdie(); 2623 2624 snap = get_running_snapshot(wip->inst); 2625 2626 if (scf_iter_instance_pgs_typed_composed(pg_iter, wip->inst, snap, 2627 SCF_GROUP_DEPENDENCY) != SCF_SUCCESS) 2628 scfdie(); 2629 2630 while ((ret = scf_iter_next_pg(pg_iter, g_pg)) == 1) 2631 print_detailed_dependency(g_pg); 2632 if (ret == -1) 2633 scfdie(); 2634 2635 scf_iter_destroy(pg_iter); 2636 2637 if (opt_processes) 2638 detailed_list_processes(wip); 2639 2640 /* "application" type property groups */ 2641 if (opt_verbose == 1) 2642 print_application_properties(wip, snap); 2643 2644 scf_snapshot_destroy(snap); 2645 2646 return (0); 2647 } 2648 2649 static int 2650 print_log(void *unused __unused, scf_walkinfo_t *wip) 2651 { 2652 scf_propertygroup_t *rpg; 2653 char buf[MAXPATHLEN]; 2654 2655 if ((rpg = scf_pg_create(h)) == NULL) 2656 scfdie(); 2657 2658 if (scf_instance_get_pg(wip->inst, SCF_PG_RESTARTER, rpg) != 0) { 2659 if (scf_error() != SCF_ERROR_NOT_FOUND) 2660 scfdie(); 2661 2662 goto out; 2663 } 2664 2665 if (pg_get_single_val(rpg, SCF_PROPERTY_LOGFILE, 2666 SCF_TYPE_ASTRING, buf, sizeof (buf), 0) == 0) { 2667 (void) printf("%s\n", buf); 2668 } else if (pg_get_single_val(rpg, SCF_PROPERTY_ALT_LOGFILE, 2669 SCF_TYPE_ASTRING, buf, sizeof (buf), 0) == 0) { 2670 (void) printf("%s\n", buf); 2671 } 2672 2673 out: 2674 scf_pg_destroy(rpg); 2675 2676 return (0); 2677 } 2678 2679 int 2680 qsort_str_compare(const void *p1, const void *p2) 2681 { 2682 return (strcmp((const char *)p1, (const char *)p2)); 2683 } 2684 2685 /* 2686 * get_notify_param_classes() 2687 * return the fma classes that don't have a tag in fma_tags[], otherwise NULL 2688 */ 2689 static char ** 2690 get_notify_param_classes() 2691 { 2692 scf_handle_t *h = _scf_handle_create_and_bind(SCF_VERSION); 2693 scf_instance_t *inst = scf_instance_create(h); 2694 scf_snapshot_t *snap = scf_snapshot_create(h); 2695 scf_snaplevel_t *slvl = scf_snaplevel_create(h); 2696 scf_propertygroup_t *pg = scf_pg_create(h); 2697 scf_iter_t *iter = scf_iter_create(h); 2698 int size = 4; 2699 int n = 0; 2700 size_t sz = scf_limit(SCF_LIMIT_MAX_NAME_LENGTH) + 1; 2701 int err; 2702 char *pgname = safe_malloc(sz); 2703 char **buf = safe_malloc(size * sizeof (char *)); 2704 2705 if (h == NULL || inst == NULL || snap == NULL || slvl == NULL || 2706 pg == NULL || iter == NULL) { 2707 uu_die(gettext("Failed object creation: %s\n"), 2708 scf_strerror(scf_error())); 2709 } 2710 2711 if (scf_handle_decode_fmri(h, SCF_NOTIFY_PARAMS_INST, NULL, NULL, inst, 2712 NULL, NULL, SCF_DECODE_FMRI_EXACT) != 0) 2713 uu_die(gettext("Failed to decode %s: %s\n"), 2714 SCF_NOTIFY_PARAMS_INST, scf_strerror(scf_error())); 2715 2716 if (scf_instance_get_snapshot(inst, "running", snap) != 0) 2717 uu_die(gettext("Failed to get snapshot: %s\n"), 2718 scf_strerror(scf_error())); 2719 2720 if (scf_snapshot_get_base_snaplevel(snap, slvl) != 0) 2721 uu_die(gettext("Failed to get base snaplevel: %s\n"), 2722 scf_strerror(scf_error())); 2723 2724 if (scf_iter_snaplevel_pgs_typed(iter, slvl, 2725 SCF_NOTIFY_PARAMS_PG_TYPE) != 0) 2726 uu_die(gettext("Failed to get iterator: %s\n"), 2727 scf_strerror(scf_error())); 2728 2729 while ((err = scf_iter_next_pg(iter, pg)) == 1) { 2730 char *c; 2731 2732 if (scf_pg_get_name(pg, pgname, sz) == -1) 2733 uu_die(gettext("Failed to get pg name: %s\n"), 2734 scf_strerror(scf_error())); 2735 if ((c = strrchr(pgname, ',')) != NULL) 2736 *c = '\0'; 2737 if (has_fma_tag(pgname)) 2738 continue; 2739 if (!is_fma_token(pgname)) 2740 /* 2741 * We don't emmit a warning here so that we don't 2742 * pollute the output 2743 */ 2744 continue; 2745 2746 if (n + 1 >= size) { 2747 size *= 2; 2748 buf = realloc(buf, size * sizeof (char *)); 2749 if (buf == NULL) 2750 uu_die(gettext("Out of memory.\n")); 2751 } 2752 buf[n] = safe_strdup(pgname); 2753 ++n; 2754 } 2755 /* 2756 * NULL terminate buf 2757 */ 2758 buf[n] = NULL; 2759 if (err == -1) 2760 uu_die(gettext("Failed to iterate pgs: %s\n"), 2761 scf_strerror(scf_error())); 2762 2763 /* sort the classes */ 2764 qsort((void *)buf, n, sizeof (char *), qsort_str_compare); 2765 2766 free(pgname); 2767 scf_iter_destroy(iter); 2768 scf_pg_destroy(pg); 2769 scf_snaplevel_destroy(slvl); 2770 scf_snapshot_destroy(snap); 2771 scf_instance_destroy(inst); 2772 scf_handle_destroy(h); 2773 2774 return (buf); 2775 } 2776 2777 /* 2778 * get_fma_notify_params() 2779 * populates an nvlist_t with notifycation parameters for a given FMA class 2780 * returns 0 if the nvlist is populated, 1 otherwise; 2781 */ 2782 int 2783 get_fma_notify_params(nvlist_t *nvl, const char *class) 2784 { 2785 if (_scf_get_fma_notify_params(class, nvl, 0) != 0) { 2786 /* 2787 * if the preferences have just been deleted 2788 * or does not exist, just skip. 2789 */ 2790 if (scf_error() != SCF_ERROR_NOT_FOUND && 2791 scf_error() != SCF_ERROR_DELETED) 2792 uu_warn(gettext( 2793 "Failed get_fma_notify_params %s\n"), 2794 scf_strerror(scf_error())); 2795 2796 return (1); 2797 } 2798 2799 return (0); 2800 } 2801 2802 /* 2803 * print_notify_fma() 2804 * outputs the notification paramets of FMA events. 2805 * It first outputs classes in fma_tags[], then outputs the other classes 2806 * sorted alphabetically 2807 */ 2808 static void 2809 print_notify_fma(void) 2810 { 2811 nvlist_t *nvl; 2812 char **tmp = NULL; 2813 char **classes, *p; 2814 const char *class; 2815 uint32_t i; 2816 2817 if (nvlist_alloc(&nvl, NV_UNIQUE_NAME, 0) != 0) 2818 uu_die(gettext("Out of memory.\n")); 2819 2820 for (i = 0; (class = get_fma_class(i)) != NULL; ++i) { 2821 if (get_fma_notify_params(nvl, class) == 0) 2822 listnotify_print(nvl, get_fma_tag(i)); 2823 } 2824 2825 if ((classes = get_notify_param_classes()) == NULL) 2826 goto cleanup; 2827 2828 tmp = classes; 2829 for (p = *tmp; p; ++tmp, p = *tmp) { 2830 if (get_fma_notify_params(nvl, p) == 0) 2831 listnotify_print(nvl, re_tag(p)); 2832 2833 free(p); 2834 } 2835 2836 free(classes); 2837 2838 cleanup: 2839 nvlist_free(nvl); 2840 } 2841 2842 /* 2843 * print_notify_fmri() 2844 * prints notifycation parameters for an SMF instance. 2845 */ 2846 static void 2847 print_notify_fmri(const char *fmri) 2848 { 2849 nvlist_t *nvl; 2850 2851 if (nvlist_alloc(&nvl, NV_UNIQUE_NAME, 0) != 0) 2852 uu_die(gettext("Out of memory.\n")); 2853 2854 if (_scf_get_svc_notify_params(fmri, nvl, SCF_TRANSITION_ALL, 0, 0) != 2855 SCF_SUCCESS) { 2856 if (scf_error() != SCF_ERROR_NOT_FOUND && 2857 scf_error() != SCF_ERROR_DELETED) 2858 uu_warn(gettext( 2859 "Failed _scf_get_svc_notify_params: %s\n"), 2860 scf_strerror(scf_error())); 2861 } else { 2862 if (strcmp(SCF_INSTANCE_GLOBAL, fmri) == 0) 2863 safe_printf( 2864 gettext("System wide notification parameters:\n")); 2865 safe_printf("%s:\n", fmri); 2866 listnotify_print(nvl, NULL); 2867 } 2868 nvlist_free(nvl); 2869 } 2870 2871 /* 2872 * print_notify_special() 2873 * prints notification parameters for FMA events and system wide SMF state 2874 * transitions parameters 2875 */ 2876 static void 2877 print_notify_special() 2878 { 2879 safe_printf("Notification parameters for FMA Events\n"); 2880 print_notify_fma(); 2881 print_notify_fmri(SCF_INSTANCE_GLOBAL); 2882 } 2883 2884 /* 2885 * print_notify() 2886 * callback function to print notification parameters for SMF state transition 2887 * instances. It skips global and notify-params instances as they should be 2888 * printed by print_notify_special() 2889 */ 2890 /* ARGSUSED */ 2891 static int 2892 print_notify(void *unused, scf_walkinfo_t *wip) 2893 { 2894 if (strcmp(SCF_INSTANCE_GLOBAL, wip->fmri) == 0 || 2895 strcmp(SCF_NOTIFY_PARAMS_INST, wip->fmri) == 0) 2896 return (0); 2897 2898 print_notify_fmri(wip->fmri); 2899 2900 return (0); 2901 } 2902 2903 /* 2904 * Append a one-lined description of each process in inst's contract(s) and 2905 * return the augmented string. 2906 */ 2907 static char * 2908 add_processes(scf_walkinfo_t *wip, char *line, scf_propertygroup_t *lpg) 2909 { 2910 pid_t *pids = NULL; 2911 uint_t i, n = 0; 2912 2913 if (lpg == NULL) { 2914 if (instance_processes(wip->inst, wip->fmri, &pids, &n) != 0) 2915 return (line); 2916 } else { 2917 /* Legacy services */ 2918 scf_iter_t *iter; 2919 2920 if ((iter = scf_iter_create(h)) == NULL) 2921 scfdie(); 2922 2923 (void) propvals_to_pids(lpg, scf_property_contract, &pids, &n, 2924 g_prop, g_val, iter); 2925 2926 scf_iter_destroy(iter); 2927 } 2928 2929 if (n == 0) 2930 return (line); 2931 2932 qsort(pids, n, sizeof (*pids), pidcmp); 2933 2934 for (i = 0; i < n; ++i) { 2935 char *cp, stime[9]; 2936 psinfo_t psi; 2937 const char *name = NULL; 2938 int len = 1 + 15 + 8 + 3 + 6 + 1 + PRFNSZ; 2939 2940 if (proc_get_psinfo(pids[i], &psi) != 0) 2941 continue; 2942 2943 line = realloc(line, strlen(line) + len); 2944 if (line == NULL) 2945 uu_die(gettext("Out of memory.\n")); 2946 2947 cp = strchr(line, '\0'); 2948 2949 if (!IS_ZOMBIE(&psi)) { 2950 #define DAY (24 * 60 * 60) 2951 #define YEAR (12 * 30 * 24 * 60 * 60) 2952 2953 struct tm *tm; 2954 2955 tm = localtime(&psi.pr_start.tv_sec); 2956 2957 /* 2958 * Print time if started within the past 24 hours, 2959 * print date if within the past 12 months, print year 2960 * if started greater than 12 months ago. 2961 */ 2962 if (now - psi.pr_start.tv_sec < DAY) { 2963 (void) strftime(stime, sizeof (stime), 2964 gettext(FORMAT_TIME), tm); 2965 } else if (now - psi.pr_start.tv_sec < YEAR) { 2966 (void) strftime(stime, sizeof (stime), 2967 gettext(FORMAT_DATE), tm); 2968 } else { 2969 (void) strftime(stime, sizeof (stime), 2970 gettext(FORMAT_YEAR), tm); 2971 } 2972 2973 name = psi.pr_fname; 2974 #undef DAY 2975 #undef YEAR 2976 } else { 2977 (void) snprintf(stime, sizeof (stime), "-"); 2978 name = "<defunct>"; 2979 } 2980 2981 (void) snprintf(cp, len, "\n %-8s %6ld %.*s", 2982 stime, pids[i], PRFNSZ, name); 2983 } 2984 2985 free(pids); 2986 2987 return (line); 2988 } 2989 2990 /*ARGSUSED*/ 2991 static int 2992 list_instance(void *unused, scf_walkinfo_t *wip) 2993 { 2994 struct avl_string *lp; 2995 char *cp; 2996 int i; 2997 uu_avl_index_t idx; 2998 2999 /* 3000 * If the user has specified a restarter, check for a match first 3001 */ 3002 if (restarters != NULL) { 3003 struct pfmri_list *rest; 3004 int match; 3005 char *restarter_fmri; 3006 const char *scope_name, *svc_name, *inst_name, *pg_name; 3007 3008 /* legacy services don't have restarters */ 3009 if (wip->pg != NULL) 3010 return (0); 3011 3012 restarter_fmri = safe_malloc(max_scf_fmri_length + 1); 3013 3014 if (inst_get_single_val(wip->inst, SCF_PG_GENERAL, 3015 SCF_PROPERTY_RESTARTER, SCF_TYPE_ASTRING, restarter_fmri, 3016 max_scf_fmri_length + 1, 0, 0, 1) != 0) 3017 (void) strcpy(restarter_fmri, SCF_SERVICE_STARTD); 3018 3019 if (scf_parse_svc_fmri(restarter_fmri, &scope_name, &svc_name, 3020 &inst_name, &pg_name, NULL) != SCF_SUCCESS) { 3021 free(restarter_fmri); 3022 return (0); 3023 } 3024 3025 match = 0; 3026 for (rest = restarters; rest != NULL; rest = rest->next) { 3027 if (strcmp(rest->scope, scope_name) == 0 && 3028 strcmp(rest->service, svc_name) == 0 && 3029 strcmp(rest->instance, inst_name) == 0) 3030 match = 1; 3031 } 3032 3033 free(restarter_fmri); 3034 3035 if (!match) 3036 return (0); 3037 } 3038 3039 if (wip->pg == NULL && ht_buckets != NULL && ht_add(wip->fmri)) { 3040 /* It was already there. */ 3041 return (0); 3042 } 3043 3044 lp = safe_malloc(sizeof (*lp)); 3045 3046 lp->str = NULL; 3047 for (i = 0; i < opt_cnum; ++i) { 3048 columns[opt_columns[i]].sprint(&lp->str, wip); 3049 } 3050 cp = lp->str + strlen(lp->str); 3051 cp--; 3052 while (*cp == ' ') 3053 cp--; 3054 *(cp+1) = '\0'; 3055 3056 /* If we're supposed to list the processes, too, do that now. */ 3057 if (opt_processes) 3058 lp->str = add_processes(wip, lp->str, wip->pg); 3059 3060 /* Create the sort key. */ 3061 cp = lp->key = safe_malloc(sortkey_sz); 3062 for (i = 0; i < opt_snum; ++i) { 3063 int j = opt_sort[i] & 0xff; 3064 3065 assert(columns[j].get_sortkey != NULL); 3066 columns[j].get_sortkey(cp, opt_sort[i] & ~0xff, wip); 3067 cp += columns[j].sortkey_width; 3068 } 3069 3070 /* Insert into AVL tree. */ 3071 uu_avl_node_init(lp, &lp->node, lines_pool); 3072 (void) uu_avl_find(lines, lp, NULL, &idx); 3073 uu_avl_insert(lines, lp, idx); 3074 3075 return (0); 3076 } 3077 3078 static int 3079 list_if_enabled(void *unused, scf_walkinfo_t *wip) 3080 { 3081 if (wip->pg != NULL || 3082 instance_enabled(wip->inst, B_FALSE) == 1 || 3083 instance_enabled(wip->inst, B_TRUE) == 1) 3084 return (list_instance(unused, wip)); 3085 3086 return (0); 3087 } 3088 3089 /* 3090 * Service FMRI selection: Lookup and call list_instance() for the instances. 3091 * Instance FMRI selection: Lookup and call list_instance(). 3092 * 3093 * Note: This is shoehorned into a walk_dependencies() callback prototype so 3094 * it can be used in list_dependencies. 3095 */ 3096 static int 3097 list_svc_or_inst_fmri(void *complain, scf_walkinfo_t *wip) 3098 { 3099 char *fmri; 3100 const char *svc_name, *inst_name, *pg_name, *save; 3101 scf_iter_t *iter; 3102 int ret; 3103 3104 fmri = safe_strdup(wip->fmri); 3105 3106 if (scf_parse_svc_fmri(fmri, NULL, &svc_name, &inst_name, &pg_name, 3107 NULL) != SCF_SUCCESS) { 3108 if (complain) 3109 uu_warn(gettext("FMRI \"%s\" is invalid.\n"), 3110 wip->fmri); 3111 exit_status = UU_EXIT_FATAL; 3112 free(fmri); 3113 return (0); 3114 } 3115 3116 /* 3117 * Yes, this invalidates *_name, but we only care whether they're NULL 3118 * or not. 3119 */ 3120 free(fmri); 3121 3122 if (svc_name == NULL || pg_name != NULL) { 3123 if (complain) 3124 uu_warn(gettext("FMRI \"%s\" does not designate a " 3125 "service or instance.\n"), wip->fmri); 3126 return (0); 3127 } 3128 3129 if (inst_name != NULL) { 3130 /* instance */ 3131 if (scf_handle_decode_fmri(h, wip->fmri, wip->scope, wip->svc, 3132 wip->inst, NULL, NULL, 0) != SCF_SUCCESS) { 3133 if (scf_error() != SCF_ERROR_NOT_FOUND) 3134 scfdie(); 3135 3136 if (complain) 3137 uu_warn(gettext( 3138 "Instance \"%s\" does not exist.\n"), 3139 wip->fmri); 3140 return (0); 3141 } 3142 3143 return (list_instance(NULL, wip)); 3144 } 3145 3146 /* service: Walk the instances. */ 3147 if (scf_handle_decode_fmri(h, wip->fmri, wip->scope, wip->svc, NULL, 3148 NULL, NULL, 0) != SCF_SUCCESS) { 3149 if (scf_error() != SCF_ERROR_NOT_FOUND) 3150 scfdie(); 3151 3152 if (complain) 3153 uu_warn(gettext("Service \"%s\" does not exist.\n"), 3154 wip->fmri); 3155 3156 exit_status = UU_EXIT_FATAL; 3157 3158 return (0); 3159 } 3160 3161 iter = scf_iter_create(h); 3162 if (iter == NULL) 3163 scfdie(); 3164 3165 if (scf_iter_service_instances(iter, wip->svc) != SCF_SUCCESS) 3166 scfdie(); 3167 3168 if ((fmri = malloc(max_scf_fmri_length + 1)) == NULL) { 3169 scf_iter_destroy(iter); 3170 exit_status = UU_EXIT_FATAL; 3171 return (0); 3172 } 3173 3174 save = wip->fmri; 3175 wip->fmri = fmri; 3176 while ((ret = scf_iter_next_instance(iter, wip->inst)) == 1) { 3177 if (scf_instance_to_fmri(wip->inst, fmri, 3178 max_scf_fmri_length + 1) <= 0) 3179 scfdie(); 3180 (void) list_instance(NULL, wip); 3181 } 3182 free(fmri); 3183 wip->fmri = save; 3184 if (ret == -1) 3185 scfdie(); 3186 3187 exit_status = UU_EXIT_OK; 3188 3189 scf_iter_destroy(iter); 3190 3191 return (0); 3192 } 3193 3194 /* 3195 * Dependency selection: Straightforward since each instance lists the 3196 * services it depends on. 3197 */ 3198 3199 static void 3200 walk_dependencies(scf_walkinfo_t *wip, scf_walk_callback callback, void *data) 3201 { 3202 scf_snapshot_t *snap; 3203 scf_iter_t *iter, *viter; 3204 int ret, vret; 3205 char *dep; 3206 3207 assert(wip->inst != NULL); 3208 3209 if ((iter = scf_iter_create(h)) == NULL || 3210 (viter = scf_iter_create(h)) == NULL) 3211 scfdie(); 3212 3213 snap = get_running_snapshot(wip->inst); 3214 3215 if (scf_iter_instance_pgs_typed_composed(iter, wip->inst, snap, 3216 SCF_GROUP_DEPENDENCY) != SCF_SUCCESS) 3217 scfdie(); 3218 3219 dep = safe_malloc(max_scf_value_length + 1); 3220 3221 while ((ret = scf_iter_next_pg(iter, g_pg)) == 1) { 3222 scf_type_t ty; 3223 3224 /* Ignore exclude_any dependencies. */ 3225 if (scf_pg_get_property(g_pg, SCF_PROPERTY_GROUPING, g_prop) != 3226 SCF_SUCCESS) { 3227 if (scf_error() != SCF_ERROR_NOT_FOUND) 3228 scfdie(); 3229 3230 continue; 3231 } 3232 3233 if (scf_property_type(g_prop, &ty) != SCF_SUCCESS) 3234 scfdie(); 3235 3236 if (ty != SCF_TYPE_ASTRING) 3237 continue; 3238 3239 if (scf_property_get_value(g_prop, g_val) != SCF_SUCCESS) { 3240 if (scf_error() != SCF_ERROR_CONSTRAINT_VIOLATED) 3241 scfdie(); 3242 3243 continue; 3244 } 3245 3246 if (scf_value_get_astring(g_val, dep, 3247 max_scf_value_length + 1) < 0) 3248 scfdie(); 3249 3250 if (strcmp(dep, SCF_DEP_EXCLUDE_ALL) == 0) 3251 continue; 3252 3253 if (scf_pg_get_property(g_pg, SCF_PROPERTY_ENTITIES, g_prop) != 3254 SCF_SUCCESS) { 3255 if (scf_error() != SCF_ERROR_NOT_FOUND) 3256 scfdie(); 3257 3258 continue; 3259 } 3260 3261 if (scf_iter_property_values(viter, g_prop) != SCF_SUCCESS) 3262 scfdie(); 3263 3264 while ((vret = scf_iter_next_value(viter, g_val)) == 1) { 3265 if (scf_value_get_astring(g_val, dep, 3266 max_scf_value_length + 1) < 0) 3267 scfdie(); 3268 3269 wip->fmri = dep; 3270 if (callback(data, wip) != 0) 3271 goto out; 3272 } 3273 if (vret == -1) 3274 scfdie(); 3275 } 3276 if (ret == -1) 3277 scfdie(); 3278 3279 out: 3280 scf_iter_destroy(viter); 3281 scf_iter_destroy(iter); 3282 scf_snapshot_destroy(snap); 3283 } 3284 3285 static int 3286 list_dependencies(void *data, scf_walkinfo_t *wip) 3287 { 3288 walk_dependencies(wip, list_svc_or_inst_fmri, data); 3289 return (0); 3290 } 3291 3292 3293 /* 3294 * Dependent selection: The "providing" service's or instance's FMRI is parsed 3295 * into the provider_* variables, the instances are walked, and any instance 3296 * which lists an FMRI which parses to these components is selected. This is 3297 * inefficient in the face of multiple operands, but that should be uncommon. 3298 */ 3299 3300 static char *provider_scope; 3301 static char *provider_svc; 3302 static char *provider_inst; /* NULL for services */ 3303 3304 /*ARGSUSED*/ 3305 static int 3306 check_against_provider(void *arg, scf_walkinfo_t *wip) 3307 { 3308 char *cfmri; 3309 const char *scope_name, *svc_name, *inst_name, *pg_name; 3310 int *matchp = arg; 3311 3312 cfmri = safe_strdup(wip->fmri); 3313 3314 if (scf_parse_svc_fmri(cfmri, &scope_name, &svc_name, &inst_name, 3315 &pg_name, NULL) != SCF_SUCCESS) { 3316 free(cfmri); 3317 return (0); 3318 } 3319 3320 if (svc_name == NULL || pg_name != NULL) { 3321 free(cfmri); 3322 return (0); 3323 } 3324 3325 /* 3326 * If the user has specified an instance, then also match dependencies 3327 * on the service itself. 3328 */ 3329 *matchp = (strcmp(provider_scope, scope_name) == 0 && 3330 strcmp(provider_svc, svc_name) == 0 && 3331 (provider_inst == NULL ? (inst_name == NULL) : 3332 (inst_name == NULL || strcmp(provider_inst, inst_name) == 0))); 3333 3334 free(cfmri); 3335 3336 /* Stop on matches. */ 3337 return (*matchp); 3338 } 3339 3340 static int 3341 list_if_dependent(void *unused, scf_walkinfo_t *wip) 3342 { 3343 /* Only proceed if this instance depends on provider_*. */ 3344 int match = 0; 3345 3346 (void) walk_dependencies(wip, check_against_provider, &match); 3347 3348 if (match) 3349 return (list_instance(unused, wip)); 3350 3351 return (0); 3352 } 3353 3354 /*ARGSUSED*/ 3355 static int 3356 list_dependents(void *unused, scf_walkinfo_t *wip) 3357 { 3358 char *save; 3359 int ret; 3360 3361 if (scf_scope_get_name(wip->scope, provider_scope, 3362 max_scf_fmri_length) <= 0 || 3363 scf_service_get_name(wip->svc, provider_svc, 3364 max_scf_fmri_length) <= 0) 3365 scfdie(); 3366 3367 save = provider_inst; 3368 if (wip->inst == NULL) 3369 provider_inst = NULL; 3370 else if (scf_instance_get_name(wip->inst, provider_inst, 3371 max_scf_fmri_length) <= 0) 3372 scfdie(); 3373 3374 ret = scf_walk_fmri(h, 0, NULL, 0, list_if_dependent, NULL, NULL, 3375 uu_warn); 3376 3377 provider_inst = save; 3378 3379 return (ret); 3380 } 3381 3382 /* 3383 * main() & helpers 3384 */ 3385 3386 static void 3387 add_sort_column(const char *col, int reverse) 3388 { 3389 int i; 3390 3391 ++opt_snum; 3392 3393 opt_sort = realloc(opt_sort, opt_snum * sizeof (*opt_sort)); 3394 if (opt_sort == NULL) 3395 uu_die(gettext("Too many sort criteria: out of memory.\n")); 3396 3397 for (i = 0; i < ncolumns; ++i) { 3398 if (strcasecmp(col, columns[i].name) == 0) 3399 break; 3400 } 3401 3402 if (i < ncolumns) 3403 opt_sort[opt_snum - 1] = (reverse ? i | 0x100 : i); 3404 else 3405 uu_die(gettext("Unrecognized sort column \"%s\".\n"), col); 3406 3407 sortkey_sz += columns[i].sortkey_width; 3408 } 3409 3410 static void 3411 add_restarter(const char *fmri) 3412 { 3413 char *cfmri; 3414 const char *pg_name; 3415 struct pfmri_list *rest; 3416 3417 cfmri = safe_strdup(fmri); 3418 rest = safe_malloc(sizeof (*rest)); 3419 3420 if (scf_parse_svc_fmri(cfmri, &rest->scope, &rest->service, 3421 &rest->instance, &pg_name, NULL) != SCF_SUCCESS) 3422 uu_die(gettext("Restarter FMRI \"%s\" is invalid.\n"), fmri); 3423 3424 if (rest->instance == NULL || pg_name != NULL) 3425 uu_die(gettext("Restarter FMRI \"%s\" does not designate an " 3426 "instance.\n"), fmri); 3427 3428 rest->next = restarters; 3429 restarters = rest; 3430 return; 3431 3432 err: 3433 free(cfmri); 3434 free(rest); 3435 } 3436 3437 /* ARGSUSED */ 3438 static int 3439 line_cmp(const void *l_arg, const void *r_arg, void *private) 3440 { 3441 const struct avl_string *l = l_arg; 3442 const struct avl_string *r = r_arg; 3443 3444 return (memcmp(l->key, r->key, sortkey_sz)); 3445 } 3446 3447 /* ARGSUSED */ 3448 static int 3449 print_line(void *e, void *private) 3450 { 3451 struct avl_string *lp = e; 3452 3453 (void) puts(lp->str); 3454 3455 return (UU_WALK_NEXT); 3456 } 3457 3458 /* ARGSUSED */ 3459 static void 3460 errignore(const char *str, ...) 3461 {} 3462 3463 int 3464 main(int argc, char **argv) 3465 { 3466 char opt, opt_mode; 3467 int i, n; 3468 char *columns_str = NULL; 3469 char *cp; 3470 const char *progname; 3471 int err, missing = 1, ignored, *errarg; 3472 uint_t nzents = 0, zent = 0; 3473 zoneid_t *zids = NULL; 3474 char zonename[ZONENAME_MAX]; 3475 void (*errfunc)(const char *, ...); 3476 3477 int show_all = 0; 3478 int show_header = 1; 3479 int show_zones = 0; 3480 3481 const char * const options = "aHpvno:R:s:S:dDlL?xZz:"; 3482 3483 (void) setlocale(LC_ALL, ""); 3484 3485 locale = setlocale(LC_MESSAGES, NULL); 3486 if (locale) { 3487 locale = safe_strdup(locale); 3488 _scf_sanitize_locale(locale); 3489 } 3490 3491 (void) textdomain(TEXT_DOMAIN); 3492 progname = uu_setpname(argv[0]); 3493 3494 exit_status = UU_EXIT_OK; 3495 3496 max_scf_name_length = scf_limit(SCF_LIMIT_MAX_NAME_LENGTH); 3497 max_scf_value_length = scf_limit(SCF_LIMIT_MAX_VALUE_LENGTH); 3498 max_scf_fmri_length = scf_limit(SCF_LIMIT_MAX_FMRI_LENGTH); 3499 max_scf_type_length = scf_limit(SCF_LIMIT_MAX_PG_TYPE_LENGTH); 3500 3501 if (max_scf_name_length == -1 || max_scf_value_length == -1 || 3502 max_scf_fmri_length == -1 || max_scf_type_length == -1) 3503 scfdie(); 3504 3505 now = time(NULL); 3506 assert(now != -1); 3507 3508 /* 3509 * opt_mode is the mode of operation. 0 for plain, 'd' for 3510 * dependencies, 'D' for dependents, and 'l' for detailed (long). We 3511 * need to know now so we know which options are valid. 3512 */ 3513 opt_mode = 0; 3514 while ((opt = getopt(argc, argv, options)) != -1) { 3515 switch (opt) { 3516 case '?': 3517 if (optopt == '?') { 3518 print_help(progname); 3519 return (UU_EXIT_OK); 3520 } else { 3521 argserr(progname); 3522 /* NOTREACHED */ 3523 } 3524 3525 case 'd': 3526 case 'D': 3527 case 'l': 3528 case 'L': 3529 if (opt_mode != 0) 3530 argserr(progname); 3531 3532 opt_mode = opt; 3533 break; 3534 3535 case 'n': 3536 if (opt_mode != 0) 3537 argserr(progname); 3538 3539 opt_mode = opt; 3540 break; 3541 3542 case 'x': 3543 if (opt_mode != 0) 3544 argserr(progname); 3545 3546 opt_mode = opt; 3547 break; 3548 3549 default: 3550 break; 3551 } 3552 } 3553 3554 sortkey_sz = 0; 3555 3556 optind = 1; /* Reset getopt() */ 3557 while ((opt = getopt(argc, argv, options)) != -1) { 3558 switch (opt) { 3559 case 'a': 3560 if (opt_mode != 0) 3561 argserr(progname); 3562 show_all = 1; 3563 break; 3564 3565 case 'H': 3566 if (opt_mode == 'l' || opt_mode == 'x') 3567 argserr(progname); 3568 show_header = 0; 3569 break; 3570 3571 case 'p': 3572 if (opt_mode == 'x') 3573 argserr(progname); 3574 opt_processes = 1; 3575 break; 3576 3577 case 'v': 3578 opt_verbose = 1; 3579 break; 3580 3581 case 'o': 3582 if (opt_mode == 'l' || opt_mode == 'x') 3583 argserr(progname); 3584 columns_str = optarg; 3585 break; 3586 3587 case 'R': 3588 if (opt_mode != 0 || opt_mode == 'x') 3589 argserr(progname); 3590 3591 add_restarter(optarg); 3592 break; 3593 3594 case 's': 3595 case 'S': 3596 if (opt_mode != 0) 3597 argserr(progname); 3598 3599 add_sort_column(optarg, optopt == 'S'); 3600 break; 3601 3602 case 'd': 3603 case 'D': 3604 case 'l': 3605 case 'L': 3606 case 'n': 3607 case 'x': 3608 assert(opt_mode == optopt); 3609 break; 3610 3611 case 'z': 3612 if (getzoneid() != GLOBAL_ZONEID) 3613 uu_die(gettext("svcs -z may only be used from " 3614 "the global zone\n")); 3615 if (show_zones) 3616 argserr(progname); 3617 3618 opt_zone = optarg; 3619 break; 3620 3621 case 'Z': 3622 if (getzoneid() != GLOBAL_ZONEID) 3623 uu_die(gettext("svcs -Z may only be used from " 3624 "the global zone\n")); 3625 if (opt_zone != NULL) 3626 argserr(progname); 3627 3628 show_zones = 1; 3629 break; 3630 3631 case '?': 3632 argserr(progname); 3633 3634 default: 3635 assert(0); 3636 abort(); 3637 } 3638 } 3639 3640 /* 3641 * -a is only meaningful when given no arguments 3642 */ 3643 if (show_all && optind != argc) 3644 uu_warn(gettext("-a ignored when used with arguments.\n")); 3645 3646 while (show_zones) { 3647 uint_t found; 3648 3649 if (zone_list(NULL, &nzents) != 0) 3650 uu_die(gettext("could not get number of zones")); 3651 3652 if ((zids = malloc(nzents * sizeof (zoneid_t))) == NULL) { 3653 uu_die(gettext("could not allocate array for " 3654 "%d zone IDs"), nzents); 3655 } 3656 3657 found = nzents; 3658 3659 if (zone_list(zids, &found) != 0) 3660 uu_die(gettext("could not get zone list")); 3661 3662 /* 3663 * If the number of zones has not changed between our calls to 3664 * zone_list(), we're done -- otherwise, we must free our array 3665 * of zone IDs and take another lap. 3666 */ 3667 if (found == nzents) 3668 break; 3669 3670 free(zids); 3671 } 3672 3673 argc -= optind; 3674 argv += optind; 3675 3676 again: 3677 h = scf_handle_create(SCF_VERSION); 3678 if (h == NULL) 3679 scfdie(); 3680 3681 if (opt_zone != NULL || zids != NULL) { 3682 scf_value_t *zone; 3683 3684 assert(opt_zone == NULL || zids == NULL); 3685 3686 if (opt_zone == NULL) { 3687 if (getzonenamebyid(zids[zent++], 3688 zonename, sizeof (zonename)) < 0) { 3689 uu_warn(gettext("could not get name for " 3690 "zone %d; ignoring"), zids[zent - 1]); 3691 goto nextzone; 3692 } 3693 3694 g_zonename = zonename; 3695 } else { 3696 g_zonename = opt_zone; 3697 } 3698 3699 if ((zone = scf_value_create(h)) == NULL) 3700 scfdie(); 3701 3702 if (scf_value_set_astring(zone, g_zonename) != SCF_SUCCESS) 3703 scfdie(); 3704 3705 if (scf_handle_decorate(h, "zone", zone) != SCF_SUCCESS) 3706 uu_die(gettext("invalid zone '%s'\n"), g_zonename); 3707 3708 scf_value_destroy(zone); 3709 } 3710 3711 if (scf_handle_bind(h) == -1) { 3712 if (g_zonename != NULL) { 3713 uu_warn(gettext("Could not bind to repository " 3714 "server for zone %s: %s\n"), g_zonename, 3715 scf_strerror(scf_error())); 3716 3717 if (!show_zones) 3718 return (UU_EXIT_FATAL); 3719 3720 goto nextzone; 3721 } 3722 3723 uu_die(gettext("Could not bind to repository server: %s. " 3724 "Exiting.\n"), scf_strerror(scf_error())); 3725 } 3726 3727 if ((g_pg = scf_pg_create(h)) == NULL || 3728 (g_prop = scf_property_create(h)) == NULL || 3729 (g_val = scf_value_create(h)) == NULL) 3730 scfdie(); 3731 3732 if (show_zones) { 3733 /* 3734 * It's hard to avoid editorializing here, but suffice it to 3735 * say that scf_walk_fmri() takes an error handler, the 3736 * interface to which has been regrettably misdesigned: the 3737 * handler itself takes exclusively a string -- even though 3738 * scf_walk_fmri() has detailed, programmatic knowledge 3739 * of the error condition at the time it calls its errfunc. 3740 * That is, only the error message and not the error semantics 3741 * are given to the handler. This is poor interface at best, 3742 * but it is particularly problematic when we are talking to 3743 * multiple repository servers (as when we are iterating over 3744 * all zones) as we do not want to treat failure to find a 3745 * match in one zone as overall failure. Ideally, we would 3746 * simply ignore SCF_MSG_PATTERN_NOINSTANCE and correctly 3747 * process the others, but alas, no such interface exists -- 3748 * and we must settle for instead ignoring all errfunc-called 3749 * errors in the case that we are iterating over all zones... 3750 */ 3751 errfunc = errignore; 3752 errarg = missing ? &missing : &ignored; 3753 missing = 0; 3754 } else { 3755 errfunc = uu_warn; 3756 errarg = &exit_status; 3757 } 3758 3759 /* 3760 * If we're in long mode, take care of it now before we deal with the 3761 * sorting and the columns, since we won't use them anyway. 3762 */ 3763 if (opt_mode == 'l') { 3764 if (argc == 0) 3765 argserr(progname); 3766 3767 if ((err = scf_walk_fmri(h, argc, argv, SCF_WALK_MULTIPLE, 3768 print_detailed, NULL, errarg, errfunc)) != 0) { 3769 uu_warn(gettext("failed to iterate over " 3770 "instances: %s\n"), scf_strerror(err)); 3771 exit_status = UU_EXIT_FATAL; 3772 } 3773 3774 goto nextzone; 3775 } 3776 3777 if (opt_mode == 'L') { 3778 if ((err = scf_walk_fmri(h, argc, argv, SCF_WALK_MULTIPLE, 3779 print_log, NULL, &exit_status, uu_warn)) != 0) { 3780 uu_warn(gettext("failed to iterate over " 3781 "instances: %s\n"), scf_strerror(err)); 3782 exit_status = UU_EXIT_FATAL; 3783 } 3784 3785 goto nextzone; 3786 } 3787 3788 if (opt_mode == 'n') { 3789 print_notify_special(); 3790 if ((err = scf_walk_fmri(h, argc, argv, SCF_WALK_MULTIPLE, 3791 print_notify, NULL, errarg, errfunc)) != 0) { 3792 uu_warn(gettext("failed to iterate over " 3793 "instances: %s\n"), scf_strerror(err)); 3794 exit_status = UU_EXIT_FATAL; 3795 } 3796 3797 goto nextzone; 3798 } 3799 3800 if (opt_mode == 'x') { 3801 explain(opt_verbose, argc, argv); 3802 goto nextzone; 3803 } 3804 3805 if (columns_str == NULL) { 3806 if (opt_snum == 0) { 3807 if (show_zones) 3808 add_sort_column("zone", 0); 3809 3810 /* Default sort. */ 3811 add_sort_column("state", 0); 3812 add_sort_column("stime", 0); 3813 add_sort_column("fmri", 0); 3814 } 3815 3816 if (!opt_verbose) { 3817 columns_str = safe_strdup(show_zones ? 3818 "zone,state,stime,fmri" : "state,stime,fmri"); 3819 } else { 3820 columns_str = safe_strdup(show_zones ? 3821 "zone,state,nstate,stime,ctid,fmri" : 3822 "state,nstate,stime,ctid,fmri"); 3823 } 3824 } 3825 3826 if (opt_columns == NULL) { 3827 /* Decode columns_str into opt_columns. */ 3828 line_sz = 0; 3829 3830 opt_cnum = 1; 3831 for (cp = columns_str; *cp != '\0'; ++cp) 3832 if (*cp == ',') 3833 ++opt_cnum; 3834 3835 if (*columns_str == '\0') 3836 uu_die(gettext("No columns specified.\n")); 3837 3838 opt_columns = malloc(opt_cnum * sizeof (*opt_columns)); 3839 if (opt_columns == NULL) 3840 uu_die(gettext("Too many columns.\n")); 3841 3842 for (n = 0; *columns_str != '\0'; ++n) { 3843 i = getcolumnopt(&columns_str); 3844 if (i == -1) 3845 uu_die(gettext("Unknown column \"%s\".\n"), 3846 columns_str); 3847 3848 if (strcmp(columns[i].name, "N") == 0 || 3849 strcmp(columns[i].name, "SN") == 0 || 3850 strcmp(columns[i].name, "NSTA") == 0 || 3851 strcmp(columns[i].name, "NSTATE") == 0) 3852 opt_nstate_shown = 1; 3853 3854 opt_columns[n] = i; 3855 line_sz += columns[i].width + 1; 3856 } 3857 3858 if ((lines_pool = uu_avl_pool_create("lines_pool", 3859 sizeof (struct avl_string), offsetof(struct avl_string, 3860 node), line_cmp, UU_AVL_DEBUG)) == NULL || 3861 (lines = uu_avl_create(lines_pool, NULL, 0)) == NULL) 3862 uu_die(gettext("Unexpected libuutil error: %s\n"), 3863 uu_strerror(uu_error())); 3864 } 3865 3866 switch (opt_mode) { 3867 case 0: 3868 /* 3869 * If we already have a hash table (e.g., because we are 3870 * processing multiple zones), destroy it before creating 3871 * a new one. 3872 */ 3873 if (ht_buckets != NULL) 3874 ht_free(); 3875 3876 ht_init(); 3877 3878 /* Always show all FMRIs when given arguments or restarters */ 3879 if (argc != 0 || restarters != NULL) 3880 show_all = 1; 3881 3882 if ((err = scf_walk_fmri(h, argc, argv, 3883 SCF_WALK_MULTIPLE | SCF_WALK_LEGACY, 3884 show_all ? list_instance : list_if_enabled, NULL, 3885 errarg, errfunc)) != 0) { 3886 uu_warn(gettext("failed to iterate over " 3887 "instances: %s\n"), scf_strerror(err)); 3888 exit_status = UU_EXIT_FATAL; 3889 } 3890 break; 3891 3892 case 'd': 3893 if (argc == 0) 3894 argserr(progname); 3895 3896 if ((err = scf_walk_fmri(h, argc, argv, 3897 SCF_WALK_MULTIPLE, list_dependencies, NULL, 3898 errarg, errfunc)) != 0) { 3899 uu_warn(gettext("failed to iterate over " 3900 "instances: %s\n"), scf_strerror(err)); 3901 exit_status = UU_EXIT_FATAL; 3902 } 3903 break; 3904 3905 case 'D': 3906 if (argc == 0) 3907 argserr(progname); 3908 3909 provider_scope = safe_malloc(max_scf_fmri_length); 3910 provider_svc = safe_malloc(max_scf_fmri_length); 3911 provider_inst = safe_malloc(max_scf_fmri_length); 3912 3913 if ((err = scf_walk_fmri(h, argc, argv, 3914 SCF_WALK_MULTIPLE | SCF_WALK_SERVICE, 3915 list_dependents, NULL, &exit_status, uu_warn)) != 0) { 3916 uu_warn(gettext("failed to iterate over " 3917 "instances: %s\n"), scf_strerror(err)); 3918 exit_status = UU_EXIT_FATAL; 3919 } 3920 3921 free(provider_scope); 3922 free(provider_svc); 3923 free(provider_inst); 3924 break; 3925 3926 case 'n': 3927 break; 3928 3929 default: 3930 assert(0); 3931 abort(); 3932 } 3933 3934 nextzone: 3935 if (show_zones && zent < nzents && exit_status == 0) { 3936 scf_handle_destroy(h); 3937 goto again; 3938 } 3939 3940 if (show_zones && exit_status == 0) 3941 exit_status = missing; 3942 3943 if (opt_columns == NULL) 3944 return (exit_status); 3945 3946 if (show_header) 3947 print_header(); 3948 3949 (void) uu_avl_walk(lines, print_line, NULL, 0); 3950 3951 return (exit_status); 3952 } 3953