提交 53049978 编写于 作者: D David S. Miller

pkt_sched: Make qdisc grafting locking more specific.

Lock the root of the qdisc being operated upon.

All explicit references to qdisc_tree_lock() are now gone.
The only remaining uses are via the sch_tree_{lock,unlock}()
and tcf_tree_{lock,unlock}() macros.
Signed-off-by: NDavid S. Miller <davem@davemloft.net>
上级 ead81cc5
...@@ -441,15 +441,29 @@ static struct Qdisc * ...@@ -441,15 +441,29 @@ static struct Qdisc *
dev_graft_qdisc(struct net_device *dev, struct Qdisc *qdisc) dev_graft_qdisc(struct net_device *dev, struct Qdisc *qdisc)
{ {
struct netdev_queue *dev_queue; struct netdev_queue *dev_queue;
spinlock_t *root_lock;
struct Qdisc *oqdisc; struct Qdisc *oqdisc;
int ingress;
if (dev->flags & IFF_UP) if (dev->flags & IFF_UP)
dev_deactivate(dev); dev_deactivate(dev);
qdisc_lock_tree(dev); ingress = 0;
if (qdisc && qdisc->flags&TCQ_F_INGRESS) { if (qdisc && qdisc->flags&TCQ_F_INGRESS)
ingress = 1;
if (ingress) {
dev_queue = &dev->rx_queue; dev_queue = &dev->rx_queue;
oqdisc = dev_queue->qdisc; oqdisc = dev_queue->qdisc;
} else {
dev_queue = netdev_get_tx_queue(dev, 0);
oqdisc = dev_queue->qdisc_sleeping;
}
root_lock = qdisc_root_lock(oqdisc);
spin_lock_bh(root_lock);
if (ingress) {
/* Prune old scheduler */ /* Prune old scheduler */
if (oqdisc && atomic_read(&oqdisc->refcnt) <= 1) { if (oqdisc && atomic_read(&oqdisc->refcnt) <= 1) {
/* delete */ /* delete */
...@@ -460,9 +474,6 @@ dev_graft_qdisc(struct net_device *dev, struct Qdisc *qdisc) ...@@ -460,9 +474,6 @@ dev_graft_qdisc(struct net_device *dev, struct Qdisc *qdisc)
} }
} else { } else {
dev_queue = netdev_get_tx_queue(dev, 0);
oqdisc = dev_queue->qdisc_sleeping;
/* Prune old scheduler */ /* Prune old scheduler */
if (oqdisc && atomic_read(&oqdisc->refcnt) <= 1) if (oqdisc && atomic_read(&oqdisc->refcnt) <= 1)
qdisc_reset(oqdisc); qdisc_reset(oqdisc);
...@@ -474,7 +485,7 @@ dev_graft_qdisc(struct net_device *dev, struct Qdisc *qdisc) ...@@ -474,7 +485,7 @@ dev_graft_qdisc(struct net_device *dev, struct Qdisc *qdisc)
dev_queue->qdisc = &noop_qdisc; dev_queue->qdisc = &noop_qdisc;
} }
qdisc_unlock_tree(dev); spin_unlock_bh(root_lock);
if (dev->flags & IFF_UP) if (dev->flags & IFF_UP)
dev_activate(dev); dev_activate(dev);
...@@ -765,10 +776,12 @@ static int tc_get_qdisc(struct sk_buff *skb, struct nlmsghdr *n, void *arg) ...@@ -765,10 +776,12 @@ static int tc_get_qdisc(struct sk_buff *skb, struct nlmsghdr *n, void *arg)
if ((err = qdisc_graft(dev, p, clid, NULL, &q)) != 0) if ((err = qdisc_graft(dev, p, clid, NULL, &q)) != 0)
return err; return err;
if (q) { if (q) {
spinlock_t *root_lock = qdisc_root_lock(q);
qdisc_notify(skb, n, clid, q, NULL); qdisc_notify(skb, n, clid, q, NULL);
qdisc_lock_tree(dev); spin_unlock_bh(root_lock);
qdisc_destroy(q); qdisc_destroy(q);
qdisc_unlock_tree(dev); spin_unlock_bh(root_lock);
} }
} else { } else {
qdisc_notify(skb, n, clid, NULL, q); qdisc_notify(skb, n, clid, NULL, q);
...@@ -911,20 +924,24 @@ static int tc_modify_qdisc(struct sk_buff *skb, struct nlmsghdr *n, void *arg) ...@@ -911,20 +924,24 @@ static int tc_modify_qdisc(struct sk_buff *skb, struct nlmsghdr *n, void *arg)
graft: graft:
if (1) { if (1) {
struct Qdisc *old_q = NULL; struct Qdisc *old_q = NULL;
spinlock_t *root_lock;
err = qdisc_graft(dev, p, clid, q, &old_q); err = qdisc_graft(dev, p, clid, q, &old_q);
if (err) { if (err) {
if (q) { if (q) {
qdisc_lock_tree(dev); root_lock = qdisc_root_lock(q);
spin_lock_bh(root_lock);
qdisc_destroy(q); qdisc_destroy(q);
qdisc_unlock_tree(dev); spin_unlock_bh(root_lock);
} }
return err; return err;
} }
qdisc_notify(skb, n, clid, old_q, q); qdisc_notify(skb, n, clid, old_q, q);
if (old_q) { if (old_q) {
qdisc_lock_tree(dev); root_lock = qdisc_root_lock(old_q);
spin_lock_bh(root_lock);
qdisc_destroy(old_q); qdisc_destroy(old_q);
qdisc_unlock_tree(dev); spin_unlock_bh(root_lock);
} }
} }
return 0; return 0;
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册