VirtualBox

source: vbox/trunk/src/VBox/Runtime/r0drv/netbsd/memobj-r0drv-netbsd.c@ 100208

Last change on this file since 100208 was 98103, checked in by vboxsync, 2 years ago

Copyright year updates by scm.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 21.1 KB
Line 
1/* $Id: memobj-r0drv-netbsd.c 98103 2023-01-17 14:15:46Z vboxsync $ */
2/** @file
3 * IPRT - Ring-0 Memory Objects, NetBSD.
4 */
5
6/*
7 * Contributed by knut st. osmundsen, Andriy Gapon, Arto Huusko.
8 *
9 * Copyright (C) 2007-2023 Oracle and/or its affiliates.
10 *
11 * This file is part of VirtualBox base platform packages, as
12 * available from https://www.virtualbox.org.
13 *
14 * This program is free software; you can redistribute it and/or
15 * modify it under the terms of the GNU General Public License
16 * as published by the Free Software Foundation, in version 3 of the
17 * License.
18 *
19 * This program is distributed in the hope that it will be useful, but
20 * WITHOUT ANY WARRANTY; without even the implied warranty of
21 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
22 * General Public License for more details.
23 *
24 * You should have received a copy of the GNU General Public License
25 * along with this program; if not, see <https://www.gnu.org/licenses>.
26 *
27 * The contents of this file may alternatively be used under the terms
28 * of the Common Development and Distribution License Version 1.0
29 * (CDDL), a copy of it is provided in the "COPYING.CDDL" file included
30 * in the VirtualBox distribution, in which case the provisions of the
31 * CDDL are applicable instead of those of the GPL.
32 *
33 * You may elect to license modified versions of this file under the
34 * terms and conditions of either the GPL or the CDDL or both.
35 *
36 * SPDX-License-Identifier: GPL-3.0-only OR CDDL-1.0
37 * --------------------------------------------------------------------
38 *
39 * Copyright (c) 2007 knut st. osmundsen <[email protected]>
40 * Copyright (c) 2011 Andriy Gapon <[email protected]>
41 * Copyright (c) 2014 Arto Huusko
42 *
43 * Permission is hereby granted, free of charge, to any person
44 * obtaining a copy of this software and associated documentation
45 * files (the "Software"), to deal in the Software without
46 * restriction, including without limitation the rights to use,
47 * copy, modify, merge, publish, distribute, sublicense, and/or sell
48 * copies of the Software, and to permit persons to whom the
49 * Software is furnished to do so, subject to the following
50 * conditions:
51 *
52 * The above copyright notice and this permission notice shall be
53 * included in all copies or substantial portions of the Software.
54 *
55 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
56 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
57 * OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
58 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
59 * HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
60 * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
61 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
62 * OTHER DEALINGS IN THE SOFTWARE.
63 */
64
65
66/*********************************************************************************************************************************
67* Header Files *
68*********************************************************************************************************************************/
69#include "the-netbsd-kernel.h"
70
71#include <iprt/memobj.h>
72#include <iprt/mem.h>
73#include <iprt/err.h>
74#include <iprt/assert.h>
75#include <iprt/log.h>
76#include <iprt/param.h>
77#include <iprt/process.h>
78#include "internal/memobj.h"
79
80
81/*********************************************************************************************************************************
82* Structures and Typedefs *
83*********************************************************************************************************************************/
84/**
85 * The NetBSD version of the memory object structure.
86 */
87typedef struct RTR0MEMOBJNETBSD
88{
89 /** The core structure. */
90 RTR0MEMOBJINTERNAL Core;
91 size_t size;
92 struct pglist pglist;
93} RTR0MEMOBJNETBSD, *PRTR0MEMOBJNETBSD;
94
95
96typedef struct vm_map* vm_map_t;
97
98/**
99 * Gets the virtual memory map the specified object is mapped into.
100 *
101 * @returns VM map handle on success, NULL if no map.
102 * @param pMem The memory object.
103 */
104static vm_map_t rtR0MemObjNetBSDGetMap(PRTR0MEMOBJINTERNAL pMem)
105{
106 switch (pMem->enmType)
107 {
108 case RTR0MEMOBJTYPE_PAGE:
109 case RTR0MEMOBJTYPE_LOW:
110 case RTR0MEMOBJTYPE_CONT:
111 return kernel_map;
112
113 case RTR0MEMOBJTYPE_PHYS:
114 case RTR0MEMOBJTYPE_PHYS_NC:
115 return NULL; /* pretend these have no mapping atm. */
116
117 case RTR0MEMOBJTYPE_LOCK:
118 return pMem->u.Lock.R0Process == NIL_RTR0PROCESS
119 ? kernel_map
120 : &((struct proc *)pMem->u.Lock.R0Process)->p_vmspace->vm_map;
121
122 case RTR0MEMOBJTYPE_RES_VIRT:
123 return pMem->u.ResVirt.R0Process == NIL_RTR0PROCESS
124 ? kernel_map
125 : &((struct proc *)pMem->u.ResVirt.R0Process)->p_vmspace->vm_map;
126
127 case RTR0MEMOBJTYPE_MAPPING:
128 return pMem->u.Mapping.R0Process == NIL_RTR0PROCESS
129 ? kernel_map
130 : &((struct proc *)pMem->u.Mapping.R0Process)->p_vmspace->vm_map;
131
132 default:
133 return NULL;
134 }
135}
136
137
138DECLHIDDEN(int) rtR0MemObjNativeFree(RTR0MEMOBJ pMem)
139{
140 PRTR0MEMOBJNETBSD pMemNetBSD = (PRTR0MEMOBJNETBSD)pMem;
141 int rc;
142
143 switch (pMemNetBSD->Core.enmType)
144 {
145 case RTR0MEMOBJTYPE_PAGE:
146 {
147 kmem_free(pMemNetBSD->Core.pv, pMemNetBSD->Core.cb);
148 break;
149 }
150 case RTR0MEMOBJTYPE_LOW:
151 case RTR0MEMOBJTYPE_CONT:
152 {
153 /* Unmap */
154 pmap_kremove((vaddr_t)pMemNetBSD->Core.pv, pMemNetBSD->Core.cb);
155 /* Free the virtual space */
156 uvm_km_free(kernel_map, (vaddr_t)pMemNetBSD->Core.pv, pMemNetBSD->Core.cb, UVM_KMF_VAONLY);
157 /* Free the physical pages */
158 uvm_pglistfree(&pMemNetBSD->pglist);
159 break;
160 }
161 case RTR0MEMOBJTYPE_PHYS:
162 case RTR0MEMOBJTYPE_PHYS_NC:
163 {
164 /* Free the physical pages */
165 uvm_pglistfree(&pMemNetBSD->pglist);
166 break;
167 }
168 case RTR0MEMOBJTYPE_LOCK:
169 if (pMemNetBSD->Core.u.Lock.R0Process != NIL_RTR0PROCESS)
170 {
171 uvm_map_pageable(
172 &((struct proc *)pMemNetBSD->Core.u.Lock.R0Process)->p_vmspace->vm_map,
173 (vaddr_t)pMemNetBSD->Core.pv,
174 ((vaddr_t)pMemNetBSD->Core.pv) + pMemNetBSD->Core.cb,
175 1, 0);
176 }
177 break;
178 case RTR0MEMOBJTYPE_RES_VIRT:
179 if (pMemNetBSD->Core.u.Lock.R0Process == NIL_RTR0PROCESS)
180 {
181 uvm_km_free(kernel_map, (vaddr_t)pMemNetBSD->Core.pv, pMemNetBSD->Core.cb, UVM_KMF_VAONLY);
182 }
183 break;
184 case RTR0MEMOBJTYPE_MAPPING:
185 if (pMemNetBSD->Core.u.Lock.R0Process == NIL_RTR0PROCESS)
186 {
187 pmap_kremove((vaddr_t)pMemNetBSD->Core.pv, pMemNetBSD->Core.cb);
188 uvm_km_free(kernel_map, (vaddr_t)pMemNetBSD->Core.pv, pMemNetBSD->Core.cb, UVM_KMF_VAONLY);
189 }
190 break;
191
192 default:
193 AssertMsgFailed(("enmType=%d\n", pMemNetBSD->Core.enmType));
194 return VERR_INTERNAL_ERROR;
195 }
196
197 return VINF_SUCCESS;
198}
199
200static int rtR0MemObjNetBSDAllocHelper(PRTR0MEMOBJNETBSD pMemNetBSD, size_t cb, bool fExecutable,
201 paddr_t VmPhysAddrHigh, bool fContiguous)
202{
203 /* Virtual space first */
204 vaddr_t virt = uvm_km_alloc(kernel_map, cb, 0, UVM_KMF_VAONLY | UVM_KMF_WAITVA | UVM_KMF_CANFAIL);
205 if (virt == 0)
206 return VERR_NO_MEMORY;
207
208 struct pglist *rlist = &pMemNetBSD->pglist;
209
210 int nsegs = fContiguous ? 1 : INT_MAX;
211
212 /* Physical pages */
213 if (uvm_pglistalloc(cb, 0, VmPhysAddrHigh, PAGE_SIZE, 0, rlist, nsegs, 1) != 0)
214 {
215 uvm_km_free(kernel_map, virt, cb, UVM_KMF_VAONLY);
216 return VERR_NO_MEMORY; /** @todo inaccurate status code */
217 }
218
219 /* Map */
220 struct vm_page *page;
221 vm_prot_t prot = VM_PROT_READ | VM_PROT_WRITE;
222 if (fExecutable)
223 prot |= VM_PROT_EXECUTE;
224 vaddr_t virt2 = virt;
225 TAILQ_FOREACH(page, rlist, pageq.queue)
226 {
227 pmap_kenter_pa(virt2, VM_PAGE_TO_PHYS(page), prot, 0);
228 virt2 += PAGE_SIZE;
229 }
230
231 pMemNetBSD->Core.fFlags |= RTR0MEMOBJ_FLAGS_UNINITIALIZED_AT_ALLOC; /*?*/
232 pMemNetBSD->Core.pv = (void *)virt;
233 if (fContiguous)
234 {
235 page = TAILQ_FIRST(rlist);
236 pMemNetBSD->Core.u.Cont.Phys = VM_PAGE_TO_PHYS(page);
237 }
238 return VINF_SUCCESS;
239}
240
241
242DECLHIDDEN(int) rtR0MemObjNativeAllocPage(PPRTR0MEMOBJINTERNAL ppMem, size_t cb, bool fExecutable, const char *pszTag)
243{
244 PRTR0MEMOBJNETBSD pMemNetBSD = (PRTR0MEMOBJNETBSD)rtR0MemObjNew(sizeof(*pMemNetBSD), RTR0MEMOBJTYPE_PAGE, NULL, cb, pszTag);
245 if (pMemNetBSD)
246 {
247 void *pvMem = kmem_alloc(cb, KM_SLEEP);
248 if (pvMem)
249 {
250 if (fExecutable)
251 pmap_protect(pmap_kernel(), (vaddr_t)pvMem, (vaddr_t)pvMem + cb, VM_PROT_READ | VM_PROT_WRITE | VM_PROT_EXECUTE);
252
253 pMemNetBSD->Core.fFlags |= RTR0MEMOBJ_FLAGS_UNINITIALIZED_AT_ALLOC;
254 pMemNetBSD->Core.pv = pvMem;
255 *ppMem = &pMemNetBSD->Core;
256 return VINF_SUCCESS;
257 }
258 rtR0MemObjDelete(&pMemNetBSD->Core);
259 return VERR_NO_PAGE_MEMORY;
260 }
261 return VERR_NO_MEMORY;
262}
263
264
265DECLHIDDEN(int) rtR0MemObjNativeAllocLarge(PPRTR0MEMOBJINTERNAL ppMem, size_t cb, size_t cbLargePage, uint32_t fFlags,
266 const char *pszTag)
267{
268 return rtR0MemObjFallbackAllocLarge(ppMem, cb, cbLargePage, fFlags, pszTag);
269}
270
271
272DECLHIDDEN(int) rtR0MemObjNativeAllocLow(PPRTR0MEMOBJINTERNAL ppMem, size_t cb, bool fExecutable, const char *pszTag)
273{
274 PRTR0MEMOBJNETBSD pMemNetBSD = (PRTR0MEMOBJNETBSD)rtR0MemObjNew(sizeof(*pMemNetBSD), RTR0MEMOBJTYPE_LOW, NULL, cb, pszTag);
275 if (pMemNetBSD)
276 {
277 int rc = rtR0MemObjNetBSDAllocHelper(pMemNetBSD, cb, fExecutable, _4G - 1, false /*fContiguous*/);
278 if (RT_SUCCESS(rc))
279 {
280 *ppMem = &pMemNetBSD->Core;
281 return VINF_SUCCESS;
282 }
283 rtR0MemObjDelete(&pMemNetBSD->Core);
284 return rc;
285 }
286 return VERR_NO_MEMORY;
287}
288
289
290DECLHIDDEN(int) rtR0MemObjNativeAllocCont(PPRTR0MEMOBJINTERNAL ppMem, size_t cb, bool fExecutable, const char *pszTag)
291{
292 PRTR0MEMOBJNETBSD pMemNetBSD = (PRTR0MEMOBJNETBSD)rtR0MemObjNew(sizeof(*pMemNetBSD), RTR0MEMOBJTYPE_CONT, NULL, cb, pszTag);
293 if (pMemNetBSD)
294 {
295 int rc = rtR0MemObjNetBSDAllocHelper(pMemNetBSD, cb, fExecutable, _4G - 1, true /*fContiguous*/);
296 if (RT_SUCCESS(rc))
297 {
298 *ppMem = &pMemNetBSD->Core;
299 return VINF_SUCCESS;
300 }
301 rtR0MemObjDelete(&pMemNetBSD->Core);
302 return rc;
303 }
304 return VERR_NO_MEMORY;
305}
306
307
308static int rtR0MemObjNetBSDAllocPhysPages(PPRTR0MEMOBJINTERNAL ppMem, RTR0MEMOBJTYPE enmType, size_t cb,
309 RTHCPHYS PhysHighest, size_t uAlignment, bool fContiguous, const char *pszTag)
310{
311 /* create the object. */
312 PRTR0MEMOBJNETBSD pMemNetBSD = (PRTR0MEMOBJNETBSD)rtR0MemObjNew(sizeof(*pMemNetBSD), enmType, NULL, cb, pszTag);
313 if (pMemNetBSD)
314 {
315 paddr_t const VmPhysAddrHigh = PhysHighest != NIL_RTHCPHYS ? PhysHighest : ~(paddr_t)0;
316 int const nsegs = fContiguous ? 1 : INT_MAX;
317 int rc = uvm_pglistalloc(cb, 0, VmPhysAddrHigh, uAlignment, 0, &pMemNetBSD->pglist, nsegs, 1);
318 if (!rc)
319 {
320 pMemNetBSD->Core.fFlags |= RTR0MEMOBJ_FLAGS_UNINITIALIZED_AT_ALLOC; /*?*/
321 if (fContiguous)
322 {
323 Assert(enmType == RTR0MEMOBJTYPE_PHYS);
324 const struct vm_page * const pg = TAILQ_FIRST(&pMemNetBSD->pglist);
325 pMemNetBSD->Core.u.Phys.PhysBase = VM_PAGE_TO_PHYS(pg);
326 pMemNetBSD->Core.u.Phys.fAllocated = true;
327 }
328 *ppMem = &pMemNetBSD->Core;
329 return VINF_SUCCESS;
330 }
331 rtR0MemObjDelete(&pMemNetBSD->Core);
332 return VERR_NO_PAGE_MEMORY;
333 }
334 return VERR_NO_MEMORY;
335}
336
337
338DECLHIDDEN(int) rtR0MemObjNativeAllocPhys(PPRTR0MEMOBJINTERNAL ppMem, size_t cb, RTHCPHYS PhysHighest, size_t uAlignment,
339 const char *pszTag)
340{
341 return rtR0MemObjNetBSDAllocPhysPages(ppMem, RTR0MEMOBJTYPE_PHYS, cb, PhysHighest, uAlignment, true, pszTag);
342}
343
344
345DECLHIDDEN(int) rtR0MemObjNativeAllocPhysNC(PPRTR0MEMOBJINTERNAL ppMem, size_t cb, RTHCPHYS PhysHighest, const char *pszTag)
346{
347 return rtR0MemObjNetBSDAllocPhysPages(ppMem, RTR0MEMOBJTYPE_PHYS_NC, cb, PhysHighest, PAGE_SIZE, false, pszTag);
348}
349
350
351DECLHIDDEN(int) rtR0MemObjNativeEnterPhys(PPRTR0MEMOBJINTERNAL ppMem, RTHCPHYS Phys, size_t cb, uint32_t uCachePolicy,
352 const char *pszTag)
353{
354 AssertReturn(uCachePolicy == RTMEM_CACHE_POLICY_DONT_CARE, VERR_NOT_SUPPORTED);
355
356 /* create the object. */
357 PRTR0MEMOBJNETBSD pMemNetBSD = (PRTR0MEMOBJNETBSD)rtR0MemObjNew(sizeof(*pMemNetBSD), RTR0MEMOBJTYPE_PHYS, NULL, cb, pszTag);
358 if (pMemNetBSD)
359 {
360 /* there is no allocation here, it needs to be mapped somewhere first. */
361 pMemNetBSD->Core.u.Phys.fAllocated = false;
362 pMemNetBSD->Core.u.Phys.PhysBase = Phys;
363 pMemNetBSD->Core.u.Phys.uCachePolicy = uCachePolicy;
364 TAILQ_INIT(&pMemNetBSD->pglist);
365 *ppMem = &pMemNetBSD->Core;
366 return VINF_SUCCESS;
367 }
368 return VERR_NO_MEMORY;
369}
370
371
372DECLHIDDEN(int) rtR0MemObjNativeLockUser(PPRTR0MEMOBJINTERNAL ppMem, RTR3PTR R3Ptr, size_t cb, uint32_t fAccess,
373 RTR0PROCESS R0Process, const char *pszTag)
374{
375 PRTR0MEMOBJNETBSD pMemNetBSD = (PRTR0MEMOBJNETBSD)rtR0MemObjNew(sizeof(*pMemNetBSD), RTR0MEMOBJTYPE_LOCK,
376 (void *)R3Ptr, cb, pszTag);
377 if (pMemNetBSD)
378 {
379 int rc = uvm_map_pageable(&((struct proc *)R0Process)->p_vmspace->vm_map, R3Ptr, R3Ptr + cb,
380 0 /*new_pageable*/, 0 /*lockflags*/);
381 if (!rc)
382 {
383 pMemNetBSD->Core.u.Lock.R0Process = R0Process;
384 *ppMem = &pMemNetBSD->Core;
385 return VINF_SUCCESS;
386 }
387 rtR0MemObjDelete(&pMemNetBSD->Core);
388 return VERR_LOCK_FAILED;
389 }
390 return VERR_NO_MEMORY;
391}
392
393
394DECLHIDDEN(int) rtR0MemObjNativeLockKernel(PPRTR0MEMOBJINTERNAL ppMem, void *pv, size_t cb, uint32_t fAccess, const char *pszTag)
395{
396 /* Kernel memory (always?) wired; all memory allocated by vbox code is? */
397 PRTR0MEMOBJNETBSD pMemNetBSD = (PRTR0MEMOBJNETBSD)rtR0MemObjNew(sizeof(*pMemNetBSD), RTR0MEMOBJTYPE_LOCK, pv, cb, pszTag);
398 if (pMemNetBSD)
399 {
400 pMemNetBSD->Core.u.Lock.R0Process = NIL_RTR0PROCESS;
401 pMemNetBSD->Core.pv = pv;
402 *ppMem = &pMemNetBSD->Core;
403 return VINF_SUCCESS;
404 }
405 return VERR_NO_MEMORY;
406}
407
408DECLHIDDEN(int) rtR0MemObjNativeReserveKernel(PPRTR0MEMOBJINTERNAL ppMem, void *pvFixed, size_t cb, size_t uAlignment,
409 const char *pszTag)
410{
411 if (pvFixed != (void *)-1)
412 {
413 /* can we support this? or can we assume the virtual space is already reserved? */
414 printf("reserve specified kernel virtual address not supported\n");
415 return VERR_NOT_SUPPORTED;
416 }
417
418 PRTR0MEMOBJNETBSD pMemNetBSD = (PRTR0MEMOBJNETBSD)rtR0MemObjNew(sizeof(*pMemNetBSD), RTR0MEMOBJTYPE_RES_VIRT,
419 NULL, cb, pszTag);
420 if (pMemNetBSD)
421 {
422 vaddr_t virt = uvm_km_alloc(kernel_map, cb, uAlignment, UVM_KMF_VAONLY | UVM_KMF_WAITVA | UVM_KMF_CANFAIL);
423 if (virt != 0)
424 {
425 pMemNetBSD->Core.u.ResVirt.R0Process = NIL_RTR0PROCESS;
426 pMemNetBSD->Core.pv = (void *)virt;
427 *ppMem = &pMemNetBSD->Core;
428 return VINF_SUCCESS;
429 }
430 rtR0MemObjDelete(&pMemNetBSD->Core);
431 return VERR_NO_MEMORY;
432 }
433 return VERR_NO_MEMORY;
434}
435
436
437DECLHIDDEN(int) rtR0MemObjNativeReserveUser(PPRTR0MEMOBJINTERNAL ppMem, RTR3PTR R3PtrFixed, size_t cb, size_t uAlignment,
438 RTR0PROCESS R0Process, const char *pszTag)
439{
440 RT_NOREF(ppMem, R3PtrFixed, cb, uAlignment, R0Process, pszTag);
441 printf("NativeReserveUser\n");
442 return VERR_NOT_SUPPORTED;
443}
444
445
446DECLHIDDEN(int) rtR0MemObjNativeMapKernel(PPRTR0MEMOBJINTERNAL ppMem, RTR0MEMOBJ pMemToMap, void *pvFixed, size_t uAlignment,
447 unsigned fProt, size_t offSub, size_t cbSub, const char *pszTag)
448{
449 if (pvFixed != (void *)-1)
450 {
451 /* can we support this? or can we assume the virtual space is already reserved? */
452 printf("map to specified kernel virtual address not supported\n");
453 return VERR_NOT_SUPPORTED;
454 }
455
456 PRTR0MEMOBJNETBSD pMemNetBSD0 = (PRTR0MEMOBJNETBSD)pMemToMap;
457 if ((pMemNetBSD0->Core.enmType != RTR0MEMOBJTYPE_PHYS)
458 && (pMemNetBSD0->Core.enmType != RTR0MEMOBJTYPE_PHYS_NC))
459 {
460 printf("memory to map is not physical\n");
461 return VERR_NOT_SUPPORTED;
462 }
463 size_t sz = cbSub > 0 ? cbSub : pMemNetBSD0->Core.cb;
464
465 PRTR0MEMOBJNETBSD pMemNetBSD = (PRTR0MEMOBJNETBSD)rtR0MemObjNew(sizeof(*pMemNetBSD), RTR0MEMOBJTYPE_MAPPING, NULL, sz, pszTag);
466
467 vaddr_t virt = uvm_km_alloc(kernel_map, sz, uAlignment, UVM_KMF_VAONLY | UVM_KMF_WAITVA | UVM_KMF_CANFAIL);
468 if (virt != 0)
469 {
470 vm_prot_t prot = 0;
471 if (fProt & RTMEM_PROT_READ)
472 prot |= VM_PROT_READ;
473 if (fProt & RTMEM_PROT_WRITE)
474 prot |= VM_PROT_WRITE;
475 if (fProt & RTMEM_PROT_EXEC)
476 prot |= VM_PROT_EXECUTE;
477
478 struct vm_page *page;
479 vaddr_t virt2 = virt;
480 size_t map_pos = 0;
481 TAILQ_FOREACH(page, &pMemNetBSD0->pglist, pageq.queue)
482 {
483 if (map_pos >= offSub)
484 {
485 if (cbSub > 0 && (map_pos >= offSub + cbSub))
486 break;
487
488 pmap_kenter_pa(virt2, VM_PAGE_TO_PHYS(page), prot, 0);
489 virt2 += PAGE_SIZE;
490 }
491 map_pos += PAGE_SIZE;
492 }
493
494 pMemNetBSD->Core.pv = (void *)virt;
495 pMemNetBSD->Core.u.Mapping.R0Process = NIL_RTR0PROCESS;
496 *ppMem = &pMemNetBSD->Core;
497 return VINF_SUCCESS;
498 }
499
500 rtR0MemObjDelete(&pMemNetBSD->Core);
501 return VERR_NO_MEMORY;
502}
503
504
505DECLHIDDEN(int) rtR0MemObjNativeMapUser(PPRTR0MEMOBJINTERNAL ppMem, RTR0MEMOBJ pMemToMap, RTR3PTR R3PtrFixed, size_t uAlignment,
506 unsigned fProt, RTR0PROCESS R0Process, size_t offSub, size_t cbSub, const char *pszTag)
507{
508 RT_NOREF(ppMem, pMemToMap, R3PtrFixed, uAlignment, fProt, R0Process, offSub, cbSub, pszTag);
509 printf("NativeMapUser\n");
510 return VERR_NOT_SUPPORTED;
511}
512
513
514DECLHIDDEN(int) rtR0MemObjNativeProtect(PRTR0MEMOBJINTERNAL pMem, size_t offSub, size_t cbSub, uint32_t fProt)
515{
516 vm_map_t const pVmMap = rtR0MemObjNetBSDGetMap(pMem);
517 if (pVmMap)
518 {
519 vaddr_t const AddrStart = (vaddr_t)pMem->pv + offSub;
520 vm_prot_t ProtectionFlags = 0;
521 if (fProt & RTMEM_PROT_READ)
522 ProtectionFlags |= UVM_PROT_R;
523 if (fProt & RTMEM_PROT_WRITE)
524 ProtectionFlags |= UVM_PROT_W;
525 if (fProt & RTMEM_PROT_EXEC)
526 ProtectionFlags |= UVM_PROT_X;
527
528 int rc = uvm_map_protect(pVmMap, AddrStart, AddrStart + cbSub, ProtectionFlags, 0);
529 if (!rc)
530 return VINF_SUCCESS;
531 return RTErrConvertFromErrno(rc);
532 }
533 return VERR_NOT_SUPPORTED;
534}
535
536
537DECLHIDDEN(RTHCPHYS) rtR0MemObjNativeGetPagePhysAddr(PRTR0MEMOBJINTERNAL pMem, size_t iPage)
538{
539 PRTR0MEMOBJNETBSD pMemNetBSD = (PRTR0MEMOBJNETBSD)pMem;
540
541 switch (pMemNetBSD->Core.enmType)
542 {
543 case RTR0MEMOBJTYPE_PAGE:
544 case RTR0MEMOBJTYPE_LOW:
545 {
546 vaddr_t va = (vaddr_t)pMemNetBSD->Core.pv + ptoa(iPage);
547 paddr_t pa = 0;
548 pmap_extract(pmap_kernel(), va, &pa);
549 return pa;
550 }
551 case RTR0MEMOBJTYPE_CONT:
552 return pMemNetBSD->Core.u.Cont.Phys + ptoa(iPage);
553 case RTR0MEMOBJTYPE_PHYS:
554 return pMemNetBSD->Core.u.Phys.PhysBase + ptoa(iPage);
555 case RTR0MEMOBJTYPE_PHYS_NC:
556 {
557 struct vm_page *page;
558 size_t i = 0;
559 TAILQ_FOREACH(page, &pMemNetBSD->pglist, pageq.queue)
560 {
561 if (i == iPage)
562 break;
563 i++;
564 }
565 return VM_PAGE_TO_PHYS(page);
566 }
567 case RTR0MEMOBJTYPE_LOCK:
568 case RTR0MEMOBJTYPE_MAPPING:
569 {
570 pmap_t pmap;
571 if (pMem->u.Lock.R0Process == NIL_RTR0PROCESS)
572 pmap = pmap_kernel();
573 else
574 pmap = ((struct proc *)pMem->u.Lock.R0Process)->p_vmspace->vm_map.pmap;
575 vaddr_t va = (vaddr_t)pMemNetBSD->Core.pv + ptoa(iPage);
576 paddr_t pa = 0;
577 pmap_extract(pmap, va, &pa);
578 return pa;
579 }
580 case RTR0MEMOBJTYPE_RES_VIRT:
581 return NIL_RTHCPHYS;
582 default:
583 return NIL_RTHCPHYS;
584 }
585}
586
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette