Set up go function pointer table
[akaros.git] / user / electric-fence / page.c
1 /* For email below, drop spaces and <spam-buster> tag.
2  * MODIFIED:  March 20, 2014 (jric<spam-buster> @ <spam-buster> chegg DOT com)
3  */
4 #include "efence.h"
5 #include <fcntl.h>
6 #include <parlib/uthread.h>
7 #include <stdio.h>
8 #include <stdlib.h>
9 #include <string.h>
10 #include <sys/mman.h>
11 #include <unistd.h>
12
13 static caddr_t startAddr = (caddr_t)0;
14
15 /*
16  * Create memory.
17  */
18 void *Page_Create(size_t size)
19 {
20         caddr_t allocation;
21
22         /*
23          * In this version, "startAddr" is a _hint_, not a demand.
24          * When the memory I map here is contiguous with other
25          * mappings, the allocator can coalesce the memory from two
26          * or more mappings into one large contiguous chunk, and thus
27          * might be able to find a fit that would not otherwise have
28          * been possible. I could _force_ it to be contiguous by using
29          * the MMAP_FIXED flag, but I don't want to stomp on memory mappings
30          * generated by other software, etc.
31          */
32         allocation = (caddr_t)mmap(startAddr, size, PROT_READ | PROT_WRITE,
33                                    MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
34
35         /*
36          * Set the "address hint" for the next mmap() so that it will abut
37          * the mapping we just created.
38          *
39          * HP/UX 9.01 has a kernel bug that makes mmap() fail sometimes
40          * when given a non-zero address hint, so we'll leave the hint set
41          * to zero on that system. HP recently told me this is now fixed.
42          * Someone please tell me when it is probable to assume that most
43          * of those systems that were running 9.01 have been upgraded.
44          */
45         startAddr = allocation + size;
46
47         if (allocation == (caddr_t)-1)
48                 EF_Exit("mmap() failed: %r");
49
50         return (void *)allocation;
51 }
52
53 static void mprotectFailed(void)
54 {
55         EF_Exit("mprotect() failed: %r");
56 }
57
58 void Page_AllowAccess(void *address, size_t size)
59 {
60         if (mprotect((caddr_t)address, size, PROT_READ | PROT_WRITE) < 0)
61                 mprotectFailed();
62 }
63
64 void Page_DenyAccess(void *address, size_t size)
65 {
66         if (mprotect((caddr_t)address, size, PROT_NONE) < 0)
67                 mprotectFailed();
68 }
69
70 void Page_Delete(void *address, size_t size)
71 {
72         Page_DenyAccess(address, size);
73 }
74
75 size_t Page_Size(void)
76 {
77         return PGSIZE;
78 }