mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 02:30:34 +02:00 
			
		
		
		
	mm, compaction: restrict async compaction to pageblocks of same migratetype
The migrate scanner in async compaction is currently limited to MIGRATE_MOVABLE pageblocks. This is a heuristic intended to reduce latency, based on the assumption that non-MOVABLE pageblocks are unlikely to contain movable pages. However, with the exception of THP's, most high-order allocations are not movable. Should the async compaction succeed, this increases the chance that the non-MOVABLE allocations will fallback to a MOVABLE pageblock, making the long-term fragmentation worse. This patch attempts to help the situation by changing async direct compaction so that the migrate scanner only scans the pageblocks of the requested migratetype. If it's a non-MOVABLE type and there are such pageblocks that do contain movable pages, chances are that the allocation can succeed within one of such pageblocks, removing the need for a fallback. If that fails, the subsequent sync attempt will ignore this restriction. In testing based on 4.9 kernel with stress-highalloc from mmtests configured for order-4 GFP_KERNEL allocations, this patch has reduced the number of unmovable allocations falling back to movable pageblocks by 30%. The number of movable allocations falling back is reduced by 12%. Link: http://lkml.kernel.org/r/20170307131545.28577-8-vbabka@suse.cz Signed-off-by: Vlastimil Babka <vbabka@suse.cz> Cc: Mel Gorman <mgorman@techsingularity.net> Cc: Johannes Weiner <hannes@cmpxchg.org> Cc: Joonsoo Kim <iamjoonsoo.kim@lge.com> Cc: David Rientjes <rientjes@google.com> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
This commit is contained in:
		
							parent
							
								
									d39773a062
								
							
						
					
					
						commit
						282722b0d2
					
				
					 2 changed files with 22 additions and 9 deletions
				
			
		| 
						 | 
				
			
			@ -986,10 +986,17 @@ isolate_migratepages_range(struct compact_control *cc, unsigned long start_pfn,
 | 
			
		|||
static bool suitable_migration_source(struct compact_control *cc,
 | 
			
		||||
							struct page *page)
 | 
			
		||||
{
 | 
			
		||||
	if (cc->mode != MIGRATE_ASYNC)
 | 
			
		||||
	int block_mt;
 | 
			
		||||
 | 
			
		||||
	if ((cc->mode != MIGRATE_ASYNC) || !cc->direct_compaction)
 | 
			
		||||
		return true;
 | 
			
		||||
 | 
			
		||||
	return is_migrate_movable(get_pageblock_migratetype(page));
 | 
			
		||||
	block_mt = get_pageblock_migratetype(page);
 | 
			
		||||
 | 
			
		||||
	if (cc->migratetype == MIGRATE_MOVABLE)
 | 
			
		||||
		return is_migrate_movable(block_mt);
 | 
			
		||||
	else
 | 
			
		||||
		return block_mt == cc->migratetype;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* Returns true if the page is within a block suitable for migration to */
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -3665,6 +3665,7 @@ __alloc_pages_slowpath(gfp_t gfp_mask, unsigned int order,
 | 
			
		|||
						struct alloc_context *ac)
 | 
			
		||||
{
 | 
			
		||||
	bool can_direct_reclaim = gfp_mask & __GFP_DIRECT_RECLAIM;
 | 
			
		||||
	const bool costly_order = order > PAGE_ALLOC_COSTLY_ORDER;
 | 
			
		||||
	struct page *page = NULL;
 | 
			
		||||
	unsigned int alloc_flags;
 | 
			
		||||
	unsigned long did_some_progress;
 | 
			
		||||
| 
						 | 
				
			
			@ -3732,12 +3733,17 @@ __alloc_pages_slowpath(gfp_t gfp_mask, unsigned int order,
 | 
			
		|||
 | 
			
		||||
	/*
 | 
			
		||||
	 * For costly allocations, try direct compaction first, as it's likely
 | 
			
		||||
	 * that we have enough base pages and don't need to reclaim. Don't try
 | 
			
		||||
	 * that for allocations that are allowed to ignore watermarks, as the
 | 
			
		||||
	 * ALLOC_NO_WATERMARKS attempt didn't yet happen.
 | 
			
		||||
	 * that we have enough base pages and don't need to reclaim. For non-
 | 
			
		||||
	 * movable high-order allocations, do that as well, as compaction will
 | 
			
		||||
	 * try prevent permanent fragmentation by migrating from blocks of the
 | 
			
		||||
	 * same migratetype.
 | 
			
		||||
	 * Don't try this for allocations that are allowed to ignore
 | 
			
		||||
	 * watermarks, as the ALLOC_NO_WATERMARKS attempt didn't yet happen.
 | 
			
		||||
	 */
 | 
			
		||||
	if (can_direct_reclaim && order > PAGE_ALLOC_COSTLY_ORDER &&
 | 
			
		||||
		!gfp_pfmemalloc_allowed(gfp_mask)) {
 | 
			
		||||
	if (can_direct_reclaim &&
 | 
			
		||||
			(costly_order ||
 | 
			
		||||
			   (order > 0 && ac->migratetype != MIGRATE_MOVABLE))
 | 
			
		||||
			&& !gfp_pfmemalloc_allowed(gfp_mask)) {
 | 
			
		||||
		page = __alloc_pages_direct_compact(gfp_mask, order,
 | 
			
		||||
						alloc_flags, ac,
 | 
			
		||||
						INIT_COMPACT_PRIORITY,
 | 
			
		||||
| 
						 | 
				
			
			@ -3749,7 +3755,7 @@ __alloc_pages_slowpath(gfp_t gfp_mask, unsigned int order,
 | 
			
		|||
		 * Checks for costly allocations with __GFP_NORETRY, which
 | 
			
		||||
		 * includes THP page fault allocations
 | 
			
		||||
		 */
 | 
			
		||||
		if (gfp_mask & __GFP_NORETRY) {
 | 
			
		||||
		if (costly_order && (gfp_mask & __GFP_NORETRY)) {
 | 
			
		||||
			/*
 | 
			
		||||
			 * If compaction is deferred for high-order allocations,
 | 
			
		||||
			 * it is because sync compaction recently failed. If
 | 
			
		||||
| 
						 | 
				
			
			@ -3830,7 +3836,7 @@ __alloc_pages_slowpath(gfp_t gfp_mask, unsigned int order,
 | 
			
		|||
	 * Do not retry costly high order allocations unless they are
 | 
			
		||||
	 * __GFP_REPEAT
 | 
			
		||||
	 */
 | 
			
		||||
	if (order > PAGE_ALLOC_COSTLY_ORDER && !(gfp_mask & __GFP_REPEAT))
 | 
			
		||||
	if (costly_order && !(gfp_mask & __GFP_REPEAT))
 | 
			
		||||
		goto nopage;
 | 
			
		||||
 | 
			
		||||
	if (should_reclaim_retry(gfp_mask, order, ac, alloc_flags,
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Loading…
	
		Reference in a new issue