Commit d6263677 authored by Edwin Peer's avatar Edwin Peer Committed by David S. Miller

bnxt_en: support lane configuration via ethtool

Recent kernels support changing the number of link lanes via ethtool.
This is useful for determining the appropriate signal mode to use when
a given link speed can be achieved using different lane configurations.

Accept the ethtool lanes parameter when configuring forced speed.  If
there is no lanes parameter, select a default.
Signed-off-by: default avatarEdwin Peer <edwin.peer@broadcom.com>
Signed-off-by: default avatarMichael Chan <michael.chan@broadcom.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent ecdad2a6
...@@ -2019,13 +2019,15 @@ static int bnxt_get_link_ksettings(struct net_device *dev, ...@@ -2019,13 +2019,15 @@ static int bnxt_get_link_ksettings(struct net_device *dev,
return 0; return 0;
} }
static int bnxt_force_link_speed(struct net_device *dev, u32 ethtool_speed) static int
bnxt_force_link_speed(struct net_device *dev, u32 ethtool_speed, u32 lanes)
{ {
struct bnxt *bp = netdev_priv(dev); struct bnxt *bp = netdev_priv(dev);
struct bnxt_link_info *link_info = &bp->link_info; struct bnxt_link_info *link_info = &bp->link_info;
u16 support_pam4_spds = link_info->support_pam4_speeds; u16 support_pam4_spds = link_info->support_pam4_speeds;
u16 support_spds = link_info->support_speeds; u16 support_spds = link_info->support_speeds;
u8 sig_mode = BNXT_SIG_MODE_NRZ; u8 sig_mode = BNXT_SIG_MODE_NRZ;
u32 lanes_needed = 1;
u16 fw_speed = 0; u16 fw_speed = 0;
switch (ethtool_speed) { switch (ethtool_speed) {
...@@ -2046,37 +2048,46 @@ static int bnxt_force_link_speed(struct net_device *dev, u32 ethtool_speed) ...@@ -2046,37 +2048,46 @@ static int bnxt_force_link_speed(struct net_device *dev, u32 ethtool_speed)
fw_speed = PORT_PHY_CFG_REQ_FORCE_LINK_SPEED_10GB; fw_speed = PORT_PHY_CFG_REQ_FORCE_LINK_SPEED_10GB;
break; break;
case SPEED_20000: case SPEED_20000:
if (support_spds & BNXT_LINK_SPEED_MSK_20GB) if (support_spds & BNXT_LINK_SPEED_MSK_20GB) {
fw_speed = PORT_PHY_CFG_REQ_FORCE_LINK_SPEED_20GB; fw_speed = PORT_PHY_CFG_REQ_FORCE_LINK_SPEED_20GB;
lanes_needed = 2;
}
break; break;
case SPEED_25000: case SPEED_25000:
if (support_spds & BNXT_LINK_SPEED_MSK_25GB) if (support_spds & BNXT_LINK_SPEED_MSK_25GB)
fw_speed = PORT_PHY_CFG_REQ_FORCE_LINK_SPEED_25GB; fw_speed = PORT_PHY_CFG_REQ_FORCE_LINK_SPEED_25GB;
break; break;
case SPEED_40000: case SPEED_40000:
if (support_spds & BNXT_LINK_SPEED_MSK_40GB) if (support_spds & BNXT_LINK_SPEED_MSK_40GB) {
fw_speed = PORT_PHY_CFG_REQ_FORCE_LINK_SPEED_40GB; fw_speed = PORT_PHY_CFG_REQ_FORCE_LINK_SPEED_40GB;
lanes_needed = 4;
}
break; break;
case SPEED_50000: case SPEED_50000:
if (support_spds & BNXT_LINK_SPEED_MSK_50GB) { if ((support_spds & BNXT_LINK_SPEED_MSK_50GB) && lanes != 1) {
fw_speed = PORT_PHY_CFG_REQ_FORCE_LINK_SPEED_50GB; fw_speed = PORT_PHY_CFG_REQ_FORCE_LINK_SPEED_50GB;
lanes_needed = 2;
} else if (support_pam4_spds & BNXT_LINK_PAM4_SPEED_MSK_50GB) { } else if (support_pam4_spds & BNXT_LINK_PAM4_SPEED_MSK_50GB) {
fw_speed = PORT_PHY_CFG_REQ_FORCE_PAM4_LINK_SPEED_50GB; fw_speed = PORT_PHY_CFG_REQ_FORCE_PAM4_LINK_SPEED_50GB;
sig_mode = BNXT_SIG_MODE_PAM4; sig_mode = BNXT_SIG_MODE_PAM4;
} }
break; break;
case SPEED_100000: case SPEED_100000:
if (support_spds & BNXT_LINK_SPEED_MSK_100GB) { if ((support_spds & BNXT_LINK_SPEED_MSK_100GB) &&
lanes != 2 && lanes != 1) {
fw_speed = PORT_PHY_CFG_REQ_FORCE_LINK_SPEED_100GB; fw_speed = PORT_PHY_CFG_REQ_FORCE_LINK_SPEED_100GB;
lanes_needed = 4;
} else if (support_pam4_spds & BNXT_LINK_PAM4_SPEED_MSK_100GB) { } else if (support_pam4_spds & BNXT_LINK_PAM4_SPEED_MSK_100GB) {
fw_speed = PORT_PHY_CFG_REQ_FORCE_PAM4_LINK_SPEED_100GB; fw_speed = PORT_PHY_CFG_REQ_FORCE_PAM4_LINK_SPEED_100GB;
sig_mode = BNXT_SIG_MODE_PAM4; sig_mode = BNXT_SIG_MODE_PAM4;
lanes_needed = 2;
} }
break; break;
case SPEED_200000: case SPEED_200000:
if (support_pam4_spds & BNXT_LINK_PAM4_SPEED_MSK_200GB) { if (support_pam4_spds & BNXT_LINK_PAM4_SPEED_MSK_200GB) {
fw_speed = PORT_PHY_CFG_REQ_FORCE_PAM4_LINK_SPEED_200GB; fw_speed = PORT_PHY_CFG_REQ_FORCE_PAM4_LINK_SPEED_200GB;
sig_mode = BNXT_SIG_MODE_PAM4; sig_mode = BNXT_SIG_MODE_PAM4;
lanes_needed = 4;
} }
break; break;
} }
...@@ -2086,6 +2097,11 @@ static int bnxt_force_link_speed(struct net_device *dev, u32 ethtool_speed) ...@@ -2086,6 +2097,11 @@ static int bnxt_force_link_speed(struct net_device *dev, u32 ethtool_speed)
return -EINVAL; return -EINVAL;
} }
if (lanes && lanes != lanes_needed) {
netdev_err(dev, "unsupported number of lanes for speed\n");
return -EINVAL;
}
if (link_info->req_link_speed == fw_speed && if (link_info->req_link_speed == fw_speed &&
link_info->req_signal_mode == sig_mode && link_info->req_signal_mode == sig_mode &&
link_info->autoneg == 0) link_info->autoneg == 0)
...@@ -2130,7 +2146,7 @@ static int bnxt_set_link_ksettings(struct net_device *dev, ...@@ -2130,7 +2146,7 @@ static int bnxt_set_link_ksettings(struct net_device *dev,
struct bnxt_link_info *link_info = &bp->link_info; struct bnxt_link_info *link_info = &bp->link_info;
const struct ethtool_link_settings *base = &lk_ksettings->base; const struct ethtool_link_settings *base = &lk_ksettings->base;
bool set_pause = false; bool set_pause = false;
u32 speed; u32 speed, lanes = 0;
int rc = 0; int rc = 0;
if (!BNXT_PHY_CFG_ABLE(bp)) if (!BNXT_PHY_CFG_ABLE(bp))
...@@ -2171,7 +2187,8 @@ static int bnxt_set_link_ksettings(struct net_device *dev, ...@@ -2171,7 +2187,8 @@ static int bnxt_set_link_ksettings(struct net_device *dev,
goto set_setting_exit; goto set_setting_exit;
} }
speed = base->speed; speed = base->speed;
rc = bnxt_force_link_speed(dev, speed); lanes = lk_ksettings->lanes;
rc = bnxt_force_link_speed(dev, speed, lanes);
if (rc) { if (rc) {
if (rc == -EALREADY) if (rc == -EALREADY)
rc = 0; rc = 0;
...@@ -4377,6 +4394,7 @@ void bnxt_ethtool_free(struct bnxt *bp) ...@@ -4377,6 +4394,7 @@ void bnxt_ethtool_free(struct bnxt *bp)
} }
const struct ethtool_ops bnxt_ethtool_ops = { const struct ethtool_ops bnxt_ethtool_ops = {
.cap_link_lanes_supported = 1,
.supported_coalesce_params = ETHTOOL_COALESCE_USECS | .supported_coalesce_params = ETHTOOL_COALESCE_USECS |
ETHTOOL_COALESCE_MAX_FRAMES | ETHTOOL_COALESCE_MAX_FRAMES |
ETHTOOL_COALESCE_USECS_IRQ | ETHTOOL_COALESCE_USECS_IRQ |
......
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