[PATCH] vmscan: Fix temp_priority race

The temp_priority field in zone is racy, as we can walk through a reclaim
path, and just before we copy it into prev_priority, it can be overwritten
(say with DEF_PRIORITY) by another reclaimer.

The same bug is contained in both try_to_free_pages and balance_pgdat, but
it is fixed slightly differently.  In balance_pgdat, we keep a separate
priority record per zone in a local array.  In try_to_free_pages there is
no need to do this, as the priority level is the same for all zones that we
reclaim from.

Impact of this bug is that temp_priority is copied into prev_priority, and
setting this artificially high causes reclaimers to set distress
artificially low.  They then fail to reclaim mapped pages, when they are,
in fact, under severe memory pressure (their priority may be as low as 0).
This causes the OOM killer to fire incorrectly.

From: Andrew Morton <akpm@osdl.org>

__zone_reclaim() isn't modifying zone->prev_priority.  But zone->prev_priority
is used in the decision whether or not to bring mapped pages onto the inactive
list.  Hence there's a risk here that __zone_reclaim() will fail because
zone->prev_priority ir large (ie: low urgency) and lots of mapped pages end up
stuck on the active list.

Fix that up by decreasing (ie making more urgent) zone->prev_priority as
__zone_reclaim() scans the zone's pages.

This bug perhaps explains why ZONE_RECLAIM_PRIORITY was created.  It should be
possible to remove that now, and to just start out at DEF_PRIORITY?

Cc: Nick Piggin <nickpiggin@yahoo.com.au>
Cc: Christoph Lameter <clameter@engr.sgi.com>
Cc: <stable@kernel.org>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
This commit is contained in:
Martin Bligh 2006-10-28 10:38:24 -07:00 committed by Linus Torvalds
parent 2ae88149a2
commit 3bb1a852ab
4 changed files with 43 additions and 22 deletions

View File

@ -218,13 +218,9 @@ struct zone {
* under - it drives the swappiness decision: whether to unmap mapped * under - it drives the swappiness decision: whether to unmap mapped
* pages. * pages.
* *
* temp_priority is used to remember the scanning priority at which * Access to both this field is quite racy even on uniprocessor. But
* this zone was successfully refilled to free_pages == pages_high.
*
* Access to both these fields is quite racy even on uniprocessor. But
* it is expected to average out OK. * it is expected to average out OK.
*/ */
int temp_priority;
int prev_priority; int prev_priority;

View File

@ -2407,7 +2407,7 @@ static void __meminit free_area_init_core(struct pglist_data *pgdat,
zone->zone_pgdat = pgdat; zone->zone_pgdat = pgdat;
zone->free_pages = 0; zone->free_pages = 0;
zone->temp_priority = zone->prev_priority = DEF_PRIORITY; zone->prev_priority = DEF_PRIORITY;
zone_pcp_init(zone); zone_pcp_init(zone);
INIT_LIST_HEAD(&zone->active_list); INIT_LIST_HEAD(&zone->active_list);

View File

@ -723,6 +723,20 @@ done:
return nr_reclaimed; return nr_reclaimed;
} }
/*
* We are about to scan this zone at a certain priority level. If that priority
* level is smaller (ie: more urgent) than the previous priority, then note
* that priority level within the zone. This is done so that when the next
* process comes in to scan this zone, it will immediately start out at this
* priority level rather than having to build up its own scanning priority.
* Here, this priority affects only the reclaim-mapped threshold.
*/
static inline void note_zone_scanning_priority(struct zone *zone, int priority)
{
if (priority < zone->prev_priority)
zone->prev_priority = priority;
}
static inline int zone_is_near_oom(struct zone *zone) static inline int zone_is_near_oom(struct zone *zone)
{ {
return zone->pages_scanned >= (zone->nr_active + zone->nr_inactive)*3; return zone->pages_scanned >= (zone->nr_active + zone->nr_inactive)*3;
@ -972,9 +986,7 @@ static unsigned long shrink_zones(int priority, struct zone **zones,
if (!cpuset_zone_allowed(zone, __GFP_HARDWALL)) if (!cpuset_zone_allowed(zone, __GFP_HARDWALL))
continue; continue;
zone->temp_priority = priority; note_zone_scanning_priority(zone, priority);
if (zone->prev_priority > priority)
zone->prev_priority = priority;
if (zone->all_unreclaimable && priority != DEF_PRIORITY) if (zone->all_unreclaimable && priority != DEF_PRIORITY)
continue; /* Let kswapd poll it */ continue; /* Let kswapd poll it */
@ -1024,7 +1036,6 @@ unsigned long try_to_free_pages(struct zone **zones, gfp_t gfp_mask)
if (!cpuset_zone_allowed(zone, __GFP_HARDWALL)) if (!cpuset_zone_allowed(zone, __GFP_HARDWALL))
continue; continue;
zone->temp_priority = DEF_PRIORITY;
lru_pages += zone->nr_active + zone->nr_inactive; lru_pages += zone->nr_active + zone->nr_inactive;
} }
@ -1065,13 +1076,22 @@ unsigned long try_to_free_pages(struct zone **zones, gfp_t gfp_mask)
if (!sc.all_unreclaimable) if (!sc.all_unreclaimable)
ret = 1; ret = 1;
out: out:
/*
* Now that we've scanned all the zones at this priority level, note
* that level within the zone so that the next thread which performs
* scanning of this zone will immediately start out at this priority
* level. This affects only the decision whether or not to bring
* mapped pages onto the inactive list.
*/
if (priority < 0)
priority = 0;
for (i = 0; zones[i] != 0; i++) { for (i = 0; zones[i] != 0; i++) {
struct zone *zone = zones[i]; struct zone *zone = zones[i];
if (!cpuset_zone_allowed(zone, __GFP_HARDWALL)) if (!cpuset_zone_allowed(zone, __GFP_HARDWALL))
continue; continue;
zone->prev_priority = zone->temp_priority; zone->prev_priority = priority;
} }
return ret; return ret;
} }
@ -1111,6 +1131,11 @@ static unsigned long balance_pgdat(pg_data_t *pgdat, int order)
.swap_cluster_max = SWAP_CLUSTER_MAX, .swap_cluster_max = SWAP_CLUSTER_MAX,
.swappiness = vm_swappiness, .swappiness = vm_swappiness,
}; };
/*
* temp_priority is used to remember the scanning priority at which
* this zone was successfully refilled to free_pages == pages_high.
*/
int temp_priority[MAX_NR_ZONES];
loop_again: loop_again:
total_scanned = 0; total_scanned = 0;
@ -1118,11 +1143,8 @@ loop_again:
sc.may_writepage = !laptop_mode; sc.may_writepage = !laptop_mode;
count_vm_event(PAGEOUTRUN); count_vm_event(PAGEOUTRUN);
for (i = 0; i < pgdat->nr_zones; i++) { for (i = 0; i < pgdat->nr_zones; i++)
struct zone *zone = pgdat->node_zones + i; temp_priority[i] = DEF_PRIORITY;
zone->temp_priority = DEF_PRIORITY;
}
for (priority = DEF_PRIORITY; priority >= 0; priority--) { for (priority = DEF_PRIORITY; priority >= 0; priority--) {
int end_zone = 0; /* Inclusive. 0 = ZONE_DMA */ int end_zone = 0; /* Inclusive. 0 = ZONE_DMA */
@ -1183,10 +1205,9 @@ scan:
if (!zone_watermark_ok(zone, order, zone->pages_high, if (!zone_watermark_ok(zone, order, zone->pages_high,
end_zone, 0)) end_zone, 0))
all_zones_ok = 0; all_zones_ok = 0;
zone->temp_priority = priority; temp_priority[i] = priority;
if (zone->prev_priority > priority)
zone->prev_priority = priority;
sc.nr_scanned = 0; sc.nr_scanned = 0;
note_zone_scanning_priority(zone, priority);
nr_reclaimed += shrink_zone(priority, zone, &sc); nr_reclaimed += shrink_zone(priority, zone, &sc);
reclaim_state->reclaimed_slab = 0; reclaim_state->reclaimed_slab = 0;
nr_slab = shrink_slab(sc.nr_scanned, GFP_KERNEL, nr_slab = shrink_slab(sc.nr_scanned, GFP_KERNEL,
@ -1226,10 +1247,15 @@ scan:
break; break;
} }
out: out:
/*
* Note within each zone the priority level at which this zone was
* brought into a happy state. So that the next thread which scans this
* zone will start out at that priority level.
*/
for (i = 0; i < pgdat->nr_zones; i++) { for (i = 0; i < pgdat->nr_zones; i++) {
struct zone *zone = pgdat->node_zones + i; struct zone *zone = pgdat->node_zones + i;
zone->prev_priority = zone->temp_priority; zone->prev_priority = temp_priority[i];
} }
if (!all_zones_ok) { if (!all_zones_ok) {
cond_resched(); cond_resched();
@ -1614,6 +1640,7 @@ static int __zone_reclaim(struct zone *zone, gfp_t gfp_mask, unsigned int order)
*/ */
priority = ZONE_RECLAIM_PRIORITY; priority = ZONE_RECLAIM_PRIORITY;
do { do {
note_zone_scanning_priority(zone, priority);
nr_reclaimed += shrink_zone(priority, zone, &sc); nr_reclaimed += shrink_zone(priority, zone, &sc);
priority--; priority--;
} while (priority >= 0 && nr_reclaimed < nr_pages); } while (priority >= 0 && nr_reclaimed < nr_pages);

View File

@ -587,11 +587,9 @@ static int zoneinfo_show(struct seq_file *m, void *arg)
seq_printf(m, seq_printf(m,
"\n all_unreclaimable: %u" "\n all_unreclaimable: %u"
"\n prev_priority: %i" "\n prev_priority: %i"
"\n temp_priority: %i"
"\n start_pfn: %lu", "\n start_pfn: %lu",
zone->all_unreclaimable, zone->all_unreclaimable,
zone->prev_priority, zone->prev_priority,
zone->temp_priority,
zone->zone_start_pfn); zone->zone_start_pfn);
spin_unlock_irqrestore(&zone->lock, flags); spin_unlock_irqrestore(&zone->lock, flags);
seq_putc(m, '\n'); seq_putc(m, '\n');