1da2e3ebdSchin /***********************************************************************
2da2e3ebdSchin *                                                                      *
3da2e3ebdSchin *               This software is part of the ast package               *
4*b30d1939SAndy Fiddaman *          Copyright (c) 1985-2012 AT&T Intellectual Property          *
5da2e3ebdSchin *                      and is licensed under the                       *
6*b30d1939SAndy Fiddaman *                 Eclipse Public License, Version 1.0                  *
77c2fbfb3SApril Chin *                    by AT&T Intellectual Property                     *
8da2e3ebdSchin *                                                                      *
9da2e3ebdSchin *                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)         *
12da2e3ebdSchin *                                                                      *
13da2e3ebdSchin *              Information and Software Systems Research               *
14da2e3ebdSchin *                            AT&T Research                             *
15da2e3ebdSchin *                           Florham Park NJ                            *
16da2e3ebdSchin *                                                                      *
17da2e3ebdSchin *                 Glenn Fowler <gsf@research.att.com>                  *
18da2e3ebdSchin *                  David Korn <dgk@research.att.com>                   *
19da2e3ebdSchin *                   Phong Vo <kpv@research.att.com>                    *
20da2e3ebdSchin *                                                                      *
21da2e3ebdSchin ***********************************************************************/
22da2e3ebdSchin #if defined(_UWIN) && defined(_BLD_ast)
23da2e3ebdSchin 
_STUB_vmlast()24da2e3ebdSchin void _STUB_vmlast(){}
25da2e3ebdSchin 
26da2e3ebdSchin #else
27da2e3ebdSchin 
28da2e3ebdSchin #include	"vmhdr.h"
29da2e3ebdSchin 
30da2e3ebdSchin /*	Allocation with freeing and reallocing of last allocated block only.
31da2e3ebdSchin **
32da2e3ebdSchin **	Written by Kiem-Phong Vo, kpv@research.att.com, 01/16/94.
33da2e3ebdSchin */
34da2e3ebdSchin 
35da2e3ebdSchin #if __STD_C
lastalloc(Vmalloc_t * vm,size_t size,int local)36*b30d1939SAndy Fiddaman static Void_t* lastalloc(Vmalloc_t* vm, size_t size, int local)
37da2e3ebdSchin #else
38*b30d1939SAndy Fiddaman static Void_t* lastalloc(vm, size, local)
39da2e3ebdSchin Vmalloc_t*	vm;
40da2e3ebdSchin size_t		size;
41*b30d1939SAndy Fiddaman int		local;
42da2e3ebdSchin #endif
43da2e3ebdSchin {
44*b30d1939SAndy Fiddaman 	Block_t		*tp, *next;
45*b30d1939SAndy Fiddaman 	Seg_t		*seg, *last;
46*b30d1939SAndy Fiddaman 	size_t		s;
47*b30d1939SAndy Fiddaman 	Vmdata_t	*vd = vm->data;
48*b30d1939SAndy Fiddaman 	size_t		orgsize = size;
49*b30d1939SAndy Fiddaman 
50*b30d1939SAndy Fiddaman 	SETLOCK(vm, local);
51da2e3ebdSchin 
52da2e3ebdSchin 	size = size < ALIGN ? ALIGN : ROUND(size,ALIGN);
53*b30d1939SAndy Fiddaman 	for(last = NIL(Seg_t*), seg = vd->seg; seg; last = seg, seg = seg->next)
54*b30d1939SAndy Fiddaman 	{	if(!(tp = seg->free) || (SIZE(tp)+sizeof(Head_t)) < size)
55*b30d1939SAndy Fiddaman 			continue;
56*b30d1939SAndy Fiddaman 		if(last)
57*b30d1939SAndy Fiddaman 		{	last->next = seg->next;
58*b30d1939SAndy Fiddaman 			seg->next = vd->seg;
59*b30d1939SAndy Fiddaman 			vd->seg = seg;
60da2e3ebdSchin 		}
61*b30d1939SAndy Fiddaman 		goto got_block;
62*b30d1939SAndy Fiddaman 	}
63da2e3ebdSchin 
64*b30d1939SAndy Fiddaman 	/* there is no usable free space in region, try extending */
65*b30d1939SAndy Fiddaman 	if((tp = (*_Vmextend)(vm,size,NIL(Vmsearch_f))) )
66*b30d1939SAndy Fiddaman 	{	seg = SEG(tp);
67*b30d1939SAndy Fiddaman 		goto got_block;
68da2e3ebdSchin 	}
69*b30d1939SAndy Fiddaman 	else	goto done;
70da2e3ebdSchin 
71da2e3ebdSchin got_block:
72da2e3ebdSchin 	if((s = SIZE(tp)) >= size)
73da2e3ebdSchin 	{	next = (Block_t*)((Vmuchar_t*)tp+size);
74da2e3ebdSchin 		SIZE(next) = s - size;
75da2e3ebdSchin 		SEG(next) = seg;
76da2e3ebdSchin 		seg->free = next;
77da2e3ebdSchin 	}
78da2e3ebdSchin 	else	seg->free = NIL(Block_t*);
79da2e3ebdSchin 
80da2e3ebdSchin 	vd->free = seg->last = tp;
81da2e3ebdSchin 
82da2e3ebdSchin 	if(!local && (vd->mode&VM_TRACE) && _Vmtrace)
83da2e3ebdSchin 		(*_Vmtrace)(vm, NIL(Vmuchar_t*), (Vmuchar_t*)tp, orgsize, 0);
84da2e3ebdSchin 
85da2e3ebdSchin done:
86*b30d1939SAndy Fiddaman 	CLRLOCK(vm, local);
87*b30d1939SAndy Fiddaman 
88da2e3ebdSchin 	return (Void_t*)tp;
89da2e3ebdSchin }
90da2e3ebdSchin 
91da2e3ebdSchin #if __STD_C
lastfree(Vmalloc_t * vm,reg Void_t * data,int local)92*b30d1939SAndy Fiddaman static int lastfree(Vmalloc_t* vm, reg Void_t* data, int local )
93da2e3ebdSchin #else
94*b30d1939SAndy Fiddaman static int lastfree(vm, data, local)
95da2e3ebdSchin Vmalloc_t*	vm;
96*b30d1939SAndy Fiddaman Void_t*		data;
97*b30d1939SAndy Fiddaman int		local;
98da2e3ebdSchin #endif
99da2e3ebdSchin {
100*b30d1939SAndy Fiddaman 	Seg_t		*seg;
101*b30d1939SAndy Fiddaman 	Block_t		*fp;
102*b30d1939SAndy Fiddaman 	size_t		s;
103*b30d1939SAndy Fiddaman 	Vmdata_t	*vd = vm->data;
104da2e3ebdSchin 
105da2e3ebdSchin 	if(!data)
106da2e3ebdSchin 		return 0;
1077c2fbfb3SApril Chin 
108*b30d1939SAndy Fiddaman 	SETLOCK(vm, local);
109*b30d1939SAndy Fiddaman 
110da2e3ebdSchin 	if(data != (Void_t*)vd->free)
111*b30d1939SAndy Fiddaman 		data = NIL(Void_t*); /* signaling an error */
112*b30d1939SAndy Fiddaman 	else
113*b30d1939SAndy Fiddaman 	{	seg = vd->seg;
114*b30d1939SAndy Fiddaman 		if(!local && (vd->mode&VM_TRACE) && _Vmtrace)
115*b30d1939SAndy Fiddaman 		{	if(seg->free )
116*b30d1939SAndy Fiddaman 				s = (Vmuchar_t*)(seg->free) - (Vmuchar_t*)data;
117*b30d1939SAndy Fiddaman 			else	s = (Vmuchar_t*)BLOCK(seg->baddr) - (Vmuchar_t*)data;
118*b30d1939SAndy Fiddaman 			(*_Vmtrace)(vm, (Vmuchar_t*)data, NIL(Vmuchar_t*), s, 0);
119*b30d1939SAndy Fiddaman 		}
120da2e3ebdSchin 
121*b30d1939SAndy Fiddaman 		vd->free = NIL(Block_t*);
122*b30d1939SAndy Fiddaman 		fp = (Block_t*)data;
123*b30d1939SAndy Fiddaman 		SEG(fp)  = seg;
124*b30d1939SAndy Fiddaman 		SIZE(fp) = ((Vmuchar_t*)BLOCK(seg->baddr) - (Vmuchar_t*)data) - sizeof(Head_t);
125*b30d1939SAndy Fiddaman 		seg->free = fp;
126*b30d1939SAndy Fiddaman 		seg->last = NIL(Block_t*);
127da2e3ebdSchin 	}
128da2e3ebdSchin 
129*b30d1939SAndy Fiddaman 	CLRLOCK(vm, local);
130da2e3ebdSchin 
131*b30d1939SAndy Fiddaman 	return data ? 0 : -1;
132da2e3ebdSchin }
133da2e3ebdSchin 
134da2e3ebdSchin #if __STD_C
lastresize(Vmalloc_t * vm,reg Void_t * data,size_t size,int type,int local)135*b30d1939SAndy Fiddaman static Void_t* lastresize(Vmalloc_t* vm, reg Void_t* data, size_t size, int type, int local)
136da2e3ebdSchin #else
137*b30d1939SAndy Fiddaman static Void_t* lastresize(vm, data, size, type, local )
138da2e3ebdSchin Vmalloc_t*	vm;
139da2e3ebdSchin reg Void_t*	data;
140da2e3ebdSchin size_t		size;
141da2e3ebdSchin int		type;
142*b30d1939SAndy Fiddaman int		local;
143da2e3ebdSchin #endif
144da2e3ebdSchin {
145*b30d1939SAndy Fiddaman 	Block_t		*tp;
146*b30d1939SAndy Fiddaman 	Seg_t		*seg;
147*b30d1939SAndy Fiddaman 	ssize_t		s, ds;
148*b30d1939SAndy Fiddaman 	Void_t		*addr;
149*b30d1939SAndy Fiddaman 	size_t		oldsize = 0;
150*b30d1939SAndy Fiddaman 	Void_t		*orgdata = data;
151*b30d1939SAndy Fiddaman 	size_t		orgsize = size;
152*b30d1939SAndy Fiddaman 	Vmdata_t	*vd = vm->data;
153*b30d1939SAndy Fiddaman 
154da2e3ebdSchin 	if(!data)
155*b30d1939SAndy Fiddaman 	{	data = lastalloc(vm, size, local);
156*b30d1939SAndy Fiddaman 		if(data && (type&VM_RSZERO) )
157*b30d1939SAndy Fiddaman 			memset(data, 0, size);
158*b30d1939SAndy Fiddaman 		return data;
159da2e3ebdSchin 	}
160da2e3ebdSchin 	if(size <= 0)
161*b30d1939SAndy Fiddaman 	{	(void)lastfree(vm, data, local);
162da2e3ebdSchin 		return NIL(Void_t*);
163da2e3ebdSchin 	}
164da2e3ebdSchin 
165*b30d1939SAndy Fiddaman 	SETLOCK(vm, local);
166da2e3ebdSchin 
167da2e3ebdSchin 	if(data == (Void_t*)vd->free)
168da2e3ebdSchin 		seg = vd->seg;
169da2e3ebdSchin 	else
170da2e3ebdSchin 	{	/* see if it was one of ours */
171da2e3ebdSchin 		for(seg = vd->seg; seg; seg = seg->next)
172da2e3ebdSchin 			if(data >= seg->addr && data < (Void_t*)seg->baddr)
173da2e3ebdSchin 				break;
174da2e3ebdSchin 		if(!seg || (VLONG(data)%ALIGN) != 0 ||
175da2e3ebdSchin 		   (seg->last && (Vmuchar_t*)data > (Vmuchar_t*)seg->last) )
176*b30d1939SAndy Fiddaman 		{	data = NIL(Void_t*);
177*b30d1939SAndy Fiddaman 			goto done;
178da2e3ebdSchin 		}
179da2e3ebdSchin 	}
180da2e3ebdSchin 
181da2e3ebdSchin 	/* set 's' to be the current available space */
182da2e3ebdSchin 	if(data != seg->last)
183da2e3ebdSchin 	{	if(seg->last && (Vmuchar_t*)data < (Vmuchar_t*)seg->last)
184da2e3ebdSchin 			oldsize = (Vmuchar_t*)seg->last - (Vmuchar_t*)data;
185da2e3ebdSchin 		else	oldsize = (Vmuchar_t*)BLOCK(seg->baddr) - (Vmuchar_t*)data;
186da2e3ebdSchin 		s = -1;
187da2e3ebdSchin 	}
188da2e3ebdSchin 	else
189da2e3ebdSchin 	{	s = (Vmuchar_t*)BLOCK(seg->baddr) - (Vmuchar_t*)data;
190da2e3ebdSchin 		if(!(tp = seg->free) )
191da2e3ebdSchin 			oldsize = s;
192da2e3ebdSchin 		else
193da2e3ebdSchin 		{	oldsize = (Vmuchar_t*)tp - (Vmuchar_t*)data;
194da2e3ebdSchin 			seg->free = NIL(Block_t*);
195da2e3ebdSchin 		}
196da2e3ebdSchin 	}
197da2e3ebdSchin 
198da2e3ebdSchin 	size = size < ALIGN ? ALIGN : ROUND(size,ALIGN);
199da2e3ebdSchin 	if(s < 0 || (ssize_t)size > s)
200da2e3ebdSchin 	{	if(s >= 0) /* amount to extend */
201da2e3ebdSchin 		{	ds = size-s; ds = ROUND(ds,vd->incr);
202da2e3ebdSchin 			addr = (*vm->disc->memoryf)(vm, seg->addr, seg->extent,
203da2e3ebdSchin 						    seg->extent+ds, vm->disc);
204da2e3ebdSchin 			if(addr == seg->addr)
205da2e3ebdSchin 			{	s += ds;
206da2e3ebdSchin 				seg->size += ds;
207da2e3ebdSchin 				seg->extent += ds;
208da2e3ebdSchin 				seg->baddr += ds;
209da2e3ebdSchin 				SIZE(BLOCK(seg->baddr)) = BUSY;
210da2e3ebdSchin 			}
211da2e3ebdSchin 			else	goto do_alloc;
212da2e3ebdSchin 		}
213da2e3ebdSchin 		else
214da2e3ebdSchin 		{ do_alloc:
215da2e3ebdSchin 			if(!(type&(VM_RSMOVE|VM_RSCOPY)) )
216da2e3ebdSchin 				data = NIL(Void_t*);
217da2e3ebdSchin 			else
218da2e3ebdSchin 			{	tp = vd->free;
219da2e3ebdSchin 				if(!(addr = KPVALLOC(vm,size,lastalloc)) )
220da2e3ebdSchin 				{	vd->free = tp;
221da2e3ebdSchin 					data = NIL(Void_t*);
222da2e3ebdSchin 				}
223da2e3ebdSchin 				else
224da2e3ebdSchin 				{	if(type&VM_RSCOPY)
225da2e3ebdSchin 					{	ds = oldsize < size ? oldsize : size;
226da2e3ebdSchin 						memcpy(addr, data, ds);
227da2e3ebdSchin 					}
228da2e3ebdSchin 
229da2e3ebdSchin 					if(s >= 0 && seg != vd->seg)
230da2e3ebdSchin 					{	tp = (Block_t*)data;
231da2e3ebdSchin 						SEG(tp) = seg;
232da2e3ebdSchin 						SIZE(tp) = s - sizeof(Head_t);
233da2e3ebdSchin 						seg->free = tp;
234da2e3ebdSchin 					}
235da2e3ebdSchin 
236da2e3ebdSchin 					/* new block and size */
237da2e3ebdSchin 					data = addr;
238da2e3ebdSchin 					seg = vd->seg;
239da2e3ebdSchin 					s = (Vmuchar_t*)BLOCK(seg->baddr) -
240da2e3ebdSchin 					    (Vmuchar_t*)data;
241da2e3ebdSchin 					seg->free = NIL(Block_t*);
242da2e3ebdSchin 				}
243da2e3ebdSchin 			}
244da2e3ebdSchin 		}
245da2e3ebdSchin 	}
246da2e3ebdSchin 
247da2e3ebdSchin 	if(data)
248da2e3ebdSchin 	{	if(s >= (ssize_t)(size+sizeof(Head_t)) )
249da2e3ebdSchin 		{	tp = (Block_t*)((Vmuchar_t*)data + size);
250da2e3ebdSchin 			SEG(tp) = seg;
251da2e3ebdSchin 			SIZE(tp) = (s - size) - sizeof(Head_t);
252da2e3ebdSchin 			seg->free = tp;
253da2e3ebdSchin 		}
254da2e3ebdSchin 
255da2e3ebdSchin 		vd->free = seg->last = (Block_t*)data;
256da2e3ebdSchin 
257da2e3ebdSchin 		if(!local && (vd->mode&VM_TRACE) && _Vmtrace)
258da2e3ebdSchin 			(*_Vmtrace)(vm,(Vmuchar_t*)orgdata,(Vmuchar_t*)data,orgsize,0);
259da2e3ebdSchin 
260*b30d1939SAndy Fiddaman 		if((type&VM_RSZERO) && size > oldsize)
261*b30d1939SAndy Fiddaman 			memset((Void_t*)((Vmuchar_t*)data + oldsize), 0, size-oldsize);
262*b30d1939SAndy Fiddaman 	}
263da2e3ebdSchin 
264*b30d1939SAndy Fiddaman done:	CLRLOCK(vm, local);
265da2e3ebdSchin 
266da2e3ebdSchin 	return data;
267da2e3ebdSchin }
268da2e3ebdSchin 
269da2e3ebdSchin 
270da2e3ebdSchin #if __STD_C
lastaddr(Vmalloc_t * vm,Void_t * addr,int local)271*b30d1939SAndy Fiddaman static long lastaddr(Vmalloc_t* vm, Void_t* addr, int local)
272da2e3ebdSchin #else
273*b30d1939SAndy Fiddaman static long lastaddr(vm, addr, local)
274da2e3ebdSchin Vmalloc_t*	vm;
275da2e3ebdSchin Void_t*		addr;
276*b30d1939SAndy Fiddaman int		local;
277da2e3ebdSchin #endif
278da2e3ebdSchin {
279*b30d1939SAndy Fiddaman 	long		offset;
280*b30d1939SAndy Fiddaman 	Vmdata_t	*vd = vm->data;
281*b30d1939SAndy Fiddaman 
282*b30d1939SAndy Fiddaman 	SETLOCK(vm, local);
283da2e3ebdSchin 
284da2e3ebdSchin 	if(!vd->free || addr < (Void_t*)vd->free || addr >= (Void_t*)vd->seg->baddr)
285*b30d1939SAndy Fiddaman 		offset = -1L;
286*b30d1939SAndy Fiddaman 	else	offset = (long)((Vmuchar_t*)addr - (Vmuchar_t*)vd->free);
287*b30d1939SAndy Fiddaman 
288*b30d1939SAndy Fiddaman 	CLRLOCK(vm, local);
289*b30d1939SAndy Fiddaman 
290*b30d1939SAndy Fiddaman 	return offset;
291da2e3ebdSchin }
292da2e3ebdSchin 
293da2e3ebdSchin #if __STD_C
lastsize(Vmalloc_t * vm,Void_t * addr,int local)294*b30d1939SAndy Fiddaman static long lastsize(Vmalloc_t* vm, Void_t* addr, int local)
295da2e3ebdSchin #else
296*b30d1939SAndy Fiddaman static long lastsize(vm, addr, local)
297da2e3ebdSchin Vmalloc_t*	vm;
298da2e3ebdSchin Void_t*		addr;
299*b30d1939SAndy Fiddaman int		local;
300da2e3ebdSchin #endif
301da2e3ebdSchin {
302*b30d1939SAndy Fiddaman 	long		size;
303*b30d1939SAndy Fiddaman 	Vmdata_t	*vd = vm->data;
304*b30d1939SAndy Fiddaman 
305*b30d1939SAndy Fiddaman 	SETLOCK(vm, local);
306da2e3ebdSchin 
307da2e3ebdSchin 	if(!vd->free || addr != (Void_t*)vd->free )
308*b30d1939SAndy Fiddaman 		size = -1L;
309da2e3ebdSchin 	else if(vd->seg->free)
310*b30d1939SAndy Fiddaman 		size = (long)((Vmuchar_t*)vd->seg->free - (Vmuchar_t*)addr);
311*b30d1939SAndy Fiddaman 	else	size = (long)((Vmuchar_t*)vd->seg->baddr - (Vmuchar_t*)addr - sizeof(Head_t));
312*b30d1939SAndy Fiddaman 
313*b30d1939SAndy Fiddaman 	CLRLOCK(vm, local);
314*b30d1939SAndy Fiddaman 
315*b30d1939SAndy Fiddaman 	return size;
316da2e3ebdSchin }
317da2e3ebdSchin 
318da2e3ebdSchin #if __STD_C
lastcompact(Vmalloc_t * vm,int local)319*b30d1939SAndy Fiddaman static int lastcompact(Vmalloc_t* vm, int local)
320da2e3ebdSchin #else
321*b30d1939SAndy Fiddaman static int lastcompact(vm, local)
322da2e3ebdSchin Vmalloc_t*	vm;
323*b30d1939SAndy Fiddaman int		local;
324da2e3ebdSchin #endif
325da2e3ebdSchin {
326*b30d1939SAndy Fiddaman 	ssize_t		s;
327*b30d1939SAndy Fiddaman 	Block_t		*fp;
328*b30d1939SAndy Fiddaman 	Seg_t		*seg, *next;
329*b30d1939SAndy Fiddaman 	Vmdata_t	*vd = vm->data;
330*b30d1939SAndy Fiddaman 
331*b30d1939SAndy Fiddaman 	SETLOCK(vm, local);
332da2e3ebdSchin 
333da2e3ebdSchin 	for(seg = vd->seg; seg; seg = next)
334da2e3ebdSchin 	{	next = seg->next;
335da2e3ebdSchin 
336da2e3ebdSchin 		if(!(fp = seg->free))
337da2e3ebdSchin 			continue;
338da2e3ebdSchin 
339da2e3ebdSchin 		seg->free = NIL(Block_t*);
340da2e3ebdSchin 		if(seg->size == (s = SIZE(fp)&~BITS))
341da2e3ebdSchin 			s = seg->extent;
342da2e3ebdSchin 		else	s += sizeof(Head_t);
343da2e3ebdSchin 
344da2e3ebdSchin 		if((*_Vmtruncate)(vm,seg,s,1) == s)
345da2e3ebdSchin 			seg->free = fp;
346da2e3ebdSchin 	}
347da2e3ebdSchin 
348da2e3ebdSchin 	if((vd->mode&VM_TRACE) && _Vmtrace)
349da2e3ebdSchin 		(*_Vmtrace)(vm,(Vmuchar_t*)0,(Vmuchar_t*)0,0,0);
350da2e3ebdSchin 
351*b30d1939SAndy Fiddaman 	CLRLOCK(vm, local);
352da2e3ebdSchin 	return 0;
353da2e3ebdSchin }
354da2e3ebdSchin 
355da2e3ebdSchin #if __STD_C
lastalign(Vmalloc_t * vm,size_t size,size_t align,int local)356*b30d1939SAndy Fiddaman static Void_t* lastalign(Vmalloc_t* vm, size_t size, size_t align, int local)
357da2e3ebdSchin #else
358*b30d1939SAndy Fiddaman static Void_t* lastalign(vm, size, align, local)
359da2e3ebdSchin Vmalloc_t*	vm;
360da2e3ebdSchin size_t		size;
361da2e3ebdSchin size_t		align;
362*b30d1939SAndy Fiddaman int		local;
363da2e3ebdSchin #endif
364da2e3ebdSchin {
365*b30d1939SAndy Fiddaman 	Vmuchar_t	*data;
366*b30d1939SAndy Fiddaman 	Seg_t		*seg;
367*b30d1939SAndy Fiddaman 	Block_t		*next;
368*b30d1939SAndy Fiddaman 	size_t		s, orgsize = size, orgalign = align;
369*b30d1939SAndy Fiddaman 	Vmdata_t	*vd = vm->data;
370da2e3ebdSchin 
371da2e3ebdSchin 	if(size <= 0 || align <= 0)
372da2e3ebdSchin 		return NIL(Void_t*);
373da2e3ebdSchin 
374*b30d1939SAndy Fiddaman 	SETLOCK(vm, local);
375da2e3ebdSchin 
376da2e3ebdSchin 	size = size <= TINYSIZE ? TINYSIZE : ROUND(size,ALIGN);
377da2e3ebdSchin 	align = MULTIPLE(align,ALIGN);
378da2e3ebdSchin 
379da2e3ebdSchin 	s = size + align;
380da2e3ebdSchin 	if(!(data = (Vmuchar_t*)KPVALLOC(vm,s,lastalloc)) )
381da2e3ebdSchin 		goto done;
382da2e3ebdSchin 
383da2e3ebdSchin 	/* find the segment containing this block */
384da2e3ebdSchin 	for(seg = vd->seg; seg; seg = seg->next)
385da2e3ebdSchin 		if(seg->last == (Block_t*)data)
386da2e3ebdSchin 			break;
387da2e3ebdSchin 	/**/ASSERT(seg);
388da2e3ebdSchin 
389da2e3ebdSchin 	/* get a suitably aligned address */
390da2e3ebdSchin 	if((s = (size_t)(VLONG(data)%align)) != 0)
391da2e3ebdSchin 		data += align-s; /**/ASSERT((VLONG(data)%align) == 0);
392da2e3ebdSchin 
393da2e3ebdSchin 	/* free the unused tail */
394da2e3ebdSchin 	next = (Block_t*)(data+size);
395da2e3ebdSchin 	if((s = (seg->baddr - (Vmuchar_t*)next)) >= sizeof(Block_t))
396da2e3ebdSchin 	{	SEG(next) = seg;
397da2e3ebdSchin 		SIZE(next) = s - sizeof(Head_t);
398da2e3ebdSchin 		seg->free = next;
399da2e3ebdSchin 	}
400da2e3ebdSchin 
401da2e3ebdSchin 	vd->free = seg->last = (Block_t*)data;
402da2e3ebdSchin 
403*b30d1939SAndy Fiddaman 	if(!local && (vd->mode&VM_TRACE) && _Vmtrace)
404da2e3ebdSchin 		(*_Vmtrace)(vm,NIL(Vmuchar_t*),data,orgsize,orgalign);
405da2e3ebdSchin 
406da2e3ebdSchin done:
407*b30d1939SAndy Fiddaman 	CLRLOCK(vm, local);
408da2e3ebdSchin 
409da2e3ebdSchin 	return (Void_t*)data;
410da2e3ebdSchin }
411da2e3ebdSchin 
412da2e3ebdSchin /* Public method for free-1 allocation */
413da2e3ebdSchin static Vmethod_t _Vmlast =
414da2e3ebdSchin {
415da2e3ebdSchin 	lastalloc,
416da2e3ebdSchin 	lastresize,
417da2e3ebdSchin 	lastfree,
418da2e3ebdSchin 	lastaddr,
419da2e3ebdSchin 	lastsize,
420da2e3ebdSchin 	lastcompact,
421da2e3ebdSchin 	lastalign,
422da2e3ebdSchin 	VM_MTLAST
423da2e3ebdSchin };
424da2e3ebdSchin 
425da2e3ebdSchin __DEFINE__(Vmethod_t*,Vmlast,&_Vmlast);
426da2e3ebdSchin 
427da2e3ebdSchin #ifdef NoF
428da2e3ebdSchin NoF(vmlast)
429da2e3ebdSchin #endif
430da2e3ebdSchin 
431da2e3ebdSchin #endif
432