On Wed, Apr 24, 2013 at 10:26:50AM -0400, Theodore Ts'o wrote: > On Tue, Apr 23, 2013 at 03:00:08PM -0700, Andrew Morton wrote: > > That should fix things for now. Although it might be better to just do > > > > mark_page_accessed(page); /* to SetPageReferenced */ > > lru_add_drain(); /* to SetPageLRU */ > > > > Because a) this was too early to decide that the page is > > super-important and b) the second touch of this page should have a > > mark_page_accessed() in it already. > > The question is do we really want to put lru_add_drain() into the ext4 > file system code? That seems to pushing some fairly mm-specific > knowledge into file system code. I'll do this if I have to do, but > wouldn't be better if this was pushed into mark_page_accessed(), or > some other new API was exported by the mm subsystem? > I don't think we want to push lru_add_drain() into the ext4 code. It's too specific of knowledge just to work around pagevecs. Before we rework how pagevecs select what LRU to place a page, can we make sure that fixing that will fix the problem? Andrew, can you try the following patch please? Also, is there any chance you can describe in more detail what the workload does? If it fails to boot, remove the second that calls lru_add_drain_all() and try again. The patch looks deceptively simple, a downside from is is that workloads that call mark_page_accessed() frequently will contend more on the zone->lru_lock than it did previously. Moving lru_add_drain() to the ext4 could would suffer the same contention problem. Thanks. ---8<--- mm: pagevec: Move inactive pages to active lists even if on a pagevec If a page is on a pagevec aimed at the inactive list then two subsequent calls to mark_page_acessed() will still not move it to the active list. This can cause a page to be reclaimed sooner than is expected. This patch detects if an inactive page is not on the LRU and drains the pagevec before promoting it. Not-signed-off diff --git a/mm/swap.c b/mm/swap.c index 8a529a0..eac64fe 100644 --- a/mm/swap.c +++ b/mm/swap.c @@ -437,7 +437,18 @@ void activate_page(struct page *page) void mark_page_accessed(struct page *page) { if (!PageActive(page) && !PageUnevictable(page) && - PageReferenced(page) && PageLRU(page)) { + PageReferenced(page)) { + /* Page could be in pagevec */ + if (!PageLRU(page)) + lru_add_drain(); + + /* + * Weeeee, using in_atomic() like this is a hand-grenade. + * Patch is for debugging purposes only, do not merge this. + */ + if (!PageLRU(page) && !in_atomic()) + lru_add_drain_all(); + activate_page(page); ClearPageReferenced(page); } else if (!PageReferenced(page)) { -- To unsubscribe, send a message with 'unsubscribe linux-mm' in the body to majordomo@xxxxxxxxx. For more info on Linux MM, see: http://www.linux-mm.org/ . Don't email: <a href=mailto:"dont@xxxxxxxxx"> email@xxxxxxxxx </a>