Message ID | 20210215061559.1187396-1-nathan@nathanrossi.com |
---|---|
State | New |
Headers | show |
Series | net: dsa: mv88e6xxx: prevent 2500BASEX mode override | expand |
On Mon, 15 Feb 2021 06:15:59 +0000 Nathan Rossi <nathan@nathanrossi.com> wrote: > diff --git a/drivers/net/dsa/mv88e6xxx/chip.c b/drivers/net/dsa/mv88e6xxx/chip.c > index 54aa942eed..5c52906b29 100644 > --- a/drivers/net/dsa/mv88e6xxx/chip.c > +++ b/drivers/net/dsa/mv88e6xxx/chip.c > @@ -650,6 +650,13 @@ static void mv88e6xxx_validate(struct dsa_switch *ds, int port, > if (chip->info->ops->phylink_validate) > chip->info->ops->phylink_validate(chip, port, mask, state); > > + /* Advertise 2500BASEX only if 1000BASEX is not configured, this > + * prevents phylink_helper_basex_speed from always overriding the > + * 1000BASEX mode since auto negotiation is always enabled. > + */ > + if (state->interface == PHY_INTERFACE_MODE_1000BASEX) > + phylink_clear(mask, 2500baseX_Full); > + I don't quite like this. This problem should be either solved in phylink_helper_basex_speed() or somewhere in the mv88e6xxx code, but near the call to phylink_helper_basex_speed(). Putting a solution to the behaviour of phylink_helper_basex_speed() it into the validate() method when phylink_helper_basex_speed() is called from a different place will complicate debugging in the future. If we start solving problems in this kind of way, the driver will become totally unreadable, IMO. Marek
On Tue, 16 Feb 2021 at 01:58, Marek Behun <marek.behun@nic.cz> wrote: > > On Mon, 15 Feb 2021 15:29:44 +0000 > Russell King - ARM Linux admin <linux@armlinux.org.uk> wrote: > > > On Mon, Feb 15, 2021 at 04:16:27PM +0100, Marek Behun wrote: > > > On Mon, 15 Feb 2021 14:57:57 +0000 > > > Russell King - ARM Linux admin <linux@armlinux.org.uk> wrote: > > > > > > > On Mon, Feb 15, 2021 at 02:47:56PM +0100, Marek Behun wrote: > > > > > On Mon, 15 Feb 2021 06:15:59 +0000 > > > > > Nathan Rossi <nathan@nathanrossi.com> wrote: > > > > > > > > > > > diff --git a/drivers/net/dsa/mv88e6xxx/chip.c b/drivers/net/dsa/mv88e6xxx/chip.c > > > > > > index 54aa942eed..5c52906b29 100644 > > > > > > --- a/drivers/net/dsa/mv88e6xxx/chip.c > > > > > > +++ b/drivers/net/dsa/mv88e6xxx/chip.c > > > > > > @@ -650,6 +650,13 @@ static void mv88e6xxx_validate(struct dsa_switch *ds, int port, > > > > > > if (chip->info->ops->phylink_validate) > > > > > > chip->info->ops->phylink_validate(chip, port, mask, state); > > > > > > > > > > > > + /* Advertise 2500BASEX only if 1000BASEX is not configured, this > > > > > > + * prevents phylink_helper_basex_speed from always overriding the > > > > > > + * 1000BASEX mode since auto negotiation is always enabled. > > > > > > + */ > > > > > > + if (state->interface == PHY_INTERFACE_MODE_1000BASEX) > > > > > > + phylink_clear(mask, 2500baseX_Full); > > > > > > + > > > > > > > > > > I don't quite like this. This problem should be either solved in > > > > > phylink_helper_basex_speed() or somewhere in the mv88e6xxx code, but near > > > > > the call to phylink_helper_basex_speed(). > > > > > > > > > > Putting a solution to the behaviour of phylink_helper_basex_speed() it > > > > > into the validate() method when phylink_helper_basex_speed() is called > > > > > from a different place will complicate debugging in the future. If > > > > > we start solving problems in this kind of way, the driver will become > > > > > totally unreadable, IMO. > > > > > > > > If we can't switch between 1000base-X and 2500base-X, then we should > > > > not be calling phylink_helper_basex_speed() - and only one of those > > > > two capabilities should be set in the validation callback. I thought > > > > there were DSA switches where we could program the CMODE to switch > > > > between these two... > > > > > > There are. At least Peridot, Topaz and Amethyst support switching > > > between these modes. But only on some ports. > > > > > > This problem happnes on Peridot X, I think. > > > > > > On Peridot X there are > > > - port 0: RGMII > > > - ports 9-10: capable of 1, 2.5 and 10G SerDes (10G via > > > XAUI/RXAUI, so multiple lanes) > > > - ports 1-8: with copper PHYs > > > - some of these can instead be set to use the unused SerDes lanes > > > of ports 9-10, but only in 1000base-x mode > > > > > > So the problem can happen if you set port 9 or 10 to only use one > > > SerDes lane, and use the spare lanes for the 1G ports. > > > On these ports 2500base-x is not supported, only 1000base-x (maybe > > > sgmii, I don't remember) > > > > It sounds like the modes are not reporting correctly then before calling > > phylink_helper_basex_speed(). If the port can only be used at 1000base-X, > > then it should not be allowing 2500base-X to be set prior to calling > > phylink_helper_basex_speed(). > > > > Hmm. It doesn't seem that way. Ports 1-8 only set support for 1000baseT > and 1000baseX. And for lower modes if state->interface is not 8023z. > > Nathan, what switch do you use and on which port does this happen? > After looking at this issue again in more depth, it seems I had managed to confuse myself with the uplinks and their actual modes. Specifically the fixed link uplink was being configured as a 1000base-x but should have been configured as sgmii phy-mode in the device tree. This resolves the issues associated with auto negotiation (not enabled for sgmii), on 88e6390 (port 9 uplink) and 88e6193x (port 0 uplink) switches. Sorry for the noise, and any confusion. However it should be noted that the issue mentioned in this patch might still be applicable if fixed 1000base-x links are expected to work. Thanks, Nathan
diff --git a/drivers/net/dsa/mv88e6xxx/chip.c b/drivers/net/dsa/mv88e6xxx/chip.c index 54aa942eed..5c52906b29 100644 --- a/drivers/net/dsa/mv88e6xxx/chip.c +++ b/drivers/net/dsa/mv88e6xxx/chip.c @@ -650,6 +650,13 @@ static void mv88e6xxx_validate(struct dsa_switch *ds, int port, if (chip->info->ops->phylink_validate) chip->info->ops->phylink_validate(chip, port, mask, state); + /* Advertise 2500BASEX only if 1000BASEX is not configured, this + * prevents phylink_helper_basex_speed from always overriding the + * 1000BASEX mode since auto negotiation is always enabled. + */ + if (state->interface == PHY_INTERFACE_MODE_1000BASEX) + phylink_clear(mask, 2500baseX_Full); + bitmap_and(supported, supported, mask, __ETHTOOL_LINK_MODE_MASK_NBITS); bitmap_and(state->advertising, state->advertising, mask, __ETHTOOL_LINK_MODE_MASK_NBITS);