]> git.ipfire.org Git - thirdparty/linux.git/commitdiff
MIPS: Malta: Fix !EVA SOC-it PCI MMIO
authorMaciej W. Rozycki <macro@orcam.me.uk>
Mon, 20 Oct 2025 01:11:49 +0000 (02:11 +0100)
committerThomas Bogendoerfer <tsbogend@alpha.franken.de>
Tue, 11 Nov 2025 11:58:49 +0000 (12:58 +0100)
Fix a regression that has caused accesses to the PCI MMIO window to
complete unclaimed in non-EVA configurations with the SOC-it family of
system controllers, preventing PCI devices from working that use MMIO.

In the non-EVA case PHYS_OFFSET is set to 0, meaning that PCI_BAR0 is
set with an empty mask (and PCI_HEAD4 matches addresses starting from 0
accordingly).  Consequently all addresses are matched for incoming DMA
accesses from PCI.  This seems to confuse the system controller's logic
and outgoing bus cycles targeting the PCI MMIO window seem not to make
it to the intended devices.

This happens as well when a wider mask is used with PCI_BAR0, such as
0x80000000 or 0xe0000000, that makes addresses match that overlap with
the PCI MMIO window, which starts at 0x10000000 in our configuration.

Set the mask in PCI_BAR0 to 0xf0000000 for non-EVA then, covering the
non-EVA maximum 256 MiB of RAM, which is what YAMON does and which used
to work correctly up to the offending commit.  Set PCI_P2SCMSKL to match
PCI_BAR0 as required by the system controller's specification, and match
PCI_P2SCMAPL to PCI_HEAD4 for identity mapping.

Verified with:

Core board type/revision =      0x0d (Core74K) / 0x01
System controller/revision =    MIPS SOC-it 101 OCP / 1.3   SDR-FW-4:1
Processor Company ID/options =  0x01 (MIPS Technologies, Inc.) / 0x1c
Processor ID/revision =         0x97 (MIPS 74Kf) / 0x4c

for non-EVA and with:

Core board type/revision =      0x0c (CoreFPGA-5) / 0x00
System controller/revision =    MIPS ROC-it2 / 0.0   FW-1:1 (CLK_unknown) GIC
Processor Company ID/options =  0x01 (MIPS Technologies, Inc.) / 0x00
Processor ID/revision =         0xa0 (MIPS interAptiv UP) / 0x20

for EVA/non-EVA, fixing:

defxx 0000:00:12.0: assign IRQ: got 10
defxx: v1.12 2021/03/10  Lawrence V. Stefani and others
0000:00:12.0: Could not read adapter factory MAC address!

vs:

defxx 0000:00:12.0: assign IRQ: got 10
defxx: v1.12 2021/03/10  Lawrence V. Stefani and others
0000:00:12.0: DEFPA at MMIO addr = 0x10142000, IRQ = 10, Hardware addr = 00-00-f8-xx-xx-xx
0000:00:12.0: registered as fddi0

for non-EVA and causing no change for EVA.

Signed-off-by: Maciej W. Rozycki <macro@orcam.me.uk>
Fixes: 422dd256642b ("MIPS: Malta: Allow PCI devices DMA to lower 2GB physical")
Cc: stable@vger.kernel.org # v4.9+
Signed-off-by: Thomas Bogendoerfer <tsbogend@alpha.franken.de>
arch/mips/mti-malta/malta-init.c

index 000d6d50520a89c70ea200f2bbf4934f2d43bf44..82b0fd8576a241fb70293ebd99a7dcf5f435a2dc 100644 (file)
@@ -241,16 +241,22 @@ mips_pci_controller:
 #endif
 
                /*
-                * Setup the Malta max (2GB) memory for PCI DMA in host bridge
-                * in transparent addressing mode.
+                * Set up memory mapping in host bridge for PCI DMA masters,
+                * in transparent addressing mode.  For EVA use the Malta
+                * maximum of 2 GiB memory in the alias space at 0x80000000
+                * as per PHYS_OFFSET.  Otherwise use 256 MiB of memory in
+                * the regular space, avoiding mapping the PCI MMIO window
+                * for DMA as it seems to confuse the system controller's
+                * logic, causing PCI MMIO to stop working.
                 */
-               mask = PHYS_OFFSET | PCI_BASE_ADDRESS_MEM_PREFETCH;
-               MSC_WRITE(MSC01_PCI_BAR0, mask);
-               MSC_WRITE(MSC01_PCI_HEAD4, mask);
+               mask = PHYS_OFFSET ? PHYS_OFFSET : 0xf0000000;
+               MSC_WRITE(MSC01_PCI_BAR0,
+                         mask | PCI_BASE_ADDRESS_MEM_PREFETCH);
+               MSC_WRITE(MSC01_PCI_HEAD4,
+                         PHYS_OFFSET | PCI_BASE_ADDRESS_MEM_PREFETCH);
 
-               mask &= MSC01_PCI_BAR0_SIZE_MSK;
                MSC_WRITE(MSC01_PCI_P2SCMSKL, mask);
-               MSC_WRITE(MSC01_PCI_P2SCMAPL, mask);
+               MSC_WRITE(MSC01_PCI_P2SCMAPL, PHYS_OFFSET);
 
                /* Don't handle target retries indefinitely.  */
                if ((data & MSC01_PCI_CFG_MAXRTRY_MSK) ==