mlxsw: spectrum: Prevent auto negotiation on number of lanes
authorShalom Toledo <shalomt@mellanox.com>
Wed, 28 Aug 2019 15:54:35 +0000 (18:54 +0300)
committerDavid S. Miller <davem@davemloft.net>
Thu, 29 Aug 2019 01:24:04 +0000 (18:24 -0700)
After 50G-1-lane and 100G-2-lanes link modes were introduced, the driver
is facing situations in which the hardware auto negotiates not only on
speed and type, but also on number of lanes.

Prevent auto negotiation on number of lanes by allowing only port speeds
that can be supported on a given port according to its width.

Signed-off-by: Shalom Toledo <shalomt@mellanox.com>
Acked-by: Jiri Pirko <jiri@mellanox.com>
Signed-off-by: Ido Schimmel <idosch@mellanox.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
drivers/net/ethernet/mellanox/mlxsw/spectrum.c
drivers/net/ethernet/mellanox/mlxsw/spectrum.h

index 4d1f8b9c46a71efc356ac48da82e4a1b87ae1c31..d4e7971e525d139fd10d0228b5985e81a5a7da0b 100644 (file)
@@ -2655,7 +2655,7 @@ mlxsw_sp1_from_ptys_supported_port(struct mlxsw_sp *mlxsw_sp,
 
 static void
 mlxsw_sp1_from_ptys_link(struct mlxsw_sp *mlxsw_sp, u32 ptys_eth_proto,
-                        unsigned long *mode)
+                        u8 width, unsigned long *mode)
 {
        int i;
 
@@ -2696,7 +2696,7 @@ mlxsw_sp1_from_ptys_speed_duplex(struct mlxsw_sp *mlxsw_sp, bool carrier_ok,
 }
 
 static u32
-mlxsw_sp1_to_ptys_advert_link(struct mlxsw_sp *mlxsw_sp,
+mlxsw_sp1_to_ptys_advert_link(struct mlxsw_sp *mlxsw_sp, u8 width,
                              const struct ethtool_link_ksettings *cmd)
 {
        u32 ptys_proto = 0;
@@ -2710,7 +2710,8 @@ mlxsw_sp1_to_ptys_advert_link(struct mlxsw_sp *mlxsw_sp,
        return ptys_proto;
 }
 
-static u32 mlxsw_sp1_to_ptys_speed(struct mlxsw_sp *mlxsw_sp, u32 speed)
+static u32 mlxsw_sp1_to_ptys_speed(struct mlxsw_sp *mlxsw_sp, u8 width,
+                                  u32 speed)
 {
        u32 ptys_proto = 0;
        int i;
@@ -2898,11 +2899,31 @@ mlxsw_sp2_mask_ethtool_200gaui_4_200gbase_cr4_kr4[] = {
 #define MLXSW_SP2_MASK_ETHTOOL_200GAUI_4_200GBASE_CR4_KR4_LEN \
        ARRAY_SIZE(mlxsw_sp2_mask_ethtool_200gaui_4_200gbase_cr4_kr4)
 
+#define MLXSW_SP_PORT_MASK_WIDTH_1X    BIT(0)
+#define MLXSW_SP_PORT_MASK_WIDTH_2X    BIT(1)
+#define MLXSW_SP_PORT_MASK_WIDTH_4X    BIT(2)
+
+static u8 mlxsw_sp_port_mask_width_get(u8 width)
+{
+       switch (width) {
+       case 1:
+               return MLXSW_SP_PORT_MASK_WIDTH_1X;
+       case 2:
+               return MLXSW_SP_PORT_MASK_WIDTH_2X;
+       case 4:
+               return MLXSW_SP_PORT_MASK_WIDTH_4X;
+       default:
+               WARN_ON_ONCE(1);
+               return 0;
+       }
+}
+
 struct mlxsw_sp2_port_link_mode {
        const enum ethtool_link_mode_bit_indices *mask_ethtool;
        int m_ethtool_len;
        u32 mask;
        u32 speed;
+       u8 mask_width;
 };
 
 static const struct mlxsw_sp2_port_link_mode mlxsw_sp2_port_link_mode[] = {
@@ -2910,72 +2931,97 @@ static const struct mlxsw_sp2_port_link_mode mlxsw_sp2_port_link_mode[] = {
                .mask           = MLXSW_REG_PTYS_EXT_ETH_SPEED_SGMII_100M,
                .mask_ethtool   = mlxsw_sp2_mask_ethtool_sgmii_100m,
                .m_ethtool_len  = MLXSW_SP2_MASK_ETHTOOL_SGMII_100M_LEN,
+               .mask_width     = MLXSW_SP_PORT_MASK_WIDTH_1X |
+                                 MLXSW_SP_PORT_MASK_WIDTH_2X |
+                                 MLXSW_SP_PORT_MASK_WIDTH_4X,
                .speed          = SPEED_100,
        },
        {
                .mask           = MLXSW_REG_PTYS_EXT_ETH_SPEED_1000BASE_X_SGMII,
                .mask_ethtool   = mlxsw_sp2_mask_ethtool_1000base_x_sgmii,
                .m_ethtool_len  = MLXSW_SP2_MASK_ETHTOOL_1000BASE_X_SGMII_LEN,
+               .mask_width     = MLXSW_SP_PORT_MASK_WIDTH_1X |
+                                 MLXSW_SP_PORT_MASK_WIDTH_2X |
+                                 MLXSW_SP_PORT_MASK_WIDTH_4X,
                .speed          = SPEED_1000,
        },
        {
                .mask           = MLXSW_REG_PTYS_EXT_ETH_SPEED_2_5GBASE_X_2_5GMII,
                .mask_ethtool   = mlxsw_sp2_mask_ethtool_2_5gbase_x_2_5gmii,
                .m_ethtool_len  = MLXSW_SP2_MASK_ETHTOOL_2_5GBASE_X_2_5GMII_LEN,
+               .mask_width     = MLXSW_SP_PORT_MASK_WIDTH_1X |
+                                 MLXSW_SP_PORT_MASK_WIDTH_2X |
+                                 MLXSW_SP_PORT_MASK_WIDTH_4X,
                .speed          = SPEED_2500,
        },
        {
                .mask           = MLXSW_REG_PTYS_EXT_ETH_SPEED_5GBASE_R,
                .mask_ethtool   = mlxsw_sp2_mask_ethtool_5gbase_r,
                .m_ethtool_len  = MLXSW_SP2_MASK_ETHTOOL_5GBASE_R_LEN,
+               .mask_width     = MLXSW_SP_PORT_MASK_WIDTH_1X |
+                                 MLXSW_SP_PORT_MASK_WIDTH_2X |
+                                 MLXSW_SP_PORT_MASK_WIDTH_4X,
                .speed          = SPEED_5000,
        },
        {
                .mask           = MLXSW_REG_PTYS_EXT_ETH_SPEED_XFI_XAUI_1_10G,
                .mask_ethtool   = mlxsw_sp2_mask_ethtool_xfi_xaui_1_10g,
                .m_ethtool_len  = MLXSW_SP2_MASK_ETHTOOL_XFI_XAUI_1_10G_LEN,
+               .mask_width     = MLXSW_SP_PORT_MASK_WIDTH_1X |
+                                 MLXSW_SP_PORT_MASK_WIDTH_2X |
+                                 MLXSW_SP_PORT_MASK_WIDTH_4X,
                .speed          = SPEED_10000,
        },
        {
                .mask           = MLXSW_REG_PTYS_EXT_ETH_SPEED_XLAUI_4_XLPPI_4_40G,
                .mask_ethtool   = mlxsw_sp2_mask_ethtool_xlaui_4_xlppi_4_40g,
                .m_ethtool_len  = MLXSW_SP2_MASK_ETHTOOL_XLAUI_4_XLPPI_4_40G_LEN,
+               .mask_width     = MLXSW_SP_PORT_MASK_WIDTH_4X,
                .speed          = SPEED_40000,
        },
        {
                .mask           = MLXSW_REG_PTYS_EXT_ETH_SPEED_25GAUI_1_25GBASE_CR_KR,
                .mask_ethtool   = mlxsw_sp2_mask_ethtool_25gaui_1_25gbase_cr_kr,
                .m_ethtool_len  = MLXSW_SP2_MASK_ETHTOOL_25GAUI_1_25GBASE_CR_KR_LEN,
+               .mask_width     = MLXSW_SP_PORT_MASK_WIDTH_1X |
+                                 MLXSW_SP_PORT_MASK_WIDTH_2X |
+                                 MLXSW_SP_PORT_MASK_WIDTH_4X,
                .speed          = SPEED_25000,
        },
        {
                .mask           = MLXSW_REG_PTYS_EXT_ETH_SPEED_50GAUI_2_LAUI_2_50GBASE_CR2_KR2,
                .mask_ethtool   = mlxsw_sp2_mask_ethtool_50gaui_2_laui_2_50gbase_cr2_kr2,
                .m_ethtool_len  = MLXSW_SP2_MASK_ETHTOOL_50GAUI_2_LAUI_2_50GBASE_CR2_KR2_LEN,
+               .mask_width     = MLXSW_SP_PORT_MASK_WIDTH_2X |
+                                 MLXSW_SP_PORT_MASK_WIDTH_4X,
                .speed          = SPEED_50000,
        },
        {
                .mask           = MLXSW_REG_PTYS_EXT_ETH_SPEED_50GAUI_1_LAUI_1_50GBASE_CR_KR,
                .mask_ethtool   = mlxsw_sp2_mask_ethtool_50gaui_1_laui_1_50gbase_cr_kr,
                .m_ethtool_len  = MLXSW_SP2_MASK_ETHTOOL_50GAUI_1_LAUI_1_50GBASE_CR_KR_LEN,
+               .mask_width     = MLXSW_SP_PORT_MASK_WIDTH_1X,
                .speed          = SPEED_50000,
        },
        {
                .mask           = MLXSW_REG_PTYS_EXT_ETH_SPEED_CAUI_4_100GBASE_CR4_KR4,
                .mask_ethtool   = mlxsw_sp2_mask_ethtool_caui_4_100gbase_cr4_kr4,
                .m_ethtool_len  = MLXSW_SP2_MASK_ETHTOOL_CAUI_4_100GBASE_CR4_KR4_LEN,
+               .mask_width     = MLXSW_SP_PORT_MASK_WIDTH_4X,
                .speed          = SPEED_100000,
        },
        {
                .mask           = MLXSW_REG_PTYS_EXT_ETH_SPEED_100GAUI_2_100GBASE_CR2_KR2,
                .mask_ethtool   = mlxsw_sp2_mask_ethtool_100gaui_2_100gbase_cr2_kr2,
                .m_ethtool_len  = MLXSW_SP2_MASK_ETHTOOL_100GAUI_2_100GBASE_CR2_KR2_LEN,
+               .mask_width     = MLXSW_SP_PORT_MASK_WIDTH_2X,
                .speed          = SPEED_100000,
        },
        {
                .mask           = MLXSW_REG_PTYS_EXT_ETH_SPEED_200GAUI_4_200GBASE_CR4_KR4,
                .mask_ethtool   = mlxsw_sp2_mask_ethtool_200gaui_4_200gbase_cr4_kr4,
                .m_ethtool_len  = MLXSW_SP2_MASK_ETHTOOL_200GAUI_4_200GBASE_CR4_KR4_LEN,
+               .mask_width     = MLXSW_SP_PORT_MASK_WIDTH_4X,
                .speed          = SPEED_200000,
        },
 };
@@ -3003,12 +3049,14 @@ mlxsw_sp2_set_bit_ethtool(const struct mlxsw_sp2_port_link_mode *link_mode,
 
 static void
 mlxsw_sp2_from_ptys_link(struct mlxsw_sp *mlxsw_sp, u32 ptys_eth_proto,
-                        unsigned long *mode)
+                        u8 width, unsigned long *mode)
 {
+       u8 mask_width = mlxsw_sp_port_mask_width_get(width);
        int i;
 
        for (i = 0; i < MLXSW_SP2_PORT_LINK_MODE_LEN; i++) {
-               if (ptys_eth_proto & mlxsw_sp2_port_link_mode[i].mask)
+               if ((ptys_eth_proto & mlxsw_sp2_port_link_mode[i].mask) &&
+                   (mask_width & mlxsw_sp2_port_link_mode[i].mask_width))
                        mlxsw_sp2_set_bit_ethtool(&mlxsw_sp2_port_link_mode[i],
                                                  mode);
        }
@@ -3059,27 +3107,32 @@ mlxsw_sp2_test_bit_ethtool(const struct mlxsw_sp2_port_link_mode *link_mode,
 }
 
 static u32
-mlxsw_sp2_to_ptys_advert_link(struct mlxsw_sp *mlxsw_sp,
+mlxsw_sp2_to_ptys_advert_link(struct mlxsw_sp *mlxsw_sp, u8 width,
                              const struct ethtool_link_ksettings *cmd)
 {
+       u8 mask_width = mlxsw_sp_port_mask_width_get(width);
        u32 ptys_proto = 0;
        int i;
 
        for (i = 0; i < MLXSW_SP2_PORT_LINK_MODE_LEN; i++) {
-               if (mlxsw_sp2_test_bit_ethtool(&mlxsw_sp2_port_link_mode[i],
+               if ((mask_width & mlxsw_sp2_port_link_mode[i].mask_width) &&
+                   mlxsw_sp2_test_bit_ethtool(&mlxsw_sp2_port_link_mode[i],
                                               cmd->link_modes.advertising))
                        ptys_proto |= mlxsw_sp2_port_link_mode[i].mask;
        }
        return ptys_proto;
 }
 
-static u32 mlxsw_sp2_to_ptys_speed(struct mlxsw_sp *mlxsw_sp, u32 speed)
+static u32 mlxsw_sp2_to_ptys_speed(struct mlxsw_sp *mlxsw_sp,
+                                  u8 width, u32 speed)
 {
+       u8 mask_width = mlxsw_sp_port_mask_width_get(width);
        u32 ptys_proto = 0;
        int i;
 
        for (i = 0; i < MLXSW_SP2_PORT_LINK_MODE_LEN; i++) {
-               if (speed == mlxsw_sp2_port_link_mode[i].speed)
+               if ((speed == mlxsw_sp2_port_link_mode[i].speed) &&
+                   (mask_width & mlxsw_sp2_port_link_mode[i].mask_width))
                        ptys_proto |= mlxsw_sp2_port_link_mode[i].mask;
        }
        return ptys_proto;
@@ -3163,7 +3216,7 @@ mlxsw_sp2_port_type_speed_ops = {
 
 static void
 mlxsw_sp_port_get_link_supported(struct mlxsw_sp *mlxsw_sp, u32 eth_proto_cap,
-                                struct ethtool_link_ksettings *cmd)
+                                u8 width, struct ethtool_link_ksettings *cmd)
 {
        const struct mlxsw_sp_port_type_speed_ops *ops;
 
@@ -3174,12 +3227,13 @@ mlxsw_sp_port_get_link_supported(struct mlxsw_sp *mlxsw_sp, u32 eth_proto_cap,
        ethtool_link_ksettings_add_link_mode(cmd, supported, Pause);
 
        ops->from_ptys_supported_port(mlxsw_sp, eth_proto_cap, cmd);
-       ops->from_ptys_link(mlxsw_sp, eth_proto_cap, cmd->link_modes.supported);
+       ops->from_ptys_link(mlxsw_sp, eth_proto_cap, width,
+                           cmd->link_modes.supported);
 }
 
 static void
 mlxsw_sp_port_get_link_advertise(struct mlxsw_sp *mlxsw_sp,
-                                u32 eth_proto_admin, bool autoneg,
+                                u32 eth_proto_admin, bool autoneg, u8 width,
                                 struct ethtool_link_ksettings *cmd)
 {
        const struct mlxsw_sp_port_type_speed_ops *ops;
@@ -3190,7 +3244,7 @@ mlxsw_sp_port_get_link_advertise(struct mlxsw_sp *mlxsw_sp,
                return;
 
        ethtool_link_ksettings_add_link_mode(cmd, advertising, Autoneg);
-       ops->from_ptys_link(mlxsw_sp, eth_proto_admin,
+       ops->from_ptys_link(mlxsw_sp, eth_proto_admin, width,
                            cmd->link_modes.advertising);
 }
 
@@ -3245,10 +3299,11 @@ static int mlxsw_sp_port_get_link_ksettings(struct net_device *dev,
        ops->reg_ptys_eth_unpack(mlxsw_sp, ptys_pl, &eth_proto_cap,
                                 &eth_proto_admin, &eth_proto_oper);
 
-       mlxsw_sp_port_get_link_supported(mlxsw_sp, eth_proto_cap, cmd);
+       mlxsw_sp_port_get_link_supported(mlxsw_sp, eth_proto_cap,
+                                        mlxsw_sp_port->mapping.width, cmd);
 
        mlxsw_sp_port_get_link_advertise(mlxsw_sp, eth_proto_admin, autoneg,
-                                        cmd);
+                                        mlxsw_sp_port->mapping.width, cmd);
 
        cmd->base.autoneg = autoneg ? AUTONEG_ENABLE : AUTONEG_DISABLE;
        connector_type = mlxsw_reg_ptys_connector_type_get(ptys_pl);
@@ -3282,8 +3337,10 @@ mlxsw_sp_port_set_link_ksettings(struct net_device *dev,
 
        autoneg = cmd->base.autoneg == AUTONEG_ENABLE;
        eth_proto_new = autoneg ?
-               ops->to_ptys_advert_link(mlxsw_sp, cmd) :
-               ops->to_ptys_speed(mlxsw_sp, cmd->base.speed);
+               ops->to_ptys_advert_link(mlxsw_sp, mlxsw_sp_port->mapping.width,
+                                        cmd) :
+               ops->to_ptys_speed(mlxsw_sp, mlxsw_sp_port->mapping.width,
+                                  cmd->base.speed);
 
        eth_proto_new = eth_proto_new & eth_proto_cap;
        if (!eth_proto_new) {
index 20c14bba9ccb9ecce1e60ae85c7d891a21c8e417..12f14a42b43a2fb4b0e4b050bb5882b474f24cc3 100644 (file)
@@ -279,14 +279,14 @@ struct mlxsw_sp_port_type_speed_ops {
                                         u32 ptys_eth_proto,
                                         struct ethtool_link_ksettings *cmd);
        void (*from_ptys_link)(struct mlxsw_sp *mlxsw_sp, u32 ptys_eth_proto,
-                              unsigned long *mode);
+                              u8 width, unsigned long *mode);
        u32 (*from_ptys_speed)(struct mlxsw_sp *mlxsw_sp, u32 ptys_eth_proto);
        void (*from_ptys_speed_duplex)(struct mlxsw_sp *mlxsw_sp,
                                       bool carrier_ok, u32 ptys_eth_proto,
                                       struct ethtool_link_ksettings *cmd);
-       u32 (*to_ptys_advert_link)(struct mlxsw_sp *mlxsw_sp,
+       u32 (*to_ptys_advert_link)(struct mlxsw_sp *mlxsw_sp, u8 width,
                                   const struct ethtool_link_ksettings *cmd);
-       u32 (*to_ptys_speed)(struct mlxsw_sp *mlxsw_sp, u32 speed);
+       u32 (*to_ptys_speed)(struct mlxsw_sp *mlxsw_sp, u8 width, u32 speed);
        u32 (*to_ptys_upper_speed)(struct mlxsw_sp *mlxsw_sp, u32 upper_speed);
        int (*port_speed_base)(struct mlxsw_sp *mlxsw_sp, u8 local_port,
                               u32 *base_speed);