mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/chenhuacai/linux-loongson
synced 2025-09-07 22:23:00 +00:00
stmmac: dwmac-mediatek: add support for mt8195
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>
This commit is contained in:
parent
150b6adda6
commit
f2d356a6ab
@ -39,6 +39,33 @@
|
|||||||
#define ETH_FINE_DLY_GTXC BIT(1)
|
#define ETH_FINE_DLY_GTXC BIT(1)
|
||||||
#define ETH_FINE_DLY_RXC BIT(0)
|
#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 {
|
struct mac_delay_struct {
|
||||||
u32 tx_delay;
|
u32 tx_delay;
|
||||||
u32 rx_delay;
|
u32 rx_delay;
|
||||||
@ -57,11 +84,13 @@ struct mediatek_dwmac_plat_data {
|
|||||||
phy_interface_t phy_mode;
|
phy_interface_t phy_mode;
|
||||||
bool rmii_clk_from_mac;
|
bool rmii_clk_from_mac;
|
||||||
bool rmii_rxc;
|
bool rmii_rxc;
|
||||||
|
bool mac_wol;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct mediatek_dwmac_variant {
|
struct mediatek_dwmac_variant {
|
||||||
int (*dwmac_set_phy_interface)(struct mediatek_dwmac_plat_data *plat);
|
int (*dwmac_set_phy_interface)(struct mediatek_dwmac_plat_data *plat);
|
||||||
int (*dwmac_set_delay)(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 */
|
/* clock ids to be requested */
|
||||||
const char * const *clk_list;
|
const char * const *clk_list;
|
||||||
@ -77,6 +106,10 @@ static const char * const mt2712_dwmac_clk_l[] = {
|
|||||||
"axi", "apb", "mac_main", "ptp_ref"
|
"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)
|
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;
|
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,
|
.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)
|
static int mediatek_dwmac_config_dt(struct mediatek_dwmac_plat_data *plat)
|
||||||
{
|
{
|
||||||
struct mac_delay_struct *mac_delay = &plat->mac_delay;
|
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");
|
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_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->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;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -408,6 +629,50 @@ static int mediatek_dwmac_clks_config(void *priv, bool enabled)
|
|||||||
|
|
||||||
return ret;
|
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)
|
static int mediatek_dwmac_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct mediatek_dwmac_plat_data *priv_plat;
|
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))
|
if (IS_ERR(plat_dat))
|
||||||
return PTR_ERR(plat_dat);
|
return PTR_ERR(plat_dat);
|
||||||
|
|
||||||
plat_dat->interface = priv_plat->phy_mode;
|
mediatek_dwmac_common_data(pdev, plat_dat, priv_plat);
|
||||||
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_init(pdev, priv_plat);
|
mediatek_dwmac_init(pdev, priv_plat);
|
||||||
|
|
||||||
ret = stmmac_dvr_probe(&pdev->dev, plat_dat, &stmmac_res);
|
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[] = {
|
static const struct of_device_id mediatek_dwmac_match[] = {
|
||||||
{ .compatible = "mediatek,mt2712-gmac",
|
{ .compatible = "mediatek,mt2712-gmac",
|
||||||
.data = &mt2712_gmac_variant },
|
.data = &mt2712_gmac_variant },
|
||||||
|
{ .compatible = "mediatek,mt8195-gmac",
|
||||||
|
.data = &mt8195_gmac_variant },
|
||||||
{ }
|
{ }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user