~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~

TOMOYO Linux Cross Reference
Linux/arch/arm64/mm/contpte.c

Version: ~ [ linux-6.11.5 ] ~ [ linux-6.10.14 ] ~ [ linux-6.9.12 ] ~ [ linux-6.8.12 ] ~ [ linux-6.7.12 ] ~ [ linux-6.6.58 ] ~ [ linux-6.5.13 ] ~ [ linux-6.4.16 ] ~ [ linux-6.3.13 ] ~ [ linux-6.2.16 ] ~ [ linux-6.1.114 ] ~ [ linux-6.0.19 ] ~ [ linux-5.19.17 ] ~ [ linux-5.18.19 ] ~ [ linux-5.17.15 ] ~ [ linux-5.16.20 ] ~ [ linux-5.15.169 ] ~ [ linux-5.14.21 ] ~ [ linux-5.13.19 ] ~ [ linux-5.12.19 ] ~ [ linux-5.11.22 ] ~ [ linux-5.10.228 ] ~ [ linux-5.9.16 ] ~ [ linux-5.8.18 ] ~ [ linux-5.7.19 ] ~ [ linux-5.6.19 ] ~ [ linux-5.5.19 ] ~ [ linux-5.4.284 ] ~ [ linux-5.3.18 ] ~ [ linux-5.2.21 ] ~ [ linux-5.1.21 ] ~ [ linux-5.0.21 ] ~ [ linux-4.20.17 ] ~ [ linux-4.19.322 ] ~ [ linux-4.18.20 ] ~ [ linux-4.17.19 ] ~ [ linux-4.16.18 ] ~ [ linux-4.15.18 ] ~ [ linux-4.14.336 ] ~ [ linux-4.13.16 ] ~ [ linux-4.12.14 ] ~ [ linux-4.11.12 ] ~ [ linux-4.10.17 ] ~ [ linux-4.9.337 ] ~ [ linux-4.4.302 ] ~ [ linux-3.10.108 ] ~ [ linux-2.6.32.71 ] ~ [ linux-2.6.0 ] ~ [ linux-2.4.37.11 ] ~ [ unix-v6-master ] ~ [ ccs-tools-1.8.9 ] ~ [ policy-sample ] ~
Architecture: ~ [ i386 ] ~ [ alpha ] ~ [ m68k ] ~ [ mips ] ~ [ ppc ] ~ [ sparc ] ~ [ sparc64 ] ~

  1 // SPDX-License-Identifier: GPL-2.0-only
  2 /*
  3  * Copyright (C) 2023 ARM Ltd.
  4  */
  5 
  6 #include <linux/mm.h>
  7 #include <linux/efi.h>
  8 #include <linux/export.h>
  9 #include <asm/tlbflush.h>
 10 
 11 static inline bool mm_is_user(struct mm_struct *mm)
 12 {
 13         /*
 14          * Don't attempt to apply the contig bit to kernel mappings, because
 15          * dynamically adding/removing the contig bit can cause page faults.
 16          * These racing faults are ok for user space, since they get serialized
 17          * on the PTL. But kernel mappings can't tolerate faults.
 18          */
 19         if (unlikely(mm_is_efi(mm)))
 20                 return false;
 21         return mm != &init_mm;
 22 }
 23 
 24 static inline pte_t *contpte_align_down(pte_t *ptep)
 25 {
 26         return PTR_ALIGN_DOWN(ptep, sizeof(*ptep) * CONT_PTES);
 27 }
 28 
 29 static void contpte_try_unfold_partial(struct mm_struct *mm, unsigned long addr,
 30                                         pte_t *ptep, unsigned int nr)
 31 {
 32         /*
 33          * Unfold any partially covered contpte block at the beginning and end
 34          * of the range.
 35          */
 36 
 37         if (ptep != contpte_align_down(ptep) || nr < CONT_PTES)
 38                 contpte_try_unfold(mm, addr, ptep, __ptep_get(ptep));
 39 
 40         if (ptep + nr != contpte_align_down(ptep + nr)) {
 41                 unsigned long last_addr = addr + PAGE_SIZE * (nr - 1);
 42                 pte_t *last_ptep = ptep + nr - 1;
 43 
 44                 contpte_try_unfold(mm, last_addr, last_ptep,
 45                                    __ptep_get(last_ptep));
 46         }
 47 }
 48 
 49 static void contpte_convert(struct mm_struct *mm, unsigned long addr,
 50                             pte_t *ptep, pte_t pte)
 51 {
 52         struct vm_area_struct vma = TLB_FLUSH_VMA(mm, 0);
 53         unsigned long start_addr;
 54         pte_t *start_ptep;
 55         int i;
 56 
 57         start_ptep = ptep = contpte_align_down(ptep);
 58         start_addr = addr = ALIGN_DOWN(addr, CONT_PTE_SIZE);
 59         pte = pfn_pte(ALIGN_DOWN(pte_pfn(pte), CONT_PTES), pte_pgprot(pte));
 60 
 61         for (i = 0; i < CONT_PTES; i++, ptep++, addr += PAGE_SIZE) {
 62                 pte_t ptent = __ptep_get_and_clear(mm, addr, ptep);
 63 
 64                 if (pte_dirty(ptent))
 65                         pte = pte_mkdirty(pte);
 66 
 67                 if (pte_young(ptent))
 68                         pte = pte_mkyoung(pte);
 69         }
 70 
 71         __flush_tlb_range(&vma, start_addr, addr, PAGE_SIZE, true, 3);
 72 
 73         __set_ptes(mm, start_addr, start_ptep, pte, CONT_PTES);
 74 }
 75 
 76 void __contpte_try_fold(struct mm_struct *mm, unsigned long addr,
 77                         pte_t *ptep, pte_t pte)
 78 {
 79         /*
 80          * We have already checked that the virtual and pysical addresses are
 81          * correctly aligned for a contpte mapping in contpte_try_fold() so the
 82          * remaining checks are to ensure that the contpte range is fully
 83          * covered by a single folio, and ensure that all the ptes are valid
 84          * with contiguous PFNs and matching prots. We ignore the state of the
 85          * access and dirty bits for the purpose of deciding if its a contiguous
 86          * range; the folding process will generate a single contpte entry which
 87          * has a single access and dirty bit. Those 2 bits are the logical OR of
 88          * their respective bits in the constituent pte entries. In order to
 89          * ensure the contpte range is covered by a single folio, we must
 90          * recover the folio from the pfn, but special mappings don't have a
 91          * folio backing them. Fortunately contpte_try_fold() already checked
 92          * that the pte is not special - we never try to fold special mappings.
 93          * Note we can't use vm_normal_page() for this since we don't have the
 94          * vma.
 95          */
 96 
 97         unsigned long folio_start, folio_end;
 98         unsigned long cont_start, cont_end;
 99         pte_t expected_pte, subpte;
100         struct folio *folio;
101         struct page *page;
102         unsigned long pfn;
103         pte_t *orig_ptep;
104         pgprot_t prot;
105 
106         int i;
107 
108         if (!mm_is_user(mm))
109                 return;
110 
111         page = pte_page(pte);
112         folio = page_folio(page);
113         folio_start = addr - (page - &folio->page) * PAGE_SIZE;
114         folio_end = folio_start + folio_nr_pages(folio) * PAGE_SIZE;
115         cont_start = ALIGN_DOWN(addr, CONT_PTE_SIZE);
116         cont_end = cont_start + CONT_PTE_SIZE;
117 
118         if (folio_start > cont_start || folio_end < cont_end)
119                 return;
120 
121         pfn = ALIGN_DOWN(pte_pfn(pte), CONT_PTES);
122         prot = pte_pgprot(pte_mkold(pte_mkclean(pte)));
123         expected_pte = pfn_pte(pfn, prot);
124         orig_ptep = ptep;
125         ptep = contpte_align_down(ptep);
126 
127         for (i = 0; i < CONT_PTES; i++) {
128                 subpte = pte_mkold(pte_mkclean(__ptep_get(ptep)));
129                 if (!pte_same(subpte, expected_pte))
130                         return;
131                 expected_pte = pte_advance_pfn(expected_pte, 1);
132                 ptep++;
133         }
134 
135         pte = pte_mkcont(pte);
136         contpte_convert(mm, addr, orig_ptep, pte);
137 }
138 EXPORT_SYMBOL_GPL(__contpte_try_fold);
139 
140 void __contpte_try_unfold(struct mm_struct *mm, unsigned long addr,
141                         pte_t *ptep, pte_t pte)
142 {
143         /*
144          * We have already checked that the ptes are contiguous in
145          * contpte_try_unfold(), so just check that the mm is user space.
146          */
147         if (!mm_is_user(mm))
148                 return;
149 
150         pte = pte_mknoncont(pte);
151         contpte_convert(mm, addr, ptep, pte);
152 }
153 EXPORT_SYMBOL_GPL(__contpte_try_unfold);
154 
155 pte_t contpte_ptep_get(pte_t *ptep, pte_t orig_pte)
156 {
157         /*
158          * Gather access/dirty bits, which may be populated in any of the ptes
159          * of the contig range. We are guaranteed to be holding the PTL, so any
160          * contiguous range cannot be unfolded or otherwise modified under our
161          * feet.
162          */
163 
164         pte_t pte;
165         int i;
166 
167         ptep = contpte_align_down(ptep);
168 
169         for (i = 0; i < CONT_PTES; i++, ptep++) {
170                 pte = __ptep_get(ptep);
171 
172                 if (pte_dirty(pte))
173                         orig_pte = pte_mkdirty(orig_pte);
174 
175                 if (pte_young(pte))
176                         orig_pte = pte_mkyoung(orig_pte);
177         }
178 
179         return orig_pte;
180 }
181 EXPORT_SYMBOL_GPL(contpte_ptep_get);
182 
183 pte_t contpte_ptep_get_lockless(pte_t *orig_ptep)
184 {
185         /*
186          * The ptep_get_lockless() API requires us to read and return *orig_ptep
187          * so that it is self-consistent, without the PTL held, so we may be
188          * racing with other threads modifying the pte. Usually a READ_ONCE()
189          * would suffice, but for the contpte case, we also need to gather the
190          * access and dirty bits from across all ptes in the contiguous block,
191          * and we can't read all of those neighbouring ptes atomically, so any
192          * contiguous range may be unfolded/modified/refolded under our feet.
193          * Therefore we ensure we read a _consistent_ contpte range by checking
194          * that all ptes in the range are valid and have CONT_PTE set, that all
195          * pfns are contiguous and that all pgprots are the same (ignoring
196          * access/dirty). If we find a pte that is not consistent, then we must
197          * be racing with an update so start again. If the target pte does not
198          * have CONT_PTE set then that is considered consistent on its own
199          * because it is not part of a contpte range.
200          */
201 
202         pgprot_t orig_prot;
203         unsigned long pfn;
204         pte_t orig_pte;
205         pgprot_t prot;
206         pte_t *ptep;
207         pte_t pte;
208         int i;
209 
210 retry:
211         orig_pte = __ptep_get(orig_ptep);
212 
213         if (!pte_valid_cont(orig_pte))
214                 return orig_pte;
215 
216         orig_prot = pte_pgprot(pte_mkold(pte_mkclean(orig_pte)));
217         ptep = contpte_align_down(orig_ptep);
218         pfn = pte_pfn(orig_pte) - (orig_ptep - ptep);
219 
220         for (i = 0; i < CONT_PTES; i++, ptep++, pfn++) {
221                 pte = __ptep_get(ptep);
222                 prot = pte_pgprot(pte_mkold(pte_mkclean(pte)));
223 
224                 if (!pte_valid_cont(pte) ||
225                    pte_pfn(pte) != pfn ||
226                    pgprot_val(prot) != pgprot_val(orig_prot))
227                         goto retry;
228 
229                 if (pte_dirty(pte))
230                         orig_pte = pte_mkdirty(orig_pte);
231 
232                 if (pte_young(pte))
233                         orig_pte = pte_mkyoung(orig_pte);
234         }
235 
236         return orig_pte;
237 }
238 EXPORT_SYMBOL_GPL(contpte_ptep_get_lockless);
239 
240 void contpte_set_ptes(struct mm_struct *mm, unsigned long addr,
241                                         pte_t *ptep, pte_t pte, unsigned int nr)
242 {
243         unsigned long next;
244         unsigned long end;
245         unsigned long pfn;
246         pgprot_t prot;
247 
248         /*
249          * The set_ptes() spec guarantees that when nr > 1, the initial state of
250          * all ptes is not-present. Therefore we never need to unfold or
251          * otherwise invalidate a range before we set the new ptes.
252          * contpte_set_ptes() should never be called for nr < 2.
253          */
254         VM_WARN_ON(nr == 1);
255 
256         if (!mm_is_user(mm))
257                 return __set_ptes(mm, addr, ptep, pte, nr);
258 
259         end = addr + (nr << PAGE_SHIFT);
260         pfn = pte_pfn(pte);
261         prot = pte_pgprot(pte);
262 
263         do {
264                 next = pte_cont_addr_end(addr, end);
265                 nr = (next - addr) >> PAGE_SHIFT;
266                 pte = pfn_pte(pfn, prot);
267 
268                 if (((addr | next | (pfn << PAGE_SHIFT)) & ~CONT_PTE_MASK) == 0)
269                         pte = pte_mkcont(pte);
270                 else
271                         pte = pte_mknoncont(pte);
272 
273                 __set_ptes(mm, addr, ptep, pte, nr);
274 
275                 addr = next;
276                 ptep += nr;
277                 pfn += nr;
278 
279         } while (addr != end);
280 }
281 EXPORT_SYMBOL_GPL(contpte_set_ptes);
282 
283 void contpte_clear_full_ptes(struct mm_struct *mm, unsigned long addr,
284                                 pte_t *ptep, unsigned int nr, int full)
285 {
286         contpte_try_unfold_partial(mm, addr, ptep, nr);
287         __clear_full_ptes(mm, addr, ptep, nr, full);
288 }
289 EXPORT_SYMBOL_GPL(contpte_clear_full_ptes);
290 
291 pte_t contpte_get_and_clear_full_ptes(struct mm_struct *mm,
292                                 unsigned long addr, pte_t *ptep,
293                                 unsigned int nr, int full)
294 {
295         contpte_try_unfold_partial(mm, addr, ptep, nr);
296         return __get_and_clear_full_ptes(mm, addr, ptep, nr, full);
297 }
298 EXPORT_SYMBOL_GPL(contpte_get_and_clear_full_ptes);
299 
300 int contpte_ptep_test_and_clear_young(struct vm_area_struct *vma,
301                                         unsigned long addr, pte_t *ptep)
302 {
303         /*
304          * ptep_clear_flush_young() technically requires us to clear the access
305          * flag for a _single_ pte. However, the core-mm code actually tracks
306          * access/dirty per folio, not per page. And since we only create a
307          * contig range when the range is covered by a single folio, we can get
308          * away with clearing young for the whole contig range here, so we avoid
309          * having to unfold.
310          */
311 
312         int young = 0;
313         int i;
314 
315         ptep = contpte_align_down(ptep);
316         addr = ALIGN_DOWN(addr, CONT_PTE_SIZE);
317 
318         for (i = 0; i < CONT_PTES; i++, ptep++, addr += PAGE_SIZE)
319                 young |= __ptep_test_and_clear_young(vma, addr, ptep);
320 
321         return young;
322 }
323 EXPORT_SYMBOL_GPL(contpte_ptep_test_and_clear_young);
324 
325 int contpte_ptep_clear_flush_young(struct vm_area_struct *vma,
326                                         unsigned long addr, pte_t *ptep)
327 {
328         int young;
329 
330         young = contpte_ptep_test_and_clear_young(vma, addr, ptep);
331 
332         if (young) {
333                 /*
334                  * See comment in __ptep_clear_flush_young(); same rationale for
335                  * eliding the trailing DSB applies here.
336                  */
337                 addr = ALIGN_DOWN(addr, CONT_PTE_SIZE);
338                 __flush_tlb_range_nosync(vma, addr, addr + CONT_PTE_SIZE,
339                                          PAGE_SIZE, true, 3);
340         }
341 
342         return young;
343 }
344 EXPORT_SYMBOL_GPL(contpte_ptep_clear_flush_young);
345 
346 void contpte_wrprotect_ptes(struct mm_struct *mm, unsigned long addr,
347                                         pte_t *ptep, unsigned int nr)
348 {
349         /*
350          * If wrprotecting an entire contig range, we can avoid unfolding. Just
351          * set wrprotect and wait for the later mmu_gather flush to invalidate
352          * the tlb. Until the flush, the page may or may not be wrprotected.
353          * After the flush, it is guaranteed wrprotected. If it's a partial
354          * range though, we must unfold, because we can't have a case where
355          * CONT_PTE is set but wrprotect applies to a subset of the PTEs; this
356          * would cause it to continue to be unpredictable after the flush.
357          */
358 
359         contpte_try_unfold_partial(mm, addr, ptep, nr);
360         __wrprotect_ptes(mm, addr, ptep, nr);
361 }
362 EXPORT_SYMBOL_GPL(contpte_wrprotect_ptes);
363 
364 void contpte_clear_young_dirty_ptes(struct vm_area_struct *vma,
365                                     unsigned long addr, pte_t *ptep,
366                                     unsigned int nr, cydp_t flags)
367 {
368         /*
369          * We can safely clear access/dirty without needing to unfold from
370          * the architectures perspective, even when contpte is set. If the
371          * range starts or ends midway through a contpte block, we can just
372          * expand to include the full contpte block. While this is not
373          * exactly what the core-mm asked for, it tracks access/dirty per
374          * folio, not per page. And since we only create a contpte block
375          * when it is covered by a single folio, we can get away with
376          * clearing access/dirty for the whole block.
377          */
378         unsigned long start = addr;
379         unsigned long end = start + nr * PAGE_SIZE;
380 
381         if (pte_cont(__ptep_get(ptep + nr - 1)))
382                 end = ALIGN(end, CONT_PTE_SIZE);
383 
384         if (pte_cont(__ptep_get(ptep))) {
385                 start = ALIGN_DOWN(start, CONT_PTE_SIZE);
386                 ptep = contpte_align_down(ptep);
387         }
388 
389         __clear_young_dirty_ptes(vma, start, ptep, (end - start) / PAGE_SIZE, flags);
390 }
391 EXPORT_SYMBOL_GPL(contpte_clear_young_dirty_ptes);
392 
393 int contpte_ptep_set_access_flags(struct vm_area_struct *vma,
394                                         unsigned long addr, pte_t *ptep,
395                                         pte_t entry, int dirty)
396 {
397         unsigned long start_addr;
398         pte_t orig_pte;
399         int i;
400 
401         /*
402          * Gather the access/dirty bits for the contiguous range. If nothing has
403          * changed, its a noop.
404          */
405         orig_pte = pte_mknoncont(ptep_get(ptep));
406         if (pte_val(orig_pte) == pte_val(entry))
407                 return 0;
408 
409         /*
410          * We can fix up access/dirty bits without having to unfold the contig
411          * range. But if the write bit is changing, we must unfold.
412          */
413         if (pte_write(orig_pte) == pte_write(entry)) {
414                 /*
415                  * For HW access management, we technically only need to update
416                  * the flag on a single pte in the range. But for SW access
417                  * management, we need to update all the ptes to prevent extra
418                  * faults. Avoid per-page tlb flush in __ptep_set_access_flags()
419                  * and instead flush the whole range at the end.
420                  */
421                 ptep = contpte_align_down(ptep);
422                 start_addr = addr = ALIGN_DOWN(addr, CONT_PTE_SIZE);
423 
424                 for (i = 0; i < CONT_PTES; i++, ptep++, addr += PAGE_SIZE)
425                         __ptep_set_access_flags(vma, addr, ptep, entry, 0);
426 
427                 if (dirty)
428                         __flush_tlb_range(vma, start_addr, addr,
429                                                         PAGE_SIZE, true, 3);
430         } else {
431                 __contpte_try_unfold(vma->vm_mm, addr, ptep, orig_pte);
432                 __ptep_set_access_flags(vma, addr, ptep, entry, dirty);
433         }
434 
435         return 1;
436 }
437 EXPORT_SYMBOL_GPL(contpte_ptep_set_access_flags);
438 

~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~

kernel.org | git.kernel.org | LWN.net | Project Home | SVN repository | Mail admin

Linux® is a registered trademark of Linus Torvalds in the United States and other countries.
TOMOYO® is a registered trademark of NTT DATA CORPORATION.

sflogo.php