mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-18 02:46:06 +00:00
mm/damon: validate if the pmd entry is present before accessing
pmd_huge() is used to validate if the pmd entry is mapped by a huge page, also including the case of non-present (migration or hwpoisoned) pmd entry on arm64 or x86 architectures. This means that pmd_pfn() can not get the correct pfn number for a non-present pmd entry, which will cause damon_get_page() to get an incorrect page struct (also may be NULL by pfn_to_online_page()), making the access statistics incorrect. This means that the DAMON may make incorrect decision according to the incorrect statistics, for example, DAMON may can not reclaim cold page in time due to this cold page was regarded as accessed mistakenly if DAMOS_PAGEOUT operation is specified. Moreover it does not make sense that we still waste time to get the page of the non-present entry. Just treat it as not-accessed and skip it, which maintains consistency with non-present pte level entries. So add pmd entry present validation to fix the above issues. Link: https://lkml.kernel.org/r/58b1d1f5fbda7db49ca886d9ef6783e3dcbbbc98.1660805030.git.baolin.wang@linux.alibaba.com Fixes: 3f49584b262c ("mm/damon: implement primitives for the virtual memory address spaces") Signed-off-by: Baolin Wang <baolin.wang@linux.alibaba.com> Reviewed-by: SeongJae Park <sj@kernel.org> Reviewed-by: Muchun Song <songmuchun@bytedance.com> Cc: Mike Kravetz <mike.kravetz@oracle.com> Cc: <stable@vger.kernel.org> Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
This commit is contained in:
parent
6a3edd2939
commit
c8b9aff419
@ -304,6 +304,11 @@ static int damon_mkold_pmd_entry(pmd_t *pmd, unsigned long addr,
|
|||||||
|
|
||||||
if (pmd_huge(*pmd)) {
|
if (pmd_huge(*pmd)) {
|
||||||
ptl = pmd_lock(walk->mm, pmd);
|
ptl = pmd_lock(walk->mm, pmd);
|
||||||
|
if (!pmd_present(*pmd)) {
|
||||||
|
spin_unlock(ptl);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
if (pmd_huge(*pmd)) {
|
if (pmd_huge(*pmd)) {
|
||||||
damon_pmdp_mkold(pmd, walk->mm, addr);
|
damon_pmdp_mkold(pmd, walk->mm, addr);
|
||||||
spin_unlock(ptl);
|
spin_unlock(ptl);
|
||||||
@ -431,6 +436,11 @@ static int damon_young_pmd_entry(pmd_t *pmd, unsigned long addr,
|
|||||||
#ifdef CONFIG_TRANSPARENT_HUGEPAGE
|
#ifdef CONFIG_TRANSPARENT_HUGEPAGE
|
||||||
if (pmd_huge(*pmd)) {
|
if (pmd_huge(*pmd)) {
|
||||||
ptl = pmd_lock(walk->mm, pmd);
|
ptl = pmd_lock(walk->mm, pmd);
|
||||||
|
if (!pmd_present(*pmd)) {
|
||||||
|
spin_unlock(ptl);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
if (!pmd_huge(*pmd)) {
|
if (!pmd_huge(*pmd)) {
|
||||||
spin_unlock(ptl);
|
spin_unlock(ptl);
|
||||||
goto regular_page;
|
goto regular_page;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user