VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/posix/alloc-posix.cpp@ 2981

Last change on this file since 2981 was 2981, checked in by vboxsync, 18 years ago

InnoTek -> innotek: all the headers and comments.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 7.4 KB
Line 
1/* $Id: alloc-posix.cpp 2981 2007-06-01 16:01:28Z vboxsync $ */
2/** @file
3 * innotek Portable Runtime - Memory Allocation, POSIX.
4 */
5
6/*
7 * Copyright (C) 2006-2007 innotek GmbH
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 as published by the Free Software Foundation,
13 * in version 2 as it comes in the "COPYING" file of the VirtualBox OSE
14 * distribution. VirtualBox OSE is distributed in the hope that it will
15 * be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * If you received this file as part of a commercial VirtualBox
18 * distribution, then only the terms of your commercial VirtualBox
19 * license agreement apply instead of the previous paragraph.
20 */
21
22
23/*******************************************************************************
24* Header Files *
25*******************************************************************************/
26#include <iprt/alloc.h>
27#include <iprt/assert.h>
28#include <iprt/param.h>
29#include <iprt/err.h>
30#include <iprt/string.h>
31
32#include <stdlib.h>
33#include <malloc.h>
34#include <errno.h>
35#include <sys/mman.h>
36
37#if !defined(RT_USE_MMAP) && (defined(__LINUX__))
38# define RT_USE_MMAP
39#endif
40
41/*******************************************************************************
42* Structures and Typedefs *
43*******************************************************************************/
44#ifdef RT_USE_MMAP
45/**
46 * RTMemExecAlloc() header used when using mmap for allocating the memory.
47 */
48typedef struct RTMEMEXECHDR
49{
50 /** Magic number (RTMEMEXECHDR_MAGIC). */
51 size_t uMagic;
52 /** The size we requested from mmap. */
53 size_t cb;
54# if ARCH_BITS == 32
55 uint32_t Alignment[2];
56# endif
57} RTMEMEXECHDR, *PRTMEMEXECHDR;
58
59/** Magic for RTMEMEXECHDR. */
60#define RTMEMEXECHDR_MAGIC (~(size_t)0xfeedbabe)
61
62#endif /* RT_USE_MMAP */
63
64
65
66#ifdef IN_RING3
67
68/**
69 * Allocates memory which may contain code.
70 *
71 * @returns Pointer to the allocated memory.
72 * @returns NULL on failure.
73 * @param cb Size in bytes of the memory block to allocate.
74 */
75RTDECL(void *) RTMemExecAlloc(size_t cb)
76{
77 AssertMsg(cb, ("Allocating ZERO bytes is really not a good idea! Good luck with the next assertion!\n"));
78
79#ifdef RT_USE_MMAP
80 /*
81 * Use mmap to get low memory.
82 */
83 size_t cbAlloc = RT_ALIGN_Z(cb + sizeof(RTMEMEXECHDR), PAGE_SIZE);
84 void *pv = mmap(NULL, cbAlloc, PROT_READ | PROT_WRITE | PROT_EXEC, MAP_PRIVATE | MAP_ANONYMOUS
85#if defined(__AMD64__) && defined(MAP_32BIT)
86 | MAP_32BIT
87#endif
88 , -1, 0);
89 AssertMsgReturn(pv != MAP_FAILED, ("errno=%d cb=%#zx\n", errno, cb), NULL);
90 PRTMEMEXECHDR pHdr = (PRTMEMEXECHDR)pv;
91 pHdr->uMagic = RTMEMEXECHDR_MAGIC;
92 pHdr->cb = cbAlloc;
93 pv = pHdr + 1;
94
95#else
96 /*
97 * Allocate first.
98 */
99 cb = RT_ALIGN_Z(cb, 32);
100 void *pv = NULL;
101 int rc = posix_memalign(&pv, 32, cb);
102 AssertMsg(!rc && pv, ("posix_memalign(%zd) failed!!! rc=%d\n", cb, rc));
103 if (pv && !rc)
104 {
105 /*
106 * Add PROT_EXEC flag to the page.
107 *
108 * This is in violation of the SuS where I think it saith that mprotect() shall
109 * only be used with mmap()'ed memory. Works on linux and OS/2 LIBC v0.6.
110 */
111 memset(pv, 0xcc, cb);
112 void *pvProt = (void *)((uintptr_t)pv & ~PAGE_OFFSET_MASK);
113 size_t cbProt = ((uintptr_t)pv & PAGE_OFFSET_MASK) + cb;
114 cbProt = RT_ALIGN_Z(cbProt, PAGE_SIZE);
115 rc = mprotect(pvProt, cbProt, PROT_READ | PROT_WRITE | PROT_EXEC);
116 if (rc)
117 {
118 AssertMsgFailed(("mprotect(%p, %#zx,,) -> rc=%d, errno=%d\n", pvProt, cbProt, rc, errno));
119 free(pv);
120 pv = NULL;
121 }
122 }
123#endif
124 return pv;
125}
126
127
128/**
129 * Free executable/read/write memory allocated by RTMemExecAlloc().
130 *
131 * @param pv Pointer to memory block.
132 */
133RTDECL(void) RTMemExecFree(void *pv)
134{
135 if (pv)
136 {
137#ifdef RT_USE_MMAP
138 PRTMEMEXECHDR pHdr = (PRTMEMEXECHDR)pv - 1;
139 AssertMsgReturnVoid(RT_ALIGN_P(pHdr, PAGE_SIZE) == pHdr, ("pHdr=%p pv=%p\n", pHdr, pv));
140 AssertMsgReturnVoid(pHdr->uMagic == RTMEMEXECHDR_MAGIC, ("pHdr=%p(uMagic=%#zx) pv=%p\n", pHdr, pHdr->uMagic, pv));
141 int rc = munmap(pHdr, pHdr->cb);
142 AssertMsg(!rc, ("munmap -> %d errno=%d\n", rc, errno)); NOREF(rc);
143#else
144 free(pv);
145#endif
146 }
147}
148
149
150/**
151 * Allocate page aligned memory.
152 *
153 * @returns Pointer to the allocated memory.
154 * @returns NULL if we're out of memory.
155 * @param cb Size of the memory block. Will be rounded up to page size.
156 */
157RTDECL(void *) RTMemPageAlloc(size_t cb)
158{
159#if 0 /** @todo huh? we're using posix_memalign in the next function... */
160 void *pv;
161 int rc = posix_memalign(&pv, PAGE_SIZE, RT_ALIGN_Z(cb, PAGE_SIZE));
162 if (!rc)
163 return pv;
164 return NULL;
165#else
166 return memalign(PAGE_SIZE, cb);
167#endif
168}
169
170
171/**
172 * Allocate zero'ed page aligned memory.
173 *
174 * @returns Pointer to the allocated memory.
175 * @returns NULL if we're out of memory.
176 * @param cb Size of the memory block. Will be rounded up to page size.
177 */
178RTDECL(void *) RTMemPageAllocZ(size_t cb)
179{
180 void *pv;
181 int rc = posix_memalign(&pv, PAGE_SIZE, RT_ALIGN_Z(cb, PAGE_SIZE));
182 if (!rc)
183 {
184 bzero(pv, RT_ALIGN_Z(cb, PAGE_SIZE));
185 return pv;
186 }
187 return NULL;
188}
189
190
191/**
192 * Free a memory block allocated with RTMemPageAlloc() or RTMemPageAllocZ().
193 *
194 * @param pv Pointer to the block as it was returned by the allocation function.
195 * NULL will be ignored.
196 */
197RTDECL(void) RTMemPageFree(void *pv)
198{
199 if (pv)
200 free(pv);
201}
202
203
204/**
205 * Change the page level protection of a memory region.
206 *
207 * @returns iprt status code.
208 * @param pv Start of the region. Will be rounded down to nearest page boundary.
209 * @param cb Size of the region. Will be rounded up to the nearest page boundary.
210 * @param fProtect The new protection, a combination of the RTMEM_PROT_* defines.
211 */
212RTDECL(int) RTMemProtect(void *pv, size_t cb, unsigned fProtect)
213{
214 /*
215 * Validate input.
216 */
217 if (cb == 0)
218 {
219 AssertMsgFailed(("!cb\n"));
220 return VERR_INVALID_PARAMETER;
221 }
222 if (fProtect & ~(RTMEM_PROT_NONE | RTMEM_PROT_READ | RTMEM_PROT_WRITE | RTMEM_PROT_EXEC))
223 {
224 AssertMsgFailed(("fProtect=%#x\n", fProtect));
225 return VERR_INVALID_PARAMETER;
226 }
227
228 /*
229 * Convert the flags.
230 */
231 int fProt;
232#if RTMEM_PROT_NONE == PROT_NONE \
233 && RTMEM_PROT_READ == PROT_READ \
234 && RTMEM_PROT_WRITE == PROT_WRITE \
235 && RTMEM_PROT_EXEC == PROT_EXEC
236 fProt = fProtect;
237#else
238 Assert(!RTMEM_PROT_NONE);
239 if (!fProtect)
240 fProt = PROT_NONE;
241 else
242 {
243 fProt = 0;
244 if (fProtect & RTMEM_PROT_READ)
245 fProt |= PROT_READ;
246 if (fProtect & RTMEM_PROT_WRITE)
247 fProt |= PROT_WRITE;
248 if (fProtect & RTMEM_PROT_EXEC)
249 fProt |= PROT_EXEC;
250 }
251#endif
252
253 /*
254 * Align the request.
255 */
256 cb += (uintptr_t)pv & PAGE_OFFSET_MASK;
257 pv = (void *)((uintptr_t)pv & ~PAGE_OFFSET_MASK);
258
259 /*
260 * Change the page attributes.
261 */
262 int rc = mprotect(pv, cb, fProt);
263 if (!rc)
264 return rc;
265 return RTErrConvertFromErrno(errno);
266}
267
268#endif
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