phys_addr_t min_addr,
                          int nid, bool exact_nid);
 
-void split_free_page(struct page *free_page,
-                               int order, unsigned long split_pfn_offset);
+int split_free_page(struct page *free_page,
+                       unsigned int order, unsigned long split_pfn_offset);
 
 #if defined CONFIG_COMPACTION || defined CONFIG_CMA
 
 
  * @order:             the order of the page
  * @split_pfn_offset:  split offset within the page
  *
+ * Return -ENOENT if the free page is changed, otherwise 0
+ *
  * It is used when the free page crosses two pageblocks with different migratetypes
  * at split_pfn_offset within the page. The split free page will be put into
  * separate migratetype lists afterwards. Otherwise, the function achieves
  * nothing.
  */
-void split_free_page(struct page *free_page,
-                               int order, unsigned long split_pfn_offset)
+int split_free_page(struct page *free_page,
+                       unsigned int order, unsigned long split_pfn_offset)
 {
        struct zone *zone = page_zone(free_page);
        unsigned long free_page_pfn = page_to_pfn(free_page);
        unsigned long pfn;
        unsigned long flags;
        int free_page_order;
+       int mt;
+       int ret = 0;
 
        if (split_pfn_offset == 0)
-               return;
+               return ret;
 
        spin_lock_irqsave(&zone->lock, flags);
+
+       if (!PageBuddy(free_page) || buddy_order(free_page) != order) {
+               ret = -ENOENT;
+               goto out;
+       }
+
+       mt = get_pageblock_migratetype(free_page);
+       if (likely(!is_migrate_isolate(mt)))
+               __mod_zone_freepage_state(zone, -(1UL << order), mt);
+
        del_page_from_free_list(free_page, zone, order);
        for (pfn = free_page_pfn;
             pfn < free_page_pfn + (1UL << order);) {
                int mt = get_pfnblock_migratetype(pfn_to_page(pfn), pfn);
 
-               free_page_order = min_t(int,
+               free_page_order = min_t(unsigned int,
                                        pfn ? __ffs(pfn) : order,
                                        __fls(split_pfn_offset));
                __free_one_page(pfn_to_page(pfn), pfn, zone, free_page_order,
                if (split_pfn_offset == 0)
                        split_pfn_offset = (1UL << order) - (pfn - free_page_pfn);
        }
+out:
        spin_unlock_irqrestore(&zone->lock, flags);
+       return ret;
 }
 /*
  * A bad page could be due to a number of fields. Instead of multiple branches,
 
                if (PageBuddy(page)) {
                        int order = buddy_order(page);
 
-                       if (pfn + (1UL << order) > boundary_pfn)
-                               split_free_page(page, order, boundary_pfn - pfn);
-                       pfn += (1UL << order);
+                       if (pfn + (1UL << order) > boundary_pfn) {
+                               /* free page changed before split, check it again */
+                               if (split_free_page(page, order, boundary_pfn - pfn))
+                                       continue;
+                       }
+
+                       pfn += 1UL << order;
                        continue;
                }
                /*