rtnetlink: add ndo_fdb_dump_context

rtnl_fdb_dump() and various ndo_fdb_dump() helpers share
a hidden layout of cb->ctx.

Before switching rtnl_fdb_dump() to for_each_netdev_dump()
in the following patch, make this more explicit.

Signed-off-by: Eric Dumazet <edumazet@google.com>
Reviewed-by: Ido Schimmel <idosch@nvidia.com>
Reviewed-by: Kuniyuki Iwashima <kuniyu@amazon.com>
Link: https://patch.msgid.link/20241209100747.2269613-2-edumazet@google.com
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
This commit is contained in:
Eric Dumazet 2024-12-09 10:07:45 +00:00 committed by Jakub Kicinski
parent 4eb0308d78
commit be325f08c4
7 changed files with 32 additions and 18 deletions

View File

@ -780,13 +780,14 @@ struct ethsw_dump_ctx {
static int dpaa2_switch_fdb_dump_nl(struct fdb_dump_entry *entry, static int dpaa2_switch_fdb_dump_nl(struct fdb_dump_entry *entry,
struct ethsw_dump_ctx *dump) struct ethsw_dump_ctx *dump)
{ {
struct ndo_fdb_dump_context *ctx = (void *)dump->cb->ctx;
int is_dynamic = entry->type & DPSW_FDB_ENTRY_DINAMIC; int is_dynamic = entry->type & DPSW_FDB_ENTRY_DINAMIC;
u32 portid = NETLINK_CB(dump->cb->skb).portid; u32 portid = NETLINK_CB(dump->cb->skb).portid;
u32 seq = dump->cb->nlh->nlmsg_seq; u32 seq = dump->cb->nlh->nlmsg_seq;
struct nlmsghdr *nlh; struct nlmsghdr *nlh;
struct ndmsg *ndm; struct ndmsg *ndm;
if (dump->idx < dump->cb->args[2]) if (dump->idx < ctx->fdb_idx)
goto skip; goto skip;
nlh = nlmsg_put(dump->skb, portid, seq, RTM_NEWNEIGH, nlh = nlmsg_put(dump->skb, portid, seq, RTM_NEWNEIGH,

View File

@ -758,12 +758,13 @@ static int ocelot_port_fdb_do_dump(const unsigned char *addr, u16 vid,
bool is_static, void *data) bool is_static, void *data)
{ {
struct ocelot_dump_ctx *dump = data; struct ocelot_dump_ctx *dump = data;
struct ndo_fdb_dump_context *ctx = (void *)dump->cb->ctx;
u32 portid = NETLINK_CB(dump->cb->skb).portid; u32 portid = NETLINK_CB(dump->cb->skb).portid;
u32 seq = dump->cb->nlh->nlmsg_seq; u32 seq = dump->cb->nlh->nlmsg_seq;
struct nlmsghdr *nlh; struct nlmsghdr *nlh;
struct ndmsg *ndm; struct ndmsg *ndm;
if (dump->idx < dump->cb->args[2]) if (dump->idx < ctx->fdb_idx)
goto skip; goto skip;
nlh = nlmsg_put(dump->skb, portid, seq, RTM_NEWNEIGH, nlh = nlmsg_put(dump->skb, portid, seq, RTM_NEWNEIGH,

View File

@ -1352,6 +1352,7 @@ static int vxlan_fdb_dump(struct sk_buff *skb, struct netlink_callback *cb,
struct net_device *dev, struct net_device *dev,
struct net_device *filter_dev, int *idx) struct net_device *filter_dev, int *idx)
{ {
struct ndo_fdb_dump_context *ctx = (void *)cb->ctx;
struct vxlan_dev *vxlan = netdev_priv(dev); struct vxlan_dev *vxlan = netdev_priv(dev);
unsigned int h; unsigned int h;
int err = 0; int err = 0;
@ -1364,7 +1365,7 @@ static int vxlan_fdb_dump(struct sk_buff *skb, struct netlink_callback *cb,
struct vxlan_rdst *rd; struct vxlan_rdst *rd;
if (rcu_access_pointer(f->nh)) { if (rcu_access_pointer(f->nh)) {
if (*idx < cb->args[2]) if (*idx < ctx->fdb_idx)
goto skip_nh; goto skip_nh;
err = vxlan_fdb_info(skb, vxlan, f, err = vxlan_fdb_info(skb, vxlan, f,
NETLINK_CB(cb->skb).portid, NETLINK_CB(cb->skb).portid,
@ -1381,7 +1382,7 @@ static int vxlan_fdb_dump(struct sk_buff *skb, struct netlink_callback *cb,
} }
list_for_each_entry_rcu(rd, &f->remotes, list) { list_for_each_entry_rcu(rd, &f->remotes, list) {
if (*idx < cb->args[2]) if (*idx < ctx->fdb_idx)
goto skip; goto skip;
err = vxlan_fdb_info(skb, vxlan, f, err = vxlan_fdb_info(skb, vxlan, f,

View File

@ -178,6 +178,13 @@ void rtnetlink_init(void);
void __rtnl_unlock(void); void __rtnl_unlock(void);
void rtnl_kfree_skbs(struct sk_buff *head, struct sk_buff *tail); void rtnl_kfree_skbs(struct sk_buff *head, struct sk_buff *tail);
/* Shared by rtnl_fdb_dump() and various ndo_fdb_dump() helpers. */
struct ndo_fdb_dump_context {
unsigned long s_h;
unsigned long s_idx;
unsigned long fdb_idx;
};
extern int ndo_dflt_fdb_dump(struct sk_buff *skb, extern int ndo_dflt_fdb_dump(struct sk_buff *skb,
struct netlink_callback *cb, struct netlink_callback *cb,
struct net_device *dev, struct net_device *dev,

View File

@ -955,6 +955,7 @@ int br_fdb_dump(struct sk_buff *skb,
struct net_device *filter_dev, struct net_device *filter_dev,
int *idx) int *idx)
{ {
struct ndo_fdb_dump_context *ctx = (void *)cb->ctx;
struct net_bridge *br = netdev_priv(dev); struct net_bridge *br = netdev_priv(dev);
struct net_bridge_fdb_entry *f; struct net_bridge_fdb_entry *f;
int err = 0; int err = 0;
@ -970,7 +971,7 @@ int br_fdb_dump(struct sk_buff *skb,
rcu_read_lock(); rcu_read_lock();
hlist_for_each_entry_rcu(f, &br->fdb_list, fdb_node) { hlist_for_each_entry_rcu(f, &br->fdb_list, fdb_node) {
if (*idx < cb->args[2]) if (*idx < ctx->fdb_idx)
goto skip; goto skip;
if (filter_dev && (!f->dst || f->dst->dev != filter_dev)) { if (filter_dev && (!f->dst || f->dst->dev != filter_dev)) {
if (filter_dev != dev) if (filter_dev != dev)

View File

@ -4762,15 +4762,16 @@ static int nlmsg_populate_fdb(struct sk_buff *skb,
int *idx, int *idx,
struct netdev_hw_addr_list *list) struct netdev_hw_addr_list *list)
{ {
struct ndo_fdb_dump_context *ctx = (void *)cb->ctx;
struct netdev_hw_addr *ha; struct netdev_hw_addr *ha;
int err;
u32 portid, seq; u32 portid, seq;
int err;
portid = NETLINK_CB(cb->skb).portid; portid = NETLINK_CB(cb->skb).portid;
seq = cb->nlh->nlmsg_seq; seq = cb->nlh->nlmsg_seq;
list_for_each_entry(ha, &list->list, list) { list_for_each_entry(ha, &list->list, list) {
if (*idx < cb->args[2]) if (*idx < ctx->fdb_idx)
goto skip; goto skip;
err = nlmsg_populate_fdb_fill(skb, dev, ha->addr, 0, err = nlmsg_populate_fdb_fill(skb, dev, ha->addr, 0,
@ -4909,10 +4910,9 @@ static int valid_fdb_dump_legacy(const struct nlmsghdr *nlh,
static int rtnl_fdb_dump(struct sk_buff *skb, struct netlink_callback *cb) static int rtnl_fdb_dump(struct sk_buff *skb, struct netlink_callback *cb)
{ {
struct net_device *dev; const struct net_device_ops *ops = NULL, *cops = NULL;
struct net_device *br_dev = NULL; struct ndo_fdb_dump_context *ctx = (void *)cb->ctx;
const struct net_device_ops *ops = NULL; struct net_device *dev, *br_dev = NULL;
const struct net_device_ops *cops = NULL;
struct net *net = sock_net(skb->sk); struct net *net = sock_net(skb->sk);
struct hlist_head *head; struct hlist_head *head;
int brport_idx = 0; int brport_idx = 0;
@ -4922,6 +4922,8 @@ static int rtnl_fdb_dump(struct sk_buff *skb, struct netlink_callback *cb)
int err = 0; int err = 0;
int fidx = 0; int fidx = 0;
NL_ASSERT_CTX_FITS(struct ndo_fdb_dump_context);
if (cb->strict_check) if (cb->strict_check)
err = valid_fdb_dump_strict(cb->nlh, &br_idx, &brport_idx, err = valid_fdb_dump_strict(cb->nlh, &br_idx, &brport_idx,
cb->extack); cb->extack);
@ -4939,8 +4941,8 @@ static int rtnl_fdb_dump(struct sk_buff *skb, struct netlink_callback *cb)
ops = br_dev->netdev_ops; ops = br_dev->netdev_ops;
} }
s_h = cb->args[0]; s_h = ctx->s_h;
s_idx = cb->args[1]; s_idx = ctx->s_idx;
for (h = s_h; h < NETDEV_HASHENTRIES; h++, s_idx = 0) { for (h = s_h; h < NETDEV_HASHENTRIES; h++, s_idx = 0) {
idx = 0; idx = 0;
@ -4992,7 +4994,7 @@ static int rtnl_fdb_dump(struct sk_buff *skb, struct netlink_callback *cb)
cops = NULL; cops = NULL;
/* reset fdb offset to 0 for rest of the interfaces */ /* reset fdb offset to 0 for rest of the interfaces */
cb->args[2] = 0; ctx->fdb_idx = 0;
fidx = 0; fidx = 0;
cont: cont:
idx++; idx++;
@ -5000,9 +5002,9 @@ static int rtnl_fdb_dump(struct sk_buff *skb, struct netlink_callback *cb)
} }
out: out:
cb->args[0] = h; ctx->s_h = h;
cb->args[1] = idx; ctx->s_idx = idx;
cb->args[2] = fidx; ctx->fdb_idx = fidx;
return skb->len; return skb->len;
} }

View File

@ -515,12 +515,13 @@ dsa_user_port_fdb_do_dump(const unsigned char *addr, u16 vid,
bool is_static, void *data) bool is_static, void *data)
{ {
struct dsa_user_dump_ctx *dump = data; struct dsa_user_dump_ctx *dump = data;
struct ndo_fdb_dump_context *ctx = (void *)dump->cb->ctx;
u32 portid = NETLINK_CB(dump->cb->skb).portid; u32 portid = NETLINK_CB(dump->cb->skb).portid;
u32 seq = dump->cb->nlh->nlmsg_seq; u32 seq = dump->cb->nlh->nlmsg_seq;
struct nlmsghdr *nlh; struct nlmsghdr *nlh;
struct ndmsg *ndm; struct ndmsg *ndm;
if (dump->idx < dump->cb->args[2]) if (dump->idx < ctx->fdb_idx)
goto skip; goto skip;
nlh = nlmsg_put(dump->skb, portid, seq, RTM_NEWNEIGH, nlh = nlmsg_put(dump->skb, portid, seq, RTM_NEWNEIGH,