Commit 15f585bd authored by Gao feng's avatar Gao feng Committed by Pablo Neira Ayuso

netfilter: nf_ct_generic: add namespace support

This patch adds namespace support for the generic layer 4 protocol
tracker.
Acked-by: default avatarEric W. Biederman <ebiederm@xmission.com>
Signed-off-by: default avatarGao feng <gaofeng@cn.fujitsu.com>
Signed-off-by: default avatarPablo Neira Ayuso <pablo@netfilter.org>
parent 524a53e5
...@@ -28,8 +28,8 @@ extern unsigned int nf_conntrack_in(struct net *net, ...@@ -28,8 +28,8 @@ extern unsigned int nf_conntrack_in(struct net *net,
extern int nf_conntrack_init(struct net *net); extern int nf_conntrack_init(struct net *net);
extern void nf_conntrack_cleanup(struct net *net); extern void nf_conntrack_cleanup(struct net *net);
extern int nf_conntrack_proto_init(void); extern int nf_conntrack_proto_init(struct net *net);
extern void nf_conntrack_proto_fini(void); extern void nf_conntrack_proto_fini(struct net *net);
extern bool extern bool
nf_ct_get_tuple(const struct sk_buff *skb, nf_ct_get_tuple(const struct sk_buff *skb,
......
...@@ -20,7 +20,13 @@ struct nf_proto_net { ...@@ -20,7 +20,13 @@ struct nf_proto_net {
unsigned int users; unsigned int users;
}; };
struct nf_generic_net {
struct nf_proto_net pn;
unsigned int timeout;
};
struct nf_ip_net { struct nf_ip_net {
struct nf_generic_net generic;
#if defined(CONFIG_SYSCTL) && defined(CONFIG_NF_CONNTRACK_PROC_COMPAT) #if defined(CONFIG_SYSCTL) && defined(CONFIG_NF_CONNTRACK_PROC_COMPAT)
struct ctl_table_header *ctl_table_header; struct ctl_table_header *ctl_table_header;
struct ctl_table *ctl_table; struct ctl_table *ctl_table;
......
...@@ -1333,7 +1333,6 @@ static void nf_conntrack_cleanup_init_net(void) ...@@ -1333,7 +1333,6 @@ static void nf_conntrack_cleanup_init_net(void)
while (untrack_refs() > 0) while (untrack_refs() > 0)
schedule(); schedule();
nf_conntrack_proto_fini();
#ifdef CONFIG_NF_CONNTRACK_ZONES #ifdef CONFIG_NF_CONNTRACK_ZONES
nf_ct_extend_unregister(&nf_ct_zone_extend); nf_ct_extend_unregister(&nf_ct_zone_extend);
#endif #endif
...@@ -1372,7 +1371,7 @@ void nf_conntrack_cleanup(struct net *net) ...@@ -1372,7 +1371,7 @@ void nf_conntrack_cleanup(struct net *net)
netfilter framework. Roll on, two-stage module netfilter framework. Roll on, two-stage module
delete... */ delete... */
synchronize_net(); synchronize_net();
nf_conntrack_proto_fini(net);
nf_conntrack_cleanup_net(net); nf_conntrack_cleanup_net(net);
if (net_eq(net, &init_net)) { if (net_eq(net, &init_net)) {
...@@ -1496,11 +1495,6 @@ static int nf_conntrack_init_init_net(void) ...@@ -1496,11 +1495,6 @@ static int nf_conntrack_init_init_net(void)
printk(KERN_INFO "nf_conntrack version %s (%u buckets, %d max)\n", printk(KERN_INFO "nf_conntrack version %s (%u buckets, %d max)\n",
NF_CONNTRACK_VERSION, nf_conntrack_htable_size, NF_CONNTRACK_VERSION, nf_conntrack_htable_size,
nf_conntrack_max); nf_conntrack_max);
ret = nf_conntrack_proto_init();
if (ret < 0)
goto err_proto;
#ifdef CONFIG_NF_CONNTRACK_ZONES #ifdef CONFIG_NF_CONNTRACK_ZONES
ret = nf_ct_extend_register(&nf_ct_zone_extend); ret = nf_ct_extend_register(&nf_ct_zone_extend);
if (ret < 0) if (ret < 0)
...@@ -1518,9 +1512,7 @@ static int nf_conntrack_init_init_net(void) ...@@ -1518,9 +1512,7 @@ static int nf_conntrack_init_init_net(void)
#ifdef CONFIG_NF_CONNTRACK_ZONES #ifdef CONFIG_NF_CONNTRACK_ZONES
err_extend: err_extend:
nf_conntrack_proto_fini();
#endif #endif
err_proto:
return ret; return ret;
} }
...@@ -1583,9 +1575,7 @@ static int nf_conntrack_init_net(struct net *net) ...@@ -1583,9 +1575,7 @@ static int nf_conntrack_init_net(struct net *net)
ret = nf_conntrack_helper_init(net); ret = nf_conntrack_helper_init(net);
if (ret < 0) if (ret < 0)
goto err_helper; goto err_helper;
return 0; return 0;
err_helper: err_helper:
nf_conntrack_timeout_fini(net); nf_conntrack_timeout_fini(net);
err_timeout: err_timeout:
...@@ -1622,6 +1612,9 @@ int nf_conntrack_init(struct net *net) ...@@ -1622,6 +1612,9 @@ int nf_conntrack_init(struct net *net)
if (ret < 0) if (ret < 0)
goto out_init_net; goto out_init_net;
} }
ret = nf_conntrack_proto_init(net);
if (ret < 0)
goto out_proto;
ret = nf_conntrack_init_net(net); ret = nf_conntrack_init_net(net);
if (ret < 0) if (ret < 0)
goto out_net; goto out_net;
...@@ -1637,6 +1630,8 @@ int nf_conntrack_init(struct net *net) ...@@ -1637,6 +1630,8 @@ int nf_conntrack_init(struct net *net)
return 0; return 0;
out_net: out_net:
nf_conntrack_proto_fini(net);
out_proto:
if (net_eq(net, &init_net)) if (net_eq(net, &init_net))
nf_conntrack_cleanup_init_net(); nf_conntrack_cleanup_init_net();
out_init_net: out_init_net:
......
...@@ -302,10 +302,16 @@ EXPORT_SYMBOL_GPL(nf_conntrack_l3proto_unregister); ...@@ -302,10 +302,16 @@ EXPORT_SYMBOL_GPL(nf_conntrack_l3proto_unregister);
static struct nf_proto_net *nf_ct_l4proto_net(struct net *net, static struct nf_proto_net *nf_ct_l4proto_net(struct net *net,
struct nf_conntrack_l4proto *l4proto) struct nf_conntrack_l4proto *l4proto)
{ {
if (l4proto->net_id) switch (l4proto->l4proto) {
return net_generic(net, *l4proto->net_id); case 255: /* l4proto_generic */
else return (struct nf_proto_net *)&net->ct.nf_ct_proto.generic;
return NULL; default:
if (l4proto->net_id)
return net_generic(net, *l4proto->net_id);
else
return NULL;
}
return NULL;
} }
static static
...@@ -487,28 +493,34 @@ void nf_conntrack_l4proto_unregister(struct net *net, ...@@ -487,28 +493,34 @@ void nf_conntrack_l4proto_unregister(struct net *net,
} }
EXPORT_SYMBOL_GPL(nf_conntrack_l4proto_unregister); EXPORT_SYMBOL_GPL(nf_conntrack_l4proto_unregister);
int nf_conntrack_proto_init(void) int nf_conntrack_proto_init(struct net *net)
{ {
unsigned int i; unsigned int i;
int err; int err;
err = nf_conntrack_l4proto_generic.init_net(net);
err = nf_ct_l4proto_register_sysctl(&init_net, &nf_conntrack_l4proto_generic); if (err < 0)
return err;
err = nf_ct_l4proto_register_sysctl(net,
&nf_conntrack_l4proto_generic);
if (err < 0) if (err < 0)
return err; return err;
for (i = 0; i < AF_MAX; i++) if (net == &init_net) {
rcu_assign_pointer(nf_ct_l3protos[i], for (i = 0; i < AF_MAX; i++)
&nf_conntrack_l3proto_generic); rcu_assign_pointer(nf_ct_l3protos[i],
&nf_conntrack_l3proto_generic);
}
return 0; return 0;
} }
void nf_conntrack_proto_fini(void) void nf_conntrack_proto_fini(struct net *net)
{ {
unsigned int i; unsigned int i;
nf_ct_l4proto_unregister_sysctl(net,
nf_ct_l4proto_unregister_sysctl(&init_net, &nf_conntrack_l4proto_generic); &nf_conntrack_l4proto_generic);
if (net == &init_net) {
/* free l3proto protocol tables */ /* free l3proto protocol tables */
for (i = 0; i < PF_MAX; i++) for (i = 0; i < PF_MAX; i++)
kfree(nf_ct_protos[i]); kfree(nf_ct_protos[i]);
}
} }
...@@ -14,6 +14,11 @@ ...@@ -14,6 +14,11 @@
static unsigned int nf_ct_generic_timeout __read_mostly = 600*HZ; static unsigned int nf_ct_generic_timeout __read_mostly = 600*HZ;
static inline struct nf_generic_net *generic_pernet(struct net *net)
{
return &net->ct.nf_ct_proto.generic;
}
static bool generic_pkt_to_tuple(const struct sk_buff *skb, static bool generic_pkt_to_tuple(const struct sk_buff *skb,
unsigned int dataoff, unsigned int dataoff,
struct nf_conntrack_tuple *tuple) struct nf_conntrack_tuple *tuple)
...@@ -42,7 +47,7 @@ static int generic_print_tuple(struct seq_file *s, ...@@ -42,7 +47,7 @@ static int generic_print_tuple(struct seq_file *s,
static unsigned int *generic_get_timeouts(struct net *net) static unsigned int *generic_get_timeouts(struct net *net)
{ {
return &nf_ct_generic_timeout; return &(generic_pernet(net)->timeout);
} }
/* Returns verdict for packet, or -1 for invalid. */ /* Returns verdict for packet, or -1 for invalid. */
...@@ -110,7 +115,6 @@ static struct ctl_table_header *generic_sysctl_header; ...@@ -110,7 +115,6 @@ static struct ctl_table_header *generic_sysctl_header;
static struct ctl_table generic_sysctl_table[] = { static struct ctl_table generic_sysctl_table[] = {
{ {
.procname = "nf_conntrack_generic_timeout", .procname = "nf_conntrack_generic_timeout",
.data = &nf_ct_generic_timeout,
.maxlen = sizeof(unsigned int), .maxlen = sizeof(unsigned int),
.mode = 0644, .mode = 0644,
.proc_handler = proc_dointvec_jiffies, .proc_handler = proc_dointvec_jiffies,
...@@ -121,7 +125,6 @@ static struct ctl_table generic_sysctl_table[] = { ...@@ -121,7 +125,6 @@ static struct ctl_table generic_sysctl_table[] = {
static struct ctl_table generic_compat_sysctl_table[] = { static struct ctl_table generic_compat_sysctl_table[] = {
{ {
.procname = "ip_conntrack_generic_timeout", .procname = "ip_conntrack_generic_timeout",
.data = &nf_ct_generic_timeout,
.maxlen = sizeof(unsigned int), .maxlen = sizeof(unsigned int),
.mode = 0644, .mode = 0644,
.proc_handler = proc_dointvec_jiffies, .proc_handler = proc_dointvec_jiffies,
...@@ -131,6 +134,34 @@ static struct ctl_table generic_compat_sysctl_table[] = { ...@@ -131,6 +134,34 @@ static struct ctl_table generic_compat_sysctl_table[] = {
#endif /* CONFIG_NF_CONNTRACK_PROC_COMPAT */ #endif /* CONFIG_NF_CONNTRACK_PROC_COMPAT */
#endif /* CONFIG_SYSCTL */ #endif /* CONFIG_SYSCTL */
static int generic_init_net(struct net *net)
{
struct nf_generic_net *gn = generic_pernet(net);
struct nf_proto_net *pn = (struct nf_proto_net *)gn;
gn->timeout = nf_ct_generic_timeout;
#ifdef CONFIG_SYSCTL
pn->ctl_table = kmemdup(generic_sysctl_table,
sizeof(generic_sysctl_table),
GFP_KERNEL);
if (!pn->ctl_table)
return -ENOMEM;
pn->ctl_table[0].data = &gn->timeout;
#ifdef CONFIG_NF_CONNTRACK_PROC_COMPAT
pn->ctl_compat_table = kmemdup(generic_compat_sysctl_table,
sizeof(generic_compat_sysctl_table),
GFP_KERNEL);
if (!pn->ctl_compat_table) {
kfree(pn->ctl_table);
pn->ctl_table = NULL;
return -ENOMEM;
}
pn->ctl_compat_table[0].data = &gn->timeout;
#endif
#endif
return 0;
}
struct nf_conntrack_l4proto nf_conntrack_l4proto_generic __read_mostly = struct nf_conntrack_l4proto nf_conntrack_l4proto_generic __read_mostly =
{ {
.l3proto = PF_UNSPEC, .l3proto = PF_UNSPEC,
...@@ -158,4 +189,5 @@ struct nf_conntrack_l4proto nf_conntrack_l4proto_generic __read_mostly = ...@@ -158,4 +189,5 @@ struct nf_conntrack_l4proto nf_conntrack_l4proto_generic __read_mostly =
.ctl_compat_table = generic_compat_sysctl_table, .ctl_compat_table = generic_compat_sysctl_table,
#endif #endif
#endif #endif
.init_net = generic_init_net,
}; };
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