1 /* 2 * CDDL HEADER START 3 * 4 * The contents of this file are subject to the terms of the 5 * Common Development and Distribution License (the "License"). 6 * You may not use this file except in compliance with the License. 7 * 8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 9 * or http://www.opensolaris.org/os/licensing. 10 * See the License for the specific language governing permissions 11 * and limitations under the License. 12 * 13 * When distributing Covered Code, include this CDDL HEADER in each 14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 15 * If applicable, add the following below this CDDL HEADER, with the 16 * fields enclosed by brackets "[]" replaced with your own identifying 17 * information: Portions Copyright [yyyy] [name of copyright owner] 18 * 19 * CDDL HEADER END 20 */ 21 /* 22 * Copyright 2009 Sun Microsystems, Inc. All rights reserved. 23 * Use is subject to license terms. 24 */ 25 26 /* 27 * Copyright 2018 Joyent, Inc. 28 */ 29 30 #include <sys/types.h> 31 #include <sys/processor.h> 32 #include <sys/pset.h> 33 #include <sys/lwp.h> 34 #include <sys/priocntl.h> 35 #include <sys/fxpriocntl.h> 36 #include <time.h> 37 #include <stdio.h> 38 #include <stdlib.h> 39 #include <inttypes.h> 40 #include <unistd.h> 41 #include <limits.h> 42 #include <string.h> 43 #include <strings.h> 44 #include <thread.h> 45 #include <errno.h> 46 #include <libintl.h> 47 #include <locale.h> 48 #include <kstat.h> 49 #include <synch.h> 50 #include <libcpc.h> 51 #include <sys/resource.h> 52 53 #include "cpucmds.h" 54 #include "statcommon.h" 55 56 static struct options { 57 int debug; 58 int dotitle; 59 int dohelp; 60 int dotick; 61 int dosoaker; 62 int doperiod; 63 char *pgmname; 64 uint_t mseconds; 65 uint_t nsamples; 66 uint_t nsets; 67 uint_t mseconds_rest; 68 cpc_setgrp_t *master; 69 } __options; 70 71 /* 72 * States for soaker threads. 73 */ 74 #define SOAK_PAUSE 0 75 #define SOAK_RUN 1 76 77 struct tstate { 78 processorid_t cpuid; 79 int chip_id; 80 cpc_setgrp_t *sgrp; 81 int status; 82 thread_t tid; 83 int soak_state; 84 mutex_t soak_lock; 85 cond_t soak_cv; 86 }; 87 88 static const struct options *opts = (const struct options *)&__options; 89 90 static cpc_t *cpc; 91 92 struct tstate *gstate; 93 static int ncpus; 94 static int max_chip_id; 95 static int *chip_designees; /* cpuid of CPU which counts for phs chip */ 96 static int smt = 0; /* If set, cpustat needs to be SMT-aware. */ 97 static pcinfo_t fxinfo = { 0, "FX", NULL }; /* FX scheduler class info */ 98 99 static uint_t timestamp_fmt = NODATE; 100 101 /*ARGSUSED*/ 102 static void 103 cpustat_errfn(const char *fn, int subcode, const char *fmt, va_list ap) 104 { 105 (void) fprintf(stderr, "%s: ", opts->pgmname); 106 if (opts->debug) 107 (void) fprintf(stderr, "%s: ", fn); 108 (void) vfprintf(stderr, fmt, ap); 109 } 110 111 static int cpustat(void); 112 static int get_chipid(kstat_ctl_t *kc, processorid_t cpuid); 113 static void *soaker(void *arg); 114 115 116 #if !defined(TEXT_DOMAIN) 117 #define TEXT_DOMAIN "SYS_TEST" 118 #endif 119 120 int 121 main(int argc, char *argv[]) 122 { 123 struct options *opts = &__options; 124 int c, errcnt = 0, ret; 125 cpc_setgrp_t *sgrp; 126 char *errstr; 127 double period; 128 char *endp; 129 struct rlimit rl; 130 131 (void) setlocale(LC_ALL, ""); 132 (void) textdomain(TEXT_DOMAIN); 133 134 if ((opts->pgmname = strrchr(argv[0], '/')) == NULL) 135 opts->pgmname = argv[0]; 136 else 137 opts->pgmname++; 138 139 /* Make sure we can open enough files */ 140 rl.rlim_max = rl.rlim_cur = RLIM_INFINITY; 141 if (setrlimit(RLIMIT_NOFILE, &rl) != 0) { 142 errstr = strerror(errno); 143 (void) fprintf(stderr, 144 gettext("%s: setrlimit failed - %s\n"), 145 opts->pgmname, errstr); 146 } 147 148 if ((cpc = cpc_open(CPC_VER_CURRENT)) == NULL) { 149 errstr = strerror(errno); 150 (void) fprintf(stderr, gettext("%s: cannot access performance " 151 "counters - %s\n"), opts->pgmname, errstr); 152 return (1); 153 } 154 155 (void) cpc_seterrhndlr(cpc, cpustat_errfn); 156 strtoset_errfn = cpustat_errfn; 157 158 /* 159 * Check to see if cpustat needs to be SMT-aware. 160 */ 161 smt = smt_limited_cpc_hw(cpc); 162 163 /* 164 * Establish some defaults 165 */ 166 opts->mseconds = 5000; 167 opts->nsamples = UINT_MAX; 168 opts->dotitle = 1; 169 if ((opts->master = cpc_setgrp_new(cpc, smt)) == NULL) { 170 (void) fprintf(stderr, gettext("%s: out of heap\n"), 171 opts->pgmname); 172 return (1); 173 } 174 175 while ((c = getopt(argc, argv, "Dc:hntT:sp:")) != EOF && errcnt == 0) 176 switch (c) { 177 case 'D': /* enable debugging */ 178 opts->debug++; 179 break; 180 case 'c': /* specify statistics */ 181 if ((sgrp = cpc_setgrp_newset(opts->master, 182 optarg, &errcnt)) != NULL) 183 opts->master = sgrp; 184 break; 185 case 'n': /* no titles */ 186 opts->dotitle = 0; 187 break; 188 case 'p': /* periodic behavior */ 189 opts->doperiod = 1; 190 period = strtod(optarg, &endp); 191 if (*endp != '\0') { 192 (void) fprintf(stderr, gettext("%s: invalid " 193 "parameter \"%s\"\n"), opts->pgmname, 194 optarg); 195 errcnt++; 196 } 197 break; 198 case 's': /* run soaker thread */ 199 opts->dosoaker = 1; 200 break; 201 case 't': /* print %tick */ 202 opts->dotick = 1; 203 break; 204 case 'T': 205 if (optarg) { 206 if (*optarg == 'u') 207 timestamp_fmt = UDATE; 208 else if (*optarg == 'd') 209 timestamp_fmt = DDATE; 210 else 211 errcnt++; 212 } else { 213 errcnt++; 214 } 215 break; 216 case 'h': /* help */ 217 opts->dohelp = 1; 218 break; 219 case '?': 220 default: 221 errcnt++; 222 break; 223 } 224 225 switch (argc - optind) { 226 case 0: 227 break; 228 case 2: 229 opts->nsamples = strtol(argv[optind + 1], &endp, 10); 230 if (*endp != '\0') { 231 (void) fprintf(stderr, 232 gettext("%s: invalid argument \"%s\"\n"), 233 opts->pgmname, argv[optind + 1]); 234 errcnt++; 235 break; 236 } 237 /*FALLTHROUGH*/ 238 case 1: 239 opts->mseconds = (uint_t)(strtod(argv[optind], &endp) * 1000.0); 240 if (*endp != '\0') { 241 (void) fprintf(stderr, 242 gettext("%s: invalid argument \"%s\"\n"), 243 opts->pgmname, argv[optind]); 244 errcnt++; 245 } 246 break; 247 default: 248 errcnt++; 249 break; 250 } 251 252 if (opts->nsamples == 0 || opts->mseconds == 0) 253 errcnt++; 254 255 if (errcnt != 0 || opts->dohelp || 256 (opts->nsets = cpc_setgrp_numsets(opts->master)) == 0) { 257 (void) fprintf(opts->dohelp ? stdout : stderr, gettext( 258 "Usage:\n\t%s -c spec [-c spec]... [-p period] [-T u|d]\n" 259 "\t\t[-sntD] [interval [count]]\n\n" 260 "\t-c spec\t specify processor events to be monitored\n" 261 "\t-n\t suppress titles\n" 262 "\t-p period cycle through event list periodically\n" 263 "\t-s\t run user soaker thread for system-only events\n" 264 "\t-t\t include %s register\n" 265 "\t-T d|u\t Display a timestamp in date (d) or unix " 266 "time_t (u)\n" 267 "\t-D\t enable debug mode\n" 268 "\t-h\t print extended usage information\n\n" 269 "\tUse cputrack(1) to monitor per-process statistics.\n"), 270 opts->pgmname, CPC_TICKREG_NAME); 271 if (opts->dohelp) { 272 (void) putchar('\n'); 273 (void) capabilities(cpc, stdout); 274 exit(0); 275 } 276 exit(2); 277 } 278 279 /* 280 * If the user requested periodic behavior, calculate the rest time 281 * between cycles. 282 */ 283 if (opts->doperiod) { 284 opts->mseconds_rest = (uint_t)((period * 1000.0) - 285 (opts->mseconds * opts->nsets)); 286 if ((int)opts->mseconds_rest < 0) 287 opts->mseconds_rest = 0; 288 if (opts->nsamples != UINT_MAX) 289 opts->nsamples *= opts->nsets; 290 } 291 292 cpc_setgrp_reset(opts->master); 293 (void) setvbuf(stdout, NULL, _IOLBF, 0); 294 295 /* 296 * By design, cpustat (regrettably) has multiple threads racing in 297 * write(2) to generate output. As there are no guarantees made with 298 * respect to the atomicity of concurrent writes on non-O_APPEND file 299 * descriptors, we must set O_APPEND on stdout to assure that no output 300 * is lost. If cpustat is rearchitected such that only one thread is 301 * generating output (which would also assure that the output is always 302 * in a consistent order), this code should be removed. 303 */ 304 if (fcntl(1, F_SETFL, fcntl(1, F_GETFL) | O_APPEND) == -1) { 305 (void) fprintf(stderr, gettext("%s: cannot set output to be " 306 "append-only - %s\n"), opts->pgmname, strerror(errno)); 307 return (1); 308 } 309 310 /* 311 * If no system-mode only sets were created, no soaker threads will be 312 * needed. 313 */ 314 if (opts->dosoaker == 1 && cpc_setgrp_has_sysonly(opts->master) == 0) 315 opts->dosoaker = 0; 316 317 ret = cpustat(); 318 319 (void) cpc_close(cpc); 320 321 return (ret); 322 } 323 324 static void 325 print_title(cpc_setgrp_t *sgrp) 326 { 327 (void) printf("%7s %3s %5s ", "time", "cpu", "event"); 328 if (opts->dotick) 329 (void) printf("%9s ", CPC_TICKREG_NAME); 330 (void) printf("%s\n", cpc_setgrp_gethdr(sgrp)); 331 } 332 333 static void 334 print_sample(processorid_t cpuid, cpc_buf_t *buf, int nreq, const char *setname, 335 int sibling) 336 { 337 char line[1024]; 338 int ccnt; 339 int i; 340 uint64_t val; 341 uint64_t tick; 342 hrtime_t hrtime; 343 344 hrtime = cpc_buf_hrtime(cpc, buf); 345 tick = cpc_buf_tick(cpc, buf); 346 347 ccnt = snprintf(line, sizeof (line), "%7.3f %3d %5s ", 348 mstimestamp(hrtime), (int)cpuid, "tick"); 349 if (opts->dotick) 350 ccnt += snprintf(line + ccnt, sizeof (line) - ccnt, 351 "%9" PRId64 " ", tick); 352 for (i = 0; i < nreq; i++) { 353 (void) cpc_buf_get(cpc, buf, i, &val); 354 ccnt += snprintf(line + ccnt, sizeof (line) - ccnt, 355 "%9" PRId64 " ", val); 356 } 357 if (opts->nsets > 1) 358 ccnt += snprintf(line + ccnt, sizeof (line) - ccnt, 359 " # %s\n", setname); 360 else 361 ccnt += snprintf(line + ccnt, sizeof (line) - ccnt, "\n"); 362 363 if (sibling) { 364 /* 365 * This sample is being printed for a "sibling" CPU -- that is, 366 * a CPU which does not have its own CPC set bound. It is being 367 * measured via a set bound to another CPU sharing its physical 368 * processor. 369 */ 370 int designee = chip_designees[gstate[cpuid].chip_id]; 371 char *p; 372 373 if ((p = strrchr(line, '#')) == NULL) 374 p = strrchr(line, '\n'); 375 376 if (p != NULL) { 377 *p = '\0'; 378 ccnt = strlen(line); 379 ccnt += snprintf(line + ccnt, sizeof (line) - ccnt, 380 "# counter shared with CPU %d\n", designee); 381 } 382 } 383 384 if (timestamp_fmt != NODATE) 385 print_timestamp(timestamp_fmt); 386 if (ccnt > sizeof (line)) 387 ccnt = sizeof (line); 388 if (ccnt > 0) 389 (void) write(1, line, ccnt); 390 391 /* 392 * If this CPU is the chip designee for any other CPUs, print a line for 393 * them here. 394 */ 395 if (smt && (sibling == 0)) { 396 for (i = 0; i < ncpus; i++) { 397 if ((i != cpuid) && (gstate[i].cpuid != -1) && 398 (chip_designees[gstate[i].chip_id] == cpuid)) 399 print_sample(i, buf, nreq, setname, 1); 400 } 401 } 402 } 403 404 static void 405 print_total(int ncpus, cpc_buf_t *buf, int nreq, const char *setname) 406 { 407 int i; 408 uint64_t val; 409 410 (void) printf("%7.3f %3d %5s ", mstimestamp(cpc_buf_hrtime(cpc, buf)), 411 ncpus, "total"); 412 if (opts->dotick) 413 (void) printf("%9" PRId64 " ", cpc_buf_tick(cpc, buf)); 414 for (i = 0; i < nreq; i++) { 415 (void) cpc_buf_get(cpc, buf, i, &val); 416 (void) printf("%9" PRId64 " ", val); 417 } 418 if (opts->nsets > 1) 419 (void) printf(" # %s", setname); 420 (void) fputc('\n', stdout); 421 } 422 423 #define NSECS_PER_MSEC 1000000ll 424 #define NSECS_PER_SEC 1000000000ll 425 426 static void * 427 gtick(void *arg) 428 { 429 struct tstate *state = arg; 430 char *errstr; 431 uint_t nsamples; 432 uint_t sample_cnt = 1; 433 hrtime_t ht, htdelta, restdelta; 434 cpc_setgrp_t *sgrp = state->sgrp; 435 cpc_set_t *this = cpc_setgrp_getset(sgrp); 436 const char *name = cpc_setgrp_getname(sgrp); 437 cpc_buf_t **data1, **data2, **scratch; 438 cpc_buf_t *tmp; 439 int nreqs; 440 thread_t tid; 441 442 htdelta = NSECS_PER_MSEC * opts->mseconds; 443 restdelta = NSECS_PER_MSEC * opts->mseconds_rest; 444 ht = gethrtime(); 445 446 /* 447 * If this CPU is SMT, we run one gtick() thread per _physical_ CPU, 448 * instead of per cpu_t. The following check returns if it detects that 449 * this cpu_t has not been designated to do the counting for this 450 * physical CPU. 451 */ 452 if (smt && chip_designees[state->chip_id] != state->cpuid) 453 return (NULL); 454 455 /* 456 * If we need to run a soaker thread on this CPU, start it here. 457 */ 458 if (opts->dosoaker) { 459 if (cond_init(&state->soak_cv, USYNC_THREAD, NULL) != 0) 460 goto bad; 461 if (mutex_init(&state->soak_lock, USYNC_THREAD, 462 NULL) != 0) 463 goto bad; 464 (void) mutex_lock(&state->soak_lock); 465 state->soak_state = SOAK_PAUSE; 466 if (thr_create(NULL, 0, soaker, state, NULL, &tid) != 0) 467 goto bad; 468 469 while (state->soak_state == SOAK_PAUSE) 470 (void) cond_wait(&state->soak_cv, 471 &state->soak_lock); 472 (void) mutex_unlock(&state->soak_lock); 473 474 /* 475 * If the soaker needs to pause for the first set, stop it now. 476 */ 477 if (cpc_setgrp_sysonly(sgrp) == 0) { 478 (void) mutex_lock(&state->soak_lock); 479 state->soak_state = SOAK_PAUSE; 480 (void) mutex_unlock(&state->soak_lock); 481 } 482 } 483 if (cpc_bind_cpu(cpc, state->cpuid, this, 0) == -1) 484 goto bad; 485 486 for (nsamples = opts->nsamples; nsamples; nsamples--, sample_cnt++) { 487 hrtime_t htnow; 488 struct timespec ts; 489 490 nreqs = cpc_setgrp_getbufs(sgrp, &data1, &data2, &scratch); 491 492 ht += htdelta; 493 htnow = gethrtime(); 494 if (ht <= htnow) 495 continue; 496 ts.tv_sec = (time_t)((ht - htnow) / NSECS_PER_SEC); 497 ts.tv_nsec = (suseconds_t)((ht - htnow) % NSECS_PER_SEC); 498 499 (void) nanosleep(&ts, NULL); 500 501 if (opts->nsets == 1) { 502 /* 503 * If we're dealing with one set, buffer usage is: 504 * 505 * data1 = most recent data snapshot 506 * data2 = previous data snapshot 507 * scratch = used for diffing data1 and data2 508 * 509 * Save the snapshot from the previous sample in data2 510 * before putting the current sample in data1. 511 */ 512 tmp = *data1; 513 *data1 = *data2; 514 *data2 = tmp; 515 if (cpc_set_sample(cpc, this, *data1) != 0) 516 goto bad; 517 cpc_buf_sub(cpc, *scratch, *data1, *data2); 518 519 print_sample(state->cpuid, *scratch, nreqs, name, 0); 520 } else { 521 /* 522 * More than one set is in use (multiple -c options 523 * given). Buffer usage in this case is: 524 * 525 * data1 = total counts for this set since program began 526 * data2 = unused 527 * scratch = most recent data snapshot 528 */ 529 name = cpc_setgrp_getname(sgrp); 530 nreqs = cpc_setgrp_getbufs(sgrp, &data1, &data2, 531 &scratch); 532 533 if (cpc_set_sample(cpc, this, *scratch) != 0) 534 goto bad; 535 536 cpc_buf_add(cpc, *data1, *data1, *scratch); 537 538 if (cpc_unbind(cpc, this) != 0) 539 (void) fprintf(stderr, gettext("%s: error " 540 "unbinding on cpu %d - %s\n"), 541 opts->pgmname, state->cpuid, 542 strerror(errno)); 543 544 this = cpc_setgrp_nextset(sgrp); 545 546 print_sample(state->cpuid, *scratch, nreqs, name, 0); 547 548 /* 549 * If periodic behavior was requested, rest here. 550 */ 551 if (opts->doperiod && opts->mseconds_rest > 0 && 552 (sample_cnt % opts->nsets) == 0) { 553 /* 554 * Stop the soaker while the tool rests. 555 */ 556 if (opts->dosoaker) { 557 (void) mutex_lock(&state->soak_lock); 558 if (state->soak_state == SOAK_RUN) 559 state->soak_state = SOAK_PAUSE; 560 (void) mutex_unlock(&state->soak_lock); 561 } 562 563 htnow = gethrtime(); 564 ht += restdelta; 565 ts.tv_sec = (time_t)((ht - htnow) / 566 NSECS_PER_SEC); 567 ts.tv_nsec = (suseconds_t)((ht - htnow) % 568 NSECS_PER_SEC); 569 570 (void) nanosleep(&ts, NULL); 571 } 572 573 /* 574 * Start or stop the soaker if needed. 575 */ 576 if (opts->dosoaker) { 577 (void) mutex_lock(&state->soak_lock); 578 if (cpc_setgrp_sysonly(sgrp) && 579 state->soak_state == SOAK_PAUSE) { 580 /* 581 * Soaker is paused but the next set is 582 * sysonly: start the soaker. 583 */ 584 state->soak_state = SOAK_RUN; 585 (void) cond_signal(&state->soak_cv); 586 } else if (cpc_setgrp_sysonly(sgrp) == 0 && 587 state->soak_state == SOAK_RUN) 588 /* 589 * Soaker is running but the next set 590 * counts user events: stop the soaker. 591 */ 592 state->soak_state = SOAK_PAUSE; 593 (void) mutex_unlock(&state->soak_lock); 594 } 595 596 if (cpc_bind_cpu(cpc, state->cpuid, this, 0) != 0) 597 goto bad; 598 } 599 } 600 601 if (cpc_unbind(cpc, this) != 0) 602 (void) fprintf(stderr, gettext("%s: error unbinding on" 603 " cpu %d - %s\n"), opts->pgmname, 604 state->cpuid, strerror(errno)); 605 606 /* 607 * We're done, so stop the soaker if needed. 608 */ 609 if (opts->dosoaker) { 610 (void) mutex_lock(&state->soak_lock); 611 if (state->soak_state == SOAK_RUN) 612 state->soak_state = SOAK_PAUSE; 613 (void) mutex_unlock(&state->soak_lock); 614 } 615 616 return (NULL); 617 bad: 618 state->status = 3; 619 errstr = strerror(errno); 620 (void) fprintf(stderr, gettext("%s: cpu%d - %s\n"), 621 opts->pgmname, state->cpuid, errstr); 622 return (NULL); 623 } 624 625 static int 626 cpustat(void) 627 { 628 cpc_setgrp_t *accum; 629 cpc_set_t *start; 630 int c, i, retval; 631 int lwps = 0; 632 psetid_t mypset, cpupset; 633 char *errstr; 634 cpc_buf_t **data1, **data2, **scratch; 635 int nreqs; 636 kstat_ctl_t *kc; 637 638 ncpus = (int)sysconf(_SC_NPROCESSORS_CONF); 639 if ((gstate = calloc(ncpus, sizeof (*gstate))) == NULL) { 640 (void) fprintf(stderr, gettext( 641 "%s: out of heap\n"), opts->pgmname); 642 return (1); 643 } 644 645 max_chip_id = sysconf(_SC_CPUID_MAX); 646 if ((chip_designees = malloc(max_chip_id * sizeof (int))) == NULL) { 647 (void) fprintf(stderr, gettext( 648 "%s: out of heap\n"), opts->pgmname); 649 return (1); 650 } 651 for (i = 0; i < max_chip_id; i++) 652 chip_designees[i] = -1; 653 654 if (smt) { 655 if ((kc = kstat_open()) == NULL) { 656 (void) fprintf(stderr, gettext( 657 "%s: kstat_open() failed: %s\n"), opts->pgmname, 658 strerror(errno)); 659 return (1); 660 } 661 } 662 663 if (opts->dosoaker) 664 if (priocntl(0, 0, PC_GETCID, &fxinfo) == -1) { 665 (void) fprintf(stderr, gettext( 666 "%s: couldn't get FX scheduler class: %s\n"), 667 opts->pgmname, strerror(errno)); 668 return (1); 669 } 670 671 /* 672 * Only include processors that are participating in the system 673 */ 674 for (c = 0, i = 0; i < ncpus; c++) { 675 switch (p_online(c, P_STATUS)) { 676 case P_ONLINE: 677 case P_NOINTR: 678 if (smt) { 679 680 gstate[i].chip_id = get_chipid(kc, c); 681 if (gstate[i].chip_id != -1 && 682 chip_designees[gstate[i].chip_id] == -1) 683 chip_designees[gstate[i].chip_id] = c; 684 } 685 686 gstate[i++].cpuid = c; 687 break; 688 case P_OFFLINE: 689 case P_POWEROFF: 690 case P_FAULTED: 691 case P_SPARE: 692 gstate[i++].cpuid = -1; 693 break; 694 default: 695 gstate[i++].cpuid = -1; 696 (void) fprintf(stderr, 697 gettext("%s: cpu%d in unknown state\n"), 698 opts->pgmname, c); 699 break; 700 case -1: 701 break; 702 } 703 } 704 705 /* 706 * Examine the processor sets; if we're in one, only attempt 707 * to report on the set we're in. 708 */ 709 if (pset_bind(PS_QUERY, P_PID, P_MYID, &mypset) == -1) { 710 errstr = strerror(errno); 711 (void) fprintf(stderr, gettext("%s: pset_bind - %s\n"), 712 opts->pgmname, errstr); 713 } else { 714 for (i = 0; i < ncpus; i++) { 715 struct tstate *this = &gstate[i]; 716 717 if (this->cpuid == -1) 718 continue; 719 720 if (pset_assign(PS_QUERY, 721 this->cpuid, &cpupset) == -1) { 722 errstr = strerror(errno); 723 (void) fprintf(stderr, 724 gettext("%s: pset_assign - %s\n"), 725 opts->pgmname, errstr); 726 continue; 727 } 728 729 if (mypset != cpupset) 730 this->cpuid = -1; 731 } 732 } 733 734 if (opts->dotitle) 735 print_title(opts->master); 736 zerotime(); 737 738 for (i = 0; i < ncpus; i++) { 739 struct tstate *this = &gstate[i]; 740 741 if (this->cpuid == -1) 742 continue; 743 this->sgrp = cpc_setgrp_clone(opts->master); 744 if (this->sgrp == NULL) { 745 this->cpuid = -1; 746 continue; 747 } 748 if (thr_create(NULL, 0, gtick, this, 749 THR_BOUND|THR_NEW_LWP, &this->tid) == 0) 750 lwps++; 751 else { 752 (void) fprintf(stderr, 753 gettext("%s: cannot create thread for cpu%d\n"), 754 opts->pgmname, this->cpuid); 755 this->status = 4; 756 } 757 } 758 759 if (lwps != 0) 760 for (i = 0; i < ncpus; i++) 761 (void) thr_join(gstate[i].tid, NULL, NULL); 762 763 if ((accum = cpc_setgrp_clone(opts->master)) == NULL) { 764 (void) fprintf(stderr, gettext("%s: out of heap\n"), 765 opts->pgmname); 766 return (1); 767 } 768 769 retval = 0; 770 for (i = 0; i < ncpus; i++) { 771 struct tstate *this = &gstate[i]; 772 773 if (this->cpuid == -1) 774 continue; 775 cpc_setgrp_accum(accum, this->sgrp); 776 cpc_setgrp_free(this->sgrp); 777 this->sgrp = NULL; 778 if (this->status != 0) 779 retval = 1; 780 } 781 782 cpc_setgrp_reset(accum); 783 start = cpc_setgrp_getset(accum); 784 do { 785 nreqs = cpc_setgrp_getbufs(accum, &data1, &data2, &scratch); 786 print_total(lwps, *data1, nreqs, cpc_setgrp_getname(accum)); 787 } while (cpc_setgrp_nextset(accum) != start); 788 789 cpc_setgrp_free(accum); 790 accum = NULL; 791 792 free(gstate); 793 return (retval); 794 } 795 796 static int 797 get_chipid(kstat_ctl_t *kc, processorid_t cpuid) 798 { 799 kstat_t *ksp; 800 kstat_named_t *k; 801 802 if ((ksp = kstat_lookup(kc, "cpu_info", cpuid, NULL)) == NULL) 803 return (-1); 804 805 if (kstat_read(kc, ksp, NULL) == -1) { 806 (void) fprintf(stderr, 807 gettext("%s: kstat_read() failed for cpu %d: %s\n"), 808 opts->pgmname, cpuid, strerror(errno)); 809 return (-1); 810 } 811 812 if ((k = (kstat_named_t *)kstat_data_lookup(ksp, "chip_id")) == NULL) { 813 (void) fprintf(stderr, 814 gettext("%s: chip_id not found for cpu %d: %s\n"), 815 opts->pgmname, cpuid, strerror(errno)); 816 return (-1); 817 } 818 819 return (k->value.i32); 820 } 821 822 static void * 823 soaker(void *arg) 824 { 825 struct tstate *state = arg; 826 pcparms_t pcparms; 827 fxparms_t *fx = (fxparms_t *)pcparms.pc_clparms; 828 829 if (processor_bind(P_LWPID, P_MYID, state->cpuid, NULL) != 0) 830 (void) fprintf(stderr, gettext("%s: couldn't bind soaker " 831 "thread to cpu%d: %s\n"), opts->pgmname, state->cpuid, 832 strerror(errno)); 833 834 /* 835 * Put the soaker thread in the fixed priority (FX) class so it runs 836 * at the lowest possible global priority. 837 */ 838 pcparms.pc_cid = fxinfo.pc_cid; 839 fx->fx_upri = 0; 840 fx->fx_uprilim = 0; 841 fx->fx_tqsecs = fx->fx_tqnsecs = FX_TQDEF; 842 843 if (priocntl(P_LWPID, P_MYID, PC_SETPARMS, &pcparms) != 0) 844 (void) fprintf(stderr, gettext("%s: couldn't put soaker " 845 "thread in FX sched class: %s\n"), opts->pgmname, 846 strerror(errno)); 847 848 /* 849 * Let the parent thread know we're ready to roll. 850 */ 851 (void) mutex_lock(&state->soak_lock); 852 state->soak_state = SOAK_RUN; 853 (void) cond_signal(&state->soak_cv); 854 (void) mutex_unlock(&state->soak_lock); 855 856 for (;;) { 857 spin: 858 (void) mutex_lock(&state->soak_lock); 859 if (state->soak_state == SOAK_RUN) { 860 (void) mutex_unlock(&state->soak_lock); 861 goto spin; 862 } 863 864 while (state->soak_state == SOAK_PAUSE) 865 (void) cond_wait(&state->soak_cv, 866 &state->soak_lock); 867 (void) mutex_unlock(&state->soak_lock); 868 } 869 870 /*NOTREACHED*/ 871 return (NULL); 872 } 873