* [PATCH net-next 0/7] net: ethtool: Introduce ethnl dump helpers
@ 2025-03-05 14:19 Maxime Chevallier
2025-03-05 14:19 ` [PATCH net-next 1/7] net: ethtool: netlink: Allow per-netdevice DUMP operations Maxime Chevallier
` (7 more replies)
0 siblings, 8 replies; 14+ messages in thread
From: Maxime Chevallier @ 2025-03-05 14:19 UTC (permalink / raw)
To: davem, Andrew Lunn, Jakub Kicinski, Eric Dumazet, Paolo Abeni,
Heiner Kallweit
Cc: Maxime Chevallier, netdev, linux-kernel, thomas.petazzoni,
linux-arm-kernel, Christophe Leroy, Herve Codina,
Florian Fainelli, Russell King, Vladimir Oltean,
Köry Maincent, Oleksij Rempel, Simon Horman, Romain Gantois,
Piergiorgio Beruto
Hi everyone,
This series adds some scaffolding into ethnl to ease the support of
DUMP operations.
As of today when using ethnl's default ops, the DUMP requests will
simply perform a GET for each netdev.
That hits limitations for commands that may return multiple messages for
a single netdev, such as :
- RSS (listing contexts)
- All PHY-specific commands (PLCA, PSE-PD, phy)
- tsinfo (one item for the netdev + one per phy)
Commands that need a non-default DUMP support have to re-implement
->dumpit() themselves, which prevents using most of ethnl's internal
circuitry.
This series therefore introduces a better support for dump operations in
ethnl.
The patches 1 and 2 introduce the support for filtered DUMPs, where an
ifindex/ifname can be passed in the request header for the DUMP
operation. This is for when we want to dump everything a netdev
supports, but without doing so for every single netdev. ethtool's
"--show-phys ethX" option for example performs a filtered dump.
Patch 3 introduces 3 new ethnl ops :
->dump_start() to initialize a dump context
->dump_one_dev(), that can be implemented per-command to dump
everything on a given netdev
->dump_done() to release the context
The default behaviour for dumps remains the same, calling the whole
->doit() path for each netdev.
Patch 4 introduces a set of ->dump_start(), ->dump_one_dev() and
->dump_done() callback implementations that can simply be plugged into
the existing commands that list objects per-phy, making the
phy-targeting command behaviour more coherent.
Patch 5 uses that new set of helpers to rewrite the phy.c support, which
now uses the regulat ethnl_ops instead of fully custom genl ops. This
one is the hardest to review, sorry about that, I couldn't really manage
to incrementally rework that file :(
Patches 6 and 7 are where the new dump infra shines, adding per-netdev
per-phy dump support for PLCA and PSE-PD.
We could also consider converting tsinfo/tsconfig, rss and tunnels to
these new ->dump_***() operations as well, but that's out of this
series' scope.
I've tested that series with some netdevsim PHY patches that I plan to
submit (they can be found here [1]), with the refcount tracker
for net/netns enabled to make sure the lock usage is somewhat coherent.
Thanks,
Maxime
[1]: https://github.com/minimaxwell/linux/tree/mc/netdevsim-phy
Maxime Chevallier (7):
net: ethtool: netlink: Allow per-netdevice DUMP operations
net: ethtool: netlink: Rename ethnl_default_dump_one
net: ethtool: netlink: Introduce command-specific dump_one_dev
net: ethtool: netlink: Introduce per-phy DUMP helpers
net: ethtool: phy: Convert the PHY_GET command to generic phy dump
net: ethtool: plca: Use per-PHY DUMP operations
net: ethtool: pse-pd: Use per-PHY DUMP operations
net/ethtool/netlink.c | 161 ++++++++++++++------
net/ethtool/netlink.h | 46 +++++-
net/ethtool/phy.c | 335 ++++++++++++------------------------------
net/ethtool/plca.c | 12 ++
net/ethtool/pse-pd.c | 6 +
5 files changed, 277 insertions(+), 283 deletions(-)
--
2.48.1
^ permalink raw reply [flat|nested] 14+ messages in thread
* [PATCH net-next 1/7] net: ethtool: netlink: Allow per-netdevice DUMP operations
2025-03-05 14:19 [PATCH net-next 0/7] net: ethtool: Introduce ethnl dump helpers Maxime Chevallier
@ 2025-03-05 14:19 ` Maxime Chevallier
2025-03-07 12:21 ` Simon Horman
2025-03-05 14:19 ` [PATCH net-next 2/7] net: ethtool: netlink: Rename ethnl_default_dump_one Maxime Chevallier
` (6 subsequent siblings)
7 siblings, 1 reply; 14+ messages in thread
From: Maxime Chevallier @ 2025-03-05 14:19 UTC (permalink / raw)
To: davem, Andrew Lunn, Jakub Kicinski, Eric Dumazet, Paolo Abeni,
Heiner Kallweit
Cc: Maxime Chevallier, netdev, linux-kernel, thomas.petazzoni,
linux-arm-kernel, Christophe Leroy, Herve Codina,
Florian Fainelli, Russell King, Vladimir Oltean,
Köry Maincent, Oleksij Rempel, Simon Horman, Romain Gantois,
Piergiorgio Beruto
We have a number of netlink commands in the ethnl family that may have
multiple objects to dump even for a single net_device, including :
- PLCA, PSE-PD, phy: one message per PHY device
- tsinfo: one message per timestamp source (netdev + phys)
- rss: One per RSS context
To get this behaviour, these netlink commands need to roll a custom
->dumpit().
To prepare making per-netdev DUMP more generic in ethnl, introduce a
member in the ethnl ops to indicate if a given command may allow
pernetdev DUMPs (also referred to as filtered DUMPs).
Signed-off-by: Maxime Chevallier <maxime.chevallier@bootlin.com>
---
net/ethtool/netlink.c | 45 ++++++++++++++++++++++++++++---------------
net/ethtool/netlink.h | 1 +
2 files changed, 30 insertions(+), 16 deletions(-)
diff --git a/net/ethtool/netlink.c b/net/ethtool/netlink.c
index 734849a57369..0815b28ba32f 100644
--- a/net/ethtool/netlink.c
+++ b/net/ethtool/netlink.c
@@ -578,21 +578,34 @@ static int ethnl_default_dumpit(struct sk_buff *skb,
int ret = 0;
rcu_read_lock();
- for_each_netdev_dump(net, dev, ctx->pos_ifindex) {
- dev_hold(dev);
+ if (ctx->req_info->dev) {
+ dev = ctx->req_info->dev;
rcu_read_unlock();
-
- ret = ethnl_default_dump_one(skb, dev, ctx, genl_info_dump(cb));
-
+ /* Filtered DUMP request targeted to a single netdev. We already
+ * hold a ref to the netdev from ->start()
+ */
+ ret = ethnl_default_dump_one_dev(skb, dev, ctx,
+ genl_info_dump(cb));
rcu_read_lock();
- dev_put(dev);
-
- if (ret < 0 && ret != -EOPNOTSUPP) {
- if (likely(skb->len))
- ret = skb->len;
- break;
+ netdev_put(ctx->req_info->dev, &ctx->req_info->dev_tracker);
+ } else {
+ for_each_netdev_dump(net, dev, ctx->pos_ifindex) {
+ dev_hold(dev);
+ rcu_read_unlock();
+
+ ret = ethnl_default_dump_one(skb, dev, ctx,
+ genl_info_dump(cb));
+
+ rcu_read_lock();
+ dev_put(dev);
+
+ if (ret < 0 && ret != -EOPNOTSUPP) {
+ if (likely(skb->len))
+ ret = skb->len;
+ break;
+ }
+ ret = 0;
}
- ret = 0;
}
rcu_read_unlock();
@@ -626,10 +639,10 @@ static int ethnl_default_start(struct netlink_callback *cb)
}
ret = ethnl_default_parse(req_info, &info->info, ops, false);
- if (req_info->dev) {
- /* We ignore device specification in dump requests but as the
- * same parser as for non-dump (doit) requests is used, it
- * would take reference to the device if it finds one
+ if (req_info->dev && !ops->allow_pernetdev_dump) {
+ /* We ignore device specification in unfiltered dump requests
+ * but as the same parser as for non-dump (doit) requests is
+ * used, it would take reference to the device if it finds one
*/
netdev_put(req_info->dev, &req_info->dev_tracker);
req_info->dev = NULL;
diff --git a/net/ethtool/netlink.h b/net/ethtool/netlink.h
index ec6ab5443a6f..4db27182741f 100644
--- a/net/ethtool/netlink.h
+++ b/net/ethtool/netlink.h
@@ -388,6 +388,7 @@ struct ethnl_request_ops {
unsigned int req_info_size;
unsigned int reply_data_size;
bool allow_nodev_do;
+ bool allow_pernetdev_dump;
u8 set_ntf_cmd;
int (*parse_request)(struct ethnl_req_info *req_info,
--
2.48.1
^ permalink raw reply related [flat|nested] 14+ messages in thread
* [PATCH net-next 2/7] net: ethtool: netlink: Rename ethnl_default_dump_one
2025-03-05 14:19 [PATCH net-next 0/7] net: ethtool: Introduce ethnl dump helpers Maxime Chevallier
2025-03-05 14:19 ` [PATCH net-next 1/7] net: ethtool: netlink: Allow per-netdevice DUMP operations Maxime Chevallier
@ 2025-03-05 14:19 ` Maxime Chevallier
2025-03-05 14:19 ` [PATCH net-next 3/7] net: ethtool: netlink: Introduce command-specific dump_one_dev Maxime Chevallier
` (5 subsequent siblings)
7 siblings, 0 replies; 14+ messages in thread
From: Maxime Chevallier @ 2025-03-05 14:19 UTC (permalink / raw)
To: davem, Andrew Lunn, Jakub Kicinski, Eric Dumazet, Paolo Abeni,
Heiner Kallweit
Cc: Maxime Chevallier, netdev, linux-kernel, thomas.petazzoni,
linux-arm-kernel, Christophe Leroy, Herve Codina,
Florian Fainelli, Russell King, Vladimir Oltean,
Köry Maincent, Oleksij Rempel, Simon Horman, Romain Gantois,
Piergiorgio Beruto
As we work on getting more objects out of a per-netdev DUMP, rename
ethnl_default_dump_one() into ethnl_default_dump_one_dev(), making it
explicit that this dumps everything for one netdev.
Signed-off-by: Maxime Chevallier <maxime.chevallier@bootlin.com>
---
net/ethtool/netlink.c | 10 +++++-----
1 file changed, 5 insertions(+), 5 deletions(-)
diff --git a/net/ethtool/netlink.c b/net/ethtool/netlink.c
index 0815b28ba32f..6dddaea2babb 100644
--- a/net/ethtool/netlink.c
+++ b/net/ethtool/netlink.c
@@ -533,9 +533,9 @@ static int ethnl_default_doit(struct sk_buff *skb, struct genl_info *info)
return ret;
}
-static int ethnl_default_dump_one(struct sk_buff *skb, struct net_device *dev,
- const struct ethnl_dump_ctx *ctx,
- const struct genl_info *info)
+static int ethnl_default_dump_one_dev(struct sk_buff *skb, struct net_device *dev,
+ const struct ethnl_dump_ctx *ctx,
+ const struct genl_info *info)
{
void *ehdr;
int ret;
@@ -593,8 +593,8 @@ static int ethnl_default_dumpit(struct sk_buff *skb,
dev_hold(dev);
rcu_read_unlock();
- ret = ethnl_default_dump_one(skb, dev, ctx,
- genl_info_dump(cb));
+ ret = ethnl_default_dump_one_dev(skb, dev, ctx,
+ genl_info_dump(cb));
rcu_read_lock();
dev_put(dev);
--
2.48.1
^ permalink raw reply related [flat|nested] 14+ messages in thread
* [PATCH net-next 3/7] net: ethtool: netlink: Introduce command-specific dump_one_dev
2025-03-05 14:19 [PATCH net-next 0/7] net: ethtool: Introduce ethnl dump helpers Maxime Chevallier
2025-03-05 14:19 ` [PATCH net-next 1/7] net: ethtool: netlink: Allow per-netdevice DUMP operations Maxime Chevallier
2025-03-05 14:19 ` [PATCH net-next 2/7] net: ethtool: netlink: Rename ethnl_default_dump_one Maxime Chevallier
@ 2025-03-05 14:19 ` Maxime Chevallier
2025-03-05 14:19 ` [PATCH net-next 4/7] net: ethtool: netlink: Introduce per-phy DUMP helpers Maxime Chevallier
` (4 subsequent siblings)
7 siblings, 0 replies; 14+ messages in thread
From: Maxime Chevallier @ 2025-03-05 14:19 UTC (permalink / raw)
To: davem, Andrew Lunn, Jakub Kicinski, Eric Dumazet, Paolo Abeni,
Heiner Kallweit
Cc: Maxime Chevallier, netdev, linux-kernel, thomas.petazzoni,
linux-arm-kernel, Christophe Leroy, Herve Codina,
Florian Fainelli, Russell King, Vladimir Oltean,
Köry Maincent, Oleksij Rempel, Simon Horman, Romain Gantois,
Piergiorgio Beruto
Prepare more generic ethnl DUMP hanldling, by allowing netlink commands
to register their own dump_one_dev() callback. This avoids having to
roll with a fully custom genl ->dumpit callback, allowing the re-use of some
ethnl plumbing.
Fallback to the default dump_one_dev behaviour when no custom callback
is found.
The command dump context is maintained within the ethnl_dump_ctx, that
we move in netlink.h so that command handlers can access it.
This context can be allocated/freed in new ->dump_start() and
->dump_done() callbacks.
Signed-off-by: Maxime Chevallier <maxime.chevallier@bootlin.com>
---
net/ethtool/netlink.c | 58 +++++++++++++++++++++++++------------------
net/ethtool/netlink.h | 35 ++++++++++++++++++++++++++
2 files changed, 69 insertions(+), 24 deletions(-)
diff --git a/net/ethtool/netlink.c b/net/ethtool/netlink.c
index 6dddaea2babb..c0215f4acc05 100644
--- a/net/ethtool/netlink.c
+++ b/net/ethtool/netlink.c
@@ -336,24 +336,6 @@ int ethnl_multicast(struct sk_buff *skb, struct net_device *dev)
/* GET request helpers */
-/**
- * struct ethnl_dump_ctx - context structure for generic dumpit() callback
- * @ops: request ops of currently processed message type
- * @req_info: parsed request header of processed request
- * @reply_data: data needed to compose the reply
- * @pos_ifindex: saved iteration position - ifindex
- *
- * These parameters are kept in struct netlink_callback as context preserved
- * between iterations. They are initialized by ethnl_default_start() and used
- * in ethnl_default_dumpit() and ethnl_default_done().
- */
-struct ethnl_dump_ctx {
- const struct ethnl_request_ops *ops;
- struct ethnl_req_info *req_info;
- struct ethnl_reply_data *reply_data;
- unsigned long pos_ifindex;
-};
-
static const struct ethnl_request_ops *
ethnl_default_requests[__ETHTOOL_MSG_USER_CNT] = {
[ETHTOOL_MSG_STRSET_GET] = ðnl_strset_request_ops,
@@ -533,9 +515,9 @@ static int ethnl_default_doit(struct sk_buff *skb, struct genl_info *info)
return ret;
}
-static int ethnl_default_dump_one_dev(struct sk_buff *skb, struct net_device *dev,
- const struct ethnl_dump_ctx *ctx,
- const struct genl_info *info)
+static int ethnl_default_dump_one(struct sk_buff *skb,
+ const struct ethnl_dump_ctx *ctx,
+ const struct genl_info *info)
{
void *ehdr;
int ret;
@@ -546,13 +528,13 @@ static int ethnl_default_dump_one_dev(struct sk_buff *skb, struct net_device *de
if (!ehdr)
return -EMSGSIZE;
- ethnl_init_reply_data(ctx->reply_data, ctx->ops, dev);
rtnl_lock();
ret = ctx->ops->prepare_data(ctx->req_info, ctx->reply_data, info);
rtnl_unlock();
if (ret < 0)
goto out;
- ret = ethnl_fill_reply_header(skb, dev, ctx->ops->hdr_attr);
+ ret = ethnl_fill_reply_header(skb, ctx->reply_data->dev,
+ ctx->ops->hdr_attr);
if (ret < 0)
goto out;
ret = ctx->ops->fill_reply(skb, ctx->req_info, ctx->reply_data);
@@ -560,11 +542,29 @@ static int ethnl_default_dump_one_dev(struct sk_buff *skb, struct net_device *de
out:
if (ctx->ops->cleanup_data)
ctx->ops->cleanup_data(ctx->reply_data);
- ctx->reply_data->dev = NULL;
+
if (ret < 0)
genlmsg_cancel(skb, ehdr);
else
genlmsg_end(skb, ehdr);
+
+ return ret;
+}
+
+static int ethnl_default_dump_one_dev(struct sk_buff *skb, struct net_device *dev,
+ struct ethnl_dump_ctx *ctx,
+ const struct genl_info *info)
+{
+ int ret;
+
+ ethnl_init_reply_data(ctx->reply_data, ctx->ops, dev);
+
+ if (ctx->ops->dump_one_dev)
+ ret = ctx->ops->dump_one_dev(skb, ctx, info);
+ else
+ ret = ethnl_default_dump_one(skb, ctx, info);
+
+ ctx->reply_data->dev = NULL;
return ret;
}
@@ -593,6 +593,7 @@ static int ethnl_default_dumpit(struct sk_buff *skb,
dev_hold(dev);
rcu_read_unlock();
+ ctx->req_info->dev = dev;
ret = ethnl_default_dump_one_dev(skb, dev, ctx,
genl_info_dump(cb));
@@ -655,6 +656,12 @@ static int ethnl_default_start(struct netlink_callback *cb)
ctx->reply_data = reply_data;
ctx->pos_ifindex = 0;
+ if (ctx->ops->dump_start) {
+ ret = ctx->ops->dump_start(ctx);
+ if (ret)
+ goto free_reply_data;
+ }
+
return 0;
free_reply_data:
@@ -670,6 +677,9 @@ static int ethnl_default_done(struct netlink_callback *cb)
{
struct ethnl_dump_ctx *ctx = ethnl_dump_context(cb);
+ if (ctx->ops->dump_done)
+ ctx->ops->dump_done(ctx);
+
kfree(ctx->reply_data);
kfree(ctx->req_info);
diff --git a/net/ethtool/netlink.h b/net/ethtool/netlink.h
index 4db27182741f..d7506b08e5d6 100644
--- a/net/ethtool/netlink.h
+++ b/net/ethtool/netlink.h
@@ -307,6 +307,26 @@ struct ethnl_reply_data {
struct net_device *dev;
};
+/**
+ * struct ethnl_dump_ctx - context structure for generic dumpit() callback
+ * @ops: request ops of currently processed message type
+ * @req_info: parsed request header of processed request
+ * @reply_data: data needed to compose the reply
+ * @pos_ifindex: saved iteration position - ifindex
+ * @cmd_ctx: command-specific context to maintain across the dump.
+ *
+ * These parameters are kept in struct netlink_callback as context preserved
+ * between iterations. They are initialized by ethnl_default_start() and used
+ * in ethnl_default_dumpit() and ethnl_default_done().
+ */
+struct ethnl_dump_ctx {
+ const struct ethnl_request_ops *ops;
+ struct ethnl_req_info *req_info;
+ struct ethnl_reply_data *reply_data;
+ unsigned long pos_ifindex;
+ void *cmd_ctx;
+};
+
int ethnl_ops_begin(struct net_device *dev);
void ethnl_ops_complete(struct net_device *dev);
@@ -372,6 +392,15 @@ int ethnl_sock_priv_set(struct sk_buff *skb, struct net_device *dev, u32 portid,
* - 0 if no configuration has changed
* - 1 if configuration changed and notification should be generated
* - negative errno on errors
+ * @dump_start:
+ * Optional callback to prepare a dump operation, should there be a need
+ * to maintain some context across the dump.
+ * @dump_one_dev;
+ * Optional callback to generate all messages for a given netdev. This
+ * is relevant only when a request can produce different results for the
+ * same netdev depending on command-specific attributes.
+ * @dump_done:
+ * Optional callback to cleanup any context allocated in ->dump_start()
*
* Description of variable parts of GET request handling when using the
* unified infrastructure. When used, a pointer to an instance of this
@@ -408,6 +437,12 @@ struct ethnl_request_ops {
struct genl_info *info);
int (*set)(struct ethnl_req_info *req_info,
struct genl_info *info);
+
+ int (*dump_start)(struct ethnl_dump_ctx *ctx);
+ int (*dump_one_dev)(struct sk_buff *skb,
+ struct ethnl_dump_ctx *ctx,
+ const struct genl_info *info);
+ void (*dump_done)(struct ethnl_dump_ctx *ctx);
};
/* request handlers */
--
2.48.1
^ permalink raw reply related [flat|nested] 14+ messages in thread
* [PATCH net-next 4/7] net: ethtool: netlink: Introduce per-phy DUMP helpers
2025-03-05 14:19 [PATCH net-next 0/7] net: ethtool: Introduce ethnl dump helpers Maxime Chevallier
` (2 preceding siblings ...)
2025-03-05 14:19 ` [PATCH net-next 3/7] net: ethtool: netlink: Introduce command-specific dump_one_dev Maxime Chevallier
@ 2025-03-05 14:19 ` Maxime Chevallier
2025-03-05 14:19 ` [PATCH net-next 5/7] net: ethtool: phy: Convert the PHY_GET command to generic phy dump Maxime Chevallier
` (3 subsequent siblings)
7 siblings, 0 replies; 14+ messages in thread
From: Maxime Chevallier @ 2025-03-05 14:19 UTC (permalink / raw)
To: davem, Andrew Lunn, Jakub Kicinski, Eric Dumazet, Paolo Abeni,
Heiner Kallweit
Cc: Maxime Chevallier, netdev, linux-kernel, thomas.petazzoni,
linux-arm-kernel, Christophe Leroy, Herve Codina,
Florian Fainelli, Russell King, Vladimir Oltean,
Köry Maincent, Oleksij Rempel, Simon Horman, Romain Gantois,
Piergiorgio Beruto
As there are multiple ethnl commands that report messages based on
phy_device information, let's introduce a set of ethnl generic dump
helpers to allow DUMP support for each PHY on a given netdev.
This logic iterates over the phy_link_topology of each netdev (or a
single netdev for filtered DUMP), and call ethnl_default_dump_one() with
the req_info populated with ifindex + phyindex.
This allows re-using all the existing infra for phy-targetting commands
that already use ethnl generic helpers.
Signed-off-by: Maxime Chevallier <maxime.chevallier@bootlin.com>
---
net/ethtool/netlink.c | 53 +++++++++++++++++++++++++++++++++++++++++++
net/ethtool/netlink.h | 6 +++++
2 files changed, 59 insertions(+)
diff --git a/net/ethtool/netlink.c b/net/ethtool/netlink.c
index c0215f4acc05..474362af1aea 100644
--- a/net/ethtool/netlink.c
+++ b/net/ethtool/netlink.c
@@ -551,6 +551,59 @@ static int ethnl_default_dump_one(struct sk_buff *skb,
return ret;
}
+/* Specific context for phy-targeting command DUMP operatins. We keep in context
+ * the latest phy_index we dumped, in case of an interrupted DUMP.
+ */
+struct ethnl_dump_ctx_perphy {
+ unsigned long phy_index;
+};
+
+int ethnl_dump_start_perphy(struct ethnl_dump_ctx *ctx)
+{
+ struct ethnl_dump_ctx_perphy *dump_ctx;
+
+ dump_ctx = kzalloc(sizeof(*dump_ctx), GFP_KERNEL);
+ if (!dump_ctx)
+ return -ENOMEM;
+
+ ctx->cmd_ctx = dump_ctx;
+
+ return 0;
+}
+
+void ethnl_dump_done_perphy(struct ethnl_dump_ctx *ctx)
+{
+ kfree(ctx->cmd_ctx);
+}
+
+int ethnl_dump_one_dev_perphy(struct sk_buff *skb,
+ struct ethnl_dump_ctx *ctx,
+ const struct genl_info *info)
+{
+ struct ethnl_dump_ctx_perphy *dump_ctx = ctx->cmd_ctx;
+ struct net_device *dev = ctx->reply_data->dev;
+ struct phy_device_node *pdn;
+ int ret = 0;
+
+ if (!dev->link_topo)
+ return 0;
+
+ xa_for_each_start(&dev->link_topo->phys, dump_ctx->phy_index,
+ pdn, dump_ctx->phy_index) {
+ ctx->req_info->phy_index = dump_ctx->phy_index;
+
+ /* We can re-use the original dump_one as ->prepare_data in
+ * commands use ethnl_req_get_phydev(), which gets the PHY from
+ * what's in req_info
+ */
+ ret = ethnl_default_dump_one(skb, ctx, info);
+ if (ret)
+ break;
+ }
+
+ return ret;
+}
+
static int ethnl_default_dump_one_dev(struct sk_buff *skb, struct net_device *dev,
struct ethnl_dump_ctx *ctx,
const struct genl_info *info)
diff --git a/net/ethtool/netlink.h b/net/ethtool/netlink.h
index d7506b08e5d6..835376a9c84c 100644
--- a/net/ethtool/netlink.h
+++ b/net/ethtool/netlink.h
@@ -327,6 +327,12 @@ struct ethnl_dump_ctx {
void *cmd_ctx;
};
+/* Generic callbacks to be used by PHY targeting commands */
+int ethnl_dump_start_perphy(struct ethnl_dump_ctx *ctx);
+int ethnl_dump_one_dev_perphy(struct sk_buff *skb, struct ethnl_dump_ctx *ctx,
+ const struct genl_info *info);
+void ethnl_dump_done_perphy(struct ethnl_dump_ctx *ctx);
+
int ethnl_ops_begin(struct net_device *dev);
void ethnl_ops_complete(struct net_device *dev);
--
2.48.1
^ permalink raw reply related [flat|nested] 14+ messages in thread
* [PATCH net-next 5/7] net: ethtool: phy: Convert the PHY_GET command to generic phy dump
2025-03-05 14:19 [PATCH net-next 0/7] net: ethtool: Introduce ethnl dump helpers Maxime Chevallier
` (3 preceding siblings ...)
2025-03-05 14:19 ` [PATCH net-next 4/7] net: ethtool: netlink: Introduce per-phy DUMP helpers Maxime Chevallier
@ 2025-03-05 14:19 ` Maxime Chevallier
2025-03-05 14:19 ` [PATCH net-next 6/7] net: ethtool: plca: Use per-PHY DUMP operations Maxime Chevallier
` (2 subsequent siblings)
7 siblings, 0 replies; 14+ messages in thread
From: Maxime Chevallier @ 2025-03-05 14:19 UTC (permalink / raw)
To: davem, Andrew Lunn, Jakub Kicinski, Eric Dumazet, Paolo Abeni,
Heiner Kallweit
Cc: Maxime Chevallier, netdev, linux-kernel, thomas.petazzoni,
linux-arm-kernel, Christophe Leroy, Herve Codina,
Florian Fainelli, Russell King, Vladimir Oltean,
Köry Maincent, Oleksij Rempel, Simon Horman, Romain Gantois,
Piergiorgio Beruto
Now that we have an infrastructure in ethnl for perphy DUMPs, we can get
rid of the custom ->doit and ->dumpit to deal with PHY listing commands.
As most of the code was custom, this basically means re-writing how we
deal with PHY listing.
Signed-off-by: Maxime Chevallier <maxime.chevallier@bootlin.com>
---
net/ethtool/netlink.c | 9 +-
net/ethtool/netlink.h | 4 -
net/ethtool/phy.c | 335 ++++++++++++------------------------------
3 files changed, 103 insertions(+), 245 deletions(-)
diff --git a/net/ethtool/netlink.c b/net/ethtool/netlink.c
index 474362af1aea..b4d774786849 100644
--- a/net/ethtool/netlink.c
+++ b/net/ethtool/netlink.c
@@ -379,6 +379,7 @@ ethnl_default_requests[__ETHTOOL_MSG_USER_CNT] = {
[ETHTOOL_MSG_MM_SET] = ðnl_mm_request_ops,
[ETHTOOL_MSG_TSCONFIG_GET] = ðnl_tsconfig_request_ops,
[ETHTOOL_MSG_TSCONFIG_SET] = ðnl_tsconfig_request_ops,
+ [ETHTOOL_MSG_PHY_GET] = ðnl_phy_request_ops,
};
static struct ethnl_dump_ctx *ethnl_dump_context(struct netlink_callback *cb)
@@ -1332,10 +1333,10 @@ static const struct genl_ops ethtool_genl_ops[] = {
},
{
.cmd = ETHTOOL_MSG_PHY_GET,
- .doit = ethnl_phy_doit,
- .start = ethnl_phy_start,
- .dumpit = ethnl_phy_dumpit,
- .done = ethnl_phy_done,
+ .doit = ethnl_default_doit,
+ .start = ethnl_default_start,
+ .dumpit = ethnl_default_dumpit,
+ .done = ethnl_default_done,
.policy = ethnl_phy_get_policy,
.maxattr = ARRAY_SIZE(ethnl_phy_get_policy) - 1,
},
diff --git a/net/ethtool/netlink.h b/net/ethtool/netlink.h
index 835376a9c84c..bc6dde699a32 100644
--- a/net/ethtool/netlink.h
+++ b/net/ethtool/netlink.h
@@ -541,10 +541,6 @@ int ethnl_tunnel_info_dumpit(struct sk_buff *skb, struct netlink_callback *cb);
int ethnl_act_module_fw_flash(struct sk_buff *skb, struct genl_info *info);
int ethnl_rss_dump_start(struct netlink_callback *cb);
int ethnl_rss_dumpit(struct sk_buff *skb, struct netlink_callback *cb);
-int ethnl_phy_start(struct netlink_callback *cb);
-int ethnl_phy_doit(struct sk_buff *skb, struct genl_info *info);
-int ethnl_phy_dumpit(struct sk_buff *skb, struct netlink_callback *cb);
-int ethnl_phy_done(struct netlink_callback *cb);
int ethnl_tsinfo_start(struct netlink_callback *cb);
int ethnl_tsinfo_dumpit(struct sk_buff *skb, struct netlink_callback *cb);
int ethnl_tsinfo_done(struct netlink_callback *cb);
diff --git a/net/ethtool/phy.c b/net/ethtool/phy.c
index ed8f690f6bac..90c27dd1a5c6 100644
--- a/net/ethtool/phy.c
+++ b/net/ethtool/phy.c
@@ -11,296 +11,157 @@
#include <linux/sfp.h>
struct phy_req_info {
- struct ethnl_req_info base;
- struct phy_device_node *pdn;
+ struct ethnl_req_info base;
};
-#define PHY_REQINFO(__req_base) \
- container_of(__req_base, struct phy_req_info, base)
+struct phy_reply_data {
+ struct ethnl_reply_data base;
+ u32 phyindex;
+ char *drvname;
+ char *name;
+ unsigned int upstream_type;
+ char *upstream_sfp_name;
+ unsigned int upstream_index;
+ char *downstream_sfp_name;
+};
+
+#define PHY_REPDATA(__reply_base) \
+ container_of(__reply_base, struct phy_reply_data, base)
const struct nla_policy ethnl_phy_get_policy[ETHTOOL_A_PHY_HEADER + 1] = {
[ETHTOOL_A_PHY_HEADER] = NLA_POLICY_NESTED(ethnl_header_policy),
};
-/* Caller holds rtnl */
-static ssize_t
-ethnl_phy_reply_size(const struct ethnl_req_info *req_base,
- struct netlink_ext_ack *extack)
+static int phy_reply_size(const struct ethnl_req_info *req_info,
+ const struct ethnl_reply_data *reply_data)
{
- struct phy_req_info *req_info = PHY_REQINFO(req_base);
- struct phy_device_node *pdn = req_info->pdn;
- struct phy_device *phydev = pdn->phy;
+ struct phy_reply_data *rep_data = PHY_REPDATA(reply_data);
size_t size = 0;
- ASSERT_RTNL();
-
/* ETHTOOL_A_PHY_INDEX */
size += nla_total_size(sizeof(u32));
/* ETHTOOL_A_DRVNAME */
- if (phydev->drv)
- size += nla_total_size(strlen(phydev->drv->name) + 1);
+ if (rep_data->drvname)
+ size += nla_total_size(strlen(rep_data->drvname) + 1);
/* ETHTOOL_A_NAME */
- size += nla_total_size(strlen(dev_name(&phydev->mdio.dev)) + 1);
+ size += nla_total_size(strlen(rep_data->name) + 1);
/* ETHTOOL_A_PHY_UPSTREAM_TYPE */
size += nla_total_size(sizeof(u32));
- if (phy_on_sfp(phydev)) {
- const char *upstream_sfp_name = sfp_get_name(pdn->parent_sfp_bus);
-
- /* ETHTOOL_A_PHY_UPSTREAM_SFP_NAME */
- if (upstream_sfp_name)
- size += nla_total_size(strlen(upstream_sfp_name) + 1);
+ if (rep_data->upstream_sfp_name)
+ size += nla_total_size(strlen(rep_data->upstream_sfp_name) + 1);
- /* ETHTOOL_A_PHY_UPSTREAM_INDEX */
+ if (rep_data->upstream_index)
size += nla_total_size(sizeof(u32));
- }
-
- /* ETHTOOL_A_PHY_DOWNSTREAM_SFP_NAME */
- if (phydev->sfp_bus) {
- const char *sfp_name = sfp_get_name(phydev->sfp_bus);
- if (sfp_name)
- size += nla_total_size(strlen(sfp_name) + 1);
- }
+ if (rep_data->downstream_sfp_name)
+ size += nla_total_size(strlen(rep_data->downstream_sfp_name) + 1);
return size;
}
-static int
-ethnl_phy_fill_reply(const struct ethnl_req_info *req_base, struct sk_buff *skb)
+static int phy_prepare_data(const struct ethnl_req_info *req_info,
+ struct ethnl_reply_data *reply_data,
+ const struct genl_info *info)
{
- struct phy_req_info *req_info = PHY_REQINFO(req_base);
- struct phy_device_node *pdn = req_info->pdn;
- struct phy_device *phydev = pdn->phy;
- enum phy_upstream ptype;
+ struct phy_reply_data *rep_data = PHY_REPDATA(reply_data);
+ struct phy_link_topology *topo = reply_data->dev->link_topo;
+ struct nlattr **tb = info->attrs;
+ struct phy_device_node *pdn;
+ struct phy_device *phydev;
- ptype = pdn->upstream_type;
+ /* RTNL is held by th caller */
+ phydev = ethnl_req_get_phydev(req_info, tb, ETHTOOL_A_PHY_HEADER,
+ info->extack);
+ if (IS_ERR_OR_NULL(phydev))
+ return -EOPNOTSUPP;
- if (nla_put_u32(skb, ETHTOOL_A_PHY_INDEX, phydev->phyindex) ||
- nla_put_string(skb, ETHTOOL_A_PHY_NAME, dev_name(&phydev->mdio.dev)) ||
- nla_put_u32(skb, ETHTOOL_A_PHY_UPSTREAM_TYPE, ptype))
- return -EMSGSIZE;
+ pdn = xa_load(&topo->phys, phydev->phyindex);
+ if (!pdn)
+ return -EOPNOTSUPP;
- if (phydev->drv &&
- nla_put_string(skb, ETHTOOL_A_PHY_DRVNAME, phydev->drv->name))
- return -EMSGSIZE;
+ rep_data->phyindex = phydev->phyindex;
+ rep_data->name = kstrdup(dev_name(&phydev->mdio.dev), GFP_KERNEL);
+ rep_data->drvname = kstrdup(phydev->drv->name, GFP_KERNEL);
+ rep_data->upstream_type = pdn->upstream_type;
- if (ptype == PHY_UPSTREAM_PHY) {
+ if (pdn->upstream_type == PHY_UPSTREAM_PHY) {
struct phy_device *upstream = pdn->upstream.phydev;
- const char *sfp_upstream_name;
-
- /* Parent index */
- if (nla_put_u32(skb, ETHTOOL_A_PHY_UPSTREAM_INDEX, upstream->phyindex))
- return -EMSGSIZE;
-
- if (pdn->parent_sfp_bus) {
- sfp_upstream_name = sfp_get_name(pdn->parent_sfp_bus);
- if (sfp_upstream_name &&
- nla_put_string(skb, ETHTOOL_A_PHY_UPSTREAM_SFP_NAME,
- sfp_upstream_name))
- return -EMSGSIZE;
- }
+ rep_data->upstream_index = upstream->phyindex;
}
- if (phydev->sfp_bus) {
- const char *sfp_name = sfp_get_name(phydev->sfp_bus);
+ if (pdn->parent_sfp_bus)
+ rep_data->upstream_sfp_name = kstrdup(sfp_get_name(pdn->parent_sfp_bus),
+ GFP_KERNEL);
- if (sfp_name &&
- nla_put_string(skb, ETHTOOL_A_PHY_DOWNSTREAM_SFP_NAME,
- sfp_name))
- return -EMSGSIZE;
- }
+ if (phydev->sfp_bus)
+ rep_data->downstream_sfp_name = kstrdup(sfp_get_name(phydev->sfp_bus),
+ GFP_KERNEL);
return 0;
}
-static int ethnl_phy_parse_request(struct ethnl_req_info *req_base,
- struct nlattr **tb,
- struct netlink_ext_ack *extack)
+static int phy_fill_reply(struct sk_buff *skb,
+ const struct ethnl_req_info *req_info,
+ const struct ethnl_reply_data *reply_data)
{
- struct phy_link_topology *topo = req_base->dev->link_topo;
- struct phy_req_info *req_info = PHY_REQINFO(req_base);
- struct phy_device *phydev;
-
- phydev = ethnl_req_get_phydev(req_base, tb[ETHTOOL_A_PHY_HEADER],
- extack);
- if (!phydev)
- return 0;
-
- if (IS_ERR(phydev))
- return PTR_ERR(phydev);
-
- if (!topo)
- return 0;
-
- req_info->pdn = xa_load(&topo->phys, phydev->phyindex);
-
- return 0;
-}
-
-int ethnl_phy_doit(struct sk_buff *skb, struct genl_info *info)
-{
- struct phy_req_info req_info = {};
- struct nlattr **tb = info->attrs;
- struct sk_buff *rskb;
- void *reply_payload;
- int reply_len;
- int ret;
-
- ret = ethnl_parse_header_dev_get(&req_info.base,
- tb[ETHTOOL_A_PHY_HEADER],
- genl_info_net(info), info->extack,
- true);
- if (ret < 0)
- return ret;
-
- rtnl_lock();
-
- ret = ethnl_phy_parse_request(&req_info.base, tb, info->extack);
- if (ret < 0)
- goto err_unlock_rtnl;
-
- /* No PHY, return early */
- if (!req_info.pdn)
- goto err_unlock_rtnl;
-
- ret = ethnl_phy_reply_size(&req_info.base, info->extack);
- if (ret < 0)
- goto err_unlock_rtnl;
- reply_len = ret + ethnl_reply_header_size();
-
- rskb = ethnl_reply_init(reply_len, req_info.base.dev,
- ETHTOOL_MSG_PHY_GET_REPLY,
- ETHTOOL_A_PHY_HEADER,
- info, &reply_payload);
- if (!rskb) {
- ret = -ENOMEM;
- goto err_unlock_rtnl;
- }
+ struct phy_reply_data *rep_data = PHY_REPDATA(reply_data);
- ret = ethnl_phy_fill_reply(&req_info.base, rskb);
- if (ret)
- goto err_free_msg;
-
- rtnl_unlock();
- ethnl_parse_header_dev_put(&req_info.base);
- genlmsg_end(rskb, reply_payload);
-
- return genlmsg_reply(rskb, info);
-
-err_free_msg:
- nlmsg_free(rskb);
-err_unlock_rtnl:
- rtnl_unlock();
- ethnl_parse_header_dev_put(&req_info.base);
- return ret;
-}
-
-struct ethnl_phy_dump_ctx {
- struct phy_req_info *phy_req_info;
- unsigned long ifindex;
- unsigned long phy_index;
-};
-
-int ethnl_phy_start(struct netlink_callback *cb)
-{
- const struct genl_info *info = genl_info_dump(cb);
- struct ethnl_phy_dump_ctx *ctx = (void *)cb->ctx;
- int ret;
+ if (nla_put_u32(skb, ETHTOOL_A_PHY_INDEX, rep_data->phyindex) ||
+ nla_put_string(skb, ETHTOOL_A_PHY_NAME, rep_data->name) ||
+ nla_put_u32(skb, ETHTOOL_A_PHY_UPSTREAM_TYPE, rep_data->upstream_type))
+ return -EMSGSIZE;
- BUILD_BUG_ON(sizeof(*ctx) > sizeof(cb->ctx));
+ if (rep_data->drvname &&
+ nla_put_string(skb, ETHTOOL_A_PHY_DRVNAME, rep_data->drvname))
+ return -EMSGSIZE;
- ctx->phy_req_info = kzalloc(sizeof(*ctx->phy_req_info), GFP_KERNEL);
- if (!ctx->phy_req_info)
- return -ENOMEM;
+ if (rep_data->upstream_index &&
+ nla_put_u32(skb, ETHTOOL_A_PHY_UPSTREAM_INDEX,
+ rep_data->upstream_index))
+ return -EMSGSIZE;
- ret = ethnl_parse_header_dev_get(&ctx->phy_req_info->base,
- info->attrs[ETHTOOL_A_PHY_HEADER],
- sock_net(cb->skb->sk), cb->extack,
- false);
- ctx->ifindex = 0;
- ctx->phy_index = 0;
+ if (rep_data->upstream_sfp_name &&
+ nla_put_string(skb, ETHTOOL_A_PHY_UPSTREAM_SFP_NAME,
+ rep_data->upstream_sfp_name))
+ return -EMSGSIZE;
- if (ret)
- kfree(ctx->phy_req_info);
+ if (rep_data->downstream_sfp_name &&
+ nla_put_string(skb, ETHTOOL_A_PHY_DOWNSTREAM_SFP_NAME,
+ rep_data->downstream_sfp_name))
+ return -EMSGSIZE;
- return ret;
+ return 0;
}
-int ethnl_phy_done(struct netlink_callback *cb)
+static void phy_cleanup_data(struct ethnl_reply_data *reply_data)
{
- struct ethnl_phy_dump_ctx *ctx = (void *)cb->ctx;
-
- if (ctx->phy_req_info->base.dev)
- ethnl_parse_header_dev_put(&ctx->phy_req_info->base);
-
- kfree(ctx->phy_req_info);
+ struct phy_reply_data *rep_data = PHY_REPDATA(reply_data);
- return 0;
+ kfree(rep_data->drvname);
+ kfree(rep_data->name);
+ kfree(rep_data->upstream_sfp_name);
+ kfree(rep_data->downstream_sfp_name);
}
-static int ethnl_phy_dump_one_dev(struct sk_buff *skb, struct net_device *dev,
- struct netlink_callback *cb)
-{
- struct ethnl_phy_dump_ctx *ctx = (void *)cb->ctx;
- struct phy_req_info *pri = ctx->phy_req_info;
- struct phy_device_node *pdn;
- int ret = 0;
- void *ehdr;
-
- if (!dev->link_topo)
- return 0;
-
- xa_for_each_start(&dev->link_topo->phys, ctx->phy_index, pdn, ctx->phy_index) {
- ehdr = ethnl_dump_put(skb, cb, ETHTOOL_MSG_PHY_GET_REPLY);
- if (!ehdr) {
- ret = -EMSGSIZE;
- break;
- }
-
- ret = ethnl_fill_reply_header(skb, dev, ETHTOOL_A_PHY_HEADER);
- if (ret < 0) {
- genlmsg_cancel(skb, ehdr);
- break;
- }
-
- pri->pdn = pdn;
- ret = ethnl_phy_fill_reply(&pri->base, skb);
- if (ret < 0) {
- genlmsg_cancel(skb, ehdr);
- break;
- }
-
- genlmsg_end(skb, ehdr);
- }
+const struct ethnl_request_ops ethnl_phy_request_ops = {
+ .request_cmd = ETHTOOL_MSG_PHY_GET,
+ .reply_cmd = ETHTOOL_MSG_PHY_GET_REPLY,
+ .hdr_attr = ETHTOOL_A_PHY_HEADER,
+ .req_info_size = sizeof(struct phy_req_info),
+ .reply_data_size = sizeof(struct phy_reply_data),
- return ret;
-}
+ .prepare_data = phy_prepare_data,
+ .reply_size = phy_reply_size,
+ .fill_reply = phy_fill_reply,
+ .cleanup_data = phy_cleanup_data,
-int ethnl_phy_dumpit(struct sk_buff *skb, struct netlink_callback *cb)
-{
- struct ethnl_phy_dump_ctx *ctx = (void *)cb->ctx;
- struct net *net = sock_net(skb->sk);
- struct net_device *dev;
- int ret = 0;
-
- rtnl_lock();
-
- if (ctx->phy_req_info->base.dev) {
- ret = ethnl_phy_dump_one_dev(skb, ctx->phy_req_info->base.dev, cb);
- } else {
- for_each_netdev_dump(net, dev, ctx->ifindex) {
- ret = ethnl_phy_dump_one_dev(skb, dev, cb);
- if (ret)
- break;
-
- ctx->phy_index = 0;
- }
- }
- rtnl_unlock();
+ .dump_start = ethnl_dump_start_perphy,
+ .dump_one_dev = ethnl_dump_one_dev_perphy,
+ .dump_done = ethnl_dump_done_perphy,
- return ret;
-}
+ .allow_pernetdev_dump = true,
+};
--
2.48.1
^ permalink raw reply related [flat|nested] 14+ messages in thread
* [PATCH net-next 6/7] net: ethtool: plca: Use per-PHY DUMP operations
2025-03-05 14:19 [PATCH net-next 0/7] net: ethtool: Introduce ethnl dump helpers Maxime Chevallier
` (4 preceding siblings ...)
2025-03-05 14:19 ` [PATCH net-next 5/7] net: ethtool: phy: Convert the PHY_GET command to generic phy dump Maxime Chevallier
@ 2025-03-05 14:19 ` Maxime Chevallier
2025-03-05 14:19 ` [PATCH net-next 7/7] net: ethtool: pse-pd: " Maxime Chevallier
2025-03-05 17:02 ` [PATCH net-next 0/7] net: ethtool: Introduce ethnl dump helpers Maxime Chevallier
7 siblings, 0 replies; 14+ messages in thread
From: Maxime Chevallier @ 2025-03-05 14:19 UTC (permalink / raw)
To: davem, Andrew Lunn, Jakub Kicinski, Eric Dumazet, Paolo Abeni,
Heiner Kallweit
Cc: Maxime Chevallier, netdev, linux-kernel, thomas.petazzoni,
linux-arm-kernel, Christophe Leroy, Herve Codina,
Florian Fainelli, Russell King, Vladimir Oltean,
Köry Maincent, Oleksij Rempel, Simon Horman, Romain Gantois,
Piergiorgio Beruto
Leverage the per-phy ethnl DUMP helpers in case we have more that one
PLCA-able PHY on the link.
This is done for both PLCA status and PLCA config.
Signed-off-by: Maxime Chevallier <maxime.chevallier@bootlin.com>
---
net/ethtool/plca.c | 12 ++++++++++++
1 file changed, 12 insertions(+)
diff --git a/net/ethtool/plca.c b/net/ethtool/plca.c
index e1f7820a6158..2148ff607561 100644
--- a/net/ethtool/plca.c
+++ b/net/ethtool/plca.c
@@ -191,6 +191,12 @@ const struct ethnl_request_ops ethnl_plca_cfg_request_ops = {
.set = ethnl_set_plca,
.set_ntf_cmd = ETHTOOL_MSG_PLCA_NTF,
+
+ .dump_start = ethnl_dump_start_perphy,
+ .dump_one_dev = ethnl_dump_one_dev_perphy,
+ .dump_done = ethnl_dump_done_perphy,
+
+ .allow_pernetdev_dump = true,
};
// PLCA get status message -------------------------------------------------- //
@@ -268,4 +274,10 @@ const struct ethnl_request_ops ethnl_plca_status_request_ops = {
.prepare_data = plca_get_status_prepare_data,
.reply_size = plca_get_status_reply_size,
.fill_reply = plca_get_status_fill_reply,
+
+ .dump_start = ethnl_dump_start_perphy,
+ .dump_one_dev = ethnl_dump_one_dev_perphy,
+ .dump_done = ethnl_dump_done_perphy,
+
+ .allow_pernetdev_dump = true,
};
--
2.48.1
^ permalink raw reply related [flat|nested] 14+ messages in thread
* [PATCH net-next 7/7] net: ethtool: pse-pd: Use per-PHY DUMP operations
2025-03-05 14:19 [PATCH net-next 0/7] net: ethtool: Introduce ethnl dump helpers Maxime Chevallier
` (5 preceding siblings ...)
2025-03-05 14:19 ` [PATCH net-next 6/7] net: ethtool: plca: Use per-PHY DUMP operations Maxime Chevallier
@ 2025-03-05 14:19 ` Maxime Chevallier
2025-03-05 17:02 ` [PATCH net-next 0/7] net: ethtool: Introduce ethnl dump helpers Maxime Chevallier
7 siblings, 0 replies; 14+ messages in thread
From: Maxime Chevallier @ 2025-03-05 14:19 UTC (permalink / raw)
To: davem, Andrew Lunn, Jakub Kicinski, Eric Dumazet, Paolo Abeni,
Heiner Kallweit
Cc: Maxime Chevallier, netdev, linux-kernel, thomas.petazzoni,
linux-arm-kernel, Christophe Leroy, Herve Codina,
Florian Fainelli, Russell King, Vladimir Oltean,
Köry Maincent, Oleksij Rempel, Simon Horman, Romain Gantois,
Piergiorgio Beruto
Leverage the per-phy ethnl DUMP helpers in case we have more that one
PSE PHY on the link.
Signed-off-by: Maxime Chevallier <maxime.chevallier@bootlin.com>
---
net/ethtool/pse-pd.c | 6 ++++++
1 file changed, 6 insertions(+)
diff --git a/net/ethtool/pse-pd.c b/net/ethtool/pse-pd.c
index 4f6b99eab2a6..f3d14be8bdd9 100644
--- a/net/ethtool/pse-pd.c
+++ b/net/ethtool/pse-pd.c
@@ -314,4 +314,10 @@ const struct ethnl_request_ops ethnl_pse_request_ops = {
.set = ethnl_set_pse,
/* PSE has no notification */
+
+ .dump_start = ethnl_dump_start_perphy,
+ .dump_one_dev = ethnl_dump_one_dev_perphy,
+ .dump_done = ethnl_dump_done_perphy,
+
+ .allow_pernetdev_dump = true,
};
--
2.48.1
^ permalink raw reply related [flat|nested] 14+ messages in thread
* Re: [PATCH net-next 0/7] net: ethtool: Introduce ethnl dump helpers
2025-03-05 14:19 [PATCH net-next 0/7] net: ethtool: Introduce ethnl dump helpers Maxime Chevallier
` (6 preceding siblings ...)
2025-03-05 14:19 ` [PATCH net-next 7/7] net: ethtool: pse-pd: " Maxime Chevallier
@ 2025-03-05 17:02 ` Maxime Chevallier
2025-03-06 2:47 ` Jakub Kicinski
7 siblings, 1 reply; 14+ messages in thread
From: Maxime Chevallier @ 2025-03-05 17:02 UTC (permalink / raw)
To: davem, Andrew Lunn, Jakub Kicinski, Eric Dumazet, Paolo Abeni,
Heiner Kallweit
Cc: netdev, linux-kernel, thomas.petazzoni, linux-arm-kernel,
Christophe Leroy, Herve Codina, Florian Fainelli, Russell King,
Vladimir Oltean, Köry Maincent, Oleksij Rempel, Simon Horman,
Romain Gantois, Piergiorgio Beruto, Stanislav Fomichev
On Wed, 5 Mar 2025 15:19:30 +0100
Maxime Chevallier <maxime.chevallier@bootlin.com> wrote:
> Hi everyone,
>
> This series adds some scaffolding into ethnl to ease the support of
> DUMP operations.
>
> As of today when using ethnl's default ops, the DUMP requests will
> simply perform a GET for each netdev.
>
> That hits limitations for commands that may return multiple messages for
> a single netdev, such as :
>
> - RSS (listing contexts)
> - All PHY-specific commands (PLCA, PSE-PD, phy)
> - tsinfo (one item for the netdev + one per phy)
>
> Commands that need a non-default DUMP support have to re-implement
> ->dumpit() themselves, which prevents using most of ethnl's internal
> circuitry.
>
> This series therefore introduces a better support for dump operations in
> ethnl.
>
> The patches 1 and 2 introduce the support for filtered DUMPs, where an
> ifindex/ifname can be passed in the request header for the DUMP
> operation. This is for when we want to dump everything a netdev
> supports, but without doing so for every single netdev. ethtool's
> "--show-phys ethX" option for example performs a filtered dump.
>
> Patch 3 introduces 3 new ethnl ops :
> ->dump_start() to initialize a dump context
> ->dump_one_dev(), that can be implemented per-command to dump
> everything on a given netdev
> ->dump_done() to release the context
>
> The default behaviour for dumps remains the same, calling the whole
> ->doit() path for each netdev.
>
> Patch 4 introduces a set of ->dump_start(), ->dump_one_dev() and
> ->dump_done() callback implementations that can simply be plugged into
> the existing commands that list objects per-phy, making the
> phy-targeting command behaviour more coherent.
>
> Patch 5 uses that new set of helpers to rewrite the phy.c support, which
> now uses the regulat ethnl_ops instead of fully custom genl ops. This
> one is the hardest to review, sorry about that, I couldn't really manage
> to incrementally rework that file :(
>
> Patches 6 and 7 are where the new dump infra shines, adding per-netdev
> per-phy dump support for PLCA and PSE-PD.
>
> We could also consider converting tsinfo/tsconfig, rss and tunnels to
> these new ->dump_***() operations as well, but that's out of this
> series' scope.
>
> I've tested that series with some netdevsim PHY patches that I plan to
> submit (they can be found here [1]), with the refcount tracker
> for net/netns enabled to make sure the lock usage is somewhat coherent.
>
> Thanks,
>
> Maxime
>
> [1]: https://github.com/minimaxwell/linux/tree/mc/netdevsim-phy
>
This series will very likely conflict with Stanislav's netdev lock
work [2], I'll of course be happy to rebase should it get merged :)
Thanks,
Maxime
[2]: https://lore.kernel.org/netdev/20250305163732.2766420-1-sdf@fomichev.me/T/#t
>
> Maxime Chevallier (7):
> net: ethtool: netlink: Allow per-netdevice DUMP operations
> net: ethtool: netlink: Rename ethnl_default_dump_one
> net: ethtool: netlink: Introduce command-specific dump_one_dev
> net: ethtool: netlink: Introduce per-phy DUMP helpers
> net: ethtool: phy: Convert the PHY_GET command to generic phy dump
> net: ethtool: plca: Use per-PHY DUMP operations
> net: ethtool: pse-pd: Use per-PHY DUMP operations
>
> net/ethtool/netlink.c | 161 ++++++++++++++------
> net/ethtool/netlink.h | 46 +++++-
> net/ethtool/phy.c | 335 ++++++++++++------------------------------
> net/ethtool/plca.c | 12 ++
> net/ethtool/pse-pd.c | 6 +
> 5 files changed, 277 insertions(+), 283 deletions(-)
>
^ permalink raw reply [flat|nested] 14+ messages in thread
* Re: [PATCH net-next 0/7] net: ethtool: Introduce ethnl dump helpers
2025-03-05 17:02 ` [PATCH net-next 0/7] net: ethtool: Introduce ethnl dump helpers Maxime Chevallier
@ 2025-03-06 2:47 ` Jakub Kicinski
2025-03-06 7:42 ` Maxime Chevallier
0 siblings, 1 reply; 14+ messages in thread
From: Jakub Kicinski @ 2025-03-06 2:47 UTC (permalink / raw)
To: Maxime Chevallier
Cc: davem, Andrew Lunn, Eric Dumazet, Paolo Abeni, Heiner Kallweit,
netdev, linux-kernel, thomas.petazzoni, linux-arm-kernel,
Christophe Leroy, Herve Codina, Florian Fainelli, Russell King,
Vladimir Oltean, Köry Maincent, Oleksij Rempel, Simon Horman,
Romain Gantois, Piergiorgio Beruto, Stanislav Fomichev
On Wed, 5 Mar 2025 18:02:52 +0100 Maxime Chevallier wrote:
> This series will very likely conflict with Stanislav's netdev lock
> work [2], I'll of course be happy to rebase should it get merged :)
Also this doesn't build. Please hold off on reposting for a couple of
days - because of the large conflict radius the previous few revisions
of the locking series haven't even gotten into the selftest CI queue :(
^ permalink raw reply [flat|nested] 14+ messages in thread
* Re: [PATCH net-next 0/7] net: ethtool: Introduce ethnl dump helpers
2025-03-06 2:47 ` Jakub Kicinski
@ 2025-03-06 7:42 ` Maxime Chevallier
0 siblings, 0 replies; 14+ messages in thread
From: Maxime Chevallier @ 2025-03-06 7:42 UTC (permalink / raw)
To: Jakub Kicinski
Cc: davem, Andrew Lunn, Eric Dumazet, Paolo Abeni, Heiner Kallweit,
netdev, linux-kernel, thomas.petazzoni, linux-arm-kernel,
Christophe Leroy, Herve Codina, Florian Fainelli, Russell King,
Vladimir Oltean, Köry Maincent, Oleksij Rempel, Simon Horman,
Romain Gantois, Piergiorgio Beruto, Stanislav Fomichev
On Wed, 5 Mar 2025 18:47:49 -0800
Jakub Kicinski <kuba@kernel.org> wrote:
> On Wed, 5 Mar 2025 18:02:52 +0100 Maxime Chevallier wrote:
> > This series will very likely conflict with Stanislav's netdev lock
> > work [2], I'll of course be happy to rebase should it get merged :)
>
> Also this doesn't build. Please hold off on reposting for a couple of
> days - because of the large conflict radius the previous few revisions
> of the locking series haven't even gotten into the selftest CI queue :(
Sure no problem. I'll try t figure out what's up with the build in the
meantime, looks like I inverted a few changes that break
bissecatability.
I'll keep an eye on Stanislav's work in the meantime as well.
Thanks,
Maxime
^ permalink raw reply [flat|nested] 14+ messages in thread
* Re: [PATCH net-next 1/7] net: ethtool: netlink: Allow per-netdevice DUMP operations
2025-03-05 14:19 ` [PATCH net-next 1/7] net: ethtool: netlink: Allow per-netdevice DUMP operations Maxime Chevallier
@ 2025-03-07 12:21 ` Simon Horman
2025-03-07 13:18 ` Maxime Chevallier
0 siblings, 1 reply; 14+ messages in thread
From: Simon Horman @ 2025-03-07 12:21 UTC (permalink / raw)
To: Maxime Chevallier
Cc: davem, Andrew Lunn, Jakub Kicinski, Eric Dumazet, Paolo Abeni,
Heiner Kallweit, netdev, linux-kernel, thomas.petazzoni,
linux-arm-kernel, Christophe Leroy, Herve Codina,
Florian Fainelli, Russell King, Vladimir Oltean,
Köry Maincent, Oleksij Rempel, Romain Gantois,
Piergiorgio Beruto
On Wed, Mar 05, 2025 at 03:19:31PM +0100, Maxime Chevallier wrote:
> We have a number of netlink commands in the ethnl family that may have
> multiple objects to dump even for a single net_device, including :
>
> - PLCA, PSE-PD, phy: one message per PHY device
> - tsinfo: one message per timestamp source (netdev + phys)
> - rss: One per RSS context
>
> To get this behaviour, these netlink commands need to roll a custom
> ->dumpit().
>
> To prepare making per-netdev DUMP more generic in ethnl, introduce a
> member in the ethnl ops to indicate if a given command may allow
> pernetdev DUMPs (also referred to as filtered DUMPs).
>
> Signed-off-by: Maxime Chevallier <maxime.chevallier@bootlin.com>
> ---
> net/ethtool/netlink.c | 45 ++++++++++++++++++++++++++++---------------
> net/ethtool/netlink.h | 1 +
> 2 files changed, 30 insertions(+), 16 deletions(-)
>
> diff --git a/net/ethtool/netlink.c b/net/ethtool/netlink.c
> index 734849a57369..0815b28ba32f 100644
> --- a/net/ethtool/netlink.c
> +++ b/net/ethtool/netlink.c
> @@ -578,21 +578,34 @@ static int ethnl_default_dumpit(struct sk_buff *skb,
> int ret = 0;
>
> rcu_read_lock();
> - for_each_netdev_dump(net, dev, ctx->pos_ifindex) {
> - dev_hold(dev);
> + if (ctx->req_info->dev) {
> + dev = ctx->req_info->dev;
> rcu_read_unlock();
> -
> - ret = ethnl_default_dump_one(skb, dev, ctx, genl_info_dump(cb));
> -
> + /* Filtered DUMP request targeted to a single netdev. We already
> + * hold a ref to the netdev from ->start()
> + */
> + ret = ethnl_default_dump_one_dev(skb, dev, ctx,
> + genl_info_dump(cb));
Hi Maxime,
ethnl_default_dump_one_dev() is called here but it doesn't exist
until the following patch is applied, which breaks bisection.
> rcu_read_lock();
> - dev_put(dev);
> -
> - if (ret < 0 && ret != -EOPNOTSUPP) {
> - if (likely(skb->len))
> - ret = skb->len;
> - break;
> + netdev_put(ctx->req_info->dev, &ctx->req_info->dev_tracker);
> + } else {
> + for_each_netdev_dump(net, dev, ctx->pos_ifindex) {
> + dev_hold(dev);
> + rcu_read_unlock();
> +
> + ret = ethnl_default_dump_one(skb, dev, ctx,
> + genl_info_dump(cb));
> +
> + rcu_read_lock();
> + dev_put(dev);
> +
> + if (ret < 0 && ret != -EOPNOTSUPP) {
> + if (likely(skb->len))
> + ret = skb->len;
> + break;
> + }
> + ret = 0;
> }
> - ret = 0;
> }
> rcu_read_unlock();
>
...
> diff --git a/net/ethtool/netlink.h b/net/ethtool/netlink.h
> index ec6ab5443a6f..4db27182741f 100644
> --- a/net/ethtool/netlink.h
> +++ b/net/ethtool/netlink.h
> @@ -388,6 +388,7 @@ struct ethnl_request_ops {
> unsigned int req_info_size;
> unsigned int reply_data_size;
> bool allow_nodev_do;
> + bool allow_pernetdev_dump;
nit: allow_pernetdev_dump should also be added to the Kernel doc for
struct ethnl_request_ops
Flagged by ./scripts/kernel-doc -none
There also appear to be similar minor issues with subsequent
patches in this series.
> u8 set_ntf_cmd;
>
> int (*parse_request)(struct ethnl_req_info *req_info,
> --
> 2.48.1
>
^ permalink raw reply [flat|nested] 14+ messages in thread
* Re: [PATCH net-next 1/7] net: ethtool: netlink: Allow per-netdevice DUMP operations
2025-03-07 12:21 ` Simon Horman
@ 2025-03-07 13:18 ` Maxime Chevallier
2025-03-07 15:14 ` Simon Horman
0 siblings, 1 reply; 14+ messages in thread
From: Maxime Chevallier @ 2025-03-07 13:18 UTC (permalink / raw)
To: Simon Horman
Cc: davem, Andrew Lunn, Jakub Kicinski, Eric Dumazet, Paolo Abeni,
Heiner Kallweit, netdev, linux-kernel, thomas.petazzoni,
linux-arm-kernel, Christophe Leroy, Herve Codina,
Florian Fainelli, Russell King, Vladimir Oltean,
Köry Maincent, Oleksij Rempel, Romain Gantois,
Piergiorgio Beruto
Hi Simon,
On Fri, 7 Mar 2025 12:21:19 +0000
Simon Horman <horms@kernel.org> wrote:
> On Wed, Mar 05, 2025 at 03:19:31PM +0100, Maxime Chevallier wrote:
> > We have a number of netlink commands in the ethnl family that may have
> > multiple objects to dump even for a single net_device, including :
> >
> > - PLCA, PSE-PD, phy: one message per PHY device
> > - tsinfo: one message per timestamp source (netdev + phys)
> > - rss: One per RSS context
> >
> > To get this behaviour, these netlink commands need to roll a custom
> > ->dumpit().
> >
> > To prepare making per-netdev DUMP more generic in ethnl, introduce a
> > member in the ethnl ops to indicate if a given command may allow
> > pernetdev DUMPs (also referred to as filtered DUMPs).
> >
> > Signed-off-by: Maxime Chevallier <maxime.chevallier@bootlin.com>
> > ---
> > net/ethtool/netlink.c | 45 ++++++++++++++++++++++++++++---------------
> > net/ethtool/netlink.h | 1 +
> > 2 files changed, 30 insertions(+), 16 deletions(-)
> >
> > diff --git a/net/ethtool/netlink.c b/net/ethtool/netlink.c
> > index 734849a57369..0815b28ba32f 100644
> > --- a/net/ethtool/netlink.c
> > +++ b/net/ethtool/netlink.c
> > @@ -578,21 +578,34 @@ static int ethnl_default_dumpit(struct sk_buff *skb,
> > int ret = 0;
> >
> > rcu_read_lock();
> > - for_each_netdev_dump(net, dev, ctx->pos_ifindex) {
> > - dev_hold(dev);
> > + if (ctx->req_info->dev) {
> > + dev = ctx->req_info->dev;
> > rcu_read_unlock();
> > -
> > - ret = ethnl_default_dump_one(skb, dev, ctx, genl_info_dump(cb));
> > -
> > + /* Filtered DUMP request targeted to a single netdev. We already
> > + * hold a ref to the netdev from ->start()
> > + */
> > + ret = ethnl_default_dump_one_dev(skb, dev, ctx,
> > + genl_info_dump(cb));
>
> Hi Maxime,
>
> ethnl_default_dump_one_dev() is called here but it doesn't exist
> until the following patch is applied, which breaks bisection.
Yeah I messed-up in my rebase and bisection broke :(
I'll send a new version in a few days, as Jakub said, let's give some
time for the netdev_lock series to move forward and go through CI, I'll
need to rebase on it at some point.
>
> > rcu_read_lock();
> > - dev_put(dev);
> > -
> > - if (ret < 0 && ret != -EOPNOTSUPP) {
> > - if (likely(skb->len))
> > - ret = skb->len;
> > - break;
> > + netdev_put(ctx->req_info->dev, &ctx->req_info->dev_tracker);
> > + } else {
> > + for_each_netdev_dump(net, dev, ctx->pos_ifindex) {
> > + dev_hold(dev);
> > + rcu_read_unlock();
> > +
> > + ret = ethnl_default_dump_one(skb, dev, ctx,
> > + genl_info_dump(cb));
> > +
> > + rcu_read_lock();
> > + dev_put(dev);
> > +
> > + if (ret < 0 && ret != -EOPNOTSUPP) {
> > + if (likely(skb->len))
> > + ret = skb->len;
> > + break;
> > + }
> > + ret = 0;
> > }
> > - ret = 0;
> > }
> > rcu_read_unlock();
> >
>
> ...
>
> > diff --git a/net/ethtool/netlink.h b/net/ethtool/netlink.h
> > index ec6ab5443a6f..4db27182741f 100644
> > --- a/net/ethtool/netlink.h
> > +++ b/net/ethtool/netlink.h
> > @@ -388,6 +388,7 @@ struct ethnl_request_ops {
> > unsigned int req_info_size;
> > unsigned int reply_data_size;
> > bool allow_nodev_do;
> > + bool allow_pernetdev_dump;
>
> nit: allow_pernetdev_dump should also be added to the Kernel doc for
> struct ethnl_request_ops
>
> Flagged by ./scripts/kernel-doc -none
>
> There also appear to be similar minor issues with subsequent
> patches in this series.
Ack, I'll make sure the doc is up to date and properly formatted :)
Thanks,
Maxime
> > u8 set_ntf_cmd;
> >
> > int (*parse_request)(struct ethnl_req_info *req_info,
> > --
> > 2.48.1
> >
^ permalink raw reply [flat|nested] 14+ messages in thread
* Re: [PATCH net-next 1/7] net: ethtool: netlink: Allow per-netdevice DUMP operations
2025-03-07 13:18 ` Maxime Chevallier
@ 2025-03-07 15:14 ` Simon Horman
0 siblings, 0 replies; 14+ messages in thread
From: Simon Horman @ 2025-03-07 15:14 UTC (permalink / raw)
To: Maxime Chevallier
Cc: davem, Andrew Lunn, Jakub Kicinski, Eric Dumazet, Paolo Abeni,
Heiner Kallweit, netdev, linux-kernel, thomas.petazzoni,
linux-arm-kernel, Christophe Leroy, Herve Codina,
Florian Fainelli, Russell King, Vladimir Oltean,
Köry Maincent, Oleksij Rempel, Romain Gantois,
Piergiorgio Beruto
On Fri, Mar 07, 2025 at 02:18:19PM +0100, Maxime Chevallier wrote:
> Hi Simon,
>
> On Fri, 7 Mar 2025 12:21:19 +0000
> Simon Horman <horms@kernel.org> wrote:
>
> > On Wed, Mar 05, 2025 at 03:19:31PM +0100, Maxime Chevallier wrote:
> > > We have a number of netlink commands in the ethnl family that may have
> > > multiple objects to dump even for a single net_device, including :
> > >
> > > - PLCA, PSE-PD, phy: one message per PHY device
> > > - tsinfo: one message per timestamp source (netdev + phys)
> > > - rss: One per RSS context
> > >
> > > To get this behaviour, these netlink commands need to roll a custom
> > > ->dumpit().
> > >
> > > To prepare making per-netdev DUMP more generic in ethnl, introduce a
> > > member in the ethnl ops to indicate if a given command may allow
> > > pernetdev DUMPs (also referred to as filtered DUMPs).
> > >
> > > Signed-off-by: Maxime Chevallier <maxime.chevallier@bootlin.com>
> > > ---
> > > net/ethtool/netlink.c | 45 ++++++++++++++++++++++++++++---------------
> > > net/ethtool/netlink.h | 1 +
> > > 2 files changed, 30 insertions(+), 16 deletions(-)
> > >
> > > diff --git a/net/ethtool/netlink.c b/net/ethtool/netlink.c
> > > index 734849a57369..0815b28ba32f 100644
> > > --- a/net/ethtool/netlink.c
> > > +++ b/net/ethtool/netlink.c
> > > @@ -578,21 +578,34 @@ static int ethnl_default_dumpit(struct sk_buff *skb,
> > > int ret = 0;
> > >
> > > rcu_read_lock();
> > > - for_each_netdev_dump(net, dev, ctx->pos_ifindex) {
> > > - dev_hold(dev);
> > > + if (ctx->req_info->dev) {
> > > + dev = ctx->req_info->dev;
> > > rcu_read_unlock();
> > > -
> > > - ret = ethnl_default_dump_one(skb, dev, ctx, genl_info_dump(cb));
> > > -
> > > + /* Filtered DUMP request targeted to a single netdev. We already
> > > + * hold a ref to the netdev from ->start()
> > > + */
> > > + ret = ethnl_default_dump_one_dev(skb, dev, ctx,
> > > + genl_info_dump(cb));
> >
> > Hi Maxime,
> >
> > ethnl_default_dump_one_dev() is called here but it doesn't exist
> > until the following patch is applied, which breaks bisection.
>
> Yeah I messed-up in my rebase and bisection broke :(
>
> I'll send a new version in a few days, as Jakub said, let's give some
> time for the netdev_lock series to move forward and go through CI, I'll
> need to rebase on it at some point.
Thanks. Apologies for duplicating Jakub's comments to some extent.
I only saw them after I'd sent my previous email to you.
...
^ permalink raw reply [flat|nested] 14+ messages in thread
end of thread, other threads:[~2025-03-07 15:18 UTC | newest]
Thread overview: 14+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2025-03-05 14:19 [PATCH net-next 0/7] net: ethtool: Introduce ethnl dump helpers Maxime Chevallier
2025-03-05 14:19 ` [PATCH net-next 1/7] net: ethtool: netlink: Allow per-netdevice DUMP operations Maxime Chevallier
2025-03-07 12:21 ` Simon Horman
2025-03-07 13:18 ` Maxime Chevallier
2025-03-07 15:14 ` Simon Horman
2025-03-05 14:19 ` [PATCH net-next 2/7] net: ethtool: netlink: Rename ethnl_default_dump_one Maxime Chevallier
2025-03-05 14:19 ` [PATCH net-next 3/7] net: ethtool: netlink: Introduce command-specific dump_one_dev Maxime Chevallier
2025-03-05 14:19 ` [PATCH net-next 4/7] net: ethtool: netlink: Introduce per-phy DUMP helpers Maxime Chevallier
2025-03-05 14:19 ` [PATCH net-next 5/7] net: ethtool: phy: Convert the PHY_GET command to generic phy dump Maxime Chevallier
2025-03-05 14:19 ` [PATCH net-next 6/7] net: ethtool: plca: Use per-PHY DUMP operations Maxime Chevallier
2025-03-05 14:19 ` [PATCH net-next 7/7] net: ethtool: pse-pd: " Maxime Chevallier
2025-03-05 17:02 ` [PATCH net-next 0/7] net: ethtool: Introduce ethnl dump helpers Maxime Chevallier
2025-03-06 2:47 ` Jakub Kicinski
2025-03-06 7:42 ` Maxime Chevallier
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).