d0cac39e4e
Based upon a report by Meelis Roos. Sparc64 SBUS and PCI controllers use a combination of IMAP and ICLR registers to manage device interrupts. The IMAP register contains the "valid" enable bit as well as CPU targetting information. Whereas the ICLR register is written with zero at the end of handling an interrupt to reset the state machine for that interrupt to IDLE so it can be sent again. For PCI slot and SBUS slot devices we can have multiple interrupts sharing the same IMAP register. There are individual ICLR registers but only one IMAP register for managing those. We represent each shared case with individual virtual IRQs so the generic IRQ layer thinks there is only one user of the IRQ instance. In such shared IMAP cases this is wrong, so if there are multiple active users then a free_irq() call will prematurely turn off the interrupt by clearing the Valid bit in the IMAP register even though there are other active users. Fix this by simply doing nothing in sun4u_disable_irq() and checking IRQF_DISABLED during IRQ dispatch. This situation doesn't exist in the hypervisor sun4v cases, so I left those alone. Tested-by: Meelis Roos <mroos@linux.ee> Signed-off-by: David S. Miller <davem@davemloft.net> |
||
---|---|---|
.. | ||
.gitignore | ||
Makefile | ||
apc.c | ||
asm-offsets.c | ||
audit.c | ||
auxio_32.c | ||
auxio_64.c | ||
central.c | ||
cherrs.S | ||
chmc.c | ||
compat_audit.c | ||
cpu.c | ||
devices.c | ||
dma.c | ||
dma.h | ||
ds.c | ||
dtlb_miss.S | ||
dtlb_prot.S | ||
ebus.c | ||
entry.S | ||
entry.h | ||
etrap_32.S | ||
etrap_64.S | ||
fpu_traps.S | ||
ftrace.c | ||
getsetcc.S | ||
head_32.S | ||
head_64.S | ||
helpers.S | ||
hvapi.c | ||
hvcalls.S | ||
hvtramp.S | ||
idprom.c | ||
init_task.c | ||
iommu.c | ||
iommu_common.h | ||
ioport.c | ||
irq.h | ||
irq_32.c | ||
irq_64.c | ||
itlb_miss.S | ||
ivec.S | ||
kernel.h | ||
kgdb_32.c | ||
kgdb_64.c | ||
kprobes.c | ||
kstack.h | ||
ktlb.S | ||
ldc.c | ||
led.c | ||
mdesc.c | ||
misctrap.S | ||
module.c | ||
muldiv.c | ||
nmi.c | ||
of_device_32.c | ||
of_device_64.c | ||
pci.c | ||
pci_common.c | ||
pci_fire.c | ||
pci_impl.h | ||
pci_msi.c | ||
pci_psycho.c | ||
pci_sabre.c | ||
pci_schizo.c | ||
pci_sun4v.c | ||
pci_sun4v.h | ||
pci_sun4v_asm.S | ||
pcic.c | ||
pcr.c | ||
pmc.c | ||
power.c | ||
process_32.c | ||
process_64.c | ||
prom.h | ||
prom_32.c | ||
prom_64.c | ||
prom_common.c | ||
prom_irqtrans.c | ||
psycho_common.c | ||
psycho_common.h | ||
ptrace_32.c | ||
ptrace_64.c | ||
reboot.c | ||
rtrap_32.S | ||
rtrap_64.S | ||
sbus.c | ||
setup_32.c | ||
setup_64.c | ||
signal32.c | ||
signal_32.c | ||
signal_64.c | ||
smp_32.c | ||
smp_64.c | ||
sparc_ksyms_32.c | ||
sparc_ksyms_64.c | ||
spiterrs.S | ||
sstate.c | ||
stacktrace.c | ||
starfire.c | ||
sun4c_irq.c | ||
sun4d_irq.c | ||
sun4d_smp.c | ||
sun4m_irq.c | ||
sun4m_smp.c | ||
sun4v_ivec.S | ||
sun4v_tlb_miss.S | ||
sys32.S | ||
sys_sparc32.c | ||
sys_sparc_32.c | ||
sys_sparc_64.c | ||
syscalls.S | ||
sysfs.c | ||
systbls.h | ||
systbls_32.S | ||
systbls_64.S | ||
tadpole.c | ||
tick14.c | ||
time_32.c | ||
time_64.c | ||
trampoline_32.S | ||
trampoline_64.S | ||
traps_32.c | ||
traps_64.c | ||
tsb.S | ||
ttable.S | ||
una_asm_32.S | ||
una_asm_64.S | ||
unaligned_32.c | ||
unaligned_64.c | ||
us2e_cpufreq.c | ||
us3_cpufreq.c | ||
utrap.S | ||
vio.c | ||
viohs.c | ||
visemul.c | ||
vmlinux.lds.S | ||
windows.c | ||
winfixup.S | ||
wof.S | ||
wuf.S |