Merge branch 'phy-listing-link_topology-tracking'
Maxime Chevallier says: ==================== Introduce PHY listing and link_topology tracking This is V11 for the link topology addition, allowing to track all PHYs that are linked to netdevices. This V11 addresses the various netlink-related issues that were raised by Jakub, and fixes some typos in the documentation. As a remainder, here's what the PHY listings would look like : - eth0 has a 88x3310 acting as media converter, and an SFP module with an embedded 88e1111 PHY - eth2 has a 88e1510 PHY PHY for eth0: PHY index: 1 Driver name: mv88x3310 PHY device name: f212a600.mdio-mii:00 Downstream SFP bus name: sfp-eth0 PHY id: 0 Upstream type: MAC PHY for eth0: PHY index: 2 Driver name: Marvell 88E1111 PHY device name: i2c:sfp-eth0:16 PHY id: 21040322 Upstream type: PHY Upstream PHY index: 1 Upstream SFP name: sfp-eth0 PHY for eth2: PHY index: 1 Driver name: Marvell 88E1510 PHY device name: f212a200.mdio-mii:00 PHY id: 21040593 Upstream type: MAC Ethtool patches : https://github.com/minimaxwell/ethtool/tree/link-topo-v6 ==================== Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
commit
267e31750a
@ -57,6 +57,7 @@ Structure of this header is
|
||||
``ETHTOOL_A_HEADER_DEV_INDEX`` u32 device ifindex
|
||||
``ETHTOOL_A_HEADER_DEV_NAME`` string device name
|
||||
``ETHTOOL_A_HEADER_FLAGS`` u32 flags common for all requests
|
||||
``ETHTOOL_A_HEADER_PHY_INDEX`` u32 phy device index
|
||||
============================== ====== =============================
|
||||
|
||||
``ETHTOOL_A_HEADER_DEV_INDEX`` and ``ETHTOOL_A_HEADER_DEV_NAME`` identify the
|
||||
@ -81,6 +82,12 @@ the behaviour is backward compatible, i.e. requests from old clients not aware
|
||||
of the flag should be interpreted the way the client expects. A client must
|
||||
not set flags it does not understand.
|
||||
|
||||
``ETHTOOL_A_HEADER_PHY_INDEX`` identifies the Ethernet PHY the message relates to.
|
||||
As there are numerous commands that are related to PHY configuration, and because
|
||||
there may be more than one PHY on the link, the PHY index can be passed in the
|
||||
request for the commands that needs it. It is, however, not mandatory, and if it
|
||||
is not passed for commands that target a PHY, the net_device.phydev pointer
|
||||
is used.
|
||||
|
||||
Bit sets
|
||||
========
|
||||
|
@ -8015,6 +8015,8 @@ F: include/linux/mii.h
|
||||
F: include/linux/of_net.h
|
||||
F: include/linux/phy.h
|
||||
F: include/linux/phy_fixed.h
|
||||
F: include/linux/phy_link_topology.h
|
||||
F: include/linux/phy_link_topology_core.h
|
||||
F: include/linux/phylib_stubs.h
|
||||
F: include/linux/platform_data/mdio-bcm-unimac.h
|
||||
F: include/linux/platform_data/mdio-gpio.h
|
||||
|
@ -2,7 +2,7 @@
|
||||
# Makefile for Linux PHY drivers
|
||||
|
||||
libphy-y := phy.o phy-c45.o phy-core.o phy_device.o \
|
||||
linkmode.o
|
||||
linkmode.o phy_link_topology.o
|
||||
mdio-bus-y += mdio_bus.o mdio_device.o
|
||||
|
||||
ifdef CONFIG_MDIO_DEVICE
|
||||
|
@ -553,6 +553,8 @@ static const struct sfp_upstream_ops sfp_phy_ops = {
|
||||
.link_down = mv2222_sfp_link_down,
|
||||
.attach = phy_sfp_attach,
|
||||
.detach = phy_sfp_detach,
|
||||
.connect_phy = phy_sfp_connect_phy,
|
||||
.disconnect_phy = phy_sfp_disconnect_phy,
|
||||
};
|
||||
|
||||
static int mv2222_probe(struct phy_device *phydev)
|
||||
|
@ -3550,6 +3550,8 @@ static const struct sfp_upstream_ops m88e1510_sfp_ops = {
|
||||
.module_remove = m88e1510_sfp_remove,
|
||||
.attach = phy_sfp_attach,
|
||||
.detach = phy_sfp_detach,
|
||||
.connect_phy = phy_sfp_connect_phy,
|
||||
.disconnect_phy = phy_sfp_disconnect_phy,
|
||||
};
|
||||
|
||||
static int m88e1510_probe(struct phy_device *phydev)
|
||||
|
@ -503,6 +503,8 @@ static int mv3310_sfp_insert(void *upstream, const struct sfp_eeprom_id *id)
|
||||
static const struct sfp_upstream_ops mv3310_sfp_ops = {
|
||||
.attach = phy_sfp_attach,
|
||||
.detach = phy_sfp_detach,
|
||||
.connect_phy = phy_sfp_connect_phy,
|
||||
.disconnect_phy = phy_sfp_disconnect_phy,
|
||||
.module_insert = mv3310_sfp_insert,
|
||||
};
|
||||
|
||||
|
@ -29,6 +29,7 @@
|
||||
#include <linux/phy.h>
|
||||
#include <linux/phylib_stubs.h>
|
||||
#include <linux/phy_led_triggers.h>
|
||||
#include <linux/phy_link_topology.h>
|
||||
#include <linux/pse-pd/pse.h>
|
||||
#include <linux/property.h>
|
||||
#include <linux/rtnetlink.h>
|
||||
@ -276,6 +277,14 @@ static void phy_mdio_device_remove(struct mdio_device *mdiodev)
|
||||
|
||||
static struct phy_driver genphy_driver;
|
||||
|
||||
static struct phy_link_topology *phy_get_link_topology(struct phy_device *phydev)
|
||||
{
|
||||
if (phydev->attached_dev)
|
||||
return phydev->attached_dev->link_topo;
|
||||
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static LIST_HEAD(phy_fixup_list);
|
||||
static DEFINE_MUTEX(phy_fixup_lock);
|
||||
|
||||
@ -1369,6 +1378,46 @@ phy_standalone_show(struct device *dev, struct device_attribute *attr,
|
||||
}
|
||||
static DEVICE_ATTR_RO(phy_standalone);
|
||||
|
||||
/**
|
||||
* phy_sfp_connect_phy - Connect the SFP module's PHY to the upstream PHY
|
||||
* @upstream: pointer to the upstream phy device
|
||||
* @phy: pointer to the SFP module's phy device
|
||||
*
|
||||
* This helper allows keeping track of PHY devices on the link. It adds the
|
||||
* SFP module's phy to the phy namespace of the upstream phy
|
||||
*/
|
||||
int phy_sfp_connect_phy(void *upstream, struct phy_device *phy)
|
||||
{
|
||||
struct phy_device *phydev = upstream;
|
||||
struct phy_link_topology *topo = phy_get_link_topology(phydev);
|
||||
|
||||
if (topo)
|
||||
return phy_link_topo_add_phy(topo, phy, PHY_UPSTREAM_PHY, phydev);
|
||||
|
||||
return 0;
|
||||
}
|
||||
EXPORT_SYMBOL(phy_sfp_connect_phy);
|
||||
|
||||
/**
|
||||
* phy_sfp_disconnect_phy - Disconnect the SFP module's PHY from the upstream PHY
|
||||
* @upstream: pointer to the upstream phy device
|
||||
* @phy: pointer to the SFP module's phy device
|
||||
*
|
||||
* This helper allows keeping track of PHY devices on the link. It removes the
|
||||
* SFP module's phy to the phy namespace of the upstream phy. As the module phy
|
||||
* will be destroyed, re-inserting the same module will add a new phy with a
|
||||
* new index.
|
||||
*/
|
||||
void phy_sfp_disconnect_phy(void *upstream, struct phy_device *phy)
|
||||
{
|
||||
struct phy_device *phydev = upstream;
|
||||
struct phy_link_topology *topo = phy_get_link_topology(phydev);
|
||||
|
||||
if (topo)
|
||||
phy_link_topo_del_phy(topo, phy);
|
||||
}
|
||||
EXPORT_SYMBOL(phy_sfp_disconnect_phy);
|
||||
|
||||
/**
|
||||
* phy_sfp_attach - attach the SFP bus to the PHY upstream network device
|
||||
* @upstream: pointer to the phy device
|
||||
@ -1511,6 +1560,11 @@ int phy_attach_direct(struct net_device *dev, struct phy_device *phydev,
|
||||
|
||||
if (phydev->sfp_bus_attached)
|
||||
dev->sfp_bus = phydev->sfp_bus;
|
||||
|
||||
err = phy_link_topo_add_phy(dev->link_topo, phydev,
|
||||
PHY_UPSTREAM_MAC, dev);
|
||||
if (err)
|
||||
goto error;
|
||||
}
|
||||
|
||||
/* Some Ethernet drivers try to connect to a PHY device before
|
||||
@ -1938,6 +1992,7 @@ void phy_detach(struct phy_device *phydev)
|
||||
if (dev) {
|
||||
phydev->attached_dev->phydev = NULL;
|
||||
phydev->attached_dev = NULL;
|
||||
phy_link_topo_del_phy(dev->link_topo, phydev);
|
||||
}
|
||||
phydev->phylink = NULL;
|
||||
|
||||
|
105
drivers/net/phy/phy_link_topology.c
Normal file
105
drivers/net/phy/phy_link_topology.c
Normal file
@ -0,0 +1,105 @@
|
||||
// SPDX-License-Identifier: GPL-2.0+
|
||||
/*
|
||||
* Infrastructure to handle all PHY devices connected to a given netdev,
|
||||
* either directly or indirectly attached.
|
||||
*
|
||||
* Copyright (c) 2023 Maxime Chevallier<maxime.chevallier@bootlin.com>
|
||||
*/
|
||||
|
||||
#include <linux/phy_link_topology.h>
|
||||
#include <linux/netdevice.h>
|
||||
#include <linux/phy.h>
|
||||
#include <linux/rtnetlink.h>
|
||||
#include <linux/xarray.h>
|
||||
|
||||
struct phy_link_topology *phy_link_topo_create(struct net_device *dev)
|
||||
{
|
||||
struct phy_link_topology *topo;
|
||||
|
||||
topo = kzalloc(sizeof(*topo), GFP_KERNEL);
|
||||
if (!topo)
|
||||
return ERR_PTR(-ENOMEM);
|
||||
|
||||
xa_init_flags(&topo->phys, XA_FLAGS_ALLOC1);
|
||||
topo->next_phy_index = 1;
|
||||
|
||||
return topo;
|
||||
}
|
||||
|
||||
void phy_link_topo_destroy(struct phy_link_topology *topo)
|
||||
{
|
||||
if (!topo)
|
||||
return;
|
||||
|
||||
xa_destroy(&topo->phys);
|
||||
kfree(topo);
|
||||
}
|
||||
|
||||
int phy_link_topo_add_phy(struct phy_link_topology *topo,
|
||||
struct phy_device *phy,
|
||||
enum phy_upstream upt, void *upstream)
|
||||
{
|
||||
struct phy_device_node *pdn;
|
||||
int ret;
|
||||
|
||||
pdn = kzalloc(sizeof(*pdn), GFP_KERNEL);
|
||||
if (!pdn)
|
||||
return -ENOMEM;
|
||||
|
||||
pdn->phy = phy;
|
||||
switch (upt) {
|
||||
case PHY_UPSTREAM_MAC:
|
||||
pdn->upstream.netdev = (struct net_device *)upstream;
|
||||
if (phy_on_sfp(phy))
|
||||
pdn->parent_sfp_bus = pdn->upstream.netdev->sfp_bus;
|
||||
break;
|
||||
case PHY_UPSTREAM_PHY:
|
||||
pdn->upstream.phydev = (struct phy_device *)upstream;
|
||||
if (phy_on_sfp(phy))
|
||||
pdn->parent_sfp_bus = pdn->upstream.phydev->sfp_bus;
|
||||
break;
|
||||
default:
|
||||
ret = -EINVAL;
|
||||
goto err;
|
||||
}
|
||||
pdn->upstream_type = upt;
|
||||
|
||||
/* Attempt to re-use a previously allocated phy_index */
|
||||
if (phy->phyindex) {
|
||||
ret = xa_insert(&topo->phys, phy->phyindex, pdn, GFP_KERNEL);
|
||||
|
||||
/* Errors could be either -ENOMEM or -EBUSY. If the phy has an
|
||||
* index, and there's another entry at the same index, this is
|
||||
* unexpected and we still error-out
|
||||
*/
|
||||
if (ret)
|
||||
goto err;
|
||||
return 0;
|
||||
}
|
||||
|
||||
ret = xa_alloc_cyclic(&topo->phys, &phy->phyindex, pdn, xa_limit_32b,
|
||||
&topo->next_phy_index, GFP_KERNEL);
|
||||
if (ret)
|
||||
goto err;
|
||||
|
||||
return 0;
|
||||
|
||||
err:
|
||||
kfree(pdn);
|
||||
return ret;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(phy_link_topo_add_phy);
|
||||
|
||||
void phy_link_topo_del_phy(struct phy_link_topology *topo,
|
||||
struct phy_device *phy)
|
||||
{
|
||||
struct phy_device_node *pdn = xa_erase(&topo->phys, phy->phyindex);
|
||||
|
||||
/* We delete the PHY from the topology, however we don't re-set the
|
||||
* phy->phyindex field. If the PHY isn't gone, we can re-assign it the
|
||||
* same index next time it's added back to the topology
|
||||
*/
|
||||
|
||||
kfree(pdn);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(phy_link_topo_del_phy);
|
@ -3408,7 +3408,8 @@ static int phylink_sfp_connect_phy(void *upstream, struct phy_device *phy)
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void phylink_sfp_disconnect_phy(void *upstream)
|
||||
static void phylink_sfp_disconnect_phy(void *upstream,
|
||||
struct phy_device *phydev)
|
||||
{
|
||||
phylink_disconnect_phy(upstream);
|
||||
}
|
||||
|
@ -770,6 +770,8 @@ static const struct sfp_upstream_ops at8031_sfp_ops = {
|
||||
.attach = phy_sfp_attach,
|
||||
.detach = phy_sfp_detach,
|
||||
.module_insert = at8031_sfp_insert,
|
||||
.connect_phy = phy_sfp_connect_phy,
|
||||
.disconnect_phy = phy_sfp_disconnect_phy,
|
||||
};
|
||||
|
||||
static int at8031_parse_dt(struct phy_device *phydev)
|
||||
|
@ -699,6 +699,8 @@ static const struct sfp_upstream_ops qca807x_sfp_ops = {
|
||||
.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)
|
||||
|
@ -486,7 +486,7 @@ static void sfp_unregister_bus(struct sfp_bus *bus)
|
||||
bus->socket_ops->stop(bus->sfp);
|
||||
bus->socket_ops->detach(bus->sfp);
|
||||
if (bus->phydev && ops && ops->disconnect_phy)
|
||||
ops->disconnect_phy(bus->upstream);
|
||||
ops->disconnect_phy(bus->upstream, bus->phydev);
|
||||
}
|
||||
bus->registered = false;
|
||||
}
|
||||
@ -742,7 +742,7 @@ void sfp_remove_phy(struct sfp_bus *bus)
|
||||
const struct sfp_upstream_ops *ops = sfp_get_upstream_ops(bus);
|
||||
|
||||
if (ops && ops->disconnect_phy)
|
||||
ops->disconnect_phy(bus->upstream);
|
||||
ops->disconnect_phy(bus->upstream, bus->phydev);
|
||||
bus->phydev = NULL;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(sfp_remove_phy);
|
||||
@ -859,3 +859,14 @@ void sfp_unregister_socket(struct sfp_bus *bus)
|
||||
sfp_bus_put(bus);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(sfp_unregister_socket);
|
||||
|
||||
const char *sfp_get_name(struct sfp_bus *bus)
|
||||
{
|
||||
ASSERT_RTNL();
|
||||
|
||||
if (bus->sfp_dev)
|
||||
return dev_name(bus->sfp_dev);
|
||||
|
||||
return NULL;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(sfp_get_name);
|
||||
|
@ -40,7 +40,6 @@
|
||||
#include <net/dcbnl.h>
|
||||
#endif
|
||||
#include <net/netprio_cgroup.h>
|
||||
|
||||
#include <linux/netdev_features.h>
|
||||
#include <linux/neighbour.h>
|
||||
#include <uapi/linux/netdevice.h>
|
||||
@ -52,6 +51,7 @@
|
||||
#include <net/net_trackers.h>
|
||||
#include <net/net_debug.h>
|
||||
#include <net/dropreason-core.h>
|
||||
#include <linux/phy_link_topology_core.h>
|
||||
|
||||
struct netpoll_info;
|
||||
struct device;
|
||||
@ -1974,6 +1974,7 @@ enum netdev_reg_state {
|
||||
* @fcoe_ddp_xid: Max exchange id for FCoE LRO by ddp
|
||||
*
|
||||
* @priomap: XXX: need comments on this one
|
||||
* @link_topo: Physical link topology tracking attached PHYs
|
||||
* @phydev: Physical device may attach itself
|
||||
* for hardware timestamping
|
||||
* @sfp_bus: attached &struct sfp_bus structure.
|
||||
@ -2364,6 +2365,7 @@ struct net_device {
|
||||
#if IS_ENABLED(CONFIG_CGROUP_NET_PRIO)
|
||||
struct netprio_map __rcu *priomap;
|
||||
#endif
|
||||
struct phy_link_topology *link_topo;
|
||||
struct phy_device *phydev;
|
||||
struct sfp_bus *sfp_bus;
|
||||
struct lock_class_key *qdisc_tx_busylock;
|
||||
|
@ -550,6 +550,9 @@ struct macsec_ops;
|
||||
* @drv: Pointer to the driver for this PHY instance
|
||||
* @devlink: Create a link between phy dev and mac dev, if the external phy
|
||||
* used by current mac interface is managed by another mac interface.
|
||||
* @phyindex: Unique id across the phy's parent tree of phys to address the PHY
|
||||
* from userspace, similar to ifindex. A zero index means the PHY
|
||||
* wasn't assigned an id yet.
|
||||
* @phy_id: UID for this device found during discovery
|
||||
* @c45_ids: 802.3-c45 Device Identifiers if is_c45.
|
||||
* @is_c45: Set to true if this PHY uses clause 45 addressing.
|
||||
@ -650,6 +653,7 @@ struct phy_device {
|
||||
|
||||
struct device_link *devlink;
|
||||
|
||||
u32 phyindex;
|
||||
u32 phy_id;
|
||||
|
||||
struct phy_c45_device_ids c45_ids;
|
||||
@ -1754,6 +1758,8 @@ int phy_suspend(struct phy_device *phydev);
|
||||
int phy_resume(struct phy_device *phydev);
|
||||
int __phy_resume(struct phy_device *phydev);
|
||||
int phy_loopback(struct phy_device *phydev, bool enable);
|
||||
int phy_sfp_connect_phy(void *upstream, struct phy_device *phy);
|
||||
void phy_sfp_disconnect_phy(void *upstream, struct phy_device *phy);
|
||||
void phy_sfp_attach(void *upstream, struct sfp_bus *bus);
|
||||
void phy_sfp_detach(void *upstream, struct sfp_bus *bus);
|
||||
int phy_sfp_probe(struct phy_device *phydev,
|
||||
|
72
include/linux/phy_link_topology.h
Normal file
72
include/linux/phy_link_topology.h
Normal file
@ -0,0 +1,72 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
/*
|
||||
* PHY device list allow maintaining a list of PHY devices that are
|
||||
* part of a netdevice's link topology. PHYs can for example be chained,
|
||||
* as is the case when using a PHY that exposes an SFP module, on which an
|
||||
* SFP transceiver that embeds a PHY is connected.
|
||||
*
|
||||
* This list can then be used by userspace to leverage individual PHY
|
||||
* capabilities.
|
||||
*/
|
||||
#ifndef __PHY_LINK_TOPOLOGY_H
|
||||
#define __PHY_LINK_TOPOLOGY_H
|
||||
|
||||
#include <linux/ethtool.h>
|
||||
#include <linux/phy_link_topology_core.h>
|
||||
|
||||
struct xarray;
|
||||
struct phy_device;
|
||||
struct net_device;
|
||||
struct sfp_bus;
|
||||
|
||||
struct phy_device_node {
|
||||
enum phy_upstream upstream_type;
|
||||
|
||||
union {
|
||||
struct net_device *netdev;
|
||||
struct phy_device *phydev;
|
||||
} upstream;
|
||||
|
||||
struct sfp_bus *parent_sfp_bus;
|
||||
|
||||
struct phy_device *phy;
|
||||
};
|
||||
|
||||
struct phy_link_topology {
|
||||
struct xarray phys;
|
||||
u32 next_phy_index;
|
||||
};
|
||||
|
||||
static inline struct phy_device *
|
||||
phy_link_topo_get_phy(struct phy_link_topology *topo, u32 phyindex)
|
||||
{
|
||||
struct phy_device_node *pdn = xa_load(&topo->phys, phyindex);
|
||||
|
||||
if (pdn)
|
||||
return pdn->phy;
|
||||
|
||||
return NULL;
|
||||
}
|
||||
|
||||
#if IS_REACHABLE(CONFIG_PHYLIB)
|
||||
int phy_link_topo_add_phy(struct phy_link_topology *topo,
|
||||
struct phy_device *phy,
|
||||
enum phy_upstream upt, void *upstream);
|
||||
|
||||
void phy_link_topo_del_phy(struct phy_link_topology *lt, struct phy_device *phy);
|
||||
|
||||
#else
|
||||
static inline int phy_link_topo_add_phy(struct phy_link_topology *topo,
|
||||
struct phy_device *phy,
|
||||
enum phy_upstream upt, void *upstream)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
static inline void phy_link_topo_del_phy(struct phy_link_topology *topo,
|
||||
struct phy_device *phy)
|
||||
{
|
||||
}
|
||||
#endif
|
||||
|
||||
#endif /* __PHY_LINK_TOPOLOGY_H */
|
25
include/linux/phy_link_topology_core.h
Normal file
25
include/linux/phy_link_topology_core.h
Normal file
@ -0,0 +1,25 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0 */
|
||||
#ifndef __PHY_LINK_TOPOLOGY_CORE_H
|
||||
#define __PHY_LINK_TOPOLOGY_CORE_H
|
||||
|
||||
struct phy_link_topology;
|
||||
|
||||
#if IS_REACHABLE(CONFIG_PHYLIB)
|
||||
|
||||
struct phy_link_topology *phy_link_topo_create(struct net_device *dev);
|
||||
void phy_link_topo_destroy(struct phy_link_topology *topo);
|
||||
|
||||
#else
|
||||
|
||||
static inline struct phy_link_topology *phy_link_topo_create(struct net_device *dev)
|
||||
{
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static inline void phy_link_topo_destroy(struct phy_link_topology *topo)
|
||||
{
|
||||
}
|
||||
|
||||
#endif
|
||||
|
||||
#endif /* __PHY_LINK_TOPOLOGY_CORE_H */
|
@ -544,7 +544,7 @@ struct sfp_upstream_ops {
|
||||
void (*link_down)(void *priv);
|
||||
void (*link_up)(void *priv);
|
||||
int (*connect_phy)(void *priv, struct phy_device *);
|
||||
void (*disconnect_phy)(void *priv);
|
||||
void (*disconnect_phy)(void *priv, struct phy_device *);
|
||||
};
|
||||
|
||||
#if IS_ENABLED(CONFIG_SFP)
|
||||
@ -570,6 +570,7 @@ struct sfp_bus *sfp_bus_find_fwnode(const struct fwnode_handle *fwnode);
|
||||
int sfp_bus_add_upstream(struct sfp_bus *bus, void *upstream,
|
||||
const struct sfp_upstream_ops *ops);
|
||||
void sfp_bus_del_upstream(struct sfp_bus *bus);
|
||||
const char *sfp_get_name(struct sfp_bus *bus);
|
||||
#else
|
||||
static inline int sfp_parse_port(struct sfp_bus *bus,
|
||||
const struct sfp_eeprom_id *id,
|
||||
@ -648,6 +649,11 @@ static inline int sfp_bus_add_upstream(struct sfp_bus *bus, void *upstream,
|
||||
static inline void sfp_bus_del_upstream(struct sfp_bus *bus)
|
||||
{
|
||||
}
|
||||
|
||||
static inline const char *sfp_get_name(struct sfp_bus *bus)
|
||||
{
|
||||
return NULL;
|
||||
}
|
||||
#endif
|
||||
|
||||
#endif
|
||||
|
@ -2268,4 +2268,20 @@ struct ethtool_link_settings {
|
||||
* __u32 map_lp_advertising[link_mode_masks_nwords];
|
||||
*/
|
||||
};
|
||||
|
||||
/**
|
||||
* enum phy_upstream - Represents the upstream component a given PHY device
|
||||
* is connected to, as in what is on the other end of the MII bus. Most PHYs
|
||||
* will be attached to an Ethernet MAC controller, but in some cases, there's
|
||||
* an intermediate PHY used as a media-converter, which will driver another
|
||||
* MII interface as its output.
|
||||
* @PHY_UPSTREAM_MAC: Upstream component is a MAC (a switch port,
|
||||
* or ethernet controller)
|
||||
* @PHY_UPSTREAM_PHY: Upstream component is a PHY (likely a media converter)
|
||||
*/
|
||||
enum phy_upstream {
|
||||
PHY_UPSTREAM_MAC,
|
||||
PHY_UPSTREAM_PHY,
|
||||
};
|
||||
|
||||
#endif /* _UAPI_LINUX_ETHTOOL_H */
|
||||
|
@ -132,6 +132,7 @@ enum {
|
||||
ETHTOOL_A_HEADER_DEV_INDEX, /* u32 */
|
||||
ETHTOOL_A_HEADER_DEV_NAME, /* string */
|
||||
ETHTOOL_A_HEADER_FLAGS, /* u32 - ETHTOOL_FLAG_* */
|
||||
ETHTOOL_A_HEADER_PHY_INDEX, /* u32 */
|
||||
|
||||
/* add new constants above here */
|
||||
__ETHTOOL_A_HEADER_CNT,
|
||||
|
@ -158,6 +158,7 @@
|
||||
#include <net/page_pool/types.h>
|
||||
#include <net/page_pool/helpers.h>
|
||||
#include <net/rps.h>
|
||||
#include <linux/phy_link_topology_core.h>
|
||||
|
||||
#include "dev.h"
|
||||
#include "net-sysfs.h"
|
||||
@ -10962,6 +10963,12 @@ struct net_device *alloc_netdev_mqs(int sizeof_priv, const char *name,
|
||||
#ifdef CONFIG_NET_SCHED
|
||||
hash_init(dev->qdisc_hash);
|
||||
#endif
|
||||
dev->link_topo = phy_link_topo_create(dev);
|
||||
if (IS_ERR(dev->link_topo)) {
|
||||
dev->link_topo = NULL;
|
||||
goto free_all;
|
||||
}
|
||||
|
||||
dev->priv_flags = IFF_XMIT_DST_RELEASE | IFF_XMIT_DST_RELEASE_PERM;
|
||||
setup(dev);
|
||||
|
||||
@ -11050,6 +11057,8 @@ void free_netdev(struct net_device *dev)
|
||||
free_percpu(dev->xdp_bulkq);
|
||||
dev->xdp_bulkq = NULL;
|
||||
|
||||
phy_link_topo_destroy(dev->link_topo);
|
||||
|
||||
/* Compatibility with error handling in drivers */
|
||||
if (dev->reg_state == NETREG_UNINITIALIZED) {
|
||||
netdev_freemem(dev);
|
||||
|
@ -4,6 +4,7 @@
|
||||
#include <linux/ethtool_netlink.h>
|
||||
#include <linux/pm_runtime.h>
|
||||
#include "netlink.h"
|
||||
#include <linux/phy_link_topology.h>
|
||||
|
||||
static struct genl_family ethtool_genl_family;
|
||||
|
||||
@ -30,6 +31,24 @@ const struct nla_policy ethnl_header_policy_stats[] = {
|
||||
ETHTOOL_FLAGS_STATS),
|
||||
};
|
||||
|
||||
const struct nla_policy ethnl_header_policy_phy[] = {
|
||||
[ETHTOOL_A_HEADER_DEV_INDEX] = { .type = NLA_U32 },
|
||||
[ETHTOOL_A_HEADER_DEV_NAME] = { .type = NLA_NUL_STRING,
|
||||
.len = ALTIFNAMSIZ - 1 },
|
||||
[ETHTOOL_A_HEADER_FLAGS] = NLA_POLICY_MASK(NLA_U32,
|
||||
ETHTOOL_FLAGS_BASIC),
|
||||
[ETHTOOL_A_HEADER_PHY_INDEX] = NLA_POLICY_MIN(NLA_U32, 1),
|
||||
};
|
||||
|
||||
const struct nla_policy ethnl_header_policy_phy_stats[] = {
|
||||
[ETHTOOL_A_HEADER_DEV_INDEX] = { .type = NLA_U32 },
|
||||
[ETHTOOL_A_HEADER_DEV_NAME] = { .type = NLA_NUL_STRING,
|
||||
.len = ALTIFNAMSIZ - 1 },
|
||||
[ETHTOOL_A_HEADER_FLAGS] = NLA_POLICY_MASK(NLA_U32,
|
||||
ETHTOOL_FLAGS_STATS),
|
||||
[ETHTOOL_A_HEADER_PHY_INDEX] = NLA_POLICY_MIN(NLA_U32, 1),
|
||||
};
|
||||
|
||||
int ethnl_ops_begin(struct net_device *dev)
|
||||
{
|
||||
int ret;
|
||||
@ -89,8 +108,9 @@ int ethnl_parse_header_dev_get(struct ethnl_req_info *req_info,
|
||||
const struct nlattr *header, struct net *net,
|
||||
struct netlink_ext_ack *extack, bool require_dev)
|
||||
{
|
||||
struct nlattr *tb[ARRAY_SIZE(ethnl_header_policy)];
|
||||
struct nlattr *tb[ARRAY_SIZE(ethnl_header_policy_phy)];
|
||||
const struct nlattr *devname_attr;
|
||||
struct phy_device *phydev = NULL;
|
||||
struct net_device *dev = NULL;
|
||||
u32 flags = 0;
|
||||
int ret;
|
||||
@ -104,7 +124,7 @@ int ethnl_parse_header_dev_get(struct ethnl_req_info *req_info,
|
||||
/* No validation here, command policy should have a nested policy set
|
||||
* for the header, therefore validation should have already been done.
|
||||
*/
|
||||
ret = nla_parse_nested(tb, ARRAY_SIZE(ethnl_header_policy) - 1, header,
|
||||
ret = nla_parse_nested(tb, ARRAY_SIZE(ethnl_header_policy_phy) - 1, header,
|
||||
NULL, extack);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
@ -145,6 +165,30 @@ int ethnl_parse_header_dev_get(struct ethnl_req_info *req_info,
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
if (dev) {
|
||||
if (tb[ETHTOOL_A_HEADER_PHY_INDEX]) {
|
||||
struct nlattr *phy_id;
|
||||
|
||||
phy_id = tb[ETHTOOL_A_HEADER_PHY_INDEX];
|
||||
phydev = phy_link_topo_get_phy(dev->link_topo,
|
||||
nla_get_u32(phy_id));
|
||||
if (!phydev) {
|
||||
NL_SET_BAD_ATTR(extack, phy_id);
|
||||
return -ENODEV;
|
||||
}
|
||||
} else {
|
||||
/* If we need a PHY but no phy index is specified, fallback
|
||||
* to dev->phydev
|
||||
*/
|
||||
phydev = dev->phydev;
|
||||
}
|
||||
} else if (tb[ETHTOOL_A_HEADER_PHY_INDEX]) {
|
||||
NL_SET_ERR_MSG_ATTR(extack, header,
|
||||
"can't target a PHY without a netdev");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
req_info->phydev = phydev;
|
||||
req_info->dev = dev;
|
||||
req_info->flags = flags;
|
||||
return 0;
|
||||
|
@ -250,6 +250,7 @@ static inline unsigned int ethnl_reply_header_size(void)
|
||||
* @dev: network device the request is for (may be null)
|
||||
* @dev_tracker: refcount tracker for @dev reference
|
||||
* @flags: request flags common for all request types
|
||||
* @phydev: phy_device connected to @dev this request is for (may be null)
|
||||
*
|
||||
* This is a common base for request specific structures holding data from
|
||||
* parsed userspace request. These always embed struct ethnl_req_info at
|
||||
@ -259,6 +260,7 @@ struct ethnl_req_info {
|
||||
struct net_device *dev;
|
||||
netdevice_tracker dev_tracker;
|
||||
u32 flags;
|
||||
struct phy_device *phydev;
|
||||
};
|
||||
|
||||
static inline void ethnl_parse_header_dev_put(struct ethnl_req_info *req_info)
|
||||
@ -395,9 +397,12 @@ extern const struct ethnl_request_ops ethnl_rss_request_ops;
|
||||
extern const struct ethnl_request_ops ethnl_plca_cfg_request_ops;
|
||||
extern const struct ethnl_request_ops ethnl_plca_status_request_ops;
|
||||
extern const struct ethnl_request_ops ethnl_mm_request_ops;
|
||||
extern const struct ethnl_request_ops ethnl_phy_request_ops;
|
||||
|
||||
extern const struct nla_policy ethnl_header_policy[ETHTOOL_A_HEADER_FLAGS + 1];
|
||||
extern const struct nla_policy ethnl_header_policy_stats[ETHTOOL_A_HEADER_FLAGS + 1];
|
||||
extern const struct nla_policy ethnl_header_policy_phy[ETHTOOL_A_HEADER_PHY_INDEX + 1];
|
||||
extern const struct nla_policy ethnl_header_policy_phy_stats[ETHTOOL_A_HEADER_PHY_INDEX + 1];
|
||||
extern const struct nla_policy ethnl_strset_get_policy[ETHTOOL_A_STRSET_COUNTS_ONLY + 1];
|
||||
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];
|
||||
|
Loading…
x
Reference in New Issue
Block a user