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