netfilter: xtables: prepare for on-demand hook register

This change prepares for upcoming on-demand xtables hook registration.

We change the protoypes of the register/unregister functions.
A followup patch will then add nf_hook_register/unregister calls
to the iptables one.

Once a hook is registered packets will be picked up, so all assignments
of the form

net->ipv4.iptable_$table = new_table

have to be moved to ip(6)t_register_table, else we can see NULL
net->ipv4.iptable_$table later.

This patch doesn't change functionality; without this the actual change
simply gets too big.

Signed-off-by: Florian Westphal <fw@strlen.de>
Signed-off-by: Pablo Neira Ayuso <pablo@netfilter.org>
This commit is contained in:
Florian Westphal 2016-02-25 10:08:35 +01:00 committed by Pablo Neira Ayuso
parent 5f547391f5
commit a67dd266ad
17 changed files with 107 additions and 88 deletions

View File

@ -48,10 +48,11 @@ struct arpt_error {
} }
extern void *arpt_alloc_initial_table(const struct xt_table *); extern void *arpt_alloc_initial_table(const struct xt_table *);
extern struct xt_table *arpt_register_table(struct net *net, int arpt_register_table(struct net *net, const struct xt_table *table,
const struct xt_table *table, const struct arpt_replace *repl,
const struct arpt_replace *repl); const struct nf_hook_ops *ops, struct xt_table **res);
extern void arpt_unregister_table(struct xt_table *table); void arpt_unregister_table(struct net *net, struct xt_table *table,
const struct nf_hook_ops *ops);
extern unsigned int arpt_do_table(struct sk_buff *skb, extern unsigned int arpt_do_table(struct sk_buff *skb,
const struct nf_hook_state *state, const struct nf_hook_state *state,
struct xt_table *table); struct xt_table *table);

View File

@ -24,10 +24,11 @@
extern void ipt_init(void) __init; extern void ipt_init(void) __init;
extern struct xt_table *ipt_register_table(struct net *net, int ipt_register_table(struct net *net, const struct xt_table *table,
const struct xt_table *table, const struct ipt_replace *repl,
const struct ipt_replace *repl); const struct nf_hook_ops *ops, struct xt_table **res);
extern void ipt_unregister_table(struct net *net, struct xt_table *table); void ipt_unregister_table(struct net *net, struct xt_table *table,
const struct nf_hook_ops *ops);
/* Standard entry. */ /* Standard entry. */
struct ipt_standard { struct ipt_standard {

View File

@ -25,10 +25,11 @@
extern void ip6t_init(void) __init; extern void ip6t_init(void) __init;
extern void *ip6t_alloc_initial_table(const struct xt_table *); extern void *ip6t_alloc_initial_table(const struct xt_table *);
extern struct xt_table *ip6t_register_table(struct net *net, int ip6t_register_table(struct net *net, const struct xt_table *table,
const struct xt_table *table, const struct ip6t_replace *repl,
const struct ip6t_replace *repl); const struct nf_hook_ops *ops, struct xt_table **res);
extern void ip6t_unregister_table(struct net *net, struct xt_table *table); void ip6t_unregister_table(struct net *net, struct xt_table *table,
const struct nf_hook_ops *ops);
extern unsigned int ip6t_do_table(struct sk_buff *skb, extern unsigned int ip6t_do_table(struct sk_buff *skb,
const struct nf_hook_state *state, const struct nf_hook_state *state,
struct xt_table *table); struct xt_table *table);

View File

@ -1780,9 +1780,11 @@ static int do_arpt_get_ctl(struct sock *sk, int cmd, void __user *user, int *len
return ret; return ret;
} }
struct xt_table *arpt_register_table(struct net *net, int arpt_register_table(struct net *net,
const struct xt_table *table, const struct xt_table *table,
const struct arpt_replace *repl) const struct arpt_replace *repl,
const struct nf_hook_ops *ops,
struct xt_table **res)
{ {
int ret; int ret;
struct xt_table_info *newinfo; struct xt_table_info *newinfo;
@ -1791,10 +1793,8 @@ struct xt_table *arpt_register_table(struct net *net,
struct xt_table *new_table; struct xt_table *new_table;
newinfo = xt_alloc_table_info(repl->size); newinfo = xt_alloc_table_info(repl->size);
if (!newinfo) { if (!newinfo)
ret = -ENOMEM; return -ENOMEM;
goto out;
}
loc_cpu_entry = newinfo->entries; loc_cpu_entry = newinfo->entries;
memcpy(loc_cpu_entry, repl->entries, repl->size); memcpy(loc_cpu_entry, repl->entries, repl->size);
@ -1809,15 +1809,18 @@ struct xt_table *arpt_register_table(struct net *net,
ret = PTR_ERR(new_table); ret = PTR_ERR(new_table);
goto out_free; goto out_free;
} }
return new_table;
WRITE_ONCE(*res, new_table);
return ret;
out_free: out_free:
xt_free_table_info(newinfo); xt_free_table_info(newinfo);
out: return ret;
return ERR_PTR(ret);
} }
void arpt_unregister_table(struct xt_table *table) void arpt_unregister_table(struct net *net, struct xt_table *table,
const struct nf_hook_ops *ops)
{ {
struct xt_table_info *private; struct xt_table_info *private;
void *loc_cpu_entry; void *loc_cpu_entry;

View File

@ -38,19 +38,20 @@ static struct nf_hook_ops *arpfilter_ops __read_mostly;
static int __net_init arptable_filter_net_init(struct net *net) static int __net_init arptable_filter_net_init(struct net *net)
{ {
struct arpt_replace *repl; struct arpt_replace *repl;
int err;
repl = arpt_alloc_initial_table(&packet_filter); repl = arpt_alloc_initial_table(&packet_filter);
if (repl == NULL) if (repl == NULL)
return -ENOMEM; return -ENOMEM;
net->ipv4.arptable_filter = err = arpt_register_table(net, &packet_filter, repl, arpfilter_ops,
arpt_register_table(net, &packet_filter, repl); &net->ipv4.arptable_filter);
kfree(repl); kfree(repl);
return PTR_ERR_OR_ZERO(net->ipv4.arptable_filter); return err;
} }
static void __net_exit arptable_filter_net_exit(struct net *net) static void __net_exit arptable_filter_net_exit(struct net *net)
{ {
arpt_unregister_table(net->ipv4.arptable_filter); arpt_unregister_table(net, net->ipv4.arptable_filter, arpfilter_ops);
} }
static struct pernet_operations arptable_filter_net_ops = { static struct pernet_operations arptable_filter_net_ops = {

View File

@ -2062,9 +2062,9 @@ do_ipt_get_ctl(struct sock *sk, int cmd, void __user *user, int *len)
return ret; return ret;
} }
struct xt_table *ipt_register_table(struct net *net, int ipt_register_table(struct net *net, const struct xt_table *table,
const struct xt_table *table, const struct ipt_replace *repl,
const struct ipt_replace *repl) const struct nf_hook_ops *ops, struct xt_table **res)
{ {
int ret; int ret;
struct xt_table_info *newinfo; struct xt_table_info *newinfo;
@ -2073,10 +2073,8 @@ struct xt_table *ipt_register_table(struct net *net,
struct xt_table *new_table; struct xt_table *new_table;
newinfo = xt_alloc_table_info(repl->size); newinfo = xt_alloc_table_info(repl->size);
if (!newinfo) { if (!newinfo)
ret = -ENOMEM; return -ENOMEM;
goto out;
}
loc_cpu_entry = newinfo->entries; loc_cpu_entry = newinfo->entries;
memcpy(loc_cpu_entry, repl->entries, repl->size); memcpy(loc_cpu_entry, repl->entries, repl->size);
@ -2091,15 +2089,16 @@ struct xt_table *ipt_register_table(struct net *net,
goto out_free; goto out_free;
} }
return new_table; WRITE_ONCE(*res, new_table);
return ret;
out_free: out_free:
xt_free_table_info(newinfo); xt_free_table_info(newinfo);
out: return ret;
return ERR_PTR(ret);
} }
void ipt_unregister_table(struct net *net, struct xt_table *table) void ipt_unregister_table(struct net *net, struct xt_table *table,
const struct nf_hook_ops *ops)
{ {
struct xt_table_info *private; struct xt_table_info *private;
void *loc_cpu_entry; void *loc_cpu_entry;

View File

@ -54,6 +54,7 @@ module_param(forward, bool, 0000);
static int __net_init iptable_filter_net_init(struct net *net) static int __net_init iptable_filter_net_init(struct net *net)
{ {
struct ipt_replace *repl; struct ipt_replace *repl;
int err;
repl = ipt_alloc_initial_table(&packet_filter); repl = ipt_alloc_initial_table(&packet_filter);
if (repl == NULL) if (repl == NULL)
@ -62,15 +63,15 @@ static int __net_init iptable_filter_net_init(struct net *net)
((struct ipt_standard *)repl->entries)[1].target.verdict = ((struct ipt_standard *)repl->entries)[1].target.verdict =
forward ? -NF_ACCEPT - 1 : -NF_DROP - 1; forward ? -NF_ACCEPT - 1 : -NF_DROP - 1;
net->ipv4.iptable_filter = err = ipt_register_table(net, &packet_filter, repl, filter_ops,
ipt_register_table(net, &packet_filter, repl); &net->ipv4.iptable_filter);
kfree(repl); kfree(repl);
return PTR_ERR_OR_ZERO(net->ipv4.iptable_filter); return err;
} }
static void __net_exit iptable_filter_net_exit(struct net *net) static void __net_exit iptable_filter_net_exit(struct net *net)
{ {
ipt_unregister_table(net, net->ipv4.iptable_filter); ipt_unregister_table(net, net->ipv4.iptable_filter, filter_ops);
} }
static struct pernet_operations iptable_filter_net_ops = { static struct pernet_operations iptable_filter_net_ops = {

View File

@ -96,19 +96,20 @@ static struct nf_hook_ops *mangle_ops __read_mostly;
static int __net_init iptable_mangle_net_init(struct net *net) static int __net_init iptable_mangle_net_init(struct net *net)
{ {
struct ipt_replace *repl; struct ipt_replace *repl;
int ret;
repl = ipt_alloc_initial_table(&packet_mangler); repl = ipt_alloc_initial_table(&packet_mangler);
if (repl == NULL) if (repl == NULL)
return -ENOMEM; return -ENOMEM;
net->ipv4.iptable_mangle = ret = ipt_register_table(net, &packet_mangler, repl, mangle_ops,
ipt_register_table(net, &packet_mangler, repl); &net->ipv4.iptable_mangle);
kfree(repl); kfree(repl);
return PTR_ERR_OR_ZERO(net->ipv4.iptable_mangle); return ret;
} }
static void __net_exit iptable_mangle_net_exit(struct net *net) static void __net_exit iptable_mangle_net_exit(struct net *net)
{ {
ipt_unregister_table(net, net->ipv4.iptable_mangle); ipt_unregister_table(net, net->ipv4.iptable_mangle, mangle_ops);
} }
static struct pernet_operations iptable_mangle_net_ops = { static struct pernet_operations iptable_mangle_net_ops = {

View File

@ -98,18 +98,20 @@ static struct nf_hook_ops nf_nat_ipv4_ops[] __read_mostly = {
static int __net_init iptable_nat_net_init(struct net *net) static int __net_init iptable_nat_net_init(struct net *net)
{ {
struct ipt_replace *repl; struct ipt_replace *repl;
int ret;
repl = ipt_alloc_initial_table(&nf_nat_ipv4_table); repl = ipt_alloc_initial_table(&nf_nat_ipv4_table);
if (repl == NULL) if (repl == NULL)
return -ENOMEM; return -ENOMEM;
net->ipv4.nat_table = ipt_register_table(net, &nf_nat_ipv4_table, repl); ret = ipt_register_table(net, &nf_nat_ipv4_table, repl,
nf_nat_ipv4_ops, &net->ipv4.nat_table);
kfree(repl); kfree(repl);
return PTR_ERR_OR_ZERO(net->ipv4.nat_table); return ret;
} }
static void __net_exit iptable_nat_net_exit(struct net *net) static void __net_exit iptable_nat_net_exit(struct net *net)
{ {
ipt_unregister_table(net, net->ipv4.nat_table); ipt_unregister_table(net, net->ipv4.nat_table, nf_nat_ipv4_ops);
} }
static struct pernet_operations iptable_nat_net_ops = { static struct pernet_operations iptable_nat_net_ops = {

View File

@ -37,19 +37,20 @@ static struct nf_hook_ops *rawtable_ops __read_mostly;
static int __net_init iptable_raw_net_init(struct net *net) static int __net_init iptable_raw_net_init(struct net *net)
{ {
struct ipt_replace *repl; struct ipt_replace *repl;
int ret;
repl = ipt_alloc_initial_table(&packet_raw); repl = ipt_alloc_initial_table(&packet_raw);
if (repl == NULL) if (repl == NULL)
return -ENOMEM; return -ENOMEM;
net->ipv4.iptable_raw = ret = ipt_register_table(net, &packet_raw, repl, rawtable_ops,
ipt_register_table(net, &packet_raw, repl); &net->ipv4.iptable_raw);
kfree(repl); kfree(repl);
return PTR_ERR_OR_ZERO(net->ipv4.iptable_raw); return ret;
} }
static void __net_exit iptable_raw_net_exit(struct net *net) static void __net_exit iptable_raw_net_exit(struct net *net)
{ {
ipt_unregister_table(net, net->ipv4.iptable_raw); ipt_unregister_table(net, net->ipv4.iptable_raw, rawtable_ops);
} }
static struct pernet_operations iptable_raw_net_ops = { static struct pernet_operations iptable_raw_net_ops = {

View File

@ -54,19 +54,20 @@ static struct nf_hook_ops *sectbl_ops __read_mostly;
static int __net_init iptable_security_net_init(struct net *net) static int __net_init iptable_security_net_init(struct net *net)
{ {
struct ipt_replace *repl; struct ipt_replace *repl;
int ret;
repl = ipt_alloc_initial_table(&security_table); repl = ipt_alloc_initial_table(&security_table);
if (repl == NULL) if (repl == NULL)
return -ENOMEM; return -ENOMEM;
net->ipv4.iptable_security = ret = ipt_register_table(net, &security_table, repl, sectbl_ops,
ipt_register_table(net, &security_table, repl); &net->ipv4.iptable_security);
kfree(repl); kfree(repl);
return PTR_ERR_OR_ZERO(net->ipv4.iptable_security); return ret;
} }
static void __net_exit iptable_security_net_exit(struct net *net) static void __net_exit iptable_security_net_exit(struct net *net)
{ {
ipt_unregister_table(net, net->ipv4.iptable_security); ipt_unregister_table(net, net->ipv4.iptable_security, sectbl_ops);
} }
static struct pernet_operations iptable_security_net_ops = { static struct pernet_operations iptable_security_net_ops = {

View File

@ -2071,9 +2071,10 @@ do_ip6t_get_ctl(struct sock *sk, int cmd, void __user *user, int *len)
return ret; return ret;
} }
struct xt_table *ip6t_register_table(struct net *net, int ip6t_register_table(struct net *net, const struct xt_table *table,
const struct xt_table *table, const struct ip6t_replace *repl,
const struct ip6t_replace *repl) const struct nf_hook_ops *ops,
struct xt_table **res)
{ {
int ret; int ret;
struct xt_table_info *newinfo; struct xt_table_info *newinfo;
@ -2082,10 +2083,8 @@ struct xt_table *ip6t_register_table(struct net *net,
struct xt_table *new_table; struct xt_table *new_table;
newinfo = xt_alloc_table_info(repl->size); newinfo = xt_alloc_table_info(repl->size);
if (!newinfo) { if (!newinfo)
ret = -ENOMEM; return -ENOMEM;
goto out;
}
loc_cpu_entry = newinfo->entries; loc_cpu_entry = newinfo->entries;
memcpy(loc_cpu_entry, repl->entries, repl->size); memcpy(loc_cpu_entry, repl->entries, repl->size);
@ -2099,15 +2098,17 @@ struct xt_table *ip6t_register_table(struct net *net,
ret = PTR_ERR(new_table); ret = PTR_ERR(new_table);
goto out_free; goto out_free;
} }
return new_table;
WRITE_ONCE(*res, new_table);
return ret;
out_free: out_free:
xt_free_table_info(newinfo); xt_free_table_info(newinfo);
out: return ret;
return ERR_PTR(ret);
} }
void ip6t_unregister_table(struct net *net, struct xt_table *table) void ip6t_unregister_table(struct net *net, struct xt_table *table,
const struct nf_hook_ops *ops)
{ {
struct xt_table_info *private; struct xt_table_info *private;
void *loc_cpu_entry; void *loc_cpu_entry;

View File

@ -47,6 +47,7 @@ module_param(forward, bool, 0000);
static int __net_init ip6table_filter_net_init(struct net *net) static int __net_init ip6table_filter_net_init(struct net *net)
{ {
struct ip6t_replace *repl; struct ip6t_replace *repl;
int err;
repl = ip6t_alloc_initial_table(&packet_filter); repl = ip6t_alloc_initial_table(&packet_filter);
if (repl == NULL) if (repl == NULL)
@ -55,15 +56,15 @@ static int __net_init ip6table_filter_net_init(struct net *net)
((struct ip6t_standard *)repl->entries)[1].target.verdict = ((struct ip6t_standard *)repl->entries)[1].target.verdict =
forward ? -NF_ACCEPT - 1 : -NF_DROP - 1; forward ? -NF_ACCEPT - 1 : -NF_DROP - 1;
net->ipv6.ip6table_filter = err = ip6t_register_table(net, &packet_filter, repl, filter_ops,
ip6t_register_table(net, &packet_filter, repl); &net->ipv6.ip6table_filter);
kfree(repl); kfree(repl);
return PTR_ERR_OR_ZERO(net->ipv6.ip6table_filter); return err;
} }
static void __net_exit ip6table_filter_net_exit(struct net *net) static void __net_exit ip6table_filter_net_exit(struct net *net)
{ {
ip6t_unregister_table(net, net->ipv6.ip6table_filter); ip6t_unregister_table(net, net->ipv6.ip6table_filter, filter_ops);
} }
static struct pernet_operations ip6table_filter_net_ops = { static struct pernet_operations ip6table_filter_net_ops = {

View File

@ -91,19 +91,20 @@ static struct nf_hook_ops *mangle_ops __read_mostly;
static int __net_init ip6table_mangle_net_init(struct net *net) static int __net_init ip6table_mangle_net_init(struct net *net)
{ {
struct ip6t_replace *repl; struct ip6t_replace *repl;
int ret;
repl = ip6t_alloc_initial_table(&packet_mangler); repl = ip6t_alloc_initial_table(&packet_mangler);
if (repl == NULL) if (repl == NULL)
return -ENOMEM; return -ENOMEM;
net->ipv6.ip6table_mangle = ret = ip6t_register_table(net, &packet_mangler, repl, mangle_ops,
ip6t_register_table(net, &packet_mangler, repl); &net->ipv6.ip6table_mangle);
kfree(repl); kfree(repl);
return PTR_ERR_OR_ZERO(net->ipv6.ip6table_mangle); return ret;
} }
static void __net_exit ip6table_mangle_net_exit(struct net *net) static void __net_exit ip6table_mangle_net_exit(struct net *net)
{ {
ip6t_unregister_table(net, net->ipv6.ip6table_mangle); ip6t_unregister_table(net, net->ipv6.ip6table_mangle, mangle_ops);
} }
static struct pernet_operations ip6table_mangle_net_ops = { static struct pernet_operations ip6table_mangle_net_ops = {

View File

@ -100,18 +100,20 @@ static struct nf_hook_ops nf_nat_ipv6_ops[] __read_mostly = {
static int __net_init ip6table_nat_net_init(struct net *net) static int __net_init ip6table_nat_net_init(struct net *net)
{ {
struct ip6t_replace *repl; struct ip6t_replace *repl;
int ret;
repl = ip6t_alloc_initial_table(&nf_nat_ipv6_table); repl = ip6t_alloc_initial_table(&nf_nat_ipv6_table);
if (repl == NULL) if (repl == NULL)
return -ENOMEM; return -ENOMEM;
net->ipv6.ip6table_nat = ip6t_register_table(net, &nf_nat_ipv6_table, repl); ret = ip6t_register_table(net, &nf_nat_ipv6_table, repl,
nf_nat_ipv6_ops, &net->ipv6.ip6table_nat);
kfree(repl); kfree(repl);
return PTR_ERR_OR_ZERO(net->ipv6.ip6table_nat); return ret;
} }
static void __net_exit ip6table_nat_net_exit(struct net *net) static void __net_exit ip6table_nat_net_exit(struct net *net)
{ {
ip6t_unregister_table(net, net->ipv6.ip6table_nat); ip6t_unregister_table(net, net->ipv6.ip6table_nat, nf_nat_ipv6_ops);
} }
static struct pernet_operations ip6table_nat_net_ops = { static struct pernet_operations ip6table_nat_net_ops = {

View File

@ -30,19 +30,20 @@ static struct nf_hook_ops *rawtable_ops __read_mostly;
static int __net_init ip6table_raw_net_init(struct net *net) static int __net_init ip6table_raw_net_init(struct net *net)
{ {
struct ip6t_replace *repl; struct ip6t_replace *repl;
int ret;
repl = ip6t_alloc_initial_table(&packet_raw); repl = ip6t_alloc_initial_table(&packet_raw);
if (repl == NULL) if (repl == NULL)
return -ENOMEM; return -ENOMEM;
net->ipv6.ip6table_raw = ret = ip6t_register_table(net, &packet_raw, repl, rawtable_ops,
ip6t_register_table(net, &packet_raw, repl); &net->ipv6.ip6table_raw);
kfree(repl); kfree(repl);
return PTR_ERR_OR_ZERO(net->ipv6.ip6table_raw); return ret;
} }
static void __net_exit ip6table_raw_net_exit(struct net *net) static void __net_exit ip6table_raw_net_exit(struct net *net)
{ {
ip6t_unregister_table(net, net->ipv6.ip6table_raw); ip6t_unregister_table(net, net->ipv6.ip6table_raw, rawtable_ops);
} }
static struct pernet_operations ip6table_raw_net_ops = { static struct pernet_operations ip6table_raw_net_ops = {

View File

@ -47,19 +47,20 @@ static struct nf_hook_ops *sectbl_ops __read_mostly;
static int __net_init ip6table_security_net_init(struct net *net) static int __net_init ip6table_security_net_init(struct net *net)
{ {
struct ip6t_replace *repl; struct ip6t_replace *repl;
int ret;
repl = ip6t_alloc_initial_table(&security_table); repl = ip6t_alloc_initial_table(&security_table);
if (repl == NULL) if (repl == NULL)
return -ENOMEM; return -ENOMEM;
net->ipv6.ip6table_security = ret = ip6t_register_table(net, &security_table, repl, sectbl_ops,
ip6t_register_table(net, &security_table, repl); &net->ipv6.ip6table_security);
kfree(repl); kfree(repl);
return PTR_ERR_OR_ZERO(net->ipv6.ip6table_security); return ret;
} }
static void __net_exit ip6table_security_net_exit(struct net *net) static void __net_exit ip6table_security_net_exit(struct net *net)
{ {
ip6t_unregister_table(net, net->ipv6.ip6table_security); ip6t_unregister_table(net, net->ipv6.ip6table_security, sectbl_ops);
} }
static struct pernet_operations ip6table_security_net_ops = { static struct pernet_operations ip6table_security_net_ops = {