stmmac: dwmac-mediatek: add support for mt8195
authorBiao Huang <biao.huang@mediatek.com>
Mon, 14 Mar 2022 07:57:12 +0000 (15:57 +0800)
committerDavid S. Miller <davem@davemloft.net>
Wed, 16 Mar 2022 12:49:23 +0000 (12:49 +0000)
Add Ethernet support for MediaTek SoCs from the mt8195 family.

Signed-off-by: Biao Huang <biao.huang@mediatek.com>
Acked-by: AngeloGioacchino Del Regno <angelogioacchino.delregno@collabora.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
drivers/net/ethernet/stmicro/stmmac/dwmac-mediatek.c

index b2507a2ba3262c77d75b47ffe70c8a0a9f3affc4..6ff88df5876733dd6dddd746547af0d1c939b78b 100644 (file)
 #define ETH_FINE_DLY_GTXC      BIT(1)
 #define ETH_FINE_DLY_RXC       BIT(0)
 
+/* Peri Configuration register for mt8195 */
+#define MT8195_PERI_ETH_CTRL0          0xFD0
+#define MT8195_RMII_CLK_SRC_INTERNAL   BIT(28)
+#define MT8195_RMII_CLK_SRC_RXC                BIT(27)
+#define MT8195_ETH_INTF_SEL            GENMASK(26, 24)
+#define MT8195_RGMII_TXC_PHASE_CTRL    BIT(22)
+#define MT8195_EXT_PHY_MODE            BIT(21)
+#define MT8195_DLY_GTXC_INV            BIT(12)
+#define MT8195_DLY_GTXC_ENABLE         BIT(5)
+#define MT8195_DLY_GTXC_STAGES         GENMASK(4, 0)
+
+#define MT8195_PERI_ETH_CTRL1          0xFD4
+#define MT8195_DLY_RXC_INV             BIT(25)
+#define MT8195_DLY_RXC_ENABLE          BIT(18)
+#define MT8195_DLY_RXC_STAGES          GENMASK(17, 13)
+#define MT8195_DLY_TXC_INV             BIT(12)
+#define MT8195_DLY_TXC_ENABLE          BIT(5)
+#define MT8195_DLY_TXC_STAGES          GENMASK(4, 0)
+
+#define MT8195_PERI_ETH_CTRL2          0xFD8
+#define MT8195_DLY_RMII_RXC_INV                BIT(25)
+#define MT8195_DLY_RMII_RXC_ENABLE     BIT(18)
+#define MT8195_DLY_RMII_RXC_STAGES     GENMASK(17, 13)
+#define MT8195_DLY_RMII_TXC_INV                BIT(12)
+#define MT8195_DLY_RMII_TXC_ENABLE     BIT(5)
+#define MT8195_DLY_RMII_TXC_STAGES     GENMASK(4, 0)
+
 struct mac_delay_struct {
        u32 tx_delay;
        u32 rx_delay;
@@ -57,11 +84,13 @@ struct mediatek_dwmac_plat_data {
        phy_interface_t phy_mode;
        bool rmii_clk_from_mac;
        bool rmii_rxc;
+       bool mac_wol;
 };
 
 struct mediatek_dwmac_variant {
        int (*dwmac_set_phy_interface)(struct mediatek_dwmac_plat_data *plat);
        int (*dwmac_set_delay)(struct mediatek_dwmac_plat_data *plat);
+       void (*dwmac_fix_mac_speed)(void *priv, unsigned int speed);
 
        /* clock ids to be requested */
        const char * const *clk_list;
@@ -77,6 +106,10 @@ static const char * const mt2712_dwmac_clk_l[] = {
        "axi", "apb", "mac_main", "ptp_ref"
 };
 
+static const char * const mt8195_dwmac_clk_l[] = {
+       "axi", "apb", "mac_cg", "mac_main", "ptp_ref"
+};
+
 static int mt2712_set_interface(struct mediatek_dwmac_plat_data *plat)
 {
        int rmii_clk_from_mac = plat->rmii_clk_from_mac ? RMII_CLK_SRC_INTERNAL : 0;
@@ -256,6 +289,193 @@ static const struct mediatek_dwmac_variant mt2712_gmac_variant = {
                .tx_delay_max = 17600,
 };
 
+static int mt8195_set_interface(struct mediatek_dwmac_plat_data *plat)
+{
+       int rmii_clk_from_mac = plat->rmii_clk_from_mac ? MT8195_RMII_CLK_SRC_INTERNAL : 0;
+       int rmii_rxc = plat->rmii_rxc ? MT8195_RMII_CLK_SRC_RXC : 0;
+       u32 intf_val = 0;
+
+       /* select phy interface in top control domain */
+       switch (plat->phy_mode) {
+       case PHY_INTERFACE_MODE_MII:
+               intf_val |= FIELD_PREP(MT8195_ETH_INTF_SEL, PHY_INTF_MII);
+               break;
+       case PHY_INTERFACE_MODE_RMII:
+               intf_val |= (rmii_rxc | rmii_clk_from_mac);
+               intf_val |= FIELD_PREP(MT8195_ETH_INTF_SEL, PHY_INTF_RMII);
+               break;
+       case PHY_INTERFACE_MODE_RGMII:
+       case PHY_INTERFACE_MODE_RGMII_TXID:
+       case PHY_INTERFACE_MODE_RGMII_RXID:
+       case PHY_INTERFACE_MODE_RGMII_ID:
+               intf_val |= FIELD_PREP(MT8195_ETH_INTF_SEL, PHY_INTF_RGMII);
+               break;
+       default:
+               dev_err(plat->dev, "phy interface not supported\n");
+               return -EINVAL;
+       }
+
+       /* MT8195 only support external PHY */
+       intf_val |= MT8195_EXT_PHY_MODE;
+
+       regmap_write(plat->peri_regmap, MT8195_PERI_ETH_CTRL0, intf_val);
+
+       return 0;
+}
+
+static void mt8195_delay_ps2stage(struct mediatek_dwmac_plat_data *plat)
+{
+       struct mac_delay_struct *mac_delay = &plat->mac_delay;
+
+       /* 290ps per stage */
+       mac_delay->tx_delay /= 290;
+       mac_delay->rx_delay /= 290;
+}
+
+static void mt8195_delay_stage2ps(struct mediatek_dwmac_plat_data *plat)
+{
+       struct mac_delay_struct *mac_delay = &plat->mac_delay;
+
+       /* 290ps per stage */
+       mac_delay->tx_delay *= 290;
+       mac_delay->rx_delay *= 290;
+}
+
+static int mt8195_set_delay(struct mediatek_dwmac_plat_data *plat)
+{
+       struct mac_delay_struct *mac_delay = &plat->mac_delay;
+       u32 gtxc_delay_val = 0, delay_val = 0, rmii_delay_val = 0;
+
+       mt8195_delay_ps2stage(plat);
+
+       switch (plat->phy_mode) {
+       case PHY_INTERFACE_MODE_MII:
+               delay_val |= FIELD_PREP(MT8195_DLY_TXC_ENABLE, !!mac_delay->tx_delay);
+               delay_val |= FIELD_PREP(MT8195_DLY_TXC_STAGES, mac_delay->tx_delay);
+               delay_val |= FIELD_PREP(MT8195_DLY_TXC_INV, mac_delay->tx_inv);
+
+               delay_val |= FIELD_PREP(MT8195_DLY_RXC_ENABLE, !!mac_delay->rx_delay);
+               delay_val |= FIELD_PREP(MT8195_DLY_RXC_STAGES, mac_delay->rx_delay);
+               delay_val |= FIELD_PREP(MT8195_DLY_RXC_INV, mac_delay->rx_inv);
+               break;
+       case PHY_INTERFACE_MODE_RMII:
+               if (plat->rmii_clk_from_mac) {
+                       /* case 1: mac provides the rmii reference clock,
+                        * and the clock output to TXC pin.
+                        * The egress timing can be adjusted by RMII_TXC delay macro circuit.
+                        * The ingress timing can be adjusted by RMII_RXC delay macro circuit.
+                        */
+                       rmii_delay_val |= FIELD_PREP(MT8195_DLY_RMII_TXC_ENABLE,
+                                                    !!mac_delay->tx_delay);
+                       rmii_delay_val |= FIELD_PREP(MT8195_DLY_RMII_TXC_STAGES,
+                                                    mac_delay->tx_delay);
+                       rmii_delay_val |= FIELD_PREP(MT8195_DLY_RMII_TXC_INV,
+                                                    mac_delay->tx_inv);
+
+                       rmii_delay_val |= FIELD_PREP(MT8195_DLY_RMII_RXC_ENABLE,
+                                                    !!mac_delay->rx_delay);
+                       rmii_delay_val |= FIELD_PREP(MT8195_DLY_RMII_RXC_STAGES,
+                                                    mac_delay->rx_delay);
+                       rmii_delay_val |= FIELD_PREP(MT8195_DLY_RMII_RXC_INV,
+                                                    mac_delay->rx_inv);
+               } else {
+                       /* case 2: the rmii reference clock is from external phy,
+                        * and the property "rmii_rxc" indicates which pin(TXC/RXC)
+                        * the reference clk is connected to. The reference clock is a
+                        * received signal, so rx_delay/rx_inv are used to indicate
+                        * the reference clock timing adjustment
+                        */
+                       if (plat->rmii_rxc) {
+                               /* the rmii reference clock from outside is connected
+                                * to RXC pin, the reference clock will be adjusted
+                                * by RXC delay macro circuit.
+                                */
+                               delay_val |= FIELD_PREP(MT8195_DLY_RXC_ENABLE,
+                                                       !!mac_delay->rx_delay);
+                               delay_val |= FIELD_PREP(MT8195_DLY_RXC_STAGES,
+                                                       mac_delay->rx_delay);
+                               delay_val |= FIELD_PREP(MT8195_DLY_RXC_INV,
+                                                       mac_delay->rx_inv);
+                       } else {
+                               /* the rmii reference clock from outside is connected
+                                * to TXC pin, the reference clock will be adjusted
+                                * by TXC delay macro circuit.
+                                */
+                               delay_val |= FIELD_PREP(MT8195_DLY_TXC_ENABLE,
+                                                       !!mac_delay->rx_delay);
+                               delay_val |= FIELD_PREP(MT8195_DLY_TXC_STAGES,
+                                                       mac_delay->rx_delay);
+                               delay_val |= FIELD_PREP(MT8195_DLY_TXC_INV,
+                                                       mac_delay->rx_inv);
+                       }
+               }
+               break;
+       case PHY_INTERFACE_MODE_RGMII:
+       case PHY_INTERFACE_MODE_RGMII_TXID:
+       case PHY_INTERFACE_MODE_RGMII_RXID:
+       case PHY_INTERFACE_MODE_RGMII_ID:
+               gtxc_delay_val |= FIELD_PREP(MT8195_DLY_GTXC_ENABLE, !!mac_delay->tx_delay);
+               gtxc_delay_val |= FIELD_PREP(MT8195_DLY_GTXC_STAGES, mac_delay->tx_delay);
+               gtxc_delay_val |= FIELD_PREP(MT8195_DLY_GTXC_INV, mac_delay->tx_inv);
+
+               delay_val |= FIELD_PREP(MT8195_DLY_RXC_ENABLE, !!mac_delay->rx_delay);
+               delay_val |= FIELD_PREP(MT8195_DLY_RXC_STAGES, mac_delay->rx_delay);
+               delay_val |= FIELD_PREP(MT8195_DLY_RXC_INV, mac_delay->rx_inv);
+
+               break;
+       default:
+               dev_err(plat->dev, "phy interface not supported\n");
+               return -EINVAL;
+       }
+
+       regmap_update_bits(plat->peri_regmap,
+                          MT8195_PERI_ETH_CTRL0,
+                          MT8195_RGMII_TXC_PHASE_CTRL |
+                          MT8195_DLY_GTXC_INV |
+                          MT8195_DLY_GTXC_ENABLE |
+                          MT8195_DLY_GTXC_STAGES,
+                          gtxc_delay_val);
+       regmap_write(plat->peri_regmap, MT8195_PERI_ETH_CTRL1, delay_val);
+       regmap_write(plat->peri_regmap, MT8195_PERI_ETH_CTRL2, rmii_delay_val);
+
+       mt8195_delay_stage2ps(plat);
+
+       return 0;
+}
+
+static void mt8195_fix_mac_speed(void *priv, unsigned int speed)
+{
+       struct mediatek_dwmac_plat_data *priv_plat = priv;
+
+       if ((phy_interface_mode_is_rgmii(priv_plat->phy_mode))) {
+               /* prefer 2ns fixed delay which is controlled by TXC_PHASE_CTRL,
+                * when link speed is 1Gbps with RGMII interface,
+                * Fall back to delay macro circuit for 10/100Mbps link speed.
+                */
+               if (speed == SPEED_1000)
+                       regmap_update_bits(priv_plat->peri_regmap,
+                                          MT8195_PERI_ETH_CTRL0,
+                                          MT8195_RGMII_TXC_PHASE_CTRL |
+                                          MT8195_DLY_GTXC_ENABLE |
+                                          MT8195_DLY_GTXC_INV |
+                                          MT8195_DLY_GTXC_STAGES,
+                                          MT8195_RGMII_TXC_PHASE_CTRL);
+               else
+                       mt8195_set_delay(priv_plat);
+       }
+}
+
+static const struct mediatek_dwmac_variant mt8195_gmac_variant = {
+       .dwmac_set_phy_interface = mt8195_set_interface,
+       .dwmac_set_delay = mt8195_set_delay,
+       .dwmac_fix_mac_speed = mt8195_fix_mac_speed,
+       .clk_list = mt8195_dwmac_clk_l,
+       .num_clks = ARRAY_SIZE(mt8195_dwmac_clk_l),
+       .dma_bit_mask = 35,
+       .rx_delay_max = 9280,
+       .tx_delay_max = 9280,
+};
+
 static int mediatek_dwmac_config_dt(struct mediatek_dwmac_plat_data *plat)
 {
        struct mac_delay_struct *mac_delay = &plat->mac_delay;
@@ -296,6 +516,7 @@ static int mediatek_dwmac_config_dt(struct mediatek_dwmac_plat_data *plat)
        mac_delay->rx_inv = of_property_read_bool(plat->np, "mediatek,rxc-inverse");
        plat->rmii_rxc = of_property_read_bool(plat->np, "mediatek,rmii-rxc");
        plat->rmii_clk_from_mac = of_property_read_bool(plat->np, "mediatek,rmii-clk-from-mac");
+       plat->mac_wol = of_property_read_bool(plat->np, "mediatek,mac-wol");
 
        return 0;
 }
@@ -408,6 +629,50 @@ static int mediatek_dwmac_clks_config(void *priv, bool enabled)
 
        return ret;
 }
+
+static int mediatek_dwmac_common_data(struct platform_device *pdev,
+                                     struct plat_stmmacenet_data *plat,
+                                     struct mediatek_dwmac_plat_data *priv_plat)
+{
+       int i;
+
+       plat->interface = priv_plat->phy_mode;
+       plat->use_phy_wol = priv_plat->mac_wol ? 0 : 1;
+       plat->riwt_off = 1;
+       plat->maxmtu = ETH_DATA_LEN;
+       plat->addr64 = priv_plat->variant->dma_bit_mask;
+       plat->bsp_priv = priv_plat;
+       plat->init = mediatek_dwmac_init;
+       plat->exit = mediatek_dwmac_exit;
+       plat->clks_config = mediatek_dwmac_clks_config;
+       if (priv_plat->variant->dwmac_fix_mac_speed)
+               plat->fix_mac_speed = priv_plat->variant->dwmac_fix_mac_speed;
+
+       plat->safety_feat_cfg = devm_kzalloc(&pdev->dev,
+                                            sizeof(*plat->safety_feat_cfg),
+                                            GFP_KERNEL);
+       if (!plat->safety_feat_cfg)
+               return -ENOMEM;
+
+       plat->safety_feat_cfg->tsoee = 1;
+       plat->safety_feat_cfg->mrxpee = 0;
+       plat->safety_feat_cfg->mestee = 1;
+       plat->safety_feat_cfg->mrxee = 1;
+       plat->safety_feat_cfg->mtxee = 1;
+       plat->safety_feat_cfg->epsi = 0;
+       plat->safety_feat_cfg->edpp = 1;
+       plat->safety_feat_cfg->prtyen = 1;
+       plat->safety_feat_cfg->tmouten = 1;
+
+       for (i = 0; i < plat->tx_queues_to_use; i++) {
+               /* Default TX Q0 to use TSO and rest TXQ for TBS */
+               if (i > 0)
+                       plat->tx_queues_cfg[i].tbs_en = 1;
+       }
+
+       return 0;
+}
+
 static int mediatek_dwmac_probe(struct platform_device *pdev)
 {
        struct mediatek_dwmac_plat_data *priv_plat;
@@ -444,16 +709,7 @@ static int mediatek_dwmac_probe(struct platform_device *pdev)
        if (IS_ERR(plat_dat))
                return PTR_ERR(plat_dat);
 
-       plat_dat->interface = priv_plat->phy_mode;
-       plat_dat->use_phy_wol = 1;
-       plat_dat->riwt_off = 1;
-       plat_dat->maxmtu = ETH_DATA_LEN;
-       plat_dat->addr64 = priv_plat->variant->dma_bit_mask;
-       plat_dat->bsp_priv = priv_plat;
-       plat_dat->init = mediatek_dwmac_init;
-       plat_dat->exit = mediatek_dwmac_exit;
-       plat_dat->clks_config = mediatek_dwmac_clks_config;
-
+       mediatek_dwmac_common_data(pdev, plat_dat, priv_plat);
        mediatek_dwmac_init(pdev, priv_plat);
 
        ret = stmmac_dvr_probe(&pdev->dev, plat_dat, &stmmac_res);
@@ -468,6 +724,8 @@ static int mediatek_dwmac_probe(struct platform_device *pdev)
 static const struct of_device_id mediatek_dwmac_match[] = {
        { .compatible = "mediatek,mt2712-gmac",
          .data = &mt2712_gmac_variant },
+       { .compatible = "mediatek,mt8195-gmac",
+         .data = &mt8195_gmac_variant },
        { }
 };