Commit 4203f2a7 authored by Liping Zhang's avatar Liping Zhang Committed by Greg Kroah-Hartman

netfilter: nfnl_cthelper: fix incorrect helper->expect_class_max


[ Upstream commit ae5c6821 ]

The helper->expect_class_max must be set to the total number of
expect_policy minus 1, since we will use the statement "if (class >
helper->expect_class_max)" to validate the CTA_EXPECT_CLASS attr in
ctnetlink_alloc_expect.

So for compatibility, set the helper->expect_class_max to the
NFCTH_POLICY_SET_NUM attr's value minus 1.

Also: it's invalid when the NFCTH_POLICY_SET_NUM attr's value is zero.
1. this will result "expect_policy = kzalloc(0, GFP_KERNEL);";
2. we cannot set the helper->expect_class_max to a proper value.

So if nla_get_be32(tb[NFCTH_POLICY_SET_NUM]) is zero, report -EINVAL to
the userspace.
Signed-off-by: default avatarLiping Zhang <zlpnobody@gmail.com>
Signed-off-by: default avatarPablo Neira Ayuso <pablo@netfilter.org>
Signed-off-by: default avatarSasha Levin <alexander.levin@verizon.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent fa029020
...@@ -161,6 +161,7 @@ nfnl_cthelper_parse_expect_policy(struct nf_conntrack_helper *helper, ...@@ -161,6 +161,7 @@ nfnl_cthelper_parse_expect_policy(struct nf_conntrack_helper *helper,
int i, ret; int i, ret;
struct nf_conntrack_expect_policy *expect_policy; struct nf_conntrack_expect_policy *expect_policy;
struct nlattr *tb[NFCTH_POLICY_SET_MAX+1]; struct nlattr *tb[NFCTH_POLICY_SET_MAX+1];
unsigned int class_max;
ret = nla_parse_nested(tb, NFCTH_POLICY_SET_MAX, attr, ret = nla_parse_nested(tb, NFCTH_POLICY_SET_MAX, attr,
nfnl_cthelper_expect_policy_set); nfnl_cthelper_expect_policy_set);
...@@ -170,19 +171,18 @@ nfnl_cthelper_parse_expect_policy(struct nf_conntrack_helper *helper, ...@@ -170,19 +171,18 @@ nfnl_cthelper_parse_expect_policy(struct nf_conntrack_helper *helper,
if (!tb[NFCTH_POLICY_SET_NUM]) if (!tb[NFCTH_POLICY_SET_NUM])
return -EINVAL; return -EINVAL;
helper->expect_class_max = class_max = ntohl(nla_get_be32(tb[NFCTH_POLICY_SET_NUM]));
ntohl(nla_get_be32(tb[NFCTH_POLICY_SET_NUM])); if (class_max == 0)
return -EINVAL;
if (helper->expect_class_max != 0 && if (class_max > NF_CT_MAX_EXPECT_CLASSES)
helper->expect_class_max > NF_CT_MAX_EXPECT_CLASSES)
return -EOVERFLOW; return -EOVERFLOW;
expect_policy = kzalloc(sizeof(struct nf_conntrack_expect_policy) * expect_policy = kzalloc(sizeof(struct nf_conntrack_expect_policy) *
helper->expect_class_max, GFP_KERNEL); class_max, GFP_KERNEL);
if (expect_policy == NULL) if (expect_policy == NULL)
return -ENOMEM; return -ENOMEM;
for (i=0; i<helper->expect_class_max; i++) { for (i = 0; i < class_max; i++) {
if (!tb[NFCTH_POLICY_SET+i]) if (!tb[NFCTH_POLICY_SET+i])
goto err; goto err;
...@@ -191,6 +191,8 @@ nfnl_cthelper_parse_expect_policy(struct nf_conntrack_helper *helper, ...@@ -191,6 +191,8 @@ nfnl_cthelper_parse_expect_policy(struct nf_conntrack_helper *helper,
if (ret < 0) if (ret < 0)
goto err; goto err;
} }
helper->expect_class_max = class_max - 1;
helper->expect_policy = expect_policy; helper->expect_policy = expect_policy;
return 0; return 0;
err: err:
...@@ -377,10 +379,10 @@ nfnl_cthelper_dump_policy(struct sk_buff *skb, ...@@ -377,10 +379,10 @@ nfnl_cthelper_dump_policy(struct sk_buff *skb,
goto nla_put_failure; goto nla_put_failure;
if (nla_put_be32(skb, NFCTH_POLICY_SET_NUM, if (nla_put_be32(skb, NFCTH_POLICY_SET_NUM,
htonl(helper->expect_class_max))) htonl(helper->expect_class_max + 1)))
goto nla_put_failure; goto nla_put_failure;
for (i=0; i<helper->expect_class_max; i++) { for (i = 0; i < helper->expect_class_max + 1; i++) {
nest_parms2 = nla_nest_start(skb, nest_parms2 = nla_nest_start(skb,
(NFCTH_POLICY_SET+i) | NLA_F_NESTED); (NFCTH_POLICY_SET+i) | NLA_F_NESTED);
if (nest_parms2 == NULL) if (nest_parms2 == NULL)
......
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