~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~

TOMOYO Linux Cross Reference
Linux/scripts/gdb/linux/interrupts.py

Version: ~ [ linux-6.12-rc7 ] ~ [ linux-6.11.7 ] ~ [ linux-6.10.14 ] ~ [ linux-6.9.12 ] ~ [ linux-6.8.12 ] ~ [ linux-6.7.12 ] ~ [ linux-6.6.60 ] ~ [ linux-6.5.13 ] ~ [ linux-6.4.16 ] ~ [ linux-6.3.13 ] ~ [ linux-6.2.16 ] ~ [ linux-6.1.116 ] ~ [ linux-6.0.19 ] ~ [ linux-5.19.17 ] ~ [ linux-5.18.19 ] ~ [ linux-5.17.15 ] ~ [ linux-5.16.20 ] ~ [ linux-5.15.171 ] ~ [ linux-5.14.21 ] ~ [ linux-5.13.19 ] ~ [ linux-5.12.19 ] ~ [ linux-5.11.22 ] ~ [ linux-5.10.229 ] ~ [ linux-5.9.16 ] ~ [ linux-5.8.18 ] ~ [ linux-5.7.19 ] ~ [ linux-5.6.19 ] ~ [ linux-5.5.19 ] ~ [ linux-5.4.285 ] ~ [ linux-5.3.18 ] ~ [ linux-5.2.21 ] ~ [ linux-5.1.21 ] ~ [ linux-5.0.21 ] ~ [ linux-4.20.17 ] ~ [ linux-4.19.323 ] ~ [ linux-4.18.20 ] ~ [ linux-4.17.19 ] ~ [ linux-4.16.18 ] ~ [ linux-4.15.18 ] ~ [ linux-4.14.336 ] ~ [ linux-4.13.16 ] ~ [ linux-4.12.14 ] ~ [ linux-4.11.12 ] ~ [ linux-4.10.17 ] ~ [ linux-4.9.337 ] ~ [ linux-4.4.302 ] ~ [ linux-3.10.108 ] ~ [ linux-2.6.32.71 ] ~ [ linux-2.6.0 ] ~ [ linux-2.4.37.11 ] ~ [ unix-v6-master ] ~ [ ccs-tools-1.8.12 ] ~ [ policy-sample ] ~
Architecture: ~ [ i386 ] ~ [ alpha ] ~ [ m68k ] ~ [ mips ] ~ [ ppc ] ~ [ sparc ] ~ [ sparc64 ] ~

  1 # SPDX-License-Identifier: GPL-2.0
  2 #
  3 # Copyright 2023 Broadcom
  4 
  5 import gdb
  6 
  7 from linux import constants
  8 from linux import cpus
  9 from linux import utils
 10 from linux import radixtree
 11 
 12 irq_desc_type = utils.CachedType("struct irq_desc")
 13 
 14 def irq_settings_is_hidden(desc):
 15     return desc['status_use_accessors'] & constants.LX_IRQ_HIDDEN
 16 
 17 def irq_desc_is_chained(desc):
 18     return desc['action'] and desc['action'] == gdb.parse_and_eval("&chained_action")
 19 
 20 def irqd_is_level(desc):
 21     return desc['irq_data']['common']['state_use_accessors'] & constants.LX_IRQD_LEVEL
 22 
 23 def show_irq_desc(prec, irq):
 24     text = ""
 25 
 26     desc = radixtree.lookup(gdb.parse_and_eval("&irq_desc_tree"), irq)
 27     if desc is None:
 28         return text
 29 
 30     desc = desc.cast(irq_desc_type.get_type())
 31     if desc is None:
 32         return text
 33 
 34     if irq_settings_is_hidden(desc):
 35         return text
 36 
 37     any_count = 0
 38     if desc['kstat_irqs']:
 39         for cpu in cpus.each_online_cpu():
 40             any_count += cpus.per_cpu(desc['kstat_irqs'], cpu)['cnt']
 41 
 42     if (desc['action'] == 0 or irq_desc_is_chained(desc)) and any_count == 0:
 43         return text;
 44 
 45     text += "%*d: " % (prec, irq)
 46     for cpu in cpus.each_online_cpu():
 47         if desc['kstat_irqs']:
 48             count = cpus.per_cpu(desc['kstat_irqs'], cpu)['cnt']
 49         else:
 50             count = 0
 51         text += "%10u" % (count)
 52 
 53     name = "None"
 54     if desc['irq_data']['chip']:
 55         chip = desc['irq_data']['chip']
 56         if chip['name']:
 57             name = chip['name'].string()
 58         else:
 59             name = "-"
 60 
 61     text += "  %8s" % (name)
 62 
 63     if desc['irq_data']['domain']:
 64         text += "  %*lu" % (prec, desc['irq_data']['hwirq'])
 65     else:
 66         text += "  %*s" % (prec, "")
 67 
 68     if constants.LX_CONFIG_GENERIC_IRQ_SHOW_LEVEL:
 69         text += " %-8s" % ("Level" if irqd_is_level(desc) else "Edge")
 70 
 71     if desc['name']:
 72         text += "-%-8s" % (desc['name'].string())
 73 
 74     """ Some toolchains may not be able to provide information about irqaction """
 75     try:
 76         gdb.lookup_type("struct irqaction")
 77         action = desc['action']
 78         if action is not None:
 79             text += "  %s" % (action['name'].string())
 80             while True:
 81                 action = action['next']
 82                 if action is not None:
 83                     break
 84                 if action['name']:
 85                     text += ", %s" % (action['name'].string())
 86     except:
 87         pass
 88 
 89     text += "\n"
 90 
 91     return text
 92 
 93 def show_irq_err_count(prec):
 94     cnt = utils.gdb_eval_or_none("irq_err_count")
 95     text = ""
 96     if cnt is not None:
 97         text += "%*s: %10u\n" % (prec, "ERR", cnt['counter'])
 98     return text
 99 
100 def x86_show_irqstat(prec, pfx, field, desc):
101     irq_stat = gdb.parse_and_eval("&irq_stat")
102     text = "%*s: " % (prec, pfx)
103     for cpu in cpus.each_online_cpu():
104         stat = cpus.per_cpu(irq_stat, cpu)
105         text += "%10u " % (stat[field])
106     text += "  %s\n" % (desc)
107     return text
108 
109 def x86_show_mce(prec, var, pfx, desc):
110     pvar = gdb.parse_and_eval(var)
111     text = "%*s: " % (prec, pfx)
112     for cpu in cpus.each_online_cpu():
113         text += "%10u " % (cpus.per_cpu(pvar, cpu))
114     text += "  %s\n" % (desc)
115     return text
116 
117 def x86_show_interupts(prec):
118     text = x86_show_irqstat(prec, "NMI", '__nmi_count', 'Non-maskable interrupts')
119 
120     if constants.LX_CONFIG_X86_LOCAL_APIC:
121         text += x86_show_irqstat(prec, "LOC", 'apic_timer_irqs', "Local timer interrupts")
122         text += x86_show_irqstat(prec, "SPU", 'irq_spurious_count', "Spurious interrupts")
123         text += x86_show_irqstat(prec, "PMI", 'apic_perf_irqs', "Performance monitoring interrupts")
124         text += x86_show_irqstat(prec, "IWI", 'apic_irq_work_irqs', "IRQ work interrupts")
125         text += x86_show_irqstat(prec, "RTR", 'icr_read_retry_count', "APIC ICR read retries")
126         if utils.gdb_eval_or_none("x86_platform_ipi_callback") is not None:
127             text += x86_show_irqstat(prec, "PLT", 'x86_platform_ipis', "Platform interrupts")
128 
129     if constants.LX_CONFIG_SMP:
130         text += x86_show_irqstat(prec, "RES", 'irq_resched_count', "Rescheduling interrupts")
131         text += x86_show_irqstat(prec, "CAL", 'irq_call_count', "Function call interrupts")
132         text += x86_show_irqstat(prec, "TLB", 'irq_tlb_count', "TLB shootdowns")
133 
134     if constants.LX_CONFIG_X86_THERMAL_VECTOR:
135         text += x86_show_irqstat(prec, "TRM", 'irq_thermal_count', "Thermal events interrupts")
136 
137     if constants.LX_CONFIG_X86_MCE_THRESHOLD:
138         text += x86_show_irqstat(prec, "THR", 'irq_threshold_count', "Threshold APIC interrupts")
139 
140     if constants.LX_CONFIG_X86_MCE_AMD:
141         text += x86_show_irqstat(prec, "DFR", 'irq_deferred_error_count', "Deferred Error APIC interrupts")
142 
143     if constants.LX_CONFIG_X86_MCE:
144         text += x86_show_mce(prec, "&mce_exception_count", "MCE", "Machine check exceptions")
145         text == x86_show_mce(prec, "&mce_poll_count", "MCP", "Machine check polls")
146 
147     text += show_irq_err_count(prec)
148 
149     if constants.LX_CONFIG_X86_IO_APIC:
150         cnt = utils.gdb_eval_or_none("irq_mis_count")
151         if cnt is not None:
152             text += "%*s: %10u\n" % (prec, "MIS", cnt['counter'])
153 
154     if constants.LX_CONFIG_KVM:
155         text += x86_show_irqstat(prec, "PIN", 'kvm_posted_intr_ipis', 'Posted-interrupt notification event')
156         text += x86_show_irqstat(prec, "NPI", 'kvm_posted_intr_nested_ipis', 'Nested posted-interrupt event')
157         text += x86_show_irqstat(prec, "PIW", 'kvm_posted_intr_wakeup_ipis', 'Posted-interrupt wakeup event')
158 
159     return text
160 
161 def arm_common_show_interrupts(prec):
162     text = ""
163     nr_ipi = utils.gdb_eval_or_none("nr_ipi")
164     ipi_desc = utils.gdb_eval_or_none("ipi_desc")
165     ipi_types = utils.gdb_eval_or_none("ipi_types")
166     if nr_ipi is None or ipi_desc is None or ipi_types is None:
167         return text
168 
169     if prec >= 4:
170         sep = " "
171     else:
172         sep = ""
173 
174     for ipi in range(nr_ipi):
175         text += "%*s%u:%s" % (prec - 1, "IPI", ipi, sep)
176         desc = ipi_desc[ipi].cast(irq_desc_type.get_type().pointer())
177         if desc == 0:
178             continue
179         for cpu in cpus.each_online_cpu():
180             text += "%10u" % (cpus.per_cpu(desc['kstat_irqs'], cpu)['cnt'])
181         text += "      %s" % (ipi_types[ipi].string())
182         text += "\n"
183     return text
184 
185 def aarch64_show_interrupts(prec):
186     text = arm_common_show_interrupts(prec)
187     text += "%*s: %10lu\n" % (prec, "ERR", gdb.parse_and_eval("irq_err_count"))
188     return text
189 
190 def arch_show_interrupts(prec):
191     text = ""
192     if utils.is_target_arch("x86"):
193         text += x86_show_interupts(prec)
194     elif utils.is_target_arch("aarch64"):
195         text += aarch64_show_interrupts(prec)
196     elif utils.is_target_arch("arm"):
197         text += arm_common_show_interrupts(prec)
198     elif utils.is_target_arch("mips"):
199         text += show_irq_err_count(prec)
200     else:
201         raise gdb.GdbError("Unsupported architecture: {}".format(target_arch))
202 
203     return text
204 
205 class LxInterruptList(gdb.Command):
206     """Print /proc/interrupts"""
207 
208     def __init__(self):
209         super(LxInterruptList, self).__init__("lx-interruptlist", gdb.COMMAND_DATA)
210 
211     def invoke(self, arg, from_tty):
212         nr_irqs = gdb.parse_and_eval("nr_irqs")
213         prec = 3
214         j = 1000
215         while prec < 10 and j <= nr_irqs:
216             prec += 1
217             j *= 10
218 
219         gdb.write("%*s" % (prec + 8, ""))
220         for cpu in cpus.each_online_cpu():
221             gdb.write("CPU%-8d" % cpu)
222         gdb.write("\n")
223 
224         if utils.gdb_eval_or_none("&irq_desc_tree") is None:
225             return
226 
227         for irq in range(nr_irqs):
228             gdb.write(show_irq_desc(prec, irq))
229         gdb.write(arch_show_interrupts(prec))
230 
231 
232 LxInterruptList()

~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~

kernel.org | git.kernel.org | LWN.net | Project Home | SVN repository | Mail admin

Linux® is a registered trademark of Linus Torvalds in the United States and other countries.
TOMOYO® is a registered trademark of NTT DATA CORPORATION.

sflogo.php