aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--include/net/flow.h1
-rw-r--r--net/core/flow.c19
-rw-r--r--net/xfrm/xfrm_policy.c7
3 files changed, 26 insertions, 1 deletions
diff --git a/include/net/flow.h b/include/net/flow.h
index bee3741e5a6f..64fd24836650 100644
--- a/include/net/flow.h
+++ b/include/net/flow.h
@@ -219,6 +219,7 @@ struct flow_cache_object *flow_cache_lookup(struct net *net,
219 u8 dir, flow_resolve_t resolver, 219 u8 dir, flow_resolve_t resolver,
220 void *ctx); 220 void *ctx);
221int flow_cache_init(struct net *net); 221int flow_cache_init(struct net *net);
222void flow_cache_fini(struct net *net);
222 223
223void flow_cache_flush(struct net *net); 224void flow_cache_flush(struct net *net);
224void flow_cache_flush_deferred(struct net *net); 225void flow_cache_flush_deferred(struct net *net);
diff --git a/net/core/flow.c b/net/core/flow.c
index 102f8ea2eb6e..31cfb365e0c6 100644
--- a/net/core/flow.c
+++ b/net/core/flow.c
@@ -484,3 +484,22 @@ err:
484 return -ENOMEM; 484 return -ENOMEM;
485} 485}
486EXPORT_SYMBOL(flow_cache_init); 486EXPORT_SYMBOL(flow_cache_init);
487
488void flow_cache_fini(struct net *net)
489{
490 int i;
491 struct flow_cache *fc = &net->xfrm.flow_cache_global;
492
493 del_timer_sync(&fc->rnd_timer);
494 unregister_hotcpu_notifier(&fc->hotcpu_notifier);
495
496 for_each_possible_cpu(i) {
497 struct flow_cache_percpu *fcp = per_cpu_ptr(fc->percpu, i);
498 kfree(fcp->hash_table);
499 fcp->hash_table = NULL;
500 }
501
502 free_percpu(fc->percpu);
503 fc->percpu = NULL;
504}
505EXPORT_SYMBOL(flow_cache_fini);
diff --git a/net/xfrm/xfrm_policy.c b/net/xfrm/xfrm_policy.c
index a75fae4b045a..f02f511b7107 100644
--- a/net/xfrm/xfrm_policy.c
+++ b/net/xfrm/xfrm_policy.c
@@ -2913,15 +2913,19 @@ static int __net_init xfrm_net_init(struct net *net)
2913 rv = xfrm_sysctl_init(net); 2913 rv = xfrm_sysctl_init(net);
2914 if (rv < 0) 2914 if (rv < 0)
2915 goto out_sysctl; 2915 goto out_sysctl;
2916 rv = flow_cache_init(net);
2917 if (rv < 0)
2918 goto out;
2916 2919
2917 /* Initialize the per-net locks here */ 2920 /* Initialize the per-net locks here */
2918 spin_lock_init(&net->xfrm.xfrm_state_lock); 2921 spin_lock_init(&net->xfrm.xfrm_state_lock);
2919 rwlock_init(&net->xfrm.xfrm_policy_lock); 2922 rwlock_init(&net->xfrm.xfrm_policy_lock);
2920 mutex_init(&net->xfrm.xfrm_cfg_mutex); 2923 mutex_init(&net->xfrm.xfrm_cfg_mutex);
2921 2924
2922 flow_cache_init(net);
2923 return 0; 2925 return 0;
2924 2926
2927out:
2928 xfrm_sysctl_fini(net);
2925out_sysctl: 2929out_sysctl:
2926 xfrm_policy_fini(net); 2930 xfrm_policy_fini(net);
2927out_policy: 2931out_policy:
@@ -2934,6 +2938,7 @@ out_statistics:
2934 2938
2935static void __net_exit xfrm_net_exit(struct net *net) 2939static void __net_exit xfrm_net_exit(struct net *net)
2936{ 2940{
2941 flow_cache_fini(net);
2937 xfrm_sysctl_fini(net); 2942 xfrm_sysctl_fini(net);
2938 xfrm_policy_fini(net); 2943 xfrm_policy_fini(net);
2939 xfrm_state_fini(net); 2944 xfrm_state_fini(net);