diff options
author | Cong Wang <xiyou.wangcong@gmail.com> | 2017-11-06 16:47:20 -0500 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2017-11-08 20:03:09 -0500 |
commit | 0b2a59894b7657fab46b50f176bd772aa495044f (patch) | |
tree | 3861183d9cc6d207b5dda58ab48203bcacdd1144 /net/sched/cls_basic.c | |
parent | e4b95c41df36befcfd117210900cd790bc2cd048 (diff) |
cls_basic: use tcf_exts_get_net() before call_rcu()
Hold netns refcnt before call_rcu() and release it after
the tcf_exts_destroy() is done.
Note, on ->destroy() path we have to respect the return value
of tcf_exts_get_net(), on other paths it should always return
true, so we don't need to care.
Cc: Lucas Bates <lucasb@mojatatu.com>
Cc: Jamal Hadi Salim <jhs@mojatatu.com>
Cc: Jiri Pirko <jiri@resnulli.us>
Signed-off-by: Cong Wang <xiyou.wangcong@gmail.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/sched/cls_basic.c')
-rw-r--r-- | net/sched/cls_basic.c | 20 |
1 files changed, 15 insertions, 5 deletions
diff --git a/net/sched/cls_basic.c b/net/sched/cls_basic.c index f177649a2419..e43c56d5b96a 100644 --- a/net/sched/cls_basic.c +++ b/net/sched/cls_basic.c | |||
@@ -85,16 +85,21 @@ static int basic_init(struct tcf_proto *tp) | |||
85 | return 0; | 85 | return 0; |
86 | } | 86 | } |
87 | 87 | ||
88 | static void __basic_delete_filter(struct basic_filter *f) | ||
89 | { | ||
90 | tcf_exts_destroy(&f->exts); | ||
91 | tcf_em_tree_destroy(&f->ematches); | ||
92 | tcf_exts_put_net(&f->exts); | ||
93 | kfree(f); | ||
94 | } | ||
95 | |||
88 | static void basic_delete_filter_work(struct work_struct *work) | 96 | static void basic_delete_filter_work(struct work_struct *work) |
89 | { | 97 | { |
90 | struct basic_filter *f = container_of(work, struct basic_filter, work); | 98 | struct basic_filter *f = container_of(work, struct basic_filter, work); |
91 | 99 | ||
92 | rtnl_lock(); | 100 | rtnl_lock(); |
93 | tcf_exts_destroy(&f->exts); | 101 | __basic_delete_filter(f); |
94 | tcf_em_tree_destroy(&f->ematches); | ||
95 | rtnl_unlock(); | 102 | rtnl_unlock(); |
96 | |||
97 | kfree(f); | ||
98 | } | 103 | } |
99 | 104 | ||
100 | static void basic_delete_filter(struct rcu_head *head) | 105 | static void basic_delete_filter(struct rcu_head *head) |
@@ -113,7 +118,10 @@ static void basic_destroy(struct tcf_proto *tp) | |||
113 | list_for_each_entry_safe(f, n, &head->flist, link) { | 118 | list_for_each_entry_safe(f, n, &head->flist, link) { |
114 | list_del_rcu(&f->link); | 119 | list_del_rcu(&f->link); |
115 | tcf_unbind_filter(tp, &f->res); | 120 | tcf_unbind_filter(tp, &f->res); |
116 | call_rcu(&f->rcu, basic_delete_filter); | 121 | if (tcf_exts_get_net(&f->exts)) |
122 | call_rcu(&f->rcu, basic_delete_filter); | ||
123 | else | ||
124 | __basic_delete_filter(f); | ||
117 | } | 125 | } |
118 | kfree_rcu(head, rcu); | 126 | kfree_rcu(head, rcu); |
119 | } | 127 | } |
@@ -125,6 +133,7 @@ static int basic_delete(struct tcf_proto *tp, void *arg, bool *last) | |||
125 | 133 | ||
126 | list_del_rcu(&f->link); | 134 | list_del_rcu(&f->link); |
127 | tcf_unbind_filter(tp, &f->res); | 135 | tcf_unbind_filter(tp, &f->res); |
136 | tcf_exts_get_net(&f->exts); | ||
128 | call_rcu(&f->rcu, basic_delete_filter); | 137 | call_rcu(&f->rcu, basic_delete_filter); |
129 | *last = list_empty(&head->flist); | 138 | *last = list_empty(&head->flist); |
130 | return 0; | 139 | return 0; |
@@ -219,6 +228,7 @@ static int basic_change(struct net *net, struct sk_buff *in_skb, | |||
219 | if (fold) { | 228 | if (fold) { |
220 | list_replace_rcu(&fold->link, &fnew->link); | 229 | list_replace_rcu(&fold->link, &fnew->link); |
221 | tcf_unbind_filter(tp, &fold->res); | 230 | tcf_unbind_filter(tp, &fold->res); |
231 | tcf_exts_get_net(&fold->exts); | ||
222 | call_rcu(&fold->rcu, basic_delete_filter); | 232 | call_rcu(&fold->rcu, basic_delete_filter); |
223 | } else { | 233 | } else { |
224 | list_add_rcu(&fnew->link, &head->flist); | 234 | list_add_rcu(&fnew->link, &head->flist); |