Add an AllocateRW to match AllocateRWX.
[oota-llvm.git] / lib / System / Unix / Memory.inc
1 //===- Unix/Memory.cpp - Generic UNIX System Configuration ------*- C++ -*-===//
2 // 
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 // 
8 //===----------------------------------------------------------------------===//
9 //
10 // This file defines some functions for various memory management utilities.
11 //
12 //===----------------------------------------------------------------------===//
13
14 #include "Unix.h"
15 #include "llvm/System/Process.h"
16
17 #ifdef HAVE_SYS_MMAN_H
18 #include <sys/mman.h>
19 #endif
20
21 /// AllocateRWXMemory - Allocate a slab of memory with read/write/execute
22 /// permissions.  This is typically used for JIT applications where we want
23 /// to emit code to the memory then jump to it.  Getting this type of memory
24 /// is very OS specific.
25 ///
26 llvm::sys::MemoryBlock 
27 llvm::sys::Memory::AllocateRWX(unsigned NumBytes, const MemoryBlock* NearBlock,
28                                std::string *ErrMsg) {
29   if (NumBytes == 0) return MemoryBlock();
30
31   long pageSize = Process::GetPageSize();
32   unsigned NumPages = (NumBytes+pageSize-1)/pageSize;
33
34   int fd = -1;
35 #ifdef NEED_DEV_ZERO_FOR_MMAP
36   static int zero_fd = open("/dev/zero", O_RDWR);
37   if (zero_fd == -1) {
38     MakeErrMsg(ErrMsg, "Can't open /dev/zero device");
39     return MemoryBlock();
40   }
41   fd = zero_fd;
42 #endif
43
44   int flags = MAP_PRIVATE |
45 #ifdef HAVE_MMAP_ANONYMOUS
46   MAP_ANONYMOUS
47 #else
48   MAP_ANON
49 #endif
50   ;
51
52   void* start = NearBlock ? (unsigned char*)NearBlock->base() + 
53                             NearBlock->size() : 0;
54
55   void *pa = ::mmap(start, pageSize*NumPages, PROT_READ|PROT_WRITE|PROT_EXEC,
56                     flags, fd, 0);
57   if (pa == MAP_FAILED) {
58     if (NearBlock) //Try again without a near hint
59       return AllocateRWX(NumBytes, 0);
60
61     MakeErrMsg(ErrMsg, "Can't allocate RWX Memory");
62     return MemoryBlock();
63   }
64   MemoryBlock result;
65   result.Address = pa;
66   result.Size = NumPages*pageSize;
67   return result;
68 }
69
70 /// AllocateRWMemory - Allocate a slab of memory with read/write permissions. 
71 /// This memory needs to have executable permissions set before it can be used
72 /// to execute JIT'ed code.
73 llvm::sys::MemoryBlock 
74 llvm::sys::Memory::AllocateRW(unsigned NumBytes, const MemoryBlock* NearBlock,
75                                std::string *ErrMsg) {
76   if (NumBytes == 0) return MemoryBlock();
77
78   long pageSize = Process::GetPageSize();
79   unsigned NumPages = (NumBytes+pageSize-1)/pageSize;
80
81   int fd = -1;
82 #ifdef NEED_DEV_ZERO_FOR_MMAP
83   static int zero_fd = open("/dev/zero", O_RDWR);
84   if (zero_fd == -1) {
85     MakeErrMsg(ErrMsg, "Can't open /dev/zero device");
86     return MemoryBlock();
87   }
88   fd = zero_fd;
89 #endif
90
91   int flags = MAP_PRIVATE |
92 #ifdef HAVE_MMAP_ANONYMOUS
93   MAP_ANONYMOUS
94 #else
95   MAP_ANON
96 #endif
97   ;
98
99   void* start = NearBlock ? (unsigned char*)NearBlock->base() + 
100                             NearBlock->size() : 0;
101
102   void *pa = ::mmap(start, pageSize*NumPages, PROT_READ|PROT_WRITE,
103                     flags, fd, 0);
104   if (pa == MAP_FAILED) {
105     if (NearBlock) //Try again without a near hint
106       return AllocateRWX(NumBytes, 0);
107
108     MakeErrMsg(ErrMsg, "Can't allocate RWX Memory");
109     return MemoryBlock();
110   }
111   MemoryBlock result;
112   result.Address = pa;
113   result.Size = NumPages*pageSize;
114   return result;
115 }
116
117 bool llvm::sys::Memory::ReleaseRWX(MemoryBlock &M, std::string *ErrMsg) {
118   if (M.Address == 0 || M.Size == 0) return false;
119   if (0 != ::munmap(M.Address, M.Size))
120     return MakeErrMsg(ErrMsg, "Can't release RWX Memory");
121   return false;
122 }
123