aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorHerbert Xu <herbert@gondor.apana.org.au>2005-04-16 18:24:10 -0400
committerLinus Torvalds <torvalds@ppc970.osdl.org>2005-04-16 18:24:10 -0400
commit6775cab98b89b2caa10dce4b07e2c81999e45517 (patch)
treec24ede63d9f2eb303744d5c96c8f5294cee3c1db
parent2a27805127aee1e7e62854bcf9ca8c355c23b73e (diff)
[PATCH] Fix dst_destroy() race
When we are not the real parent of the dst (e.g., when we're xfrm_dst and the child is an rtentry), it may already be on the GC list. In fact the current code is buggy to, we need to check dst->flags before the dec as dst may no longer be valid afterwards. Signed-off-by: Herbert Xu <herbert@gondor.apana.org.au> Signed-off-by: David S. Miller <davem@davemloft.net> Signed-off-by: Andrew Morton <akpm@osdl.org> Signed-off-by: Linus Torvalds <torvalds@osdl.org>
-rw-r--r--net/core/dst.c6
1 files changed, 4 insertions, 2 deletions
diff --git a/net/core/dst.c b/net/core/dst.c
index 3bf6cc434814..fc434ade5270 100644
--- a/net/core/dst.c
+++ b/net/core/dst.c
@@ -198,13 +198,15 @@ again:
198 198
199 dst = child; 199 dst = child;
200 if (dst) { 200 if (dst) {
201 int nohash = dst->flags & DST_NOHASH;
202
201 if (atomic_dec_and_test(&dst->__refcnt)) { 203 if (atomic_dec_and_test(&dst->__refcnt)) {
202 /* We were real parent of this dst, so kill child. */ 204 /* We were real parent of this dst, so kill child. */
203 if (dst->flags&DST_NOHASH) 205 if (nohash)
204 goto again; 206 goto again;
205 } else { 207 } else {
206 /* Child is still referenced, return it for freeing. */ 208 /* Child is still referenced, return it for freeing. */
207 if (dst->flags&DST_NOHASH) 209 if (nohash)
208 return dst; 210 return dst;
209 /* Child is still in his hash table */ 211 /* Child is still in his hash table */
210 } 212 }