From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1753231AbaFATcq (ORCPT ); Sun, 1 Jun 2014 15:32:46 -0400 Received: from casper.infradead.org ([85.118.1.10]:33427 "EHLO casper.infradead.org" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1753138AbaFATcl (ORCPT ); Sun, 1 Jun 2014 15:32:41 -0400 Date: Sun, 1 Jun 2014 21:32:40 +0200 From: Peter Zijlstra To: Hugh Dickins Cc: Dave Jones , Andrew Morton , linux-kernel@vger.kernel.org, linux-mm@kvack.org Subject: Re: sleeping function warning from __put_anon_vma Message-ID: <20140601193240.GF16155@laptop.programming.kicks-ass.net> References: <20140530000944.GA29942@redhat.com> MIME-Version: 1.0 Content-Type: text/plain; charset=us-ascii Content-Disposition: inline In-Reply-To: User-Agent: Mutt/1.5.21 (2012-12-30) Sender: linux-kernel-owner@vger.kernel.org List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On Sat, May 31, 2014 at 01:33:13PM -0700, Hugh Dickins wrote: > [PATCH] mm: fix sleeping function warning from __put_anon_vma > > Trinity reports BUG: > sleeping function called from invalid context at kernel/locking/rwsem.c:47 > in_atomic(): 0, irqs_disabled(): 0, pid: 5787, name: trinity-c27 > __might_sleep < down_write < __put_anon_vma < page_get_anon_vma < > migrate_pages < compact_zone < compact_zone_order < try_to_compact_pages .. > > Right, since conversion to mutex then rwsem, we should not put_anon_vma() > from inside an rcu_read_lock()ed section: fix the two places that did so. > > Fixes: 88c22088bf23 ("mm: optimize page_lock_anon_vma() fast-path") > Reported-by: Dave Jones > Signed-off-by: Hugh Dickins > Needs-Ack-from: Peter Zijlstra > --- > > mm/rmap.c | 8 ++++++-- > 1 file changed, 6 insertions(+), 2 deletions(-) > > --- 3.15-rc7/mm/rmap.c 2014-04-13 17:24:36.680507189 -0700 > +++ linux/mm/rmap.c 2014-05-31 12:02:08.496088637 -0700 > @@ -426,12 +426,14 @@ struct anon_vma *page_get_anon_vma(struc > * above cannot corrupt). > */ > if (!page_mapped(page)) { > + rcu_read_unlock(); > put_anon_vma(anon_vma); > anon_vma = NULL; > + goto outer; > } > out: > rcu_read_unlock(); > - > +outer: > return anon_vma; > } I think we can do that without the goto if we write something like: if (!page_mapped(page)) { rcu_read_unlock(); put_anon_vma(anon_vma); return NULL; } > @@ -477,9 +479,10 @@ struct anon_vma *page_lock_anon_vma_read > } > > if (!page_mapped(page)) { > + rcu_read_unlock(); > put_anon_vma(anon_vma); > anon_vma = NULL; > - goto out; > + goto outer; > } > > /* we pinned the anon_vma, its safe to sleep */ > @@ -501,6 +504,7 @@ struct anon_vma *page_lock_anon_vma_read > > out: > rcu_read_unlock(); > +outer: > return anon_vma; > } Same here too, I suppose. Interesting that we never managed to hit this one; it might also make sense to put a might_sleep() in anon_vma_free(). Other than that, I don't see anything really odd, then again, its sunday evening and my thinking cap isn't exactly on proper.