LCOV - code coverage report
Current view: top level - mm - page_vma_mapped.c (source / functions) Hit Total Coverage
Test: gcov_data_raspi2_real_modules_combined.info Lines: 22 33 66.7 %
Date: 2020-09-30 20:25:40 Functions: 3 4 75.0 %
Branches: 12 38 31.6 %

           Branch data     Line data    Source code
       1                 :            : // SPDX-License-Identifier: GPL-2.0
       2                 :            : #include <linux/mm.h>
       3                 :            : #include <linux/rmap.h>
       4                 :            : #include <linux/hugetlb.h>
       5                 :            : #include <linux/swap.h>
       6                 :            : #include <linux/swapops.h>
       7                 :            : 
       8                 :            : #include "internal.h"
       9                 :            : 
      10                 :            : static inline bool not_found(struct page_vma_mapped_walk *pvmw)
      11                 :            : {
      12                 :            :         page_vma_mapped_walk_done(pvmw);
      13                 :            :         return false;
      14                 :            : }
      15                 :            : 
      16                 :        140 : static bool map_pte(struct page_vma_mapped_walk *pvmw)
      17                 :            : {
      18                 :        280 :         pvmw->pte = pte_offset_map(pvmw->pmd, pvmw->address);
      19         [ -  + ]:        140 :         if (!(pvmw->flags & PVMW_SYNC)) {
      20         [ #  # ]:          0 :                 if (pvmw->flags & PVMW_MIGRATION) {
      21         [ #  # ]:          0 :                         if (!is_swap_pte(*pvmw->pte))
      22                 :            :                                 return false;
      23                 :            :                 } else {
      24                 :            :                         /*
      25                 :            :                          * We get here when we are trying to unmap a private
      26                 :            :                          * device page from the process address space. Such
      27                 :            :                          * page is not CPU accessible and thus is mapped as
      28                 :            :                          * a special swap entry, nonetheless it still does
      29                 :            :                          * count as a valid regular mapping for the page (and
      30                 :            :                          * is accounted as such in page maps count).
      31                 :            :                          *
      32                 :            :                          * So handle this special case as if it was a normal
      33                 :            :                          * page mapping ie lock CPU page table and returns
      34                 :            :                          * true.
      35                 :            :                          *
      36                 :            :                          * For more details on device private memory see HMM
      37                 :            :                          * (include/linux/hmm.h or mm/hmm.c).
      38                 :            :                          */
      39         [ #  # ]:          0 :                         if (is_swap_pte(*pvmw->pte)) {
      40                 :            :                                 swp_entry_t entry;
      41                 :            : 
      42                 :            :                                 /* Handle un-addressable ZONE_DEVICE memory */
      43                 :            :                                 entry = pte_to_swp_entry(*pvmw->pte);
      44                 :            :                                 if (!is_device_private_entry(entry))
      45                 :            :                                         return false;
      46         [ #  # ]:          0 :                         } else if (!pte_present(*pvmw->pte))
      47                 :            :                                 return false;
      48                 :            :                 }
      49                 :            :         }
      50                 :        140 :         pvmw->ptl = pte_lockptr(pvmw->vma->vm_mm, pvmw->pmd);
      51                 :            :         spin_lock(pvmw->ptl);
      52                 :        140 :         return true;
      53                 :            : }
      54                 :            : 
      55                 :            : static inline bool pfn_in_hpage(struct page *hpage, unsigned long pfn)
      56                 :            : {
      57                 :        140 :         unsigned long hpage_pfn = page_to_pfn(hpage);
      58                 :            : 
      59                 :            :         /* THP can be referenced by any subpage */
      60                 :        140 :         return pfn >= hpage_pfn && pfn - hpage_pfn < hpage_nr_pages(hpage);
      61                 :            : }
      62                 :            : 
      63                 :            : /**
      64                 :            :  * check_pte - check if @pvmw->page is mapped at the @pvmw->pte
      65                 :            :  *
      66                 :            :  * page_vma_mapped_walk() found a place where @pvmw->page is *potentially*
      67                 :            :  * mapped. check_pte() has to validate this.
      68                 :            :  *
      69                 :            :  * @pvmw->pte may point to empty PTE, swap PTE or PTE pointing to arbitrary
      70                 :            :  * page.
      71                 :            :  *
      72                 :            :  * If PVMW_MIGRATION flag is set, returns true if @pvmw->pte contains migration
      73                 :            :  * entry that points to @pvmw->page or any subpage in case of THP.
      74                 :            :  *
      75                 :            :  * If PVMW_MIGRATION flag is not set, returns true if @pvmw->pte points to
      76                 :            :  * @pvmw->page or any subpage in case of THP.
      77                 :            :  *
      78                 :            :  * Otherwise, return false.
      79                 :            :  *
      80                 :            :  */
      81                 :        140 : static bool check_pte(struct page_vma_mapped_walk *pvmw)
      82                 :            : {
      83                 :            :         unsigned long pfn;
      84                 :            : 
      85         [ -  + ]:        140 :         if (pvmw->flags & PVMW_MIGRATION) {
      86                 :            :                 swp_entry_t entry;
      87         [ #  # ]:          0 :                 if (!is_swap_pte(*pvmw->pte))
      88                 :            :                         return false;
      89                 :            :                 entry = pte_to_swp_entry(*pvmw->pte);
      90                 :            : 
      91         [ #  # ]:          0 :                 if (!is_migration_entry(entry))
      92                 :            :                         return false;
      93                 :            : 
      94                 :            :                 pfn = migration_entry_to_pfn(entry);
      95         [ +  - ]:        280 :         } else if (is_swap_pte(*pvmw->pte)) {
      96                 :            :                 swp_entry_t entry;
      97                 :            : 
      98                 :            :                 /* Handle un-addressable ZONE_DEVICE memory */
      99                 :            :                 entry = pte_to_swp_entry(*pvmw->pte);
     100                 :            :                 if (!is_device_private_entry(entry))
     101                 :            :                         return false;
     102                 :            : 
     103                 :            :                 pfn = device_private_entry_to_pfn(entry);
     104                 :            :         } else {
     105         [ +  - ]:        140 :                 if (!pte_present(*pvmw->pte))
     106                 :            :                         return false;
     107                 :            : 
     108                 :        140 :                 pfn = pte_pfn(*pvmw->pte);
     109                 :            :         }
     110                 :            : 
     111                 :        280 :         return pfn_in_hpage(pvmw->page, pfn);
     112                 :            : }
     113                 :            : 
     114                 :            : /**
     115                 :            :  * page_vma_mapped_walk - check if @pvmw->page is mapped in @pvmw->vma at
     116                 :            :  * @pvmw->address
     117                 :            :  * @pvmw: pointer to struct page_vma_mapped_walk. page, vma, address and flags
     118                 :            :  * must be set. pmd, pte and ptl must be NULL.
     119                 :            :  *
     120                 :            :  * Returns true if the page is mapped in the vma. @pvmw->pmd and @pvmw->pte point
     121                 :            :  * to relevant page table entries. @pvmw->ptl is locked. @pvmw->address is
     122                 :            :  * adjusted if needed (for PTE-mapped THPs).
     123                 :            :  *
     124                 :            :  * If @pvmw->pmd is set but @pvmw->pte is not, you have found PMD-mapped page
     125                 :            :  * (usually THP). For PTE-mapped THP, you should run page_vma_mapped_walk() in
     126                 :            :  * a loop to find all PTEs that map the THP.
     127                 :            :  *
     128                 :            :  * For HugeTLB pages, @pvmw->pte is set to the relevant page table entry
     129                 :            :  * regardless of which page table level the page is mapped at. @pvmw->pmd is
     130                 :            :  * NULL.
     131                 :            :  *
     132                 :            :  * Retruns false if there are no more page table entries for the page in
     133                 :            :  * the vma. @pvmw->ptl is unlocked and @pvmw->pte is unmapped.
     134                 :            :  *
     135                 :            :  * If you need to stop the walk before page_vma_mapped_walk() returned false,
     136                 :            :  * use page_vma_mapped_walk_done(). It will do the housekeeping.
     137                 :            :  */
     138                 :        280 : bool page_vma_mapped_walk(struct page_vma_mapped_walk *pvmw)
     139                 :            : {
     140                 :        280 :         struct mm_struct *mm = pvmw->vma->vm_mm;
     141                 :            :         struct page *page = pvmw->page;
     142                 :            :         pgd_t *pgd;
     143                 :            :         p4d_t *p4d;
     144                 :            :         pud_t *pud;
     145                 :            :         pmd_t pmde;
     146                 :            : 
     147                 :            :         /* The only possible pmd mapping has been handled on last iteration */
     148   [ +  +  -  + ]:        280 :         if (pvmw->pmd && !pvmw->pte)
     149                 :            :                 return not_found(pvmw);
     150                 :            : 
     151         [ +  + ]:        280 :         if (pvmw->pte)
     152                 :            :                 goto next_pte;
     153                 :            : 
     154                 :            :         if (unlikely(PageHuge(pvmw->page))) {
     155                 :            :                 /* when pud is not present, pte will be NULL */
     156                 :            :                 pvmw->pte = huge_pte_offset(mm, pvmw->address, page_size(page));
     157                 :            :                 if (!pvmw->pte)
     158                 :            :                         return false;
     159                 :            : 
     160                 :            :                 pvmw->ptl = huge_pte_lockptr(page_hstate(page), mm, pvmw->pte);
     161                 :            :                 spin_lock(pvmw->ptl);
     162                 :            :                 if (!check_pte(pvmw))
     163                 :            :                         return not_found(pvmw);
     164                 :            :                 return true;
     165                 :            :         }
     166                 :            : restart:
     167                 :        140 :         pgd = pgd_offset(mm, pvmw->address);
     168                 :            :         if (!pgd_present(*pgd))
     169                 :            :                 return false;
     170                 :            :         p4d = p4d_offset(pgd, pvmw->address);
     171                 :            :         if (!p4d_present(*p4d))
     172                 :            :                 return false;
     173                 :            :         pud = pud_offset(p4d, pvmw->address);
     174                 :            :         if (!pud_present(*pud))
     175                 :            :                 return false;
     176                 :        140 :         pvmw->pmd = pmd_offset(pud, pvmw->address);
     177                 :            :         /*
     178                 :            :          * Make sure the pmd value isn't cached in a register by the
     179                 :            :          * compiler and used as a stale value after we've observed a
     180                 :            :          * subsequent update.
     181                 :            :          */
     182                 :            :         pmde = READ_ONCE(*pvmw->pmd);
     183                 :            :         if (pmd_trans_huge(pmde) || is_pmd_migration_entry(pmde)) {
     184                 :            :                 pvmw->ptl = pmd_lock(mm, pvmw->pmd);
     185                 :            :                 if (likely(pmd_trans_huge(*pvmw->pmd))) {
     186                 :            :                         if (pvmw->flags & PVMW_MIGRATION)
     187                 :            :                                 return not_found(pvmw);
     188                 :            :                         if (pmd_page(*pvmw->pmd) != page)
     189                 :            :                                 return not_found(pvmw);
     190                 :            :                         return true;
     191                 :            :                 } else if (!pmd_present(*pvmw->pmd)) {
     192                 :            :                         if (thp_migration_supported()) {
     193                 :            :                                 if (!(pvmw->flags & PVMW_MIGRATION))
     194                 :            :                                         return not_found(pvmw);
     195                 :            :                                 if (is_migration_entry(pmd_to_swp_entry(*pvmw->pmd))) {
     196                 :            :                                         swp_entry_t entry = pmd_to_swp_entry(*pvmw->pmd);
     197                 :            : 
     198                 :            :                                         if (migration_entry_to_page(entry) != page)
     199                 :            :                                                 return not_found(pvmw);
     200                 :            :                                         return true;
     201                 :            :                                 }
     202                 :            :                         }
     203                 :            :                         return not_found(pvmw);
     204                 :            :                 } else {
     205                 :            :                         /* THP pmd was split under us: handle on pte level */
     206                 :            :                         spin_unlock(pvmw->ptl);
     207                 :            :                         pvmw->ptl = NULL;
     208                 :            :                 }
     209         [ +  - ]:        140 :         } else if (!pmd_present(pmde)) {
     210                 :            :                 return false;
     211                 :            :         }
     212         [ +  - ]:        140 :         if (!map_pte(pvmw))
     213                 :            :                 goto next_pte;
     214                 :            :         while (1) {
     215         [ -  + ]:        140 :                 if (check_pte(pvmw))
     216                 :            :                         return true;
     217                 :            : next_pte:
     218                 :            :                 /* Seek to next pte only makes sense for THP */
     219                 :            :                 if (!PageTransHuge(pvmw->page) || PageHuge(pvmw->page))
     220                 :            :                         return not_found(pvmw);
     221                 :            :                 do {
     222                 :            :                         pvmw->address += PAGE_SIZE;
     223                 :            :                         if (pvmw->address >= pvmw->vma->vm_end ||
     224                 :            :                             pvmw->address >=
     225                 :            :                                         __vma_address(pvmw->page, pvmw->vma) +
     226                 :            :                                         hpage_nr_pages(pvmw->page) * PAGE_SIZE)
     227                 :            :                                 return not_found(pvmw);
     228                 :            :                         /* Did we cross page table boundary? */
     229                 :            :                         if (pvmw->address % PMD_SIZE == 0) {
     230                 :            :                                 pte_unmap(pvmw->pte);
     231                 :            :                                 if (pvmw->ptl) {
     232                 :            :                                         spin_unlock(pvmw->ptl);
     233                 :            :                                         pvmw->ptl = NULL;
     234                 :            :                                 }
     235                 :            :                                 goto restart;
     236                 :            :                         } else {
     237                 :            :                                 pvmw->pte++;
     238                 :            :                         }
     239                 :            :                 } while (pte_none(*pvmw->pte));
     240                 :            : 
     241                 :            :                 if (!pvmw->ptl) {
     242                 :            :                         pvmw->ptl = pte_lockptr(mm, pvmw->pmd);
     243                 :            :                         spin_lock(pvmw->ptl);
     244                 :            :                 }
     245                 :            :         }
     246                 :            : }
     247                 :            : 
     248                 :            : /**
     249                 :            :  * page_mapped_in_vma - check whether a page is really mapped in a VMA
     250                 :            :  * @page: the page to test
     251                 :            :  * @vma: the VMA to test
     252                 :            :  *
     253                 :            :  * Returns 1 if the page is mapped into the page tables of the VMA, 0
     254                 :            :  * if the page is not mapped into the page tables of this VMA.  Only
     255                 :            :  * valid for normal file or anonymous VMAs.
     256                 :            :  */
     257                 :          0 : int page_mapped_in_vma(struct page *page, struct vm_area_struct *vma)
     258                 :            : {
     259                 :          0 :         struct page_vma_mapped_walk pvmw = {
     260                 :            :                 .page = page,
     261                 :            :                 .vma = vma,
     262                 :            :                 .flags = PVMW_SYNC,
     263                 :            :         };
     264                 :            :         unsigned long start, end;
     265                 :            : 
     266                 :            :         start = __vma_address(page, vma);
     267                 :            :         end = start + PAGE_SIZE * (hpage_nr_pages(page) - 1);
     268                 :            : 
     269   [ #  #  #  # ]:          0 :         if (unlikely(end < vma->vm_start || start >= vma->vm_end))
     270                 :            :                 return 0;
     271                 :          0 :         pvmw.address = max(start, vma->vm_start);
     272         [ #  # ]:          0 :         if (!page_vma_mapped_walk(&pvmw))
     273                 :            :                 return 0;
     274                 :            :         page_vma_mapped_walk_done(&pvmw);
     275                 :            :         return 1;
     276                 :            : }

Generated by: LCOV version 1.14