netfilter: nf_tables: fix nft_counters_enabled underflow at nf_tables_addchain()
[ Upstream commit 921ebde3c0d22c8cba74ce8eb3cc4626abff1ccd ] syzbot is reporting underflow of nft_counters_enabled counter at nf_tables_addchain() [1], for commit 43eb8949cfdffa76 ("netfilter: nf_tables: do not leave chain stats enabled on error") missed that nf_tables_chain_destroy() after nft_basechain_init() in the error path of nf_tables_addchain() decrements the counter because nft_basechain_init() makes nft_is_base_chain() return true by setting NFT_CHAIN_BASE flag. Increment the counter immediately after returning from nft_basechain_init(). Link: https://syzkaller.appspot.com/bug?extid=b5d82a651b71cd8a75ab [1] Reported-by: syzbot <syzbot+b5d82a651b71cd8a75ab@syzkaller.appspotmail.com> Signed-off-by: Tetsuo Handa <penguin-kernel@I-love.SAKURA.ne.jp> Tested-by: syzbot <syzbot+b5d82a651b71cd8a75ab@syzkaller.appspotmail.com> Fixes: 43eb8949cfdffa76 ("netfilter: nf_tables: do not leave chain stats enabled on error") Signed-off-by: Florian Westphal <fw@strlen.de> Signed-off-by: Sasha Levin <sashal@kernel.org>
This commit is contained in:

committed by
Greg Kroah-Hartman

parent
1e7e55374d
commit
710e3f526b
@@ -2001,7 +2001,6 @@ static int nf_tables_addchain(struct nft_ctx *ctx, u8 family, u8 genmask,
|
|||||||
u8 policy, u32 flags)
|
u8 policy, u32 flags)
|
||||||
{
|
{
|
||||||
const struct nlattr * const *nla = ctx->nla;
|
const struct nlattr * const *nla = ctx->nla;
|
||||||
struct nft_stats __percpu *stats = NULL;
|
|
||||||
struct nft_table *table = ctx->table;
|
struct nft_table *table = ctx->table;
|
||||||
struct nft_base_chain *basechain;
|
struct nft_base_chain *basechain;
|
||||||
struct net *net = ctx->net;
|
struct net *net = ctx->net;
|
||||||
@@ -2015,6 +2014,7 @@ static int nf_tables_addchain(struct nft_ctx *ctx, u8 family, u8 genmask,
|
|||||||
return -EOVERFLOW;
|
return -EOVERFLOW;
|
||||||
|
|
||||||
if (nla[NFTA_CHAIN_HOOK]) {
|
if (nla[NFTA_CHAIN_HOOK]) {
|
||||||
|
struct nft_stats __percpu *stats = NULL;
|
||||||
struct nft_chain_hook hook;
|
struct nft_chain_hook hook;
|
||||||
|
|
||||||
if (flags & NFT_CHAIN_BINDING)
|
if (flags & NFT_CHAIN_BINDING)
|
||||||
@@ -2047,6 +2047,8 @@ static int nf_tables_addchain(struct nft_ctx *ctx, u8 family, u8 genmask,
|
|||||||
kfree(basechain);
|
kfree(basechain);
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
if (stats)
|
||||||
|
static_branch_inc(&nft_counters_enabled);
|
||||||
} else {
|
} else {
|
||||||
if (flags & NFT_CHAIN_BASE)
|
if (flags & NFT_CHAIN_BASE)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
@@ -2121,9 +2123,6 @@ static int nf_tables_addchain(struct nft_ctx *ctx, u8 family, u8 genmask,
|
|||||||
goto err_unregister_hook;
|
goto err_unregister_hook;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (stats)
|
|
||||||
static_branch_inc(&nft_counters_enabled);
|
|
||||||
|
|
||||||
table->use++;
|
table->use++;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
Reference in New Issue
Block a user