File: blockalloc.h

package info (click to toggle)
psyco-doc 1.6-1
  • links: PTS
  • area: contrib
  • in suites: lenny
  • size: 1,832 kB
  • ctags: 3,236
  • sloc: ansic: 23,895; python: 5,646; perl: 1,309; makefile: 153
file content (118 lines) | stat: -rw-r--r-- 3,855 bytes parent folder | download | duplicates (6)
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
 /***************************************************************/
/***           Generic fixed-sized-blocks allocator            ***/
 /***************************************************************/

#ifndef _BLOCKALLOC_H
#define _BLOCKALLOC_H


#include "psyco.h"


/* Define two functions to allocate and free items of type 'type'.
   The functions are named psyco_llalloc_[name] and psyco_llfree_[name]. */
#define BLOCKALLOC_INTERFACE(name, type)			\
	BLOCKALLOC_INTERN_DEF(name, type, EXTERNVAR, EXTERNFN)


/* Implements the required internal variables and functions.
   The allocator will reserve and subdivide blocks of the given size. */
#define BLOCKALLOC_IMPLEMENTATION(name, type, blocksize)	\
	BLOCKALLOC_INTERN_IMPL(name, type, blocksize, DEFINEVAR, DEFINEFN)


/* Same as the combination of the above two when the result need only
   be local to a .c source (i.e. static). */
#define BLOCKALLOC_STATIC(name, type, blocksize)				\
	BLOCKALLOC_INTERN_DEF(name, type, staticforward, static /*forward*/)	\
	BLOCKALLOC_INTERN_IMPL(name, type, blocksize, statichere, static)


/***************************************************************/
 /***   dummy version, used in conjunction with a memory      ***/
  /***   checker like linuxmemchk                              ***/

#ifdef PSYCO_NO_LINKED_LISTS

#define BLOCKALLOC_INTERN_DEF(name, type, EVAR, EFN)		\
PSY_INLINE type* psyco_llalloc_##name(void) {			\
	type* vi = (type*) PyMem_MALLOC(sizeof(type));		\
	if (vi == NULL)						\
		OUT_OF_MEMORY();				\
	return vi;						\
}								\
PSY_INLINE void psyco_llfree_##name(type* vi) {			\
	PyMem_FREE((char*) vi);					\
}

#define BLOCKALLOC_INTERN_IMPL(name, type, blocksize, DVAR, DFN)  /* nothing */


/***************************************************************/
 /***   full linked-list block-allocated version              ***/

#else /* if !PSYCO_NO_LINKED_LISTS */

#define BLOCKALLOC_INTERN_DEF(name, type, EVAR, EFN)		\
EVAR void** psyco_linked_list_##name;				\
EFN  void*  psyco_ll_newblock_##name(void);			\
PSY_INLINE type* psyco_llalloc_##name(void) {			\
	type* vi;						\
	if (psyco_linked_list_##name == NULL)			\
		vi = (type*) psyco_ll_newblock_##name();	\
	else {							\
		vi = (type*) psyco_linked_list_##name;		\
		psyco_linked_list_##name = *(void**) vi;	\
		BLOCKALLOC_DEBUG_CHECK(				\
			(type*) psyco_linked_list_##name, type)	\
	}							\
        BLOCKALLOC_DEBUG_CHECK(vi, type)			\
	return vi;						\
}								\
PSY_INLINE void psyco_llfree_##name(type* vi) {			\
	if (PSYCO_DEBUG)					\
		memset(vi, 0xCD, sizeof(type));			\
	*(void**) vi = psyco_linked_list_##name;		\
	psyco_linked_list_##name = (void**) vi;			\
}

#define BLOCKALLOC_INTERN_IMPL(name, type, blocksize, DVAR, DFN)	\
DVAR void** psyco_linked_list_##name = NULL;				\
DFN  void* psyco_ll_newblock_##name()					\
{									\
	int block_count = (blocksize)/sizeof(type);			\
	size_t sze = block_count * sizeof(type);			\
	type* p;							\
	type* prev = (type*) psyco_linked_list_##name;			\
	type* block = (type*) PyMem_MALLOC(sze);			\
	psyco_memory_usage += sze;					\
	extra_assert(block_count > 0 && sizeof(type) >= sizeof(void*));	\
	if (block == NULL)						\
		OUT_OF_MEMORY();					\
	if (PSYCO_DEBUG)						\
		memset(block, 0xCD, sze);				\
	for (p=block+block_count; --p!=block; ) {			\
		*(type**)p = prev;					\
		prev = p;						\
	}								\
	psyco_linked_list_##name = *(void***) prev;			\
	return prev;							\
}

#if PSYCO_DEBUG
# define BLOCKALLOC_DEBUG_CHECK(vi, type)				\
	if (vi) {							\
		int i;							\
		for (i=sizeof(void*); i<sizeof(type); i++)		\
			extra_assert(((unsigned char*)(vi))[i] == 0xCD);\
	}
#else
# define BLOCKALLOC_DEBUG_CHECK(vi, type)   /* nothing */
#endif

/***************************************************************/

#endif /* !PSYCO_NO_LINKED_LISTS */


#endif /* _BLOCKALLOC_H */