public inbox for linux-pci@vger.kernel.org
 help / color / mirror / Atom feed
* [PATCH] PCI: pcie-altera: Set MPS to MPSS on Agilex 7 Root Ports
@ 2025-11-10 17:00 Mahesh Vaidya
  2025-11-10 20:06 ` Niklas Cassel
  0 siblings, 1 reply; 4+ messages in thread
From: Mahesh Vaidya @ 2025-11-10 17:00 UTC (permalink / raw)
  To: joyce.ooi, linux-pci
  Cc: subhransu.sekhar.prusty, krishna.kumar.simmadhari.ramadass,
	nanditha.jayarajan, Mahesh Vaidya

The Altera Agilex 7 Root Port (RP) defaults its Device Control
(DEVCTL) register's MPS setting to 128 bytes upon power-on.
When the kernel's PCIe core enumerates the bus (using the default
PCIE_BUS_DEFAULT policy), it observes this 128-byte current setting
and limits all downstream Endpoints to 128 bytes.
This occurs even if both the RP and the Endpoint support a higher MPSS
(e.g., 256 or 512 bytes), resulting in sub-optimal DMA performance.

This patch fixes the issue by reading the RP's actual MPSS from its
Device Capability (DEVCAP) register and writing this value into the
DEVCTL register, overriding the 128-byte default value.
As this fix is called in driver's probe function before the PCI bus
is scanned, it ensures that when the kernel's PCI core enumerates the
downstream port, it reads the correct, maximum-supported MPS from the
RP's DEVCTL and can negotiate the optimal MPS for the Endpoint.

Signed-off-by: Mahesh Vaidya <mahesh.vaidya@altera.com>
Signed-off-by: Subhransu S. Prusty <subhransu.sekhar.prusty@altera.com>
---
 drivers/pci/controller/pcie-altera.c | 38 ++++++++++++++++++++++++++++
 1 file changed, 38 insertions(+)

diff --git a/drivers/pci/controller/pcie-altera.c b/drivers/pci/controller/pcie-altera.c
index 3dbb7adc421c..4e23df9f5d7c 100644
--- a/drivers/pci/controller/pcie-altera.c
+++ b/drivers/pci/controller/pcie-altera.c
@@ -783,6 +783,38 @@ static void altera_pcie_retrain(struct altera_pcie *pcie)
 	}
 }
 
+/**
+ * altera_pcie_set_mps_to_mpss - Set RP MPS to its maximum supported value
+ * @pcie: Altera PCIe controller instance
+ *
+ * The Max Payload Size (MPS) in the Device Control (DEVCTL) register of
+ * the PCIe root port defaults to 128 bytes. This must be updated to the
+ * Max Payload Size Supported (MPSS) value of the Device Capabilities
+ * (DEVCAP) register, before the bus is scanned otherwise the kernel will
+ * limit all downstream devices to negotiate to 128 bytes.
+ *
+ * We cannot use pcie_set_mps() here, as this logic must run
+ * before the RP's pci_dev is created and the bus is enumerated.
+ */
+static void altera_pcie_set_mps_to_mpss(struct altera_pcie *pcie)
+{
+	u16 devcap, devctl;
+
+	altera_read_cap_word(pcie, pcie->root_bus_nr, RP_DEVFN, PCI_EXP_DEVCAP,
+			     &devcap);
+	altera_read_cap_word(pcie, pcie->root_bus_nr, RP_DEVFN, PCI_EXP_DEVCTL,
+			     &devctl);
+
+	/* Clear MPS bits in Device Control register */
+	devctl &= ~PCI_EXP_DEVCTL_PAYLOAD;
+
+	/* Set MPS in Device Control to MPSS from Device Capabilities */
+	devctl |= (devcap & PCI_EXP_DEVCAP_PAYLOAD) << 5;
+
+	altera_write_cap_word(pcie, pcie->root_bus_nr, RP_DEVFN, PCI_EXP_DEVCTL,
+			      devctl);
+}
+
 static int altera_pcie_intx_map(struct irq_domain *domain, unsigned int irq,
 				irq_hw_number_t hwirq)
 {
@@ -1031,6 +1063,12 @@ static int altera_pcie_probe(struct platform_device *pdev)
 		writel(CFG_AER,
 		       pcie->hip_base + pcie->pcie_data->port_conf_offset +
 		       pcie->pcie_data->port_irq_enable_offset);
+
+		/*
+		 * Set RP's MPS value to its MPSS value before bus scan to avoid
+		 * kernel defaulting to 128 bytes.
+		 */
+		altera_pcie_set_mps_to_mpss(pcie);
 	}
 
 	bridge->sysdata = pcie;
-- 
2.34.1


^ permalink raw reply related	[flat|nested] 4+ messages in thread

end of thread, other threads:[~2025-11-11 16:00 UTC | newest]

Thread overview: 4+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2025-11-10 17:00 [PATCH] PCI: pcie-altera: Set MPS to MPSS on Agilex 7 Root Ports Mahesh Vaidya
2025-11-10 20:06 ` Niklas Cassel
2025-11-11 15:21   ` Mahesh Vaidya
2025-11-11 16:00     ` Niklas Cassel

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox