/* * CDDL HEADER START * * The contents of this file are subject to the terms of the * Common Development and Distribution License (the "License"). * You may not use this file except in compliance with the License. * * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE * or http://www.opensolaris.org/os/licensing. * See the License for the specific language governing permissions * and limitations under the License. * * When distributing Covered Code, include this CDDL HEADER in each * file and include the License file at usr/src/OPENSOLARIS.LICENSE. * If applicable, add the following below this CDDL HEADER, with the * fields enclosed by brackets "[]" replaced with your own identifying * information: Portions Copyright [yyyy] [name of copyright owner] * * CDDL HEADER END */ /* * Copyright 2008 Sun Microsystems, Inc. All rights reserved. * Use is subject to license terms. */ #pragma ident "%Z%%M% %I% %E% SMI" #include "intr_common.h" #include #include #include int option_flags; uintptr_t gld_intr_addr; static struct av_head softvec_tbl[LOCK_LEVEL + 1]; static char *businfo_array[] = { " ", "CBUS", "CBUSII", "EISA", "FUTURE", "INTERN", "ISA", "MBI", "MBII", "PCIe", "MPI", "MPSA", "NUBUS", "PCI", "PCMCIA", "TC", "VL", "VME", "XPRESS", " " }; void interrupt_help(void) { mdb_printf("Prints the interrupt usage on the system.\n" "By default, only interrupt service routine names are printed.\n\n" "Switches:\n" " -d instead of ISR, print \n" " -i show like intrstat, cpu# ISR/\n"); } void soft_interrupt_help(void) { mdb_printf("Prints the soft interrupt usage on the system.\n" "By default, only interrupt service routine names are printed.\n\n" "Switch:\n" " -d instead of ISR, print \n"); } /* * This is copied from avintr.c * NOTE: Ensure that this definition stays in sync */ typedef struct av_softinfo { cpuset_t av_pending; /* pending bitmasks */ } av_softinfo_t; /* ARGSUSED */ int soft_interrupt_dump(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv) { int i; av_softinfo_t avsoftinfo; struct autovec avhp; ddi_softint_hdl_impl_t hdlp; option_flags = 0; if (mdb_getopts(argc, argv, 'd', MDB_OPT_SETBITS, INTR_DISPLAY_DRVR_INST, &option_flags, NULL) != argc) return (DCMD_USAGE); if (mdb_readvar(&softvec_tbl, "softvect") == -1) { mdb_warn("failed to read autovect"); return (DCMD_ERR); } /* Print the header first */ mdb_printf("%ADDR PEND PIL ARG1 " "ARG2 ISR(s)%\n"); /* Walk all the entries */ for (i = 0; i < LOCK_LEVEL + 1; i++) { /* Read the entry, if invalid continue */ if (mdb_vread(&avhp, sizeof (struct autovec), (uintptr_t)softvec_tbl[i].avh_link) == -1) continue; do { if (!avhp.av_vector || (mdb_vread(&hdlp, sizeof (ddi_softint_hdl_impl_t), (uintptr_t)avhp.av_intr_id) == -1) || (mdb_vread(&avsoftinfo, sizeof (av_softinfo_t), (uintptr_t)hdlp.ih_pending) == -1)) continue; /* Print each soft interrupt entry */ mdb_printf("%-16p %-2d %-2d %-16p %-16p", avhp.av_intr_id, mdb_cpuset_find( (uintptr_t)&avsoftinfo.av_pending) != -1 ? 1 : 0, avhp.av_prilevel, avhp.av_intarg1, avhp.av_intarg2); interrupt_print_isr((uintptr_t)avhp.av_vector, (uintptr_t)avhp.av_intarg1, (uintptr_t)hdlp.ih_dip); mdb_printf("\n"); } while (mdb_vread(&avhp, sizeof (struct autovec), (uintptr_t)avhp.av_link) != -1); } return (DCMD_OK); } void interrupt_print_isr(uintptr_t vector, uintptr_t arg1, uintptr_t dip) { uintptr_t isr_addr = vector; struct dev_info dev_info; /* * figure out the real ISR function name from gld_intr() */ if (isr_addr == gld_intr_addr) { gld_mac_info_t macinfo; if (mdb_vread(&macinfo, sizeof (gld_mac_info_t), arg1) != -1) { /* verify gld data structure and get the real ISR */ if (macinfo.gldm_GLD_version == GLD_VERSION) isr_addr = (uintptr_t)macinfo.gldm_intr; } } if ((option_flags & INTR_DISPLAY_DRVR_INST) && dip) { char drvr_name[MODMAXNAMELEN + 1]; if (dip && mdb_devinfo2driver(dip, drvr_name, sizeof (drvr_name)) == 0) { (void) mdb_vread(&dev_info, sizeof (dev_info), dip); mdb_printf("%s#%d", drvr_name, dev_info.devi_instance); } else { mdb_printf("%a", isr_addr); } } else { mdb_printf("%a", isr_addr); } } /* * get_interrupt_type: * * Get some interrupt related useful information * * NOTE: a0 is clock, c0/d0/e0 are x-calls, e1 is apic_error_intr * d1/d3 are cbe_fire interrupts */ static char * get_interrupt_type(short index) { if (index == RESERVE_INDEX) return ("IPI"); else if (index == ACPI_INDEX) return ("Fixed"); else if (index == MSI_INDEX) return ("MSI"); else if (index == MSIX_INDEX) return ("MSI-X"); else return ("Fixed"); } void apic_interrupt_dump(apic_irq_t *irqp, struct av_head *avp, int i, ushort_t *evtchnp, char level) { int bus_type; int j; char *intr_type; char ioapic_iline[10]; char ipl[3]; char cpu_assigned[4]; char evtchn[8]; ushort_t assigned_cpu; struct autovec avhp; /* If invalid index; continue */ if (!irqp->airq_mps_intr_index || irqp->airq_mps_intr_index == FREE_INDEX) return; /* Figure out interrupt type and trigger information */ intr_type = get_interrupt_type(irqp->airq_mps_intr_index); /* Figure out IOAPIC number and ILINE number */ if (APIC_IS_MSI_OR_MSIX_INDEX(irqp->airq_mps_intr_index)) (void) mdb_snprintf(ioapic_iline, 10, "- "); else { if (!irqp->airq_ioapicindex && !irqp->airq_intin_no) { if (strcmp(intr_type, "Fixed") == 0) (void) mdb_snprintf(ioapic_iline, 10, "0x%x/0x%x", irqp->airq_ioapicindex, irqp->airq_intin_no); else if (irqp->airq_mps_intr_index == RESERVE_INDEX) (void) mdb_snprintf(ioapic_iline, 10, "- "); else (void) mdb_snprintf(ioapic_iline, 10, " "); } else (void) mdb_snprintf(ioapic_iline, 10, "0x%x/0x%x", irqp->airq_ioapicindex, irqp->airq_intin_no); } evtchn[0] = '\0'; if (evtchnp != NULL) (void) mdb_snprintf(evtchn, 8, "%-7hd", *evtchnp); assigned_cpu = irqp->airq_temp_cpu; if (assigned_cpu == IRQ_UNINIT || assigned_cpu == IRQ_UNBOUND) assigned_cpu = irqp->airq_cpu; bus_type = irqp->airq_iflag.bustype; if (irqp->airq_mps_intr_index == RESERVE_INDEX) { (void) mdb_snprintf(cpu_assigned, 4, "all"); (void) mdb_snprintf(ipl, 3, "%d", avp->avh_hi_pri); } else { (void) mdb_snprintf(cpu_assigned, 4, "%d", assigned_cpu); (void) mdb_snprintf(ipl, 3, "%d", irqp->airq_ipl); } /* Print each interrupt entry */ if (option_flags & INTR_DISPLAY_INTRSTAT) mdb_printf("%-4s", cpu_assigned); else mdb_printf("%-3d 0x%x %s%-3s %-6s %-3s %-6s %-4s%-3d %-9s ", i, irqp->airq_vector, evtchn, ipl, (bus_type ? businfo_array[bus_type] : " "), (level ? "Lvl" : "Edg"), intr_type, cpu_assigned, irqp->airq_share, ioapic_iline); /* If valid dip found; print driver name */ if (irqp->airq_dip) { (void) mdb_vread(&avhp, sizeof (struct autovec), (uintptr_t)avp->avh_link); /* * Loop thru all the shared IRQs */ if (irqp->airq_share) interrupt_print_isr((uintptr_t)avhp.av_vector, (uintptr_t)avhp.av_intarg1, (uintptr_t)avhp.av_dip); for (j = 1; irqp->airq_mps_intr_index != FREE_INDEX && j < irqp->airq_share; j++) { if (mdb_vread(&avhp, sizeof (struct autovec), (uintptr_t)avhp.av_link) != -1) { mdb_printf(", "); interrupt_print_isr((uintptr_t)avhp.av_vector, (uintptr_t)avhp.av_intarg1, (uintptr_t)avhp.av_dip); } else { break; } } } else { if (irqp->airq_mps_intr_index == RESERVE_INDEX && !irqp->airq_share) mdb_printf("poke_cpu"); else if (mdb_vread(&avhp, sizeof (struct autovec), (uintptr_t)avp->avh_link) != -1) mdb_printf("%a", avhp.av_vector); } mdb_printf("\n"); }