Commit 53681407 authored by Jiri Pirko's avatar Jiri Pirko Committed by David S. Miller

net: sched: fix notifications for action-held chains

Chains that only have action references serve as placeholders.
Until a non-action reference is created, user should not be aware
of the chain. Also he should not receive any notifications about it.
So send notifications for the new chain only in case the chain gets
the first non-action reference. Symmetrically to that, when
the last non-action reference is dropped, send the notification about
deleted chain.
Reported-by: default avatarCong Wang <xiyou.wangcong@gmail.com>
Signed-off-by: default avatarJiri Pirko <jiri@mellanox.com>
Acked-by: default avatarCong Wang <xiyou.wangcong@gmail.com>

v1->v2:
- made __tcf_chain_{get,put}() static as suggested by Cong
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 3d32f4c5
...@@ -262,16 +262,6 @@ static void tcf_chain_hold(struct tcf_chain *chain) ...@@ -262,16 +262,6 @@ static void tcf_chain_hold(struct tcf_chain *chain)
++chain->refcnt; ++chain->refcnt;
} }
static void tcf_chain_hold_by_act(struct tcf_chain *chain)
{
++chain->action_refcnt;
}
static void tcf_chain_release_by_act(struct tcf_chain *chain)
{
--chain->action_refcnt;
}
static bool tcf_chain_held_by_acts_only(struct tcf_chain *chain) static bool tcf_chain_held_by_acts_only(struct tcf_chain *chain)
{ {
/* In case all the references are action references, this /* In case all the references are action references, this
...@@ -295,52 +285,77 @@ static struct tcf_chain *tcf_chain_lookup(struct tcf_block *block, ...@@ -295,52 +285,77 @@ static struct tcf_chain *tcf_chain_lookup(struct tcf_block *block,
static int tc_chain_notify(struct tcf_chain *chain, struct sk_buff *oskb, static int tc_chain_notify(struct tcf_chain *chain, struct sk_buff *oskb,
u32 seq, u16 flags, int event, bool unicast); u32 seq, u16 flags, int event, bool unicast);
struct tcf_chain *tcf_chain_get(struct tcf_block *block, u32 chain_index, static struct tcf_chain *__tcf_chain_get(struct tcf_block *block,
bool create) u32 chain_index, bool create,
bool by_act)
{ {
struct tcf_chain *chain = tcf_chain_lookup(block, chain_index); struct tcf_chain *chain = tcf_chain_lookup(block, chain_index);
if (chain) { if (chain) {
tcf_chain_hold(chain); tcf_chain_hold(chain);
return chain; } else {
if (!create)
return NULL;
chain = tcf_chain_create(block, chain_index);
if (!chain)
return NULL;
} }
if (!create) if (by_act)
return NULL; ++chain->action_refcnt;
chain = tcf_chain_create(block, chain_index);
if (!chain) /* Send notification only in case we got the first
return NULL; * non-action reference. Until then, the chain acts only as
tc_chain_notify(chain, NULL, 0, NLM_F_CREATE | NLM_F_EXCL, * a placeholder for actions pointing to it and user ought
RTM_NEWCHAIN, false); * not know about them.
*/
if (chain->refcnt - chain->action_refcnt == 1 && !by_act)
tc_chain_notify(chain, NULL, 0, NLM_F_CREATE | NLM_F_EXCL,
RTM_NEWCHAIN, false);
return chain; return chain;
} }
struct tcf_chain *tcf_chain_get(struct tcf_block *block, u32 chain_index,
bool create)
{
return __tcf_chain_get(block, chain_index, create, false);
}
EXPORT_SYMBOL(tcf_chain_get); EXPORT_SYMBOL(tcf_chain_get);
struct tcf_chain *tcf_chain_get_by_act(struct tcf_block *block, u32 chain_index) struct tcf_chain *tcf_chain_get_by_act(struct tcf_block *block, u32 chain_index)
{ {
struct tcf_chain *chain = tcf_chain_get(block, chain_index, true); return __tcf_chain_get(block, chain_index, true, true);
tcf_chain_hold_by_act(chain);
return chain;
} }
EXPORT_SYMBOL(tcf_chain_get_by_act); EXPORT_SYMBOL(tcf_chain_get_by_act);
static void tc_chain_tmplt_del(struct tcf_chain *chain); static void tc_chain_tmplt_del(struct tcf_chain *chain);
void tcf_chain_put(struct tcf_chain *chain) static void __tcf_chain_put(struct tcf_chain *chain, bool by_act)
{ {
if (--chain->refcnt == 0) { if (by_act)
chain->action_refcnt--;
chain->refcnt--;
/* The last dropped non-action reference will trigger notification. */
if (chain->refcnt - chain->action_refcnt == 0 && !by_act)
tc_chain_notify(chain, NULL, 0, 0, RTM_DELCHAIN, false); tc_chain_notify(chain, NULL, 0, 0, RTM_DELCHAIN, false);
if (chain->refcnt == 0) {
tc_chain_tmplt_del(chain); tc_chain_tmplt_del(chain);
tcf_chain_destroy(chain); tcf_chain_destroy(chain);
} }
} }
void tcf_chain_put(struct tcf_chain *chain)
{
__tcf_chain_put(chain, false);
}
EXPORT_SYMBOL(tcf_chain_put); EXPORT_SYMBOL(tcf_chain_put);
void tcf_chain_put_by_act(struct tcf_chain *chain) void tcf_chain_put_by_act(struct tcf_chain *chain)
{ {
tcf_chain_release_by_act(chain); __tcf_chain_put(chain, true);
tcf_chain_put(chain);
} }
EXPORT_SYMBOL(tcf_chain_put_by_act); EXPORT_SYMBOL(tcf_chain_put_by_act);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment