aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMark Johnston <markj@FreeBSD.org>2025-12-03 13:41:30 +0000
committerMark Johnston <markj@FreeBSD.org>2025-12-03 13:41:30 +0000
commit20a0a2f8f36f5db704b674106fa61d8d95b6b071 (patch)
tree5300012051df93d4b1f23dfdd9e116354f82ef34
parent35dd53a9e13265f7a479649776453efc5b737a0f (diff)
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
-rw-r--r--sys/riscv/riscv/pmap.c22
1 files 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);
}