Commit 42c03844 authored by Sean Wang's avatar Sean Wang Committed by David S. Miller

net-next: mediatek: add support for MediaTek MT7622 SoC

This patch adds the driver for ethernet controller on MT7622 SoC. It has
the similar handling logic as the previously MT7623 does, but there are
additions against with MT7623 SoC, the shared SGMII given for the dual
GMACs and including 5-ports 10/100 embedded switch support (ESW) as the
GMAC1 option, thus more clocks consumers for the extra feature are
introduced here. So for ease portability and maintenance, those
differences all are being kept inside the platform data as other drivers
usually do. Currently testing successfully is done with those patches for
the conditions such as GMAC2 with IP1001 PHY via RGMII and GMAC1/2 with
RTL8211F PHY via SGMII.
Signed-off-by: default avatarSean Wang <sean.wang@mediatek.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 2ec50f57
...@@ -53,7 +53,8 @@ static const struct mtk_ethtool_stats { ...@@ -53,7 +53,8 @@ static const struct mtk_ethtool_stats {
}; };
static const char * const mtk_clks_source_name[] = { static const char * const mtk_clks_source_name[] = {
"ethif", "esw", "gp1", "gp2", "trgpll" "ethif", "esw", "gp0", "gp1", "gp2", "trgpll", "sgmii_tx250m",
"sgmii_rx250m", "sgmii_cdr_ref", "sgmii_cdr_fb", "sgmii_ck", "eth2pll"
}; };
void mtk_w32(struct mtk_eth *eth, u32 val, unsigned reg) void mtk_w32(struct mtk_eth *eth, u32 val, unsigned reg)
...@@ -163,6 +164,47 @@ static void mtk_gmac0_rgmii_adjust(struct mtk_eth *eth, int speed) ...@@ -163,6 +164,47 @@ static void mtk_gmac0_rgmii_adjust(struct mtk_eth *eth, int speed)
mtk_w32(eth, val, TRGMII_TCK_CTRL); mtk_w32(eth, val, TRGMII_TCK_CTRL);
} }
static void mtk_gmac_sgmii_hw_setup(struct mtk_eth *eth, int mac_id)
{
u32 val;
/* Setup the link timer and QPHY power up inside SGMIISYS */
regmap_write(eth->sgmiisys, SGMSYS_PCS_LINK_TIMER,
SGMII_LINK_TIMER_DEFAULT);
regmap_read(eth->sgmiisys, SGMSYS_SGMII_MODE, &val);
val |= SGMII_REMOTE_FAULT_DIS;
regmap_write(eth->sgmiisys, SGMSYS_SGMII_MODE, val);
regmap_read(eth->sgmiisys, SGMSYS_PCS_CONTROL_1, &val);
val |= SGMII_AN_RESTART;
regmap_write(eth->sgmiisys, SGMSYS_PCS_CONTROL_1, val);
regmap_read(eth->sgmiisys, SGMSYS_QPHY_PWR_STATE_CTRL, &val);
val &= ~SGMII_PHYA_PWD;
regmap_write(eth->sgmiisys, SGMSYS_QPHY_PWR_STATE_CTRL, val);
/* Determine MUX for which GMAC uses the SGMII interface */
if (MTK_HAS_CAPS(eth->soc->caps, MTK_DUAL_GMAC_SHARED_SGMII)) {
regmap_read(eth->ethsys, ETHSYS_SYSCFG0, &val);
val &= ~SYSCFG0_SGMII_MASK;
val |= !mac_id ? SYSCFG0_SGMII_GMAC1 : SYSCFG0_SGMII_GMAC2;
regmap_write(eth->ethsys, ETHSYS_SYSCFG0, val);
dev_info(eth->dev, "setup shared sgmii for gmac=%d\n",
mac_id);
}
/* Setup the GMAC1 going through SGMII path when SoC also support
* ESW on GMAC1
*/
if (MTK_HAS_CAPS(eth->soc->caps, MTK_GMAC1_ESW | MTK_GMAC1_SGMII) &&
!mac_id) {
mtk_w32(eth, 0, MTK_MAC_MISC);
dev_info(eth->dev, "setup gmac1 going through sgmii");
}
}
static void mtk_phy_link_adjust(struct net_device *dev) static void mtk_phy_link_adjust(struct net_device *dev)
{ {
struct mtk_mac *mac = netdev_priv(dev); struct mtk_mac *mac = netdev_priv(dev);
...@@ -270,6 +312,7 @@ static int mtk_phy_connect(struct net_device *dev) ...@@ -270,6 +312,7 @@ static int mtk_phy_connect(struct net_device *dev)
if (!np) if (!np)
return -ENODEV; return -ENODEV;
mac->ge_mode = 0;
switch (of_get_phy_mode(np)) { switch (of_get_phy_mode(np)) {
case PHY_INTERFACE_MODE_TRGMII: case PHY_INTERFACE_MODE_TRGMII:
mac->trgmii = true; mac->trgmii = true;
...@@ -277,7 +320,10 @@ static int mtk_phy_connect(struct net_device *dev) ...@@ -277,7 +320,10 @@ static int mtk_phy_connect(struct net_device *dev)
case PHY_INTERFACE_MODE_RGMII_RXID: case PHY_INTERFACE_MODE_RGMII_RXID:
case PHY_INTERFACE_MODE_RGMII_ID: case PHY_INTERFACE_MODE_RGMII_ID:
case PHY_INTERFACE_MODE_RGMII: case PHY_INTERFACE_MODE_RGMII:
mac->ge_mode = 0; break;
case PHY_INTERFACE_MODE_SGMII:
if (MTK_HAS_CAPS(eth->soc->caps, MTK_SGMII))
mtk_gmac_sgmii_hw_setup(eth, mac->id);
break; break;
case PHY_INTERFACE_MODE_MII: case PHY_INTERFACE_MODE_MII:
mac->ge_mode = 1; mac->ge_mode = 1;
...@@ -2423,6 +2469,7 @@ static int mtk_get_chip_id(struct mtk_eth *eth, u32 *chip_id) ...@@ -2423,6 +2469,7 @@ static int mtk_get_chip_id(struct mtk_eth *eth, u32 *chip_id)
static bool mtk_is_hwlro_supported(struct mtk_eth *eth) static bool mtk_is_hwlro_supported(struct mtk_eth *eth)
{ {
switch (eth->chip_id) { switch (eth->chip_id) {
case MT7622_ETH:
case MT7623_ETH: case MT7623_ETH:
return true; return true;
} }
...@@ -2462,6 +2509,16 @@ static int mtk_probe(struct platform_device *pdev) ...@@ -2462,6 +2509,16 @@ static int mtk_probe(struct platform_device *pdev)
return PTR_ERR(eth->ethsys); return PTR_ERR(eth->ethsys);
} }
if (MTK_HAS_CAPS(eth->soc->caps, MTK_SGMII)) {
eth->sgmiisys =
syscon_regmap_lookup_by_phandle(pdev->dev.of_node,
"mediatek,sgmiisys");
if (IS_ERR(eth->sgmiisys)) {
dev_err(&pdev->dev, "no sgmiisys regmap found\n");
return PTR_ERR(eth->sgmiisys);
}
}
eth->pctl = syscon_regmap_lookup_by_phandle(pdev->dev.of_node, eth->pctl = syscon_regmap_lookup_by_phandle(pdev->dev.of_node,
"mediatek,pctl"); "mediatek,pctl");
if (IS_ERR(eth->pctl)) { if (IS_ERR(eth->pctl)) {
...@@ -2595,6 +2652,11 @@ static const struct mtk_soc_data mt2701_data = { ...@@ -2595,6 +2652,11 @@ static const struct mtk_soc_data mt2701_data = {
.required_clks = MT7623_CLKS_BITMAP .required_clks = MT7623_CLKS_BITMAP
}; };
static const struct mtk_soc_data mt7622_data = {
.caps = MTK_DUAL_GMAC_SHARED_SGMII | MTK_GMAC1_ESW,
.required_clks = MT7622_CLKS_BITMAP
};
static const struct mtk_soc_data mt7623_data = { static const struct mtk_soc_data mt7623_data = {
.caps = MTK_GMAC1_TRGMII, .caps = MTK_GMAC1_TRGMII,
.required_clks = MT7623_CLKS_BITMAP .required_clks = MT7623_CLKS_BITMAP
...@@ -2602,6 +2664,7 @@ static const struct mtk_soc_data mt7623_data = { ...@@ -2602,6 +2664,7 @@ static const struct mtk_soc_data mt7623_data = {
const struct of_device_id of_mtk_match[] = { const struct of_device_id of_mtk_match[] = {
{ .compatible = "mediatek,mt2701-eth", .data = &mt2701_data}, { .compatible = "mediatek,mt2701-eth", .data = &mt2701_data},
{ .compatible = "mediatek,mt7622-eth", .data = &mt7622_data},
{ .compatible = "mediatek,mt7623-eth", .data = &mt7623_data}, { .compatible = "mediatek,mt7623-eth", .data = &mt7623_data},
{}, {},
}; };
......
...@@ -302,6 +302,9 @@ ...@@ -302,6 +302,9 @@
#define PHY_IAC_REG_SHIFT 25 #define PHY_IAC_REG_SHIFT 25
#define PHY_IAC_TIMEOUT HZ #define PHY_IAC_TIMEOUT HZ
#define MTK_MAC_MISC 0x1000c
#define MTK_MUX_TO_ESW BIT(0)
/* Mac control registers */ /* Mac control registers */
#define MTK_MAC_MCR(x) (0x10100 + (x * 0x100)) #define MTK_MAC_MCR(x) (0x10100 + (x * 0x100))
#define MAC_MCR_MAX_RX_1536 BIT(24) #define MAC_MCR_MAX_RX_1536 BIT(24)
...@@ -357,11 +360,15 @@ ...@@ -357,11 +360,15 @@
#define ETHSYS_CHIPID0_3 0x0 #define ETHSYS_CHIPID0_3 0x0
#define ETHSYS_CHIPID4_7 0x4 #define ETHSYS_CHIPID4_7 0x4
#define MT7623_ETH 7623 #define MT7623_ETH 7623
#define MT7622_ETH 7622
/* ethernet subsystem config register */ /* ethernet subsystem config register */
#define ETHSYS_SYSCFG0 0x14 #define ETHSYS_SYSCFG0 0x14
#define SYSCFG0_GE_MASK 0x3 #define SYSCFG0_GE_MASK 0x3
#define SYSCFG0_GE_MODE(x, y) (x << (12 + (y * 2))) #define SYSCFG0_GE_MODE(x, y) (x << (12 + (y * 2)))
#define SYSCFG0_SGMII_MASK (3 << 8)
#define SYSCFG0_SGMII_GMAC1 ((2 << 8) & GENMASK(9, 8))
#define SYSCFG0_SGMII_GMAC2 ((3 << 8) & GENMASK(9, 8))
/* ethernet subsystem clock register */ /* ethernet subsystem clock register */
#define ETHSYS_CLKCFG0 0x2c #define ETHSYS_CLKCFG0 0x2c
...@@ -372,6 +379,23 @@ ...@@ -372,6 +379,23 @@
#define RSTCTRL_FE BIT(6) #define RSTCTRL_FE BIT(6)
#define RSTCTRL_PPE BIT(31) #define RSTCTRL_PPE BIT(31)
/* SGMII subsystem config registers */
/* Register to auto-negotiation restart */
#define SGMSYS_PCS_CONTROL_1 0x0
#define SGMII_AN_RESTART BIT(9)
/* Register to programmable link timer, the unit in 2 * 8ns */
#define SGMSYS_PCS_LINK_TIMER 0x18
#define SGMII_LINK_TIMER_DEFAULT (0x186a0 & GENMASK(19, 0))
/* Register to control remote fault */
#define SGMSYS_SGMII_MODE 0x20
#define SGMII_REMOTE_FAULT_DIS BIT(8)
/* Register to power up QPHY */
#define SGMSYS_QPHY_PWR_STATE_CTRL 0xe8
#define SGMII_PHYA_PWD BIT(4)
struct mtk_rx_dma { struct mtk_rx_dma {
unsigned int rxd1; unsigned int rxd1;
unsigned int rxd2; unsigned int rxd2;
...@@ -437,15 +461,31 @@ enum mtk_tx_flags { ...@@ -437,15 +461,31 @@ enum mtk_tx_flags {
enum mtk_clks_map { enum mtk_clks_map {
MTK_CLK_ETHIF, MTK_CLK_ETHIF,
MTK_CLK_ESW, MTK_CLK_ESW,
MTK_CLK_GP0,
MTK_CLK_GP1, MTK_CLK_GP1,
MTK_CLK_GP2, MTK_CLK_GP2,
MTK_CLK_TRGPLL, MTK_CLK_TRGPLL,
MTK_CLK_SGMII_TX_250M,
MTK_CLK_SGMII_RX_250M,
MTK_CLK_SGMII_CDR_REF,
MTK_CLK_SGMII_CDR_FB,
MTK_CLK_SGMII_CK,
MTK_CLK_ETH2PLL,
MTK_CLK_MAX MTK_CLK_MAX
}; };
#define MT7623_CLKS_BITMAP (BIT(MTK_CLK_ETHIF) | BIT(MTK_CLK_ESW) | \ #define MT7623_CLKS_BITMAP (BIT(MTK_CLK_ETHIF) | BIT(MTK_CLK_ESW) | \
BIT(MTK_CLK_GP1) | BIT(MTK_CLK_GP2) | \ BIT(MTK_CLK_GP1) | BIT(MTK_CLK_GP2) | \
BIT(MTK_CLK_TRGPLL)) BIT(MTK_CLK_TRGPLL))
#define MT7622_CLKS_BITMAP (BIT(MTK_CLK_ETHIF) | BIT(MTK_CLK_ESW) | \
BIT(MTK_CLK_GP0) | BIT(MTK_CLK_GP1) | \
BIT(MTK_CLK_GP2) | \
BIT(MTK_CLK_SGMII_TX_250M) | \
BIT(MTK_CLK_SGMII_RX_250M) | \
BIT(MTK_CLK_SGMII_CDR_REF) | \
BIT(MTK_CLK_SGMII_CDR_FB) | \
BIT(MTK_CLK_SGMII_CK) | \
BIT(MTK_CLK_ETH2PLL))
enum mtk_dev_state { enum mtk_dev_state {
MTK_HW_INIT, MTK_HW_INIT,
MTK_RESETTING MTK_RESETTING
...@@ -516,9 +556,16 @@ struct mtk_rx_ring { ...@@ -516,9 +556,16 @@ struct mtk_rx_ring {
#define MTK_TRGMII BIT(0) #define MTK_TRGMII BIT(0)
#define MTK_GMAC1_TRGMII (BIT(1) | MTK_TRGMII) #define MTK_GMAC1_TRGMII (BIT(1) | MTK_TRGMII)
#define MTK_ESW BIT(4)
#define MTK_GMAC1_ESW (BIT(5) | MTK_ESW)
#define MTK_SGMII BIT(8)
#define MTK_GMAC1_SGMII (BIT(9) | MTK_SGMII)
#define MTK_GMAC2_SGMII (BIT(10) | MTK_SGMII)
#define MTK_DUAL_GMAC_SHARED_SGMII (BIT(11) | MTK_GMAC1_SGMII | \
MTK_GMAC2_SGMII)
#define MTK_HAS_CAPS(caps, _x) (((caps) & (_x)) == (_x)) #define MTK_HAS_CAPS(caps, _x) (((caps) & (_x)) == (_x))
/* struct mtk_soc_data - This is the structure holding all differences /* struct mtk_eth_data - This is the structure holding all differences
* among various plaforms * among various plaforms
* @caps Flags shown the extra capability for the SoC * @caps Flags shown the extra capability for the SoC
* @required_clks Flags shown the bitmap for required clocks on * @required_clks Flags shown the bitmap for required clocks on
...@@ -547,6 +594,8 @@ struct mtk_soc_data { ...@@ -547,6 +594,8 @@ struct mtk_soc_data {
* @msg_enable: Ethtool msg level * @msg_enable: Ethtool msg level
* @ethsys: The register map pointing at the range used to setup * @ethsys: The register map pointing at the range used to setup
* MII modes * MII modes
* @sgmiisys: The register map pointing at the range used to setup
* SGMII modes
* @pctl: The register map pointing at the range used to setup * @pctl: The register map pointing at the range used to setup
* GMAC port drive/slew values * GMAC port drive/slew values
* @dma_refcnt: track how many netdevs are using the DMA engine * @dma_refcnt: track how many netdevs are using the DMA engine
...@@ -560,7 +609,7 @@ struct mtk_soc_data { ...@@ -560,7 +609,7 @@ struct mtk_soc_data {
* @clks: clock array for all clocks required * @clks: clock array for all clocks required
* @mii_bus: If there is a bus we need to create an instance for it * @mii_bus: If there is a bus we need to create an instance for it
* @pending_work: The workqueue used to reset the dma ring * @pending_work: The workqueue used to reset the dma ring
* @state Initialization and runtime state of the device * @state: Initialization and runtime state of the device
* @soc: Holding specific data among vaious SoCs * @soc: Holding specific data among vaious SoCs
*/ */
...@@ -577,6 +626,7 @@ struct mtk_eth { ...@@ -577,6 +626,7 @@ struct mtk_eth {
u32 msg_enable; u32 msg_enable;
unsigned long sysclk; unsigned long sysclk;
struct regmap *ethsys; struct regmap *ethsys;
struct regmap *sgmiisys;
struct regmap *pctl; struct regmap *pctl;
u32 chip_id; u32 chip_id;
bool hwlro; bool hwlro;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment