Choose pages from the per-cpu list based on migration type
[sfrench/cifs-2.6.git] / mm / page_alloc.c
index 332f46d70d2d75b6750777fd0ce9af572a6820a4..e3e726bd2858f6b97df2e92e4cf7f3ea9073fbd2 100644 (file)
@@ -158,6 +158,22 @@ int nr_node_ids __read_mostly = MAX_NUMNODES;
 EXPORT_SYMBOL(nr_node_ids);
 #endif
 
+static inline int get_pageblock_migratetype(struct page *page)
+{
+       return get_pageblock_flags_group(page, PB_migrate, PB_migrate_end);
+}
+
+static void set_pageblock_migratetype(struct page *page, int migratetype)
+{
+       set_pageblock_flags_group(page, (unsigned long)migratetype,
+                                       PB_migrate, PB_migrate_end);
+}
+
+static inline int gfpflags_to_migratetype(gfp_t gfp_flags)
+{
+       return ((gfp_flags & __GFP_MOVABLE) != 0);
+}
+
 #ifdef CONFIG_DEBUG_VM
 static int page_outside_zone_boundaries(struct zone *zone, struct page *page)
 {
@@ -412,6 +428,7 @@ static inline void __free_one_page(struct page *page,
 {
        unsigned long page_idx;
        int order_size = 1 << order;
+       int migratetype = get_pageblock_migratetype(page);
 
        if (unlikely(PageCompound(page)))
                destroy_compound_page(page, order);
@@ -424,7 +441,6 @@ static inline void __free_one_page(struct page *page,
        __mod_zone_page_state(zone, NR_FREE_PAGES, order_size);
        while (order < MAX_ORDER-1) {
                unsigned long combined_idx;
-               struct free_area *area;
                struct page *buddy;
 
                buddy = __page_find_buddy(page, page_idx, order);
@@ -432,8 +448,7 @@ static inline void __free_one_page(struct page *page,
                        break;          /* Move the buddy up one level. */
 
                list_del(&buddy->lru);
-               area = zone->free_area + order;
-               area->nr_free--;
+               zone->free_area[order].nr_free--;
                rmv_page_order(buddy);
                combined_idx = __find_combined_index(page_idx, order);
                page = page + (combined_idx - page_idx);
@@ -441,7 +456,8 @@ static inline void __free_one_page(struct page *page,
                order++;
        }
        set_page_order(page, order);
-       list_add(&page->lru, &zone->free_area[order].free_list);
+       list_add(&page->lru,
+               &zone->free_area[order].free_list[migratetype]);
        zone->free_area[order].nr_free++;
 }
 
@@ -575,7 +591,8 @@ void fastcall __init __free_pages_bootmem(struct page *page, unsigned int order)
  * -- wli
  */
 static inline void expand(struct zone *zone, struct page *page,
-       int low, int high, struct free_area *area)
+       int low, int high, struct free_area *area,
+       int migratetype)
 {
        unsigned long size = 1 << high;
 
@@ -584,7 +601,7 @@ static inline void expand(struct zone *zone, struct page *page,
                high--;
                size >>= 1;
                VM_BUG_ON(bad_range(zone, &page[size]));
-               list_add(&page[size].lru, &area->free_list);
+               list_add(&page[size].lru, &area->free_list[migratetype]);
                area->nr_free++;
                set_page_order(&page[size], high);
        }
@@ -636,31 +653,95 @@ static int prep_new_page(struct page *page, int order, gfp_t gfp_flags)
        return 0;
 }
 
+/*
+ * This array describes the order lists are fallen back to when
+ * the free lists for the desirable migrate type are depleted
+ */
+static int fallbacks[MIGRATE_TYPES][MIGRATE_TYPES-1] = {
+       [MIGRATE_UNMOVABLE] = { MIGRATE_MOVABLE   },
+       [MIGRATE_MOVABLE]   = { MIGRATE_UNMOVABLE },
+};
+
+/* Remove an element from the buddy allocator from the fallback list */
+static struct page *__rmqueue_fallback(struct zone *zone, int order,
+                                               int start_migratetype)
+{
+       struct free_area * area;
+       int current_order;
+       struct page *page;
+       int migratetype, i;
+
+       /* Find the largest possible block of pages in the other list */
+       for (current_order = MAX_ORDER-1; current_order >= order;
+                                               --current_order) {
+               for (i = 0; i < MIGRATE_TYPES - 1; i++) {
+                       migratetype = fallbacks[start_migratetype][i];
+
+                       area = &(zone->free_area[current_order]);
+                       if (list_empty(&area->free_list[migratetype]))
+                               continue;
+
+                       page = list_entry(area->free_list[migratetype].next,
+                                       struct page, lru);
+                       area->nr_free--;
+
+                       /*
+                        * If breaking a large block of pages, place the buddies
+                        * on the preferred allocation list
+                        */
+                       if (unlikely(current_order >= MAX_ORDER / 2))
+                               migratetype = start_migratetype;
+
+                       /* Remove the page from the freelists */
+                       list_del(&page->lru);
+                       rmv_page_order(page);
+                       __mod_zone_page_state(zone, NR_FREE_PAGES,
+                                                       -(1UL << order));
+
+                       if (current_order == MAX_ORDER - 1)
+                               set_pageblock_migratetype(page,
+                                                       start_migratetype);
+
+                       expand(zone, page, order, current_order, area, migratetype);
+                       return page;
+               }
+       }
+
+       return NULL;
+}
+
 /* 
  * Do the hard work of removing an element from the buddy allocator.
  * Call me with the zone->lock already held.
  */
-static struct page *__rmqueue(struct zone *zone, unsigned int order)
+static struct page *__rmqueue(struct zone *zone, unsigned int order,
+                                               int migratetype)
 {
        struct free_area * area;
        unsigned int current_order;
        struct page *page;
 
+       /* Find a page of the appropriate size in the preferred list */
        for (current_order = order; current_order < MAX_ORDER; ++current_order) {
-               area = zone->free_area + current_order;
-               if (list_empty(&area->free_list))
+               area = &(zone->free_area[current_order]);
+               if (list_empty(&area->free_list[migratetype]))
                        continue;
 
-               page = list_entry(area->free_list.next, struct page, lru);
+               page = list_entry(area->free_list[migratetype].next,
+                                                       struct page, lru);
                list_del(&page->lru);
                rmv_page_order(page);
                area->nr_free--;
                __mod_zone_page_state(zone, NR_FREE_PAGES, - (1UL << order));
-               expand(zone, page, order, current_order, area);
-               return page;
+               expand(zone, page, order, current_order, area, migratetype);
+               goto got_page;
        }
 
-       return NULL;
+       page = __rmqueue_fallback(zone, order, migratetype);
+
+got_page:
+
+       return page;
 }
 
 /* 
@@ -669,16 +750,18 @@ static struct page *__rmqueue(struct zone *zone, unsigned int order)
  * Returns the number of new pages which were placed at *list.
  */
 static int rmqueue_bulk(struct zone *zone, unsigned int order, 
-                       unsigned long count, struct list_head *list)
+                       unsigned long count, struct list_head *list,
+                       int migratetype)
 {
        int i;
        
        spin_lock(&zone->lock);
        for (i = 0; i < count; ++i) {
-               struct page *page = __rmqueue(zone, order);
+               struct page *page = __rmqueue(zone, order, migratetype);
                if (unlikely(page == NULL))
                        break;
-               list_add_tail(&page->lru, list);
+               list_add(&page->lru, list);
+               set_page_private(page, migratetype);
        }
        spin_unlock(&zone->lock);
        return i;
@@ -740,7 +823,7 @@ void mark_free_pages(struct zone *zone)
 {
        unsigned long pfn, max_zone_pfn;
        unsigned long flags;
-       int order;
+       int order, t;
        struct list_head *curr;
 
        if (!zone->spanned_pages)
@@ -757,15 +840,15 @@ void mark_free_pages(struct zone *zone)
                                swsusp_unset_page_free(page);
                }
 
-       for (order = MAX_ORDER - 1; order >= 0; --order)
-               list_for_each(curr, &zone->free_area[order].free_list) {
+       for_each_migratetype_order(order, t) {
+               list_for_each(curr, &zone->free_area[order].free_list[t]) {
                        unsigned long i;
 
                        pfn = page_to_pfn(list_entry(curr, struct page, lru));
                        for (i = 0; i < (1UL << order); i++)
                                swsusp_set_page_free(pfn_to_page(pfn + i));
                }
-
+       }
        spin_unlock_irqrestore(&zone->lock, flags);
 }
 
@@ -805,6 +888,7 @@ static void fastcall free_hot_cold_page(struct page *page, int cold)
        local_irq_save(flags);
        __count_vm_event(PGFREE);
        list_add(&page->lru, &pcp->list);
+       set_page_private(page, get_pageblock_migratetype(page));
        pcp->count++;
        if (pcp->count >= pcp->high) {
                free_pages_bulk(zone, pcp->batch, &pcp->list, 0);
@@ -854,6 +938,7 @@ static struct page *buffered_rmqueue(struct zonelist *zonelist,
        struct page *page;
        int cold = !!(gfp_flags & __GFP_COLD);
        int cpu;
+       int migratetype = gfpflags_to_migratetype(gfp_flags);
 
 again:
        cpu  = get_cpu();
@@ -864,16 +949,34 @@ again:
                local_irq_save(flags);
                if (!pcp->count) {
                        pcp->count = rmqueue_bulk(zone, 0,
-                                               pcp->batch, &pcp->list);
+                                       pcp->batch, &pcp->list, migratetype);
                        if (unlikely(!pcp->count))
                                goto failed;
                }
-               page = list_entry(pcp->list.next, struct page, lru);
-               list_del(&page->lru);
-               pcp->count--;
+               /* Find a page of the appropriate migrate type */
+               list_for_each_entry(page, &pcp->list, lru) {
+                       if (page_private(page) == migratetype) {
+                               list_del(&page->lru);
+                               pcp->count--;
+                               break;
+                       }
+               }
+
+               /*
+                * Check if a page of the appropriate migrate type
+                * was found. If not, allocate more to the pcp list
+                */
+               if (&page->lru == &pcp->list) {
+                       pcp->count += rmqueue_bulk(zone, 0,
+                                       pcp->batch, &pcp->list, migratetype);
+                       page = list_entry(pcp->list.next, struct page, lru);
+                       VM_BUG_ON(page_private(page) != migratetype);
+                       list_del(&page->lru);
+                       pcp->count--;
+               }
        } else {
                spin_lock_irqsave(&zone->lock, flags);
-               page = __rmqueue(zone, order);
+               page = __rmqueue(zone, order, migratetype);
                spin_unlock(&zone->lock);
                if (!page)
                        goto failed;
@@ -2233,6 +2336,16 @@ void __meminit memmap_init_zone(unsigned long size, int nid, unsigned long zone,
                init_page_count(page);
                reset_page_mapcount(page);
                SetPageReserved(page);
+
+               /*
+                * Mark the block movable so that blocks are reserved for
+                * movable at startup. This will force kernel allocations
+                * to reserve their blocks rather than leaking throughout
+                * the address space during boot when many long-lived
+                * kernel allocations are made
+                */
+               set_pageblock_migratetype(page, MIGRATE_MOVABLE);
+
                INIT_LIST_HEAD(&page->lru);
 #ifdef WANT_PAGE_VIRTUAL
                /* The shift won't overflow because ZONE_NORMAL is below 4G. */
@@ -2245,9 +2358,9 @@ void __meminit memmap_init_zone(unsigned long size, int nid, unsigned long zone,
 static void __meminit zone_init_free_lists(struct pglist_data *pgdat,
                                struct zone *zone, unsigned long size)
 {
-       int order;
-       for (order = 0; order < MAX_ORDER ; order++) {
-               INIT_LIST_HEAD(&zone->free_area[order].free_list);
+       int order, t;
+       for_each_migratetype_order(order, t) {
+               INIT_LIST_HEAD(&zone->free_area[order].free_list[t]);
                zone->free_area[order].nr_free = 0;
        }
 }