-
Notifications
You must be signed in to change notification settings - Fork 10
/
Copy pathstaticalloc.c
140 lines (113 loc) · 4.1 KB
/
staticalloc.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
/* staticalloc.c implements a simple allocator that provides access to a single,
* chunk of memory, without attempting to portion it at all.
*
* written nml 2005-03-10
*
*/
#include "firstinclude.h"
#include "staticalloc.h"
#include "def.h"
#include "mem.h"
#include "_mem.h"
#include "zvalgrind.h"
#include "zstdint.h"
#include <assert.h>
#include <stdlib.h>
#include <limits.h>
struct staticalloc {
union {
int size; /* size of available memory. The sign
* indicates whether it is currently
* allocated or not - negative indicates
* allocated, positive indicates not */
void *dummy; /* here to ensure that staticalloc is one
* word in size, in a world of LP64 and
* LLP64 stupidity */
} u;
};
struct staticalloc *staticalloc_new(void *area, unsigned int size) {
struct staticalloc *alloc = NULL;
/* can't handle size of > INT_MAX because we use the sign bit to track
* allocation. Seems like a good tradeoff. */
if ((size <= INT_MAX) && (size > sizeof(*alloc))) {
alloc = area;
alloc->u.size = ((int) size) - sizeof(*alloc);
VALGRIND_MAKE_NOACCESS(MEM_PTRADD(alloc, sizeof(*alloc)),
alloc->u.size);
assert(alloc->u.size);
}
return alloc;
}
void staticalloc_delete(struct staticalloc *alloc) {
assert(alloc->u.size);
VALGRIND_MAKE_WRITABLE(MEM_PTRADD(alloc, sizeof(*alloc)), alloc->u.size);
alloc->u.size = 0; /* zero out memory, so it can't be reused */
return;
}
void *staticalloc_malloc(struct staticalloc *alloc, unsigned int size) {
void *ptr;
assert(alloc->u.size);
if ((alloc->u.size > 0) && (((unsigned int) alloc->u.size) >= size)) {
VALGRIND_MAKE_WRITABLE(MEM_PTRADD(alloc, sizeof(*alloc)),
alloc->u.size);
alloc->u.size *= -1; /* indicate that the chunk is allocated */
ptr = MEM_PTRADD(alloc, sizeof(*alloc));
VALGRIND_MALLOCLIKE_BLOCK(ptr, size, 0, 0);
return ptr;
} else {
/* our memory chunk is currently out, please come again... */
return NULL;
}
}
void staticalloc_free(struct staticalloc *alloc, void *ptr) {
assert(alloc->u.size);
if (alloc->u.size < 0 && (ptr == MEM_PTRADD(alloc, sizeof(*alloc)))) {
alloc->u.size *= -1; /* indicate that the chunk is free */
VALGRIND_FREELIKE_BLOCK(ptr, 0);
VALGRIND_MAKE_NOACCESS(MEM_PTRADD(alloc, sizeof(*alloc)),
alloc->u.size);
} else {
assert(!CRASH); /* user error, it's already free */
}
}
unsigned int staticalloc_allocated(struct staticalloc *alloc) {
assert(alloc->u.size);
return (alloc->u.size < 0);
}
int staticalloc_is_managed(struct staticalloc *alloc, void *ptr) {
assert(alloc->u.size);
return (ptr == MEM_PTRADD(alloc, sizeof(*alloc)));
}
unsigned int staticalloc_overhead() {
return sizeof(struct staticalloc);
}
#ifdef STATICALLOC_TEST
#define BYTES sizeof(int)
int main() {
STATICALLOC_DECL(alloc, BYTES);
void *ptr;
int *intptr;
ptr = staticalloc_malloc(alloc, BYTES);
assert(ptr); /* allocated should have worked */
assert(ptr == &alloc_stackspace[1]); /* should give us address one word
* from the start of the space */
assert(!staticalloc_malloc(alloc, 1)); /* shouldn't be able to allocated
* again */
intptr = ptr;
*intptr = 0; /* access it */
staticalloc_free(alloc, ptr);
/* invalid stuff */
#if 0
*intptr = 0; /* access it when we don't own it */
do {
STATICALLOC_DECL(leak, 1);
assert(staticalloc_malloc(leak, 1));
} while (0);
#endif
assert(staticalloc_malloc(alloc, 1) == ptr);
assert(!staticalloc_malloc(alloc, 1));
staticalloc_free(alloc, ptr);
staticalloc_delete(alloc);
return EXIT_SUCCESS;
}
#endif