1da2e3ebdSchin /***********************************************************************
2da2e3ebdSchin * *
3da2e3ebdSchin * This software is part of the ast package *
4*3e14f97fSRoger A. Faulkner * Copyright (c) 1985-2010 AT&T Intellectual Property *
5da2e3ebdSchin * and is licensed under the *
6da2e3ebdSchin * Common Public License, Version 1.0 *
77c2fbfb3SApril Chin * by AT&T Intellectual Property *
8da2e3ebdSchin * *
9da2e3ebdSchin * A copy of the License is available at *
10da2e3ebdSchin * http://www.opensource.org/licenses/cpl1.0.txt *
11da2e3ebdSchin * (with md5 checksum 059e8cd6165cb4c31e351f2b69388fd9) *
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