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 2007 Sun Microsystems, Inc. All rights reserved. 24 * Use is subject to license terms. 25 */ 26 27 /* 28 * Copyright (c) 2013, Joyent, Inc. All rights reserved. 29 * Copyright (c) 2012 by Delphix. All rights reserved. 30 */ 31 32 #include <sys/resource.h> 33 #include <sys/mman.h> 34 #include <sys/types.h> 35 36 #include <strings.h> 37 #include <signal.h> 38 #include <stdlib.h> 39 #include <unistd.h> 40 #include <limits.h> 41 #ifdef illumos 42 #include <alloca.h> 43 #endif 44 #include <errno.h> 45 #include <fcntl.h> 46 47 #include <dt_impl.h> 48 #include <dt_string.h> 49 50 static int 51 dt_opt_agg(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 52 { 53 dt_aggregate_t *agp = &dtp->dt_aggregate; 54 55 if (arg != NULL) 56 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 57 58 agp->dtat_flags |= option; 59 return (0); 60 } 61 62 /*ARGSUSED*/ 63 static int 64 dt_opt_amin(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 65 { 66 char str[DTRACE_ATTR2STR_MAX]; 67 dtrace_attribute_t attr; 68 69 if (arg == NULL || dtrace_str2attr(arg, &attr) == -1) 70 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 71 72 dt_dprintf("set compiler attribute minimum to %s\n", 73 dtrace_attr2str(attr, str, sizeof (str))); 74 75 if (dtp->dt_pcb != NULL) { 76 dtp->dt_pcb->pcb_cflags |= DTRACE_C_EATTR; 77 dtp->dt_pcb->pcb_amin = attr; 78 } else { 79 dtp->dt_cflags |= DTRACE_C_EATTR; 80 dtp->dt_amin = attr; 81 } 82 83 return (0); 84 } 85 86 static void 87 dt_coredump(void) 88 { 89 const char msg[] = "libdtrace DEBUG: [ forcing coredump ]\n"; 90 91 struct sigaction act; 92 struct rlimit lim; 93 94 (void) write(STDERR_FILENO, msg, sizeof (msg) - 1); 95 96 act.sa_handler = SIG_DFL; 97 act.sa_flags = 0; 98 99 (void) sigemptyset(&act.sa_mask); 100 (void) sigaction(SIGABRT, &act, NULL); 101 102 lim.rlim_cur = RLIM_INFINITY; 103 lim.rlim_max = RLIM_INFINITY; 104 105 (void) setrlimit(RLIMIT_CORE, &lim); 106 abort(); 107 } 108 109 /*ARGSUSED*/ 110 static int 111 dt_opt_core(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 112 { 113 static int enabled = 0; 114 115 if (arg != NULL) 116 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 117 118 if (enabled++ || atexit(dt_coredump) == 0) 119 return (0); 120 121 return (dt_set_errno(dtp, errno)); 122 } 123 124 /*ARGSUSED*/ 125 static int 126 dt_opt_cpp_hdrs(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 127 { 128 if (arg != NULL) 129 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 130 131 if (dtp->dt_pcb != NULL) 132 return (dt_set_errno(dtp, EDT_BADOPTCTX)); 133 134 if (dt_cpp_add_arg(dtp, "-H") == NULL) 135 return (dt_set_errno(dtp, EDT_NOMEM)); 136 137 return (0); 138 } 139 140 /*ARGSUSED*/ 141 static int 142 dt_opt_cpp_path(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 143 { 144 char *cpp; 145 146 if (arg == NULL) 147 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 148 149 if (dtp->dt_pcb != NULL) 150 return (dt_set_errno(dtp, EDT_BADOPTCTX)); 151 152 if ((cpp = strdup(arg)) == NULL) 153 return (dt_set_errno(dtp, EDT_NOMEM)); 154 155 dtp->dt_cpp_argv[0] = (char *)strbasename(cpp); 156 free(dtp->dt_cpp_path); 157 dtp->dt_cpp_path = cpp; 158 159 return (0); 160 } 161 162 static int 163 dt_opt_cpp_opts(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 164 { 165 char *buf; 166 size_t len; 167 const char *opt = (const char *)option; 168 169 if (opt == NULL || arg == NULL) 170 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 171 172 if (dtp->dt_pcb != NULL) 173 return (dt_set_errno(dtp, EDT_BADOPTCTX)); 174 175 len = strlen(opt) + strlen(arg) + 1; 176 buf = alloca(len); 177 178 (void) strcpy(buf, opt); 179 (void) strcat(buf, arg); 180 181 if (dt_cpp_add_arg(dtp, buf) == NULL) 182 return (dt_set_errno(dtp, EDT_NOMEM)); 183 184 return (0); 185 } 186 187 /*ARGSUSED*/ 188 static int 189 dt_opt_ctypes(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 190 { 191 int fd; 192 193 if (arg == NULL) 194 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 195 196 if ((fd = open64(arg, O_CREAT | O_WRONLY, 0666)) == -1) 197 return (dt_set_errno(dtp, errno)); 198 199 (void) close(dtp->dt_cdefs_fd); 200 dtp->dt_cdefs_fd = fd; 201 return (0); 202 } 203 204 /*ARGSUSED*/ 205 static int 206 dt_opt_droptags(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 207 { 208 dtp->dt_droptags = 1; 209 return (0); 210 } 211 212 /*ARGSUSED*/ 213 static int 214 dt_opt_dtypes(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 215 { 216 int fd; 217 218 if (arg == NULL) 219 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 220 221 if ((fd = open64(arg, O_CREAT | O_WRONLY, 0666)) == -1) 222 return (dt_set_errno(dtp, errno)); 223 224 (void) close(dtp->dt_ddefs_fd); 225 dtp->dt_ddefs_fd = fd; 226 return (0); 227 } 228 229 /*ARGSUSED*/ 230 static int 231 dt_opt_debug(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 232 { 233 if (arg != NULL) 234 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 235 236 _dtrace_debug = 1; 237 return (0); 238 } 239 240 /*ARGSUSED*/ 241 static int 242 dt_opt_iregs(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 243 { 244 int n; 245 246 if (arg == NULL || (n = atoi(arg)) <= 0) 247 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 248 249 dtp->dt_conf.dtc_difintregs = n; 250 return (0); 251 } 252 253 /*ARGSUSED*/ 254 static int 255 dt_opt_lazyload(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 256 { 257 dtp->dt_lazyload = 1; 258 259 return (0); 260 } 261 262 /*ARGSUSED*/ 263 static int 264 dt_opt_ld_path(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 265 { 266 char *ld; 267 268 if (arg == NULL) 269 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 270 271 if (dtp->dt_pcb != NULL) 272 return (dt_set_errno(dtp, EDT_BADOPTCTX)); 273 274 if ((ld = strdup(arg)) == NULL) 275 return (dt_set_errno(dtp, EDT_NOMEM)); 276 277 free(dtp->dt_ld_path); 278 dtp->dt_ld_path = ld; 279 280 return (0); 281 } 282 283 /*ARGSUSED*/ 284 static int 285 dt_opt_libdir(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 286 { 287 dt_dirpath_t *dp; 288 289 if (arg == NULL) 290 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 291 292 if ((dp = malloc(sizeof (dt_dirpath_t))) == NULL || 293 (dp->dir_path = strdup(arg)) == NULL) { 294 free(dp); 295 return (dt_set_errno(dtp, EDT_NOMEM)); 296 } 297 298 dt_list_append(&dtp->dt_lib_path, dp); 299 return (0); 300 } 301 302 /*ARGSUSED*/ 303 static int 304 dt_opt_linkmode(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 305 { 306 if (arg == NULL) 307 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 308 309 if (strcmp(arg, "kernel") == 0) 310 dtp->dt_linkmode = DT_LINK_KERNEL; 311 else if (strcmp(arg, "primary") == 0) 312 dtp->dt_linkmode = DT_LINK_PRIMARY; 313 else if (strcmp(arg, "dynamic") == 0) 314 dtp->dt_linkmode = DT_LINK_DYNAMIC; 315 else if (strcmp(arg, "static") == 0) 316 dtp->dt_linkmode = DT_LINK_STATIC; 317 else 318 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 319 320 return (0); 321 } 322 323 /*ARGSUSED*/ 324 static int 325 dt_opt_linktype(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 326 { 327 if (arg == NULL) 328 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 329 330 if (strcasecmp(arg, "elf") == 0) 331 dtp->dt_linktype = DT_LTYP_ELF; 332 else if (strcasecmp(arg, "dof") == 0) 333 dtp->dt_linktype = DT_LTYP_DOF; 334 else 335 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 336 337 return (0); 338 } 339 340 /*ARGSUSED*/ 341 static int 342 dt_opt_encoding(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 343 { 344 if (arg == NULL) 345 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 346 347 if (strcmp(arg, "ascii") == 0) 348 dtp->dt_encoding = DT_ENCODING_ASCII; 349 else if (strcmp(arg, "utf8") == 0) 350 dtp->dt_encoding = DT_ENCODING_UTF8; 351 else 352 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 353 354 return (0); 355 } 356 357 /*ARGSUSED*/ 358 static int 359 dt_opt_evaltime(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 360 { 361 if (arg == NULL) 362 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 363 364 if (strcmp(arg, "exec") == 0) 365 dtp->dt_prcmode = DT_PROC_STOP_CREATE; 366 else if (strcmp(arg, "preinit") == 0) 367 dtp->dt_prcmode = DT_PROC_STOP_PREINIT; 368 else if (strcmp(arg, "postinit") == 0) 369 dtp->dt_prcmode = DT_PROC_STOP_POSTINIT; 370 else if (strcmp(arg, "main") == 0) 371 dtp->dt_prcmode = DT_PROC_STOP_MAIN; 372 else 373 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 374 375 return (0); 376 } 377 378 /*ARGSUSED*/ 379 static int 380 dt_opt_pgmax(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 381 { 382 int n; 383 384 if (arg == NULL || (n = atoi(arg)) < 0) 385 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 386 387 dtp->dt_procs->dph_lrulim = n; 388 return (0); 389 } 390 391 static int 392 dt_opt_setenv(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 393 { 394 char **p; 395 char *var; 396 int i; 397 398 /* 399 * We can't effectively set environment variables from #pragma lines 400 * since the processes have already been spawned. 401 */ 402 if (dtp->dt_pcb != NULL) 403 return (dt_set_errno(dtp, EDT_BADOPTCTX)); 404 405 if (arg == NULL) 406 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 407 408 if (!option && strchr(arg, '=') != NULL) 409 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 410 411 for (i = 1, p = dtp->dt_proc_env; *p != NULL; i++, p++) 412 continue; 413 414 for (p = dtp->dt_proc_env; *p != NULL; p++) { 415 var = strchr(*p, '='); 416 if (var == NULL) 417 var = *p + strlen(*p); 418 if (strncmp(*p, arg, var - *p) == 0) { 419 dt_free(dtp, *p); 420 *p = dtp->dt_proc_env[i - 1]; 421 dtp->dt_proc_env[i - 1] = NULL; 422 i--; 423 } 424 } 425 426 if (option) { 427 if ((var = strdup(arg)) == NULL) 428 return (dt_set_errno(dtp, EDT_NOMEM)); 429 430 if ((p = dt_alloc(dtp, sizeof (char *) * (i + 1))) == NULL) { 431 dt_free(dtp, var); 432 return (dt_set_errno(dtp, EDT_NOMEM)); 433 } 434 435 bcopy(dtp->dt_proc_env, p, sizeof (char *) * i); 436 dt_free(dtp, dtp->dt_proc_env); 437 dtp->dt_proc_env = p; 438 439 dtp->dt_proc_env[i - 1] = var; 440 dtp->dt_proc_env[i] = NULL; 441 } 442 443 return (0); 444 } 445 446 /*ARGSUSED*/ 447 static int 448 dt_opt_stdc(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 449 { 450 if (arg == NULL) 451 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 452 453 if (dtp->dt_pcb != NULL) 454 return (dt_set_errno(dtp, EDT_BADOPTCTX)); 455 456 if (strcmp(arg, "a") == 0) 457 dtp->dt_stdcmode = DT_STDC_XA; 458 else if (strcmp(arg, "c") == 0) 459 dtp->dt_stdcmode = DT_STDC_XC; 460 else if (strcmp(arg, "s") == 0) 461 dtp->dt_stdcmode = DT_STDC_XS; 462 else if (strcmp(arg, "t") == 0) 463 dtp->dt_stdcmode = DT_STDC_XT; 464 else 465 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 466 467 return (0); 468 } 469 470 /*ARGSUSED*/ 471 static int 472 dt_opt_syslibdir(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 473 { 474 dt_dirpath_t *dp = dt_list_next(&dtp->dt_lib_path); 475 char *path; 476 477 if (arg == NULL) 478 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 479 480 if ((path = strdup(arg)) == NULL) 481 return (dt_set_errno(dtp, EDT_NOMEM)); 482 483 free(dp->dir_path); 484 dp->dir_path = path; 485 486 return (0); 487 } 488 489 /*ARGSUSED*/ 490 static int 491 dt_opt_tree(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 492 { 493 int m; 494 495 if (arg == NULL || (m = atoi(arg)) <= 0) 496 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 497 498 dtp->dt_treedump = m; 499 return (0); 500 } 501 502 /*ARGSUSED*/ 503 static int 504 dt_opt_tregs(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 505 { 506 int n; 507 508 if (arg == NULL || (n = atoi(arg)) <= 0) 509 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 510 511 dtp->dt_conf.dtc_diftupregs = n; 512 return (0); 513 } 514 515 /*ARGSUSED*/ 516 static int 517 dt_opt_xlate(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 518 { 519 if (arg == NULL) 520 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 521 522 if (strcmp(arg, "dynamic") == 0) 523 dtp->dt_xlatemode = DT_XL_DYNAMIC; 524 else if (strcmp(arg, "static") == 0) 525 dtp->dt_xlatemode = DT_XL_STATIC; 526 else 527 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 528 529 return (0); 530 } 531 532 /*ARGSUSED*/ 533 static int 534 dt_opt_cflags(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 535 { 536 if (arg != NULL) 537 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 538 539 if (dtp->dt_pcb != NULL) 540 dtp->dt_pcb->pcb_cflags |= option; 541 else 542 dtp->dt_cflags |= option; 543 544 return (0); 545 } 546 547 static int 548 dt_opt_dflags(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 549 { 550 if (arg != NULL) 551 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 552 553 dtp->dt_dflags |= option; 554 return (0); 555 } 556 557 static int 558 dt_opt_invcflags(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 559 { 560 if (arg != NULL) 561 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 562 563 if (dtp->dt_pcb != NULL) 564 dtp->dt_pcb->pcb_cflags &= ~option; 565 else 566 dtp->dt_cflags &= ~option; 567 568 return (0); 569 } 570 571 /*ARGSUSED*/ 572 static int 573 dt_opt_version(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 574 { 575 dt_version_t v; 576 577 if (arg == NULL) 578 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 579 580 if (dt_version_str2num(arg, &v) == -1) 581 return (dt_set_errno(dtp, EDT_VERSINVAL)); 582 583 if (!dt_version_defined(v)) 584 return (dt_set_errno(dtp, EDT_VERSUNDEF)); 585 586 return (dt_reduce(dtp, v)); 587 } 588 589 static int 590 dt_opt_runtime(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 591 { 592 char *end; 593 dtrace_optval_t val = 0; 594 int i; 595 596 const struct { 597 char *positive; 598 char *negative; 599 } couples[] = { 600 { "yes", "no" }, 601 { "enable", "disable" }, 602 { "enabled", "disabled" }, 603 { "true", "false" }, 604 { "on", "off" }, 605 { "set", "unset" }, 606 { NULL } 607 }; 608 609 if (arg != NULL) { 610 if (arg[0] == '\0') { 611 val = DTRACEOPT_UNSET; 612 goto out; 613 } 614 615 for (i = 0; couples[i].positive != NULL; i++) { 616 if (strcasecmp(couples[i].positive, arg) == 0) { 617 val = 1; 618 goto out; 619 } 620 621 if (strcasecmp(couples[i].negative, arg) == 0) { 622 val = DTRACEOPT_UNSET; 623 goto out; 624 } 625 } 626 627 errno = 0; 628 val = strtoull(arg, &end, 0); 629 630 if (*end != '\0' || errno != 0 || val < 0) 631 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 632 } 633 634 out: 635 dtp->dt_options[option] = val; 636 return (0); 637 } 638 639 static int 640 dt_optval_parse(const char *arg, dtrace_optval_t *rval) 641 { 642 dtrace_optval_t mul = 1; 643 size_t len; 644 char *end; 645 646 len = strlen(arg); 647 errno = 0; 648 649 switch (arg[len - 1]) { 650 case 't': 651 case 'T': 652 mul *= 1024; 653 /*FALLTHRU*/ 654 case 'g': 655 case 'G': 656 mul *= 1024; 657 /*FALLTHRU*/ 658 case 'm': 659 case 'M': 660 mul *= 1024; 661 /*FALLTHRU*/ 662 case 'k': 663 case 'K': 664 mul *= 1024; 665 /*FALLTHRU*/ 666 default: 667 break; 668 } 669 670 errno = 0; 671 *rval = strtoull(arg, &end, 0) * mul; 672 673 if ((mul > 1 && end != &arg[len - 1]) || (mul == 1 && *end != '\0') || 674 *rval < 0 || errno != 0) 675 return (-1); 676 677 return (0); 678 } 679 680 static int 681 dt_opt_size(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 682 { 683 dtrace_optval_t val = 0; 684 685 if (arg != NULL && dt_optval_parse(arg, &val) != 0) 686 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 687 688 dtp->dt_options[option] = val; 689 return (0); 690 } 691 692 static int 693 dt_opt_rate(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 694 { 695 char *end; 696 int i; 697 dtrace_optval_t mul = 1, val = 0; 698 699 const struct { 700 char *name; 701 hrtime_t mul; 702 } suffix[] = { 703 { "ns", NANOSEC / NANOSEC }, 704 { "nsec", NANOSEC / NANOSEC }, 705 { "us", NANOSEC / MICROSEC }, 706 { "usec", NANOSEC / MICROSEC }, 707 { "ms", NANOSEC / MILLISEC }, 708 { "msec", NANOSEC / MILLISEC }, 709 { "s", NANOSEC / SEC }, 710 { "sec", NANOSEC / SEC }, 711 { "m", NANOSEC * (hrtime_t)60 }, 712 { "min", NANOSEC * (hrtime_t)60 }, 713 { "h", NANOSEC * (hrtime_t)60 * (hrtime_t)60 }, 714 { "hour", NANOSEC * (hrtime_t)60 * (hrtime_t)60 }, 715 { "d", NANOSEC * (hrtime_t)(24 * 60 * 60) }, 716 { "day", NANOSEC * (hrtime_t)(24 * 60 * 60) }, 717 { "hz", 0 }, 718 { NULL } 719 }; 720 721 if (arg != NULL) { 722 errno = 0; 723 val = strtoull(arg, &end, 0); 724 725 for (i = 0; suffix[i].name != NULL; i++) { 726 if (strcasecmp(suffix[i].name, end) == 0) { 727 mul = suffix[i].mul; 728 break; 729 } 730 } 731 732 if (suffix[i].name == NULL && *end != '\0' || val < 0) 733 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 734 735 if (mul == 0) { 736 /* 737 * The rate has been specified in frequency-per-second. 738 */ 739 if (val != 0) 740 val = NANOSEC / val; 741 } else { 742 val *= mul; 743 } 744 } 745 746 dtp->dt_options[option] = val; 747 return (0); 748 } 749 750 /* 751 * When setting the strsize option, set the option in the dt_options array 752 * using dt_opt_size() as usual, and then update the definition of the CTF 753 * type for the D intrinsic "string" to be an array of the corresponding size. 754 * If any errors occur, reset dt_options[option] to its previous value. 755 */ 756 static int 757 dt_opt_strsize(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 758 { 759 dtrace_optval_t val = dtp->dt_options[option]; 760 ctf_file_t *fp = DT_STR_CTFP(dtp); 761 ctf_id_t type = ctf_type_resolve(fp, DT_STR_TYPE(dtp)); 762 ctf_arinfo_t r; 763 764 if (dt_opt_size(dtp, arg, option) != 0) 765 return (-1); /* dt_errno is set for us */ 766 767 if (dtp->dt_options[option] > UINT_MAX) { 768 dtp->dt_options[option] = val; 769 return (dt_set_errno(dtp, EOVERFLOW)); 770 } 771 772 if (ctf_array_info(fp, type, &r) == CTF_ERR) { 773 dtp->dt_options[option] = val; 774 dtp->dt_ctferr = ctf_errno(fp); 775 return (dt_set_errno(dtp, EDT_CTF)); 776 } 777 778 r.ctr_nelems = (uint_t)dtp->dt_options[option]; 779 780 if (ctf_set_array(fp, type, &r) == CTF_ERR || 781 ctf_update(fp) == CTF_ERR) { 782 dtp->dt_options[option] = val; 783 dtp->dt_ctferr = ctf_errno(fp); 784 return (dt_set_errno(dtp, EDT_CTF)); 785 } 786 787 return (0); 788 } 789 790 static const struct { 791 const char *dtbp_name; 792 int dtbp_policy; 793 } _dtrace_bufpolicies[] = { 794 { "ring", DTRACEOPT_BUFPOLICY_RING }, 795 { "fill", DTRACEOPT_BUFPOLICY_FILL }, 796 { "switch", DTRACEOPT_BUFPOLICY_SWITCH }, 797 { NULL, 0 } 798 }; 799 800 /*ARGSUSED*/ 801 static int 802 dt_opt_bufpolicy(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 803 { 804 dtrace_optval_t policy = DTRACEOPT_UNSET; 805 int i; 806 807 if (arg == NULL) 808 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 809 810 for (i = 0; _dtrace_bufpolicies[i].dtbp_name != NULL; i++) { 811 if (strcmp(_dtrace_bufpolicies[i].dtbp_name, arg) == 0) { 812 policy = _dtrace_bufpolicies[i].dtbp_policy; 813 break; 814 } 815 } 816 817 if (policy == DTRACEOPT_UNSET) 818 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 819 820 dtp->dt_options[DTRACEOPT_BUFPOLICY] = policy; 821 822 return (0); 823 } 824 825 static const struct { 826 const char *dtbr_name; 827 int dtbr_policy; 828 } _dtrace_bufresize[] = { 829 { "auto", DTRACEOPT_BUFRESIZE_AUTO }, 830 { "manual", DTRACEOPT_BUFRESIZE_MANUAL }, 831 { NULL, 0 } 832 }; 833 834 /*ARGSUSED*/ 835 static int 836 dt_opt_bufresize(dtrace_hdl_t *dtp, const char *arg, uintptr_t option) 837 { 838 dtrace_optval_t policy = DTRACEOPT_UNSET; 839 int i; 840 841 if (arg == NULL) 842 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 843 844 for (i = 0; _dtrace_bufresize[i].dtbr_name != NULL; i++) { 845 if (strcmp(_dtrace_bufresize[i].dtbr_name, arg) == 0) { 846 policy = _dtrace_bufresize[i].dtbr_policy; 847 break; 848 } 849 } 850 851 if (policy == DTRACEOPT_UNSET) 852 return (dt_set_errno(dtp, EDT_BADOPTVAL)); 853 854 dtp->dt_options[DTRACEOPT_BUFRESIZE] = policy; 855 856 return (0); 857 } 858 859 int 860 dt_options_load(dtrace_hdl_t *dtp) 861 { 862 dof_hdr_t hdr, *dof; 863 dof_sec_t *sec; 864 size_t offs; 865 int i; 866 867 /* 868 * To load the option values, we need to ask the kernel to provide its 869 * DOF, which we'll sift through to look for OPTDESC sections. 870 */ 871 bzero(&hdr, sizeof (dof_hdr_t)); 872 hdr.dofh_loadsz = sizeof (dof_hdr_t); 873 874 #ifdef illumos 875 if (dt_ioctl(dtp, DTRACEIOC_DOFGET, &hdr) == -1) 876 #else 877 dof = &hdr; 878 if (dt_ioctl(dtp, DTRACEIOC_DOFGET, &dof) == -1) 879 #endif 880 return (dt_set_errno(dtp, errno)); 881 882 if (hdr.dofh_loadsz < sizeof (dof_hdr_t)) 883 return (dt_set_errno(dtp, EINVAL)); 884 885 dof = alloca(hdr.dofh_loadsz); 886 bzero(dof, sizeof (dof_hdr_t)); 887 dof->dofh_loadsz = hdr.dofh_loadsz; 888 889 for (i = 0; i < DTRACEOPT_MAX; i++) 890 dtp->dt_options[i] = DTRACEOPT_UNSET; 891 892 #ifdef illumos 893 if (dt_ioctl(dtp, DTRACEIOC_DOFGET, dof) == -1) 894 #else 895 if (dt_ioctl(dtp, DTRACEIOC_DOFGET, &dof) == -1) 896 #endif 897 return (dt_set_errno(dtp, errno)); 898 899 for (i = 0; i < dof->dofh_secnum; i++) { 900 sec = (dof_sec_t *)(uintptr_t)((uintptr_t)dof + 901 dof->dofh_secoff + i * dof->dofh_secsize); 902 903 if (sec->dofs_type != DOF_SECT_OPTDESC) 904 continue; 905 906 break; 907 } 908 909 for (offs = 0; offs < sec->dofs_size; offs += sec->dofs_entsize) { 910 dof_optdesc_t *opt = (dof_optdesc_t *)(uintptr_t) 911 ((uintptr_t)dof + sec->dofs_offset + offs); 912 913 if (opt->dofo_strtab != DOF_SECIDX_NONE) 914 continue; 915 916 if (opt->dofo_option >= DTRACEOPT_MAX) 917 continue; 918 919 dtp->dt_options[opt->dofo_option] = opt->dofo_value; 920 } 921 922 return (0); 923 } 924 925 typedef struct dt_option { 926 const char *o_name; 927 int (*o_func)(dtrace_hdl_t *, const char *, uintptr_t); 928 uintptr_t o_option; 929 } dt_option_t; 930 931 /* 932 * Compile-time options. 933 */ 934 static const dt_option_t _dtrace_ctoptions[] = { 935 { "aggpercpu", dt_opt_agg, DTRACE_A_PERCPU }, 936 { "amin", dt_opt_amin }, 937 { "argref", dt_opt_cflags, DTRACE_C_ARGREF }, 938 { "core", dt_opt_core }, 939 { "cpp", dt_opt_cflags, DTRACE_C_CPP }, 940 { "cpphdrs", dt_opt_cpp_hdrs }, 941 { "cpppath", dt_opt_cpp_path }, 942 { "ctypes", dt_opt_ctypes }, 943 { "defaultargs", dt_opt_cflags, DTRACE_C_DEFARG }, 944 { "dtypes", dt_opt_dtypes }, 945 { "debug", dt_opt_debug }, 946 { "define", dt_opt_cpp_opts, (uintptr_t)"-D" }, 947 { "droptags", dt_opt_droptags }, 948 { "empty", dt_opt_cflags, DTRACE_C_EMPTY }, 949 { "encoding", dt_opt_encoding }, 950 { "errtags", dt_opt_cflags, DTRACE_C_ETAGS }, 951 { "evaltime", dt_opt_evaltime }, 952 { "incdir", dt_opt_cpp_opts, (uintptr_t)"-I" }, 953 { "iregs", dt_opt_iregs }, 954 { "kdefs", dt_opt_invcflags, DTRACE_C_KNODEF }, 955 { "knodefs", dt_opt_cflags, DTRACE_C_KNODEF }, 956 { "late", dt_opt_xlate }, 957 { "lazyload", dt_opt_lazyload }, 958 { "ldpath", dt_opt_ld_path }, 959 { "libdir", dt_opt_libdir }, 960 { "linkmode", dt_opt_linkmode }, 961 { "linktype", dt_opt_linktype }, 962 { "nolibs", dt_opt_cflags, DTRACE_C_NOLIBS }, 963 { "pgmax", dt_opt_pgmax }, 964 { "pspec", dt_opt_cflags, DTRACE_C_PSPEC }, 965 { "setenv", dt_opt_setenv, 1 }, 966 { "stdc", dt_opt_stdc }, 967 { "strip", dt_opt_dflags, DTRACE_D_STRIP }, 968 { "syslibdir", dt_opt_syslibdir }, 969 { "tree", dt_opt_tree }, 970 { "tregs", dt_opt_tregs }, 971 { "udefs", dt_opt_invcflags, DTRACE_C_UNODEF }, 972 { "undef", dt_opt_cpp_opts, (uintptr_t)"-U" }, 973 { "unodefs", dt_opt_cflags, DTRACE_C_UNODEF }, 974 { "unsetenv", dt_opt_setenv, 0 }, 975 { "verbose", dt_opt_cflags, DTRACE_C_DIFV }, 976 { "version", dt_opt_version }, 977 { "zdefs", dt_opt_cflags, DTRACE_C_ZDEFS }, 978 { NULL, NULL, 0 } 979 }; 980 981 /* 982 * Run-time options. 983 */ 984 static const dt_option_t _dtrace_rtoptions[] = { 985 { "aggsize", dt_opt_size, DTRACEOPT_AGGSIZE }, 986 { "bufsize", dt_opt_size, DTRACEOPT_BUFSIZE }, 987 { "bufpolicy", dt_opt_bufpolicy, DTRACEOPT_BUFPOLICY }, 988 { "bufresize", dt_opt_bufresize, DTRACEOPT_BUFRESIZE }, 989 { "cleanrate", dt_opt_rate, DTRACEOPT_CLEANRATE }, 990 { "cpu", dt_opt_runtime, DTRACEOPT_CPU }, 991 { "destructive", dt_opt_runtime, DTRACEOPT_DESTRUCTIVE }, 992 { "dynvarsize", dt_opt_size, DTRACEOPT_DYNVARSIZE }, 993 { "grabanon", dt_opt_runtime, DTRACEOPT_GRABANON }, 994 { "jstackframes", dt_opt_runtime, DTRACEOPT_JSTACKFRAMES }, 995 { "jstackstrsize", dt_opt_size, DTRACEOPT_JSTACKSTRSIZE }, 996 { "nspec", dt_opt_runtime, DTRACEOPT_NSPEC }, 997 { "specsize", dt_opt_size, DTRACEOPT_SPECSIZE }, 998 { "stackframes", dt_opt_runtime, DTRACEOPT_STACKFRAMES }, 999 { "statusrate", dt_opt_rate, DTRACEOPT_STATUSRATE }, 1000 { "strsize", dt_opt_strsize, DTRACEOPT_STRSIZE }, 1001 { "ustackframes", dt_opt_runtime, DTRACEOPT_USTACKFRAMES }, 1002 { "temporal", dt_opt_runtime, DTRACEOPT_TEMPORAL }, 1003 { NULL, NULL, 0 } 1004 }; 1005 1006 /* 1007 * Dynamic run-time options. 1008 */ 1009 static const dt_option_t _dtrace_drtoptions[] = { 1010 { "agghist", dt_opt_runtime, DTRACEOPT_AGGHIST }, 1011 { "aggpack", dt_opt_runtime, DTRACEOPT_AGGPACK }, 1012 { "aggrate", dt_opt_rate, DTRACEOPT_AGGRATE }, 1013 { "aggsortkey", dt_opt_runtime, DTRACEOPT_AGGSORTKEY }, 1014 { "aggsortkeypos", dt_opt_runtime, DTRACEOPT_AGGSORTKEYPOS }, 1015 { "aggsortpos", dt_opt_runtime, DTRACEOPT_AGGSORTPOS }, 1016 { "aggsortrev", dt_opt_runtime, DTRACEOPT_AGGSORTREV }, 1017 { "aggzoom", dt_opt_runtime, DTRACEOPT_AGGZOOM }, 1018 { "flowindent", dt_opt_runtime, DTRACEOPT_FLOWINDENT }, 1019 { "quiet", dt_opt_runtime, DTRACEOPT_QUIET }, 1020 { "rawbytes", dt_opt_runtime, DTRACEOPT_RAWBYTES }, 1021 { "stackindent", dt_opt_runtime, DTRACEOPT_STACKINDENT }, 1022 { "switchrate", dt_opt_rate, DTRACEOPT_SWITCHRATE }, 1023 { NULL, NULL, 0 } 1024 }; 1025 1026 int 1027 dtrace_getopt(dtrace_hdl_t *dtp, const char *opt, dtrace_optval_t *val) 1028 { 1029 const dt_option_t *op; 1030 1031 if (opt == NULL) 1032 return (dt_set_errno(dtp, EINVAL)); 1033 1034 /* 1035 * We only need to search the run-time options -- it's not legal 1036 * to get the values of compile-time options. 1037 */ 1038 for (op = _dtrace_rtoptions; op->o_name != NULL; op++) { 1039 if (strcmp(op->o_name, opt) == 0) { 1040 *val = dtp->dt_options[op->o_option]; 1041 return (0); 1042 } 1043 } 1044 1045 for (op = _dtrace_drtoptions; op->o_name != NULL; op++) { 1046 if (strcmp(op->o_name, opt) == 0) { 1047 *val = dtp->dt_options[op->o_option]; 1048 return (0); 1049 } 1050 } 1051 1052 return (dt_set_errno(dtp, EDT_BADOPTNAME)); 1053 } 1054 1055 int 1056 dtrace_setopt(dtrace_hdl_t *dtp, const char *opt, const char *val) 1057 { 1058 const dt_option_t *op; 1059 1060 if (opt == NULL) 1061 return (dt_set_errno(dtp, EINVAL)); 1062 1063 for (op = _dtrace_ctoptions; op->o_name != NULL; op++) { 1064 if (strcmp(op->o_name, opt) == 0) 1065 return (op->o_func(dtp, val, op->o_option)); 1066 } 1067 1068 for (op = _dtrace_drtoptions; op->o_name != NULL; op++) { 1069 if (strcmp(op->o_name, opt) == 0) 1070 return (op->o_func(dtp, val, op->o_option)); 1071 } 1072 1073 for (op = _dtrace_rtoptions; op->o_name != NULL; op++) { 1074 if (strcmp(op->o_name, opt) == 0) { 1075 /* 1076 * Only dynamic run-time options may be set while 1077 * tracing is active. 1078 */ 1079 if (dtp->dt_active) 1080 return (dt_set_errno(dtp, EDT_ACTIVE)); 1081 1082 return (op->o_func(dtp, val, op->o_option)); 1083 } 1084 } 1085 1086 return (dt_set_errno(dtp, EDT_BADOPTNAME)); 1087 } 1088