1 /* 2 * Copyright (c) 1997, Stefan Esser <se@freebsd.org> 3 * All rights reserved. 4 * 5 * Redistribution and use in source and binary forms, with or without 6 * modification, are permitted provided that the following conditions 7 * are met: 8 * 1. Redistributions of source code must retain the above copyright 9 * notice unmodified, this list of conditions, and the following 10 * disclaimer. 11 * 2. Redistributions in binary form must reproduce the above copyright 12 * notice, this list of conditions and the following disclaimer in the 13 * documentation and/or other materials provided with the distribution. 14 * 15 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR 16 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 17 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 18 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, 19 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 20 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 21 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 22 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 23 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF 24 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 25 */ 26 27 #include <sys/cdefs.h> 28 __FBSDID("$FreeBSD$"); 29 30 #include "opt_ddb.h" 31 32 #include <sys/param.h> 33 #include <sys/bus.h> 34 #include <sys/conf.h> 35 #include <sys/rtprio.h> 36 #include <sys/systm.h> 37 #include <sys/interrupt.h> 38 #include <sys/kernel.h> 39 #include <sys/kthread.h> 40 #include <sys/ktr.h> 41 #include <sys/limits.h> 42 #include <sys/lock.h> 43 #include <sys/malloc.h> 44 #include <sys/mutex.h> 45 #include <sys/proc.h> 46 #include <sys/random.h> 47 #include <sys/resourcevar.h> 48 #include <sys/sysctl.h> 49 #include <sys/unistd.h> 50 #include <sys/vmmeter.h> 51 #include <machine/atomic.h> 52 #include <machine/cpu.h> 53 #include <machine/md_var.h> 54 #include <machine/stdarg.h> 55 #ifdef DDB 56 #include <ddb/ddb.h> 57 #include <ddb/db_sym.h> 58 #endif 59 60 struct int_entropy { 61 struct proc *proc; 62 uintptr_t vector; 63 }; 64 65 struct ithd *clk_ithd; 66 struct ithd *tty_ithd; 67 void *softclock_ih; 68 void *vm_ih; 69 70 static MALLOC_DEFINE(M_ITHREAD, "ithread", "Interrupt Threads"); 71 72 static int intr_storm_threshold = 500; 73 TUNABLE_INT("hw.intr_storm_threshold", &intr_storm_threshold); 74 SYSCTL_INT(_hw, OID_AUTO, intr_storm_threshold, CTLFLAG_RW, 75 &intr_storm_threshold, 0, 76 "Number of consecutive interrupts before storm protection is enabled"); 77 78 static void ithread_loop(void *); 79 static void ithread_update(struct ithd *); 80 static void start_softintr(void *); 81 82 u_char 83 ithread_priority(enum intr_type flags) 84 { 85 u_char pri; 86 87 flags &= (INTR_TYPE_TTY | INTR_TYPE_BIO | INTR_TYPE_NET | 88 INTR_TYPE_CAM | INTR_TYPE_MISC | INTR_TYPE_CLK | INTR_TYPE_AV); 89 switch (flags) { 90 case INTR_TYPE_TTY: 91 pri = PI_TTYLOW; 92 break; 93 case INTR_TYPE_BIO: 94 /* 95 * XXX We need to refine this. BSD/OS distinguishes 96 * between tape and disk priorities. 97 */ 98 pri = PI_DISK; 99 break; 100 case INTR_TYPE_NET: 101 pri = PI_NET; 102 break; 103 case INTR_TYPE_CAM: 104 pri = PI_DISK; /* XXX or PI_CAM? */ 105 break; 106 case INTR_TYPE_AV: /* Audio/video */ 107 pri = PI_AV; 108 break; 109 case INTR_TYPE_CLK: 110 pri = PI_REALTIME; 111 break; 112 case INTR_TYPE_MISC: 113 pri = PI_DULL; /* don't care */ 114 break; 115 default: 116 /* We didn't specify an interrupt level. */ 117 panic("ithread_priority: no interrupt type in flags"); 118 } 119 120 return pri; 121 } 122 123 /* 124 * Regenerate the name (p_comm) and priority for a threaded interrupt thread. 125 */ 126 static void 127 ithread_update(struct ithd *ithd) 128 { 129 struct intrhand *ih; 130 struct thread *td; 131 struct proc *p; 132 int entropy; 133 134 mtx_assert(&ithd->it_lock, MA_OWNED); 135 td = ithd->it_td; 136 if (td == NULL) 137 return; 138 p = td->td_proc; 139 140 strlcpy(p->p_comm, ithd->it_name, sizeof(p->p_comm)); 141 142 ih = TAILQ_FIRST(&ithd->it_handlers); 143 if (ih == NULL) { 144 mtx_lock_spin(&sched_lock); 145 td->td_priority = PRI_MAX_ITHD; 146 td->td_base_pri = PRI_MAX_ITHD; 147 mtx_unlock_spin(&sched_lock); 148 ithd->it_flags &= ~IT_ENTROPY; 149 return; 150 } 151 entropy = 0; 152 mtx_lock_spin(&sched_lock); 153 td->td_priority = ih->ih_pri; 154 td->td_base_pri = ih->ih_pri; 155 mtx_unlock_spin(&sched_lock); 156 TAILQ_FOREACH(ih, &ithd->it_handlers, ih_next) { 157 if (strlen(p->p_comm) + strlen(ih->ih_name) + 1 < 158 sizeof(p->p_comm)) { 159 strcat(p->p_comm, " "); 160 strcat(p->p_comm, ih->ih_name); 161 } else if (strlen(p->p_comm) + 1 == sizeof(p->p_comm)) { 162 if (p->p_comm[sizeof(p->p_comm) - 2] == '+') 163 p->p_comm[sizeof(p->p_comm) - 2] = '*'; 164 else 165 p->p_comm[sizeof(p->p_comm) - 2] = '+'; 166 } else 167 strcat(p->p_comm, "+"); 168 if (ih->ih_flags & IH_ENTROPY) 169 entropy++; 170 } 171 if (entropy) 172 ithd->it_flags |= IT_ENTROPY; 173 else 174 ithd->it_flags &= ~IT_ENTROPY; 175 CTR2(KTR_INTR, "%s: updated %s", __func__, p->p_comm); 176 } 177 178 int 179 ithread_create(struct ithd **ithread, uintptr_t vector, int flags, 180 void (*disable)(uintptr_t), void (*enable)(uintptr_t), const char *fmt, ...) 181 { 182 struct ithd *ithd; 183 struct thread *td; 184 struct proc *p; 185 int error; 186 va_list ap; 187 188 /* The only valid flag during creation is IT_SOFT. */ 189 if ((flags & ~IT_SOFT) != 0) 190 return (EINVAL); 191 192 ithd = malloc(sizeof(struct ithd), M_ITHREAD, M_WAITOK | M_ZERO); 193 ithd->it_vector = vector; 194 ithd->it_disable = disable; 195 ithd->it_enable = enable; 196 ithd->it_flags = flags; 197 TAILQ_INIT(&ithd->it_handlers); 198 mtx_init(&ithd->it_lock, "ithread", NULL, MTX_DEF); 199 200 va_start(ap, fmt); 201 vsnprintf(ithd->it_name, sizeof(ithd->it_name), fmt, ap); 202 va_end(ap); 203 204 error = kthread_create(ithread_loop, ithd, &p, RFSTOPPED | RFHIGHPID, 205 0, "%s", ithd->it_name); 206 if (error) { 207 mtx_destroy(&ithd->it_lock); 208 free(ithd, M_ITHREAD); 209 return (error); 210 } 211 td = FIRST_THREAD_IN_PROC(p); /* XXXKSE */ 212 mtx_lock_spin(&sched_lock); 213 td->td_ksegrp->kg_pri_class = PRI_ITHD; 214 td->td_priority = PRI_MAX_ITHD; 215 TD_SET_IWAIT(td); 216 mtx_unlock_spin(&sched_lock); 217 ithd->it_td = td; 218 td->td_ithd = ithd; 219 if (ithread != NULL) 220 *ithread = ithd; 221 CTR2(KTR_INTR, "%s: created %s", __func__, ithd->it_name); 222 return (0); 223 } 224 225 int 226 ithread_destroy(struct ithd *ithread) 227 { 228 229 struct thread *td; 230 if (ithread == NULL) 231 return (EINVAL); 232 233 td = ithread->it_td; 234 mtx_lock(&ithread->it_lock); 235 if (!TAILQ_EMPTY(&ithread->it_handlers)) { 236 mtx_unlock(&ithread->it_lock); 237 return (EINVAL); 238 } 239 ithread->it_flags |= IT_DEAD; 240 mtx_lock_spin(&sched_lock); 241 if (TD_AWAITING_INTR(td)) { 242 TD_CLR_IWAIT(td); 243 setrunqueue(td); 244 } 245 mtx_unlock_spin(&sched_lock); 246 mtx_unlock(&ithread->it_lock); 247 CTR2(KTR_INTR, "%s: killing %s", __func__, ithread->it_name); 248 return (0); 249 } 250 251 int 252 ithread_add_handler(struct ithd* ithread, const char *name, 253 driver_intr_t handler, void *arg, u_char pri, enum intr_type flags, 254 void **cookiep) 255 { 256 struct intrhand *ih, *temp_ih; 257 258 if (ithread == NULL || name == NULL || handler == NULL) 259 return (EINVAL); 260 261 ih = malloc(sizeof(struct intrhand), M_ITHREAD, M_WAITOK | M_ZERO); 262 ih->ih_handler = handler; 263 ih->ih_argument = arg; 264 ih->ih_name = name; 265 ih->ih_ithread = ithread; 266 ih->ih_pri = pri; 267 if (flags & INTR_FAST) 268 ih->ih_flags = IH_FAST; 269 else if (flags & INTR_EXCL) 270 ih->ih_flags = IH_EXCLUSIVE; 271 if (flags & INTR_MPSAFE) 272 ih->ih_flags |= IH_MPSAFE; 273 if (flags & INTR_ENTROPY) 274 ih->ih_flags |= IH_ENTROPY; 275 276 mtx_lock(&ithread->it_lock); 277 if ((flags & INTR_EXCL) != 0 && !TAILQ_EMPTY(&ithread->it_handlers)) 278 goto fail; 279 if (!TAILQ_EMPTY(&ithread->it_handlers)) { 280 temp_ih = TAILQ_FIRST(&ithread->it_handlers); 281 if (temp_ih->ih_flags & IH_EXCLUSIVE) 282 goto fail; 283 if ((ih->ih_flags & IH_FAST) && !(temp_ih->ih_flags & IH_FAST)) 284 goto fail; 285 if (!(ih->ih_flags & IH_FAST) && (temp_ih->ih_flags & IH_FAST)) 286 goto fail; 287 } 288 289 TAILQ_FOREACH(temp_ih, &ithread->it_handlers, ih_next) 290 if (temp_ih->ih_pri > ih->ih_pri) 291 break; 292 if (temp_ih == NULL) 293 TAILQ_INSERT_TAIL(&ithread->it_handlers, ih, ih_next); 294 else 295 TAILQ_INSERT_BEFORE(temp_ih, ih, ih_next); 296 ithread_update(ithread); 297 mtx_unlock(&ithread->it_lock); 298 299 if (cookiep != NULL) 300 *cookiep = ih; 301 CTR3(KTR_INTR, "%s: added %s to %s", __func__, ih->ih_name, 302 ithread->it_name); 303 return (0); 304 305 fail: 306 mtx_unlock(&ithread->it_lock); 307 free(ih, M_ITHREAD); 308 return (EINVAL); 309 } 310 311 int 312 ithread_remove_handler(void *cookie) 313 { 314 struct intrhand *handler = (struct intrhand *)cookie; 315 struct ithd *ithread; 316 #ifdef INVARIANTS 317 struct intrhand *ih; 318 #endif 319 320 if (handler == NULL) 321 return (EINVAL); 322 ithread = handler->ih_ithread; 323 KASSERT(ithread != NULL, 324 ("interrupt handler \"%s\" has a NULL interrupt thread", 325 handler->ih_name)); 326 CTR3(KTR_INTR, "%s: removing %s from %s", __func__, handler->ih_name, 327 ithread->it_name); 328 mtx_lock(&ithread->it_lock); 329 #ifdef INVARIANTS 330 TAILQ_FOREACH(ih, &ithread->it_handlers, ih_next) 331 if (ih == handler) 332 goto ok; 333 mtx_unlock(&ithread->it_lock); 334 panic("interrupt handler \"%s\" not found in interrupt thread \"%s\"", 335 ih->ih_name, ithread->it_name); 336 ok: 337 #endif 338 /* 339 * If the interrupt thread is already running, then just mark this 340 * handler as being dead and let the ithread do the actual removal. 341 * 342 * During a cold boot while cold is set, msleep() does not sleep, 343 * so we have to remove the handler here rather than letting the 344 * thread do it. 345 */ 346 mtx_lock_spin(&sched_lock); 347 if (!TD_AWAITING_INTR(ithread->it_td) && !cold) { 348 handler->ih_flags |= IH_DEAD; 349 350 /* 351 * Ensure that the thread will process the handler list 352 * again and remove this handler if it has already passed 353 * it on the list. 354 */ 355 ithread->it_need = 1; 356 } else 357 TAILQ_REMOVE(&ithread->it_handlers, handler, ih_next); 358 mtx_unlock_spin(&sched_lock); 359 if ((handler->ih_flags & IH_DEAD) != 0) 360 msleep(handler, &ithread->it_lock, PUSER, "itrmh", 0); 361 ithread_update(ithread); 362 mtx_unlock(&ithread->it_lock); 363 free(handler, M_ITHREAD); 364 return (0); 365 } 366 367 int 368 ithread_schedule(struct ithd *ithread, int do_switch) 369 { 370 struct int_entropy entropy; 371 struct thread *td; 372 struct thread *ctd; 373 struct proc *p; 374 375 /* 376 * If no ithread or no handlers, then we have a stray interrupt. 377 */ 378 if ((ithread == NULL) || TAILQ_EMPTY(&ithread->it_handlers)) 379 return (EINVAL); 380 381 ctd = curthread; 382 /* 383 * If any of the handlers for this ithread claim to be good 384 * sources of entropy, then gather some. 385 */ 386 if (harvest.interrupt && ithread->it_flags & IT_ENTROPY) { 387 entropy.vector = ithread->it_vector; 388 entropy.proc = ctd->td_proc; 389 random_harvest(&entropy, sizeof(entropy), 2, 0, 390 RANDOM_INTERRUPT); 391 } 392 393 td = ithread->it_td; 394 p = td->td_proc; 395 KASSERT(p != NULL, ("ithread %s has no process", ithread->it_name)); 396 CTR4(KTR_INTR, "%s: pid %d: (%s) need = %d", 397 __func__, p->p_pid, p->p_comm, ithread->it_need); 398 399 /* 400 * Set it_need to tell the thread to keep running if it is already 401 * running. Then, grab sched_lock and see if we actually need to 402 * put this thread on the runqueue. If so and the do_switch flag is 403 * true and it is safe to switch, then switch to the ithread 404 * immediately. Otherwise, set the needresched flag to guarantee 405 * that this ithread will run before any userland processes. 406 */ 407 ithread->it_need = 1; 408 mtx_lock_spin(&sched_lock); 409 if (TD_AWAITING_INTR(td)) { 410 CTR2(KTR_INTR, "%s: setrunqueue %d", __func__, p->p_pid); 411 TD_CLR_IWAIT(td); 412 setrunqueue(td); 413 if (do_switch && 414 (ctd->td_critnest == 1) ) { 415 KASSERT((TD_IS_RUNNING(ctd)), 416 ("ithread_schedule: Bad state for curthread.")); 417 if (ctd->td_flags & TDF_IDLETD) 418 ctd->td_state = TDS_CAN_RUN; /* XXXKSE */ 419 mi_switch(SW_INVOL); 420 } else { 421 curthread->td_flags |= TDF_NEEDRESCHED; 422 } 423 } else { 424 CTR4(KTR_INTR, "%s: pid %d: it_need %d, state %d", 425 __func__, p->p_pid, ithread->it_need, td->td_state); 426 } 427 mtx_unlock_spin(&sched_lock); 428 429 return (0); 430 } 431 432 int 433 swi_add(struct ithd **ithdp, const char *name, driver_intr_t handler, 434 void *arg, int pri, enum intr_type flags, void **cookiep) 435 { 436 struct ithd *ithd; 437 int error; 438 439 if (flags & (INTR_FAST | INTR_ENTROPY)) 440 return (EINVAL); 441 442 ithd = (ithdp != NULL) ? *ithdp : NULL; 443 444 if (ithd != NULL) { 445 if ((ithd->it_flags & IT_SOFT) == 0) 446 return(EINVAL); 447 } else { 448 error = ithread_create(&ithd, pri, IT_SOFT, NULL, NULL, 449 "swi%d:", pri); 450 if (error) 451 return (error); 452 453 if (ithdp != NULL) 454 *ithdp = ithd; 455 } 456 return (ithread_add_handler(ithd, name, handler, arg, 457 (pri * RQ_PPQ) + PI_SOFT, flags, cookiep)); 458 } 459 460 461 /* 462 * Schedule a heavyweight software interrupt process. 463 */ 464 void 465 swi_sched(void *cookie, int flags) 466 { 467 struct intrhand *ih = (struct intrhand *)cookie; 468 struct ithd *it = ih->ih_ithread; 469 int error; 470 471 atomic_add_int(&cnt.v_intr, 1); /* one more global interrupt */ 472 473 CTR3(KTR_INTR, "swi_sched pid %d(%s) need=%d", 474 it->it_td->td_proc->p_pid, it->it_td->td_proc->p_comm, it->it_need); 475 476 /* 477 * Set ih_need for this handler so that if the ithread is already 478 * running it will execute this handler on the next pass. Otherwise, 479 * it will execute it the next time it runs. 480 */ 481 atomic_store_rel_int(&ih->ih_need, 1); 482 if (!(flags & SWI_DELAY)) { 483 error = ithread_schedule(it, !cold && !dumping); 484 KASSERT(error == 0, ("stray software interrupt")); 485 } 486 } 487 488 /* 489 * This is the main code for interrupt threads. 490 */ 491 static void 492 ithread_loop(void *arg) 493 { 494 struct ithd *ithd; /* our thread context */ 495 struct intrhand *ih; /* and our interrupt handler chain */ 496 struct thread *td; 497 struct proc *p; 498 int count, warming, warned; 499 500 td = curthread; 501 p = td->td_proc; 502 ithd = (struct ithd *)arg; /* point to myself */ 503 KASSERT(ithd->it_td == td && td->td_ithd == ithd, 504 ("%s: ithread and proc linkage out of sync", __func__)); 505 warming = 10 * intr_storm_threshold; 506 warned = 0; 507 508 /* 509 * As long as we have interrupts outstanding, go through the 510 * list of handlers, giving each one a go at it. 511 */ 512 for (;;) { 513 /* 514 * If we are an orphaned thread, then just die. 515 */ 516 if (ithd->it_flags & IT_DEAD) { 517 CTR3(KTR_INTR, "%s: pid %d: (%s) exiting", __func__, 518 p->p_pid, p->p_comm); 519 td->td_ithd = NULL; 520 mtx_destroy(&ithd->it_lock); 521 free(ithd, M_ITHREAD); 522 kthread_exit(0); 523 } 524 525 CTR4(KTR_INTR, "%s: pid %d: (%s) need=%d", __func__, 526 p->p_pid, p->p_comm, ithd->it_need); 527 count = 0; 528 while (ithd->it_need) { 529 /* 530 * Service interrupts. If another interrupt 531 * arrives while we are running, they will set 532 * it_need to denote that we should make 533 * another pass. 534 */ 535 atomic_store_rel_int(&ithd->it_need, 0); 536 restart: 537 TAILQ_FOREACH(ih, &ithd->it_handlers, ih_next) { 538 if (ithd->it_flags & IT_SOFT && !ih->ih_need) 539 continue; 540 atomic_store_rel_int(&ih->ih_need, 0); 541 CTR6(KTR_INTR, 542 "%s: pid %d ih=%p: %p(%p) flg=%x", __func__, 543 p->p_pid, (void *)ih, 544 (void *)ih->ih_handler, ih->ih_argument, 545 ih->ih_flags); 546 547 if ((ih->ih_flags & IH_DEAD) != 0) { 548 mtx_lock(&ithd->it_lock); 549 TAILQ_REMOVE(&ithd->it_handlers, ih, 550 ih_next); 551 wakeup(ih); 552 mtx_unlock(&ithd->it_lock); 553 goto restart; 554 } 555 if ((ih->ih_flags & IH_MPSAFE) == 0) 556 mtx_lock(&Giant); 557 ih->ih_handler(ih->ih_argument); 558 if ((ih->ih_flags & IH_MPSAFE) == 0) 559 mtx_unlock(&Giant); 560 } 561 if (ithd->it_enable != NULL) { 562 ithd->it_enable(ithd->it_vector); 563 564 /* 565 * Storm detection needs a delay here 566 * to see slightly delayed interrupts 567 * on some machines, but we don't 568 * want to always delay, so only delay 569 * while warming up. 570 */ 571 if (warming != 0) { 572 DELAY(1); 573 --warming; 574 } 575 } 576 577 /* 578 * If we detect an interrupt storm, sleep until 579 * the next hardclock tick. We sleep at the 580 * end of the loop instead of at the beginning 581 * to ensure that we see slightly delayed 582 * interrupts. 583 */ 584 if (count >= intr_storm_threshold) { 585 if (!warned) { 586 printf( 587 "Interrupt storm detected on \"%s\"; throttling interrupt source\n", 588 p->p_comm); 589 warned = 1; 590 } 591 tsleep(&count, td->td_priority, "istorm", 1); 592 593 /* 594 * Fudge the count to re-throttle if the 595 * interrupt is still active. Our storm 596 * detection is too primitive to detect 597 * whether the storm has gone away 598 * reliably, even if we were to waste a 599 * lot of time spinning for the next 600 * intr_storm_threshold interrupts, so 601 * we assume that the storm hasn't gone 602 * away unless the interrupt repeats 603 * less often the hardclock interrupt. 604 */ 605 count = INT_MAX - 1; 606 } 607 count++; 608 } 609 WITNESS_WARN(WARN_PANIC, NULL, "suspending ithread"); 610 mtx_assert(&Giant, MA_NOTOWNED); 611 612 /* 613 * Processed all our interrupts. Now get the sched 614 * lock. This may take a while and it_need may get 615 * set again, so we have to check it again. 616 */ 617 mtx_lock_spin(&sched_lock); 618 if (!ithd->it_need) { 619 TD_SET_IWAIT(td); 620 CTR2(KTR_INTR, "%s: pid %d: done", __func__, p->p_pid); 621 mi_switch(SW_VOL); 622 CTR2(KTR_INTR, "%s: pid %d: resumed", __func__, p->p_pid); 623 } 624 mtx_unlock_spin(&sched_lock); 625 } 626 } 627 628 #ifdef DDB 629 /* 630 * Dump details about an interrupt handler 631 */ 632 static void 633 db_dump_intrhand(struct intrhand *ih) 634 { 635 int comma; 636 637 db_printf("\t%-10s ", ih->ih_name); 638 switch (ih->ih_pri) { 639 case PI_REALTIME: 640 db_printf("CLK "); 641 break; 642 case PI_AV: 643 db_printf("AV "); 644 break; 645 case PI_TTYHIGH: 646 case PI_TTYLOW: 647 db_printf("TTY "); 648 break; 649 case PI_TAPE: 650 db_printf("TAPE"); 651 break; 652 case PI_NET: 653 db_printf("NET "); 654 break; 655 case PI_DISK: 656 case PI_DISKLOW: 657 db_printf("DISK"); 658 break; 659 case PI_DULL: 660 db_printf("DULL"); 661 break; 662 default: 663 if (ih->ih_pri >= PI_SOFT) 664 db_printf("SWI "); 665 else 666 db_printf("%4u", ih->ih_pri); 667 break; 668 } 669 db_printf(" "); 670 db_printsym((uintptr_t)ih->ih_handler, DB_STGY_PROC); 671 db_printf("(%p)", ih->ih_argument); 672 if (ih->ih_need || 673 (ih->ih_flags & (IH_FAST | IH_EXCLUSIVE | IH_ENTROPY | IH_DEAD | 674 IH_MPSAFE)) != 0) { 675 db_printf(" {"); 676 comma = 0; 677 if (ih->ih_flags & IH_FAST) { 678 db_printf("FAST"); 679 comma = 1; 680 } 681 if (ih->ih_flags & IH_EXCLUSIVE) { 682 if (comma) 683 db_printf(", "); 684 db_printf("EXCL"); 685 comma = 1; 686 } 687 if (ih->ih_flags & IH_ENTROPY) { 688 if (comma) 689 db_printf(", "); 690 db_printf("ENTROPY"); 691 comma = 1; 692 } 693 if (ih->ih_flags & IH_DEAD) { 694 if (comma) 695 db_printf(", "); 696 db_printf("DEAD"); 697 comma = 1; 698 } 699 if (ih->ih_flags & IH_MPSAFE) { 700 if (comma) 701 db_printf(", "); 702 db_printf("MPSAFE"); 703 comma = 1; 704 } 705 if (ih->ih_need) { 706 if (comma) 707 db_printf(", "); 708 db_printf("NEED"); 709 } 710 db_printf("}"); 711 } 712 db_printf("\n"); 713 } 714 715 /* 716 * Dump details about an ithread 717 */ 718 void 719 db_dump_ithread(struct ithd *ithd, int handlers) 720 { 721 struct proc *p; 722 struct intrhand *ih; 723 int comma; 724 725 if (ithd->it_td != NULL) { 726 p = ithd->it_td->td_proc; 727 db_printf("%s (pid %d)", p->p_comm, p->p_pid); 728 } else 729 db_printf("%s: (no thread)", ithd->it_name); 730 if ((ithd->it_flags & (IT_SOFT | IT_ENTROPY | IT_DEAD)) != 0 || 731 ithd->it_need) { 732 db_printf(" {"); 733 comma = 0; 734 if (ithd->it_flags & IT_SOFT) { 735 db_printf("SOFT"); 736 comma = 1; 737 } 738 if (ithd->it_flags & IT_ENTROPY) { 739 if (comma) 740 db_printf(", "); 741 db_printf("ENTROPY"); 742 comma = 1; 743 } 744 if (ithd->it_flags & IT_DEAD) { 745 if (comma) 746 db_printf(", "); 747 db_printf("DEAD"); 748 comma = 1; 749 } 750 if (ithd->it_need) { 751 if (comma) 752 db_printf(", "); 753 db_printf("NEED"); 754 } 755 db_printf("}"); 756 } 757 db_printf("\n"); 758 759 if (handlers) 760 TAILQ_FOREACH(ih, &ithd->it_handlers, ih_next) 761 db_dump_intrhand(ih); 762 } 763 #endif /* DDB */ 764 765 /* 766 * Start standard software interrupt threads 767 */ 768 static void 769 start_softintr(void *dummy) 770 { 771 struct proc *p; 772 773 if (swi_add(&clk_ithd, "clock", softclock, NULL, SWI_CLOCK, 774 INTR_MPSAFE, &softclock_ih) || 775 swi_add(NULL, "vm", swi_vm, NULL, SWI_VM, INTR_MPSAFE, &vm_ih)) 776 panic("died while creating standard software ithreads"); 777 778 p = clk_ithd->it_td->td_proc; 779 PROC_LOCK(p); 780 p->p_flag |= P_NOLOAD; 781 PROC_UNLOCK(p); 782 } 783 SYSINIT(start_softintr, SI_SUB_SOFTINTR, SI_ORDER_FIRST, start_softintr, NULL) 784 785 /* 786 * Sysctls used by systat and others: hw.intrnames and hw.intrcnt. 787 * The data for this machine dependent, and the declarations are in machine 788 * dependent code. The layout of intrnames and intrcnt however is machine 789 * independent. 790 * 791 * We do not know the length of intrcnt and intrnames at compile time, so 792 * calculate things at run time. 793 */ 794 static int 795 sysctl_intrnames(SYSCTL_HANDLER_ARGS) 796 { 797 return (sysctl_handle_opaque(oidp, intrnames, eintrnames - intrnames, 798 req)); 799 } 800 801 SYSCTL_PROC(_hw, OID_AUTO, intrnames, CTLTYPE_OPAQUE | CTLFLAG_RD, 802 NULL, 0, sysctl_intrnames, "", "Interrupt Names"); 803 804 static int 805 sysctl_intrcnt(SYSCTL_HANDLER_ARGS) 806 { 807 return (sysctl_handle_opaque(oidp, intrcnt, 808 (char *)eintrcnt - (char *)intrcnt, req)); 809 } 810 811 SYSCTL_PROC(_hw, OID_AUTO, intrcnt, CTLTYPE_OPAQUE | CTLFLAG_RD, 812 NULL, 0, sysctl_intrcnt, "", "Interrupt Counts"); 813 814 #ifdef DDB 815 /* 816 * DDB command to dump the interrupt statistics. 817 */ 818 DB_SHOW_COMMAND(intrcnt, db_show_intrcnt) 819 { 820 u_long *i; 821 char *cp; 822 int quit; 823 824 cp = intrnames; 825 db_setup_paging(db_simple_pager, &quit, DB_LINES_PER_PAGE); 826 for (i = intrcnt, quit = 0; i != eintrcnt && !quit; i++) { 827 if (*cp == '\0') 828 break; 829 if (*i != 0) 830 db_printf("%s\t%lu\n", cp, *i); 831 cp += strlen(cp) + 1; 832 } 833 } 834 #endif 835