VirtualBox

source: vbox/trunk/src/VBox/Runtime/r0drv/solaris/alloc-r0drv-solaris.c

Last change on this file was 98103, checked in by vboxsync, 17 months ago

Copyright year updates by scm.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 6.5 KB
Line 
1/* $Id: alloc-r0drv-solaris.c 98103 2023-01-17 14:15:46Z vboxsync $ */
2/** @file
3 * IPRT - Memory Allocation, Ring-0 Driver, Solaris.
4 */
5
6/*
7 * Copyright (C) 2006-2023 Oracle and/or its affiliates.
8 *
9 * This file is part of VirtualBox base platform packages, as
10 * available from https://www.virtualbox.org.
11 *
12 * This program is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU General Public License
14 * as published by the Free Software Foundation, in version 3 of the
15 * License.
16 *
17 * This program is distributed in the hope that it will be useful, but
18 * WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
20 * General Public License for more details.
21 *
22 * You should have received a copy of the GNU General Public License
23 * along with this program; if not, see <https://www.gnu.org/licenses>.
24 *
25 * The contents of this file may alternatively be used under the terms
26 * of the Common Development and Distribution License Version 1.0
27 * (CDDL), a copy of it is provided in the "COPYING.CDDL" file included
28 * in the VirtualBox distribution, in which case the provisions of the
29 * CDDL are applicable instead of those of the GPL.
30 *
31 * You may elect to license modified versions of this file under the
32 * terms and conditions of either the GPL or the CDDL or both.
33 *
34 * SPDX-License-Identifier: GPL-3.0-only OR CDDL-1.0
35 */
36
37
38/*********************************************************************************************************************************
39* Header Files *
40*********************************************************************************************************************************/
41#include "the-solaris-kernel.h"
42#include "internal/iprt.h"
43#include <iprt/mem.h>
44
45#include <iprt/assert.h>
46#include <iprt/errcore.h>
47#include <iprt/log.h>
48#include <iprt/param.h>
49#include <iprt/thread.h>
50#include "r0drv/alloc-r0drv.h"
51
52
53/*********************************************************************************************************************************
54* Structures and Typedefs *
55*********************************************************************************************************************************/
56static ddi_dma_attr_t s_rtR0SolDmaAttr =
57{
58 DMA_ATTR_V0, /* Version Number */
59 (uint64_t)0, /* Lower limit */
60 (uint64_t)0, /* High limit */
61 (uint64_t)0xffffffff, /* Counter limit */
62 (uint64_t)PAGESIZE, /* Alignment */
63 (uint64_t)PAGESIZE, /* Burst size */
64 (uint64_t)PAGESIZE, /* Effective DMA size */
65 (uint64_t)0xffffffff, /* Max DMA xfer size */
66 (uint64_t)0xffffffff, /* Segment boundary */
67 1, /* Scatter-gather list length (1 for contiguous) */
68 1, /* Device granularity */
69 0 /* Bus-specific flags */
70};
71
72extern void *contig_alloc(size_t cb, ddi_dma_attr_t *pDmaAttr, size_t uAlign, int fCanSleep);
73
74
75/**
76 * OS specific allocation function.
77 */
78DECLHIDDEN(int) rtR0MemAllocEx(size_t cb, uint32_t fFlags, PRTMEMHDR *ppHdr)
79{
80 size_t cbAllocated = cb;
81 PRTMEMHDR pHdr;
82 unsigned fKmFlags = fFlags & RTMEMHDR_FLAG_ANY_CTX_ALLOC ? KM_NOSLEEP : KM_SLEEP;
83 if (fFlags & RTMEMHDR_FLAG_ZEROED)
84 pHdr = (PRTMEMHDR)kmem_zalloc(cb + sizeof(*pHdr), fKmFlags);
85 else
86 pHdr = (PRTMEMHDR)kmem_alloc(cb + sizeof(*pHdr), fKmFlags);
87 if (RT_UNLIKELY(!pHdr))
88 {
89 LogRel(("rtMemAllocEx(%u, %#x) failed\n", (unsigned)cb + sizeof(*pHdr), fFlags));
90 return VERR_NO_MEMORY;
91 }
92
93 pHdr->u32Magic = RTMEMHDR_MAGIC;
94 pHdr->fFlags = fFlags;
95 pHdr->cb = cbAllocated;
96 pHdr->cbReq = cb;
97
98 *ppHdr = pHdr;
99 return VINF_SUCCESS;
100}
101
102
103/**
104 * OS specific free function.
105 */
106DECLHIDDEN(void) rtR0MemFree(PRTMEMHDR pHdr)
107{
108 pHdr->u32Magic += 1;
109 kmem_free(pHdr, pHdr->cb + sizeof(*pHdr));
110}
111
112
113/**
114 * Allocates physical memory which satisfy the given constraints.
115 *
116 * @param uPhysHi The upper physical address limit (inclusive).
117 * @param puPhys Where to store the physical address of the allocated
118 * memory. Optional, can be NULL.
119 * @param cb Size of allocation.
120 * @param uAlignment Alignment.
121 * @param fContig Whether the memory must be physically contiguous or
122 * not.
123 * @returns Virtual address of allocated memory block or NULL if allocation
124 * failed.
125 */
126DECLHIDDEN(void *) rtR0SolMemAlloc(uint64_t uPhysHi, uint64_t *puPhys, size_t cb, uint64_t uAlignment, bool fContig)
127{
128 if ((cb & PAGEOFFSET) != 0)
129 return NULL;
130
131 size_t cPages = (cb + PAGESIZE - 1) >> PAGESHIFT;
132 if (!cPages)
133 return NULL;
134
135 ddi_dma_attr_t DmaAttr = s_rtR0SolDmaAttr;
136 DmaAttr.dma_attr_addr_hi = uPhysHi;
137 DmaAttr.dma_attr_align = uAlignment;
138 if (!fContig)
139 DmaAttr.dma_attr_sgllen = cPages > INT_MAX ? INT_MAX - 1 : cPages;
140 else
141 AssertRelease(DmaAttr.dma_attr_sgllen == 1);
142
143 void *pvMem = contig_alloc(cb, &DmaAttr, PAGESIZE, 1 /* can sleep */);
144 if (!pvMem)
145 {
146 LogRel(("rtR0SolMemAlloc failed. cb=%u Align=%u fContig=%d\n", (unsigned)cb, (unsigned)uAlignment, fContig));
147 return NULL;
148 }
149
150 pfn_t PageFrameNum = hat_getpfnum(kas.a_hat, (caddr_t)pvMem);
151 AssertRelease(PageFrameNum != PFN_INVALID);
152 if (puPhys)
153 *puPhys = (uint64_t)PageFrameNum << PAGESHIFT;
154
155 return pvMem;
156}
157
158
159/**
160 * Frees memory allocated using rtR0SolMemAlloc().
161 *
162 * @param pv The memory to free.
163 * @param cb Size of the memory block
164 */
165DECLHIDDEN(void) rtR0SolMemFree(void *pv, size_t cb)
166{
167 if (RT_LIKELY(pv))
168 g_pfnrtR0Sol_contig_free(pv, cb);
169}
170
171
172RTR0DECL(void *) RTMemContAlloc(PRTCCPHYS pPhys, size_t cb)
173{
174 AssertPtrReturn(pPhys, NULL);
175 AssertReturn(cb > 0, NULL);
176 RT_ASSERT_PREEMPTIBLE();
177
178 /* Allocate physically contiguous (< 4GB) page-aligned memory. */
179 uint64_t uPhys;
180 void *pvMem = rtR0SolMemAlloc((uint64_t)_4G - 1, &uPhys, cb, PAGESIZE, true /* fContig */);
181 if (RT_UNLIKELY(!pvMem))
182 {
183 LogRel(("RTMemContAlloc failed to allocate %u bytes\n", cb));
184 return NULL;
185 }
186
187 Assert(uPhys < _4G);
188 *pPhys = uPhys;
189 return pvMem;
190}
191
192
193RTR0DECL(void) RTMemContFree(void *pv, size_t cb)
194{
195 RT_ASSERT_PREEMPTIBLE();
196 rtR0SolMemFree(pv, cb);
197}
198
Note: See TracBrowser for help on using the repository browser.

© 2023 Oracle
ContactPrivacy policyTerms of Use