Annotation of researchv9/jerq/src/lib/j/drek/gcalloc.c, revision 1.1.1.1

1.1       root        1: #include <setup.h>
                      2: 
                      3: #define        oops(x, s)      /*if(x)sstep(s)*/
                      4: long NLONGS, NAVAIL;
                      5: /*
                      6:  * Garbage-collecting allocator
                      7:  *
                      8:  * called as:  gcalloc(nbytes, where)
                      9:  *     nbytes is unsigned long
                     10:  *     where is a pointer to the location which will point to the area
                     11:  *     (e.g. base in a Bitmap), to be updated when necessary
                     12:  */
                     13: 
                     14: /*
                     15:  * The arena is allocated in longs, to speed up compaction
                     16:  * Garbage is compacted towards the top (high address end) of the arena,
                     17:  * i.e. away from the regular, non-compacting, allocator.
                     18:  * A block has a struct header as its top HEADERSIZE longs, and
                     19:  * a pointer to the struct header as its bottom long, for use by gcfree.
                     20:  * header.caller points to the allocating process so things can be cleaned
                     21:  * up when it exits.
                     22:  * header.pval points to the USER location where the pointer to the block
                     23:  * is stored; this cell must be updated after a compaction.
                     24:  * Deallocated blocks have header.pval odd.
                     25:  */
                     26: extern char *allocendp;                /* real allocendp defined in alloc.c */
                     27: #define        STARTAREA       (long *)(allocendp)
                     28: long *ENDAREA;
                     29: /*static*/ long *nextlong;
                     30: struct header{
                     31:        union uuu{
                     32:                long    Uival;          /* integer value of where */
                     33:                long    **Upval;        /* where */
                     34:        }u;
                     35:        long    *caller;                /* which process allocated */
                     36:        unsigned long nlongs;           /* # of longs in the user's data */
                     37: };
                     38: #define        ival    u.Uival
                     39: #define        pval    u.Upval
                     40: #define        HEADERSIZE      sizeof(struct header)/sizeof(long)
                     41: #define        hp      ((struct header *)p)
                     42: 
                     43: gcinit()
                     44: {
                     45:        ENDAREA= maxaddr[VALMAXADDR]-4;
                     46:        nextlong = ENDAREA;     /* init; this runs in rom, so .data no good */
                     47: }
                     48: 
                     49: char *
                     50: realgcalloc(nbytes, where, caller)
                     51:        register unsigned long nbytes;
                     52:        long **where;
                     53:        char *caller;   /* really a Proc * */
                     54: {
                     55:        register retry = 0;
                     56:        register long *p;
                     57:        
                     58:        if((long)caller&1)      /* head off a possible disaster */
                     59:                return 0;
                     60:        nbytes+=sizeof(long)-1;
                     61:        nbytes>>=2;     /* convert bytes to longs */
                     62: #define        Nlongs  nbytes
                     63:        Nlongs+=HEADERSIZE+1;
                     64:        while(nextlong-STARTAREA < Nlongs){
                     65: NLONGS=Nlongs;
                     66: NAVAIL=nextlong-STARTAREA;
                     67:                if(retry++)
                     68:                        return 0;
                     69:                compact();
                     70:        }
                     71:        p=nextlong-HEADERSIZE;
                     72:        hp->caller=(long *)caller;
                     73:        hp->pval=where;
                     74:        hp->nlongs=Nlongs-HEADERSIZE-1;
                     75:        nextlong-=Nlongs;
                     76:        *nextlong=(long)p;      /* so we can find the header when we free */
                     77:        return (char *)(*(hp->pval)=nextlong+1);
                     78: }
                     79: char *
                     80: gcalloc(n, w)
                     81:        unsigned long n;
                     82:        long **w;
                     83: {
                     84:        return realgcalloc(n, w, (char *)0);
                     85: }
                     86: gcfree(cp)
                     87:        register char *cp;
                     88: {
                     89: #define        p       ((long *)cp)
                     90:        /* nice and safe */
                     91:        oops((p<=nextlong || p>=ENDAREA),"gcfree 1");
                     92:        if(p>nextlong && p<ENDAREA){
                     93:                cp=(char *)(p[-1]);     /* pointer to the header */
                     94:                oops(p<=nextlong || p>=ENDAREA, "gcfree 2");
                     95:                if(p>nextlong && p<ENDAREA)
                     96:                        hp->ival|=1;
                     97:        }
                     98: #undef p
                     99: }
                    100: gcfreeall(whichproc)
                    101:        char *whichproc;
                    102: {
                    103:        register long *p;
                    104:        for(p=ENDAREA; p>nextlong; p-=hp->nlongs+1){
                    105:                p-=HEADERSIZE;
                    106:                oops(hp->caller > nextlong, "gcfreeall 1");
                    107:                oops(hp->pval > nextlong, "gcfreeall 2");
                    108:                if(hp->caller==(long *)whichproc)
                    109:                        hp->ival|=1;
                    110:        }
                    111: }
                    112: #include <jerq.h>
                    113: static
                    114: compact()
                    115: {
                    116:        register long *w, *p;
                    117:        register struct header *header;
                    118:        register unsigned long n;
                    119:        static Rectangle r={0, 0, 50, 50};
                    120: 
                    121:        rectf(&display, r, F_XOR);
                    122:        w=ENDAREA;
                    123:        p=w;
                    124:        while(p>nextlong){
                    125:                header=(struct header *)(p-HEADERSIZE);
                    126: /*
                    127:                if(w==p || (header->ival&1)){
                    128:                        p-= n = header->nlongs+HEADERSIZE+1;
                    129:                        if((header->ival&1)==0)
                    130:                                w-=n;
                    131:                        continue;
                    132:                }
                    133: */
                    134:                oops(header->caller > nextlong, "compact 1");
                    135:                oops(header->pval > nextlong, "compact 2");
                    136:                if(header->ival&1){
                    137:                        p -= header->nlongs+HEADERSIZE+1;
                    138:                        continue;
                    139:                }
                    140:                *--w = n = header->nlongs;
                    141:                *--w = (long)header->caller;
                    142:                *--w = header->ival;
                    143:                p-=HEADERSIZE;
                    144:                header = (struct header *)w;
                    145:                *(((struct header *)w)->pval)=w-n; /* update *where */
                    146:                if(n>0) do
                    147:                        *--w = *--p;
                    148:                while(--n);
                    149:                --p;
                    150:                *--w = (long)header;    /* back pointer to header */
                    151:        }
                    152:        nextlong=w;
                    153:        rectf(&display, r, F_XOR);
                    154: }

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.