| 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
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
 | /*-------------------------------------------------------------------------
 *
 * aset.c--
 *    Allocation set definitions.
 *
 * Copyright (c) 1994, Regents of the University of California
 *
 *
 * IDENTIFICATION
 *    $Header: /cvsroot/pgsql/src/backend/utils/mmgr/aset.c,v 1.1.1.1 1996/07/09 06:22:09 scrappy Exp $
 *
 * NOTE
 *    XXX This is a preliminary implementation which lacks fail-fast
 *    XXX validity checking of arguments.
 *
 *-------------------------------------------------------------------------
 */
#include <stdio.h>
#include "c.h"
#include "utils/excid.h"	/* for ExhaustedMemory */
#include "utils/memutils.h"	/* where funnction declarations go */
#include "utils/elog.h"
#include "utils/palloc.h"
#undef AllocSetReset
#undef malloc
#undef free
/*
 * Internal type definitions
 */
/*
 * AllocElem --
 *	Allocation element.
 */
typedef struct AllocElemData {
    OrderedElemData	elemData;	/* elem in AllocSet */
    Size		size;
} AllocElemData;
typedef AllocElemData *AllocElem;
/*
 * Private method definitions
 */
/*
 * AllocPointerGetAllocElem --
 *	Returns allocation (internal) elem given (external) pointer.
 */
#define AllocPointerGetAllocElem(pointer)	(&((AllocElem)(pointer))[-1])
/*
 * AllocElemGetAllocPointer --
 *	Returns allocation (external) pointer given (internal) elem.
 */
#define AllocElemGetAllocPointer(alloc)	((AllocPointer)&(alloc)[1])
/*
 * AllocElemIsValid --
 *	True iff alloc is valid.
 */
#define AllocElemIsValid(alloc)	PointerIsValid(alloc)
/* non-export function prototypes */
static AllocPointer AllocSetGetFirst(AllocSet set);
static AllocPointer AllocPointerGetNext(AllocPointer pointer);
/*
 * Public routines
 */
/* 
 *	AllocPointerIsValid(pointer)
 * 	AllocSetIsValid(set)    
 *
 *		.. are now macros in aset.h -cim 4/27/91
 */
/*
 * AllocSetInit --
 *	Initializes given allocation set.
 *
 * Note:
 *	The semantics of the mode are explained above.  Limit is ignored
 *	for dynamic and static modes.
 *
 * Exceptions:
 *	BadArg if set is invalid pointer.
 *	BadArg if mode is invalid.
 */
void
AllocSetInit(AllocSet set, AllocMode mode, Size limit)
{
    AssertArg(PointerIsValid(set));
    AssertArg((int)DynamicAllocMode <= (int)mode);
    AssertArg((int)mode <= (int)BoundedAllocMode);
    
    /*
     * XXX mode is currently ignored and treated as DynamicAllocMode.
     * XXX limit is also ignored.  This affects this whole file.
     */
    
    OrderedSetInit(&set->setData, offsetof(AllocElemData, elemData));
}
/*
 * AllocSetReset --
 *	Frees memory which is allocated in the given set.
 *
 * Exceptions:
 *	BadArg if set is invalid.
 */
void
AllocSetReset(AllocSet set)
{
    AllocPointer	pointer;
    
    AssertArg(AllocSetIsValid(set));
    
    while (AllocPointerIsValid(pointer = AllocSetGetFirst(set))) {
	AllocSetFree(set, pointer);
    }
}
void
AllocSetReset_debug(char *file, int line, AllocSet set)
{
    AllocPointer	pointer;
    
    AssertArg(AllocSetIsValid(set));
    
    while (AllocPointerIsValid(pointer = AllocSetGetFirst(set))) {
	AllocSetFree(set, pointer);
    }
}
/*
 * AllocSetContains --
 *	True iff allocation set contains given allocation element.
 *
 * Exceptions:
 *	BadArg if set is invalid.
 *	BadArg if pointer is invalid.
 */
bool
AllocSetContains(AllocSet set, AllocPointer pointer)
{
    AssertArg(AllocSetIsValid(set));
    AssertArg(AllocPointerIsValid(pointer));
    
    return (OrderedSetContains(&set->setData,
			       &AllocPointerGetAllocElem(pointer)->elemData));
}
/*
 * AllocSetAlloc --
 *	Returns pointer to allocated memory of given size; memory is added
 *	to the set.
 *
 * Exceptions:
 *	BadArg if set is invalid.
 *	MemoryExhausted if allocation fails.
 */
AllocPointer
AllocSetAlloc(AllocSet set, Size size)
{
    AllocElem	alloc;
    
    AssertArg(AllocSetIsValid(set));
    
    /* allocate */
    alloc = (AllocElem)malloc(sizeof (*alloc) + size);
    
    if (!PointerIsValid(alloc)) {
	elog (FATAL, "palloc failure: memory exhausted");
    }
    
    /* add to allocation list */
    OrderedElemPushInto(&alloc->elemData, &set->setData);
    
    /* set size */
    alloc->size = size;
    
    return (AllocElemGetAllocPointer(alloc));
}
/*
 * AllocSetFree --
 *	Frees allocated memory; memory is removed from the set.
 *
 * Exceptions:
 *	BadArg if set is invalid.
 *	BadArg if pointer is invalid.
 *	BadArg if pointer is not member of set.
 */
void
AllocSetFree(AllocSet set, AllocPointer pointer)
{
    AllocElem	alloc;
    
    /* AssertArg(AllocSetIsValid(set)); */
    /* AssertArg(AllocPointerIsValid(pointer)); */
    AssertArg(AllocSetContains(set, pointer));
    
    alloc = AllocPointerGetAllocElem(pointer);
    
    /* remove from allocation set */
    OrderedElemPop(&alloc->elemData);
    
    /* free storage */
    free(alloc);
}
/*
 * AllocSetRealloc --
 *	Returns new pointer to allocated memory of given size; this memory
 *	is added to the set.  Memory associated with given pointer is copied
 *	into the new memory, and the old memory is freed.
 *
 * Exceptions:
 *	BadArg if set is invalid.
 *	BadArg if pointer is invalid.
 *	BadArg if pointer is not member of set.
 *	MemoryExhausted if allocation fails.
 */
AllocPointer
AllocSetRealloc(AllocSet set, AllocPointer pointer, Size size)
{
    AllocPointer	newPointer;
    AllocElem	alloc;
    
    /* AssertArg(AllocSetIsValid(set)); */
    /* AssertArg(AllocPointerIsValid(pointer)); */
    AssertArg(AllocSetContains(set, pointer));
    
    /*
     * Calling realloc(3) directly is not be possible (unless we use
     * our own hacked version of malloc) since we must keep the
     * allocations in the allocation set.
     */
    
    alloc = AllocPointerGetAllocElem(pointer);
    
    /* allocate new pointer */
    newPointer = AllocSetAlloc(set, size);
    
    /* fill new memory */
    memmove(newPointer, pointer, (alloc->size < size) ? alloc->size : size);
    
    /* free old pointer */
    AllocSetFree(set, pointer);
    
    return (newPointer);
}
/*
 * AllocSetIterate --
 *	Returns size of set.  Iterates through set elements calling function
 *	(if valid) on each.
 *
 * Note:
 *	This was written as an aid to debugging.  It is intended for
 *	debugging use only.
 *
 * Exceptions:
 *	BadArg if set is invalid.
 */
int
AllocSetIterate(AllocSet set,
		void (*function)(AllocPointer pointer))
{
    int		count = 0;
    AllocPointer	pointer;
    
    AssertArg(AllocSetIsValid(set));
    
    for (pointer = AllocSetGetFirst(set);
	 AllocPointerIsValid(pointer);
	 pointer = AllocPointerGetNext(pointer)) {
	
	if (PointerIsValid(function)) {
	    (*function)(pointer);
	}
	count += 1;
    }
    
    return (count);
}
int
AllocSetCount(AllocSet set)
{
    int		count = 0;
    AllocPointer	pointer;
    
    AssertArg(AllocSetIsValid(set));
    
    for (pointer = AllocSetGetFirst(set);
	 AllocPointerIsValid(pointer);
	 pointer = AllocPointerGetNext(pointer)) {
	count++;
    }
    return count;
}
/*
 * Private routines
 */
/*
 * AllocSetGetFirst --
 *	Returns "first" allocation pointer in a set.
 *
 * Note:
 *	Assumes set is valid.
 */
static AllocPointer
AllocSetGetFirst(AllocSet set)
{
    AllocElem	alloc;
    
    alloc = (AllocElem)OrderedSetGetHead(&set->setData);
    
    if (!AllocElemIsValid(alloc)) {
	return (NULL);
    }
    
    return (AllocElemGetAllocPointer(alloc));
}
/*
 * AllocPointerGetNext --
 *	Returns "successor" allocation pointer.
 *
 * Note:
 *	Assumes pointer is valid.
 */
static AllocPointer
AllocPointerGetNext(AllocPointer pointer)
{
    AllocElem	alloc;
    
    alloc = (AllocElem)
	OrderedElemGetSuccessor(&AllocPointerGetAllocElem(pointer)->elemData);
    
    if (!AllocElemIsValid(alloc)) {
	return (NULL);
    }
    
    return (AllocElemGetAllocPointer(alloc));
}
/*
 * Debugging routines
 */
/*
 * XXX AllocPointerDump --
 *	Displays allocated pointer.
 */
void
AllocPointerDump(AllocPointer pointer)
{
    printf("\t%-10d@ %0#x\n", ((long*)pointer)[-1], pointer); /* XXX */
}
/*
 * AllocSetDump --
 *	Displays allocated set.
 */
void
AllocSetDump(AllocSet set)
{
    int count;
    count = AllocSetIterate(set, AllocPointerDump);
    printf("\ttotal %d allocations\n", count);
}
 |