Skip to content
/ linux Public

Commit 04c8907

Browse files
V4belgregkh
authored andcommitted
netfilter: ctnetlink: fix use-after-free in ctnetlink_dump_exp_ct()
[ Upstream commit 5cb81ee ] ctnetlink_dump_exp_ct() stores a conntrack pointer in cb->data for the netlink dump callback ctnetlink_exp_ct_dump_table(), but drops the conntrack reference immediately after netlink_dump_start(). When the dump spans multiple rounds, the second recvmsg() triggers the dump callback which dereferences the now-freed conntrack via nfct_help(ct), leading to a use-after-free on ct->ext. The bug is that the netlink_dump_control has no .start or .done callbacks to manage the conntrack reference across dump rounds. Other dump functions in the same file (e.g. ctnetlink_get_conntrack) properly use .start/.done callbacks for this purpose. Fix this by adding .start and .done callbacks that hold and release the conntrack reference for the duration of the dump, and move the nfct_help() call after the cb->args[0] early-return check in the dump callback to avoid dereferencing ct->ext unnecessarily. BUG: KASAN: slab-use-after-free in ctnetlink_exp_ct_dump_table+0x4f/0x2e0 Read of size 8 at addr ffff88810597ebf0 by task ctnetlink_poc/133 CPU: 1 UID: 0 PID: 133 Comm: ctnetlink_poc Not tainted 7.0.0-rc2+ #3 PREEMPTLAZY Call Trace: <TASK> ctnetlink_exp_ct_dump_table+0x4f/0x2e0 netlink_dump+0x333/0x880 netlink_recvmsg+0x3e2/0x4b0 ? aa_sk_perm+0x184/0x450 sock_recvmsg+0xde/0xf0 Allocated by task 133: kmem_cache_alloc_noprof+0x134/0x440 __nf_conntrack_alloc+0xa8/0x2b0 ctnetlink_create_conntrack+0xa1/0x900 ctnetlink_new_conntrack+0x3cf/0x7d0 nfnetlink_rcv_msg+0x48e/0x510 netlink_rcv_skb+0xc9/0x1f0 nfnetlink_rcv+0xdb/0x220 netlink_unicast+0x3ec/0x590 netlink_sendmsg+0x397/0x690 __sys_sendmsg+0xf4/0x180 Freed by task 0: slab_free_after_rcu_debug+0xad/0x1e0 rcu_core+0x5c3/0x9c0 Fixes: e844a92 ("netfilter: ctnetlink: allow to dump expectation per master conntrack") Signed-off-by: Hyunwoo Kim <imv4bel@gmail.com> Signed-off-by: Florian Westphal <fw@strlen.de> Signed-off-by: Sasha Levin <sashal@kernel.org>
1 parent e01a597 commit 04c8907

File tree

1 file changed

+25
-1
lines changed

1 file changed

+25
-1
lines changed

net/netfilter/nf_conntrack_netlink.c

Lines changed: 25 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -3211,14 +3211,18 @@ ctnetlink_exp_ct_dump_table(struct sk_buff *skb, struct netlink_callback *cb)
32113211
{
32123212
struct nfgenmsg *nfmsg = nlmsg_data(cb->nlh);
32133213
struct nf_conn *ct = cb->data;
3214-
struct nf_conn_help *help = nfct_help(ct);
3214+
struct nf_conn_help *help;
32153215
u_int8_t l3proto = nfmsg->nfgen_family;
32163216
unsigned long last_id = cb->args[1];
32173217
struct nf_conntrack_expect *exp;
32183218

32193219
if (cb->args[0])
32203220
return 0;
32213221

3222+
help = nfct_help(ct);
3223+
if (!help)
3224+
return 0;
3225+
32223226
rcu_read_lock();
32233227

32243228
restart:
@@ -3248,6 +3252,24 @@ ctnetlink_exp_ct_dump_table(struct sk_buff *skb, struct netlink_callback *cb)
32483252
return skb->len;
32493253
}
32503254

3255+
static int ctnetlink_dump_exp_ct_start(struct netlink_callback *cb)
3256+
{
3257+
struct nf_conn *ct = cb->data;
3258+
3259+
if (!refcount_inc_not_zero(&ct->ct_general.use))
3260+
return -ENOENT;
3261+
return 0;
3262+
}
3263+
3264+
static int ctnetlink_dump_exp_ct_done(struct netlink_callback *cb)
3265+
{
3266+
struct nf_conn *ct = cb->data;
3267+
3268+
if (ct)
3269+
nf_ct_put(ct);
3270+
return 0;
3271+
}
3272+
32513273
static int ctnetlink_dump_exp_ct(struct net *net, struct sock *ctnl,
32523274
struct sk_buff *skb,
32533275
const struct nlmsghdr *nlh,
@@ -3263,6 +3285,8 @@ static int ctnetlink_dump_exp_ct(struct net *net, struct sock *ctnl,
32633285
struct nf_conntrack_zone zone;
32643286
struct netlink_dump_control c = {
32653287
.dump = ctnetlink_exp_ct_dump_table,
3288+
.start = ctnetlink_dump_exp_ct_start,
3289+
.done = ctnetlink_dump_exp_ct_done,
32663290
};
32673291

32683292
err = ctnetlink_parse_tuple(cda, &tuple, CTA_EXPECT_MASTER,

0 commit comments

Comments
 (0)