mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git
synced 2025-01-01 10:42:11 +00:00
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:
parent
4eb0308d78
commit
be325f08c4
@ -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,
|
||||||
|
@ -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,
|
||||||
|
@ -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,
|
||||||
|
@ -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,
|
||||||
|
@ -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)
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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,
|
||||||
|
Loading…
Reference in New Issue
Block a user