提交 787ce6d0 编写于 作者: V Vlad Buslov 提交者: David S. Miller

net: sched: use reference counting for tcf blocks on rules update

In order to remove dependency on rtnl lock on rules update path, always
take reference to block while using it on rules update path. Change
tcf_block_get() error handling to properly release block with reference
counting, instead of just destroying it, in order to accommodate potential
concurrent users.
Signed-off-by: NVlad Buslov <vladbu@mellanox.com>
Acked-by: NJiri Pirko <jiri@mellanox.com>
Signed-off-by: NDavid S. Miller <davem@davemloft.net>
上级 0607e439
...@@ -622,7 +622,7 @@ static struct tcf_block *tcf_block_find(struct net *net, struct Qdisc **q, ...@@ -622,7 +622,7 @@ static struct tcf_block *tcf_block_find(struct net *net, struct Qdisc **q,
int err = 0; int err = 0;
if (ifindex == TCM_IFINDEX_MAGIC_BLOCK) { if (ifindex == TCM_IFINDEX_MAGIC_BLOCK) {
block = tcf_block_lookup(net, block_index); block = tcf_block_refcnt_get(net, block_index);
if (!block) { if (!block) {
NL_SET_ERR_MSG(extack, "Block of given index was not found"); NL_SET_ERR_MSG(extack, "Block of given index was not found");
return ERR_PTR(-EINVAL); return ERR_PTR(-EINVAL);
...@@ -702,6 +702,14 @@ static struct tcf_block *tcf_block_find(struct net *net, struct Qdisc **q, ...@@ -702,6 +702,14 @@ static struct tcf_block *tcf_block_find(struct net *net, struct Qdisc **q,
err = -EOPNOTSUPP; err = -EOPNOTSUPP;
goto errout_qdisc; goto errout_qdisc;
} }
/* Always take reference to block in order to support execution
* of rules update path of cls API without rtnl lock. Caller
* must release block when it is finished using it. 'if' block
* of this conditional obtain reference to block by calling
* tcf_block_refcnt_get().
*/
refcount_inc(&block->refcnt);
} }
return block; return block;
...@@ -716,6 +724,9 @@ static struct tcf_block *tcf_block_find(struct net *net, struct Qdisc **q, ...@@ -716,6 +724,9 @@ static struct tcf_block *tcf_block_find(struct net *net, struct Qdisc **q,
static void tcf_block_release(struct Qdisc *q, struct tcf_block *block) static void tcf_block_release(struct Qdisc *q, struct tcf_block *block)
{ {
if (!IS_ERR_OR_NULL(block))
tcf_block_refcnt_put(block);
if (q) if (q)
qdisc_put(q); qdisc_put(q);
} }
...@@ -785,21 +796,16 @@ int tcf_block_get_ext(struct tcf_block **p_block, struct Qdisc *q, ...@@ -785,21 +796,16 @@ int tcf_block_get_ext(struct tcf_block **p_block, struct Qdisc *q,
{ {
struct net *net = qdisc_net(q); struct net *net = qdisc_net(q);
struct tcf_block *block = NULL; struct tcf_block *block = NULL;
bool created = false;
int err; int err;
if (ei->block_index) { if (ei->block_index)
/* block_index not 0 means the shared block is requested */ /* block_index not 0 means the shared block is requested */
block = tcf_block_lookup(net, ei->block_index); block = tcf_block_refcnt_get(net, ei->block_index);
if (block)
refcount_inc(&block->refcnt);
}
if (!block) { if (!block) {
block = tcf_block_create(net, q, ei->block_index, extack); block = tcf_block_create(net, q, ei->block_index, extack);
if (IS_ERR(block)) if (IS_ERR(block))
return PTR_ERR(block); return PTR_ERR(block);
created = true;
if (tcf_block_shared(block)) { if (tcf_block_shared(block)) {
err = tcf_block_insert(block, net, extack); err = tcf_block_insert(block, net, extack);
if (err) if (err)
...@@ -829,14 +835,8 @@ int tcf_block_get_ext(struct tcf_block **p_block, struct Qdisc *q, ...@@ -829,14 +835,8 @@ int tcf_block_get_ext(struct tcf_block **p_block, struct Qdisc *q,
err_chain0_head_change_cb_add: err_chain0_head_change_cb_add:
tcf_block_owner_del(block, q, ei->binder_type); tcf_block_owner_del(block, q, ei->binder_type);
err_block_owner_add: err_block_owner_add:
if (created) {
if (tcf_block_shared(block))
tcf_block_remove(block, net);
err_block_insert: err_block_insert:
kfree_rcu(block, rcu); tcf_block_refcnt_put(block);
} else {
refcount_dec(&block->refcnt);
}
return err; return err;
} }
EXPORT_SYMBOL(tcf_block_get_ext); EXPORT_SYMBOL(tcf_block_get_ext);
...@@ -1730,7 +1730,7 @@ static int tc_dump_tfilter(struct sk_buff *skb, struct netlink_callback *cb) ...@@ -1730,7 +1730,7 @@ static int tc_dump_tfilter(struct sk_buff *skb, struct netlink_callback *cb)
return err; return err;
if (tcm->tcm_ifindex == TCM_IFINDEX_MAGIC_BLOCK) { if (tcm->tcm_ifindex == TCM_IFINDEX_MAGIC_BLOCK) {
block = tcf_block_lookup(net, tcm->tcm_block_index); block = tcf_block_refcnt_get(net, tcm->tcm_block_index);
if (!block) if (!block)
goto out; goto out;
/* If we work with block index, q is NULL and parent value /* If we work with block index, q is NULL and parent value
...@@ -1789,6 +1789,8 @@ static int tc_dump_tfilter(struct sk_buff *skb, struct netlink_callback *cb) ...@@ -1789,6 +1789,8 @@ static int tc_dump_tfilter(struct sk_buff *skb, struct netlink_callback *cb)
} }
} }
if (tcm->tcm_ifindex == TCM_IFINDEX_MAGIC_BLOCK)
tcf_block_refcnt_put(block);
cb->args[0] = index; cb->args[0] = index;
out: out:
...@@ -2055,7 +2057,7 @@ static int tc_dump_chain(struct sk_buff *skb, struct netlink_callback *cb) ...@@ -2055,7 +2057,7 @@ static int tc_dump_chain(struct sk_buff *skb, struct netlink_callback *cb)
return err; return err;
if (tcm->tcm_ifindex == TCM_IFINDEX_MAGIC_BLOCK) { if (tcm->tcm_ifindex == TCM_IFINDEX_MAGIC_BLOCK) {
block = tcf_block_lookup(net, tcm->tcm_block_index); block = tcf_block_refcnt_get(net, tcm->tcm_block_index);
if (!block) if (!block)
goto out; goto out;
/* If we work with block index, q is NULL and parent value /* If we work with block index, q is NULL and parent value
...@@ -2122,6 +2124,8 @@ static int tc_dump_chain(struct sk_buff *skb, struct netlink_callback *cb) ...@@ -2122,6 +2124,8 @@ static int tc_dump_chain(struct sk_buff *skb, struct netlink_callback *cb)
index++; index++;
} }
if (tcm->tcm_ifindex == TCM_IFINDEX_MAGIC_BLOCK)
tcf_block_refcnt_put(block);
cb->args[0] = index; cb->args[0] = index;
out: out:
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册