8322 nl: misleading-indentation
[unleashed/tickless.git] / usr / src / cmd / sh / blok.c
blobba29c41180ff01cc3c65f986f463bc238ce40e01
1 /*
2 * CDDL HEADER START
4 * The contents of this file are subject to the terms of the
5 * Common Development and Distribution License (the "License").
6 * You may not use this file except in compliance with the License.
8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9 * or http://www.opensolaris.org/os/licensing.
10 * See the License for the specific language governing permissions
11 * and limitations under the License.
13 * When distributing Covered Code, include this CDDL HEADER in each
14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15 * If applicable, add the following below this CDDL HEADER, with the
16 * fields enclosed by brackets "[]" replaced with your own identifying
17 * information: Portions Copyright [yyyy] [name of copyright owner]
19 * CDDL HEADER END
23 * Copyright 2009 Sun Microsystems, Inc. All rights reserved.
24 * Use is subject to license terms.
27 /* Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T */
28 /* All Rights Reserved */
31 * UNIX shell
34 #include "defs.h"
38 * storage allocator
39 * (circular first fit strategy)
42 #define BUSY 01
43 #define busy(x) (Rcheat((x)->word) & BUSY)
45 unsigned brkincr = BRKINCR;
46 struct blk *blokp; /* current search pointer */
47 struct blk *bloktop; /* top of arena (last blok) */
49 unsigned char *brkbegin;
50 extern unsigned char *setbrk();
52 #ifdef DEBUG
54 * If DEBUG is defined, the following testing will be performed:
55 * - chkbptr() checks the linkage of blocks by following links.
56 * Note that this makes shell really slow.
57 * - fill_pat() fills the memory block with pattern like umem does.
58 * The pattern used to fill the memory area is defined below.
60 #define PAT_MAGIC 0xfeedface
61 #define PAT_INIT 0xbaddcafe
62 #define PAT_FREE 0xdeadbeef
64 static void fill_pat(struct blk *, uint32_t);
65 static void chkbptr(struct blk *);
66 #endif
68 void *
69 alloc(size_t nbytes)
71 size_t rbytes = round(nbytes + ALIGNSIZ, ALIGNSIZ);
73 if (stakbot == 0) {
74 addblok((unsigned int)0);
77 for (;;) {
78 int c = 0;
79 struct blk *p = blokp;
80 struct blk *q;
84 if (!busy(p)) {
85 while (!busy(q = p->word))
86 p->word = q->word;
87 if ((char *)q - (char *)p >= rbytes) {
88 blokp = (struct blk *)
89 ((char *)p + rbytes);
90 if (q > blokp)
91 blokp->word = p->word;
92 p->word = (struct blk *)
93 (Rcheat(blokp) | BUSY);
94 #ifdef DEBUG
95 fill_pat(p, PAT_INIT);
96 #endif
97 return ((char *)(p + 1));
100 q = p;
101 p = (struct blk *)(Rcheat(p->word) & ~BUSY);
102 } while (p > q || (c++) == 0);
103 addblok(rbytes);
107 void
108 addblok(unsigned int reqd)
110 if (stakbot == 0) {
111 brkbegin = setbrk(3 * BRKINCR);
113 * setbrk() returns 8 byte aligned address
114 * but we could need larger align in future
116 brkbegin = (unsigned char *)round(brkbegin, ALIGNSIZ);
117 bloktop = (struct blk *)brkbegin;
120 if (stakbas != staktop) {
121 unsigned char *rndstak;
122 struct blk *blokstak;
124 if (staktop >= brkend)
125 growstak(staktop);
126 pushstak(0);
127 rndstak = (unsigned char *)round(staktop, ALIGNSIZ);
128 blokstak = (struct blk *)(stakbas) - 1;
129 blokstak->word = stakbsy;
130 stakbsy = blokstak;
131 bloktop->word = (struct blk *)(Rcheat(rndstak) | BUSY);
132 bloktop = (struct blk *)(rndstak);
134 reqd += brkincr;
135 reqd &= ~(brkincr - 1);
136 blokp = bloktop;
138 * brkend points to the first invalid address.
139 * make sure bloktop is valid.
141 if ((unsigned char *)&bloktop->word >= brkend) {
142 if (setbrk((unsigned)((unsigned char *)
143 (&bloktop->word) - brkend + sizeof (struct blk))) ==
144 (unsigned char *)-1)
145 error(nospace);
147 bloktop = bloktop->word = (struct blk *)(Rcheat(bloktop) + reqd);
148 if ((unsigned char *)&bloktop->word >= brkend) {
149 if (setbrk((unsigned)((unsigned char *)
150 (&bloktop->word) - brkend + sizeof (struct blk))) ==
151 (unsigned char *)-1)
152 error(nospace);
154 bloktop->word = (struct blk *)(brkbegin + 1);
156 unsigned char *stakadr = (unsigned char *)
157 (bloktop + 2);
158 unsigned char *sp = stakadr;
159 if (reqd = (staktop-stakbot)) {
160 if (stakadr + reqd >= brkend)
161 growstak(stakadr + reqd);
162 while (reqd-- > 0)
163 *sp++ = *stakbot++;
164 sp--;
166 staktop = sp;
167 if (staktop >= brkend)
168 growstak(staktop);
169 stakbas = stakbot = stakadr;
173 void
174 free(ap)
175 void *ap;
177 struct blk *p;
179 if ((p = (struct blk *)ap) && p < bloktop && p > (struct blk *)brkbegin)
181 #ifdef DEBUG
182 chkbptr(p);
183 #endif
184 --p;
185 p->word = (struct blk *)(Rcheat(p->word) & ~BUSY);
186 #ifdef DEBUG
187 fill_pat(p, PAT_FREE);
188 #endif
195 #ifdef DEBUG
197 static void
198 fill_pat(struct blk *ptr, uint32_t pat)
200 uint32_t *ui, *eui;
202 *(uint32_t *)ptr->pad = PAT_MAGIC;
203 eui = (uint32_t *)(Rcheat(ptr->word) & ~BUSY);
204 for (ui = (uint32_t *)(ptr + 1); ui < eui; ui++)
205 *ui = pat;
208 static void
209 chkbptr(struct blk *ptr)
211 int exf = 0;
212 struct blk *p = (struct blk *)brkbegin;
213 struct blk *q;
214 int us = 0, un = 0;
216 for (;;) {
217 q = (struct blk *)(Rcheat(p->word) & ~BUSY);
219 if (p+1 == ptr)
220 exf++;
222 if (q < (struct blk *)brkbegin || q > bloktop)
223 abort();
225 if (p == bloktop)
226 break;
228 if (busy(p))
229 us += q - p;
230 else
231 un += q - p;
233 if (p >= q)
234 abort();
236 p = q;
238 if (exf == 0)
239 abort();
242 static void
243 chkmem()
245 struct blk *p = (struct blk *)brkbegin;
246 struct blk *q;
247 int us = 0, un = 0;
249 for (;;) {
250 q = (struct blk *)(Rcheat(p->word) & ~BUSY);
252 if (q < (struct blk *)brkbegin || q > bloktop)
253 abort();
255 if (p == bloktop)
256 break;
258 if (busy(p))
259 us += q - p;
260 else
261 un += q - p;
263 if (p >= q)
264 abort();
266 p = q;
269 prs("un/used/avail ");
270 prn(un);
271 blank();
272 prn(us);
273 blank();
274 prn((uintptr_t)bloktop - (uintptr_t)brkbegin - (un + us));
275 newline();
279 #endif
281 size_t
282 blklen(q)
283 char *q;
285 struct blk *pp = (struct blk *)q;
286 struct blk *p;
288 --pp;
289 p = (struct blk *)(Rcheat(pp->word) & ~BUSY);
291 return ((size_t)((long)p - (long)q));
295 * This is a really hasty hack at putting realloc() in the shell, along
296 * with alloc() and free(). I really hate having to do things like this,
297 * hacking in something before I understand _why_ libcollate does any
298 * memory (re)allocation, let alone feel comfortable with this particular
299 * implementation of realloc, assuming it actually gets used by anything.
301 * I plan to revist this, for now this is just to get sh to compile so
302 * that xcu4 builds may be done and we get xcu4 on our desktops.
304 * Eric Brunner, 10/21/94
306 * Implemented a variation on the suggested fix in Trusted Solaris 2.5,
307 * then forward ported the fix into the mainline shell.
309 * 3/3/99
311 #ifdef __STDC__
312 void *
313 realloc(pp, nbytes)
314 void *pp;
315 size_t nbytes;
316 #else
317 char *
318 realloc(pp, nbytes)
319 char *pp;
320 size_t nbytes;
321 #endif
323 char *q;
324 size_t blen;
326 if (pp == NULL)
327 return (alloc(nbytes));
328 if ((nbytes == 0) && (pp != NULL))
329 free(pp);
331 blen = blklen(pp);
333 if (blen < nbytes) { /* need to grow */
334 q = alloc(nbytes);
335 memcpy(q, pp, blen);
336 free(pp);
337 return ((char *)q);
338 } else if (blen == nbytes) { /* do nothing */
339 return (pp);
340 } else { /* free excess */
341 q = alloc(nbytes);
342 memcpy(q, pp, nbytes);
343 free(pp);
344 return ((char *)q);
347 #ifdef undef
349 * all of what follows is the _idea_ of what is going to be done
350 * getting the size of the block is a problem -- what follows
351 * is _not_ "real", since "sizeof" isn't going to tell me any
352 * thing usefull, probably have to travers the list to the next
353 * blk, then subtract ptr addrs ... and be careful not to leave
354 * holes.
356 p = (struct blk *)pp;
357 if (sizeof (p) < nbytes) { /* need to grow */
358 q = alloc(nbytes);
359 memcpy(q, pp, sizeof (p));
360 free(pp);
361 return ((char *)q);
362 } else if (sizeof (p) == nbytes) { /* do nothing */
363 return (pp);
364 } else { /* free excess */
365 q = alloc(nbytes);
366 memcpy(q, pp, nbytes);
367 free(pp);
368 return ((char *)q);
370 #endif