VirtualBox

source: vbox/trunk/src/VBox/Runtime/r0drv/solaris/memobj-r0drv-solaris.h@ 62477

Last change on this file since 62477 was 62477, checked in by vboxsync, 9 years ago

(C) 2016

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 8.7 KB
Line 
1/* $Id: memobj-r0drv-solaris.h 62477 2016-07-22 18:27:37Z vboxsync $ */
2/** @file
3 * IPRT - Ring-0 Memory Objects - Segment driver, Solaris.
4 */
5
6/*
7 * Copyright (C) 2012-2016 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28#ifndef ___r0drv_solaris_memobj_r0drv_solaris_h
29#define ___r0drv_solaris_memobj_r0drv_solaris_h
30
31/*******************************************************************************
32* Header Files *
33*******************************************************************************/
34#include "the-solaris-kernel.h"
35
36
37/*******************************************************************************
38* Structures and Typedefs *
39*******************************************************************************/
40typedef struct SEGVBOX_CRARGS
41{
42 uint64_t *paPhysAddrs;
43 size_t cbPageSize;
44 uint_t fPageAccess;
45} SEGVBOX_CRARGS;
46typedef SEGVBOX_CRARGS *PSEGVBOX_CRARGS;
47
48typedef struct SEGVBOX_DATA
49{
50 uint_t fPageAccess;
51 size_t cbPageSize;
52} SEGVBOX_DATA;
53typedef SEGVBOX_DATA *PSEGVBOX_DATA;
54
55static struct seg_ops s_SegVBoxOps;
56static vnode_t s_segVBoxVnode;
57
58
59DECLINLINE(int) rtR0SegVBoxSolCreate(seg_t *pSeg, void *pvArgs)
60{
61 struct as *pAddrSpace = pSeg->s_as;
62 PSEGVBOX_CRARGS pArgs = pvArgs;
63 PSEGVBOX_DATA pData = kmem_zalloc(sizeof(*pData), KM_SLEEP);
64
65 AssertPtr(pAddrSpace);
66 AssertPtr(pArgs);
67 AssertPtr(pData);
68
69 /*
70 * Currently we only map _4K pages but this segment driver can handle any size
71 * supported by the Solaris HAT layer.
72 */
73 size_t cbPageSize = pArgs->cbPageSize;
74 size_t uPageShift = 0;
75 switch (cbPageSize)
76 {
77 case _4K: uPageShift = 12; break;
78 case _2M: uPageShift = 21; break;
79 default: AssertReleaseMsgFailed(("Unsupported page size for mapping cbPageSize=%llx\n", cbPageSize)); break;
80 }
81
82 hat_map(pAddrSpace->a_hat, pSeg->s_base, pSeg->s_size, HAT_MAP);
83 pData->fPageAccess = pArgs->fPageAccess | PROT_USER;
84 pData->cbPageSize = cbPageSize;
85
86 pSeg->s_ops = &s_SegVBoxOps;
87 pSeg->s_data = pData;
88
89 /*
90 * Now load and lock down the mappings to the physical addresses.
91 */
92 caddr_t virtAddr = pSeg->s_base;
93 pgcnt_t cPages = (pSeg->s_size + cbPageSize - 1) >> uPageShift;
94 for (pgcnt_t iPage = 0; iPage < cPages; ++iPage, virtAddr += cbPageSize)
95 {
96 hat_devload(pAddrSpace->a_hat, virtAddr, cbPageSize, pArgs->paPhysAddrs[iPage] >> uPageShift,
97 pData->fPageAccess | HAT_UNORDERED_OK, HAT_LOAD_LOCK);
98 }
99
100 return 0;
101}
102
103
104static int rtR0SegVBoxSolDup(seg_t *pSrcSeg, seg_t *pDstSeg)
105{
106 /*
107 * Duplicate a segment and return the new segment in 'pDstSeg'.
108 */
109 PSEGVBOX_DATA pSrcData = pSrcSeg->s_data;
110 PSEGVBOX_DATA pDstData = kmem_zalloc(sizeof(*pDstData), KM_SLEEP);
111
112 AssertPtr(pDstData);
113 AssertPtr(pSrcData);
114
115 pDstData->fPageAccess = pSrcData->fPageAccess;
116 pDstData->cbPageSize = pSrcData->cbPageSize;
117 pDstSeg->s_ops = &s_SegVBoxOps;
118 pDstSeg->s_data = pDstData;
119
120 return 0;
121}
122
123
124static int rtR0SegVBoxSolUnmap(seg_t *pSeg, caddr_t virtAddr, size_t cb)
125{
126 PSEGVBOX_DATA pData = pSeg->s_data;
127
128 AssertRelease(pData);
129 AssertReleaseMsg(virtAddr >= pSeg->s_base, ("virtAddr=%p s_base=%p\n", virtAddr, pSeg->s_base));
130 AssertReleaseMsg(virtAddr + cb <= pSeg->s_base + pSeg->s_size, ("virtAddr=%p cb=%llu s_base=%p s_size=%llu\n", virtAddr,
131 cb, pSeg->s_base, pSeg->s_size));
132 size_t cbPageOffset = pData->cbPageSize - 1;
133 AssertRelease(!(cb & cbPageOffset));
134 AssertRelease(!((uintptr_t)virtAddr & cbPageOffset));
135
136 if ( virtAddr != pSeg->s_base
137 || cb != pSeg->s_size)
138 {
139 return ENOTSUP;
140 }
141
142 hat_unload(pSeg->s_as->a_hat, virtAddr, cb, HAT_UNLOAD_UNMAP | HAT_UNLOAD_UNLOCK);
143
144 seg_free(pSeg);
145 return 0;
146}
147
148
149static void rtR0SegVBoxSolFree(seg_t *pSeg)
150{
151 PSEGVBOX_DATA pData = pSeg->s_data;
152 kmem_free(pData, sizeof(*pData));
153}
154
155
156static int rtR0SegVBoxSolFault(struct hat *pHat, seg_t *pSeg, caddr_t virtAddr, size_t cb, enum fault_type FaultType,
157 enum seg_rw ReadWrite)
158{
159 /*
160 * We would demand fault if the (u)read() path would SEGOP_FAULT() on buffers mapped in via our
161 * segment driver i.e. prefaults before DMA. Don't fail in such case where we're called directly,
162 * see @bugref{5047}.
163 */
164 return 0;
165}
166
167
168static int rtR0SegVBoxSolFaultA(seg_t *pSeg, caddr_t virtAddr)
169{
170 return 0;
171}
172
173
174static int rtR0SegVBoxSolSetProt(seg_t *pSeg, caddr_t virtAddr, size_t cb, uint_t fPageAccess)
175{
176 return EACCES;
177}
178
179
180static int rtR0SegVBoxSolCheckProt(seg_t *pSeg, caddr_t virtAddr, size_t cb, uint_t fPageAccess)
181{
182 return EINVAL;
183}
184
185
186static int rtR0SegVBoxSolKluster(seg_t *pSeg, caddr_t virtAddr, ssize_t Delta)
187{
188 return -1;
189}
190
191
192static int rtR0SegVBoxSolSync(seg_t *pSeg, caddr_t virtAddr, size_t cb, int Attr, uint_t fFlags)
193{
194 return 0;
195}
196
197
198static size_t rtR0SegVBoxSolInCore(seg_t *pSeg, caddr_t virtAddr, size_t cb, char *pVec)
199{
200 PSEGVBOX_DATA pData = pSeg->s_data;
201 AssertRelease(pData);
202 size_t uPageOffset = pData->cbPageSize - 1;
203 size_t uPageMask = ~uPageOffset;
204 size_t cbLen = (cb + uPageOffset) & uPageMask;
205 for (virtAddr = 0; cbLen != 0; cbLen -= pData->cbPageSize, virtAddr += pData->cbPageSize)
206 *pVec++ = 1;
207 return cbLen;
208}
209
210
211static int rtR0SegVBoxSolLockOp(seg_t *pSeg, caddr_t virtAddr, size_t cb, int Attr, int Op, ulong_t *pLockMap, size_t off)
212{
213 return 0;
214}
215
216
217static int rtR0SegVBoxSolGetProt(seg_t *pSeg, caddr_t virtAddr, size_t cb, uint_t *pafPageAccess)
218{
219 PSEGVBOX_DATA pData = pSeg->s_data;
220 size_t iPage = seg_page(pSeg, virtAddr + cb) - seg_page(pSeg, virtAddr) + 1;
221 if (iPage)
222 {
223 do
224 {
225 iPage--;
226 pafPageAccess[iPage] = pData->fPageAccess;
227 } while (iPage);
228 }
229 return 0;
230}
231
232
233static u_offset_t rtR0SegVBoxSolGetOffset(seg_t *pSeg, caddr_t virtAddr)
234{
235 return ((uintptr_t)virtAddr - (uintptr_t)pSeg->s_base);
236}
237
238
239static int rtR0SegVBoxSolGetType(seg_t *pSeg, caddr_t virtAddr)
240{
241 return MAP_SHARED;
242}
243
244
245static int rtR0SegVBoxSolGetVp(seg_t *pSeg, caddr_t virtAddr, vnode_t **ppVnode)
246{
247 *ppVnode = &s_segVBoxVnode;
248 return 0;
249}
250
251
252static int rtR0SegVBoxSolAdvise(seg_t *pSeg, caddr_t virtAddr, size_t cb, uint_t Behav /* wut? */)
253{
254 return 0;
255}
256
257
258static void rtR0SegVBoxSolDump(seg_t *pSeg)
259{
260 /* Nothing to do. */
261}
262
263
264static int rtR0SegVBoxSolPageLock(seg_t *pSeg, caddr_t virtAddr, size_t cb, page_t ***pppPage, enum lock_type LockType, enum seg_rw ReadWrite)
265{
266 return ENOTSUP;
267}
268
269
270static int rtR0SegVBoxSolSetPageSize(seg_t *pSeg, caddr_t virtAddr, size_t cb, uint_t SizeCode)
271{
272 return ENOTSUP;
273}
274
275
276static int rtR0SegVBoxSolGetMemId(seg_t *pSeg, caddr_t virtAddr, memid_t *pMemId)
277{
278 return ENODEV;
279}
280
281
282static lgrp_mem_policy_info_t *rtR0SegVBoxSolGetPolicy(seg_t *pSeg, caddr_t virtAddr)
283{
284 return NULL;
285}
286
287
288static int rtR0SegVBoxSolCapable(seg_t *pSeg, segcapability_t Capab)
289{
290 return 0;
291}
292
293
294static struct seg_ops s_SegVBoxOps =
295{
296 rtR0SegVBoxSolDup,
297 rtR0SegVBoxSolUnmap,
298 rtR0SegVBoxSolFree,
299 rtR0SegVBoxSolFault,
300 rtR0SegVBoxSolFaultA,
301 rtR0SegVBoxSolSetProt,
302 rtR0SegVBoxSolCheckProt,
303 rtR0SegVBoxSolKluster,
304 NULL, /* swapout */
305 rtR0SegVBoxSolSync,
306 rtR0SegVBoxSolInCore,
307 rtR0SegVBoxSolLockOp,
308 rtR0SegVBoxSolGetProt,
309 rtR0SegVBoxSolGetOffset,
310 rtR0SegVBoxSolGetType,
311 rtR0SegVBoxSolGetVp,
312 rtR0SegVBoxSolAdvise,
313 rtR0SegVBoxSolDump,
314 rtR0SegVBoxSolPageLock,
315 rtR0SegVBoxSolSetPageSize,
316 rtR0SegVBoxSolGetMemId,
317 rtR0SegVBoxSolGetPolicy,
318 rtR0SegVBoxSolCapable
319};
320
321#endif /* !___r0drv_solaris_memobj_r0drv_solaris_h */
322
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