Commit 0bfcb7b7 authored by Florian Westphal's avatar Florian Westphal Committed by Pablo Neira Ayuso

netfilter: xtables: avoid NFPROTO_UNSPEC where needed

syzbot managed to call xt_cluster match via ebtables:

 WARNING: CPU: 0 PID: 11 at net/netfilter/xt_cluster.c:72 xt_cluster_mt+0x196/0x780
 [..]
 ebt_do_table+0x174b/0x2a40

Module registers to NFPROTO_UNSPEC, but it assumes ipv4/ipv6 packet
processing.  As this is only useful to restrict locally terminating
TCP/UDP traffic, register this for ipv4 and ipv6 family only.

Pablo points out that this is a general issue, direct users of the
set/getsockopt interface can call into targets/matches that were only
intended for use with ip(6)tables.

Check all UNSPEC matches and targets for similar issues:

- matches and targets are fine except if they assume skb_network_header()
  is valid -- this is only true when called from inet layer: ip(6) stack
  pulls the ip/ipv6 header into linear data area.
- targets that return XT_CONTINUE or other xtables verdicts must be
  restricted too, they are incompatbile with the ebtables traverser, e.g.
  EBT_CONTINUE is a completely different value than XT_CONTINUE.

Most matches/targets are changed to register for NFPROTO_IPV4/IPV6, as
they are provided for use by ip(6)tables.

The MARK target is also used by arptables, so register for NFPROTO_ARP too.

While at it, bail out if connbytes fails to enable the corresponding
conntrack family.

This change passes the selftests in iptables.git.

Reported-by: syzbot+256c348558aa5cf611a9@syzkaller.appspotmail.com
Closes: https://lore.kernel.org/netfilter-devel/66fec2e2.050a0220.9ec68.0047.GAE@google.com/
Fixes: 0269ea49 ("netfilter: xtables: add cluster match")
Signed-off-by: default avatarFlorian Westphal <fw@strlen.de>
Co-developed-by: default avatarPablo Neira Ayuso <pablo@netfilter.org>
Signed-off-by: default avatarPablo Neira Ayuso <pablo@netfilter.org>
parent 983e35ce
...@@ -63,24 +63,37 @@ static int checksum_tg_check(const struct xt_tgchk_param *par) ...@@ -63,24 +63,37 @@ static int checksum_tg_check(const struct xt_tgchk_param *par)
return 0; return 0;
} }
static struct xt_target checksum_tg_reg __read_mostly = { static struct xt_target checksum_tg_reg[] __read_mostly = {
.name = "CHECKSUM", {
.family = NFPROTO_UNSPEC, .name = "CHECKSUM",
.target = checksum_tg, .family = NFPROTO_IPV4,
.targetsize = sizeof(struct xt_CHECKSUM_info), .target = checksum_tg,
.table = "mangle", .targetsize = sizeof(struct xt_CHECKSUM_info),
.checkentry = checksum_tg_check, .table = "mangle",
.me = THIS_MODULE, .checkentry = checksum_tg_check,
.me = THIS_MODULE,
},
#if IS_ENABLED(CONFIG_IP6_NF_IPTABLES)
{
.name = "CHECKSUM",
.family = NFPROTO_IPV6,
.target = checksum_tg,
.targetsize = sizeof(struct xt_CHECKSUM_info),
.table = "mangle",
.checkentry = checksum_tg_check,
.me = THIS_MODULE,
},
#endif
}; };
static int __init checksum_tg_init(void) static int __init checksum_tg_init(void)
{ {
return xt_register_target(&checksum_tg_reg); return xt_register_targets(checksum_tg_reg, ARRAY_SIZE(checksum_tg_reg));
} }
static void __exit checksum_tg_exit(void) static void __exit checksum_tg_exit(void)
{ {
xt_unregister_target(&checksum_tg_reg); xt_unregister_targets(checksum_tg_reg, ARRAY_SIZE(checksum_tg_reg));
} }
module_init(checksum_tg_init); module_init(checksum_tg_init);
......
...@@ -38,9 +38,9 @@ static struct xt_target classify_tg_reg[] __read_mostly = { ...@@ -38,9 +38,9 @@ static struct xt_target classify_tg_reg[] __read_mostly = {
{ {
.name = "CLASSIFY", .name = "CLASSIFY",
.revision = 0, .revision = 0,
.family = NFPROTO_UNSPEC, .family = NFPROTO_IPV4,
.hooks = (1 << NF_INET_LOCAL_OUT) | (1 << NF_INET_FORWARD) | .hooks = (1 << NF_INET_LOCAL_OUT) | (1 << NF_INET_FORWARD) |
(1 << NF_INET_POST_ROUTING), (1 << NF_INET_POST_ROUTING),
.target = classify_tg, .target = classify_tg,
.targetsize = sizeof(struct xt_classify_target_info), .targetsize = sizeof(struct xt_classify_target_info),
.me = THIS_MODULE, .me = THIS_MODULE,
...@@ -54,6 +54,18 @@ static struct xt_target classify_tg_reg[] __read_mostly = { ...@@ -54,6 +54,18 @@ static struct xt_target classify_tg_reg[] __read_mostly = {
.targetsize = sizeof(struct xt_classify_target_info), .targetsize = sizeof(struct xt_classify_target_info),
.me = THIS_MODULE, .me = THIS_MODULE,
}, },
#if IS_ENABLED(CONFIG_IP6_NF_IPTABLES)
{
.name = "CLASSIFY",
.revision = 0,
.family = NFPROTO_IPV6,
.hooks = (1 << NF_INET_LOCAL_OUT) | (1 << NF_INET_FORWARD) |
(1 << NF_INET_POST_ROUTING),
.target = classify_tg,
.targetsize = sizeof(struct xt_classify_target_info),
.me = THIS_MODULE,
},
#endif
}; };
static int __init classify_tg_init(void) static int __init classify_tg_init(void)
......
...@@ -114,25 +114,39 @@ static void connsecmark_tg_destroy(const struct xt_tgdtor_param *par) ...@@ -114,25 +114,39 @@ static void connsecmark_tg_destroy(const struct xt_tgdtor_param *par)
nf_ct_netns_put(par->net, par->family); nf_ct_netns_put(par->net, par->family);
} }
static struct xt_target connsecmark_tg_reg __read_mostly = { static struct xt_target connsecmark_tg_reg[] __read_mostly = {
.name = "CONNSECMARK", {
.revision = 0, .name = "CONNSECMARK",
.family = NFPROTO_UNSPEC, .revision = 0,
.checkentry = connsecmark_tg_check, .family = NFPROTO_IPV4,
.destroy = connsecmark_tg_destroy, .checkentry = connsecmark_tg_check,
.target = connsecmark_tg, .destroy = connsecmark_tg_destroy,
.targetsize = sizeof(struct xt_connsecmark_target_info), .target = connsecmark_tg,
.me = THIS_MODULE, .targetsize = sizeof(struct xt_connsecmark_target_info),
.me = THIS_MODULE,
},
#if IS_ENABLED(CONFIG_IP6_NF_IPTABLES)
{
.name = "CONNSECMARK",
.revision = 0,
.family = NFPROTO_IPV6,
.checkentry = connsecmark_tg_check,
.destroy = connsecmark_tg_destroy,
.target = connsecmark_tg,
.targetsize = sizeof(struct xt_connsecmark_target_info),
.me = THIS_MODULE,
},
#endif
}; };
static int __init connsecmark_tg_init(void) static int __init connsecmark_tg_init(void)
{ {
return xt_register_target(&connsecmark_tg_reg); return xt_register_targets(connsecmark_tg_reg, ARRAY_SIZE(connsecmark_tg_reg));
} }
static void __exit connsecmark_tg_exit(void) static void __exit connsecmark_tg_exit(void)
{ {
xt_unregister_target(&connsecmark_tg_reg); xt_unregister_targets(connsecmark_tg_reg, ARRAY_SIZE(connsecmark_tg_reg));
} }
module_init(connsecmark_tg_init); module_init(connsecmark_tg_init);
......
...@@ -313,10 +313,30 @@ static void xt_ct_tg_destroy_v1(const struct xt_tgdtor_param *par) ...@@ -313,10 +313,30 @@ static void xt_ct_tg_destroy_v1(const struct xt_tgdtor_param *par)
xt_ct_tg_destroy(par, par->targinfo); xt_ct_tg_destroy(par, par->targinfo);
} }
static unsigned int
notrack_tg(struct sk_buff *skb, const struct xt_action_param *par)
{
/* Previously seen (loopback)? Ignore. */
if (skb->_nfct != 0)
return XT_CONTINUE;
nf_ct_set(skb, NULL, IP_CT_UNTRACKED);
return XT_CONTINUE;
}
static struct xt_target xt_ct_tg_reg[] __read_mostly = { static struct xt_target xt_ct_tg_reg[] __read_mostly = {
{
.name = "NOTRACK",
.revision = 0,
.family = NFPROTO_IPV4,
.target = notrack_tg,
.table = "raw",
.me = THIS_MODULE,
},
{ {
.name = "CT", .name = "CT",
.family = NFPROTO_UNSPEC, .family = NFPROTO_IPV4,
.targetsize = sizeof(struct xt_ct_target_info), .targetsize = sizeof(struct xt_ct_target_info),
.usersize = offsetof(struct xt_ct_target_info, ct), .usersize = offsetof(struct xt_ct_target_info, ct),
.checkentry = xt_ct_tg_check_v0, .checkentry = xt_ct_tg_check_v0,
...@@ -327,7 +347,7 @@ static struct xt_target xt_ct_tg_reg[] __read_mostly = { ...@@ -327,7 +347,7 @@ static struct xt_target xt_ct_tg_reg[] __read_mostly = {
}, },
{ {
.name = "CT", .name = "CT",
.family = NFPROTO_UNSPEC, .family = NFPROTO_IPV4,
.revision = 1, .revision = 1,
.targetsize = sizeof(struct xt_ct_target_info_v1), .targetsize = sizeof(struct xt_ct_target_info_v1),
.usersize = offsetof(struct xt_ct_target_info, ct), .usersize = offsetof(struct xt_ct_target_info, ct),
...@@ -339,7 +359,7 @@ static struct xt_target xt_ct_tg_reg[] __read_mostly = { ...@@ -339,7 +359,7 @@ static struct xt_target xt_ct_tg_reg[] __read_mostly = {
}, },
{ {
.name = "CT", .name = "CT",
.family = NFPROTO_UNSPEC, .family = NFPROTO_IPV4,
.revision = 2, .revision = 2,
.targetsize = sizeof(struct xt_ct_target_info_v1), .targetsize = sizeof(struct xt_ct_target_info_v1),
.usersize = offsetof(struct xt_ct_target_info, ct), .usersize = offsetof(struct xt_ct_target_info, ct),
...@@ -349,49 +369,61 @@ static struct xt_target xt_ct_tg_reg[] __read_mostly = { ...@@ -349,49 +369,61 @@ static struct xt_target xt_ct_tg_reg[] __read_mostly = {
.table = "raw", .table = "raw",
.me = THIS_MODULE, .me = THIS_MODULE,
}, },
}; #if IS_ENABLED(CONFIG_IP6_NF_IPTABLES)
{
static unsigned int .name = "NOTRACK",
notrack_tg(struct sk_buff *skb, const struct xt_action_param *par) .revision = 0,
{ .family = NFPROTO_IPV6,
/* Previously seen (loopback)? Ignore. */ .target = notrack_tg,
if (skb->_nfct != 0) .table = "raw",
return XT_CONTINUE; .me = THIS_MODULE,
},
nf_ct_set(skb, NULL, IP_CT_UNTRACKED); {
.name = "CT",
return XT_CONTINUE; .family = NFPROTO_IPV6,
} .targetsize = sizeof(struct xt_ct_target_info),
.usersize = offsetof(struct xt_ct_target_info, ct),
static struct xt_target notrack_tg_reg __read_mostly = { .checkentry = xt_ct_tg_check_v0,
.name = "NOTRACK", .destroy = xt_ct_tg_destroy_v0,
.revision = 0, .target = xt_ct_target_v0,
.family = NFPROTO_UNSPEC, .table = "raw",
.target = notrack_tg, .me = THIS_MODULE,
.table = "raw", },
.me = THIS_MODULE, {
.name = "CT",
.family = NFPROTO_IPV6,
.revision = 1,
.targetsize = sizeof(struct xt_ct_target_info_v1),
.usersize = offsetof(struct xt_ct_target_info, ct),
.checkentry = xt_ct_tg_check_v1,
.destroy = xt_ct_tg_destroy_v1,
.target = xt_ct_target_v1,
.table = "raw",
.me = THIS_MODULE,
},
{
.name = "CT",
.family = NFPROTO_IPV6,
.revision = 2,
.targetsize = sizeof(struct xt_ct_target_info_v1),
.usersize = offsetof(struct xt_ct_target_info, ct),
.checkentry = xt_ct_tg_check_v2,
.destroy = xt_ct_tg_destroy_v1,
.target = xt_ct_target_v1,
.table = "raw",
.me = THIS_MODULE,
},
#endif
}; };
static int __init xt_ct_tg_init(void) static int __init xt_ct_tg_init(void)
{ {
int ret; return xt_register_targets(xt_ct_tg_reg, ARRAY_SIZE(xt_ct_tg_reg));
ret = xt_register_target(&notrack_tg_reg);
if (ret < 0)
return ret;
ret = xt_register_targets(xt_ct_tg_reg, ARRAY_SIZE(xt_ct_tg_reg));
if (ret < 0) {
xt_unregister_target(&notrack_tg_reg);
return ret;
}
return 0;
} }
static void __exit xt_ct_tg_exit(void) static void __exit xt_ct_tg_exit(void)
{ {
xt_unregister_targets(xt_ct_tg_reg, ARRAY_SIZE(xt_ct_tg_reg)); xt_unregister_targets(xt_ct_tg_reg, ARRAY_SIZE(xt_ct_tg_reg));
xt_unregister_target(&notrack_tg_reg);
} }
module_init(xt_ct_tg_init); module_init(xt_ct_tg_init);
......
...@@ -458,28 +458,49 @@ static void idletimer_tg_destroy_v1(const struct xt_tgdtor_param *par) ...@@ -458,28 +458,49 @@ static void idletimer_tg_destroy_v1(const struct xt_tgdtor_param *par)
static struct xt_target idletimer_tg[] __read_mostly = { static struct xt_target idletimer_tg[] __read_mostly = {
{ {
.name = "IDLETIMER", .name = "IDLETIMER",
.family = NFPROTO_UNSPEC, .family = NFPROTO_IPV4,
.target = idletimer_tg_target, .target = idletimer_tg_target,
.targetsize = sizeof(struct idletimer_tg_info), .targetsize = sizeof(struct idletimer_tg_info),
.usersize = offsetof(struct idletimer_tg_info, timer), .usersize = offsetof(struct idletimer_tg_info, timer),
.checkentry = idletimer_tg_checkentry, .checkentry = idletimer_tg_checkentry,
.destroy = idletimer_tg_destroy, .destroy = idletimer_tg_destroy,
.me = THIS_MODULE, .me = THIS_MODULE,
}, },
{ {
.name = "IDLETIMER", .name = "IDLETIMER",
.family = NFPROTO_UNSPEC, .family = NFPROTO_IPV4,
.revision = 1, .revision = 1,
.target = idletimer_tg_target_v1, .target = idletimer_tg_target_v1,
.targetsize = sizeof(struct idletimer_tg_info_v1), .targetsize = sizeof(struct idletimer_tg_info_v1),
.usersize = offsetof(struct idletimer_tg_info_v1, timer), .usersize = offsetof(struct idletimer_tg_info_v1, timer),
.checkentry = idletimer_tg_checkentry_v1, .checkentry = idletimer_tg_checkentry_v1,
.destroy = idletimer_tg_destroy_v1, .destroy = idletimer_tg_destroy_v1,
.me = THIS_MODULE, .me = THIS_MODULE,
}, },
#if IS_ENABLED(CONFIG_IP6_NF_IPTABLES)
{
.name = "IDLETIMER",
.family = NFPROTO_IPV6,
.target = idletimer_tg_target,
.targetsize = sizeof(struct idletimer_tg_info),
.usersize = offsetof(struct idletimer_tg_info, timer),
.checkentry = idletimer_tg_checkentry,
.destroy = idletimer_tg_destroy,
.me = THIS_MODULE,
},
{
.name = "IDLETIMER",
.family = NFPROTO_IPV6,
.revision = 1,
.target = idletimer_tg_target_v1,
.targetsize = sizeof(struct idletimer_tg_info_v1),
.usersize = offsetof(struct idletimer_tg_info_v1, timer),
.checkentry = idletimer_tg_checkentry_v1,
.destroy = idletimer_tg_destroy_v1,
.me = THIS_MODULE,
},
#endif
}; };
static struct class *idletimer_tg_class; static struct class *idletimer_tg_class;
......
...@@ -175,26 +175,41 @@ static void led_tg_destroy(const struct xt_tgdtor_param *par) ...@@ -175,26 +175,41 @@ static void led_tg_destroy(const struct xt_tgdtor_param *par)
kfree(ledinternal); kfree(ledinternal);
} }
static struct xt_target led_tg_reg __read_mostly = { static struct xt_target led_tg_reg[] __read_mostly = {
.name = "LED", {
.revision = 0, .name = "LED",
.family = NFPROTO_UNSPEC, .revision = 0,
.target = led_tg, .family = NFPROTO_IPV4,
.targetsize = sizeof(struct xt_led_info), .target = led_tg,
.usersize = offsetof(struct xt_led_info, internal_data), .targetsize = sizeof(struct xt_led_info),
.checkentry = led_tg_check, .usersize = offsetof(struct xt_led_info, internal_data),
.destroy = led_tg_destroy, .checkentry = led_tg_check,
.me = THIS_MODULE, .destroy = led_tg_destroy,
.me = THIS_MODULE,
},
#if IS_ENABLED(CONFIG_IP6_NF_IPTABLES)
{
.name = "LED",
.revision = 0,
.family = NFPROTO_IPV6,
.target = led_tg,
.targetsize = sizeof(struct xt_led_info),
.usersize = offsetof(struct xt_led_info, internal_data),
.checkentry = led_tg_check,
.destroy = led_tg_destroy,
.me = THIS_MODULE,
},
#endif
}; };
static int __init led_tg_init(void) static int __init led_tg_init(void)
{ {
return xt_register_target(&led_tg_reg); return xt_register_targets(led_tg_reg, ARRAY_SIZE(led_tg_reg));
} }
static void __exit led_tg_exit(void) static void __exit led_tg_exit(void)
{ {
xt_unregister_target(&led_tg_reg); xt_unregister_targets(led_tg_reg, ARRAY_SIZE(led_tg_reg));
} }
module_init(led_tg_init); module_init(led_tg_init);
......
...@@ -64,25 +64,39 @@ static void nflog_tg_destroy(const struct xt_tgdtor_param *par) ...@@ -64,25 +64,39 @@ static void nflog_tg_destroy(const struct xt_tgdtor_param *par)
nf_logger_put(par->family, NF_LOG_TYPE_ULOG); nf_logger_put(par->family, NF_LOG_TYPE_ULOG);
} }
static struct xt_target nflog_tg_reg __read_mostly = { static struct xt_target nflog_tg_reg[] __read_mostly = {
.name = "NFLOG", {
.revision = 0, .name = "NFLOG",
.family = NFPROTO_UNSPEC, .revision = 0,
.checkentry = nflog_tg_check, .family = NFPROTO_IPV4,
.destroy = nflog_tg_destroy, .checkentry = nflog_tg_check,
.target = nflog_tg, .destroy = nflog_tg_destroy,
.targetsize = sizeof(struct xt_nflog_info), .target = nflog_tg,
.me = THIS_MODULE, .targetsize = sizeof(struct xt_nflog_info),
.me = THIS_MODULE,
},
#if IS_ENABLED(CONFIG_IP6_NF_IPTABLES)
{
.name = "NFLOG",
.revision = 0,
.family = NFPROTO_IPV4,
.checkentry = nflog_tg_check,
.destroy = nflog_tg_destroy,
.target = nflog_tg,
.targetsize = sizeof(struct xt_nflog_info),
.me = THIS_MODULE,
},
#endif
}; };
static int __init nflog_tg_init(void) static int __init nflog_tg_init(void)
{ {
return xt_register_target(&nflog_tg_reg); return xt_register_targets(nflog_tg_reg, ARRAY_SIZE(nflog_tg_reg));
} }
static void __exit nflog_tg_exit(void) static void __exit nflog_tg_exit(void)
{ {
xt_unregister_target(&nflog_tg_reg); xt_unregister_targets(nflog_tg_reg, ARRAY_SIZE(nflog_tg_reg));
} }
module_init(nflog_tg_init); module_init(nflog_tg_init);
......
...@@ -179,16 +179,31 @@ static void xt_rateest_tg_destroy(const struct xt_tgdtor_param *par) ...@@ -179,16 +179,31 @@ static void xt_rateest_tg_destroy(const struct xt_tgdtor_param *par)
xt_rateest_put(par->net, info->est); xt_rateest_put(par->net, info->est);
} }
static struct xt_target xt_rateest_tg_reg __read_mostly = { static struct xt_target xt_rateest_tg_reg[] __read_mostly = {
.name = "RATEEST", {
.revision = 0, .name = "RATEEST",
.family = NFPROTO_UNSPEC, .revision = 0,
.target = xt_rateest_tg, .family = NFPROTO_IPV4,
.checkentry = xt_rateest_tg_checkentry, .target = xt_rateest_tg,
.destroy = xt_rateest_tg_destroy, .checkentry = xt_rateest_tg_checkentry,
.targetsize = sizeof(struct xt_rateest_target_info), .destroy = xt_rateest_tg_destroy,
.usersize = offsetof(struct xt_rateest_target_info, est), .targetsize = sizeof(struct xt_rateest_target_info),
.me = THIS_MODULE, .usersize = offsetof(struct xt_rateest_target_info, est),
.me = THIS_MODULE,
},
#if IS_ENABLED(CONFIG_IP6_NF_IPTABLES)
{
.name = "RATEEST",
.revision = 0,
.family = NFPROTO_IPV6,
.target = xt_rateest_tg,
.checkentry = xt_rateest_tg_checkentry,
.destroy = xt_rateest_tg_destroy,
.targetsize = sizeof(struct xt_rateest_target_info),
.usersize = offsetof(struct xt_rateest_target_info, est),
.me = THIS_MODULE,
},
#endif
}; };
static __net_init int xt_rateest_net_init(struct net *net) static __net_init int xt_rateest_net_init(struct net *net)
...@@ -214,12 +229,12 @@ static int __init xt_rateest_tg_init(void) ...@@ -214,12 +229,12 @@ static int __init xt_rateest_tg_init(void)
if (err) if (err)
return err; return err;
return xt_register_target(&xt_rateest_tg_reg); return xt_register_targets(xt_rateest_tg_reg, ARRAY_SIZE(xt_rateest_tg_reg));
} }
static void __exit xt_rateest_tg_fini(void) static void __exit xt_rateest_tg_fini(void)
{ {
xt_unregister_target(&xt_rateest_tg_reg); xt_unregister_targets(xt_rateest_tg_reg, ARRAY_SIZE(xt_rateest_tg_reg));
unregister_pernet_subsys(&xt_rateest_net_ops); unregister_pernet_subsys(&xt_rateest_net_ops);
} }
......
...@@ -157,7 +157,7 @@ static struct xt_target secmark_tg_reg[] __read_mostly = { ...@@ -157,7 +157,7 @@ static struct xt_target secmark_tg_reg[] __read_mostly = {
{ {
.name = "SECMARK", .name = "SECMARK",
.revision = 0, .revision = 0,
.family = NFPROTO_UNSPEC, .family = NFPROTO_IPV4,
.checkentry = secmark_tg_check_v0, .checkentry = secmark_tg_check_v0,
.destroy = secmark_tg_destroy, .destroy = secmark_tg_destroy,
.target = secmark_tg_v0, .target = secmark_tg_v0,
...@@ -167,7 +167,7 @@ static struct xt_target secmark_tg_reg[] __read_mostly = { ...@@ -167,7 +167,7 @@ static struct xt_target secmark_tg_reg[] __read_mostly = {
{ {
.name = "SECMARK", .name = "SECMARK",
.revision = 1, .revision = 1,
.family = NFPROTO_UNSPEC, .family = NFPROTO_IPV4,
.checkentry = secmark_tg_check_v1, .checkentry = secmark_tg_check_v1,
.destroy = secmark_tg_destroy, .destroy = secmark_tg_destroy,
.target = secmark_tg_v1, .target = secmark_tg_v1,
...@@ -175,6 +175,29 @@ static struct xt_target secmark_tg_reg[] __read_mostly = { ...@@ -175,6 +175,29 @@ static struct xt_target secmark_tg_reg[] __read_mostly = {
.usersize = offsetof(struct xt_secmark_target_info_v1, secid), .usersize = offsetof(struct xt_secmark_target_info_v1, secid),
.me = THIS_MODULE, .me = THIS_MODULE,
}, },
#if IS_ENABLED(CONFIG_IP6_NF_IPTABLES)
{
.name = "SECMARK",
.revision = 0,
.family = NFPROTO_IPV6,
.checkentry = secmark_tg_check_v0,
.destroy = secmark_tg_destroy,
.target = secmark_tg_v0,
.targetsize = sizeof(struct xt_secmark_target_info),
.me = THIS_MODULE,
},
{
.name = "SECMARK",
.revision = 1,
.family = NFPROTO_IPV6,
.checkentry = secmark_tg_check_v1,
.destroy = secmark_tg_destroy,
.target = secmark_tg_v1,
.targetsize = sizeof(struct xt_secmark_target_info_v1),
.usersize = offsetof(struct xt_secmark_target_info_v1, secid),
.me = THIS_MODULE,
},
#endif
}; };
static int __init secmark_tg_init(void) static int __init secmark_tg_init(void)
......
...@@ -29,25 +29,38 @@ trace_tg(struct sk_buff *skb, const struct xt_action_param *par) ...@@ -29,25 +29,38 @@ trace_tg(struct sk_buff *skb, const struct xt_action_param *par)
return XT_CONTINUE; return XT_CONTINUE;
} }
static struct xt_target trace_tg_reg __read_mostly = { static struct xt_target trace_tg_reg[] __read_mostly = {
.name = "TRACE", {
.revision = 0, .name = "TRACE",
.family = NFPROTO_UNSPEC, .revision = 0,
.table = "raw", .family = NFPROTO_IPV4,
.target = trace_tg, .table = "raw",
.checkentry = trace_tg_check, .target = trace_tg,
.destroy = trace_tg_destroy, .checkentry = trace_tg_check,
.me = THIS_MODULE, .destroy = trace_tg_destroy,
.me = THIS_MODULE,
},
#if IS_ENABLED(CONFIG_IP6_NF_IPTABLES)
{
.name = "TRACE",
.revision = 0,
.family = NFPROTO_IPV6,
.table = "raw",
.target = trace_tg,
.checkentry = trace_tg_check,
.destroy = trace_tg_destroy,
},
#endif
}; };
static int __init trace_tg_init(void) static int __init trace_tg_init(void)
{ {
return xt_register_target(&trace_tg_reg); return xt_register_targets(trace_tg_reg, ARRAY_SIZE(trace_tg_reg));
} }
static void __exit trace_tg_exit(void) static void __exit trace_tg_exit(void)
{ {
xt_unregister_target(&trace_tg_reg); xt_unregister_targets(trace_tg_reg, ARRAY_SIZE(trace_tg_reg));
} }
module_init(trace_tg_init); module_init(trace_tg_init);
......
...@@ -208,13 +208,24 @@ static struct xt_match addrtype_mt_reg[] __read_mostly = { ...@@ -208,13 +208,24 @@ static struct xt_match addrtype_mt_reg[] __read_mostly = {
}, },
{ {
.name = "addrtype", .name = "addrtype",
.family = NFPROTO_UNSPEC, .family = NFPROTO_IPV4,
.revision = 1, .revision = 1,
.match = addrtype_mt_v1, .match = addrtype_mt_v1,
.checkentry = addrtype_mt_checkentry_v1, .checkentry = addrtype_mt_checkentry_v1,
.matchsize = sizeof(struct xt_addrtype_info_v1), .matchsize = sizeof(struct xt_addrtype_info_v1),
.me = THIS_MODULE .me = THIS_MODULE
} },
#if IS_ENABLED(CONFIG_IP6_NF_IPTABLES)
{
.name = "addrtype",
.family = NFPROTO_IPV6,
.revision = 1,
.match = addrtype_mt_v1,
.checkentry = addrtype_mt_checkentry_v1,
.matchsize = sizeof(struct xt_addrtype_info_v1),
.me = THIS_MODULE
},
#endif
}; };
static int __init addrtype_mt_init(void) static int __init addrtype_mt_init(void)
......
...@@ -146,24 +146,37 @@ static void xt_cluster_mt_destroy(const struct xt_mtdtor_param *par) ...@@ -146,24 +146,37 @@ static void xt_cluster_mt_destroy(const struct xt_mtdtor_param *par)
nf_ct_netns_put(par->net, par->family); nf_ct_netns_put(par->net, par->family);
} }
static struct xt_match xt_cluster_match __read_mostly = { static struct xt_match xt_cluster_match[] __read_mostly = {
.name = "cluster", {
.family = NFPROTO_UNSPEC, .name = "cluster",
.match = xt_cluster_mt, .family = NFPROTO_IPV4,
.checkentry = xt_cluster_mt_checkentry, .match = xt_cluster_mt,
.matchsize = sizeof(struct xt_cluster_match_info), .checkentry = xt_cluster_mt_checkentry,
.destroy = xt_cluster_mt_destroy, .matchsize = sizeof(struct xt_cluster_match_info),
.me = THIS_MODULE, .destroy = xt_cluster_mt_destroy,
.me = THIS_MODULE,
},
#if IS_ENABLED(CONFIG_IP6_NF_IPTABLES)
{
.name = "cluster",
.family = NFPROTO_IPV6,
.match = xt_cluster_mt,
.checkentry = xt_cluster_mt_checkentry,
.matchsize = sizeof(struct xt_cluster_match_info),
.destroy = xt_cluster_mt_destroy,
.me = THIS_MODULE,
},
#endif
}; };
static int __init xt_cluster_mt_init(void) static int __init xt_cluster_mt_init(void)
{ {
return xt_register_match(&xt_cluster_match); return xt_register_matches(xt_cluster_match, ARRAY_SIZE(xt_cluster_match));
} }
static void __exit xt_cluster_mt_fini(void) static void __exit xt_cluster_mt_fini(void)
{ {
xt_unregister_match(&xt_cluster_match); xt_unregister_matches(xt_cluster_match, ARRAY_SIZE(xt_cluster_match));
} }
MODULE_AUTHOR("Pablo Neira Ayuso <pablo@netfilter.org>"); MODULE_AUTHOR("Pablo Neira Ayuso <pablo@netfilter.org>");
......
...@@ -111,9 +111,11 @@ static int connbytes_mt_check(const struct xt_mtchk_param *par) ...@@ -111,9 +111,11 @@ static int connbytes_mt_check(const struct xt_mtchk_param *par)
return -EINVAL; return -EINVAL;
ret = nf_ct_netns_get(par->net, par->family); ret = nf_ct_netns_get(par->net, par->family);
if (ret < 0) if (ret < 0) {
pr_info_ratelimited("cannot load conntrack support for proto=%u\n", pr_info_ratelimited("cannot load conntrack support for proto=%u\n",
par->family); par->family);
return ret;
}
/* /*
* This filter cannot function correctly unless connection tracking * This filter cannot function correctly unless connection tracking
......
...@@ -117,26 +117,41 @@ static void connlimit_mt_destroy(const struct xt_mtdtor_param *par) ...@@ -117,26 +117,41 @@ static void connlimit_mt_destroy(const struct xt_mtdtor_param *par)
nf_ct_netns_put(par->net, par->family); nf_ct_netns_put(par->net, par->family);
} }
static struct xt_match connlimit_mt_reg __read_mostly = { static struct xt_match connlimit_mt_reg[] __read_mostly = {
.name = "connlimit", {
.revision = 1, .name = "connlimit",
.family = NFPROTO_UNSPEC, .revision = 1,
.checkentry = connlimit_mt_check, .family = NFPROTO_IPV4,
.match = connlimit_mt, .checkentry = connlimit_mt_check,
.matchsize = sizeof(struct xt_connlimit_info), .match = connlimit_mt,
.usersize = offsetof(struct xt_connlimit_info, data), .matchsize = sizeof(struct xt_connlimit_info),
.destroy = connlimit_mt_destroy, .usersize = offsetof(struct xt_connlimit_info, data),
.me = THIS_MODULE, .destroy = connlimit_mt_destroy,
.me = THIS_MODULE,
},
#if IS_ENABLED(CONFIG_IP6_NF_IPTABLES)
{
.name = "connlimit",
.revision = 1,
.family = NFPROTO_IPV6,
.checkentry = connlimit_mt_check,
.match = connlimit_mt,
.matchsize = sizeof(struct xt_connlimit_info),
.usersize = offsetof(struct xt_connlimit_info, data),
.destroy = connlimit_mt_destroy,
.me = THIS_MODULE,
},
#endif
}; };
static int __init connlimit_mt_init(void) static int __init connlimit_mt_init(void)
{ {
return xt_register_match(&connlimit_mt_reg); return xt_register_matches(connlimit_mt_reg, ARRAY_SIZE(connlimit_mt_reg));
} }
static void __exit connlimit_mt_exit(void) static void __exit connlimit_mt_exit(void)
{ {
xt_unregister_match(&connlimit_mt_reg); xt_unregister_matches(connlimit_mt_reg, ARRAY_SIZE(connlimit_mt_reg));
} }
module_init(connlimit_mt_init); module_init(connlimit_mt_init);
......
...@@ -151,7 +151,7 @@ static struct xt_target connmark_tg_reg[] __read_mostly = { ...@@ -151,7 +151,7 @@ static struct xt_target connmark_tg_reg[] __read_mostly = {
{ {
.name = "CONNMARK", .name = "CONNMARK",
.revision = 1, .revision = 1,
.family = NFPROTO_UNSPEC, .family = NFPROTO_IPV4,
.checkentry = connmark_tg_check, .checkentry = connmark_tg_check,
.target = connmark_tg, .target = connmark_tg,
.targetsize = sizeof(struct xt_connmark_tginfo1), .targetsize = sizeof(struct xt_connmark_tginfo1),
...@@ -161,13 +161,35 @@ static struct xt_target connmark_tg_reg[] __read_mostly = { ...@@ -161,13 +161,35 @@ static struct xt_target connmark_tg_reg[] __read_mostly = {
{ {
.name = "CONNMARK", .name = "CONNMARK",
.revision = 2, .revision = 2,
.family = NFPROTO_UNSPEC, .family = NFPROTO_IPV4,
.checkentry = connmark_tg_check, .checkentry = connmark_tg_check,
.target = connmark_tg_v2, .target = connmark_tg_v2,
.targetsize = sizeof(struct xt_connmark_tginfo2), .targetsize = sizeof(struct xt_connmark_tginfo2),
.destroy = connmark_tg_destroy, .destroy = connmark_tg_destroy,
.me = THIS_MODULE, .me = THIS_MODULE,
} },
#if IS_ENABLED(CONFIG_IP6_NF_IPTABLES)
{
.name = "CONNMARK",
.revision = 1,
.family = NFPROTO_IPV6,
.checkentry = connmark_tg_check,
.target = connmark_tg,
.targetsize = sizeof(struct xt_connmark_tginfo1),
.destroy = connmark_tg_destroy,
.me = THIS_MODULE,
},
{
.name = "CONNMARK",
.revision = 2,
.family = NFPROTO_IPV6,
.checkentry = connmark_tg_check,
.target = connmark_tg_v2,
.targetsize = sizeof(struct xt_connmark_tginfo2),
.destroy = connmark_tg_destroy,
.me = THIS_MODULE,
},
#endif
}; };
static struct xt_match connmark_mt_reg __read_mostly = { static struct xt_match connmark_mt_reg __read_mostly = {
......
...@@ -39,13 +39,35 @@ mark_mt(const struct sk_buff *skb, struct xt_action_param *par) ...@@ -39,13 +39,35 @@ mark_mt(const struct sk_buff *skb, struct xt_action_param *par)
return ((skb->mark & info->mask) == info->mark) ^ info->invert; return ((skb->mark & info->mask) == info->mark) ^ info->invert;
} }
static struct xt_target mark_tg_reg __read_mostly = { static struct xt_target mark_tg_reg[] __read_mostly = {
.name = "MARK", {
.revision = 2, .name = "MARK",
.family = NFPROTO_UNSPEC, .revision = 2,
.target = mark_tg, .family = NFPROTO_IPV4,
.targetsize = sizeof(struct xt_mark_tginfo2), .target = mark_tg,
.me = THIS_MODULE, .targetsize = sizeof(struct xt_mark_tginfo2),
.me = THIS_MODULE,
},
#if IS_ENABLED(CONFIG_IP_NF_ARPTABLES)
{
.name = "MARK",
.revision = 2,
.family = NFPROTO_ARP,
.target = mark_tg,
.targetsize = sizeof(struct xt_mark_tginfo2),
.me = THIS_MODULE,
},
#endif
#if IS_ENABLED(CONFIG_IP6_NF_IPTABLES)
{
.name = "MARK",
.revision = 2,
.family = NFPROTO_IPV4,
.target = mark_tg,
.targetsize = sizeof(struct xt_mark_tginfo2),
.me = THIS_MODULE,
},
#endif
}; };
static struct xt_match mark_mt_reg __read_mostly = { static struct xt_match mark_mt_reg __read_mostly = {
...@@ -61,12 +83,12 @@ static int __init mark_mt_init(void) ...@@ -61,12 +83,12 @@ static int __init mark_mt_init(void)
{ {
int ret; int ret;
ret = xt_register_target(&mark_tg_reg); ret = xt_register_targets(mark_tg_reg, ARRAY_SIZE(mark_tg_reg));
if (ret < 0) if (ret < 0)
return ret; return ret;
ret = xt_register_match(&mark_mt_reg); ret = xt_register_match(&mark_mt_reg);
if (ret < 0) { if (ret < 0) {
xt_unregister_target(&mark_tg_reg); xt_unregister_targets(mark_tg_reg, ARRAY_SIZE(mark_tg_reg));
return ret; return ret;
} }
return 0; return 0;
...@@ -75,7 +97,7 @@ static int __init mark_mt_init(void) ...@@ -75,7 +97,7 @@ static int __init mark_mt_init(void)
static void __exit mark_mt_exit(void) static void __exit mark_mt_exit(void)
{ {
xt_unregister_match(&mark_mt_reg); xt_unregister_match(&mark_mt_reg);
xt_unregister_target(&mark_tg_reg); xt_unregister_targets(mark_tg_reg, ARRAY_SIZE(mark_tg_reg));
} }
module_init(mark_mt_init); module_init(mark_mt_init);
......
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