-
Notifications
You must be signed in to change notification settings - Fork 138
/
pointer_array.c
216 lines (157 loc) · 4.8 KB
/
pointer_array.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
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
/*
Copyright (C) 2010, Parrot Foundation.
=head1 NAME
src/pointer_array.c - Implementation Pointer Array storage.
=head1 DESCRIPTION
=cut
*/
#include "parrot/parrot.h"
#include "parrot/pointer_array.h"
/* HEADERIZER HFILE: include/parrot/pointer_array.h */
/* HEADERIZER BEGIN: static */
/* Don't modify between HEADERIZER BEGIN / HEADERIZER END. Your changes will be lost. */
static void allocate_more_chunks(PARROT_INTERP,
ARGIN(Parrot_Pointer_Array *self))
__attribute__nonnull__(1)
__attribute__nonnull__(2);
#define ASSERT_ARGS_allocate_more_chunks __attribute__unused__ int _ASSERT_ARGS_CHECK = (\
PARROT_ASSERT_ARG(interp) \
, PARROT_ASSERT_ARG(self))
/* Don't modify between HEADERIZER BEGIN / HEADERIZER END. Your changes will be lost. */
/* HEADERIZER END: static */
/*
=over 4
=item C<Parrot_Pointer_Array * Parrot_pa_new(PARROT_INTERP)>
Allocate new Pointer_Array.
=cut
*/
PARROT_EXPORT
PARROT_CANNOT_RETURN_NULL
Parrot_Pointer_Array *
Parrot_pa_new(PARROT_INTERP)
{
ASSERT_ARGS(Parrot_pa_new)
Parrot_Pointer_Array *res = mem_allocate_zeroed_typed(Parrot_Pointer_Array);
return res;
}
/*
=item C<void Parrot_pa_destroy(PARROT_INTERP, Parrot_Pointer_Array *self)>
Destroy Pointer_Arra and free allocated memory.
=cut
*/
PARROT_EXPORT
void
Parrot_pa_destroy(PARROT_INTERP, ARGIN(Parrot_Pointer_Array *self))
{
ASSERT_ARGS(Parrot_pa_destroy)
size_t i;
for (i = 0; i < self->total_chunks; i++)
mem_sys_free(self->chunks[i]);
mem_sys_free(self->chunks);
}
/*
=item C<void * Parrot_pa_insert(PARROT_INTERP, Parrot_Pointer_Array *self, void
*ptr)>
Insert pointer into the array.
=cut
*/
PARROT_EXPORT
PARROT_CAN_RETURN_NULL
void *
Parrot_pa_insert(PARROT_INTERP, ARGIN(Parrot_Pointer_Array *self), ARGIN(void *ptr))
{
ASSERT_ARGS(Parrot_pa_insert)
Parrot_Pointer_Array_Chunk *chunk;
void *ret;
/* If there is no free chunks */
if (self->current_chunk >= self->total_chunks)
allocate_more_chunks(interp, self);
/* Reuse removed cell */
if (self->next_free) {
/* FIXME. Cast to UINTVAL is wrong. */
/* We have to provide parrot version of uintptr_t due lack of it in C89 */
/* See TT #1855 */
void **next = (void**)((UINTVAL)*self->next_free & ~1);
ret = self->next_free;
*self->next_free = ptr;
self->next_free = next;
return ret;
}
if (!self->chunks[self->current_chunk]->num_free)
allocate_more_chunks(interp, self);
chunk = self->chunks[self->current_chunk];
--chunk->num_free;
/* Invariant: all chunks after chunk->next_free are free */
/* We handle previously freed chunks early */
ret = &chunk->data[chunk->next_free];
chunk->data[chunk->next_free++] = ptr;
return ret;
}
/*
=item C<void Parrot_pa_remove(PARROT_INTERP, Parrot_Pointer_Array *self, void
*ptr)>
Remove pointer from array.
=cut
*/
PARROT_EXPORT
void
Parrot_pa_remove(PARROT_INTERP, ARGIN(Parrot_Pointer_Array *self), ARGIN(void *ptr))
{
ASSERT_ARGS(Parrot_pa_remove)
/* Mark sell to avoid iterating over */
*(UINTVAL*)ptr = ((UINTVAL)self->next_free) | 1;
self->next_free = (void**)ptr;
}
/*
=item C<int Parrot_pa_is_owned(PARROT_INTERP, Parrot_Pointer_Array *self, void
*orig, void *ref)>
Check that C<orig> pointer is stored in C<ref> cell. Used during system stack t
=cut
*/
PARROT_EXPORT
int
Parrot_pa_is_owned(PARROT_INTERP, ARGIN(Parrot_Pointer_Array *self),
ARGIN(void *orig), ARGIN_NULLOK(void *ref))
{
ASSERT_ARGS(Parrot_pa_is_owned)
size_t i;
/* Return early if ref is null */
if (!ref)
return 0;
/* We can't just deref pointer. It can be garbage */
/* So, ensure that C<ref> is looks like real pointer */
for (i = 0; i < self->total_chunks; i++) {
Parrot_Pointer_Array_Chunk *chunk = self->chunks[i];
if (PTR2UINTVAL(ref) < PTR2UINTVAL(chunk->data))
continue;
if (PTR2UINTVAL(ref) > PTR2UINTVAL(chunk) + CHUNK_SIZE)
continue;
return (*(void **)ref == orig);
}
return 0;
}
/*
=item C<static void allocate_more_chunks(PARROT_INTERP, Parrot_Pointer_Array
*self)>
allocate more chunks
=back
=cut
*/
static void
allocate_more_chunks(PARROT_INTERP, ARGIN(Parrot_Pointer_Array *self))
{
ASSERT_ARGS(allocate_more_chunks)
self->current_chunk = self->total_chunks++;
mem_realloc_n_typed(self->chunks,
self->total_chunks,
Parrot_Pointer_Array_Chunk*);
self->chunks[self->current_chunk] = mem_allocate_typed(Parrot_Pointer_Array_Chunk);
self->chunks[self->current_chunk]->num_free = CELL_PER_CHUNK;
self->chunks[self->current_chunk]->next_free = 0;
}
/*
* Local variables:
* c-file-style: "parrot"
* End:
* vim: expandtab shiftwidth=4 cinoptions='\:2=2' :
*/