summaryrefslogtreecommitdiff
path: root/rts/Storage.c
diff options
context:
space:
mode:
authorSimon Marlow <simonmar@microsoft.com>2006-05-30 10:02:11 +0000
committerSimon Marlow <simonmar@microsoft.com>2006-05-30 10:02:11 +0000
commite3c55aebd4f9ce7a5b4390d4726612865fd207f2 (patch)
tree87a1e86bc77f2c1d8d4dba8757e14fce324fe736 /rts/Storage.c
parent6b36d8ad3bfd1890583f3bcab96559f05bff332b (diff)
downloadhaskell-e3c55aebd4f9ce7a5b4390d4726612865fd207f2.tar.gz
replace stgMallocBytesRWX() with our own allocator
See bug #738 Allocating executable memory is getting more difficult these days. In particular, the default SELinux policy on Fedora Core 5 disallows making the heap (i.e. malloc()'d memory) executable, although it does apparently allow mmap()'ing anonymous executable memory by default. Previously, stgMallocBytesRWX() used malloc() underneath, and then tried to make the page holding the memory executable. This was rather hacky and fails with Fedora Core 5. This patch adds a mini-allocator for executable memory, based on the block allocator. We grab page-sized blocks and make them executable, then allocate small objects from the page. There's a simple free function, that will free whole pages back to the system when they are empty.
Diffstat (limited to 'rts/Storage.c')
-rw-r--r--rts/Storage.c99
1 files changed, 99 insertions, 0 deletions
diff --git a/rts/Storage.c b/rts/Storage.c
index 974be45f10..ee860e27a2 100644
--- a/rts/Storage.c
+++ b/rts/Storage.c
@@ -22,6 +22,7 @@
#include "Storage.h"
#include "Schedule.h"
#include "RetainerProfile.h" // for counting memory blocks (memInventory)
+#include "OSMem.h"
#include <stdlib.h>
#include <string.h>
@@ -968,6 +969,99 @@ calcNeeded(void)
return needed;
}
+/* ----------------------------------------------------------------------------
+ Executable memory
+
+ Executable memory must be managed separately from non-executable
+ memory. Most OSs these days require you to jump through hoops to
+ dynamically allocate executable memory, due to various security
+ measures.
+
+ Here we provide a small memory allocator for executable memory.
+ Memory is managed with a page granularity; we allocate linearly
+ in the page, and when the page is emptied (all objects on the page
+ are free) we free the page again, not forgetting to make it
+ non-executable.
+ ------------------------------------------------------------------------- */
+
+static bdescr *exec_block;
+
+void *allocateExec (nat bytes)
+{
+ void *ret;
+ nat n;
+
+ ACQUIRE_SM_LOCK;
+
+ // round up to words.
+ n = (bytes + sizeof(W_) + 1) / sizeof(W_);
+
+ if (n+1 > BLOCK_SIZE_W) {
+ barf("allocateExec: can't handle large objects");
+ }
+
+ if (exec_block == NULL ||
+ exec_block->free + n + 1 > exec_block->start + BLOCK_SIZE_W) {
+ bdescr *bd;
+ lnat pagesize = getPageSize();
+ bd = allocGroup(stg_max(1, pagesize / BLOCK_SIZE));
+ IF_DEBUG(gc, debugBelch("allocate exec block %p\n", bd->start));
+ bd->gen_no = 0;
+ bd->flags = BF_EXEC;
+ bd->link = exec_block;
+ if (exec_block != NULL) {
+ exec_block->u.back = bd;
+ }
+ bd->u.back = NULL;
+ setExecutable(bd->start, bd->blocks * BLOCK_SIZE, rtsTrue);
+ exec_block = bd;
+ }
+ *(exec_block->free) = n; // store the size of this chunk
+ exec_block->gen_no += n; // gen_no stores the number of words allocated
+ ret = exec_block->free + 1;
+ exec_block->free += n + 1;
+
+ RELEASE_SM_LOCK
+ return ret;
+}
+
+void freeExec (void *addr)
+{
+ StgPtr p = (StgPtr)addr - 1;
+ bdescr *bd = Bdescr((StgPtr)p);
+
+ if ((bd->flags & BF_EXEC) == 0) {
+ barf("freeExec: not executable");
+ }
+
+ if (*(StgPtr)p == 0) {
+ barf("freeExec: already free?");
+ }
+
+ ACQUIRE_SM_LOCK;
+
+ bd->gen_no -= *(StgPtr)p;
+ *(StgPtr)p = 0;
+
+ // Free the block if it is empty, but not if it is the block at
+ // the head of the queue.
+ if (bd->gen_no == 0 && bd != exec_block) {
+ IF_DEBUG(gc, debugBelch("free exec block %p\n", bd->start));
+ if (bd->u.back) {
+ bd->u.back->link = bd->link;
+ } else {
+ exec_block = bd->link;
+ }
+ if (bd->link) {
+ bd->link->u.back = bd->u.back;
+ }
+ setExecutable(bd->start, bd->blocks * BLOCK_SIZE, rtsFalse);
+ freeGroup(bd);
+ }
+
+ RELEASE_SM_LOCK
+}
+
/* -----------------------------------------------------------------------------
Debugging
@@ -1048,6 +1142,11 @@ memInventory(void)
// count the blocks allocated by the arena allocator
total_blocks += arenaBlocks();
+ // count the blocks containing executable memory
+ for (bd = exec_block; bd; bd = bd->link) {
+ total_blocks += bd->blocks;
+ }
+
/* count the blocks on the free list */
free_blocks = countFreeList();