Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

net/sched: fix netdevice reference leaks in attach_default_qdiscs()

In attach_default_qdiscs(), if a dev has multiple queues and queue 0 fails
to attach qdisc because there is no memory in attach_one_default_qdisc().
Then dev->qdisc will be noop_qdisc by default. But the other queues may be
able to successfully attach to default qdisc.

In this case, the fallback to noqueue process will be triggered. If the
original attached qdisc is not released and a new one is directly
attached, this will cause netdevice reference leaks.

The following is the bug log:

veth0: default qdisc (fq_codel) fail, fallback to noqueue
unregister_netdevice: waiting for veth0 to become free. Usage count = 32
leaked reference.
qdisc_alloc+0x12e/0x210
qdisc_create_dflt+0x62/0x140
attach_one_default_qdisc.constprop.41+0x44/0x70
dev_activate+0x128/0x290
__dev_open+0x12a/0x190
__dev_change_flags+0x1a2/0x1f0
dev_change_flags+0x23/0x60
do_setlink+0x332/0x1150
__rtnl_newlink+0x52f/0x8e0
rtnl_newlink+0x43/0x70
rtnetlink_rcv_msg+0x140/0x3b0
netlink_rcv_skb+0x50/0x100
netlink_unicast+0x1bb/0x290
netlink_sendmsg+0x37c/0x4e0
sock_sendmsg+0x5f/0x70
____sys_sendmsg+0x208/0x280

Fix this bug by clearing any non-noop qdiscs that may have been assigned
before trying to re-attach.

Fixes: bf6dba76d278 ("net: sched: fallback to qdisc noqueue if default qdisc setup fail")
Signed-off-by: Wang Hai <wanghai38@huawei.com>
Link: https://lore.kernel.org/r/20220826090055.24424-1-wanghai38@huawei.com
Signed-off-by: Paolo Abeni <pabeni@redhat.com>

authored by

Wang Hai and committed by
Paolo Abeni
f612466e b05972f0

+16 -15
+16 -15
net/sched/sch_generic.c
··· 1122 1122 } 1123 1123 EXPORT_SYMBOL(dev_graft_qdisc); 1124 1124 1125 + static void shutdown_scheduler_queue(struct net_device *dev, 1126 + struct netdev_queue *dev_queue, 1127 + void *_qdisc_default) 1128 + { 1129 + struct Qdisc *qdisc = dev_queue->qdisc_sleeping; 1130 + struct Qdisc *qdisc_default = _qdisc_default; 1131 + 1132 + if (qdisc) { 1133 + rcu_assign_pointer(dev_queue->qdisc, qdisc_default); 1134 + dev_queue->qdisc_sleeping = qdisc_default; 1135 + 1136 + qdisc_put(qdisc); 1137 + } 1138 + } 1139 + 1125 1140 static void attach_one_default_qdisc(struct net_device *dev, 1126 1141 struct netdev_queue *dev_queue, 1127 1142 void *_unused) ··· 1184 1169 if (qdisc == &noop_qdisc) { 1185 1170 netdev_warn(dev, "default qdisc (%s) fail, fallback to %s\n", 1186 1171 default_qdisc_ops->id, noqueue_qdisc_ops.id); 1172 + netdev_for_each_tx_queue(dev, shutdown_scheduler_queue, &noop_qdisc); 1187 1173 dev->priv_flags |= IFF_NO_QUEUE; 1188 1174 netdev_for_each_tx_queue(dev, attach_one_default_qdisc, NULL); 1189 1175 qdisc = txq->qdisc_sleeping; ··· 1461 1445 dev_init_scheduler_queue(dev, dev_ingress_queue(dev), &noop_qdisc); 1462 1446 1463 1447 timer_setup(&dev->watchdog_timer, dev_watchdog, 0); 1464 - } 1465 - 1466 - static void shutdown_scheduler_queue(struct net_device *dev, 1467 - struct netdev_queue *dev_queue, 1468 - void *_qdisc_default) 1469 - { 1470 - struct Qdisc *qdisc = dev_queue->qdisc_sleeping; 1471 - struct Qdisc *qdisc_default = _qdisc_default; 1472 - 1473 - if (qdisc) { 1474 - rcu_assign_pointer(dev_queue->qdisc, qdisc_default); 1475 - dev_queue->qdisc_sleeping = qdisc_default; 1476 - 1477 - qdisc_put(qdisc); 1478 - } 1479 1448 } 1480 1449 1481 1450 void dev_shutdown(struct net_device *dev)