Commit bcac735c authored by Serge Semin's avatar Serge Semin Committed by David S. Miller

net: pcs: xpcs: Introduce DW XPCS info structure

The being introduced structure will preserve the PCS and PMA IDs retrieved
from the respective DW XPCS MMDs or potentially pre-defined by the client
drivers. (The later change will be introduced later in the framework of
the commit adding the memory-mapped DW XPCS devices support.)

The structure fields are filled in in the xpcs_get_id() function, which
used to be responsible for the PCS Device ID getting only. Besides of the
PCS ID the method now fetches the PMA/PMD IDs too from MMD 1, which used
to be done in xpcs_dev_flag(). The retrieved PMA ID will be from now
utilized for the PMA-specific tweaks like it was introduced for the
Wangxun TxGBE PCS in the commit f629acc6 ("net: pcs: xpcs: support to
switch mode for Wangxun NICs").

Note 1. The xpcs_get_id() error-handling semantics has been changed. From
now the error number will be returned from the function. There is no point
in the next IOs or saving 0xffs and then looping over the actual device
IDs if device couldn't be reached. -ENODEV will be returned if the very
first IO operation failed thus indicating that no device could be found.

Note 2. The PCS and PMA IDs macros have been converted to enum'es. The
enum'es will be populated later in another commit with the virtual IDs
identifying the DW XPCS devices which have some platform-specifics, but
have been synthesized with the default PCS/PMA ID.
Signed-off-by: default avatarSerge Semin <fancer.lancer@gmail.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 410232ab
...@@ -237,29 +237,6 @@ int xpcs_write_vpcs(struct dw_xpcs *xpcs, int reg, u16 val) ...@@ -237,29 +237,6 @@ int xpcs_write_vpcs(struct dw_xpcs *xpcs, int reg, u16 val)
return xpcs_write_vendor(xpcs, MDIO_MMD_PCS, reg, val); return xpcs_write_vendor(xpcs, MDIO_MMD_PCS, reg, val);
} }
static int xpcs_dev_flag(struct dw_xpcs *xpcs)
{
int ret, oui;
ret = xpcs_read(xpcs, MDIO_MMD_PMAPMD, MDIO_DEVID1);
if (ret < 0)
return ret;
oui = ret;
ret = xpcs_read(xpcs, MDIO_MMD_PMAPMD, MDIO_DEVID2);
if (ret < 0)
return ret;
ret = (ret >> 10) & 0x3F;
oui |= ret << 16;
if (oui == DW_OUI_WX)
xpcs->dev_flag = DW_DEV_TXGBE;
return 0;
}
static int xpcs_poll_reset(struct dw_xpcs *xpcs, int dev) static int xpcs_poll_reset(struct dw_xpcs *xpcs, int dev)
{ {
/* Poll until the reset bit clears (50ms per retry == 0.6 sec) */ /* Poll until the reset bit clears (50ms per retry == 0.6 sec) */
...@@ -684,7 +661,7 @@ static int xpcs_config_aneg_c37_sgmii(struct dw_xpcs *xpcs, ...@@ -684,7 +661,7 @@ static int xpcs_config_aneg_c37_sgmii(struct dw_xpcs *xpcs,
{ {
int ret, mdio_ctrl, tx_conf; int ret, mdio_ctrl, tx_conf;
if (xpcs->dev_flag == DW_DEV_TXGBE) if (xpcs->info.pma == WX_TXGBE_XPCS_PMA_10G_ID)
xpcs_write_vpcs(xpcs, DW_VR_XS_PCS_DIG_CTRL1, DW_CL37_BP | DW_EN_VSMMD1); xpcs_write_vpcs(xpcs, DW_VR_XS_PCS_DIG_CTRL1, DW_CL37_BP | DW_EN_VSMMD1);
/* For AN for C37 SGMII mode, the settings are :- /* For AN for C37 SGMII mode, the settings are :-
...@@ -722,7 +699,7 @@ static int xpcs_config_aneg_c37_sgmii(struct dw_xpcs *xpcs, ...@@ -722,7 +699,7 @@ static int xpcs_config_aneg_c37_sgmii(struct dw_xpcs *xpcs,
ret |= (DW_VR_MII_PCS_MODE_C37_SGMII << ret |= (DW_VR_MII_PCS_MODE_C37_SGMII <<
DW_VR_MII_AN_CTRL_PCS_MODE_SHIFT & DW_VR_MII_AN_CTRL_PCS_MODE_SHIFT &
DW_VR_MII_PCS_MODE_MASK); DW_VR_MII_PCS_MODE_MASK);
if (xpcs->dev_flag == DW_DEV_TXGBE) { if (xpcs->info.pma == WX_TXGBE_XPCS_PMA_10G_ID) {
ret |= DW_VR_MII_AN_CTRL_8BIT; ret |= DW_VR_MII_AN_CTRL_8BIT;
/* Hardware requires it to be PHY side SGMII */ /* Hardware requires it to be PHY side SGMII */
tx_conf = DW_VR_MII_TX_CONFIG_PHY_SIDE_SGMII; tx_conf = DW_VR_MII_TX_CONFIG_PHY_SIDE_SGMII;
...@@ -744,7 +721,7 @@ static int xpcs_config_aneg_c37_sgmii(struct dw_xpcs *xpcs, ...@@ -744,7 +721,7 @@ static int xpcs_config_aneg_c37_sgmii(struct dw_xpcs *xpcs,
else else
ret &= ~DW_VR_MII_DIG_CTRL1_MAC_AUTO_SW; ret &= ~DW_VR_MII_DIG_CTRL1_MAC_AUTO_SW;
if (xpcs->dev_flag == DW_DEV_TXGBE) if (xpcs->info.pma == WX_TXGBE_XPCS_PMA_10G_ID)
ret |= DW_VR_MII_DIG_CTRL1_PHY_MODE_CTRL; ret |= DW_VR_MII_DIG_CTRL1_PHY_MODE_CTRL;
ret = xpcs_write(xpcs, MDIO_MMD_VEND2, DW_VR_MII_DIG_CTRL1, ret); ret = xpcs_write(xpcs, MDIO_MMD_VEND2, DW_VR_MII_DIG_CTRL1, ret);
...@@ -766,7 +743,7 @@ static int xpcs_config_aneg_c37_1000basex(struct dw_xpcs *xpcs, ...@@ -766,7 +743,7 @@ static int xpcs_config_aneg_c37_1000basex(struct dw_xpcs *xpcs,
int ret, mdio_ctrl, adv; int ret, mdio_ctrl, adv;
bool changed = 0; bool changed = 0;
if (xpcs->dev_flag == DW_DEV_TXGBE) if (xpcs->info.pma == WX_TXGBE_XPCS_PMA_10G_ID)
xpcs_write_vpcs(xpcs, DW_VR_XS_PCS_DIG_CTRL1, DW_CL37_BP | DW_EN_VSMMD1); xpcs_write_vpcs(xpcs, DW_VR_XS_PCS_DIG_CTRL1, DW_CL37_BP | DW_EN_VSMMD1);
/* According to Chap 7.12, to set 1000BASE-X C37 AN, AN must /* According to Chap 7.12, to set 1000BASE-X C37 AN, AN must
...@@ -857,7 +834,7 @@ int xpcs_do_config(struct dw_xpcs *xpcs, phy_interface_t interface, ...@@ -857,7 +834,7 @@ int xpcs_do_config(struct dw_xpcs *xpcs, phy_interface_t interface,
if (!compat) if (!compat)
return -ENODEV; return -ENODEV;
if (xpcs->dev_flag == DW_DEV_TXGBE) { if (xpcs->info.pma == WX_TXGBE_XPCS_PMA_10G_ID) {
ret = txgbe_xpcs_switch_mode(xpcs, interface); ret = txgbe_xpcs_switch_mode(xpcs, interface);
if (ret) if (ret)
return ret; return ret;
...@@ -1229,44 +1206,66 @@ static void xpcs_an_restart(struct phylink_pcs *pcs) ...@@ -1229,44 +1206,66 @@ static void xpcs_an_restart(struct phylink_pcs *pcs)
} }
} }
static u32 xpcs_get_id(struct dw_xpcs *xpcs) static int xpcs_get_id(struct dw_xpcs *xpcs)
{ {
int ret; int ret;
u32 id; u32 id;
/* First, search C73 PCS using PCS MMD */ /* First, search C73 PCS using PCS MMD 3. Return ENODEV if communication
* failed indicating that device couldn't be reached.
*/
ret = xpcs_read(xpcs, MDIO_MMD_PCS, MII_PHYSID1); ret = xpcs_read(xpcs, MDIO_MMD_PCS, MII_PHYSID1);
if (ret < 0) if (ret < 0)
return 0xffffffff; return -ENODEV;
id = ret << 16; id = ret << 16;
ret = xpcs_read(xpcs, MDIO_MMD_PCS, MII_PHYSID2); ret = xpcs_read(xpcs, MDIO_MMD_PCS, MII_PHYSID2);
if (ret < 0) if (ret < 0)
return 0xffffffff; return ret;
/* If Device IDs are not all zeros or all ones, id |= ret;
* we found C73 AN-type device
*/
if ((id | ret) && (id | ret) != 0xffffffff)
return id | ret;
/* Next, search C37 PCS using Vendor-Specific MII MMD */ /* If Device IDs are not all zeros or ones, then 10GBase-X/R or C73
* KR/KX4 PCS found. Otherwise fallback to detecting 1000Base-X or C37
* PCS in MII MMD 31.
*/
if (!id || id == 0xffffffff) {
ret = xpcs_read(xpcs, MDIO_MMD_VEND2, MII_PHYSID1); ret = xpcs_read(xpcs, MDIO_MMD_VEND2, MII_PHYSID1);
if (ret < 0) if (ret < 0)
return 0xffffffff; return ret;
id = ret << 16; id = ret << 16;
ret = xpcs_read(xpcs, MDIO_MMD_VEND2, MII_PHYSID2); ret = xpcs_read(xpcs, MDIO_MMD_VEND2, MII_PHYSID2);
if (ret < 0) if (ret < 0)
return 0xffffffff; return ret;
id |= ret;
}
xpcs->info.pcs = id;
/* Find out PMA/PMD ID from MMD 1 device ID registers */
ret = xpcs_read(xpcs, MDIO_MMD_PMAPMD, MDIO_DEVID1);
if (ret < 0)
return ret;
id = ret;
ret = xpcs_read(xpcs, MDIO_MMD_PMAPMD, MDIO_DEVID2);
if (ret < 0)
return ret;
/* Note the inverted dword order and masked out Model/Revision numbers
* with respect to what is done with the PCS ID...
*/
ret = (ret >> 10) & 0x3F;
id |= ret << 16;
/* If Device IDs are not all zeros, we found C37 AN-type device */ xpcs->info.pma = id;
if (id | ret)
return id | ret;
return 0xffffffff; return 0;
} }
static const struct dw_xpcs_compat synopsys_xpcs_compat[DW_XPCS_INTERFACE_MAX] = { static const struct dw_xpcs_compat synopsys_xpcs_compat[DW_XPCS_INTERFACE_MAX] = {
...@@ -1390,15 +1389,16 @@ static void xpcs_free_data(struct dw_xpcs *xpcs) ...@@ -1390,15 +1389,16 @@ static void xpcs_free_data(struct dw_xpcs *xpcs)
static int xpcs_init_id(struct dw_xpcs *xpcs) static int xpcs_init_id(struct dw_xpcs *xpcs)
{ {
u32 xpcs_id;
int i, ret; int i, ret;
xpcs_id = xpcs_get_id(xpcs); ret = xpcs_get_id(xpcs);
if (ret < 0)
return ret;
for (i = 0; i < ARRAY_SIZE(xpcs_desc_list); i++) { for (i = 0; i < ARRAY_SIZE(xpcs_desc_list); i++) {
const struct dw_xpcs_desc *desc = &xpcs_desc_list[i]; const struct dw_xpcs_desc *desc = &xpcs_desc_list[i];
if ((xpcs_id & desc->mask) != desc->id) if ((xpcs->info.pcs & desc->mask) != desc->id)
continue; continue;
xpcs->desc = desc; xpcs->desc = desc;
...@@ -1409,10 +1409,6 @@ static int xpcs_init_id(struct dw_xpcs *xpcs) ...@@ -1409,10 +1409,6 @@ static int xpcs_init_id(struct dw_xpcs *xpcs)
if (!xpcs->desc) if (!xpcs->desc)
return -ENODEV; return -ENODEV;
ret = xpcs_dev_flag(xpcs);
if (ret < 0)
return ret;
return 0; return 0;
} }
...@@ -1424,7 +1420,7 @@ static int xpcs_init_iface(struct dw_xpcs *xpcs, phy_interface_t interface) ...@@ -1424,7 +1420,7 @@ static int xpcs_init_iface(struct dw_xpcs *xpcs, phy_interface_t interface)
if (!compat) if (!compat)
return -EINVAL; return -EINVAL;
if (xpcs->dev_flag == DW_DEV_TXGBE) { if (xpcs->info.pma == WX_TXGBE_XPCS_PMA_10G_ID) {
xpcs->pcs.poll = false; xpcs->pcs.poll = false;
return 0; return 0;
} }
......
...@@ -9,11 +9,7 @@ ...@@ -9,11 +9,7 @@
#include <linux/phy.h> #include <linux/phy.h>
#include <linux/phylink.h> #include <linux/phylink.h>
#include <linux/types.h>
#define NXP_SJA1105_XPCS_ID 0x00000010
#define NXP_SJA1110_XPCS_ID 0x00000020
#define DW_XPCS_ID 0x7996ced0
#define DW_XPCS_ID_MASK 0xffffffff
/* AN mode */ /* AN mode */
#define DW_AN_C73 1 #define DW_AN_C73 1
...@@ -22,20 +18,30 @@ ...@@ -22,20 +18,30 @@
#define DW_AN_C37_1000BASEX 4 #define DW_AN_C37_1000BASEX 4
#define DW_10GBASER 5 #define DW_10GBASER 5
/* device vendor OUI */ struct dw_xpcs_desc;
#define DW_OUI_WX 0x0018fc80
/* dev_flag */ enum dw_xpcs_pcs_id {
#define DW_DEV_TXGBE BIT(0) NXP_SJA1105_XPCS_ID = 0x00000010,
NXP_SJA1110_XPCS_ID = 0x00000020,
DW_XPCS_ID = 0x7996ced0,
DW_XPCS_ID_MASK = 0xffffffff,
};
struct dw_xpcs_desc; enum dw_xpcs_pma_id {
WX_TXGBE_XPCS_PMA_10G_ID = 0x0018fc80,
};
struct dw_xpcs_info {
u32 pcs;
u32 pma;
};
struct dw_xpcs { struct dw_xpcs {
struct dw_xpcs_info info;
const struct dw_xpcs_desc *desc; const struct dw_xpcs_desc *desc;
struct mdio_device *mdiodev; struct mdio_device *mdiodev;
struct phylink_pcs pcs; struct phylink_pcs pcs;
phy_interface_t interface; phy_interface_t interface;
int dev_flag;
}; };
int xpcs_get_an_mode(struct dw_xpcs *xpcs, phy_interface_t interface); int xpcs_get_an_mode(struct dw_xpcs *xpcs, phy_interface_t interface);
......
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