Skip to content

Commit 47d4b36

Browse files
committed
netfilter: conntrack: fix rmmod double-free race
jira VULN-430 cve-pre CVE-2023-4244 commit-author Florian Westphal <fw@strlen.de> commit e6d57e9 nf_conntrack_hash_check_insert() callers free the ct entry directly, via nf_conntrack_free. This isn't safe anymore because nf_conntrack_hash_check_insert() might place the entry into the conntrack table and then delteted the entry again because it found that a conntrack extension has been removed at the same time. In this case, the just-added entry is removed again and an error is returned to the caller. Problem is that another cpu might have picked up this entry and incremented its reference count. This results in a use-after-free/double-free, once by the other cpu and once by the caller of nf_conntrack_hash_check_insert(). Fix this by making nf_conntrack_hash_check_insert() not fail anymore after the insertion, just like before the 'Fixes' commit. This is safe because a racing nf_ct_iterate() has to wait for us to release the conntrack hash spinlocks. While at it, make the function return -EAGAIN in the rmmod (genid changed) case, this makes nfnetlink replay the command (suggested by Pablo Neira). Fixes: c56716c ("netfilter: extensions: introduce extension genid count") Signed-off-by: Florian Westphal <fw@strlen.de> Signed-off-by: Pablo Neira Ayuso <pablo@netfilter.org> (cherry picked from commit e6d57e9) Signed-off-by: Marcin Wcisło <marcin.wcislo@conclusive.pl>
1 parent 709f6c7 commit 47d4b36

File tree

2 files changed

+15
-13
lines changed

2 files changed

+15
-13
lines changed

net/netfilter/nf_conntrack_core.c

Lines changed: 15 additions & 10 deletions
Original file line numberDiff line numberDiff line change
@@ -890,10 +890,8 @@ nf_conntrack_hash_check_insert(struct nf_conn *ct)
890890

891891
zone = nf_ct_zone(ct);
892892

893-
if (!nf_ct_ext_valid_pre(ct->ext)) {
894-
NF_CT_STAT_INC_ATOMIC(net, insert_failed);
895-
return -ETIMEDOUT;
896-
}
893+
if (!nf_ct_ext_valid_pre(ct->ext))
894+
return -EAGAIN;
897895

898896
local_bh_disable();
899897
do {
@@ -928,6 +926,19 @@ nf_conntrack_hash_check_insert(struct nf_conn *ct)
928926
goto chaintoolong;
929927
}
930928

929+
/* If genid has changed, we can't insert anymore because ct
930+
* extensions could have stale pointers and nf_ct_iterate_destroy
931+
* might have completed its table scan already.
932+
*
933+
* Increment of the ext genid right after this check is fine:
934+
* nf_ct_iterate_destroy blocks until locks are released.
935+
*/
936+
if (!nf_ct_ext_valid_post(ct->ext)) {
937+
err = -EAGAIN;
938+
goto out;
939+
}
940+
941+
ct->status |= IPS_CONFIRMED;
931942
smp_wmb();
932943
/* The caller holds a reference to this object */
933944
refcount_set(&ct->ct_general.use, 2);
@@ -936,12 +947,6 @@ nf_conntrack_hash_check_insert(struct nf_conn *ct)
936947
NF_CT_STAT_INC(net, insert);
937948
local_bh_enable();
938949

939-
if (!nf_ct_ext_valid_post(ct->ext)) {
940-
nf_ct_kill(ct);
941-
NF_CT_STAT_INC_ATOMIC(net, drop);
942-
return -ETIMEDOUT;
943-
}
944-
945950
return 0;
946951
chaintoolong:
947952
NF_CT_STAT_INC(net, chaintoolong);

net/netfilter/nf_conntrack_netlink.c

Lines changed: 0 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -2316,9 +2316,6 @@ ctnetlink_create_conntrack(struct net *net,
23162316
nfct_seqadj_ext_add(ct);
23172317
nfct_synproxy_ext_add(ct);
23182318

2319-
/* we must add conntrack extensions before confirmation. */
2320-
ct->status |= IPS_CONFIRMED;
2321-
23222319
if (cda[CTA_STATUS]) {
23232320
err = ctnetlink_change_status(ct, cda);
23242321
if (err < 0)

0 commit comments

Comments
 (0)