[<prev] [next>] [<thread-prev] [day] [month] [year] [list]
Message-ID: <92ccb5be-efea-4dbf-ac87-a3415b0ed3dc@csgroup.eu>
Date: Wed, 17 Sep 2025 09:03:11 +0200
From: Christophe Leroy <christophe.leroy@...roup.eu>
To: Maxime Chevallier <maxime.chevallier@...tlin.com>, davem@...emloft.net
Cc: netdev@...r.kernel.org, linux-kernel@...r.kernel.org,
linux-arm-msm@...r.kernel.org, thomas.petazzoni@...tlin.com,
Andrew Lunn <andrew@...n.ch>, Jakub Kicinski <kuba@...nel.org>,
Eric Dumazet <edumazet@...gle.com>, Paolo Abeni <pabeni@...hat.com>,
Russell King <linux@...linux.org.uk>, linux-arm-kernel@...ts.infradead.org,
Herve Codina <herve.codina@...tlin.com>,
Florian Fainelli <f.fainelli@...il.com>,
Heiner Kallweit <hkallweit1@...il.com>,
Vladimir Oltean <vladimir.oltean@....com>,
Köry Maincent <kory.maincent@...tlin.com>,
Marek Behún <kabel@...nel.org>,
Oleksij Rempel <o.rempel@...gutronix.de>,
Nicolò Veronese <nicveronese@...il.com>,
Simon Horman <horms@...nel.org>, mwojtas@...omium.org,
Antoine Tenart <atenart@...nel.org>, devicetree@...r.kernel.org,
Conor Dooley <conor+dt@...nel.org>, Krzysztof Kozlowski
<krzk+dt@...nel.org>, Rob Herring <robh@...nel.org>,
Romain Gantois <romain.gantois@...tlin.com>,
Daniel Golle <daniel@...rotopia.org>,
Dimitri Fedrau <dimitri.fedrau@...bherr.com>
Subject: Re: [PATCH net-next v12 15/18] net: phy: qca807x: Support SFP through
phy_port interface
Le 09/09/2025 à 17:26, Maxime Chevallier a écrit :
> QCA8072/8075 may be used as combo-port PHYs, with Serdes (100/1000BaseX)
> and Copper interfaces. The PHY has the ability to read the configuration
> it's in. If the configuration indicates the PHY is in combo mode, allow
> registering up to 2 ports.
>
> Register a dedicated set of port ops to handle the serdes port, and rely
> on generic phylib SFP support for the SFP handling.
>
> Signed-off-by: Maxime Chevallier <maxime.chevallier@...tlin.com>
Reviewed-by: Christophe Leroy <christophe.leroy@...roup.eu>
> ---
> drivers/net/phy/qcom/qca807x.c | 73 ++++++++++++++--------------------
> 1 file changed, 30 insertions(+), 43 deletions(-)
>
> diff --git a/drivers/net/phy/qcom/qca807x.c b/drivers/net/phy/qcom/qca807x.c
> index 070dc8c00835..d8f1ce5a7128 100644
> --- a/drivers/net/phy/qcom/qca807x.c
> +++ b/drivers/net/phy/qcom/qca807x.c
> @@ -13,7 +13,7 @@
> #include <linux/phy.h>
> #include <linux/bitfield.h>
> #include <linux/gpio/driver.h>
> -#include <linux/sfp.h>
> +#include <linux/phy_port.h>
>
> #include "../phylib.h"
> #include "qcom.h"
> @@ -643,68 +643,54 @@ static int qca807x_phy_package_config_init_once(struct phy_device *phydev)
> return ret;
> }
>
> -static int qca807x_sfp_insert(void *upstream, const struct sfp_eeprom_id *id)
> +static int qca807x_configure_serdes(struct phy_port *port, bool enable,
> + phy_interface_t interface)
> {
> - struct phy_device *phydev = upstream;
> - __ETHTOOL_DECLARE_LINK_MODE_MASK(support) = { 0, };
> - phy_interface_t iface;
> + struct phy_device *phydev = port_phydev(port);
> int ret;
> - DECLARE_PHY_INTERFACE_MASK(interfaces);
>
> - sfp_parse_support(phydev->sfp_bus, id, support, interfaces);
> - iface = sfp_select_interface(phydev->sfp_bus, support);
> + if (!phydev)
> + return -ENODEV;
>
> - dev_info(&phydev->mdio.dev, "%s SFP module inserted\n", phy_modes(iface));
> -
> - switch (iface) {
> - case PHY_INTERFACE_MODE_1000BASEX:
> - case PHY_INTERFACE_MODE_100BASEX:
> + if (enable) {
> /* Set PHY mode to PSGMII combo (1/4 copper + combo ports) mode */
> ret = phy_modify(phydev,
> QCA807X_CHIP_CONFIGURATION,
> QCA807X_CHIP_CONFIGURATION_MODE_CFG_MASK,
> QCA807X_CHIP_CONFIGURATION_MODE_PSGMII_FIBER);
> + if (ret)
> + return ret;
> /* Enable fiber mode autodection (1000Base-X or 100Base-FX) */
> ret = phy_set_bits_mmd(phydev,
> MDIO_MMD_AN,
> QCA807X_MMD7_FIBER_MODE_AUTO_DETECTION,
> QCA807X_MMD7_FIBER_MODE_AUTO_DETECTION_EN);
> - /* Select fiber page */
> - ret = phy_clear_bits(phydev,
> - QCA807X_CHIP_CONFIGURATION,
> - QCA807X_BT_BX_REG_SEL);
> -
> - phydev->port = PORT_FIBRE;
> - break;
> - default:
> - dev_err(&phydev->mdio.dev, "Incompatible SFP module inserted\n");
> - return -EINVAL;
> + if (ret)
> + return ret;
> }
>
> - return ret;
> + phydev->port = enable ? PORT_FIBRE : PORT_TP;
> +
> + return phy_modify(phydev, QCA807X_CHIP_CONFIGURATION,
> + QCA807X_BT_BX_REG_SEL,
> + enable ? 0 : QCA807X_BT_BX_REG_SEL);
> }
>
> -static void qca807x_sfp_remove(void *upstream)
> +static const struct phy_port_ops qca807x_serdes_port_ops = {
> + .configure_mii = qca807x_configure_serdes,
> +};
> +
> +static int qca807x_attach_mii_port(struct phy_device *phydev,
> + struct phy_port *port)
> {
> - struct phy_device *phydev = upstream;
> + __set_bit(PHY_INTERFACE_MODE_1000BASEX, port->interfaces);
> + __set_bit(PHY_INTERFACE_MODE_100BASEX, port->interfaces);
>
> - /* Select copper page */
> - phy_set_bits(phydev,
> - QCA807X_CHIP_CONFIGURATION,
> - QCA807X_BT_BX_REG_SEL);
> + port->ops = &qca807x_serdes_port_ops;
>
> - phydev->port = PORT_TP;
> + return 0;
> }
>
> -static const struct sfp_upstream_ops qca807x_sfp_ops = {
> - .attach = phy_sfp_attach,
> - .detach = phy_sfp_detach,
> - .module_insert = qca807x_sfp_insert,
> - .module_remove = qca807x_sfp_remove,
> - .connect_phy = phy_sfp_connect_phy,
> - .disconnect_phy = phy_sfp_disconnect_phy,
> -};
> -
> static int qca807x_probe(struct phy_device *phydev)
> {
> struct device_node *node = phydev->mdio.dev.of_node;
> @@ -745,9 +731,8 @@ static int qca807x_probe(struct phy_device *phydev)
>
> /* Attach SFP bus on combo port*/
> if (phy_read(phydev, QCA807X_CHIP_CONFIGURATION)) {
> - ret = phy_sfp_probe(phydev, &qca807x_sfp_ops);
> - if (ret)
> - return ret;
> + phydev->max_n_ports = 2;
> +
> linkmode_set_bit(ETHTOOL_LINK_MODE_FIBRE_BIT, phydev->supported);
> linkmode_set_bit(ETHTOOL_LINK_MODE_FIBRE_BIT, phydev->advertising);
> }
> @@ -825,6 +810,7 @@ static struct phy_driver qca807x_drivers[] = {
> .get_phy_stats = qca807x_get_phy_stats,
> .set_wol = at8031_set_wol,
> .get_wol = at803x_get_wol,
> + .attach_mii_port = qca807x_attach_mii_port,
> },
> {
> PHY_ID_MATCH_EXACT(PHY_ID_QCA8075),
> @@ -852,6 +838,7 @@ static struct phy_driver qca807x_drivers[] = {
> .get_phy_stats = qca807x_get_phy_stats,
> .set_wol = at8031_set_wol,
> .get_wol = at803x_get_wol,
> + .attach_mii_port = qca807x_attach_mii_port,
> },
> };
> module_phy_driver(qca807x_drivers);
Powered by blists - more mailing lists