diff mbox series

[net-next,v3,1/7] ethtool: Extend link modes settings uAPI with lanes

Message ID 20210120093713.4000363-2-danieller@nvidia.com (mailing list archive)
State Changes Requested
Delegated to: Netdev Maintainers
Headers show
Series Support setting lanes via ethtool | expand

Checks

Context Check Description
netdev/cover_letter success Link
netdev/fixes_present success Link
netdev/patch_count success Link
netdev/tree_selection success Clearly marked for net-next
netdev/subject_prefix success Link
netdev/cc_maintainers warning 13 maintainers not CCed: meirl@mellanox.com linux-doc@vger.kernel.org dmurphy@ti.com irusskikh@marvell.com saeedm@nvidia.com amitc@mellanox.com ayal@mellanox.com petr.vorel@gmail.com mchehab+huawei@kernel.org gustavo@embeddedor.com gaurav1086@gmail.com linux@rempel-privat.de corbet@lwn.net
netdev/source_inline success Was 0 now: 0
netdev/verify_signedoff success Link
netdev/module_param success Was 0 now: 0
netdev/build_32bit success Errors and warnings before: 3290 this patch: 3290
netdev/kdoc success Errors and warnings before: 75 this patch: 75
netdev/verify_fixes success Link
netdev/checkpatch warning WARNING: line length of 81 exceeds 80 columns WARNING: line length of 82 exceeds 80 columns WARNING: line length of 89 exceeds 80 columns
netdev/build_allmodconfig_warn success Errors and warnings before: 3595 this patch: 3595
netdev/header_inline success Link
netdev/stable success Stable not CCed

Commit Message

Danielle Ratson Jan. 20, 2021, 9:37 a.m. UTC
Currently, when auto negotiation is on, the user can advertise all the
linkmodes which correspond to a specific speed, but does not have a
similar selector for the number of lanes. This is significant when a
specific speed can be achieved using different number of lanes.  For
example, 2x50 or 4x25.

Add 'ETHTOOL_A_LINKMODES_LANES' attribute and expand 'struct
ethtool_link_settings' with lanes field in order to implement a new
lanes-selector that will enable the user to advertise a specific number
of lanes as well.

When auto negotiation is off, lanes parameter can be forced only if the
driver supports it. Add a capability bit in 'struct ethtool_ops' that
allows ethtool know if the driver can handle the lanes parameter when
auto negotiation is off, so if it does not, an error message will be
returned when trying to set lanes.

Example:

$ ethtool -s swp1 lanes 4
$ ethtool swp1
  Settings for swp1:
	Supported ports: [ FIBRE ]
        Supported link modes:   1000baseKX/Full
                                10000baseKR/Full
                                40000baseCR4/Full
				40000baseSR4/Full
				40000baseLR4/Full
                                25000baseCR/Full
                                25000baseSR/Full
				50000baseCR2/Full
                                100000baseSR4/Full
				100000baseCR4/Full
        Supported pause frame use: Symmetric Receive-only
        Supports auto-negotiation: Yes
        Supported FEC modes: Not reported
        Advertised link modes:  40000baseCR4/Full
				40000baseSR4/Full
				40000baseLR4/Full
                                100000baseSR4/Full
				100000baseCR4/Full
        Advertised pause frame use: No
        Advertised auto-negotiation: Yes
        Advertised FEC modes: Not reported
        Speed: Unknown!
        Duplex: Unknown! (255)
        Auto-negotiation: on
        Port: Direct Attach Copper
        PHYAD: 0
        Transceiver: internal
        Link detected: no

Signed-off-by: Danielle Ratson <danieller@nvidia.com>
---

Notes:
    v3:
    	* Change ethtool_ops.capabilities to be a bitfield and rename it.
    	* Add an according kdoc.
    	* Set min and max for the lanes policy.
    	* Change the lanes validation to be the power of two, now that the
    	  min and max are set.
    
    v2:
    	* Remove ETHTOOL_LANES defines and simply use a number instead.

 Documentation/networking/ethtool-netlink.rst |  11 +-
 include/linux/ethtool.h                      |   4 +
 include/uapi/linux/ethtool.h                 |   2 +
 include/uapi/linux/ethtool_netlink.h         |   1 +
 net/ethtool/linkmodes.c                      | 214 +++++++++++--------
 net/ethtool/netlink.h                        |   2 +-
 6 files changed, 139 insertions(+), 95 deletions(-)

Comments

Edwin Peer Jan. 20, 2021, 10:35 p.m. UTC | #1
On Wed, Jan 20, 2021 at 3:21 AM Danielle Ratson <danieller@nvidia.com> wrote:

> -#define __DEFINE_LINK_MODE_PARAMS(_speed, _type, _duplex) \
> +#define __DEFINE_LINK_MODE_PARAMS(_speed, _type, _lanes, _duplex) \
>         [ETHTOOL_LINK_MODE(_speed, _type, _duplex)] = { \
>                 .speed  = SPEED_ ## _speed, \
> +               .lanes  = _lanes, \
>                 .duplex = __DUPLEX_ ## _duplex \
>         }

What about:

#define __DECLARE_LINK_MODE_LANES(_type, _lanes)        \
static const u32 __LINK_MODE_LANES_ ## _type = _lanes;

#define __DECLARE_LINK_MODE_LANES_ALL(_type)            \
__DECLARE_LINK_MODE_LANES(_type, 1)             \
__DECLARE_LINK_MODE_LANES(_type ## 2, 2)        \
__DECLARE_LINK_MODE_LANES(_type ## 4, 4)        \
__DECLARE_LINK_MODE_LANES(_type ## 8, 8)

__DECLARE_LINK_MODE_LANES_ALL(CR)
__DECLARE_LINK_MODE_LANES_ALL(DR)
__DECLARE_LINK_MODE_LANES_ALL(ER)
__DECLARE_LINK_MODE_LANES_ALL(KR)
__DECLARE_LINK_MODE_LANES(KX, 1)
__DECLARE_LINK_MODE_LANES(KX4, 4)
__DECLARE_LINK_MODE_LANES_ALL(LR)
__DECLARE_LINK_MODE_LANES(LR_ER_FR, 1)
__DECLARE_LINK_MODE_LANES(LR2_ER2_FR2, 2)
__DECLARE_LINK_MODE_LANES(LR4_ER4_FR4, 4)
__DECLARE_LINK_MODE_LANES(LR8_ER8_FR8, 8)
__DECLARE_LINK_MODE_LANES(LRM, 1)
__DECLARE_LINK_MODE_LANES(MLD2, 2);
__DECLARE_LINK_MODE_LANES_ALL(SR);
__DECLARE_LINK_MODE_LANES(T, 1)
__DECLARE_LINK_MODE_LANES(X, 1)

#define __DEFINE_LINK_MODE_PARAMS(_speed, _type, _duplex)
         [ETHTOOL_LINK_MODE(_speed, _type, _duplex)] = { \
                 .speed  = SPEED_ ## _speed, \
                 .lanes  = __LINK_MODE_LANES ## _type, \

instead of specifying lanes for each link mode defined below?

Regards,
Edwin Peer

>  static const struct link_mode_info link_mode_params[] = {
> -       __DEFINE_LINK_MODE_PARAMS(10, T, Half),
> -       __DEFINE_LINK_MODE_PARAMS(10, T, Full),
> -       __DEFINE_LINK_MODE_PARAMS(100, T, Half),
> -       __DEFINE_LINK_MODE_PARAMS(100, T, Full),
> -       __DEFINE_LINK_MODE_PARAMS(1000, T, Half),
> -       __DEFINE_LINK_MODE_PARAMS(1000, T, Full),
> +       __DEFINE_LINK_MODE_PARAMS(10, T, 1, Half),
> +       __DEFINE_LINK_MODE_PARAMS(10, T, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(100, T, 1, Half),
> +       __DEFINE_LINK_MODE_PARAMS(100, T, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(1000, T, 1, Half),
> +       __DEFINE_LINK_MODE_PARAMS(1000, T, 1, Full),
>         __DEFINE_SPECIAL_MODE_PARAMS(Autoneg),
>         __DEFINE_SPECIAL_MODE_PARAMS(TP),
>         __DEFINE_SPECIAL_MODE_PARAMS(AUI),
>         __DEFINE_SPECIAL_MODE_PARAMS(MII),
>         __DEFINE_SPECIAL_MODE_PARAMS(FIBRE),
>         __DEFINE_SPECIAL_MODE_PARAMS(BNC),
> -       __DEFINE_LINK_MODE_PARAMS(10000, T, Full),
> +       __DEFINE_LINK_MODE_PARAMS(10000, T, 1, Full),
>         __DEFINE_SPECIAL_MODE_PARAMS(Pause),
>         __DEFINE_SPECIAL_MODE_PARAMS(Asym_Pause),
> -       __DEFINE_LINK_MODE_PARAMS(2500, X, Full),
> +       __DEFINE_LINK_MODE_PARAMS(2500, X, 1, Full),
>         __DEFINE_SPECIAL_MODE_PARAMS(Backplane),
> -       __DEFINE_LINK_MODE_PARAMS(1000, KX, Full),
> -       __DEFINE_LINK_MODE_PARAMS(10000, KX4, Full),
> -       __DEFINE_LINK_MODE_PARAMS(10000, KR, Full),
> +       __DEFINE_LINK_MODE_PARAMS(1000, KX, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(10000, KX4, 4, Full),
> +       __DEFINE_LINK_MODE_PARAMS(10000, KR, 1, Full),
>         [ETHTOOL_LINK_MODE_10000baseR_FEC_BIT] = {
>                 .speed  = SPEED_10000,
>                 .duplex = DUPLEX_FULL,
>         },
> -       __DEFINE_LINK_MODE_PARAMS(20000, MLD2, Full),
> -       __DEFINE_LINK_MODE_PARAMS(20000, KR2, Full),
> -       __DEFINE_LINK_MODE_PARAMS(40000, KR4, Full),
> -       __DEFINE_LINK_MODE_PARAMS(40000, CR4, Full),
> -       __DEFINE_LINK_MODE_PARAMS(40000, SR4, Full),
> -       __DEFINE_LINK_MODE_PARAMS(40000, LR4, Full),
> -       __DEFINE_LINK_MODE_PARAMS(56000, KR4, Full),
> -       __DEFINE_LINK_MODE_PARAMS(56000, CR4, Full),
> -       __DEFINE_LINK_MODE_PARAMS(56000, SR4, Full),
> -       __DEFINE_LINK_MODE_PARAMS(56000, LR4, Full),
> -       __DEFINE_LINK_MODE_PARAMS(25000, CR, Full),
> -       __DEFINE_LINK_MODE_PARAMS(25000, KR, Full),
> -       __DEFINE_LINK_MODE_PARAMS(25000, SR, Full),
> -       __DEFINE_LINK_MODE_PARAMS(50000, CR2, Full),
> -       __DEFINE_LINK_MODE_PARAMS(50000, KR2, Full),
> -       __DEFINE_LINK_MODE_PARAMS(100000, KR4, Full),
> -       __DEFINE_LINK_MODE_PARAMS(100000, SR4, Full),
> -       __DEFINE_LINK_MODE_PARAMS(100000, CR4, Full),
> -       __DEFINE_LINK_MODE_PARAMS(100000, LR4_ER4, Full),
> -       __DEFINE_LINK_MODE_PARAMS(50000, SR2, Full),
> -       __DEFINE_LINK_MODE_PARAMS(1000, X, Full),
> -       __DEFINE_LINK_MODE_PARAMS(10000, CR, Full),
> -       __DEFINE_LINK_MODE_PARAMS(10000, SR, Full),
> -       __DEFINE_LINK_MODE_PARAMS(10000, LR, Full),
> -       __DEFINE_LINK_MODE_PARAMS(10000, LRM, Full),
> -       __DEFINE_LINK_MODE_PARAMS(10000, ER, Full),
> -       __DEFINE_LINK_MODE_PARAMS(2500, T, Full),
> -       __DEFINE_LINK_MODE_PARAMS(5000, T, Full),
> +       __DEFINE_LINK_MODE_PARAMS(20000, MLD2, 2, Full),
> +       __DEFINE_LINK_MODE_PARAMS(20000, KR2, 2, Full),
> +       __DEFINE_LINK_MODE_PARAMS(40000, KR4, 4, Full),
> +       __DEFINE_LINK_MODE_PARAMS(40000, CR4, 4, Full),
> +       __DEFINE_LINK_MODE_PARAMS(40000, SR4, 4, Full),
> +       __DEFINE_LINK_MODE_PARAMS(40000, LR4, 4, Full),
> +       __DEFINE_LINK_MODE_PARAMS(56000, KR4, 4, Full),
> +       __DEFINE_LINK_MODE_PARAMS(56000, CR4, 4, Full),
> +       __DEFINE_LINK_MODE_PARAMS(56000, SR4, 4, Full),
> +       __DEFINE_LINK_MODE_PARAMS(56000, LR4, 4, Full),
> +       __DEFINE_LINK_MODE_PARAMS(25000, CR, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(25000, KR, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(25000, SR, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(50000, CR2, 2, Full),
> +       __DEFINE_LINK_MODE_PARAMS(50000, KR2, 2, Full),
> +       __DEFINE_LINK_MODE_PARAMS(100000, KR4, 4, Full),
> +       __DEFINE_LINK_MODE_PARAMS(100000, SR4, 4, Full),
> +       __DEFINE_LINK_MODE_PARAMS(100000, CR4, 4, Full),
> +       __DEFINE_LINK_MODE_PARAMS(100000, LR4_ER4, 4, Full),
> +       __DEFINE_LINK_MODE_PARAMS(50000, SR2, 2, Full),
> +       __DEFINE_LINK_MODE_PARAMS(1000, X, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(10000, CR, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(10000, SR, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(10000, LR, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(10000, LRM, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(10000, ER, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(2500, T, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(5000, T, 1, Full),
>         __DEFINE_SPECIAL_MODE_PARAMS(FEC_NONE),
>         __DEFINE_SPECIAL_MODE_PARAMS(FEC_RS),
>         __DEFINE_SPECIAL_MODE_PARAMS(FEC_BASER),
> -       __DEFINE_LINK_MODE_PARAMS(50000, KR, Full),
> -       __DEFINE_LINK_MODE_PARAMS(50000, SR, Full),
> -       __DEFINE_LINK_MODE_PARAMS(50000, CR, Full),
> -       __DEFINE_LINK_MODE_PARAMS(50000, LR_ER_FR, Full),
> -       __DEFINE_LINK_MODE_PARAMS(50000, DR, Full),
> -       __DEFINE_LINK_MODE_PARAMS(100000, KR2, Full),
> -       __DEFINE_LINK_MODE_PARAMS(100000, SR2, Full),
> -       __DEFINE_LINK_MODE_PARAMS(100000, CR2, Full),
> -       __DEFINE_LINK_MODE_PARAMS(100000, LR2_ER2_FR2, Full),
> -       __DEFINE_LINK_MODE_PARAMS(100000, DR2, Full),
> -       __DEFINE_LINK_MODE_PARAMS(200000, KR4, Full),
> -       __DEFINE_LINK_MODE_PARAMS(200000, SR4, Full),
> -       __DEFINE_LINK_MODE_PARAMS(200000, LR4_ER4_FR4, Full),
> -       __DEFINE_LINK_MODE_PARAMS(200000, DR4, Full),
> -       __DEFINE_LINK_MODE_PARAMS(200000, CR4, Full),
> -       __DEFINE_LINK_MODE_PARAMS(100, T1, Full),
> -       __DEFINE_LINK_MODE_PARAMS(1000, T1, Full),
> -       __DEFINE_LINK_MODE_PARAMS(400000, KR8, Full),
> -       __DEFINE_LINK_MODE_PARAMS(400000, SR8, Full),
> -       __DEFINE_LINK_MODE_PARAMS(400000, LR8_ER8_FR8, Full),
> -       __DEFINE_LINK_MODE_PARAMS(400000, DR8, Full),
> -       __DEFINE_LINK_MODE_PARAMS(400000, CR8, Full),
> +       __DEFINE_LINK_MODE_PARAMS(50000, KR, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(50000, SR, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(50000, CR, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(50000, LR_ER_FR, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(50000, DR, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(100000, KR2, 2, Full),
> +       __DEFINE_LINK_MODE_PARAMS(100000, SR2, 2, Full),
> +       __DEFINE_LINK_MODE_PARAMS(100000, CR2, 2, Full),
> +       __DEFINE_LINK_MODE_PARAMS(100000, LR2_ER2_FR2, 2, Full),
> +       __DEFINE_LINK_MODE_PARAMS(100000, DR2, 2, Full),
> +       __DEFINE_LINK_MODE_PARAMS(200000, KR4, 4, Full),
> +       __DEFINE_LINK_MODE_PARAMS(200000, SR4, 4, Full),
> +       __DEFINE_LINK_MODE_PARAMS(200000, LR4_ER4_FR4, 4, Full),
> +       __DEFINE_LINK_MODE_PARAMS(200000, DR4, 4, Full),
> +       __DEFINE_LINK_MODE_PARAMS(200000, CR4, 4, Full),
> +       __DEFINE_LINK_MODE_PARAMS(100, T1, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(1000, T1, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(400000, KR8, 8, Full),
> +       __DEFINE_LINK_MODE_PARAMS(400000, SR8, 8, Full),
> +       __DEFINE_LINK_MODE_PARAMS(400000, LR8_ER8_FR8, 8, Full),
> +       __DEFINE_LINK_MODE_PARAMS(400000, DR8, 8, Full),
> +       __DEFINE_LINK_MODE_PARAMS(400000, CR8, 8, Full),
>         __DEFINE_SPECIAL_MODE_PARAMS(FEC_LLRS),
> -       __DEFINE_LINK_MODE_PARAMS(100000, KR, Full),
> -       __DEFINE_LINK_MODE_PARAMS(100000, SR, Full),
> -       __DEFINE_LINK_MODE_PARAMS(100000, LR_ER_FR, Full),
> -       __DEFINE_LINK_MODE_PARAMS(100000, DR, Full),
> -       __DEFINE_LINK_MODE_PARAMS(100000, CR, Full),
> -       __DEFINE_LINK_MODE_PARAMS(200000, KR2, Full),
> -       __DEFINE_LINK_MODE_PARAMS(200000, SR2, Full),
> -       __DEFINE_LINK_MODE_PARAMS(200000, LR2_ER2_FR2, Full),
> -       __DEFINE_LINK_MODE_PARAMS(200000, DR2, Full),
> -       __DEFINE_LINK_MODE_PARAMS(200000, CR2, Full),
> -       __DEFINE_LINK_MODE_PARAMS(400000, KR4, Full),
> -       __DEFINE_LINK_MODE_PARAMS(400000, SR4, Full),
> -       __DEFINE_LINK_MODE_PARAMS(400000, LR4_ER4_FR4, Full),
> -       __DEFINE_LINK_MODE_PARAMS(400000, DR4, Full),
> -       __DEFINE_LINK_MODE_PARAMS(400000, CR4, Full),
> -       __DEFINE_LINK_MODE_PARAMS(100, FX, Half),
> -       __DEFINE_LINK_MODE_PARAMS(100, FX, Full),
> +       __DEFINE_LINK_MODE_PARAMS(100000, KR, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(100000, SR, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(100000, LR_ER_FR, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(100000, DR, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(100000, CR, 1, Full),
> +       __DEFINE_LINK_MODE_PARAMS(200000, KR2, 2, Full),
> +       __DEFINE_LINK_MODE_PARAMS(200000, SR2, 2, Full),
> +       __DEFINE_LINK_MODE_PARAMS(200000, LR2_ER2_FR2, 2, Full),
> +       __DEFINE_LINK_MODE_PARAMS(200000, DR2, 2, Full),
> +       __DEFINE_LINK_MODE_PARAMS(200000, CR2, 2, Full),
> +       __DEFINE_LINK_MODE_PARAMS(400000, KR4, 4, Full),
> +       __DEFINE_LINK_MODE_PARAMS(400000, SR4, 4, Full),
> +       __DEFINE_LINK_MODE_PARAMS(400000, LR4_ER4_FR4, 4, Full),
> +       __DEFINE_LINK_MODE_PARAMS(400000, DR4, 4, Full),
> +       __DEFINE_LINK_MODE_PARAMS(400000, CR4, 4, Full),
> +       __DEFINE_LINK_MODE_PARAMS(100, FX, 1, Half),
> +       __DEFINE_LINK_MODE_PARAMS(100, FX, 1, Full),
>  };
Jakub Kicinski Jan. 22, 2021, 3:44 a.m. UTC | #2
On Wed, 20 Jan 2021 11:37:07 +0200 Danielle Ratson wrote:
> Currently, when auto negotiation is on, the user can advertise all the
> linkmodes which correspond to a specific speed, but does not have a
> similar selector for the number of lanes. This is significant when a
> specific speed can be achieved using different number of lanes.  For
> example, 2x50 or 4x25.
> 
> Add 'ETHTOOL_A_LINKMODES_LANES' attribute and expand 'struct
> ethtool_link_settings' with lanes field in order to implement a new
> lanes-selector that will enable the user to advertise a specific number
> of lanes as well.
> 
> When auto negotiation is off, lanes parameter can be forced only if the
> driver supports it. Add a capability bit in 'struct ethtool_ops' that
> allows ethtool know if the driver can handle the lanes parameter when
> auto negotiation is off, so if it does not, an error message will be
> returned when trying to set lanes.

> Signed-off-by: Danielle Ratson <danieller@nvidia.com>

> diff --git a/include/uapi/linux/ethtool.h b/include/uapi/linux/ethtool.h
> index cde753bb2093..80edae2c24f7 100644
> --- a/include/uapi/linux/ethtool.h
> +++ b/include/uapi/linux/ethtool.h
> @@ -1738,6 +1738,8 @@ static inline int ethtool_validate_speed(__u32 speed)
>  	return speed <= INT_MAX || speed == (__u32)SPEED_UNKNOWN;
>  }
>  
> +#define ETHTOOL_LANES_UNKNOWN		0

I already complained about these unnecessary uAPI constants, did you
reply to that and I missed it?

Don't report the nlattr if it's unknown, we have netlink now, those
constants are from times when we returned structures and all fields 
had to have a value.

>  /* Duplex, half or full. */
>  #define DUPLEX_HALF		0x00
>  #define DUPLEX_FULL		0x01
> diff --git a/include/uapi/linux/ethtool_netlink.h b/include/uapi/linux/ethtool_netlink.h
> index e2bf36e6964b..a286635ac9b8 100644
> --- a/include/uapi/linux/ethtool_netlink.h
> +++ b/include/uapi/linux/ethtool_netlink.h
> @@ -227,6 +227,7 @@ enum {
>  	ETHTOOL_A_LINKMODES_DUPLEX,		/* u8 */
>  	ETHTOOL_A_LINKMODES_MASTER_SLAVE_CFG,	/* u8 */
>  	ETHTOOL_A_LINKMODES_MASTER_SLAVE_STATE,	/* u8 */
> +	ETHTOOL_A_LINKMODES_LANES,		/* u32 */
>  
>  	/* add new constants above here */
>  	__ETHTOOL_A_LINKMODES_CNT,
> diff --git a/net/ethtool/linkmodes.c b/net/ethtool/linkmodes.c
> index c5bcb9abc8b9..fb7d73250864 100644
> --- a/net/ethtool/linkmodes.c
> +++ b/net/ethtool/linkmodes.c
> @@ -152,12 +152,14 @@ const struct ethnl_request_ops ethnl_linkmodes_request_ops = {
>  
>  struct link_mode_info {
>  	int				speed;
> +	u32				lanes;

This is not uapi, we can make it u8 now, save a few (hundred?) bytes 
of memory and bump it to u16 later.

>  	u8				duplex;
>  };

> @@ -353,10 +358,39 @@ static int ethnl_update_linkmodes(struct genl_info *info, struct nlattr **tb,
>  
>  	*mod = false;
>  	req_speed = tb[ETHTOOL_A_LINKMODES_SPEED];
> +	req_lanes = tb[ETHTOOL_A_LINKMODES_LANES];
>  	req_duplex = tb[ETHTOOL_A_LINKMODES_DUPLEX];
>  
>  	ethnl_update_u8(&lsettings->autoneg, tb[ETHTOOL_A_LINKMODES_AUTONEG],
>  			mod);
> +
> +	if (req_lanes) {
> +		u32 lanes_cfg = nla_get_u32(tb[ETHTOOL_A_LINKMODES_LANES]);

req_lanes == tb[ETHTOOL_A_LINKMODES_LANES], right?

Please use req_lanes variable where possible.

> +
> +		if (!is_power_of_2(lanes_cfg)) {
> +			NL_SET_ERR_MSG_ATTR(info->extack,
> +					    tb[ETHTOOL_A_LINKMODES_LANES],
> +					    "lanes value is invalid");
> +			return -EINVAL;
> +		}
> +
> +		/* If autoneg is off and lanes parameter is not supported by the
> +		 * driver, return an error.
> +		 */
> +		if (!lsettings->autoneg &&
> +		    !dev->ethtool_ops->cap_link_lanes_supported) {
> +			NL_SET_ERR_MSG_ATTR(info->extack,
> +					    tb[ETHTOOL_A_LINKMODES_LANES],
> +					    "lanes configuration not supported by device");
> +			return -EOPNOTSUPP;
> +		}

This validation does not depend on the current settings at all, it's
just input validation, it can be done before rtnl_lock is taken (in a
new function).

You can move ethnl_validate_master_slave_cfg() to that function as well
(as a cleanup before this patch).

> +	} else if (!lsettings->autoneg) {
> +		/* If autoneg is off and lanes parameter is not passed from user,
> +		 * set the lanes parameter to UNKNOWN.
> +		 */
> +		ksettings->lanes = ETHTOOL_LANES_UNKNOWN;
> +	}
> +
>  	ret = ethnl_update_bitset(ksettings->link_modes.advertising,
>  				  __ETHTOOL_LINK_MODE_MASK_NBITS,
>  				  tb[ETHTOOL_A_LINKMODES_OURS], link_mode_names,
Danielle Ratson Jan. 25, 2021, 3:53 p.m. UTC | #3
> -----Original Message-----
> From: Jakub Kicinski <kuba@kernel.org>
> Sent: Friday, January 22, 2021 5:45 AM
> To: Danielle Ratson <danieller@nvidia.com>
> Cc: netdev@vger.kernel.org; davem@davemloft.net; Jiri Pirko <jiri@nvidia.com>; andrew@lunn.ch; f.fainelli@gmail.com;
> mkubecek@suse.cz; mlxsw <mlxsw@nvidia.com>; Ido Schimmel <idosch@nvidia.com>
> Subject: Re: [PATCH net-next v3 1/7] ethtool: Extend link modes settings uAPI with lanes
> 
> On Wed, 20 Jan 2021 11:37:07 +0200 Danielle Ratson wrote:
> > Currently, when auto negotiation is on, the user can advertise all the
> > linkmodes which correspond to a specific speed, but does not have a
> > similar selector for the number of lanes. This is significant when a
> > specific speed can be achieved using different number of lanes.  For
> > example, 2x50 or 4x25.
> >
> > Add 'ETHTOOL_A_LINKMODES_LANES' attribute and expand 'struct
> > ethtool_link_settings' with lanes field in order to implement a new
> > lanes-selector that will enable the user to advertise a specific
> > number of lanes as well.
> >
> > When auto negotiation is off, lanes parameter can be forced only if
> > the driver supports it. Add a capability bit in 'struct ethtool_ops'
> > that allows ethtool know if the driver can handle the lanes parameter
> > when auto negotiation is off, so if it does not, an error message will
> > be returned when trying to set lanes.
> 
> > Signed-off-by: Danielle Ratson <danieller@nvidia.com>
> 
> > diff --git a/include/uapi/linux/ethtool.h
> > b/include/uapi/linux/ethtool.h index cde753bb2093..80edae2c24f7 100644
> > --- a/include/uapi/linux/ethtool.h
> > +++ b/include/uapi/linux/ethtool.h
> > @@ -1738,6 +1738,8 @@ static inline int ethtool_validate_speed(__u32 speed)
> >  	return speed <= INT_MAX || speed == (__u32)SPEED_UNKNOWN;  }
> >
> > +#define ETHTOOL_LANES_UNKNOWN		0
> 
> I already complained about these unnecessary uAPI constants, did you reply to that and I missed it?

I guess I missed it, sorry, will fix.

> 
> Don't report the nlattr if it's unknown, we have netlink now, those constants are from times when we returned structures and all
> fields had to have a value.
> 
> >  /* Duplex, half or full. */
> >  #define DUPLEX_HALF		0x00
> >  #define DUPLEX_FULL		0x01
> > diff --git a/include/uapi/linux/ethtool_netlink.h
> > b/include/uapi/linux/ethtool_netlink.h
> > index e2bf36e6964b..a286635ac9b8 100644
> > --- a/include/uapi/linux/ethtool_netlink.h
> > +++ b/include/uapi/linux/ethtool_netlink.h
> > @@ -227,6 +227,7 @@ enum {
> >  	ETHTOOL_A_LINKMODES_DUPLEX,		/* u8 */
> >  	ETHTOOL_A_LINKMODES_MASTER_SLAVE_CFG,	/* u8 */
> >  	ETHTOOL_A_LINKMODES_MASTER_SLAVE_STATE,	/* u8 */
> > +	ETHTOOL_A_LINKMODES_LANES,		/* u32 */
> >
> >  	/* add new constants above here */
> >  	__ETHTOOL_A_LINKMODES_CNT,
> > diff --git a/net/ethtool/linkmodes.c b/net/ethtool/linkmodes.c index
> > c5bcb9abc8b9..fb7d73250864 100644
> > --- a/net/ethtool/linkmodes.c
> > +++ b/net/ethtool/linkmodes.c
> > @@ -152,12 +152,14 @@ const struct ethnl_request_ops
> > ethnl_linkmodes_request_ops = {
> >
> >  struct link_mode_info {
> >  	int				speed;
> > +	u32				lanes;
> 
> This is not uapi, we can make it u8 now, save a few (hundred?) bytes of memory and bump it to u16 later.
> 
> >  	u8				duplex;
> >  };
> 
> > @@ -353,10 +358,39 @@ static int ethnl_update_linkmodes(struct
> > genl_info *info, struct nlattr **tb,
> >
> >  	*mod = false;
> >  	req_speed = tb[ETHTOOL_A_LINKMODES_SPEED];
> > +	req_lanes = tb[ETHTOOL_A_LINKMODES_LANES];
> >  	req_duplex = tb[ETHTOOL_A_LINKMODES_DUPLEX];
> >
> >  	ethnl_update_u8(&lsettings->autoneg, tb[ETHTOOL_A_LINKMODES_AUTONEG],
> >  			mod);
> > +
> > +	if (req_lanes) {
> > +		u32 lanes_cfg = nla_get_u32(tb[ETHTOOL_A_LINKMODES_LANES]);
> 
> req_lanes == tb[ETHTOOL_A_LINKMODES_LANES], right? 

Yes, but req_lanes is a bool and doesn't fit to nla_get_u32. Do you want me to change the req_lanes type and name?

>
> Please use req_lanes variable where possible.
> 
> > +
> > +		if (!is_power_of_2(lanes_cfg)) {
> > +			NL_SET_ERR_MSG_ATTR(info->extack,
> > +					    tb[ETHTOOL_A_LINKMODES_LANES],
> > +					    "lanes value is invalid");
> > +			return -EINVAL;
> > +		}
> > +
> > +		/* If autoneg is off and lanes parameter is not supported by the
> > +		 * driver, return an error.
> > +		 */
> > +		if (!lsettings->autoneg &&
> > +		    !dev->ethtool_ops->cap_link_lanes_supported) {
> > +			NL_SET_ERR_MSG_ATTR(info->extack,
> > +					    tb[ETHTOOL_A_LINKMODES_LANES],
> > +					    "lanes configuration not supported by device");
> > +			return -EOPNOTSUPP;
> > +		}
> 
> This validation does not depend on the current settings at all, it's just input validation, it can be done before rtnl_lock is taken (in a
> new function).
> 
> You can move ethnl_validate_master_slave_cfg() to that function as well (as a cleanup before this patch).

Do you mean to move the ethnl_validate_master_slave_cfg() if from that function? Doesn't it depend on the current settings, as opposed to the supported lanes param that you wanted me to move as well?
Not sure I understand the second part of the request...

> 
> > +	} else if (!lsettings->autoneg) {
> > +		/* If autoneg is off and lanes parameter is not passed from user,
> > +		 * set the lanes parameter to UNKNOWN.
> > +		 */
> > +		ksettings->lanes = ETHTOOL_LANES_UNKNOWN;
> > +	}
> > +
> >  	ret = ethnl_update_bitset(ksettings->link_modes.advertising,
> >  				  __ETHTOOL_LINK_MODE_MASK_NBITS,
> >  				  tb[ETHTOOL_A_LINKMODES_OURS], link_mode_names,
Jakub Kicinski Jan. 25, 2021, 7:02 p.m. UTC | #4
On Mon, 25 Jan 2021 15:53:24 +0000 Danielle Ratson wrote:
> > > @@ -353,10 +358,39 @@ static int ethnl_update_linkmodes(struct
> > > genl_info *info, struct nlattr **tb,
> > >
> > >  	*mod = false;
> > >  	req_speed = tb[ETHTOOL_A_LINKMODES_SPEED];
> > > +	req_lanes = tb[ETHTOOL_A_LINKMODES_LANES];
> > >  	req_duplex = tb[ETHTOOL_A_LINKMODES_DUPLEX];
> > >
> > >  	ethnl_update_u8(&lsettings->autoneg, tb[ETHTOOL_A_LINKMODES_AUTONEG],
> > >  			mod);
> > > +
> > > +	if (req_lanes) {
> > > +		u32 lanes_cfg = nla_get_u32(tb[ETHTOOL_A_LINKMODES_LANES]);  
> > 
> > req_lanes == tb[ETHTOOL_A_LINKMODES_LANES], right?   
> 
> Yes, but req_lanes is a bool and doesn't fit to nla_get_u32. Do you want me to change the req_lanes type and name?

Ah, yes please.

> > Please use req_lanes variable where possible.
> >   
> > > +
> > > +		if (!is_power_of_2(lanes_cfg)) {
> > > +			NL_SET_ERR_MSG_ATTR(info->extack,
> > > +					    tb[ETHTOOL_A_LINKMODES_LANES],
> > > +					    "lanes value is invalid");
> > > +			return -EINVAL;
> > > +		}
> > > +
> > > +		/* If autoneg is off and lanes parameter is not supported by the
> > > +		 * driver, return an error.
> > > +		 */
> > > +		if (!lsettings->autoneg &&
> > > +		    !dev->ethtool_ops->cap_link_lanes_supported) {
> > > +			NL_SET_ERR_MSG_ATTR(info->extack,
> > > +					    tb[ETHTOOL_A_LINKMODES_LANES],
> > > +					    "lanes configuration not supported by device");
> > > +			return -EOPNOTSUPP;
> > > +		}  
> > 
> > This validation does not depend on the current settings at all,
> > it's just input validation, it can be done before rtnl_lock is
> > taken (in a new function).
> > 
> > You can move ethnl_validate_master_slave_cfg() to that function as
> > well (as a cleanup before this patch).  
> 
> Do you mean to move the ethnl_validate_master_slave_cfg() if from
> that function? 

Yes, to a separate helper.

> Doesn't it depend on the current settings, as opposed
> to the supported lanes param that you wanted me to move as well? Not
> sure I understand the second part of the request...

Sorry maybe I quoted a little too much context form the patch.

A helper like this:

static int ethnl_check_linkmodes(...)
{
	const struct nlattr *master_slave_cfg;
	
	master_slave_cfg = tb[ETHTOOL_A_LINKMODES_MASTER_SLAVE_CFG];
	if (master_slave_cfg && 
	    !ethnl_validate_master_slave_cfg(nla_get_u8(master_slave_cfg))) {
		NL_SET_ERR_MSG_ATTR(info->extack, master_slave_cfg,
				    "master/slave value is invalid");
		return -EOPNOTSUPP;
	}

	lanes_cfg = ...
	if (!is_power_of_2(...lanes_cfg)) {
		...
		return -EINVAL;
	}

	return 0;
}
 
Which you can call before the device reference is taken:

 int ethnl_set_linkmodes(struct sk_buff *skb, struct genl_info *info)
 {
 	struct ethtool_link_ksettings ksettings = {};
 	struct ethnl_req_info req_info = {};
 	struct nlattr **tb = info->attrs;
 	struct net_device *dev;
 	bool mod = false;
 	int ret;
 
+	ret = ethnl_check_linkmodes(tb);
+	if (ret)
+		return ret;
 
 	ret = ethnl_parse_header_dev_get(&req_info,
 					 tb[ETHTOOL_A_LINKMODES_HEADER],
 					 genl_info_net(info), info->extack,
 					 true);
 	if (ret < 0)
 		return ret;


But please make sure that you move the master_slave_cfg check in a
separate patch.
Danielle Ratson Feb. 2, 2021, 6:08 p.m. UTC | #5
> -----Original Message-----
> From: Edwin Peer <edwin.peer@broadcom.com>
> Sent: Thursday, January 21, 2021 12:36 AM
> To: Danielle Ratson <danieller@nvidia.com>
> Cc: netdev <netdev@vger.kernel.org>; David S . Miller <davem@davemloft.net>; Jakub Kicinski <kuba@kernel.org>; Jiri Pirko
> <jiri@nvidia.com>; Andrew Lunn <andrew@lunn.ch>; f.fainelli@gmail.com; Michal Kubecek <mkubecek@suse.cz>; mlxsw
> <mlxsw@nvidia.com>; Ido Schimmel <idosch@nvidia.com>
> Subject: Re: [PATCH net-next v3 1/7] ethtool: Extend link modes settings uAPI with lanes
> 
> On Wed, Jan 20, 2021 at 3:21 AM Danielle Ratson <danieller@nvidia.com> wrote:
> 
> > -#define __DEFINE_LINK_MODE_PARAMS(_speed, _type, _duplex) \
> > +#define __DEFINE_LINK_MODE_PARAMS(_speed, _type, _lanes, _duplex) \
> >         [ETHTOOL_LINK_MODE(_speed, _type, _duplex)] = { \
> >                 .speed  = SPEED_ ## _speed, \
> > +               .lanes  = _lanes, \
> >                 .duplex = __DUPLEX_ ## _duplex \
> >         }
> 
> What about:
> 
> #define __DECLARE_LINK_MODE_LANES(_type, _lanes)        \
> static const u32 __LINK_MODE_LANES_ ## _type = _lanes;
> 
> #define __DECLARE_LINK_MODE_LANES_ALL(_type)            \
> __DECLARE_LINK_MODE_LANES(_type, 1)             \
> __DECLARE_LINK_MODE_LANES(_type ## 2, 2)        \
> __DECLARE_LINK_MODE_LANES(_type ## 4, 4)        \
> __DECLARE_LINK_MODE_LANES(_type ## 8, 8)
> 
> __DECLARE_LINK_MODE_LANES_ALL(CR)
> __DECLARE_LINK_MODE_LANES_ALL(DR)
> __DECLARE_LINK_MODE_LANES_ALL(ER)
> __DECLARE_LINK_MODE_LANES_ALL(KR)
> __DECLARE_LINK_MODE_LANES(KX, 1)
> __DECLARE_LINK_MODE_LANES(KX4, 4)
> __DECLARE_LINK_MODE_LANES_ALL(LR)
> __DECLARE_LINK_MODE_LANES(LR_ER_FR, 1)
> __DECLARE_LINK_MODE_LANES(LR2_ER2_FR2, 2)
> __DECLARE_LINK_MODE_LANES(LR4_ER4_FR4, 4)
> __DECLARE_LINK_MODE_LANES(LR8_ER8_FR8, 8)
> __DECLARE_LINK_MODE_LANES(LRM, 1)
> __DECLARE_LINK_MODE_LANES(MLD2, 2);
> __DECLARE_LINK_MODE_LANES_ALL(SR);
> __DECLARE_LINK_MODE_LANES(T, 1)
> __DECLARE_LINK_MODE_LANES(X, 1)
> 
> #define __DEFINE_LINK_MODE_PARAMS(_speed, _type, _duplex)
>          [ETHTOOL_LINK_MODE(_speed, _type, _duplex)] = { \
>                  .speed  = SPEED_ ## _speed, \
>                  .lanes  = __LINK_MODE_LANES ## _type, \
> 
> instead of specifying lanes for each link mode defined below?
> 
> Regards,
> Edwin Peer

Thanks for the advice, due to some checkpatch issues, I used your suggestion with a small change.
Danielle

> 
> >  static const struct link_mode_info link_mode_params[] = {
> > -       __DEFINE_LINK_MODE_PARAMS(10, T, Half),
> > -       __DEFINE_LINK_MODE_PARAMS(10, T, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(100, T, Half),
> > -       __DEFINE_LINK_MODE_PARAMS(100, T, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(1000, T, Half),
> > -       __DEFINE_LINK_MODE_PARAMS(1000, T, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(10, T, 1, Half),
> > +       __DEFINE_LINK_MODE_PARAMS(10, T, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(100, T, 1, Half),
> > +       __DEFINE_LINK_MODE_PARAMS(100, T, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(1000, T, 1, Half),
> > +       __DEFINE_LINK_MODE_PARAMS(1000, T, 1, Full),
> >         __DEFINE_SPECIAL_MODE_PARAMS(Autoneg),
> >         __DEFINE_SPECIAL_MODE_PARAMS(TP),
> >         __DEFINE_SPECIAL_MODE_PARAMS(AUI),
> >         __DEFINE_SPECIAL_MODE_PARAMS(MII),
> >         __DEFINE_SPECIAL_MODE_PARAMS(FIBRE),
> >         __DEFINE_SPECIAL_MODE_PARAMS(BNC),
> > -       __DEFINE_LINK_MODE_PARAMS(10000, T, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(10000, T, 1, Full),
> >         __DEFINE_SPECIAL_MODE_PARAMS(Pause),
> >         __DEFINE_SPECIAL_MODE_PARAMS(Asym_Pause),
> > -       __DEFINE_LINK_MODE_PARAMS(2500, X, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(2500, X, 1, Full),
> >         __DEFINE_SPECIAL_MODE_PARAMS(Backplane),
> > -       __DEFINE_LINK_MODE_PARAMS(1000, KX, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(10000, KX4, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(10000, KR, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(1000, KX, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(10000, KX4, 4, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(10000, KR, 1, Full),
> >         [ETHTOOL_LINK_MODE_10000baseR_FEC_BIT] = {
> >                 .speed  = SPEED_10000,
> >                 .duplex = DUPLEX_FULL,
> >         },
> > -       __DEFINE_LINK_MODE_PARAMS(20000, MLD2, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(20000, KR2, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(40000, KR4, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(40000, CR4, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(40000, SR4, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(40000, LR4, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(56000, KR4, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(56000, CR4, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(56000, SR4, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(56000, LR4, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(25000, CR, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(25000, KR, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(25000, SR, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(50000, CR2, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(50000, KR2, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(100000, KR4, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(100000, SR4, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(100000, CR4, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(100000, LR4_ER4, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(50000, SR2, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(1000, X, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(10000, CR, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(10000, SR, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(10000, LR, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(10000, LRM, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(10000, ER, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(2500, T, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(5000, T, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(20000, MLD2, 2, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(20000, KR2, 2, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(40000, KR4, 4, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(40000, CR4, 4, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(40000, SR4, 4, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(40000, LR4, 4, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(56000, KR4, 4, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(56000, CR4, 4, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(56000, SR4, 4, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(56000, LR4, 4, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(25000, CR, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(25000, KR, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(25000, SR, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(50000, CR2, 2, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(50000, KR2, 2, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(100000, KR4, 4, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(100000, SR4, 4, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(100000, CR4, 4, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(100000, LR4_ER4, 4, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(50000, SR2, 2, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(1000, X, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(10000, CR, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(10000, SR, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(10000, LR, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(10000, LRM, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(10000, ER, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(2500, T, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(5000, T, 1, Full),
> >         __DEFINE_SPECIAL_MODE_PARAMS(FEC_NONE),
> >         __DEFINE_SPECIAL_MODE_PARAMS(FEC_RS),
> >         __DEFINE_SPECIAL_MODE_PARAMS(FEC_BASER),
> > -       __DEFINE_LINK_MODE_PARAMS(50000, KR, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(50000, SR, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(50000, CR, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(50000, LR_ER_FR, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(50000, DR, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(100000, KR2, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(100000, SR2, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(100000, CR2, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(100000, LR2_ER2_FR2, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(100000, DR2, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(200000, KR4, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(200000, SR4, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(200000, LR4_ER4_FR4, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(200000, DR4, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(200000, CR4, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(100, T1, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(1000, T1, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(400000, KR8, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(400000, SR8, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(400000, LR8_ER8_FR8, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(400000, DR8, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(400000, CR8, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(50000, KR, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(50000, SR, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(50000, CR, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(50000, LR_ER_FR, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(50000, DR, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(100000, KR2, 2, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(100000, SR2, 2, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(100000, CR2, 2, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(100000, LR2_ER2_FR2, 2, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(100000, DR2, 2, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(200000, KR4, 4, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(200000, SR4, 4, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(200000, LR4_ER4_FR4, 4, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(200000, DR4, 4, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(200000, CR4, 4, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(100, T1, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(1000, T1, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(400000, KR8, 8, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(400000, SR8, 8, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(400000, LR8_ER8_FR8, 8, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(400000, DR8, 8, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(400000, CR8, 8, Full),
> >         __DEFINE_SPECIAL_MODE_PARAMS(FEC_LLRS),
> > -       __DEFINE_LINK_MODE_PARAMS(100000, KR, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(100000, SR, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(100000, LR_ER_FR, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(100000, DR, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(100000, CR, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(200000, KR2, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(200000, SR2, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(200000, LR2_ER2_FR2, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(200000, DR2, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(200000, CR2, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(400000, KR4, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(400000, SR4, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(400000, LR4_ER4_FR4, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(400000, DR4, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(400000, CR4, Full),
> > -       __DEFINE_LINK_MODE_PARAMS(100, FX, Half),
> > -       __DEFINE_LINK_MODE_PARAMS(100, FX, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(100000, KR, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(100000, SR, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(100000, LR_ER_FR, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(100000, DR, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(100000, CR, 1, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(200000, KR2, 2, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(200000, SR2, 2, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(200000, LR2_ER2_FR2, 2, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(200000, DR2, 2, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(200000, CR2, 2, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(400000, KR4, 4, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(400000, SR4, 4, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(400000, LR4_ER4_FR4, 4, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(400000, DR4, 4, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(400000, CR4, 4, Full),
> > +       __DEFINE_LINK_MODE_PARAMS(100, FX, 1, Half),
> > +       __DEFINE_LINK_MODE_PARAMS(100, FX, 1, Full),
> >  };
diff mbox series

Patch

diff --git a/Documentation/networking/ethtool-netlink.rst b/Documentation/networking/ethtool-netlink.rst
index 30b98245979f..05073482db05 100644
--- a/Documentation/networking/ethtool-netlink.rst
+++ b/Documentation/networking/ethtool-netlink.rst
@@ -431,16 +431,17 @@  Request contents:
   ``ETHTOOL_A_LINKMODES_SPEED``               u32     link speed (Mb/s)
   ``ETHTOOL_A_LINKMODES_DUPLEX``              u8      duplex mode
   ``ETHTOOL_A_LINKMODES_MASTER_SLAVE_CFG``    u8      Master/slave port mode
+  ``ETHTOOL_A_LINKMODES_LANES``               u32     lanes
   ==========================================  ======  ==========================
 
 ``ETHTOOL_A_LINKMODES_OURS`` bit set allows setting advertised link modes. If
 autonegotiation is on (either set now or kept from before), advertised modes
 are not changed (no ``ETHTOOL_A_LINKMODES_OURS`` attribute) and at least one
-of speed and duplex is specified, kernel adjusts advertised modes to all
-supported modes matching speed, duplex or both (whatever is specified). This
-autoselection is done on ethtool side with ioctl interface, netlink interface
-is supposed to allow requesting changes without knowing what exactly kernel
-supports.
+of speed, duplex and lanes is specified, kernel adjusts advertised modes to all
+supported modes matching speed, duplex, lanes or all (whatever is specified).
+This autoselection is done on ethtool side with ioctl interface, netlink
+interface is supposed to allow requesting changes without knowing what exactly
+kernel supports.
 
 
 LINKSTATE_GET
diff --git a/include/linux/ethtool.h b/include/linux/ethtool.h
index e3da25b51ae4..1ab13c5dfb2f 100644
--- a/include/linux/ethtool.h
+++ b/include/linux/ethtool.h
@@ -128,6 +128,7 @@  struct ethtool_link_ksettings {
 		__ETHTOOL_DECLARE_LINK_MODE_MASK(advertising);
 		__ETHTOOL_DECLARE_LINK_MODE_MASK(lp_advertising);
 	} link_modes;
+	u32	lanes;
 };
 
 /**
@@ -265,6 +266,8 @@  struct ethtool_pause_stats {
 
 /**
  * struct ethtool_ops - optional netdev operations
+ * @cap_link_lanes_supported: indicates if the driver supports lanes
+ *	parameter.
  * @supported_coalesce_params: supported types of interrupt coalescing.
  * @get_drvinfo: Report driver/device information.  Should only set the
  *	@driver, @version, @fw_version and @bus_info fields.  If not
@@ -420,6 +423,7 @@  struct ethtool_pause_stats {
  * of the generic netdev features interface.
  */
 struct ethtool_ops {
+	u32     cap_link_lanes_supported:1;
 	u32	supported_coalesce_params;
 	void	(*get_drvinfo)(struct net_device *, struct ethtool_drvinfo *);
 	int	(*get_regs_len)(struct net_device *);
diff --git a/include/uapi/linux/ethtool.h b/include/uapi/linux/ethtool.h
index cde753bb2093..80edae2c24f7 100644
--- a/include/uapi/linux/ethtool.h
+++ b/include/uapi/linux/ethtool.h
@@ -1738,6 +1738,8 @@  static inline int ethtool_validate_speed(__u32 speed)
 	return speed <= INT_MAX || speed == (__u32)SPEED_UNKNOWN;
 }
 
+#define ETHTOOL_LANES_UNKNOWN		0
+
 /* Duplex, half or full. */
 #define DUPLEX_HALF		0x00
 #define DUPLEX_FULL		0x01
diff --git a/include/uapi/linux/ethtool_netlink.h b/include/uapi/linux/ethtool_netlink.h
index e2bf36e6964b..a286635ac9b8 100644
--- a/include/uapi/linux/ethtool_netlink.h
+++ b/include/uapi/linux/ethtool_netlink.h
@@ -227,6 +227,7 @@  enum {
 	ETHTOOL_A_LINKMODES_DUPLEX,		/* u8 */
 	ETHTOOL_A_LINKMODES_MASTER_SLAVE_CFG,	/* u8 */
 	ETHTOOL_A_LINKMODES_MASTER_SLAVE_STATE,	/* u8 */
+	ETHTOOL_A_LINKMODES_LANES,		/* u32 */
 
 	/* add new constants above here */
 	__ETHTOOL_A_LINKMODES_CNT,
diff --git a/net/ethtool/linkmodes.c b/net/ethtool/linkmodes.c
index c5bcb9abc8b9..fb7d73250864 100644
--- a/net/ethtool/linkmodes.c
+++ b/net/ethtool/linkmodes.c
@@ -152,12 +152,14 @@  const struct ethnl_request_ops ethnl_linkmodes_request_ops = {
 
 struct link_mode_info {
 	int				speed;
+	u32				lanes;
 	u8				duplex;
 };
 
-#define __DEFINE_LINK_MODE_PARAMS(_speed, _type, _duplex) \
+#define __DEFINE_LINK_MODE_PARAMS(_speed, _type, _lanes, _duplex) \
 	[ETHTOOL_LINK_MODE(_speed, _type, _duplex)] = { \
 		.speed	= SPEED_ ## _speed, \
+		.lanes	= _lanes, \
 		.duplex	= __DUPLEX_ ## _duplex \
 	}
 #define __DUPLEX_Half DUPLEX_HALF
@@ -165,105 +167,106 @@  struct link_mode_info {
 #define __DEFINE_SPECIAL_MODE_PARAMS(_mode) \
 	[ETHTOOL_LINK_MODE_ ## _mode ## _BIT] = { \
 		.speed	= SPEED_UNKNOWN, \
+		.lanes	= ETHTOOL_LANES_UNKNOWN, \
 		.duplex	= DUPLEX_UNKNOWN, \
 	}
 
 static const struct link_mode_info link_mode_params[] = {
-	__DEFINE_LINK_MODE_PARAMS(10, T, Half),
-	__DEFINE_LINK_MODE_PARAMS(10, T, Full),
-	__DEFINE_LINK_MODE_PARAMS(100, T, Half),
-	__DEFINE_LINK_MODE_PARAMS(100, T, Full),
-	__DEFINE_LINK_MODE_PARAMS(1000, T, Half),
-	__DEFINE_LINK_MODE_PARAMS(1000, T, Full),
+	__DEFINE_LINK_MODE_PARAMS(10, T, 1, Half),
+	__DEFINE_LINK_MODE_PARAMS(10, T, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(100, T, 1, Half),
+	__DEFINE_LINK_MODE_PARAMS(100, T, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(1000, T, 1, Half),
+	__DEFINE_LINK_MODE_PARAMS(1000, T, 1, Full),
 	__DEFINE_SPECIAL_MODE_PARAMS(Autoneg),
 	__DEFINE_SPECIAL_MODE_PARAMS(TP),
 	__DEFINE_SPECIAL_MODE_PARAMS(AUI),
 	__DEFINE_SPECIAL_MODE_PARAMS(MII),
 	__DEFINE_SPECIAL_MODE_PARAMS(FIBRE),
 	__DEFINE_SPECIAL_MODE_PARAMS(BNC),
-	__DEFINE_LINK_MODE_PARAMS(10000, T, Full),
+	__DEFINE_LINK_MODE_PARAMS(10000, T, 1, Full),
 	__DEFINE_SPECIAL_MODE_PARAMS(Pause),
 	__DEFINE_SPECIAL_MODE_PARAMS(Asym_Pause),
-	__DEFINE_LINK_MODE_PARAMS(2500, X, Full),
+	__DEFINE_LINK_MODE_PARAMS(2500, X, 1, Full),
 	__DEFINE_SPECIAL_MODE_PARAMS(Backplane),
-	__DEFINE_LINK_MODE_PARAMS(1000, KX, Full),
-	__DEFINE_LINK_MODE_PARAMS(10000, KX4, Full),
-	__DEFINE_LINK_MODE_PARAMS(10000, KR, Full),
+	__DEFINE_LINK_MODE_PARAMS(1000, KX, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(10000, KX4, 4, Full),
+	__DEFINE_LINK_MODE_PARAMS(10000, KR, 1, Full),
 	[ETHTOOL_LINK_MODE_10000baseR_FEC_BIT] = {
 		.speed	= SPEED_10000,
 		.duplex = DUPLEX_FULL,
 	},
-	__DEFINE_LINK_MODE_PARAMS(20000, MLD2, Full),
-	__DEFINE_LINK_MODE_PARAMS(20000, KR2, Full),
-	__DEFINE_LINK_MODE_PARAMS(40000, KR4, Full),
-	__DEFINE_LINK_MODE_PARAMS(40000, CR4, Full),
-	__DEFINE_LINK_MODE_PARAMS(40000, SR4, Full),
-	__DEFINE_LINK_MODE_PARAMS(40000, LR4, Full),
-	__DEFINE_LINK_MODE_PARAMS(56000, KR4, Full),
-	__DEFINE_LINK_MODE_PARAMS(56000, CR4, Full),
-	__DEFINE_LINK_MODE_PARAMS(56000, SR4, Full),
-	__DEFINE_LINK_MODE_PARAMS(56000, LR4, Full),
-	__DEFINE_LINK_MODE_PARAMS(25000, CR, Full),
-	__DEFINE_LINK_MODE_PARAMS(25000, KR, Full),
-	__DEFINE_LINK_MODE_PARAMS(25000, SR, Full),
-	__DEFINE_LINK_MODE_PARAMS(50000, CR2, Full),
-	__DEFINE_LINK_MODE_PARAMS(50000, KR2, Full),
-	__DEFINE_LINK_MODE_PARAMS(100000, KR4, Full),
-	__DEFINE_LINK_MODE_PARAMS(100000, SR4, Full),
-	__DEFINE_LINK_MODE_PARAMS(100000, CR4, Full),
-	__DEFINE_LINK_MODE_PARAMS(100000, LR4_ER4, Full),
-	__DEFINE_LINK_MODE_PARAMS(50000, SR2, Full),
-	__DEFINE_LINK_MODE_PARAMS(1000, X, Full),
-	__DEFINE_LINK_MODE_PARAMS(10000, CR, Full),
-	__DEFINE_LINK_MODE_PARAMS(10000, SR, Full),
-	__DEFINE_LINK_MODE_PARAMS(10000, LR, Full),
-	__DEFINE_LINK_MODE_PARAMS(10000, LRM, Full),
-	__DEFINE_LINK_MODE_PARAMS(10000, ER, Full),
-	__DEFINE_LINK_MODE_PARAMS(2500, T, Full),
-	__DEFINE_LINK_MODE_PARAMS(5000, T, Full),
+	__DEFINE_LINK_MODE_PARAMS(20000, MLD2, 2, Full),
+	__DEFINE_LINK_MODE_PARAMS(20000, KR2, 2, Full),
+	__DEFINE_LINK_MODE_PARAMS(40000, KR4, 4, Full),
+	__DEFINE_LINK_MODE_PARAMS(40000, CR4, 4, Full),
+	__DEFINE_LINK_MODE_PARAMS(40000, SR4, 4, Full),
+	__DEFINE_LINK_MODE_PARAMS(40000, LR4, 4, Full),
+	__DEFINE_LINK_MODE_PARAMS(56000, KR4, 4, Full),
+	__DEFINE_LINK_MODE_PARAMS(56000, CR4, 4, Full),
+	__DEFINE_LINK_MODE_PARAMS(56000, SR4, 4, Full),
+	__DEFINE_LINK_MODE_PARAMS(56000, LR4, 4, Full),
+	__DEFINE_LINK_MODE_PARAMS(25000, CR, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(25000, KR, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(25000, SR, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(50000, CR2, 2, Full),
+	__DEFINE_LINK_MODE_PARAMS(50000, KR2, 2, Full),
+	__DEFINE_LINK_MODE_PARAMS(100000, KR4, 4, Full),
+	__DEFINE_LINK_MODE_PARAMS(100000, SR4, 4, Full),
+	__DEFINE_LINK_MODE_PARAMS(100000, CR4, 4, Full),
+	__DEFINE_LINK_MODE_PARAMS(100000, LR4_ER4, 4, Full),
+	__DEFINE_LINK_MODE_PARAMS(50000, SR2, 2, Full),
+	__DEFINE_LINK_MODE_PARAMS(1000, X, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(10000, CR, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(10000, SR, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(10000, LR, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(10000, LRM, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(10000, ER, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(2500, T, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(5000, T, 1, Full),
 	__DEFINE_SPECIAL_MODE_PARAMS(FEC_NONE),
 	__DEFINE_SPECIAL_MODE_PARAMS(FEC_RS),
 	__DEFINE_SPECIAL_MODE_PARAMS(FEC_BASER),
-	__DEFINE_LINK_MODE_PARAMS(50000, KR, Full),
-	__DEFINE_LINK_MODE_PARAMS(50000, SR, Full),
-	__DEFINE_LINK_MODE_PARAMS(50000, CR, Full),
-	__DEFINE_LINK_MODE_PARAMS(50000, LR_ER_FR, Full),
-	__DEFINE_LINK_MODE_PARAMS(50000, DR, Full),
-	__DEFINE_LINK_MODE_PARAMS(100000, KR2, Full),
-	__DEFINE_LINK_MODE_PARAMS(100000, SR2, Full),
-	__DEFINE_LINK_MODE_PARAMS(100000, CR2, Full),
-	__DEFINE_LINK_MODE_PARAMS(100000, LR2_ER2_FR2, Full),
-	__DEFINE_LINK_MODE_PARAMS(100000, DR2, Full),
-	__DEFINE_LINK_MODE_PARAMS(200000, KR4, Full),
-	__DEFINE_LINK_MODE_PARAMS(200000, SR4, Full),
-	__DEFINE_LINK_MODE_PARAMS(200000, LR4_ER4_FR4, Full),
-	__DEFINE_LINK_MODE_PARAMS(200000, DR4, Full),
-	__DEFINE_LINK_MODE_PARAMS(200000, CR4, Full),
-	__DEFINE_LINK_MODE_PARAMS(100, T1, Full),
-	__DEFINE_LINK_MODE_PARAMS(1000, T1, Full),
-	__DEFINE_LINK_MODE_PARAMS(400000, KR8, Full),
-	__DEFINE_LINK_MODE_PARAMS(400000, SR8, Full),
-	__DEFINE_LINK_MODE_PARAMS(400000, LR8_ER8_FR8, Full),
-	__DEFINE_LINK_MODE_PARAMS(400000, DR8, Full),
-	__DEFINE_LINK_MODE_PARAMS(400000, CR8, Full),
+	__DEFINE_LINK_MODE_PARAMS(50000, KR, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(50000, SR, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(50000, CR, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(50000, LR_ER_FR, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(50000, DR, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(100000, KR2, 2, Full),
+	__DEFINE_LINK_MODE_PARAMS(100000, SR2, 2, Full),
+	__DEFINE_LINK_MODE_PARAMS(100000, CR2, 2, Full),
+	__DEFINE_LINK_MODE_PARAMS(100000, LR2_ER2_FR2, 2, Full),
+	__DEFINE_LINK_MODE_PARAMS(100000, DR2, 2, Full),
+	__DEFINE_LINK_MODE_PARAMS(200000, KR4, 4, Full),
+	__DEFINE_LINK_MODE_PARAMS(200000, SR4, 4, Full),
+	__DEFINE_LINK_MODE_PARAMS(200000, LR4_ER4_FR4, 4, Full),
+	__DEFINE_LINK_MODE_PARAMS(200000, DR4, 4, Full),
+	__DEFINE_LINK_MODE_PARAMS(200000, CR4, 4, Full),
+	__DEFINE_LINK_MODE_PARAMS(100, T1, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(1000, T1, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(400000, KR8, 8, Full),
+	__DEFINE_LINK_MODE_PARAMS(400000, SR8, 8, Full),
+	__DEFINE_LINK_MODE_PARAMS(400000, LR8_ER8_FR8, 8, Full),
+	__DEFINE_LINK_MODE_PARAMS(400000, DR8, 8, Full),
+	__DEFINE_LINK_MODE_PARAMS(400000, CR8, 8, Full),
 	__DEFINE_SPECIAL_MODE_PARAMS(FEC_LLRS),
-	__DEFINE_LINK_MODE_PARAMS(100000, KR, Full),
-	__DEFINE_LINK_MODE_PARAMS(100000, SR, Full),
-	__DEFINE_LINK_MODE_PARAMS(100000, LR_ER_FR, Full),
-	__DEFINE_LINK_MODE_PARAMS(100000, DR, Full),
-	__DEFINE_LINK_MODE_PARAMS(100000, CR, Full),
-	__DEFINE_LINK_MODE_PARAMS(200000, KR2, Full),
-	__DEFINE_LINK_MODE_PARAMS(200000, SR2, Full),
-	__DEFINE_LINK_MODE_PARAMS(200000, LR2_ER2_FR2, Full),
-	__DEFINE_LINK_MODE_PARAMS(200000, DR2, Full),
-	__DEFINE_LINK_MODE_PARAMS(200000, CR2, Full),
-	__DEFINE_LINK_MODE_PARAMS(400000, KR4, Full),
-	__DEFINE_LINK_MODE_PARAMS(400000, SR4, Full),
-	__DEFINE_LINK_MODE_PARAMS(400000, LR4_ER4_FR4, Full),
-	__DEFINE_LINK_MODE_PARAMS(400000, DR4, Full),
-	__DEFINE_LINK_MODE_PARAMS(400000, CR4, Full),
-	__DEFINE_LINK_MODE_PARAMS(100, FX, Half),
-	__DEFINE_LINK_MODE_PARAMS(100, FX, Full),
+	__DEFINE_LINK_MODE_PARAMS(100000, KR, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(100000, SR, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(100000, LR_ER_FR, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(100000, DR, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(100000, CR, 1, Full),
+	__DEFINE_LINK_MODE_PARAMS(200000, KR2, 2, Full),
+	__DEFINE_LINK_MODE_PARAMS(200000, SR2, 2, Full),
+	__DEFINE_LINK_MODE_PARAMS(200000, LR2_ER2_FR2, 2, Full),
+	__DEFINE_LINK_MODE_PARAMS(200000, DR2, 2, Full),
+	__DEFINE_LINK_MODE_PARAMS(200000, CR2, 2, Full),
+	__DEFINE_LINK_MODE_PARAMS(400000, KR4, 4, Full),
+	__DEFINE_LINK_MODE_PARAMS(400000, SR4, 4, Full),
+	__DEFINE_LINK_MODE_PARAMS(400000, LR4_ER4_FR4, 4, Full),
+	__DEFINE_LINK_MODE_PARAMS(400000, DR4, 4, Full),
+	__DEFINE_LINK_MODE_PARAMS(400000, CR4, 4, Full),
+	__DEFINE_LINK_MODE_PARAMS(100, FX, 1, Half),
+	__DEFINE_LINK_MODE_PARAMS(100, FX, 1, Full),
 };
 
 const struct nla_policy ethnl_linkmodes_set_policy[] = {
@@ -274,16 +277,17 @@  const struct nla_policy ethnl_linkmodes_set_policy[] = {
 	[ETHTOOL_A_LINKMODES_SPEED]		= { .type = NLA_U32 },
 	[ETHTOOL_A_LINKMODES_DUPLEX]		= { .type = NLA_U8 },
 	[ETHTOOL_A_LINKMODES_MASTER_SLAVE_CFG]	= { .type = NLA_U8 },
+	[ETHTOOL_A_LINKMODES_LANES]		= NLA_POLICY_RANGE(NLA_U32, 1, 8),
 };
 
-/* Set advertised link modes to all supported modes matching requested speed
- * and duplex values. Called when autonegotiation is on, speed or duplex is
- * requested but no link mode change. This is done in userspace with ioctl()
- * interface, move it into kernel for netlink.
+/* Set advertised link modes to all supported modes matching requested speed,
+ * lanes and duplex values. Called when autonegotiation is on, speed, lanes or
+ * duplex is requested but no link mode change. This is done in userspace with
+ * ioctl() interface, move it into kernel for netlink.
  * Returns true if advertised modes bitmap was modified.
  */
 static bool ethnl_auto_linkmodes(struct ethtool_link_ksettings *ksettings,
-				 bool req_speed, bool req_duplex)
+				 bool req_speed, bool req_lanes, bool req_duplex)
 {
 	unsigned long *advertising = ksettings->link_modes.advertising;
 	unsigned long *supported = ksettings->link_modes.supported;
@@ -302,6 +306,7 @@  static bool ethnl_auto_linkmodes(struct ethtool_link_ksettings *ksettings,
 			continue;
 		if (test_bit(i, supported) &&
 		    (!req_speed || info->speed == ksettings->base.speed) &&
+		    (!req_lanes || info->lanes == ksettings->lanes) &&
 		    (!req_duplex || info->duplex == ksettings->base.duplex))
 			set_bit(i, advertising);
 		else
@@ -327,10 +332,10 @@  static bool ethnl_validate_master_slave_cfg(u8 cfg)
 
 static int ethnl_update_linkmodes(struct genl_info *info, struct nlattr **tb,
 				  struct ethtool_link_ksettings *ksettings,
-				  bool *mod)
+				  bool *mod, const struct net_device *dev)
 {
 	struct ethtool_link_settings *lsettings = &ksettings->base;
-	bool req_speed, req_duplex;
+	bool req_speed, req_lanes, req_duplex;
 	const struct nlattr *master_slave_cfg;
 	int ret;
 
@@ -353,10 +358,39 @@  static int ethnl_update_linkmodes(struct genl_info *info, struct nlattr **tb,
 
 	*mod = false;
 	req_speed = tb[ETHTOOL_A_LINKMODES_SPEED];
+	req_lanes = tb[ETHTOOL_A_LINKMODES_LANES];
 	req_duplex = tb[ETHTOOL_A_LINKMODES_DUPLEX];
 
 	ethnl_update_u8(&lsettings->autoneg, tb[ETHTOOL_A_LINKMODES_AUTONEG],
 			mod);
+
+	if (req_lanes) {
+		u32 lanes_cfg = nla_get_u32(tb[ETHTOOL_A_LINKMODES_LANES]);
+
+		if (!is_power_of_2(lanes_cfg)) {
+			NL_SET_ERR_MSG_ATTR(info->extack,
+					    tb[ETHTOOL_A_LINKMODES_LANES],
+					    "lanes value is invalid");
+			return -EINVAL;
+		}
+
+		/* If autoneg is off and lanes parameter is not supported by the
+		 * driver, return an error.
+		 */
+		if (!lsettings->autoneg &&
+		    !dev->ethtool_ops->cap_link_lanes_supported) {
+			NL_SET_ERR_MSG_ATTR(info->extack,
+					    tb[ETHTOOL_A_LINKMODES_LANES],
+					    "lanes configuration not supported by device");
+			return -EOPNOTSUPP;
+		}
+	} else if (!lsettings->autoneg) {
+		/* If autoneg is off and lanes parameter is not passed from user,
+		 * set the lanes parameter to UNKNOWN.
+		 */
+		ksettings->lanes = ETHTOOL_LANES_UNKNOWN;
+	}
+
 	ret = ethnl_update_bitset(ksettings->link_modes.advertising,
 				  __ETHTOOL_LINK_MODE_MASK_NBITS,
 				  tb[ETHTOOL_A_LINKMODES_OURS], link_mode_names,
@@ -365,13 +399,15 @@  static int ethnl_update_linkmodes(struct genl_info *info, struct nlattr **tb,
 		return ret;
 	ethnl_update_u32(&lsettings->speed, tb[ETHTOOL_A_LINKMODES_SPEED],
 			 mod);
+	ethnl_update_u32(&ksettings->lanes, tb[ETHTOOL_A_LINKMODES_LANES],
+			 mod);
 	ethnl_update_u8(&lsettings->duplex, tb[ETHTOOL_A_LINKMODES_DUPLEX],
 			mod);
 	ethnl_update_u8(&lsettings->master_slave_cfg, master_slave_cfg, mod);
 
 	if (!tb[ETHTOOL_A_LINKMODES_OURS] && lsettings->autoneg &&
-	    (req_speed || req_duplex) &&
-	    ethnl_auto_linkmodes(ksettings, req_speed, req_duplex))
+	    (req_speed || req_lanes || req_duplex) &&
+	    ethnl_auto_linkmodes(ksettings, req_speed, req_lanes, req_duplex))
 		*mod = true;
 
 	return 0;
@@ -409,7 +445,7 @@  int ethnl_set_linkmodes(struct sk_buff *skb, struct genl_info *info)
 		goto out_ops;
 	}
 
-	ret = ethnl_update_linkmodes(info, tb, &ksettings, &mod);
+	ret = ethnl_update_linkmodes(info, tb, &ksettings, &mod, dev);
 	if (ret < 0)
 		goto out_ops;
 
diff --git a/net/ethtool/netlink.h b/net/ethtool/netlink.h
index d8efec516d86..6eabd58d81bf 100644
--- a/net/ethtool/netlink.h
+++ b/net/ethtool/netlink.h
@@ -351,7 +351,7 @@  extern const struct nla_policy ethnl_strset_get_policy[ETHTOOL_A_STRSET_COUNTS_O
 extern const struct nla_policy ethnl_linkinfo_get_policy[ETHTOOL_A_LINKINFO_HEADER + 1];
 extern const struct nla_policy ethnl_linkinfo_set_policy[ETHTOOL_A_LINKINFO_TP_MDIX_CTRL + 1];
 extern const struct nla_policy ethnl_linkmodes_get_policy[ETHTOOL_A_LINKMODES_HEADER + 1];
-extern const struct nla_policy ethnl_linkmodes_set_policy[ETHTOOL_A_LINKMODES_MASTER_SLAVE_CFG + 1];
+extern const struct nla_policy ethnl_linkmodes_set_policy[ETHTOOL_A_LINKMODES_LANES + 1];
 extern const struct nla_policy ethnl_linkstate_get_policy[ETHTOOL_A_LINKSTATE_HEADER + 1];
 extern const struct nla_policy ethnl_debug_get_policy[ETHTOOL_A_DEBUG_HEADER + 1];
 extern const struct nla_policy ethnl_debug_set_policy[ETHTOOL_A_DEBUG_MSGMASK + 1];