Skip to content

Commit

Permalink
net: Add the possibility to support a selected hwtstamp in netdevice
Browse files Browse the repository at this point in the history
Introduce the description of a hwtstamp provider, mainly defined with a
the hwtstamp source and the phydev pointer.

Add a hwtstamp provider description within the netdev structure to
allow saving the hwtstamp we want to use. This prepares for future
support of an ethtool netlink command to select the desired hwtstamp
provider. By default, the old API that does not support hwtstamp
selectability is used, meaning the hwtstamp provider pointer is unset.

Signed-off-by: Kory Maincent <kory.maincent@bootlin.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
  • Loading branch information
Kory Maincent authored and David S. Miller committed Dec 16, 2024
1 parent b18fe47 commit 35f7cad
Show file tree
Hide file tree
Showing 6 changed files with 140 additions and 7 deletions.
10 changes: 10 additions & 0 deletions drivers/net/phy/phy_device.c
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,7 @@
#include <linux/phy_link_topology.h>
#include <linux/pse-pd/pse.h>
#include <linux/property.h>
#include <linux/ptp_clock_kernel.h>
#include <linux/rtnetlink.h>
#include <linux/sfp.h>
#include <linux/skbuff.h>
Expand Down Expand Up @@ -1998,6 +1999,15 @@ void phy_detach(struct phy_device *phydev)

phy_suspend(phydev);
if (dev) {
struct hwtstamp_provider *hwprov;

hwprov = rtnl_dereference(dev->hwprov);
/* Disable timestamp if it is the one selected */
if (hwprov && hwprov->phydev == phydev) {
rcu_assign_pointer(dev->hwprov, NULL);
kfree_rcu(hwprov, rcu_head);
}

phydev->attached_dev->phydev = NULL;
phydev->attached_dev = NULL;
phy_link_topo_del_phy(dev, phydev);
Expand Down
29 changes: 29 additions & 0 deletions include/linux/net_tstamp.h
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,33 @@ enum hwtstamp_source {
HWTSTAMP_SOURCE_PHYLIB,
};

/**
* struct hwtstamp_provider_desc - hwtstamp provider description
*
* @index: index of the hwtstamp provider.
* @qualifier: hwtstamp provider qualifier.
*/
struct hwtstamp_provider_desc {
int index;
enum hwtstamp_provider_qualifier qualifier;
};

/**
* struct hwtstamp_provider - hwtstamp provider object
*
* @rcu_head: RCU callback used to free the struct.
* @source: source of the hwtstamp provider.
* @phydev: pointer of the phydev source in case a PTP coming from phylib
* @desc: hwtstamp provider description.
*/

struct hwtstamp_provider {
struct rcu_head rcu_head;
enum hwtstamp_source source;
struct phy_device *phydev;
struct hwtstamp_provider_desc desc;
};

/**
* struct kernel_hwtstamp_config - Kernel copy of struct hwtstamp_config
*
Expand All @@ -31,6 +58,7 @@ enum hwtstamp_source {
* copied the ioctl request back to user space
* @source: indication whether timestamps should come from the netdev or from
* an attached phylib PHY
* @qualifier: qualifier of the hwtstamp provider
*
* Prefer using this structure for in-kernel processing of hardware
* timestamping configuration, over the inextensible struct hwtstamp_config
Expand All @@ -43,6 +71,7 @@ struct kernel_hwtstamp_config {
struct ifreq *ifr;
bool copied_to_user;
enum hwtstamp_source source;
enum hwtstamp_provider_qualifier qualifier;
};

static inline void hwtstamp_config_to_kernel(struct kernel_hwtstamp_config *kernel_cfg,
Expand Down
4 changes: 4 additions & 0 deletions include/linux/netdevice.h
Original file line number Diff line number Diff line change
Expand Up @@ -82,6 +82,7 @@ struct xdp_metadata_ops;
struct xdp_md;
struct ethtool_netdev_state;
struct phy_link_topology;
struct hwtstamp_provider;

typedef u32 xdp_features_t;

Expand Down Expand Up @@ -2045,6 +2046,7 @@ enum netdev_reg_state {
*
* @neighbours: List heads pointing to this device's neighbours'
* dev_list, one per address-family.
* @hwprov: Tracks which PTP performs hardware packet time stamping.
*
* FIXME: cleanup struct net_device such that network protocol info
* moves out.
Expand Down Expand Up @@ -2457,6 +2459,8 @@ struct net_device {

struct hlist_head neighbours[NEIGH_NR_TABLES];

struct hwtstamp_provider __rcu *hwprov;

u8 priv[] ____cacheline_aligned
__counted_by(priv_len);
} ____cacheline_aligned;
Expand Down
11 changes: 11 additions & 0 deletions include/uapi/linux/net_tstamp.h
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,17 @@
#include <linux/types.h>
#include <linux/socket.h> /* for SO_TIMESTAMPING */

/*
* Possible type of hwtstamp provider. Mainly "precise" the default one
* is for IEEE 1588 quality and "approx" is for NICs DMA point.
*/
enum hwtstamp_provider_qualifier {
HWTSTAMP_PROVIDER_QUALIFIER_PRECISE,
HWTSTAMP_PROVIDER_QUALIFIER_APPROX,

HWTSTAMP_PROVIDER_QUALIFIER_CNT,
};

/* SO_TIMESTAMPING flags */
enum {
SOF_TIMESTAMPING_TX_HARDWARE = (1<<0),
Expand Down
41 changes: 39 additions & 2 deletions net/core/dev_ioctl.c
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@
#include <linux/rtnetlink.h>
#include <linux/net_tstamp.h>
#include <linux/phylib_stubs.h>
#include <linux/ptp_clock_kernel.h>
#include <linux/wireless.h>
#include <linux/if_bridge.h>
#include <net/dsa_stubs.h>
Expand Down Expand Up @@ -269,6 +270,21 @@ static int dev_eth_ioctl(struct net_device *dev,
int dev_get_hwtstamp_phylib(struct net_device *dev,
struct kernel_hwtstamp_config *cfg)
{
struct hwtstamp_provider *hwprov;

hwprov = rtnl_dereference(dev->hwprov);
if (hwprov) {
cfg->qualifier = hwprov->desc.qualifier;
if (hwprov->source == HWTSTAMP_SOURCE_PHYLIB &&
hwprov->phydev)
return phy_hwtstamp_get(hwprov->phydev, cfg);

if (hwprov->source == HWTSTAMP_SOURCE_NETDEV)
return dev->netdev_ops->ndo_hwtstamp_get(dev, cfg);

return -EOPNOTSUPP;
}

if (phy_is_default_hwtstamp(dev->phydev))
return phy_hwtstamp_get(dev->phydev, cfg);

Expand Down Expand Up @@ -324,11 +340,32 @@ int dev_set_hwtstamp_phylib(struct net_device *dev,
struct netlink_ext_ack *extack)
{
const struct net_device_ops *ops = dev->netdev_ops;
bool phy_ts = phy_is_default_hwtstamp(dev->phydev);
struct kernel_hwtstamp_config old_cfg = {};
struct hwtstamp_provider *hwprov;
struct phy_device *phydev;
bool changed = false;
bool phy_ts;
int err;

hwprov = rtnl_dereference(dev->hwprov);
if (hwprov) {
if (hwprov->source == HWTSTAMP_SOURCE_PHYLIB &&
hwprov->phydev) {
phy_ts = true;
phydev = hwprov->phydev;
} else if (hwprov->source == HWTSTAMP_SOURCE_NETDEV) {
phy_ts = false;
} else {
return -EOPNOTSUPP;
}

cfg->qualifier = hwprov->desc.qualifier;
} else {
phy_ts = phy_is_default_hwtstamp(dev->phydev);
if (phy_ts)
phydev = dev->phydev;
}

cfg->source = phy_ts ? HWTSTAMP_SOURCE_PHYLIB : HWTSTAMP_SOURCE_NETDEV;

if (phy_ts && dev->see_all_hwtstamp_requests) {
Expand All @@ -350,7 +387,7 @@ int dev_set_hwtstamp_phylib(struct net_device *dev,
changed = kernel_hwtstamp_config_changed(&old_cfg, cfg);

if (phy_ts) {
err = phy_hwtstamp_set(dev->phydev, cfg, extack);
err = phy_hwtstamp_set(phydev, cfg, extack);
if (err) {
if (changed)
ops->ndo_hwtstamp_set(dev, &old_cfg, NULL);
Expand Down
52 changes: 47 additions & 5 deletions net/core/timestamping.c
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@
#include <linux/ptp_classify.h>
#include <linux/skbuff.h>
#include <linux/export.h>
#include <linux/ptp_clock_kernel.h>

static unsigned int classify(const struct sk_buff *skb)
{
Expand All @@ -21,19 +22,39 @@ static unsigned int classify(const struct sk_buff *skb)

void skb_clone_tx_timestamp(struct sk_buff *skb)
{
struct hwtstamp_provider *hwprov;
struct mii_timestamper *mii_ts;
struct phy_device *phydev;
struct sk_buff *clone;
unsigned int type;

if (!skb->sk || !skb->dev ||
!phy_is_default_hwtstamp(skb->dev->phydev))
if (!skb->sk || !skb->dev)
return;

rcu_read_lock();
hwprov = rcu_dereference(skb->dev->hwprov);
if (hwprov) {
if (hwprov->source != HWTSTAMP_SOURCE_PHYLIB ||
!hwprov->phydev) {
rcu_read_unlock();
return;
}

phydev = hwprov->phydev;
} else {
phydev = skb->dev->phydev;
if (!phy_is_default_hwtstamp(phydev)) {
rcu_read_unlock();
return;
}
}
rcu_read_unlock();

type = classify(skb);
if (type == PTP_CLASS_NONE)
return;

mii_ts = skb->dev->phydev->mii_ts;
mii_ts = phydev->mii_ts;
if (likely(mii_ts->txtstamp)) {
clone = skb_clone_sk(skb);
if (!clone)
Expand All @@ -45,12 +66,33 @@ EXPORT_SYMBOL_GPL(skb_clone_tx_timestamp);

bool skb_defer_rx_timestamp(struct sk_buff *skb)
{
struct hwtstamp_provider *hwprov;
struct mii_timestamper *mii_ts;
struct phy_device *phydev;
unsigned int type;

if (!skb->dev || !phy_is_default_hwtstamp(skb->dev->phydev))
if (!skb->dev)
return false;

rcu_read_lock();
hwprov = rcu_dereference(skb->dev->hwprov);
if (hwprov) {
if (hwprov->source != HWTSTAMP_SOURCE_PHYLIB ||
!hwprov->phydev) {
rcu_read_unlock();
return false;
}

phydev = hwprov->phydev;
} else {
phydev = skb->dev->phydev;
if (!phy_is_default_hwtstamp(phydev)) {
rcu_read_unlock();
return false;
}
}
rcu_read_unlock();

if (skb_headroom(skb) < ETH_HLEN)
return false;

Expand All @@ -63,7 +105,7 @@ bool skb_defer_rx_timestamp(struct sk_buff *skb)
if (type == PTP_CLASS_NONE)
return false;

mii_ts = skb->dev->phydev->mii_ts;
mii_ts = phydev->mii_ts;
if (likely(mii_ts->rxtstamp))
return mii_ts->rxtstamp(mii_ts, skb, type);

Expand Down

0 comments on commit 35f7cad

Please sign in to comment.