On Tue, Jan 7, 2025 at 4:57 PM Kuniyuki Iwashima <kuniyu@xxxxxxxxxx> wrote: [...] > > We can fix this by linking the dev to the socket's netns and > clean them up in __net_exit hook as done in bareudp and geneve. > > ---8<--- > diff --git a/drivers/net/gtp.c b/drivers/net/gtp.c > index 89a996ad8cd0..77638a815873 100644 > --- a/drivers/net/gtp.c > +++ b/drivers/net/gtp.c > @@ -70,6 +70,7 @@ struct pdp_ctx { > /* One instance of the GTP device. */ > struct gtp_dev { > struct list_head list; > + struct list_head sock_list; > > struct sock *sk0; > struct sock *sk1u; > @@ -102,6 +103,7 @@ static unsigned int gtp_net_id __read_mostly; > > struct gtp_net { > struct list_head gtp_dev_list; > + struct list_head gtp_sock_list; After a closer look at the GTP driver, I'm confused about the gtp_dev_list here. GTP device is linked to this list at creation time, but netns can be changed afterwards. The list is used in gtp_net_exit_batch_rtnl(), but to my understanding net devices can already be deleted in default_device_exit_batch() by default. And I wonder if the use in gtp_genl_dump_pdp() can be replaced by something like for_each_netdev_rcu(). > }; > > static u32 gtp_h_initval; > @@ -1526,6 +1528,10 @@ static int gtp_newlink(struct net *src_net, struct net_device *dev, > > gn = net_generic(dev_net(dev), gtp_net_id); > list_add_rcu(>p->list, &gn->gtp_dev_list); > + > + gn = net_generic(src_net, gtp_net_id); > + list_add(>p->sock_list, &gn->gtp_sock_list); > + > dev->priv_destructor = gtp_destructor; > > netdev_dbg(dev, "registered new GTP interface\n"); > @@ -1552,6 +1558,7 @@ static void gtp_dellink(struct net_device *dev, struct list_head *head) > pdp_context_delete(pctx); > > list_del_rcu(>p->list); > + list_del(>p->sock_list); > unregister_netdevice_queue(dev, head); > } > > @@ -2465,6 +2472,8 @@ static int __net_init gtp_net_init(struct net *net) > struct gtp_net *gn = net_generic(net, gtp_net_id); > > INIT_LIST_HEAD(&gn->gtp_dev_list); > + INIT_LIST_HEAD(&gn->gtp_sock_list); > + > return 0; > } > > @@ -2475,9 +2484,12 @@ static void __net_exit gtp_net_exit_batch_rtnl(struct list_head *net_list, > > list_for_each_entry(net, net_list, exit_list) { > struct gtp_net *gn = net_generic(net, gtp_net_id); > - struct gtp_dev *gtp; > + struct gtp_dev *gtp, *next; > + > + list_for_each_entry_safe(gtp, next, &gn->gtp_dev_list, list) > + gtp_dellink(gtp->dev, dev_to_kill); > > - list_for_each_entry(gtp, &gn->gtp_dev_list, list) > + list_for_each_entry_safe(gtp, next, &gn->gtp_sock_list, sock_list) > gtp_dellink(gtp->dev, dev_to_kill); > } > } > ---8<---