Return-Path: Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1753153AbYJ0LFI (ORCPT ); Mon, 27 Oct 2008 07:05:08 -0400 Received: (majordomo@vger.kernel.org) by vger.kernel.org id S1752334AbYJ0LE5 (ORCPT ); Mon, 27 Oct 2008 07:04:57 -0400 Received: from ecfrec.frec.bull.fr ([129.183.4.8]:59609 "EHLO ecfrec.frec.bull.fr" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1752251AbYJ0LE4 (ORCPT ); Mon, 27 Oct 2008 07:04:56 -0400 Subject: Re: [PATCH 1/1] (v3) SYSVIPC - Fix the ipc structures initialization From: Nadia Derbey To: cboulte@gmail.com Cc: akpm@linux-foundation.org, manfred@colorfullife.com, linux-kernel@vger.kernel.org In-Reply-To: <4f3ee3290810270332p56e1f99q24a5eea818e626ee@mail.gmail.com> References: <20081027072826.266777838@bull.net> <20081027072836.438113395@bull.net> <4f3ee3290810270332p56e1f99q24a5eea818e626ee@mail.gmail.com> Content-Type: text/plain Date: Mon, 27 Oct 2008 12:04:09 +0100 Message-Id: <1225105450.2595.215.camel@frecb000730.frec.bull.fr> Mime-Version: 1.0 X-Mailer: Evolution 2.22.3.1 (2.22.3.1-1.fc9) Content-Transfer-Encoding: 7bit Sender: linux-kernel-owner@vger.kernel.org List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Content-Length: 3695 Lines: 111 On Mon, 2008-10-27 at 11:32 +0100, cboulte@gmail.com wrote: > On Mon, Oct 27, 2008 at 8:28 AM, wrote: > > > > This patch is a fix for Bugzilla bug > > http://bugzilla.kernel.org/show_bug.cgi?id=11796. > > > > To summarize, a simple testcase is concurrently running an infinite loop on > > msgctl(IPC_STAT) and a call to msgget(): > > > > while (1) > > msgctl(id, IPC_STAT) 1 > > | > > | > > | > > 2 id = msgget(key, IPC_CREAT) > > | > > | > > | > > > > In the interval [1-2], the id doesn't exist yet. > > > > In that test, the problem is the following: > > When we are calling ipc_addid() from msgget() the msq structure is not > > completely initialized. So idr_get_new() is inserting a pointer into the > > idr tree, and the structure which is pointed to has, among other fields, > > its lock uninitialized. > > > > Since msgctl(IPC_STAT) is looping while (1), idr_find() returns the > > pointer as soon as it is inserted into the IDR tree. And ipc_lock() > > calls spin_lock(&mqs->lock), while we have not initialized that lock > > yet. > > > > This patch moves the spin_lock_init() before the call to ipc_addid(). > > It also sets the "deleted" flag to 1 in the window between msg structure > > allocation and msg structure locking in ipc_addid(). > > > > > > Regards, > > Nadia > > > > > > Signed-off-by: Nadia Derbey > > > > --- > > ipc/util.c | 16 ++++++++++++++-- > > 1 file changed, 14 insertions(+), 2 deletions(-) > > > > Index: linux-2.6.27/ipc/util.c > > =================================================================== > > --- linux-2.6.27.orig/ipc/util.c 2008-10-23 15:20:46.000000000 +0200 > > +++ linux-2.6.27/ipc/util.c 2008-10-24 17:48:33.000000000 +0200 > > @@ -266,6 +266,17 @@ int ipc_addid(struct ipc_ids* ids, struc > > if (ids->in_use >= size) > > return -ENOSPC; > > > > + spin_lock_init(&new->lock); > > + > > + /* > > + * We have a window between the time new is inserted into the idr > > + * tree and the time it is actually locked. > > + * In order to be safe during that window set the new ipc structure > > + * as deleted: a concurrent ipc_lock() will see it as not present > > + * until the initialization phase is complete. > > + */ > > + new->deleted = 1; > > + > > err = idr_get_new(&ids->ipcs_idr, new, &id); > > if (err) > > return err; > > @@ -280,10 +291,11 @@ int ipc_addid(struct ipc_ids* ids, struc > > ids->seq = 0; > > > > new->id = ipc_buildid(id, new->seq); > > - spin_lock_init(&new->lock); > > - new->deleted = 0; > > rcu_read_lock(); > > spin_lock(&new->lock); > > + > > + new->deleted = 0; > > + > > return id; > > } > > > > > > -- > > > > Still got the lock... I'm using a 4 cpus node: Intel Xeon @ 2.8GHz... > don't know if it has an impact. ??? The bad new, is that it becomes unreprodicible on my side. For my part, I've got 2 2.8 GHz Xeon CPUs. Will review the code once more. Thanks! Nadia > The only way I found to have no lock, it's to spin_lock the ipc > _before_ inserting it into the idr. > > Best regards, c. > -- Nadia Derbey -- To unsubscribe from this list: send the line "unsubscribe linux-kernel" in the body of a message to majordomo@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html Please read the FAQ at http://www.tux.org/lkml/