1da2e3ebdSchin /***********************************************************************
2da2e3ebdSchin *                                                                      *
3da2e3ebdSchin *               This software is part of the ast package               *
4*b30d1939SAndy Fiddaman *          Copyright (c) 1985-2011 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 #pragma prototyped
23da2e3ebdSchin /*
24da2e3ebdSchin  * pointer stack routines
25da2e3ebdSchin  */
26da2e3ebdSchin 
27da2e3ebdSchin static const char id_stack[] = "\n@(#)$Id: stack (AT&T Bell Laboratories) 1984-05-01 $\0\n";
28da2e3ebdSchin 
29da2e3ebdSchin #include <ast.h>
30da2e3ebdSchin #include <stack.h>
31da2e3ebdSchin 
32da2e3ebdSchin /*
33da2e3ebdSchin  * create a new stack
34da2e3ebdSchin  */
35da2e3ebdSchin 
36da2e3ebdSchin STACK
stackalloc(register int size,void * error)37da2e3ebdSchin stackalloc(register int size, void* error)
38da2e3ebdSchin {
39da2e3ebdSchin 	register STACK			stack;
40da2e3ebdSchin 	register struct stackblock	*b;
41da2e3ebdSchin 
42da2e3ebdSchin 	if (size <= 0) size = 100;
43da2e3ebdSchin 	if (!(stack = newof(0, struct stacktable, 1, 0))) return(0);
44da2e3ebdSchin 	if (!(b = newof(0, struct stackblock, 1, 0)))
45da2e3ebdSchin 	{
46da2e3ebdSchin 		free(stack);
47da2e3ebdSchin 		return(0);
48da2e3ebdSchin 	}
49da2e3ebdSchin 	if (!(b->stack = newof(0, void*, size, 0)))
50da2e3ebdSchin 	{
51da2e3ebdSchin 		free(b);
52da2e3ebdSchin 		free(stack);
53da2e3ebdSchin 		return(0);
54da2e3ebdSchin 	}
55da2e3ebdSchin 	stack->blocks = b;
56da2e3ebdSchin 	stack->size = size;
57da2e3ebdSchin 	stack->error = error;
58da2e3ebdSchin 	stack->position.block = b;
59da2e3ebdSchin 	stack->position.index = -1;
60da2e3ebdSchin 	b->next = 0;
61da2e3ebdSchin 	b->prev = 0;
62da2e3ebdSchin 	return(stack);
63da2e3ebdSchin }
64da2e3ebdSchin 
65da2e3ebdSchin /*
66da2e3ebdSchin  * remove a stack
67da2e3ebdSchin  */
68da2e3ebdSchin 
69da2e3ebdSchin void
stackfree(register STACK stack)70da2e3ebdSchin stackfree(register STACK stack)
71da2e3ebdSchin {
72da2e3ebdSchin 	register struct stackblock*	b;
73da2e3ebdSchin 	register struct stackblock*	p;
74da2e3ebdSchin 
75da2e3ebdSchin 	b = stack->blocks;
76da2e3ebdSchin 	while (p = b)
77da2e3ebdSchin 	{
78da2e3ebdSchin 		b = p->next;
79da2e3ebdSchin 		free(p->stack);
80da2e3ebdSchin 		free(p);
81da2e3ebdSchin 	}
82da2e3ebdSchin 	free(stack);
83da2e3ebdSchin }
84da2e3ebdSchin 
85da2e3ebdSchin /*
86da2e3ebdSchin  * clear stack
87da2e3ebdSchin  */
88da2e3ebdSchin 
89da2e3ebdSchin void
stackclear(register STACK stack)90da2e3ebdSchin stackclear(register STACK stack)
91da2e3ebdSchin {
92da2e3ebdSchin 	stack->position.block = stack->blocks;
93da2e3ebdSchin 	stack->position.index = -1;
94da2e3ebdSchin }
95da2e3ebdSchin 
96da2e3ebdSchin /*
97da2e3ebdSchin  * get value on top of stack
98da2e3ebdSchin  */
99da2e3ebdSchin 
100da2e3ebdSchin void*
stackget(register STACK stack)101da2e3ebdSchin stackget(register STACK stack)
102da2e3ebdSchin {
103da2e3ebdSchin 	if (stack->position.index < 0) return(stack->error);
104da2e3ebdSchin 	else return(stack->position.block->stack[stack->position.index]);
105da2e3ebdSchin }
106da2e3ebdSchin 
107da2e3ebdSchin /*
108da2e3ebdSchin  * push value on to stack
109da2e3ebdSchin  */
110da2e3ebdSchin 
111da2e3ebdSchin int
stackpush(register STACK stack,void * value)112da2e3ebdSchin stackpush(register STACK stack, void* value)
113da2e3ebdSchin {
114da2e3ebdSchin 	register struct stackblock	*b;
115da2e3ebdSchin 
116da2e3ebdSchin 	if (++stack->position.index >= stack->size)
117da2e3ebdSchin 	{
118da2e3ebdSchin 		b = stack->position.block;
119da2e3ebdSchin 		if (b->next) b = b->next;
120da2e3ebdSchin 		else
121da2e3ebdSchin 		{
122da2e3ebdSchin 			if (!(b->next = newof(0, struct stackblock, 1, 0)))
123da2e3ebdSchin 				return(-1);
124da2e3ebdSchin 			b = b->next;
125da2e3ebdSchin 			if (!(b->stack = newof(0, void*, stack->size, 0)))
126da2e3ebdSchin 				return(-1);
127da2e3ebdSchin 			b->prev = stack->position.block;
128da2e3ebdSchin 			b->next = 0;
129da2e3ebdSchin 		}
130da2e3ebdSchin 		stack->position.block = b;
131da2e3ebdSchin 		stack->position.index = 0;
132da2e3ebdSchin 	}
133da2e3ebdSchin 	stack->position.block->stack[stack->position.index] = value;
134da2e3ebdSchin 	return(0);
135da2e3ebdSchin }
136da2e3ebdSchin 
137da2e3ebdSchin /*
138da2e3ebdSchin  * pop value off stack
139da2e3ebdSchin  */
140da2e3ebdSchin 
141da2e3ebdSchin int
stackpop(register STACK stack)142da2e3ebdSchin stackpop(register STACK stack)
143da2e3ebdSchin {
144da2e3ebdSchin 	/*
145da2e3ebdSchin 	 * return:
146da2e3ebdSchin 	 *
147da2e3ebdSchin 	 *	-1	if stack empty before pop
148da2e3ebdSchin 	 *	 0	if stack empty after pop
149da2e3ebdSchin 	 *	 1	if stack not empty before & after pop
150da2e3ebdSchin 	 */
151da2e3ebdSchin 
152da2e3ebdSchin 	if (stack->position.index < 0) return(-1);
153da2e3ebdSchin 	else if (--stack->position.index < 0)
154da2e3ebdSchin 	{
155da2e3ebdSchin 		if (!stack->position.block->prev) return(0);
156da2e3ebdSchin 		stack->position.block = stack->position.block->prev;
157da2e3ebdSchin 		stack->position.index = stack->size - 1;
158da2e3ebdSchin 		return(1);
159da2e3ebdSchin 	}
160da2e3ebdSchin 	else return(1);
161da2e3ebdSchin }
162da2e3ebdSchin 
163da2e3ebdSchin /*
164da2e3ebdSchin  * set|get stack position
165da2e3ebdSchin  */
166da2e3ebdSchin 
167da2e3ebdSchin void
stacktell(register STACK stack,int set,STACKPOS * position)168da2e3ebdSchin stacktell(register STACK stack, int set, STACKPOS* position)
169da2e3ebdSchin {
170da2e3ebdSchin 	if (set) stack->position = *position;
171da2e3ebdSchin 	else *position = stack->position;
172da2e3ebdSchin }
173