The branch main has been updated by markj:

URL: 
https://cgit.FreeBSD.org/src/commit/?id=20a0a2f8f36f5db704b674106fa61d8d95b6b071

commit 20a0a2f8f36f5db704b674106fa61d8d95b6b071
Author:     Mark Johnston <[email protected]>
AuthorDate: 2025-12-03 13:41:30 +0000
Commit:     Mark Johnston <[email protected]>
CommitDate: 2025-12-03 13:41:30 +0000

    riscv/pmap: Handle superpages in pmap_extract_and_hold()
    
    Without this change, vm_fault_quick_hold_pages() falls back to the slow
    fault handler when it encounters a superpage mapping.
    
    Reviewed by:    alc, kib
    Reported and tested by: br
    MFC after:      1 week
    Differential Revision:  https://reviews.freebsd.org/D54022
---
 sys/riscv/riscv/pmap.c | 22 ++++++++++++++++------
 1 file changed, 16 insertions(+), 6 deletions(-)

diff --git a/sys/riscv/riscv/pmap.c b/sys/riscv/riscv/pmap.c
index 26efaecc64d1..0deb8b93a6dc 100644
--- a/sys/riscv/riscv/pmap.c
+++ b/sys/riscv/riscv/pmap.c
@@ -1134,19 +1134,29 @@ pmap_extract(pmap_t pmap, vm_offset_t va)
 vm_page_t
 pmap_extract_and_hold(pmap_t pmap, vm_offset_t va, vm_prot_t prot)
 {
+       pd_entry_t *l2p, l2;
        pt_entry_t *l3p, l3;
        vm_page_t m;
 
        m = NULL;
        PMAP_LOCK(pmap);
-       l3p = pmap_l3(pmap, va);
-       if (l3p != NULL && (l3 = pmap_load(l3p)) != 0) {
-               if ((l3 & PTE_W) != 0 || (prot & VM_PROT_WRITE) == 0) {
-                       m = PTE_TO_VM_PAGE(l3);
-                       if (!vm_page_wire_mapped(m))
-                               m = NULL;
+       l2p = pmap_l2(pmap, va);
+       if (l2p == NULL || ((l2 = pmap_load(l2p)) & PTE_V) == 0) {
+               ;
+       } else if ((l2 & PTE_RWX) != 0) {
+               if ((l2 & PTE_W) != 0 || (prot & VM_PROT_WRITE) == 0) {
+                       m = PHYS_TO_VM_PAGE(L2PTE_TO_PHYS(l2) +
+                           (va & L2_OFFSET));
+               }
+       } else {
+               l3p = pmap_l2_to_l3(l2p, va);
+               if ((l3 = pmap_load(l3p)) != 0) {
+                       if ((l3 & PTE_W) != 0 || (prot & VM_PROT_WRITE) == 0)
+                               m = PTE_TO_VM_PAGE(l3);
                }
        }
+       if (m != NULL && !vm_page_wire_mapped(m))
+               m = NULL;
        PMAP_UNLOCK(pmap);
        return (m);
 }

Reply via email to