summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/soc/mediatek/mt8195/Makefile.inc1
-rw-r--r--src/soc/mediatek/mt8195/include/soc/pcie.h11
-rw-r--r--src/soc/mediatek/mt8195/pcie.c100
3 files changed, 112 insertions, 0 deletions
diff --git a/src/soc/mediatek/mt8195/Makefile.inc b/src/soc/mediatek/mt8195/Makefile.inc
index 8ab78a80e2..68fe0f39b1 100644
--- a/src/soc/mediatek/mt8195/Makefile.inc
+++ b/src/soc/mediatek/mt8195/Makefile.inc
@@ -70,6 +70,7 @@ ramstage-y += ../common/mmu_operations.c ../common/mmu_cmops.c
ramstage-y += mt6360.c
ramstage-y += ../common/mtcmos.c mtcmos.c
ramstage-$(CONFIG_COMMONLIB_STORAGE_MMC) += ../common/msdc.c msdc.c
+ramstage-$(CONFIG_PCI) += ../common/pcie.c pcie.c
ramstage-y += ../common/pll.c pll.c
ramstage-y += ../common/pmif.c
ramstage-y += ../common/rtc.c ../common/rtc_mt6359p.c
diff --git a/src/soc/mediatek/mt8195/include/soc/pcie.h b/src/soc/mediatek/mt8195/include/soc/pcie.h
new file mode 100644
index 0000000000..0978531645
--- /dev/null
+++ b/src/soc/mediatek/mt8195/include/soc/pcie.h
@@ -0,0 +1,11 @@
+/* SPDX-License-Identifier: GPL-2.0-only */
+
+#ifndef SOC_MEDIATEK_MT8195_PCIE_H
+#define SOC_MEDIATEK_MT8195_PCIE_H
+
+#include <soc/pcie_common.h>
+
+void mtk_pcie_pre_init(void);
+void mtk_pcie_get_hw_info(struct mtk_pcie_controller *ctrl);
+
+#endif
diff --git a/src/soc/mediatek/mt8195/pcie.c b/src/soc/mediatek/mt8195/pcie.c
new file mode 100644
index 0000000000..8a1fca479b
--- /dev/null
+++ b/src/soc/mediatek/mt8195/pcie.c
@@ -0,0 +1,100 @@
+/* SPDX-License-Identifier: GPL-2.0-only */
+
+#include <commonlib/stdlib.h>
+#include <console/console.h>
+#include <device/mmio.h>
+#include <device/resource.h>
+#include <delay.h>
+#include <soc/addressmap.h>
+#include <soc/gpio.h>
+#include <soc/pcie.h>
+#include <soc/pcie_common.h>
+#include <stdlib.h>
+#include <string.h>
+
+#define PCIE_REG_BASE_PORT0 0x112f0000
+#define PCIE_RST_CTRL_REG (PCIE_REG_BASE_PORT0 + 0x148)
+#define PCIE_MAC_RSTB BIT(0)
+#define PCIE_PHY_RSTB BIT(1)
+#define PCIE_BRG_RSTB BIT(2)
+#define PCIE_PE_RSTB BIT(3)
+
+
+/* MMIO range (64MB): 0x20000000 ~ 0x24000000 */
+/* Some devices still need io ranges, reserve 16MB for compatibility */
+static const struct mtk_pcie_mmio_res pcie_mmio_res_io = {
+ .cpu_addr = 0x20000000,
+ .pci_addr = 0x20000000,
+ .size = 16 * MiB,
+ .type = IORESOURCE_IO,
+};
+
+static const struct mtk_pcie_mmio_res pcie_mmio_res_mem = {
+ .cpu_addr = 0x21000000,
+ .pci_addr = 0x21000000,
+ .size = 48 * MiB,
+ .type = IORESOURCE_MEM,
+};
+
+struct pad_func {
+ gpio_t gpio;
+ u8 func;
+};
+
+#define PAD_FUNC(name, func) {GPIO(name), PAD_##name##_FUNC_##func}
+
+static const struct pad_func pcie_pins[2][3] = {
+ {
+ PAD_FUNC(PCIE_WAKE_N, WAKEN),
+ PAD_FUNC(PCIE_PERESET_N, PERSTN),
+ PAD_FUNC(PCIE_CLKREQ_N, CLKREQN),
+ },
+ {
+ PAD_FUNC(CMMCLK0, PERSTN_1),
+ PAD_FUNC(CMMCLK1, CLKREQN_1),
+ PAD_FUNC(CMMCLK2, WAKEN_1),
+ },
+};
+
+static void mtk_pcie_set_pinmux(uint8_t port)
+{
+ const struct pad_func *pins = pcie_pins[port];
+ size_t i;
+
+ for (i = 0; i < ARRAY_SIZE(pcie_pins[port]); i++) {
+ gpio_set_mode(pins[i].gpio, pins[i].func);
+ gpio_set_pull(pins[i].gpio, GPIO_PULL_ENABLE, GPIO_PULL_UP);
+ }
+}
+
+static void mtk_pcie_reset(uintptr_t reg, bool enable)
+{
+ uint32_t val;
+
+ val = read32p(reg);
+
+ if (enable)
+ val |= PCIE_MAC_RSTB | PCIE_PHY_RSTB | PCIE_BRG_RSTB |
+ PCIE_PE_RSTB;
+ else
+ val &= ~(PCIE_MAC_RSTB | PCIE_PHY_RSTB | PCIE_BRG_RSTB |
+ PCIE_PE_RSTB);
+
+ write32p(reg, val);
+}
+
+void mtk_pcie_pre_init(void)
+{
+ mtk_pcie_set_pinmux(0);
+
+ /* Assert all reset signals at early stage */
+ mtk_pcie_reset(PCIE_RST_CTRL_REG, true);
+}
+
+void mtk_pcie_get_hw_info(struct mtk_pcie_controller *ctrl)
+{
+ ctrl->base = PCIE_REG_BASE_PORT0;
+ ctrl->mmio_res_io = &pcie_mmio_res_io;
+ ctrl->mmio_res_mem = &pcie_mmio_res_mem;
+ ctrl->reset = &mtk_pcie_reset;
+}