diff options
Diffstat (limited to 'net')
-rw-r--r-- | net/netfilter/nf_conntrack_core.c | 39 |
1 files changed, 23 insertions, 16 deletions
diff --git a/net/netfilter/nf_conntrack_core.c b/net/netfilter/nf_conntrack_core.c index 37e2b88313f2..9de4bd4c0dd7 100644 --- a/net/netfilter/nf_conntrack_core.c +++ b/net/netfilter/nf_conntrack_core.c | |||
@@ -63,8 +63,6 @@ EXPORT_SYMBOL_GPL(nf_conntrack_max); | |||
63 | struct nf_conn nf_conntrack_untracked __read_mostly; | 63 | struct nf_conn nf_conntrack_untracked __read_mostly; |
64 | EXPORT_SYMBOL_GPL(nf_conntrack_untracked); | 64 | EXPORT_SYMBOL_GPL(nf_conntrack_untracked); |
65 | 65 | ||
66 | static struct kmem_cache *nf_conntrack_cachep __read_mostly; | ||
67 | |||
68 | static int nf_conntrack_hash_rnd_initted; | 66 | static int nf_conntrack_hash_rnd_initted; |
69 | static unsigned int nf_conntrack_hash_rnd; | 67 | static unsigned int nf_conntrack_hash_rnd; |
70 | 68 | ||
@@ -572,7 +570,7 @@ struct nf_conn *nf_conntrack_alloc(struct net *net, | |||
572 | * Do not use kmem_cache_zalloc(), as this cache uses | 570 | * Do not use kmem_cache_zalloc(), as this cache uses |
573 | * SLAB_DESTROY_BY_RCU. | 571 | * SLAB_DESTROY_BY_RCU. |
574 | */ | 572 | */ |
575 | ct = kmem_cache_alloc(nf_conntrack_cachep, gfp); | 573 | ct = kmem_cache_alloc(net->ct.nf_conntrack_cachep, gfp); |
576 | if (ct == NULL) { | 574 | if (ct == NULL) { |
577 | pr_debug("nf_conntrack_alloc: Can't alloc conntrack.\n"); | 575 | pr_debug("nf_conntrack_alloc: Can't alloc conntrack.\n"); |
578 | atomic_dec(&net->ct.count); | 576 | atomic_dec(&net->ct.count); |
@@ -611,7 +609,7 @@ void nf_conntrack_free(struct nf_conn *ct) | |||
611 | nf_ct_ext_destroy(ct); | 609 | nf_ct_ext_destroy(ct); |
612 | atomic_dec(&net->ct.count); | 610 | atomic_dec(&net->ct.count); |
613 | nf_ct_ext_free(ct); | 611 | nf_ct_ext_free(ct); |
614 | kmem_cache_free(nf_conntrack_cachep, ct); | 612 | kmem_cache_free(net->ct.nf_conntrack_cachep, ct); |
615 | } | 613 | } |
616 | EXPORT_SYMBOL_GPL(nf_conntrack_free); | 614 | EXPORT_SYMBOL_GPL(nf_conntrack_free); |
617 | 615 | ||
@@ -1119,7 +1117,6 @@ static void nf_conntrack_cleanup_init_net(void) | |||
1119 | 1117 | ||
1120 | nf_conntrack_helper_fini(); | 1118 | nf_conntrack_helper_fini(); |
1121 | nf_conntrack_proto_fini(); | 1119 | nf_conntrack_proto_fini(); |
1122 | kmem_cache_destroy(nf_conntrack_cachep); | ||
1123 | } | 1120 | } |
1124 | 1121 | ||
1125 | static void nf_conntrack_cleanup_net(struct net *net) | 1122 | static void nf_conntrack_cleanup_net(struct net *net) |
@@ -1137,6 +1134,8 @@ static void nf_conntrack_cleanup_net(struct net *net) | |||
1137 | nf_conntrack_ecache_fini(net); | 1134 | nf_conntrack_ecache_fini(net); |
1138 | nf_conntrack_acct_fini(net); | 1135 | nf_conntrack_acct_fini(net); |
1139 | nf_conntrack_expect_fini(net); | 1136 | nf_conntrack_expect_fini(net); |
1137 | kmem_cache_destroy(net->ct.nf_conntrack_cachep); | ||
1138 | kfree(net->ct.slabname); | ||
1140 | free_percpu(net->ct.stat); | 1139 | free_percpu(net->ct.stat); |
1141 | } | 1140 | } |
1142 | 1141 | ||
@@ -1272,15 +1271,6 @@ static int nf_conntrack_init_init_net(void) | |||
1272 | NF_CONNTRACK_VERSION, nf_conntrack_htable_size, | 1271 | NF_CONNTRACK_VERSION, nf_conntrack_htable_size, |
1273 | nf_conntrack_max); | 1272 | nf_conntrack_max); |
1274 | 1273 | ||
1275 | nf_conntrack_cachep = kmem_cache_create("nf_conntrack", | ||
1276 | sizeof(struct nf_conn), | ||
1277 | 0, SLAB_DESTROY_BY_RCU, NULL); | ||
1278 | if (!nf_conntrack_cachep) { | ||
1279 | printk(KERN_ERR "Unable to create nf_conn slab cache\n"); | ||
1280 | ret = -ENOMEM; | ||
1281 | goto err_cache; | ||
1282 | } | ||
1283 | |||
1284 | ret = nf_conntrack_proto_init(); | 1274 | ret = nf_conntrack_proto_init(); |
1285 | if (ret < 0) | 1275 | if (ret < 0) |
1286 | goto err_proto; | 1276 | goto err_proto; |
@@ -1302,8 +1292,6 @@ static int nf_conntrack_init_init_net(void) | |||
1302 | err_helper: | 1292 | err_helper: |
1303 | nf_conntrack_proto_fini(); | 1293 | nf_conntrack_proto_fini(); |
1304 | err_proto: | 1294 | err_proto: |
1305 | kmem_cache_destroy(nf_conntrack_cachep); | ||
1306 | err_cache: | ||
1307 | return ret; | 1295 | return ret; |
1308 | } | 1296 | } |
1309 | 1297 | ||
@@ -1325,6 +1313,21 @@ static int nf_conntrack_init_net(struct net *net) | |||
1325 | ret = -ENOMEM; | 1313 | ret = -ENOMEM; |
1326 | goto err_stat; | 1314 | goto err_stat; |
1327 | } | 1315 | } |
1316 | |||
1317 | net->ct.slabname = kasprintf(GFP_KERNEL, "nf_conntrack_%p", net); | ||
1318 | if (!net->ct.slabname) { | ||
1319 | ret = -ENOMEM; | ||
1320 | goto err_slabname; | ||
1321 | } | ||
1322 | |||
1323 | net->ct.nf_conntrack_cachep = kmem_cache_create(net->ct.slabname, | ||
1324 | sizeof(struct nf_conn), 0, | ||
1325 | SLAB_DESTROY_BY_RCU, NULL); | ||
1326 | if (!net->ct.nf_conntrack_cachep) { | ||
1327 | printk(KERN_ERR "Unable to create nf_conn slab cache\n"); | ||
1328 | ret = -ENOMEM; | ||
1329 | goto err_cache; | ||
1330 | } | ||
1328 | net->ct.hash = nf_ct_alloc_hashtable(&nf_conntrack_htable_size, | 1331 | net->ct.hash = nf_ct_alloc_hashtable(&nf_conntrack_htable_size, |
1329 | &net->ct.hash_vmalloc, 1); | 1332 | &net->ct.hash_vmalloc, 1); |
1330 | if (!net->ct.hash) { | 1333 | if (!net->ct.hash) { |
@@ -1352,6 +1355,10 @@ err_expect: | |||
1352 | nf_ct_free_hashtable(net->ct.hash, net->ct.hash_vmalloc, | 1355 | nf_ct_free_hashtable(net->ct.hash, net->ct.hash_vmalloc, |
1353 | nf_conntrack_htable_size); | 1356 | nf_conntrack_htable_size); |
1354 | err_hash: | 1357 | err_hash: |
1358 | kmem_cache_destroy(net->ct.nf_conntrack_cachep); | ||
1359 | err_cache: | ||
1360 | kfree(net->ct.slabname); | ||
1361 | err_slabname: | ||
1355 | free_percpu(net->ct.stat); | 1362 | free_percpu(net->ct.stat); |
1356 | err_stat: | 1363 | err_stat: |
1357 | return ret; | 1364 | return ret; |