diff options
author | Patrick McHardy <kaber@trash.net> | 2007-02-12 14:14:11 -0500 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2007-02-12 14:14:11 -0500 |
commit | 982d9a9ce389c396bc83ce29d799937f379ddcb7 (patch) | |
tree | 5f470babe5eafa37aac006bc87a1fab31db80114 | |
parent | 6b48a7d08d1bb2e3932bce1662fe411304acc18f (diff) |
[NETFILTER]: nf_conntrack: properly use RCU for nf_conntrack_destroyed callback
Signed-off-by: Patrick McHardy <kaber@trash.net>
Signed-off-by: David S. Miller <davem@davemloft.net>
-rw-r--r-- | net/ipv4/netfilter/nf_nat_core.c | 7 | ||||
-rw-r--r-- | net/netfilter/nf_conntrack_core.c | 9 |
2 files changed, 10 insertions, 6 deletions
diff --git a/net/ipv4/netfilter/nf_nat_core.c b/net/ipv4/netfilter/nf_nat_core.c index 5156d5d6c3b8..2c01378d3592 100644 --- a/net/ipv4/netfilter/nf_nat_core.c +++ b/net/ipv4/netfilter/nf_nat_core.c | |||
@@ -617,8 +617,8 @@ static int __init nf_nat_init(void) | |||
617 | } | 617 | } |
618 | 618 | ||
619 | /* FIXME: Man, this is a hack. <SIGH> */ | 619 | /* FIXME: Man, this is a hack. <SIGH> */ |
620 | NF_CT_ASSERT(nf_conntrack_destroyed == NULL); | 620 | NF_CT_ASSERT(rcu_dereference(nf_conntrack_destroyed) == NULL); |
621 | nf_conntrack_destroyed = &nf_nat_cleanup_conntrack; | 621 | rcu_assign_pointer(nf_conntrack_destroyed, nf_nat_cleanup_conntrack); |
622 | 622 | ||
623 | /* Initialize fake conntrack so that NAT will skip it */ | 623 | /* Initialize fake conntrack so that NAT will skip it */ |
624 | nf_conntrack_untracked.status |= IPS_NAT_DONE_MASK; | 624 | nf_conntrack_untracked.status |= IPS_NAT_DONE_MASK; |
@@ -642,7 +642,8 @@ static int clean_nat(struct nf_conn *i, void *data) | |||
642 | static void __exit nf_nat_cleanup(void) | 642 | static void __exit nf_nat_cleanup(void) |
643 | { | 643 | { |
644 | nf_ct_iterate_cleanup(&clean_nat, NULL); | 644 | nf_ct_iterate_cleanup(&clean_nat, NULL); |
645 | nf_conntrack_destroyed = NULL; | 645 | rcu_assign_pointer(nf_conntrack_destroyed, NULL); |
646 | synchronize_rcu(); | ||
646 | vfree(bysource); | 647 | vfree(bysource); |
647 | nf_ct_l3proto_put(l3proto); | 648 | nf_ct_l3proto_put(l3proto); |
648 | } | 649 | } |
diff --git a/net/netfilter/nf_conntrack_core.c b/net/netfilter/nf_conntrack_core.c index d59640e2377b..0cc150560fb7 100644 --- a/net/netfilter/nf_conntrack_core.c +++ b/net/netfilter/nf_conntrack_core.c | |||
@@ -318,6 +318,7 @@ destroy_conntrack(struct nf_conntrack *nfct) | |||
318 | struct nf_conn_help *help = nfct_help(ct); | 318 | struct nf_conn_help *help = nfct_help(ct); |
319 | struct nf_conntrack_l3proto *l3proto; | 319 | struct nf_conntrack_l3proto *l3proto; |
320 | struct nf_conntrack_l4proto *l4proto; | 320 | struct nf_conntrack_l4proto *l4proto; |
321 | typeof(nf_conntrack_destroyed) destroyed; | ||
321 | 322 | ||
322 | DEBUGP("destroy_conntrack(%p)\n", ct); | 323 | DEBUGP("destroy_conntrack(%p)\n", ct); |
323 | NF_CT_ASSERT(atomic_read(&nfct->use) == 0); | 324 | NF_CT_ASSERT(atomic_read(&nfct->use) == 0); |
@@ -341,10 +342,12 @@ destroy_conntrack(struct nf_conntrack *nfct) | |||
341 | ct->tuplehash[IP_CT_DIR_REPLY].tuple.dst.protonum); | 342 | ct->tuplehash[IP_CT_DIR_REPLY].tuple.dst.protonum); |
342 | if (l4proto && l4proto->destroy) | 343 | if (l4proto && l4proto->destroy) |
343 | l4proto->destroy(ct); | 344 | l4proto->destroy(ct); |
344 | rcu_read_unlock(); | ||
345 | 345 | ||
346 | if (nf_conntrack_destroyed) | 346 | destroyed = rcu_dereference(nf_conntrack_destroyed); |
347 | nf_conntrack_destroyed(ct); | 347 | if (destroyed) |
348 | destroyed(ct); | ||
349 | |||
350 | rcu_read_unlock(); | ||
348 | 351 | ||
349 | write_lock_bh(&nf_conntrack_lock); | 352 | write_lock_bh(&nf_conntrack_lock); |
350 | /* Expectations will have been removed in clean_from_lists, | 353 | /* Expectations will have been removed in clean_from_lists, |