Skip to content

Commit ab0f9e7

Browse files
Alexander Gordeevhtejun
authored andcommitted
ahci: Ensure "MSI Revert to Single Message" mode is not enforced
The AHCI specification allows hardware to choose to revert to single MSI mode when fewer messages are allocated than requested. Yet, at least ICH10 chipset reverts to single MSI mode even when enough messages are allocated in some cases (see below). This update forces the driver to not rely on initialization of multiple MSIs mode alone and always check if "MSI Revert to Single Message" (MRSM) mode was enforced by the controller and fallback to the single MSI mode in case it did. That prevents a situation when the driver configured multiple per-port IRQ handlers, but the controller sends all port's interrupts to a single IRQ, which could easily screw up the interrupt handling and lead to delays and possibly crashes. The fix was tested on a 6-port controller that successfully reverted to the single MSI mode: 00:1f.2 SATA controller: Intel Corporation 82801JI (ICH10 Family) SATA AHCI Controller (prog-if 01 [AHCI 1.0]) Subsystem: Super Micro Computer Inc Device 10a7 Flags: bus master, 66MHz, medium devsel, latency 0, IRQ 101 I/O ports at f110 [size=8] I/O ports at f100 [size=4] I/O ports at f0f0 [size=8] I/O ports at f0e0 [size=4] I/O ports at f020 [size=32] Memory at fbf00000 (32-bit, non-prefetchable) [size=2K] Capabilities: [80] MSI: Enable+ Count=1/16 Maskable- 64bit- Capabilities: [70] Power Management version 3 Capabilities: [a8] SATA HBA v1.0 Capabilities: [b0] PCI Advanced Features Kernel driver in use: ahci With 6 ports just 8 MSI vectors should be enough, but the adapter enforces the MRSM mode when less than 16 vectors are written to the Multiple Messages Enable PCI register. I instigated MRSM mode by forcing @Nvec to 8 in ahci_init_interrupts(). Signed-off-by: Alexander Gordeev <agordeev@redhat.com> Cc: linux-ide@vger.kernel.org Cc: stable@vger.kernel.org Signed-off-by: Tejun Heo <tj@kernel.org>
1 parent 9ae794a commit ab0f9e7

File tree

2 files changed

+9
-1
lines changed

2 files changed

+9
-1
lines changed

drivers/ata/ahci.c

Lines changed: 8 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1164,7 +1164,7 @@ static inline void ahci_gtf_filter_workaround(struct ata_host *host)
11641164
#endif
11651165

11661166
static int ahci_init_interrupts(struct pci_dev *pdev, unsigned int n_ports,
1167-
struct ahci_host_priv *hpriv)
1167+
struct ahci_host_priv *hpriv)
11681168
{
11691169
int nvec;
11701170

@@ -1189,6 +1189,13 @@ static int ahci_init_interrupts(struct pci_dev *pdev, unsigned int n_ports,
11891189
else if (nvec < 0)
11901190
goto intx;
11911191

1192+
/* fallback to single MSI mode if the controller enforced MRSM mode */
1193+
if (readl(hpriv->mmio + HOST_CTL) & HOST_MRSM) {
1194+
pci_disable_msi(pdev);
1195+
printk(KERN_INFO "ahci: MRSM is on, fallback to single MSI\n");
1196+
goto single_msi;
1197+
}
1198+
11921199
return nvec;
11931200

11941201
single_msi:

drivers/ata/ahci.h

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -94,6 +94,7 @@ enum {
9494
/* HOST_CTL bits */
9595
HOST_RESET = (1 << 0), /* reset controller; self-clear */
9696
HOST_IRQ_EN = (1 << 1), /* global IRQ enable */
97+
HOST_MRSM = (1 << 2), /* MSI Revert to Single Message */
9798
HOST_AHCI_EN = (1 << 31), /* AHCI enabled */
9899

99100
/* HOST_CAP bits */

0 commit comments

Comments
 (0)