mm: fix page reference leak in soft_offline_page()
[ Upstream commitdad4e5b390] The conversion to move pfn_to_online_page() internal to soft_offline_page() missed that the get_user_pages() reference taken by the madvise() path needs to be dropped when pfn_to_online_page() fails. Note the direct sysfs-path to soft_offline_page() does not perform a get_user_pages() lookup. When soft_offline_page() is handed a pfn_valid() && !pfn_to_online_page() pfn the kernel hangs at dax-device shutdown due to a leaked reference. Link: https://lkml.kernel.org/r/161058501210.1840162.8108917599181157327.stgit@dwillia2-desk3.amr.corp.intel.com Fixes:feec24a613("mm, soft-offline: convert parameter to pfn") Signed-off-by: Dan Williams <dan.j.williams@intel.com> Reviewed-by: David Hildenbrand <david@redhat.com> Reviewed-by: Oscar Salvador <osalvador@suse.de> Reviewed-by: Naoya Horiguchi <naoya.horiguchi@nec.com> Cc: Michal Hocko <mhocko@suse.com> Cc: Qian Cai <cai@lca.pw> Cc: <stable@vger.kernel.org> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org> Stable-dep-of:e2c1ab070f("mm: memory-failure: fix unexpected return value in soft_offline_page()") Signed-off-by: Sasha Levin <sashal@kernel.org>
This commit is contained in:
committed by
Greg Kroah-Hartman
parent
749630ce91
commit
406166a3ac
+16
-4
@@ -1867,6 +1867,12 @@ static int soft_offline_free_page(struct page *page)
|
|||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void put_ref_page(struct page *page)
|
||||||
|
{
|
||||||
|
if (page)
|
||||||
|
put_page(page);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* soft_offline_page - Soft offline a page.
|
* soft_offline_page - Soft offline a page.
|
||||||
* @pfn: pfn to soft-offline
|
* @pfn: pfn to soft-offline
|
||||||
@@ -1892,20 +1898,26 @@ static int soft_offline_free_page(struct page *page)
|
|||||||
int soft_offline_page(unsigned long pfn, int flags)
|
int soft_offline_page(unsigned long pfn, int flags)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
struct page *page;
|
|
||||||
bool try_again = true;
|
bool try_again = true;
|
||||||
|
struct page *page, *ref_page = NULL;
|
||||||
|
|
||||||
|
WARN_ON_ONCE(!pfn_valid(pfn) && (flags & MF_COUNT_INCREASED));
|
||||||
|
|
||||||
if (!pfn_valid(pfn))
|
if (!pfn_valid(pfn))
|
||||||
return -ENXIO;
|
return -ENXIO;
|
||||||
|
if (flags & MF_COUNT_INCREASED)
|
||||||
|
ref_page = pfn_to_page(pfn);
|
||||||
|
|
||||||
/* Only online pages can be soft-offlined (esp., not ZONE_DEVICE). */
|
/* Only online pages can be soft-offlined (esp., not ZONE_DEVICE). */
|
||||||
page = pfn_to_online_page(pfn);
|
page = pfn_to_online_page(pfn);
|
||||||
if (!page)
|
if (!page) {
|
||||||
|
put_ref_page(ref_page);
|
||||||
return -EIO;
|
return -EIO;
|
||||||
|
}
|
||||||
|
|
||||||
if (PageHWPoison(page)) {
|
if (PageHWPoison(page)) {
|
||||||
pr_info("%s: %#lx page already poisoned\n", __func__, pfn);
|
pr_info("%s: %#lx page already poisoned\n", __func__, pfn);
|
||||||
if (flags & MF_COUNT_INCREASED)
|
put_ref_page(ref_page);
|
||||||
put_page(page);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user