#define PCIE_CONF_LINK2_CTL_STS                (PCIE_CFG_OFFSET_ADDR + 0xb0)
 #define PCIE_CONF_LINK2_LCR2_LINK_SPEED        GENMASK(3, 0)
 
+enum mtk_gen3_pcie_flags {
+       SKIP_PCIE_RSTB  = BIT(0), /* Skip PERST# assertion during device
+                                  * probing or suspend/resume phase to
+                                  * avoid hw bugs/issues.
+                                  */
+};
+
 /**
  * struct mtk_gen3_pcie_pdata - differentiate between host generations
  * @power_up: pcie power_up callback
  * @phy_resets: phy reset lines SoC data.
+ * @flags: pcie device flags.
  */
 struct mtk_gen3_pcie_pdata {
        int (*power_up)(struct mtk_gen3_pcie *pcie);
                const char *id[MAX_NUM_PHY_RESETS];
                int num_resets;
        } phy_resets;
+       u32 flags;
 };
 
 /**
        val |= PCIE_DISABLE_DVFSRC_VLT_REQ;
        writel_relaxed(val, pcie->base + PCIE_MISC_CTRL_REG);
 
-       /* Assert all reset signals */
-       val = readl_relaxed(pcie->base + PCIE_RST_CTRL_REG);
-       val |= PCIE_MAC_RSTB | PCIE_PHY_RSTB | PCIE_BRG_RSTB | PCIE_PE_RSTB;
-       writel_relaxed(val, pcie->base + PCIE_RST_CTRL_REG);
-
        /*
-        * Described in PCIe CEM specification sections 2.2 (PERST# Signal)
-        * and 2.2.1 (Initial Power-Up (G3 to S0)).
-        * The deassertion of PERST# should be delayed 100ms (TPVPERL)
-        * for the power and clock to become stable.
+        * Airoha EN7581 has a hw bug asserting/releasing PCIE_PE_RSTB signal
+        * causing occasional PCIe link down. In order to overcome the issue,
+        * PCIE_RSTB signals are not asserted/released at this stage and the
+        * PCIe block is reset using en7523_reset_assert() and
+        * en7581_pci_enable().
         */
-       msleep(100);
-
-       /* De-assert reset signals */
-       val &= ~(PCIE_MAC_RSTB | PCIE_PHY_RSTB | PCIE_BRG_RSTB | PCIE_PE_RSTB);
-       writel_relaxed(val, pcie->base + PCIE_RST_CTRL_REG);
+       if (!(pcie->soc->flags & SKIP_PCIE_RSTB)) {
+               /* Assert all reset signals */
+               val = readl_relaxed(pcie->base + PCIE_RST_CTRL_REG);
+               val |= PCIE_MAC_RSTB | PCIE_PHY_RSTB | PCIE_BRG_RSTB |
+                      PCIE_PE_RSTB;
+               writel_relaxed(val, pcie->base + PCIE_RST_CTRL_REG);
+
+               /*
+                * Described in PCIe CEM specification revision 6.0.
+                *
+                * The deassertion of PERST# should be delayed 100ms (TPVPERL)
+                * for the power and clock to become stable.
+                */
+               msleep(PCIE_T_PVPERL_MS);
+
+               /* De-assert reset signals */
+               val &= ~(PCIE_MAC_RSTB | PCIE_PHY_RSTB | PCIE_BRG_RSTB |
+                        PCIE_PE_RSTB);
+               writel_relaxed(val, pcie->base + PCIE_RST_CTRL_REG);
+       }
 
        /* Check if the link is up or not */
        err = readl_poll_timeout(pcie->base + PCIE_LINK_STATUS_REG, val,
                return err;
        }
 
-       /* Pull down the PERST# pin */
-       val = readl_relaxed(pcie->base + PCIE_RST_CTRL_REG);
-       val |= PCIE_PE_RSTB;
-       writel_relaxed(val, pcie->base + PCIE_RST_CTRL_REG);
+       if (!(pcie->soc->flags & SKIP_PCIE_RSTB)) {
+               /* Assert the PERST# pin */
+               val = readl_relaxed(pcie->base + PCIE_RST_CTRL_REG);
+               val |= PCIE_PE_RSTB;
+               writel_relaxed(val, pcie->base + PCIE_RST_CTRL_REG);
+       }
 
        dev_dbg(pcie->dev, "entered L2 states successfully");
 
                .id[2] = "phy-lane2",
                .num_resets = 3,
        },
+       .flags = SKIP_PCIE_RSTB,
 };
 
 static const struct of_device_id mtk_pcie_of_match[] = {