home *** CD-ROM | disk | FTP | other *** search
- /* @(#)malloc.c 4.1 (Berkeley) 12/21/80 */
- /*
- * rtc V1 Modified : Vincent Hayward,
- * School of Electrical Engineering, Purdue University
- * Mon Apr 11 11:01:47 EST 1983
- */
-
- /*LINTLIBRARY*/
-
-
-
- /* avoid break bug */
- #ifdef pdp11
- #define GRANULE 64
- #else
- #define GRANULE 0
- #endif
- /* C storage allocator
- * circular first-fit strategy
- * works with noncontiguous, but monotonically linked, arena
- * each block is preceded by a ptr to the (pointer of)
- * the next following block
- * blocks are exact number of words long
- * aligned to the data type requirements of ALIGN
- * pointers to blocks must have BUSY bit 0
- * bit in ptr is 1 for busy, 0 for idle
- * gaps in arena are merely noted as busy blocks
- * last block of arena (pointed to by alloct) is empty and
- * has a pointer to first
- * idle blocks are coalesced during space search
- *
- * a different implementation may need to redefine
- * ALIGN, NALIGN, BLOCK, BUSY, INT
- * where INT is integer type to which a pointer can be cast
- */
- #define INT int
- #define ALIGN int
- #define NALIGN 1
- #define WORD sizeof(union store)
- #define BLOCK 1024 /* a multiple of WORD*/
- #define BUSY 1
- #define NULL 0
- #define testbusy(p) ((INT)(p)&BUSY)
- #define setbusy(p) (union store *)((INT)(p)|BUSY)
- #define clearbusy(p) (union store *)((INT)(p)&~BUSY)
-
- union store { union store *ptr;
- ALIGN dummy[NALIGN];
- int calloc; /*calloc clears an array of integers*/
- };
-
- static union store allocs[2]; /*initial arena*/
- static union store *allocp; /*search ptr*/
- static union store *alloct; /*arena top*/
- static union store *allocx; /*for benefit of realloc*/
- char *sbrk_l();
-
- char *
- malloc_l(nbytes)
- unsigned nbytes;
- {
- register union store *p, *q;
- register nw;
- static temp; /*coroutines assume no auto*/
-
- if(allocs[0].ptr==0) { /*first time*/
- allocs[0].ptr = setbusy(&allocs[1]);
- allocs[1].ptr = setbusy(&allocs[0]);
- alloct = &allocs[1];
- allocp = &allocs[0];
- }
- nw = (nbytes+WORD+WORD-1)/WORD;
- for(p=allocp; ; ) {
- for(temp=0; ; ) {
- if(!testbusy(p->ptr)) {
- while(!testbusy((q=p->ptr)->ptr)) {
- p->ptr = q->ptr;
- }
- if(q>=p+nw && p+nw>=p)
- goto found;
- }
- q = p;
- p = clearbusy(p->ptr);
- if(p>q)
- ;
- else if(q!=alloct || p!=allocs) {
- return(NULL);
- } else if(++temp>1)
- break;
- }
- temp = ((nw+BLOCK/WORD)/(BLOCK/WORD))*(BLOCK/WORD);
- q = (union store *)sbrk_l(0);
- if(q+temp+GRANULE < q) {
- return(NULL);
- }
- q = (union store *)sbrk_l((unsigned int)temp*WORD);
- if((INT)q == -1) {
- return(NULL);
- }
- alloct->ptr = q;
- if(q!=alloct+1)
- alloct->ptr = setbusy(alloct->ptr);
- alloct = q->ptr = q+temp-1;
- alloct->ptr = setbusy(allocs);
- }
- found:
- allocp = p + nw;
- if(q>allocp) {
- allocx = allocp->ptr;
- allocp->ptr = p->ptr;
- }
- p->ptr = setbusy(allocp);
- return((char *)(p+1));
- }
-
- /* freeing strategy tuned for LIFO allocation
- */
- free_l(ap)
- register char *ap;
- {
- register union store *p = (union store *)ap;
-
- allocp = --p;
- p->ptr = clearbusy(p->ptr);
- }
-
- /* realloc(p, nbytes) reallocates a block obtained from malloc()
- * and freed since last call of malloc()
- * to have new size nbytes, and old content
- * returns new location, or 0 on failure
- */
-
- char *
- realloc_l(p, nbytes)
- register union store *p;
- unsigned nbytes;
- {
- register union store *q;
- union store *s, *t;
- register unsigned nw;
- unsigned onw;
-
- if(testbusy(p[-1].ptr))
- free_l((char *)p);
- onw = p[-1].ptr - p;
- q = (union store *)malloc_l(nbytes);
- if(q==NULL || q==p)
- return((char *)q);
- s = p;
- t = q;
- nw = (nbytes+WORD-1)/WORD;
- if(nw<onw)
- onw = nw;
- while(onw--!=0)
- *t++ = *s++;
- if(q<p && q+nw>=p)
- (q+(q+nw-p))->ptr = allocx;
- return((char *)q);
- }
-
- /* @(#)calloc.c 4.1 (Berkeley) 12/21/80 */
- /* calloc - allocate and clear memory block
- */
- #define CHARPERINT (sizeof(int)/sizeof(char))
- #define NULL 0
-
- char *
- calloc_l(num, size)
- unsigned num, size;
- {
- register char *mp;
- char *malloc();
- register int *q;
- register m;
-
- num *= size;
- mp = malloc_l(num);
- if(mp == NULL)
- return(NULL);
- q = (int *) mp;
- m = (num+CHARPERINT-1)/CHARPERINT;
- while(--m>=0)
- *q++ = 0;
- return(mp);
- }
-
- /*ARGSUSED*/
-
- cfree_l(p, num, size)
- char *p;
- unsigned num, size;
- {
- free_l(p);
- }
-
-
-
- #define NPOOL BLOCK*64
-
-
- static char *
- sbrk_l(inc)
- register unsigned int inc;
- {
- static char pool[NPOOL];
- static char *pt = pool+NPOOL;
- static char *pp = pool;
- char *p;
-
- inc += inc % BLOCK;
- if (pp+inc>=pt) {
- return((char *)-1);
- }
- p = pp;
- pp += inc;
- return(p);
- }
-