VirtualBox

source: vbox/trunk/src/VBox/Runtime/r0drv/linux/alloc-r0drv-linux.c@ 33995

Last change on this file since 33995 was 32708, checked in by vboxsync, 14 years ago

iprt: build fixes

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 9.8 KB
Line 
1/* $Id: alloc-r0drv-linux.c 32708 2010-09-23 11:18:51Z vboxsync $ */
2/** @file
3 * IPRT - Memory Allocation, Ring-0 Driver, Linux.
4 */
5
6/*
7 * Copyright (C) 2006-2010 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/*******************************************************************************
29* Header Files *
30*******************************************************************************/
31#include "the-linux-kernel.h"
32#include "internal/iprt.h"
33#include <iprt/mem.h>
34
35#include <iprt/assert.h>
36#include <iprt/err.h>
37#include "r0drv/alloc-r0drv.h"
38
39#if defined(RT_ARCH_AMD64) || defined(DOXYGEN_RUNNING)
40/**
41 * We need memory in the module range (~2GB to ~0) this can only be obtained
42 * thru APIs that are not exported (see module_alloc()).
43 *
44 * So, we'll have to create a quick and dirty heap here using BSS memory.
45 * Very annoying and it's going to restrict us!
46 */
47# define RTMEMALLOC_EXEC_HEAP
48#endif
49#ifdef RTMEMALLOC_EXEC_HEAP
50# include <iprt/heap.h>
51# include <iprt/spinlock.h>
52# include <iprt/err.h>
53#endif
54
55
56/*******************************************************************************
57* Global Variables *
58*******************************************************************************/
59#ifdef RTMEMALLOC_EXEC_HEAP
60/** The heap. */
61static RTHEAPSIMPLE g_HeapExec = NIL_RTHEAPSIMPLE;
62/** Spinlock protecting the heap. */
63static RTSPINLOCK g_HeapExecSpinlock = NIL_RTSPINLOCK;
64
65
66/**
67 * API for cleaning up the heap spinlock on IPRT termination.
68 * This is as RTMemExecDonate specific to AMD64 Linux/GNU.
69 */
70void rtR0MemExecCleanup(void)
71{
72 RTSpinlockDestroy(g_HeapExecSpinlock);
73 g_HeapExecSpinlock = NIL_RTSPINLOCK;
74}
75
76
77/**
78 * Donate read+write+execute memory to the exec heap.
79 *
80 * This API is specific to AMD64 and Linux/GNU. A kernel module that desires to
81 * use RTMemExecAlloc on AMD64 Linux/GNU will have to donate some statically
82 * allocated memory in the module if it wishes for GCC generated code to work.
83 * GCC can only generate modules that work in the address range ~2GB to ~0
84 * currently.
85 *
86 * The API only accept one single donation.
87 *
88 * @returns IPRT status code.
89 * @param pvMemory Pointer to the memory block.
90 * @param cb The size of the memory block.
91 */
92RTR0DECL(int) RTR0MemExecDonate(void *pvMemory, size_t cb)
93{
94 int rc;
95 AssertReturn(g_HeapExec == NIL_RTHEAPSIMPLE, VERR_WRONG_ORDER);
96
97 rc = RTSpinlockCreate(&g_HeapExecSpinlock);
98 if (RT_SUCCESS(rc))
99 {
100 rc = RTHeapSimpleInit(&g_HeapExec, pvMemory, cb);
101 if (RT_FAILURE(rc))
102 rtR0MemExecCleanup();
103 }
104 return rc;
105}
106RT_EXPORT_SYMBOL(RTR0MemExecDonate);
107#endif /* RTMEMALLOC_EXEC_HEAP */
108
109
110
111/**
112 * OS specific allocation function.
113 */
114int rtR0MemAllocEx(size_t cb, uint32_t fFlags, PRTMEMHDR *ppHdr)
115{
116 PRTMEMHDR pHdr;
117
118 /*
119 * Allocate.
120 */
121 if (fFlags & RTMEMHDR_FLAG_EXEC)
122 {
123 if (fFlags & RTMEMHDR_FLAG_ANY_CTX)
124 return VERR_NOT_SUPPORTED;
125
126#if defined(RT_ARCH_AMD64)
127# ifdef RTMEMALLOC_EXEC_HEAP
128 if (g_HeapExec != NIL_RTHEAPSIMPLE)
129 {
130 RTSPINLOCKTMP SpinlockTmp = RTSPINLOCKTMP_INITIALIZER;
131 RTSpinlockAcquireNoInts(g_HeapExecSpinlock, &SpinlockTmp);
132 pHdr = (PRTMEMHDR)RTHeapSimpleAlloc(g_HeapExec, cb + sizeof(*pHdr), 0);
133 RTSpinlockReleaseNoInts(g_HeapExecSpinlock, &SpinlockTmp);
134 fFlags |= RTMEMHDR_FLAG_EXEC_HEAP;
135 }
136 else
137 pHdr = NULL;
138# else /* !RTMEMALLOC_EXEC_HEAP */
139 pHdr = (PRTMEMHDR)__vmalloc(cb + sizeof(*pHdr), GFP_KERNEL | __GFP_HIGHMEM, MY_PAGE_KERNEL_EXEC);
140# endif /* !RTMEMALLOC_EXEC_HEAP */
141
142#elif defined(PAGE_KERNEL_EXEC) && defined(CONFIG_X86_PAE)
143 pHdr = (PRTMEMHDR)__vmalloc(cb + sizeof(*pHdr), GFP_KERNEL | __GFP_HIGHMEM, MY_PAGE_KERNEL_EXEC);
144#else
145 pHdr = (PRTMEMHDR)vmalloc(cb + sizeof(*pHdr));
146#endif
147 }
148 else
149 {
150 if (cb <= PAGE_SIZE || (fFlags & RTMEMHDR_FLAG_ANY_CTX))
151 {
152 fFlags |= RTMEMHDR_FLAG_KMALLOC;
153 pHdr = kmalloc(cb + sizeof(*pHdr),
154 (fFlags & RTMEMHDR_FLAG_ANY_CTX_ALLOC) ? GFP_ATOMIC : GFP_KERNEL);
155 }
156 else
157 pHdr = vmalloc(cb + sizeof(*pHdr));
158 }
159 if (RT_UNLIKELY(!pHdr))
160 return VERR_NO_MEMORY;
161
162 /*
163 * Initialize.
164 */
165 pHdr->u32Magic = RTMEMHDR_MAGIC;
166 pHdr->fFlags = fFlags;
167 pHdr->cb = cb;
168 pHdr->cbReq = cb;
169
170 *ppHdr = pHdr;
171 return VINF_SUCCESS;
172}
173
174
175/**
176 * OS specific free function.
177 */
178void rtR0MemFree(PRTMEMHDR pHdr)
179{
180 pHdr->u32Magic += 1;
181 if (pHdr->fFlags & RTMEMHDR_FLAG_KMALLOC)
182 kfree(pHdr);
183#ifdef RTMEMALLOC_EXEC_HEAP
184 else if (pHdr->fFlags & RTMEMHDR_FLAG_EXEC_HEAP)
185 {
186 RTSPINLOCKTMP SpinlockTmp = RTSPINLOCKTMP_INITIALIZER;
187 RTSpinlockAcquireNoInts(g_HeapExecSpinlock, &SpinlockTmp);
188 RTHeapSimpleFree(g_HeapExec, pHdr);
189 RTSpinlockReleaseNoInts(g_HeapExecSpinlock, &SpinlockTmp);
190 }
191#endif
192 else
193 vfree(pHdr);
194}
195
196
197/**
198 * Compute order. Some functions allocate 2^order pages.
199 *
200 * @returns order.
201 * @param cPages Number of pages.
202 */
203static int CalcPowerOf2Order(unsigned long cPages)
204{
205 int iOrder;
206 unsigned long cTmp;
207
208 for (iOrder = 0, cTmp = cPages; cTmp >>= 1; ++iOrder)
209 ;
210 if (cPages & ~(1 << iOrder))
211 ++iOrder;
212
213 return iOrder;
214}
215
216
217/**
218 * Allocates physical contiguous memory (below 4GB).
219 * The allocation is page aligned and the content is undefined.
220 *
221 * @returns Pointer to the memory block. This is page aligned.
222 * @param pPhys Where to store the physical address.
223 * @param cb The allocation size in bytes. This is always
224 * rounded up to PAGE_SIZE.
225 */
226RTR0DECL(void *) RTMemContAlloc(PRTCCPHYS pPhys, size_t cb)
227{
228 int cOrder;
229 unsigned cPages;
230 struct page *paPages;
231
232 /*
233 * validate input.
234 */
235 Assert(VALID_PTR(pPhys));
236 Assert(cb > 0);
237
238 /*
239 * Allocate page pointer array.
240 */
241 cb = RT_ALIGN_Z(cb, PAGE_SIZE);
242 cPages = cb >> PAGE_SHIFT;
243 cOrder = CalcPowerOf2Order(cPages);
244#if (defined(RT_ARCH_AMD64) || defined(CONFIG_X86_PAE)) && defined(GFP_DMA32)
245 /* ZONE_DMA32: 0-4GB */
246 paPages = alloc_pages(GFP_DMA32, cOrder);
247 if (!paPages)
248#endif
249#ifdef RT_ARCH_AMD64
250 /* ZONE_DMA; 0-16MB */
251 paPages = alloc_pages(GFP_DMA, cOrder);
252#else
253 /* ZONE_NORMAL: 0-896MB */
254 paPages = alloc_pages(GFP_USER, cOrder);
255#endif
256 if (paPages)
257 {
258 /*
259 * Reserve the pages and mark them executable.
260 */
261 unsigned iPage;
262 for (iPage = 0; iPage < cPages; iPage++)
263 {
264 Assert(!PageHighMem(&paPages[iPage]));
265 if (iPage + 1 < cPages)
266 {
267 AssertMsg( (uintptr_t)phys_to_virt(page_to_phys(&paPages[iPage])) + PAGE_SIZE
268 == (uintptr_t)phys_to_virt(page_to_phys(&paPages[iPage + 1]))
269 && page_to_phys(&paPages[iPage]) + PAGE_SIZE
270 == page_to_phys(&paPages[iPage + 1]),
271 ("iPage=%i cPages=%u [0]=%#llx,%p [1]=%#llx,%p\n", iPage, cPages,
272 (long long)page_to_phys(&paPages[iPage]), phys_to_virt(page_to_phys(&paPages[iPage])),
273 (long long)page_to_phys(&paPages[iPage + 1]), phys_to_virt(page_to_phys(&paPages[iPage + 1])) ));
274 }
275
276 SetPageReserved(&paPages[iPage]);
277#if LINUX_VERSION_CODE > KERNEL_VERSION(2, 4, 20) /** @todo find the exact kernel where change_page_attr was introduced. */
278 MY_SET_PAGES_EXEC(&paPages[iPage], 1);
279#endif
280 }
281 *pPhys = page_to_phys(paPages);
282 return phys_to_virt(page_to_phys(paPages));
283 }
284
285 return NULL;
286}
287RT_EXPORT_SYMBOL(RTMemContAlloc);
288
289
290/**
291 * Frees memory allocated ysing RTMemContAlloc().
292 *
293 * @param pv Pointer to return from RTMemContAlloc().
294 * @param cb The cb parameter passed to RTMemContAlloc().
295 */
296RTR0DECL(void) RTMemContFree(void *pv, size_t cb)
297{
298 if (pv)
299 {
300 int cOrder;
301 unsigned cPages;
302 unsigned iPage;
303 struct page *paPages;
304
305 /* validate */
306 AssertMsg(!((uintptr_t)pv & PAGE_OFFSET_MASK), ("pv=%p\n", pv));
307 Assert(cb > 0);
308
309 /* calc order and get pages */
310 cb = RT_ALIGN_Z(cb, PAGE_SIZE);
311 cPages = cb >> PAGE_SHIFT;
312 cOrder = CalcPowerOf2Order(cPages);
313 paPages = virt_to_page(pv);
314
315 /*
316 * Restore page attributes freeing the pages.
317 */
318 for (iPage = 0; iPage < cPages; iPage++)
319 {
320 ClearPageReserved(&paPages[iPage]);
321#if LINUX_VERSION_CODE > KERNEL_VERSION(2, 4, 20) /** @todo find the exact kernel where change_page_attr was introduced. */
322 MY_SET_PAGES_NOEXEC(&paPages[iPage], 1);
323#endif
324 }
325 __free_pages(paPages, cOrder);
326 }
327}
328RT_EXPORT_SYMBOL(RTMemContFree);
329
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