xref: /illumos-gate/usr/src/contrib/ast/src/lib/libast/vmalloc/vmpool.c (revision b30d193948be5a7794d7ae3ba0ed9c2f72c88e0f)
1*b30d1939SAndy Fiddaman /***********************************************************************
2*b30d1939SAndy Fiddaman *                                                                      *
3*b30d1939SAndy Fiddaman *               This software is part of the ast package               *
4*b30d1939SAndy Fiddaman *          Copyright (c) 1985-2011 AT&T Intellectual Property          *
5*b30d1939SAndy Fiddaman *                      and is licensed under the                       *
6*b30d1939SAndy Fiddaman *                 Eclipse Public License, Version 1.0                  *
7*b30d1939SAndy Fiddaman *                    by AT&T Intellectual Property                     *
8*b30d1939SAndy Fiddaman *                                                                      *
9*b30d1939SAndy Fiddaman *                A copy of the License is available at                 *
10*b30d1939SAndy Fiddaman *          http://www.eclipse.org/org/documents/epl-v10.html           *
11*b30d1939SAndy Fiddaman *         (with md5 checksum b35adb5213ca9657e911e9befb180842)         *
12*b30d1939SAndy Fiddaman *                                                                      *
13*b30d1939SAndy Fiddaman *              Information and Software Systems Research               *
14*b30d1939SAndy Fiddaman *                            AT&T Research                             *
15*b30d1939SAndy Fiddaman *                           Florham Park NJ                            *
16*b30d1939SAndy Fiddaman *                                                                      *
17*b30d1939SAndy Fiddaman *                 Glenn Fowler <gsf@research.att.com>                  *
18*b30d1939SAndy Fiddaman *                  David Korn <dgk@research.att.com>                   *
19*b30d1939SAndy Fiddaman *                   Phong Vo <kpv@research.att.com>                    *
20*b30d1939SAndy Fiddaman *                                                                      *
21*b30d1939SAndy Fiddaman ***********************************************************************/
22*b30d1939SAndy Fiddaman #if defined(_UWIN) && defined(_BLD_ast)
23*b30d1939SAndy Fiddaman 
_STUB_vmpool()24*b30d1939SAndy Fiddaman void _STUB_vmpool(){}
25*b30d1939SAndy Fiddaman 
26*b30d1939SAndy Fiddaman #else
27*b30d1939SAndy Fiddaman 
28*b30d1939SAndy Fiddaman #include	"vmhdr.h"
29*b30d1939SAndy Fiddaman 
30*b30d1939SAndy Fiddaman #define POOLFREE	0x55555555L	/* block free indicator	 */
31*b30d1939SAndy Fiddaman 
32*b30d1939SAndy Fiddaman /*	Method for pool allocation.
33*b30d1939SAndy Fiddaman **	All elements in a pool have the same size.
34*b30d1939SAndy Fiddaman **	The following fields of Vmdata_t are used as:
35*b30d1939SAndy Fiddaman **		pool:	size of a block.
36*b30d1939SAndy Fiddaman **		free:	list of free blocks.
37*b30d1939SAndy Fiddaman **
38*b30d1939SAndy Fiddaman **	Written by Kiem-Phong Vo, kpv@research.att.com, 01/16/94.
39*b30d1939SAndy Fiddaman */
40*b30d1939SAndy Fiddaman 
41*b30d1939SAndy Fiddaman #if __STD_C
poolalloc(Vmalloc_t * vm,reg size_t size,int local)42*b30d1939SAndy Fiddaman static Void_t* poolalloc(Vmalloc_t* vm, reg size_t size, int local)
43*b30d1939SAndy Fiddaman #else
44*b30d1939SAndy Fiddaman static Void_t* poolalloc(vm, size, local )
45*b30d1939SAndy Fiddaman Vmalloc_t*	vm;
46*b30d1939SAndy Fiddaman reg size_t	size;
47*b30d1939SAndy Fiddaman int		local;
48*b30d1939SAndy Fiddaman #endif
49*b30d1939SAndy Fiddaman {
50*b30d1939SAndy Fiddaman 	reg Block_t	*tp, *next;
51*b30d1939SAndy Fiddaman 	reg size_t	s;
52*b30d1939SAndy Fiddaman 	reg Seg_t	*seg;
53*b30d1939SAndy Fiddaman 	reg Vmdata_t	*vd = vm->data;
54*b30d1939SAndy Fiddaman 
55*b30d1939SAndy Fiddaman 	if(size <= 0)
56*b30d1939SAndy Fiddaman 		return NIL(Void_t*);
57*b30d1939SAndy Fiddaman 
58*b30d1939SAndy Fiddaman 	if(size != vd->pool)
59*b30d1939SAndy Fiddaman 	{	if(vd->pool <= 0)
60*b30d1939SAndy Fiddaman 			vd->pool = size;
61*b30d1939SAndy Fiddaman 		else	return NIL(Void_t*);
62*b30d1939SAndy Fiddaman 	}
63*b30d1939SAndy Fiddaman 
64*b30d1939SAndy Fiddaman 	SETLOCK(vm, local);
65*b30d1939SAndy Fiddaman 
66*b30d1939SAndy Fiddaman 	if((tp = vd->free) ) /* there is a ready free block */
67*b30d1939SAndy Fiddaman 	{	vd->free = SEGLINK(tp);
68*b30d1939SAndy Fiddaman 		goto done;
69*b30d1939SAndy Fiddaman 	}
70*b30d1939SAndy Fiddaman 
71*b30d1939SAndy Fiddaman 	size = ROUND(size,ALIGN);
72*b30d1939SAndy Fiddaman 
73*b30d1939SAndy Fiddaman 	/* look thru all segments for a suitable free block */
74*b30d1939SAndy Fiddaman 	for(tp = NIL(Block_t*), seg = vd->seg; seg; seg = seg->next)
75*b30d1939SAndy Fiddaman 	{	if((tp = seg->free) &&
76*b30d1939SAndy Fiddaman 		   (s = (SIZE(tp) & ~BITS) + sizeof(Head_t)) >= size )
77*b30d1939SAndy Fiddaman 			goto got_blk;
78*b30d1939SAndy Fiddaman 	}
79*b30d1939SAndy Fiddaman 
80*b30d1939SAndy Fiddaman 	if((tp = (*_Vmextend)(vm,ROUND(size,vd->incr),NIL(Vmsearch_f))) )
81*b30d1939SAndy Fiddaman 	{	s = (SIZE(tp) & ~BITS) + sizeof(Head_t);
82*b30d1939SAndy Fiddaman 		seg = SEG(tp);
83*b30d1939SAndy Fiddaman 		goto got_blk;
84*b30d1939SAndy Fiddaman 	}
85*b30d1939SAndy Fiddaman 	else	goto done;
86*b30d1939SAndy Fiddaman 
87*b30d1939SAndy Fiddaman got_blk: /* if get here, (tp, s, seg) must be well-defined */
88*b30d1939SAndy Fiddaman 	next = (Block_t*)((Vmuchar_t*)tp+size);
89*b30d1939SAndy Fiddaman 	if((s -= size) <= (size + sizeof(Head_t)) )
90*b30d1939SAndy Fiddaman 	{	for(; s >= size; s -= size)
91*b30d1939SAndy Fiddaman 		{	SIZE(next) = POOLFREE;
92*b30d1939SAndy Fiddaman 			SEGLINK(next) = vd->free;
93*b30d1939SAndy Fiddaman 			vd->free = next;
94*b30d1939SAndy Fiddaman 			next = (Block_t*)((Vmuchar_t*)next + size);
95*b30d1939SAndy Fiddaman 		}
96*b30d1939SAndy Fiddaman 		seg->free = NIL(Block_t*);
97*b30d1939SAndy Fiddaman 	}
98*b30d1939SAndy Fiddaman 	else
99*b30d1939SAndy Fiddaman 	{	SIZE(next) = s - sizeof(Head_t);
100*b30d1939SAndy Fiddaman 		SEG(next) = seg;
101*b30d1939SAndy Fiddaman 		seg->free = next;
102*b30d1939SAndy Fiddaman 	}
103*b30d1939SAndy Fiddaman 
104*b30d1939SAndy Fiddaman done:
105*b30d1939SAndy Fiddaman 	if(!local && (vd->mode&VM_TRACE) && _Vmtrace && tp)
106*b30d1939SAndy Fiddaman 		(*_Vmtrace)(vm,NIL(Vmuchar_t*),(Vmuchar_t*)tp,vd->pool,0);
107*b30d1939SAndy Fiddaman 
108*b30d1939SAndy Fiddaman 	CLRLOCK(vm, local);
109*b30d1939SAndy Fiddaman 
110*b30d1939SAndy Fiddaman 	return (Void_t*)tp;
111*b30d1939SAndy Fiddaman }
112*b30d1939SAndy Fiddaman 
113*b30d1939SAndy Fiddaman #if __STD_C
pooladdr(Vmalloc_t * vm,reg Void_t * addr,int local)114*b30d1939SAndy Fiddaman static long pooladdr(Vmalloc_t* vm, reg Void_t* addr, int local)
115*b30d1939SAndy Fiddaman #else
116*b30d1939SAndy Fiddaman static long pooladdr(vm, addr, local)
117*b30d1939SAndy Fiddaman Vmalloc_t*	vm;
118*b30d1939SAndy Fiddaman reg Void_t*	addr;
119*b30d1939SAndy Fiddaman int		local;
120*b30d1939SAndy Fiddaman #endif
121*b30d1939SAndy Fiddaman {
122*b30d1939SAndy Fiddaman 	Block_t		*bp, *tp;
123*b30d1939SAndy Fiddaman 	Vmuchar_t	*laddr, *baddr;
124*b30d1939SAndy Fiddaman 	size_t		size;
125*b30d1939SAndy Fiddaman 	Seg_t		*seg;
126*b30d1939SAndy Fiddaman 	long		offset;
127*b30d1939SAndy Fiddaman 	Vmdata_t*	vd = vm->data;
128*b30d1939SAndy Fiddaman 
129*b30d1939SAndy Fiddaman 	SETLOCK(vm, local);
130*b30d1939SAndy Fiddaman 
131*b30d1939SAndy Fiddaman 	offset = -1L;
132*b30d1939SAndy Fiddaman 	for(seg = vd->seg; seg; seg = seg->next)
133*b30d1939SAndy Fiddaman 	{	laddr = (Vmuchar_t*)SEGBLOCK(seg);
134*b30d1939SAndy Fiddaman 		baddr = seg->baddr-sizeof(Head_t);
135*b30d1939SAndy Fiddaman 		if((Vmuchar_t*)addr < laddr || (Vmuchar_t*)addr >= baddr)
136*b30d1939SAndy Fiddaman 			continue;
137*b30d1939SAndy Fiddaman 
138*b30d1939SAndy Fiddaman 		/* the block that has this address */
139*b30d1939SAndy Fiddaman 		size = ROUND(vd->pool,ALIGN);
140*b30d1939SAndy Fiddaman 		tp = (Block_t*)(laddr + (((Vmuchar_t*)addr-laddr)/size)*size );
141*b30d1939SAndy Fiddaman 
142*b30d1939SAndy Fiddaman 		/* see if this block has been freed */
143*b30d1939SAndy Fiddaman 		if(SIZE(tp) == POOLFREE) /* may be a coincidence - make sure */
144*b30d1939SAndy Fiddaman 			for(bp = vd->free; bp; bp = SEGLINK(bp))
145*b30d1939SAndy Fiddaman 				if(bp == tp)
146*b30d1939SAndy Fiddaman 					goto done;
147*b30d1939SAndy Fiddaman 
148*b30d1939SAndy Fiddaman 		offset = (Vmuchar_t*)addr - (Vmuchar_t*)tp;
149*b30d1939SAndy Fiddaman 		goto done;
150*b30d1939SAndy Fiddaman 	}
151*b30d1939SAndy Fiddaman 
152*b30d1939SAndy Fiddaman done :
153*b30d1939SAndy Fiddaman 	CLRLOCK(vm, local);
154*b30d1939SAndy Fiddaman 
155*b30d1939SAndy Fiddaman 	return offset;
156*b30d1939SAndy Fiddaman }
157*b30d1939SAndy Fiddaman 
158*b30d1939SAndy Fiddaman #if __STD_C
poolfree(reg Vmalloc_t * vm,reg Void_t * data,int local)159*b30d1939SAndy Fiddaman static int poolfree(reg Vmalloc_t* vm, reg Void_t* data, int local )
160*b30d1939SAndy Fiddaman #else
161*b30d1939SAndy Fiddaman static int poolfree(vm, data, local)
162*b30d1939SAndy Fiddaman Vmalloc_t*	vm;
163*b30d1939SAndy Fiddaman Void_t*		data;
164*b30d1939SAndy Fiddaman int		local;
165*b30d1939SAndy Fiddaman #endif
166*b30d1939SAndy Fiddaman {
167*b30d1939SAndy Fiddaman 	Block_t		*bp;
168*b30d1939SAndy Fiddaman 	Vmdata_t	*vd = vm->data;
169*b30d1939SAndy Fiddaman 
170*b30d1939SAndy Fiddaman 	if(!data)
171*b30d1939SAndy Fiddaman 		return 0;
172*b30d1939SAndy Fiddaman 	if(vd->pool <= 0)
173*b30d1939SAndy Fiddaman 		return -1;
174*b30d1939SAndy Fiddaman 
175*b30d1939SAndy Fiddaman 	SETLOCK(vm, local);
176*b30d1939SAndy Fiddaman 
177*b30d1939SAndy Fiddaman 	/**/ASSERT(KPVADDR(vm, data, pooladdr) == 0);
178*b30d1939SAndy Fiddaman 	bp = (Block_t*)data;
179*b30d1939SAndy Fiddaman 	SIZE(bp) = POOLFREE;
180*b30d1939SAndy Fiddaman 	SEGLINK(bp) = vd->free;
181*b30d1939SAndy Fiddaman 	vd->free = bp;
182*b30d1939SAndy Fiddaman 
183*b30d1939SAndy Fiddaman 	if(!local && (vd->mode&VM_TRACE) && _Vmtrace)
184*b30d1939SAndy Fiddaman 		(*_Vmtrace)(vm, (Vmuchar_t*)data, NIL(Vmuchar_t*), vd->pool, 0);
185*b30d1939SAndy Fiddaman 
186*b30d1939SAndy Fiddaman 	CLRLOCK(vm, local);
187*b30d1939SAndy Fiddaman 
188*b30d1939SAndy Fiddaman 	return 0;
189*b30d1939SAndy Fiddaman }
190*b30d1939SAndy Fiddaman 
191*b30d1939SAndy Fiddaman #if __STD_C
poolresize(Vmalloc_t * vm,Void_t * data,size_t size,int type,int local)192*b30d1939SAndy Fiddaman static Void_t* poolresize(Vmalloc_t* vm, Void_t* data, size_t size, int type, int local )
193*b30d1939SAndy Fiddaman #else
194*b30d1939SAndy Fiddaman static Void_t* poolresize(vm, data, size, type, local )
195*b30d1939SAndy Fiddaman Vmalloc_t*	vm;
196*b30d1939SAndy Fiddaman Void_t*		data;
197*b30d1939SAndy Fiddaman size_t		size;
198*b30d1939SAndy Fiddaman int		type;
199*b30d1939SAndy Fiddaman int		local;
200*b30d1939SAndy Fiddaman #endif
201*b30d1939SAndy Fiddaman {
202*b30d1939SAndy Fiddaman 	Vmdata_t	*vd = vm->data;
203*b30d1939SAndy Fiddaman 
204*b30d1939SAndy Fiddaman 	NOTUSED(type);
205*b30d1939SAndy Fiddaman 
206*b30d1939SAndy Fiddaman 	if(!data)
207*b30d1939SAndy Fiddaman 	{	data = poolalloc(vm, size, local);
208*b30d1939SAndy Fiddaman 		if(data && (type&VM_RSZERO) )
209*b30d1939SAndy Fiddaman 			memset(data, 0, size);
210*b30d1939SAndy Fiddaman 		return data;
211*b30d1939SAndy Fiddaman 	}
212*b30d1939SAndy Fiddaman 	if(size == 0)
213*b30d1939SAndy Fiddaman 	{	(void)poolfree(vm, data, local);
214*b30d1939SAndy Fiddaman 		return NIL(Void_t*);
215*b30d1939SAndy Fiddaman 	}
216*b30d1939SAndy Fiddaman 	if(size != vd->pool)
217*b30d1939SAndy Fiddaman 		return NIL(Void_t*);
218*b30d1939SAndy Fiddaman 
219*b30d1939SAndy Fiddaman 	SETLOCK(vm, local);
220*b30d1939SAndy Fiddaman 
221*b30d1939SAndy Fiddaman 	/**/ASSERT(KPVADDR(vm, data, pooladdr) == 0);
222*b30d1939SAndy Fiddaman 
223*b30d1939SAndy Fiddaman 	if(!local && (vd->mode&VM_TRACE) && _Vmtrace)
224*b30d1939SAndy Fiddaman 		(*_Vmtrace)(vm, (Vmuchar_t*)data, (Vmuchar_t*)data, size, 0);
225*b30d1939SAndy Fiddaman 
226*b30d1939SAndy Fiddaman 	CLRLOCK(vm, local);
227*b30d1939SAndy Fiddaman 
228*b30d1939SAndy Fiddaman 	return data;
229*b30d1939SAndy Fiddaman }
230*b30d1939SAndy Fiddaman 
231*b30d1939SAndy Fiddaman #if __STD_C
poolsize(Vmalloc_t * vm,Void_t * addr,int local)232*b30d1939SAndy Fiddaman static long poolsize(Vmalloc_t* vm, Void_t* addr, int local)
233*b30d1939SAndy Fiddaman #else
234*b30d1939SAndy Fiddaman static long poolsize(vm, addr, local)
235*b30d1939SAndy Fiddaman Vmalloc_t*	vm;
236*b30d1939SAndy Fiddaman Void_t*		addr;
237*b30d1939SAndy Fiddaman int		local;
238*b30d1939SAndy Fiddaman #endif
239*b30d1939SAndy Fiddaman {
240*b30d1939SAndy Fiddaman 	return pooladdr(vm, addr, local) == 0 ? (long)vm->data->pool : -1L;
241*b30d1939SAndy Fiddaman }
242*b30d1939SAndy Fiddaman 
243*b30d1939SAndy Fiddaman #if __STD_C
poolcompact(Vmalloc_t * vm,int local)244*b30d1939SAndy Fiddaman static int poolcompact(Vmalloc_t* vm, int local)
245*b30d1939SAndy Fiddaman #else
246*b30d1939SAndy Fiddaman static int poolcompact(vm, local)
247*b30d1939SAndy Fiddaman Vmalloc_t*	vm;
248*b30d1939SAndy Fiddaman int		local;
249*b30d1939SAndy Fiddaman #endif
250*b30d1939SAndy Fiddaman {
251*b30d1939SAndy Fiddaman 	ssize_t		s;
252*b30d1939SAndy Fiddaman 	Block_t		*fp;
253*b30d1939SAndy Fiddaman 	Seg_t		*seg, *next;
254*b30d1939SAndy Fiddaman 	Vmdata_t	*vd = vm->data;
255*b30d1939SAndy Fiddaman 
256*b30d1939SAndy Fiddaman 	SETLOCK(vm, local);
257*b30d1939SAndy Fiddaman 
258*b30d1939SAndy Fiddaman 	for(seg = vd->seg; seg; seg = next)
259*b30d1939SAndy Fiddaman 	{	next = seg->next;
260*b30d1939SAndy Fiddaman 
261*b30d1939SAndy Fiddaman 		if(!(fp = seg->free))
262*b30d1939SAndy Fiddaman 			continue;
263*b30d1939SAndy Fiddaman 
264*b30d1939SAndy Fiddaman 		seg->free = NIL(Block_t*);
265*b30d1939SAndy Fiddaman 		if(seg->size == (s = SIZE(fp)&~BITS))
266*b30d1939SAndy Fiddaman 			s = seg->extent;
267*b30d1939SAndy Fiddaman 		else	s += sizeof(Head_t);
268*b30d1939SAndy Fiddaman 
269*b30d1939SAndy Fiddaman 		if((*_Vmtruncate)(vm,seg,s,1) == s)
270*b30d1939SAndy Fiddaman 			seg->free = fp;
271*b30d1939SAndy Fiddaman 	}
272*b30d1939SAndy Fiddaman 
273*b30d1939SAndy Fiddaman 	if(!local && (vd->mode&VM_TRACE) && _Vmtrace)
274*b30d1939SAndy Fiddaman 		(*_Vmtrace)(vm, (Vmuchar_t*)0, (Vmuchar_t*)0, 0, 0);
275*b30d1939SAndy Fiddaman 
276*b30d1939SAndy Fiddaman 	CLRLOCK(vm, local);
277*b30d1939SAndy Fiddaman 
278*b30d1939SAndy Fiddaman 	return 0;
279*b30d1939SAndy Fiddaman }
280*b30d1939SAndy Fiddaman 
281*b30d1939SAndy Fiddaman #if __STD_C
poolalign(Vmalloc_t * vm,size_t size,size_t align,int local)282*b30d1939SAndy Fiddaman static Void_t* poolalign(Vmalloc_t* vm, size_t size, size_t align, int local)
283*b30d1939SAndy Fiddaman #else
284*b30d1939SAndy Fiddaman static Void_t* poolalign(vm, size, align, local)
285*b30d1939SAndy Fiddaman Vmalloc_t*	vm;
286*b30d1939SAndy Fiddaman size_t		size;
287*b30d1939SAndy Fiddaman size_t		align;
288*b30d1939SAndy Fiddaman int		local;
289*b30d1939SAndy Fiddaman #endif
290*b30d1939SAndy Fiddaman {
291*b30d1939SAndy Fiddaman 	NOTUSED(vm);
292*b30d1939SAndy Fiddaman 	NOTUSED(size);
293*b30d1939SAndy Fiddaman 	NOTUSED(align);
294*b30d1939SAndy Fiddaman 	return NIL(Void_t*);
295*b30d1939SAndy Fiddaman }
296*b30d1939SAndy Fiddaman 
297*b30d1939SAndy Fiddaman /* Public interface */
298*b30d1939SAndy Fiddaman static Vmethod_t _Vmpool =
299*b30d1939SAndy Fiddaman {
300*b30d1939SAndy Fiddaman 	poolalloc,
301*b30d1939SAndy Fiddaman 	poolresize,
302*b30d1939SAndy Fiddaman 	poolfree,
303*b30d1939SAndy Fiddaman 	pooladdr,
304*b30d1939SAndy Fiddaman 	poolsize,
305*b30d1939SAndy Fiddaman 	poolcompact,
306*b30d1939SAndy Fiddaman 	poolalign,
307*b30d1939SAndy Fiddaman 	VM_MTPOOL
308*b30d1939SAndy Fiddaman };
309*b30d1939SAndy Fiddaman 
310*b30d1939SAndy Fiddaman __DEFINE__(Vmethod_t*,Vmpool,&_Vmpool);
311*b30d1939SAndy Fiddaman 
312*b30d1939SAndy Fiddaman #ifdef NoF
313*b30d1939SAndy Fiddaman NoF(vmpool)
314*b30d1939SAndy Fiddaman #endif
315*b30d1939SAndy Fiddaman 
316*b30d1939SAndy Fiddaman #endif
317