From mboxrd@z Thu Jan 1 00:00:00 1970 From: Jacek Luczak Subject: Re: [PATCH] SCTP: fix race between sctp_bind_addr_free() and sctp_bind_addr_conflict() Date: Wed, 18 May 2011 15:11:47 +0200 Message-ID: References: <1305704885.2983.4.camel@edumazet-laptop> <1305707358.2983.14.camel@edumazet-laptop> <4DD38B30.9090601@cn.fujitsu.com> <4DD3BC8F.9050802@hp.com> <1305723046.2991.19.camel@edumazet-laptop> Mime-Version: 1.0 Content-Type: text/plain; charset=ISO-8859-1 Content-Transfer-Encoding: QUOTED-PRINTABLE Cc: Vladislav Yasevich , Wei Yongjun , netdev@vger.kernel.org To: Eric Dumazet Return-path: Received: from mail-pv0-f174.google.com ([74.125.83.174]:34865 "EHLO mail-pv0-f174.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1756948Ab1ERNLr convert rfc822-to-8bit (ORCPT ); Wed, 18 May 2011 09:11:47 -0400 Received: by pvg12 with SMTP id 12so695670pvg.19 for ; Wed, 18 May 2011 06:11:47 -0700 (PDT) In-Reply-To: <1305723046.2991.19.camel@edumazet-laptop> Sender: netdev-owner@vger.kernel.org List-ID: 2011/5/18 Eric Dumazet : > Le mercredi 18 mai 2011 =E0 14:47 +0200, Jacek Luczak a =E9crit : > >> OK then, at the end what Eric suggested is IMO valid: >> >> diff --git a/net/sctp/bind_addr.c b/net/sctp/bind_addr.c >> index faf71d1..0025d90 100644 >> --- a/net/sctp/bind_addr.c >> +++ b/net/sctp/bind_addr.c >> @@ -144,10 +144,9 @@ static void sctp_bind_addr_clean(struct sctp_bi= nd_addr *bp) >> =A0 =A0 =A0 =A0 struct list_head *pos, *temp; >> >> =A0 =A0 =A0 =A0 /* Empty the bind address list. */ >> - =A0 =A0 =A0 list_for_each_safe(pos, temp, &bp->address_list) { >> - =A0 =A0 =A0 =A0 =A0 =A0 =A0 addr =3D list_entry(pos, struct sctp_s= ockaddr_entry, list); >> - =A0 =A0 =A0 =A0 =A0 =A0 =A0 list_del(pos); >> - =A0 =A0 =A0 =A0 =A0 =A0 =A0 kfree(addr); >> + =A0 =A0 =A0 list_for_each_entry(pos, &bp->address_list, list) { > > a 'safe' version is needed here, since we remove items in iterator. Yep. diff --git a/net/sctp/bind_addr.c b/net/sctp/bind_addr.c index faf71d1..6150ac5 100644 --- a/net/sctp/bind_addr.c +++ b/net/sctp/bind_addr.c @@ -140,14 +140,12 @@ void sctp_bind_addr_init(struct sctp_bind_addr *bp, __u16 port) /* Dispose of the address list. */ static void sctp_bind_addr_clean(struct sctp_bind_addr *bp) { - struct sctp_sockaddr_entry *addr; - struct list_head *pos, *temp; + struct sctp_sockaddr_entry *addr, *temp; /* Empty the bind address list. */ - list_for_each_safe(pos, temp, &bp->address_list) { - addr =3D list_entry(pos, struct sctp_sockaddr_entry, li= st); - list_del(pos); - kfree(addr); + list_for_each_entry_safe(addr, temp, &bp->address_list, list) { + list_del_rcu(&addr->list); + call_rcu(&addr->rcu, sctp_local_addr_free); SCTP_DBG_OBJCNT_DEC(addr); } } Does it now look good? -Jacek