<html><head><meta name="color-scheme" content="light dark"></head><body><pre style="word-wrap: break-word; white-space: pre-wrap;">From: Felix Fietkau &lt;nbd@nbd.name&gt;
Date: Fri, 4 Sep 2020 18:42:42 +0200
Subject: [PATCH] pci: pcie-mediatek: add support for coherent DMA

It improves performance by eliminating the need for a cache flush for DMA on
attached devices

Signed-off-by: Felix Fietkau &lt;nbd@nbd.name&gt;
---

--- a/arch/arm64/boot/dts/mediatek/mt7622.dtsi
+++ b/arch/arm64/boot/dts/mediatek/mt7622.dtsi
@@ -837,6 +837,9 @@
 		bus-range = &lt;0x00 0xff&gt;;
 		ranges = &lt;0x82000000 0 0x20000000 0x0 0x20000000 0 0x8000000&gt;;
 		status = "disabled";
+		dma-coherent;
+		mediatek,hifsys = &lt;&amp;hifsys&gt;;
+		mediatek,cci-control = &lt;&amp;cci_control2&gt;;
 
 		#interrupt-cells = &lt;1&gt;;
 		interrupt-map-mask = &lt;0 0 0 7&gt;;
@@ -881,6 +884,9 @@
 		bus-range = &lt;0x00 0xff&gt;;
 		ranges = &lt;0x82000000 0 0x28000000 0x0 0x28000000 0 0x8000000&gt;;
 		status = "disabled";
+		dma-coherent;
+		mediatek,hifsys = &lt;&amp;hifsys&gt;;
+		mediatek,cci-control = &lt;&amp;cci_control2&gt;;
 
 		#interrupt-cells = &lt;1&gt;;
 		interrupt-map-mask = &lt;0 0 0 7&gt;;
--- a/drivers/pci/controller/pcie-mediatek.c
+++ b/drivers/pci/controller/pcie-mediatek.c
@@ -20,6 +20,7 @@
 #include &lt;linux/of_address.h&gt;
 #include &lt;linux/of_pci.h&gt;
 #include &lt;linux/of_platform.h&gt;
+#include &lt;linux/of_address.h&gt;
 #include &lt;linux/pci.h&gt;
 #include &lt;linux/phy/phy.h&gt;
 #include &lt;linux/platform_device.h&gt;
@@ -139,6 +140,11 @@
 #define PCIE_LINK_STATUS_V2	0x804
 #define PCIE_PORT_LINKUP_V2	BIT(10)
 
+/* DMA channel mapping */
+#define HIFSYS_DMA_AG_MAP	0x008
+#define HIFSYS_DMA_AG_MAP_PCIE0	BIT(0)
+#define HIFSYS_DMA_AG_MAP_PCIE1	BIT(1)
+
 struct mtk_pcie_port;
 
 /**
@@ -1060,6 +1066,27 @@ static int mtk_pcie_setup(struct mtk_pci
 	struct mtk_pcie_port *port, *tmp;
 	int err, slot;
 
+	if (of_dma_is_coherent(node)) {
+		struct regmap *con;
+		u32 mask;
+
+		con = syscon_regmap_lookup_by_phandle(node,
+						      "mediatek,cci-control");
+		/* enable CPU/bus coherency */
+		if (!IS_ERR(con))
+			regmap_write(con, 0, 3);
+
+		con = syscon_regmap_lookup_by_phandle(node,
+						      "mediatek,hifsys");
+		if (IS_ERR(con)) {
+			dev_err(dev, "missing hifsys node\n");
+			return PTR_ERR(con);
+		}
+
+		mask = HIFSYS_DMA_AG_MAP_PCIE0 | HIFSYS_DMA_AG_MAP_PCIE1;
+		regmap_update_bits(con, HIFSYS_DMA_AG_MAP, mask, mask);
+	}
+
 	slot = of_get_pci_domain_nr(dev-&gt;of_node);
 	if (slot &lt; 0) {
 		for_each_available_child_of_node(node, child) {
</pre></body></html>