aboutsummaryrefslogtreecommitdiffstats
path: root/net/sched/cls_basic.c
diff options
context:
space:
mode:
authorCong Wang <xiyou.wangcong@gmail.com>2017-11-06 16:47:20 -0500
committerDavid S. Miller <davem@davemloft.net>2017-11-08 20:03:09 -0500
commit0b2a59894b7657fab46b50f176bd772aa495044f (patch)
tree3861183d9cc6d207b5dda58ab48203bcacdd1144 /net/sched/cls_basic.c
parente4b95c41df36befcfd117210900cd790bc2cd048 (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.c20
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
88static 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
88static void basic_delete_filter_work(struct work_struct *work) 96static 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
100static void basic_delete_filter(struct rcu_head *head) 105static 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);