Skip to content

Commit e6d57e9

Browse files
Florian Westphalummakynes
authored andcommitted
netfilter: conntrack: fix rmmod double-free race
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>
1 parent ac48939 commit e6d57e9

File tree

3 files changed

+15
-14
lines changed

3 files changed

+15
-14
lines changed

net/netfilter/nf_conntrack_bpf.c

Lines changed: 0 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -381,7 +381,6 @@ struct nf_conn *bpf_ct_insert_entry(struct nf_conn___init *nfct_i)
381381
struct nf_conn *nfct = (struct nf_conn *)nfct_i;
382382
int err;
383383

384-
nfct->status |= IPS_CONFIRMED;
385384
err = nf_conntrack_hash_check_insert(nfct);
386385
if (err < 0) {
387386
nf_conntrack_free(nfct);

net/netfilter/nf_conntrack_core.c

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

887887
zone = nf_ct_zone(ct);
888888

889-
if (!nf_ct_ext_valid_pre(ct->ext)) {
890-
NF_CT_STAT_INC_ATOMIC(net, insert_failed);
891-
return -ETIMEDOUT;
892-
}
889+
if (!nf_ct_ext_valid_pre(ct->ext))
890+
return -EAGAIN;
893891

894892
local_bh_disable();
895893
do {
@@ -924,6 +922,19 @@ nf_conntrack_hash_check_insert(struct nf_conn *ct)
924922
goto chaintoolong;
925923
}
926924

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

935-
if (!nf_ct_ext_valid_post(ct->ext)) {
936-
nf_ct_kill(ct);
937-
NF_CT_STAT_INC_ATOMIC(net, drop);
938-
return -ETIMEDOUT;
939-
}
940-
941946
return 0;
942947
chaintoolong:
943948
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)