]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
PCI/MSI: Add an option to write MSIX ENTRY_DATA before any reads
authorJonathan Currier <dullfire@yahoo.com>
Sat, 6 Sep 2025 20:48:26 +0000 (16:48 -0400)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Tue, 9 Sep 2025 16:54:18 +0000 (18:54 +0200)
[ Upstream commit cf761e3dacc6ad5f65a4886d00da1f9681e6805a ]

Commit 7d5ec3d36123 ("PCI/MSI: Mask all unused MSI-X entries") introduced a
readl() from ENTRY_VECTOR_CTRL before the writel() to ENTRY_DATA.

This is correct, however some hardware, like the Sun Neptune chips, the NIU
module, will cause an error and/or fatal trap if any MSIX table entry is
read before the corresponding ENTRY_DATA field is written to.

Add an optional early writel() in msix_prepare_msi_desc().

Fixes: 7d5ec3d36123 ("PCI/MSI: Mask all unused MSI-X entries")
Signed-off-by: Jonathan Currier <dullfire@yahoo.com>
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Cc: stable@vger.kernel.org
Link: https://lore.kernel.org/all/20241117234843.19236-2-dullfire@yahoo.com
[ Applied workaround to msix_setup_msi_descs() instead of msix_prepare_msi_desc() ]
Signed-off-by: Sasha Levin <sashal@kernel.org>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
drivers/pci/msi/msi.c
include/linux/pci.h

index c5cc3e453fd0c59c2f4dfba3d704d37a07f7c9f7..7110aed956c75049dc0e8e8e0a96424c563d3871 100644 (file)
@@ -534,6 +534,9 @@ static int msix_setup_msi_descs(struct pci_dev *dev, void __iomem *base,
 
                if (desc.pci.msi_attrib.can_mask) {
                        addr = pci_msix_desc_addr(&desc);
+                       /* Workaround for SUN NIU insanity, which requires write before read */
+                       if (dev->dev_flags & PCI_DEV_FLAGS_MSIX_TOUCH_ENTRY_DATA_FIRST)
+                               writel(0, addr + PCI_MSIX_ENTRY_DATA);
                        desc.pci.msix_ctrl = readl(addr + PCI_MSIX_ENTRY_VECTOR_CTRL);
                }
 
index df36df4695ed56c5d0f5e9f9f5262029e0d23842..59dfa59d6d59765896956511cf16cebccdfc6986 100644 (file)
@@ -244,6 +244,8 @@ enum pci_dev_flags {
        PCI_DEV_FLAGS_NO_RELAXED_ORDERING = (__force pci_dev_flags_t) (1 << 11),
        /* Device does honor MSI masking despite saying otherwise */
        PCI_DEV_FLAGS_HAS_MSI_MASKING = (__force pci_dev_flags_t) (1 << 12),
+       /* Device requires write to PCI_MSIX_ENTRY_DATA before any MSIX reads */
+       PCI_DEV_FLAGS_MSIX_TOUCH_ENTRY_DATA_FIRST = (__force pci_dev_flags_t) (1 << 13),
 };
 
 enum pci_irq_reroute_variant {