Commit 2df5d103 authored by David S. Miller's avatar David S. Miller

Merge git://git.kernel.org/pub/scm/linux/kernel/git/pablo/nf

Pablo Neira Ayuso says:

====================
Netfilter fixes for net

The following patchset contains Netfilter fixes for your net tree,
they are:

1) Allow nf_tables reject expression from input, forward and output hooks,
   since only there the routing information is available, otherwise we crash.

2) Fix unsafe list iteration when flushing timeout and accouting objects.

3) Fix refcount leak on timeout policy parsing failure.

4) Unlink timeout object for unconfirmed conntracks too

5) Missing validation of pkttype mangling from bridge family.

6) Fix refcount leak on ebtables on second lookup for the specific
   bridge match extension, this patch from Sabrina Dubroca.

7) Remove unnecessary ip_hdr() in nf_tables_netdev family.

Patches from 1-5 and 7 from Liping Zhang.
====================
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parents 15543692 c73c2484
...@@ -36,4 +36,8 @@ void nft_meta_set_eval(const struct nft_expr *expr, ...@@ -36,4 +36,8 @@ void nft_meta_set_eval(const struct nft_expr *expr,
void nft_meta_set_destroy(const struct nft_ctx *ctx, void nft_meta_set_destroy(const struct nft_ctx *ctx,
const struct nft_expr *expr); const struct nft_expr *expr);
int nft_meta_set_validate(const struct nft_ctx *ctx,
const struct nft_expr *expr,
const struct nft_data **data);
#endif #endif
...@@ -8,6 +8,10 @@ struct nft_reject { ...@@ -8,6 +8,10 @@ struct nft_reject {
extern const struct nla_policy nft_reject_policy[]; extern const struct nla_policy nft_reject_policy[];
int nft_reject_validate(const struct nft_ctx *ctx,
const struct nft_expr *expr,
const struct nft_data **data);
int nft_reject_init(const struct nft_ctx *ctx, int nft_reject_init(const struct nft_ctx *ctx,
const struct nft_expr *expr, const struct nft_expr *expr,
const struct nlattr * const tb[]); const struct nlattr * const tb[]);
......
...@@ -368,6 +368,8 @@ ebt_check_match(struct ebt_entry_match *m, struct xt_mtchk_param *par, ...@@ -368,6 +368,8 @@ ebt_check_match(struct ebt_entry_match *m, struct xt_mtchk_param *par,
match = xt_find_match(NFPROTO_BRIDGE, m->u.name, 0); match = xt_find_match(NFPROTO_BRIDGE, m->u.name, 0);
if (IS_ERR(match) || match->family != NFPROTO_BRIDGE) { if (IS_ERR(match) || match->family != NFPROTO_BRIDGE) {
if (!IS_ERR(match))
module_put(match->me);
request_module("ebt_%s", m->u.name); request_module("ebt_%s", m->u.name);
match = xt_find_match(NFPROTO_BRIDGE, m->u.name, 0); match = xt_find_match(NFPROTO_BRIDGE, m->u.name, 0);
} }
......
...@@ -86,6 +86,7 @@ static const struct nft_expr_ops nft_meta_bridge_set_ops = { ...@@ -86,6 +86,7 @@ static const struct nft_expr_ops nft_meta_bridge_set_ops = {
.init = nft_meta_set_init, .init = nft_meta_set_init,
.destroy = nft_meta_set_destroy, .destroy = nft_meta_set_destroy,
.dump = nft_meta_set_dump, .dump = nft_meta_set_dump,
.validate = nft_meta_set_validate,
}; };
static const struct nft_expr_ops * static const struct nft_expr_ops *
......
...@@ -46,6 +46,7 @@ static const struct nft_expr_ops nft_reject_ipv4_ops = { ...@@ -46,6 +46,7 @@ static const struct nft_expr_ops nft_reject_ipv4_ops = {
.eval = nft_reject_ipv4_eval, .eval = nft_reject_ipv4_eval,
.init = nft_reject_init, .init = nft_reject_init,
.dump = nft_reject_dump, .dump = nft_reject_dump,
.validate = nft_reject_validate,
}; };
static struct nft_expr_type nft_reject_ipv4_type __read_mostly = { static struct nft_expr_type nft_reject_ipv4_type __read_mostly = {
......
...@@ -47,6 +47,7 @@ static const struct nft_expr_ops nft_reject_ipv6_ops = { ...@@ -47,6 +47,7 @@ static const struct nft_expr_ops nft_reject_ipv6_ops = {
.eval = nft_reject_ipv6_eval, .eval = nft_reject_ipv6_eval,
.init = nft_reject_init, .init = nft_reject_init,
.dump = nft_reject_dump, .dump = nft_reject_dump,
.validate = nft_reject_validate,
}; };
static struct nft_expr_type nft_reject_ipv6_type __read_mostly = { static struct nft_expr_type nft_reject_ipv6_type __read_mostly = {
......
...@@ -30,7 +30,6 @@ nft_netdev_set_pktinfo_ipv4(struct nft_pktinfo *pkt, ...@@ -30,7 +30,6 @@ nft_netdev_set_pktinfo_ipv4(struct nft_pktinfo *pkt,
if (!iph) if (!iph)
return; return;
iph = ip_hdr(skb);
if (iph->ihl < 5 || iph->version != 4) if (iph->ihl < 5 || iph->version != 4)
return; return;
......
...@@ -343,12 +343,12 @@ static int nfnl_acct_del(struct net *net, struct sock *nfnl, ...@@ -343,12 +343,12 @@ static int nfnl_acct_del(struct net *net, struct sock *nfnl,
struct sk_buff *skb, const struct nlmsghdr *nlh, struct sk_buff *skb, const struct nlmsghdr *nlh,
const struct nlattr * const tb[]) const struct nlattr * const tb[])
{ {
char *acct_name; struct nf_acct *cur, *tmp;
struct nf_acct *cur;
int ret = -ENOENT; int ret = -ENOENT;
char *acct_name;
if (!tb[NFACCT_NAME]) { if (!tb[NFACCT_NAME]) {
list_for_each_entry(cur, &net->nfnl_acct_list, head) list_for_each_entry_safe(cur, tmp, &net->nfnl_acct_list, head)
nfnl_acct_try_del(cur); nfnl_acct_try_del(cur);
return 0; return 0;
......
...@@ -98,31 +98,28 @@ static int cttimeout_new_timeout(struct net *net, struct sock *ctnl, ...@@ -98,31 +98,28 @@ static int cttimeout_new_timeout(struct net *net, struct sock *ctnl,
break; break;
} }
l4proto = nf_ct_l4proto_find_get(l3num, l4num);
/* This protocol is not supportted, skip. */
if (l4proto->l4proto != l4num) {
ret = -EOPNOTSUPP;
goto err_proto_put;
}
if (matching) { if (matching) {
if (nlh->nlmsg_flags & NLM_F_REPLACE) { if (nlh->nlmsg_flags & NLM_F_REPLACE) {
/* You cannot replace one timeout policy by another of /* You cannot replace one timeout policy by another of
* different kind, sorry. * different kind, sorry.
*/ */
if (matching->l3num != l3num || if (matching->l3num != l3num ||
matching->l4proto->l4proto != l4num) { matching->l4proto->l4proto != l4num)
ret = -EINVAL; return -EINVAL;
goto err_proto_put;
} return ctnl_timeout_parse_policy(&matching->data,
matching->l4proto, net,
ret = ctnl_timeout_parse_policy(&matching->data, cda[CTA_TIMEOUT_DATA]);
l4proto, net,
cda[CTA_TIMEOUT_DATA]);
return ret;
} }
ret = -EBUSY;
return -EBUSY;
}
l4proto = nf_ct_l4proto_find_get(l3num, l4num);
/* This protocol is not supportted, skip. */
if (l4proto->l4proto != l4num) {
ret = -EOPNOTSUPP;
goto err_proto_put; goto err_proto_put;
} }
...@@ -305,7 +302,16 @@ static void ctnl_untimeout(struct net *net, struct ctnl_timeout *timeout) ...@@ -305,7 +302,16 @@ static void ctnl_untimeout(struct net *net, struct ctnl_timeout *timeout)
const struct hlist_nulls_node *nn; const struct hlist_nulls_node *nn;
unsigned int last_hsize; unsigned int last_hsize;
spinlock_t *lock; spinlock_t *lock;
int i; int i, cpu;
for_each_possible_cpu(cpu) {
struct ct_pcpu *pcpu = per_cpu_ptr(net->ct.pcpu_lists, cpu);
spin_lock_bh(&pcpu->lock);
hlist_nulls_for_each_entry(h, nn, &pcpu->unconfirmed, hnnode)
untimeout(h, timeout);
spin_unlock_bh(&pcpu->lock);
}
local_bh_disable(); local_bh_disable();
restart: restart:
...@@ -350,12 +356,13 @@ static int cttimeout_del_timeout(struct net *net, struct sock *ctnl, ...@@ -350,12 +356,13 @@ static int cttimeout_del_timeout(struct net *net, struct sock *ctnl,
const struct nlmsghdr *nlh, const struct nlmsghdr *nlh,
const struct nlattr * const cda[]) const struct nlattr * const cda[])
{ {
struct ctnl_timeout *cur; struct ctnl_timeout *cur, *tmp;
int ret = -ENOENT; int ret = -ENOENT;
char *name; char *name;
if (!cda[CTA_TIMEOUT_NAME]) { if (!cda[CTA_TIMEOUT_NAME]) {
list_for_each_entry(cur, &net->nfct_timeout_list, head) list_for_each_entry_safe(cur, tmp, &net->nfct_timeout_list,
head)
ctnl_timeout_try_del(net, cur); ctnl_timeout_try_del(net, cur);
return 0; return 0;
......
...@@ -291,10 +291,16 @@ int nft_meta_get_init(const struct nft_ctx *ctx, ...@@ -291,10 +291,16 @@ int nft_meta_get_init(const struct nft_ctx *ctx,
} }
EXPORT_SYMBOL_GPL(nft_meta_get_init); EXPORT_SYMBOL_GPL(nft_meta_get_init);
static int nft_meta_set_init_pkttype(const struct nft_ctx *ctx) int nft_meta_set_validate(const struct nft_ctx *ctx,
const struct nft_expr *expr,
const struct nft_data **data)
{ {
struct nft_meta *priv = nft_expr_priv(expr);
unsigned int hooks; unsigned int hooks;
if (priv->key != NFT_META_PKTTYPE)
return 0;
switch (ctx->afi->family) { switch (ctx->afi->family) {
case NFPROTO_BRIDGE: case NFPROTO_BRIDGE:
hooks = 1 << NF_BR_PRE_ROUTING; hooks = 1 << NF_BR_PRE_ROUTING;
...@@ -308,6 +314,7 @@ static int nft_meta_set_init_pkttype(const struct nft_ctx *ctx) ...@@ -308,6 +314,7 @@ static int nft_meta_set_init_pkttype(const struct nft_ctx *ctx)
return nft_chain_validate_hooks(ctx->chain, hooks); return nft_chain_validate_hooks(ctx->chain, hooks);
} }
EXPORT_SYMBOL_GPL(nft_meta_set_validate);
int nft_meta_set_init(const struct nft_ctx *ctx, int nft_meta_set_init(const struct nft_ctx *ctx,
const struct nft_expr *expr, const struct nft_expr *expr,
...@@ -327,15 +334,16 @@ int nft_meta_set_init(const struct nft_ctx *ctx, ...@@ -327,15 +334,16 @@ int nft_meta_set_init(const struct nft_ctx *ctx,
len = sizeof(u8); len = sizeof(u8);
break; break;
case NFT_META_PKTTYPE: case NFT_META_PKTTYPE:
err = nft_meta_set_init_pkttype(ctx);
if (err)
return err;
len = sizeof(u8); len = sizeof(u8);
break; break;
default: default:
return -EOPNOTSUPP; return -EOPNOTSUPP;
} }
err = nft_meta_set_validate(ctx, expr, NULL);
if (err < 0)
return err;
priv->sreg = nft_parse_register(tb[NFTA_META_SREG]); priv->sreg = nft_parse_register(tb[NFTA_META_SREG]);
err = nft_validate_register_load(priv->sreg, len); err = nft_validate_register_load(priv->sreg, len);
if (err < 0) if (err < 0)
...@@ -407,6 +415,7 @@ static const struct nft_expr_ops nft_meta_set_ops = { ...@@ -407,6 +415,7 @@ static const struct nft_expr_ops nft_meta_set_ops = {
.init = nft_meta_set_init, .init = nft_meta_set_init,
.destroy = nft_meta_set_destroy, .destroy = nft_meta_set_destroy,
.dump = nft_meta_set_dump, .dump = nft_meta_set_dump,
.validate = nft_meta_set_validate,
}; };
static const struct nft_expr_ops * static const struct nft_expr_ops *
......
...@@ -26,11 +26,27 @@ const struct nla_policy nft_reject_policy[NFTA_REJECT_MAX + 1] = { ...@@ -26,11 +26,27 @@ const struct nla_policy nft_reject_policy[NFTA_REJECT_MAX + 1] = {
}; };
EXPORT_SYMBOL_GPL(nft_reject_policy); EXPORT_SYMBOL_GPL(nft_reject_policy);
int nft_reject_validate(const struct nft_ctx *ctx,
const struct nft_expr *expr,
const struct nft_data **data)
{
return nft_chain_validate_hooks(ctx->chain,
(1 << NF_INET_LOCAL_IN) |
(1 << NF_INET_FORWARD) |
(1 << NF_INET_LOCAL_OUT));
}
EXPORT_SYMBOL_GPL(nft_reject_validate);
int nft_reject_init(const struct nft_ctx *ctx, int nft_reject_init(const struct nft_ctx *ctx,
const struct nft_expr *expr, const struct nft_expr *expr,
const struct nlattr * const tb[]) const struct nlattr * const tb[])
{ {
struct nft_reject *priv = nft_expr_priv(expr); struct nft_reject *priv = nft_expr_priv(expr);
int err;
err = nft_reject_validate(ctx, expr, NULL);
if (err < 0)
return err;
if (tb[NFTA_REJECT_TYPE] == NULL) if (tb[NFTA_REJECT_TYPE] == NULL)
return -EINVAL; return -EINVAL;
......
...@@ -66,7 +66,11 @@ static int nft_reject_inet_init(const struct nft_ctx *ctx, ...@@ -66,7 +66,11 @@ static int nft_reject_inet_init(const struct nft_ctx *ctx,
const struct nlattr * const tb[]) const struct nlattr * const tb[])
{ {
struct nft_reject *priv = nft_expr_priv(expr); struct nft_reject *priv = nft_expr_priv(expr);
int icmp_code; int icmp_code, err;
err = nft_reject_validate(ctx, expr, NULL);
if (err < 0)
return err;
if (tb[NFTA_REJECT_TYPE] == NULL) if (tb[NFTA_REJECT_TYPE] == NULL)
return -EINVAL; return -EINVAL;
...@@ -124,6 +128,7 @@ static const struct nft_expr_ops nft_reject_inet_ops = { ...@@ -124,6 +128,7 @@ static const struct nft_expr_ops nft_reject_inet_ops = {
.eval = nft_reject_inet_eval, .eval = nft_reject_inet_eval,
.init = nft_reject_inet_init, .init = nft_reject_inet_init,
.dump = nft_reject_inet_dump, .dump = nft_reject_inet_dump,
.validate = nft_reject_validate,
}; };
static struct nft_expr_type nft_reject_inet_type __read_mostly = { static struct nft_expr_type nft_reject_inet_type __read_mostly = {
......
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