|
@@ -3896,20 +3896,6 @@ follow_huge_pud(struct mm_struct *mm, unsigned long address,
|
|
|
|
|
|
#ifdef CONFIG_MEMORY_FAILURE
|
|
|
|
|
|
-/* Should be called in hugetlb_lock */
|
|
|
-static int is_hugepage_on_freelist(struct page *hpage)
|
|
|
-{
|
|
|
- struct page *page;
|
|
|
- struct page *tmp;
|
|
|
- struct hstate *h = page_hstate(hpage);
|
|
|
- int nid = page_to_nid(hpage);
|
|
|
-
|
|
|
- list_for_each_entry_safe(page, tmp, &h->hugepage_freelists[nid], lru)
|
|
|
- if (page == hpage)
|
|
|
- return 1;
|
|
|
- return 0;
|
|
|
-}
|
|
|
-
|
|
|
/*
|
|
|
* This function is called from memory failure code.
|
|
|
* Assume the caller holds page lock of the head page.
|
|
@@ -3921,7 +3907,11 @@ int dequeue_hwpoisoned_huge_page(struct page *hpage)
|
|
|
int ret = -EBUSY;
|
|
|
|
|
|
spin_lock(&hugetlb_lock);
|
|
|
- if (is_hugepage_on_freelist(hpage)) {
|
|
|
+ /*
|
|
|
+ * Just checking !page_huge_active is not enough, because that could be
|
|
|
+ * an isolated/hwpoisoned hugepage (which have >0 refcount).
|
|
|
+ */
|
|
|
+ if (!page_huge_active(hpage) && !page_count(hpage)) {
|
|
|
/*
|
|
|
* Hwpoisoned hugepage isn't linked to activelist or freelist,
|
|
|
* but dangling hpage->lru can trigger list-debug warnings
|
|
@@ -3965,25 +3955,3 @@ void putback_active_hugepage(struct page *page)
|
|
|
spin_unlock(&hugetlb_lock);
|
|
|
put_page(page);
|
|
|
}
|
|
|
-
|
|
|
-bool is_hugepage_active(struct page *page)
|
|
|
-{
|
|
|
- VM_BUG_ON_PAGE(!PageHuge(page), page);
|
|
|
- /*
|
|
|
- * This function can be called for a tail page because the caller,
|
|
|
- * scan_movable_pages, scans through a given pfn-range which typically
|
|
|
- * covers one memory block. In systems using gigantic hugepage (1GB
|
|
|
- * for x86_64,) a hugepage is larger than a memory block, and we don't
|
|
|
- * support migrating such large hugepages for now, so return false
|
|
|
- * when called for tail pages.
|
|
|
- */
|
|
|
- if (PageTail(page))
|
|
|
- return false;
|
|
|
- /*
|
|
|
- * Refcount of a hwpoisoned hugepages is 1, but they are not active,
|
|
|
- * so we should return false for them.
|
|
|
- */
|
|
|
- if (unlikely(PageHWPoison(page)))
|
|
|
- return false;
|
|
|
- return page_count(page) > 0;
|
|
|
-}
|