Return-Path: Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1755199AbZDNMZb (ORCPT ); Tue, 14 Apr 2009 08:25:31 -0400 Received: (majordomo@vger.kernel.org) by vger.kernel.org id S1754036AbZDNMZU (ORCPT ); Tue, 14 Apr 2009 08:25:20 -0400 Received: from smtp120.mail.mud.yahoo.com ([209.191.84.77]:25453 "HELO smtp120.mail.mud.yahoo.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with SMTP id S1753988AbZDNMZT (ORCPT ); Tue, 14 Apr 2009 08:25:19 -0400 DomainKey-Signature: a=rsa-sha1; q=dns; c=nofws; s=s1024; d=yahoo.com.au; h=Received:X-YMail-OSG:X-Yahoo-Newman-Property:From:To:Subject:Date:User-Agent:Cc:References:In-Reply-To:MIME-Version:Content-Type:Content-Transfer-Encoding:Content-Disposition:Message-Id; b=b/S+G9aAzCt8uqmBltc3vJQNA3716plEVeTsxRMkQPodp1JDZsJKRfEXlnuM30McvDj946zAYr9ODUEvDLvi9kafDX8MHz1RkU9NH01sfKqnReG0sOWUhRHD8uP3ulnYn5dpZd/w3trdKAEaRWkzbDT1+1qC59kw4pGZ2hmaB9w= ; X-YMail-OSG: ROSHoFMVM1kz91BdgKMnqnwloE6JHMUd0_2sqd8XuNtEu8_pZ58FNpwqjdeMYDfOXvXWfjQptd97PvGB2hLuOaIya4.3LpiPaba.j6IGNjf6OoELLnDXRGte3uSrq4nB__C7hcrOGylMccWSuJbFgsDakbsEUhjirbcLjFJoE003B3_5udzlK9FCI8yCMAawQS5NM6YAk9usgTHnyKF_zj4oFJtK7fI_ee7VCXyIZjoLKGr.sHemPOUhFEntx07nllJdzZT9z1QiLUFhihuR8tcvU93M79Oz9qm5A4q8s6AdTLEJwIWRvLa620eSk_ZqDzhZgjgyWCNCv5gC.EsvRDSs X-Yahoo-Newman-Property: ymail-3 From: Nick Piggin To: KOSAKI Motohiro Subject: Re: [RFC][PATCH v3 1/6] mm: Don't unmap gup()ed page Date: Tue, 14 Apr 2009 22:25:09 +1000 User-Agent: KMail/1.9.51 (KDE/4.0.4; ; ) Cc: LKML , Linus Torvalds , Andrew Morton , Andrea Arcangeli , Jeff Moyer , linux-mm@kvack.org, linux-fsdevel@vger.kernel.org, Hugh Dickins References: <20090414151204.C647.A69D9226@jp.fujitsu.com> <200904141925.46012.nickpiggin@yahoo.com.au> <2f11576a0904140502h295faf33qcea9a39ff7f230a5@mail.gmail.com> In-Reply-To: <2f11576a0904140502h295faf33qcea9a39ff7f230a5@mail.gmail.com> MIME-Version: 1.0 Content-Type: text/plain; charset="iso-8859-1" Content-Transfer-Encoding: 7bit Content-Disposition: inline Message-Id: <200904142225.10788.nickpiggin@yahoo.com.au> Sender: linux-kernel-owner@vger.kernel.org List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Content-Length: 4028 Lines: 94 On Tuesday 14 April 2009 22:02:47 KOSAKI Motohiro wrote: > > On Tuesday 14 April 2009 16:16:52 KOSAKI Motohiro wrote: > >> Signed-off-by: KOSAKI Motohiro > >> @@ -547,7 +549,13 @@ int reuse_swap_page(struct page *page) > >> SetPageDirty(page); > >> } > >> } > >> - return count == 1; > >> + > >> + /* > >> + * If we can re-use the swap page _and_ the end > >> + * result has only one user (the mapping), then > >> + * we reuse the whole page > >> + */ > >> + return count + page_count(page) == 2; > >> } > > > > I guess this patch does work to close the read-side race, but I slightly don't > > like using page_count for things like this. page_count can be temporarily > > raised for reasons other than access through their user mapping. Swapcache, > > page reclaim, LRU pagevecs, concurrent do_wp_page, etc. > > Yes, that's trade-off. > your early decow also can misjudge and make unnecessary copy. Yes indeed it can. Although it would only ever do so in case of pages that have had get_user_pages run against them previously, and not from random interactions from any other parts of the kernel. I would be interested, using an anon vma field as you say for keeping a gup count... it could potentially be used to avoid the extra copy. But hmm, I don't have much time to go down that path so long as the basic concept of my proposal is in question. > >> /* > >> + * Don't pull an anonymous page out from under get_user_pages. > >> + * GUP carefully breaks COW and raises page count (while holding > >> + * pte_lock, as we have here) to make sure that the page > >> + * cannot be freed. If we unmap that page here, a user write > >> + * access to the virtual address will bring back the page, but > >> + * its raised count will (ironically) be taken to mean it's not > >> + * an exclusive swap page, do_wp_page will replace it by a copy > >> + * page, and the user never get to see the data GUP was holding > >> + * the original page for. > >> + * > >> + * This test is also useful for when swapoff (unuse_process) has > >> + * to drop page lock: its reference to the page stops existing > >> + * ptes from being unmapped, so swapoff can make progress. > >> + */ > >> + if (PageSwapCache(page) && > >> + page_count(page) != page_mapcount(page) + 2) { > >> + ret = SWAP_FAIL; > >> + goto out_unmap; > >> + } > > > > I guess it does add another constraint to the VM, ie. not allowed to > > unmap an anonymous page with elevated refcount. Maybe not a big deal > > now, but I think it is enough that it should be noted. If you squint, > > this could actually be more complex/intrusive to the wider VM than my > > copy on fork (which is basically exactly like a manual do_wp_page at > > fork time). > > I agree this code effect widely kernel activity. > but actually, in past days, the kernel did the same behavior. then > almost core code is > page_count checking safe. > > but Yes, we need to afraid newer code don't works with this code... > > > > And.... I don't think this is safe against a concurrent gup_fast() > > (which helps my point). > > Could you please explain more detail ? > + if (PageSwapCache(page) && + page_count(page) != page_mapcount(page) + 2) { + ret = SWAP_FAIL; + goto out_unmap; + } Now if another thread does a get_user_pages_fast after it passes this check, it can take a gup reference to the page which is now about to be unmapped. Then after it is unmapped, if a wp fault is caused on the page, then it will not be reused and thus you lose data as explained in your big comment. -- 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/