Received: by 2002:a25:ad19:0:0:0:0:0 with SMTP id y25csp1683740ybi; Wed, 17 Jul 2019 20:04:51 -0700 (PDT) X-Google-Smtp-Source: APXvYqzkmrd2Nwew9wqEg2Fp44h9iOv6amQnwS2FLyQ+ROtXMfCTcGvziAKx02dwlXGhE4m1akFm X-Received: by 2002:a65:4d4e:: with SMTP id j14mr44691728pgt.50.1563419091659; Wed, 17 Jul 2019 20:04:51 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1563419091; cv=none; d=google.com; s=arc-20160816; b=sQd9fxb5BbqR5Qn7uYtswTpSkQO2ix0GCwydBjngovmbj0Iijdu9s77QVWbODbTHbr 8reJ07/j/RcKa8DyTnK4Pv9HIOdq+Zn4m5UW6L8A6g93uIptxjxwPtqe/K3ECVvqJl0x gZDhJgmobPTHoWhJ1sudDr+Lcm6x0jHJRXLkUKCEnTmdbAThlV9oYAISFDPs6QRSV74E icw7srQk8xm8n+EYOWwpp6V0UakzWmHBIdSGODw/P/XCKS4oKcEh1Ml9qD//gza90Qv5 rIKWKSfHeA/O6NiW769LPJPrrNGkqQeFo7cH/n1UNcxjqTFka829mAjXPII8++kkYnBo 0y0A== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding:mime-version :user-agent:references:in-reply-to:message-id:date:subject:cc:to :from:dkim-signature; bh=PTqxT421hPubiZJMAIgOQZG4KSOviR9XAF8RtWYOEiA=; b=t4n4tXQC6KxTQ8eix8eWXOd2y5EcGbZMCF40/9yAaY6yEG4hiB3ZfD/3BtcaSU8K3C zt+ncKI6KyyXahjF51qTJDi//hZ0UiZAeCZO16esPVgpAaqYl0vCk92WHK5EuWJvBqNu IvWaqNX+BOq/yucK2pRupP55A0KepJ7W4WddZqh+zVmtgFFL+4s5I5uYt4AznEWXwlHz UAVdsrf1Ki4gjwYCMldHXIV1WI355UV8SABgtrM4CSMKZaINYDzk6gUv5y7AOP4ZFqe9 4GR+8sSPCghIHE2MuQn3lM0OPY/o7kqHS2f6UkDFK308Kuy4FFK9XjOzVExT8cjHmVhJ 9P1Q== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@kernel.org header.s=default header.b=BfDDNix9; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id j74si714520pje.12.2019.07.17.20.04.34; Wed, 17 Jul 2019 20:04:51 -0700 (PDT) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=pass header.i=@kernel.org header.s=default header.b=BfDDNix9; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S2389410AbfGRDDj (ORCPT + 99 others); Wed, 17 Jul 2019 23:03:39 -0400 Received: from mail.kernel.org ([198.145.29.99]:34340 "EHLO mail.kernel.org" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S2389356AbfGRDDh (ORCPT ); Wed, 17 Jul 2019 23:03:37 -0400 Received: from localhost (115.42.148.210.bf.2iij.net [210.148.42.115]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by mail.kernel.org (Postfix) with ESMTPSA id 254AB21852; Thu, 18 Jul 2019 03:03:36 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=kernel.org; s=default; t=1563419016; bh=mV5teUn+3qRhaigH+wulIqmPjLqPh/jUeac6KggzE2M=; h=From:To:Cc:Subject:Date:In-Reply-To:References:From; b=BfDDNix9KbjhiWA+Svvlgv//5Xmbosd7oacc4IwWVqtSO6SbgzxCBxLZ4afoLAoRE Gt7kh9/xggvoCyzDLpdRjSAJjfx+IdzDQ+2gkKUEAD7YUflor7MQ/vZFGyL48rjmJp iyDKHctKX3L1h2WocXXfA2OWDumDEDTZhqNGJyC8= From: Greg Kroah-Hartman To: linux-kernel@vger.kernel.org Cc: Greg Kroah-Hartman , stable@vger.kernel.org, Robert Hodaszi , Thomas Gleixner , Marc Zyngier Subject: [PATCH 5.2 10/21] x86/ioapic: Implement irq_get_irqchip_state() callback Date: Thu, 18 Jul 2019 12:01:28 +0900 Message-Id: <20190718030032.662554163@linuxfoundation.org> X-Mailer: git-send-email 2.22.0 In-Reply-To: <20190718030030.456918453@linuxfoundation.org> References: <20190718030030.456918453@linuxfoundation.org> User-Agent: quilt/0.66 MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org From: Thomas Gleixner commit dfe0cf8b51b07e56ded571e3de0a4a9382517231 upstream. When an interrupt is shut down in free_irq() there might be an inflight interrupt pending in the IO-APIC remote IRR which is not yet serviced. That means the interrupt has been sent to the target CPUs local APIC, but the target CPU is in a state which delays the servicing. So free_irq() would proceed to free resources and to clear the vector because synchronize_hardirq() does not see an interrupt handler in progress. That can trigger a spurious interrupt warning, which is harmless and just confuses users, but it also can leave the remote IRR in a stale state because once the handler is invoked the interrupt resources might be freed already and therefore acknowledgement is not possible anymore. Implement the irq_get_irqchip_state() callback for the IO-APIC irq chip. The callback is invoked from free_irq() via __synchronize_hardirq(). Check the remote IRR bit of the interrupt and return 'in flight' if it is set and the interrupt is configured in level mode. For edge mode the remote IRR has no meaning. As this is only meaningful for level triggered interrupts this won't cure the potential spurious interrupt warning for edge triggered interrupts, but the edge trigger case does not result in stale hardware state. This has to be addressed at the vector/interrupt entry level seperately. Fixes: 464d12309e1b ("x86/vector: Switch IOAPIC to global reservation mode") Reported-by: Robert Hodaszi Signed-off-by: Thomas Gleixner Cc: Marc Zyngier Link: https://lkml.kernel.org/r/20190628111440.370295517@linutronix.de Signed-off-by: Greg Kroah-Hartman --- arch/x86/kernel/apic/io_apic.c | 46 +++++++++++++++++++++++++++++++++++++++++ 1 file changed, 46 insertions(+) --- a/arch/x86/kernel/apic/io_apic.c +++ b/arch/x86/kernel/apic/io_apic.c @@ -1893,6 +1893,50 @@ static int ioapic_set_affinity(struct ir return ret; } +/* + * Interrupt shutdown masks the ioapic pin, but the interrupt might already + * be in flight, but not yet serviced by the target CPU. That means + * __synchronize_hardirq() would return and claim that everything is calmed + * down. So free_irq() would proceed and deactivate the interrupt and free + * resources. + * + * Once the target CPU comes around to service it it will find a cleared + * vector and complain. While the spurious interrupt is harmless, the full + * release of resources might prevent the interrupt from being acknowledged + * which keeps the hardware in a weird state. + * + * Verify that the corresponding Remote-IRR bits are clear. + */ +static int ioapic_irq_get_chip_state(struct irq_data *irqd, + enum irqchip_irq_state which, + bool *state) +{ + struct mp_chip_data *mcd = irqd->chip_data; + struct IO_APIC_route_entry rentry; + struct irq_pin_list *p; + + if (which != IRQCHIP_STATE_ACTIVE) + return -EINVAL; + + *state = false; + raw_spin_lock(&ioapic_lock); + for_each_irq_pin(p, mcd->irq_2_pin) { + rentry = __ioapic_read_entry(p->apic, p->pin); + /* + * The remote IRR is only valid in level trigger mode. It's + * meaning is undefined for edge triggered interrupts and + * irrelevant because the IO-APIC treats them as fire and + * forget. + */ + if (rentry.irr && rentry.trigger) { + *state = true; + break; + } + } + raw_spin_unlock(&ioapic_lock); + return 0; +} + static struct irq_chip ioapic_chip __read_mostly = { .name = "IO-APIC", .irq_startup = startup_ioapic_irq, @@ -1902,6 +1946,7 @@ static struct irq_chip ioapic_chip __rea .irq_eoi = ioapic_ack_level, .irq_set_affinity = ioapic_set_affinity, .irq_retrigger = irq_chip_retrigger_hierarchy, + .irq_get_irqchip_state = ioapic_irq_get_chip_state, .flags = IRQCHIP_SKIP_SET_WAKE, }; @@ -1914,6 +1959,7 @@ static struct irq_chip ioapic_ir_chip __ .irq_eoi = ioapic_ir_ack_level, .irq_set_affinity = ioapic_set_affinity, .irq_retrigger = irq_chip_retrigger_hierarchy, + .irq_get_irqchip_state = ioapic_irq_get_chip_state, .flags = IRQCHIP_SKIP_SET_WAKE, };