Annotation of GNUtools/cc/objc/sarray.c, revision 1.1

1.1     ! root        1: /* Sparse Arrays for Objective C dispatch tables
        !             2:    Copyright (C) 1993 Free Software Foundation, Inc.
        !             3: 
        !             4: This file is part of GNU CC.
        !             5: 
        !             6: GNU CC is free software; you can redistribute it and/or modify
        !             7: it under the terms of the GNU General Public License as published by
        !             8: the Free Software Foundation; either version 2, or (at your option)
        !             9: any later version.
        !            10: 
        !            11: GNU CC is distributed in the hope that it will be useful,
        !            12: but WITHOUT ANY WARRANTY; without even the implied warranty of
        !            13: MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
        !            14: GNU General Public License for more details.
        !            15: 
        !            16: You should have received a copy of the GNU General Public License
        !            17: along with GNU CC; see the file COPYING.  If not, write to
        !            18: the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.  */
        !            19: 
        !            20: /* As a special exception, if you link this library with files
        !            21:    compiled with GCC to produce an executable, this does not cause
        !            22:    the resulting executable to be covered by the GNU General Public License.
        !            23:    This exception does not however invalidate any other reasons why
        !            24:    the executable file might be covered by the GNU General Public License.  */
        !            25: 
        !            26: #include "objc/sarray.h"
        !            27: #include <stdio.h>
        !            28: #include "assert.h"
        !            29: 
        !            30: int nbuckets = 0;
        !            31: int nindices = 0;
        !            32: int narrays = 0;
        !            33: int idxsize = 0;
        !            34: 
        !            35: #ifdef OBJC_SPARSE2
        !            36: const char* __objc_sparse2_id = "2 level sparse indices";
        !            37: #endif
        !            38: 
        !            39: #ifdef OBJC_SPARSE3
        !            40: const char* __objc_sparse3_id = "3 level sparse indices";
        !            41: #endif
        !            42: 
        !            43: #ifdef __alpha__
        !            44: const void *memcpy (void*, const void*, size_t);
        !            45: void free (const void*);
        !            46: #endif
        !            47: 
        !            48: void
        !            49: sarray_at_put(struct sarray* array, sidx index, void* element)
        !            50: {
        !            51: #ifdef OBJC_SPARSE3
        !            52:   struct sindex** the_index;
        !            53: #endif
        !            54:   struct sbucket** the_bucket;
        !            55: #ifdef OBJC_SPARSE3
        !            56:   size_t ioffset;
        !            57: #endif
        !            58:   size_t boffset;
        !            59:   size_t eoffset;
        !            60: #ifdef PRECOMPUTE_SELECTORS
        !            61:   union sofftype xx; 
        !            62:   xx.idx = index;
        !            63: #ifdef OBJC_SPARSE3
        !            64:   ioffset = xx.off.ioffset;
        !            65: #endif
        !            66:   boffset = xx.off.boffset;
        !            67:   eoffset = xx.off.eoffset;
        !            68: #else /* not PRECOMPUTE_SELECTORS */
        !            69: #ifdef OBJC_SPARSE3
        !            70:   ioffset = index/INDEX_CAPACITY;
        !            71:   boffset = (index/BUCKET_SIZE)%INDEX_SIZE;
        !            72:   eoffset = index%BUCKET_SIZE;
        !            73: #else
        !            74:   boffset = index/BUCKET_SIZE;
        !            75:   eoffset = index%BUCKET_SIZE;
        !            76: #endif
        !            77: #endif /* not PRECOMPUTE_SELECTORS */
        !            78: 
        !            79:   assert(soffset_decode(index) < array->capacity); /* Range check */
        !            80: 
        !            81: #ifdef OBJC_SPARSE3
        !            82:   the_index = &(array->indices[ioffset]);
        !            83:   the_bucket = &((*the_index)->buckets[boffset]);
        !            84: #else
        !            85:   the_bucket = &(array->buckets[boffset]);
        !            86: #endif
        !            87:   
        !            88:   if ((*the_bucket)->elems[eoffset] == element)
        !            89:     return;            /* great! we just avoided a lazy copy */
        !            90: 
        !            91: #ifdef OBJC_SPARSE3
        !            92: 
        !            93:   /* First, perform lazy copy/allocation of index if needed */
        !            94: 
        !            95:   if ((*the_index) == array->empty_index) {
        !            96: 
        !            97:     /* The index was previously empty, allocate a new */
        !            98:     *the_index = (struct sindex*)__objc_xmalloc(sizeof(struct sindex));
        !            99:     memcpy(*the_index, array->empty_index, sizeof(struct sindex));
        !           100:     (*the_index)->version = array->version;
        !           101:     the_bucket = &((*the_index)->buckets[boffset]);
        !           102:     nindices += 1;
        !           103:     
        !           104:   } else if ((*the_index)->version != array->version) {
        !           105: 
        !           106:     /* This index must be lazy copied */
        !           107:     struct sindex* old_index = *the_index;
        !           108:     *the_index = (struct sindex*)__objc_xmalloc(sizeof(struct sindex));
        !           109:     memcpy( *the_index,old_index, sizeof(struct sindex));
        !           110:     (*the_index)->version = array->version;
        !           111:     the_bucket = &((*the_index)->buckets[boffset]);
        !           112:     nindices += 1;
        !           113:     
        !           114:   }
        !           115: 
        !           116: #endif /* OBJC_SPARSE3 */
        !           117: 
        !           118:   /* next, perform lazy allocation/copy of the bucket if needed */
        !           119: 
        !           120:   if ((*the_bucket) == array->empty_bucket) {
        !           121: 
        !           122:     /* The bucket was previously empty (or something like that), */
        !           123:     /* allocate a new.  This is the effect of `lazy' allocation */  
        !           124:     *the_bucket = (struct sbucket*)__objc_xmalloc(sizeof(struct sbucket));
        !           125:     memcpy((void *) *the_bucket, (const void*)array->empty_bucket, sizeof(struct sbucket));
        !           126:     (*the_bucket)->version = array->version;
        !           127:     nbuckets += 1;
        !           128: 
        !           129:   } else if ((*the_bucket)->version != array->version) {
        !           130: 
        !           131:     /* Perform lazy copy. */
        !           132:     struct sbucket* old_bucket = *the_bucket;
        !           133:     *the_bucket = (struct sbucket*)__objc_xmalloc(sizeof(struct sbucket));
        !           134:     memcpy( *the_bucket,old_bucket, sizeof(struct sbucket));
        !           135:     (*the_bucket)->version = array->version;
        !           136:     nbuckets += 1;
        !           137: 
        !           138:   }
        !           139:   (*the_bucket)->elems[eoffset] = element;
        !           140: }
        !           141: 
        !           142: void
        !           143: sarray_at_put_safe(struct sarray* array, sidx index, void* element)
        !           144: {
        !           145:   if(soffset_decode(index) >= array->capacity)
        !           146:     sarray_realloc(array, soffset_decode(index)+1);
        !           147:   sarray_at_put(array, index, element);
        !           148: }
        !           149: 
        !           150: struct sarray* 
        !           151: sarray_new (int size, void* default_element)
        !           152: {
        !           153: #ifdef OBJC_SPARSE3
        !           154:   size_t num_indices = ((size-1)/(INDEX_CAPACITY))+1;
        !           155: #else /* OBJC_SPARSE2 */
        !           156:   size_t num_indices = ((size-1)/BUCKET_SIZE)+1;
        !           157: #endif
        !           158:   int counter;
        !           159:   struct sarray* arr;
        !           160: 
        !           161:   assert(size > 0);
        !           162: 
        !           163:   /* Allocate core array */
        !           164:   arr = (struct sarray*) __objc_xmalloc(sizeof(struct sarray));
        !           165:   arr->version = 0;
        !           166:   narrays  += 1;
        !           167:   
        !           168:   /* Initialize members */
        !           169: #ifdef OBJC_SPARSE3
        !           170:   arr->capacity = num_indices*INDEX_CAPACITY;
        !           171:   arr->indices = (struct sindex**) 
        !           172:     __objc_xmalloc(sizeof(struct sindex*)*num_indices);
        !           173:   idxsize  += num_indices;
        !           174: 
        !           175:   arr->empty_index = (struct sindex*) __objc_xmalloc(sizeof(struct sindex));
        !           176:   arr->empty_index->version = 0;
        !           177:   nindices += 1;
        !           178: 
        !           179: #else /* OBJC_SPARSE2 */
        !           180:   arr->capacity = num_indices*BUCKET_SIZE;
        !           181:   arr->buckets = (struct sbucket**) 
        !           182:     __objc_xmalloc(sizeof(struct sbucket*)*num_indices);
        !           183:   idxsize  += num_indices;
        !           184: 
        !           185: #endif
        !           186: 
        !           187:   arr->empty_bucket = (struct sbucket*) __objc_xmalloc(sizeof(struct sbucket));
        !           188:   arr->empty_bucket->version = 0;
        !           189:   nbuckets += 1;
        !           190: 
        !           191:   arr->ref_count = 1;
        !           192:   arr->is_copy_of = (struct sarray*)0;
        !           193:   
        !           194:   for (counter=0; counter<BUCKET_SIZE; counter++)
        !           195:     arr->empty_bucket->elems[counter] = default_element;
        !           196: 
        !           197: #ifdef OBJC_SPARSE3
        !           198:   for (counter=0; counter<INDEX_SIZE; counter++)
        !           199:     arr->empty_index->buckets[counter] = arr->empty_bucket;
        !           200: 
        !           201:   for (counter=0; counter<num_indices; counter++)
        !           202:     arr->indices[counter] = arr->empty_index;
        !           203: 
        !           204: #else /* OBJC_SPARSE2 */
        !           205: 
        !           206:   for (counter=0; counter<num_indices; counter++)
        !           207:     arr->buckets[counter] = arr->empty_bucket;
        !           208: 
        !           209: #endif
        !           210: 
        !           211:   return arr;
        !           212: }
        !           213: 
        !           214: 
        !           215: /* Reallocate the sparse array to hold `newsize' entries */
        !           216: 
        !           217: void 
        !           218: sarray_realloc(struct sarray* array, int newsize)
        !           219: {
        !           220: #ifdef OBJC_SPARSE3
        !           221:   int old_max_index = (array->capacity-1)/INDEX_CAPACITY;
        !           222:   int new_max_index = ((newsize-1)/INDEX_CAPACITY);
        !           223:   int rounded_size = (new_max_index+1)*INDEX_CAPACITY;
        !           224: 
        !           225: #else /* OBJC_SPARSE2 */
        !           226:   int old_max_index = (array->capacity-1)/BUCKET_SIZE;
        !           227:   int new_max_index = ((newsize-1)/BUCKET_SIZE);
        !           228:   int rounded_size = (new_max_index+1)*BUCKET_SIZE;
        !           229: 
        !           230: #endif
        !           231: 
        !           232:   int counter;
        !           233: 
        !           234:   assert(newsize > 0);
        !           235: 
        !           236:   /* The size is the same, just ignore the request */
        !           237:   if(rounded_size == array->capacity)
        !           238:     return;
        !           239: 
        !           240:   assert(array->ref_count == 1);       /* stop if lazy copied... */
        !           241: 
        !           242:   if(rounded_size < array->capacity) 
        !           243:     {
        !           244:       /* update capacity */
        !           245:       array->capacity = rounded_size;
        !           246: 
        !           247:       /* free buckets above new_max_index */
        !           248:       for(counter = old_max_index; counter > new_max_index; counter-- ) {
        !           249: #ifdef OBJC_SPARSE3
        !           250:        struct sindex* idx = array->indices[counter];
        !           251:        if((idx != array->empty_index) && (idx->version == array->version)) {
        !           252:          int c2; 
        !           253:          for(c2=0; c2<INDEX_SIZE; c2++) {
        !           254:            struct sbucket* bkt = idx->buckets[c2];
        !           255:            if((bkt != array->empty_bucket) && (bkt->version == array->version))
        !           256:              {
        !           257:                free(bkt);
        !           258:                nbuckets -= 1;
        !           259:              }
        !           260:          }
        !           261:          free(idx);
        !           262:          nindices -= 1;
        !           263:        }
        !           264: #else /* OBJC_SPARSE2 */
        !           265:        struct sbucket* bkt = array->buckets[counter];
        !           266:        if ((bkt != array->empty_bucket) && (bkt->version == array->version))
        !           267:          {
        !           268:            free(bkt);
        !           269:            nbuckets -= 1;
        !           270:          }
        !           271: #endif
        !           272:       }
        !           273:          
        !           274: #ifdef OBJC_SPARSE3
        !           275:       /* realloc to free the space above new_max_index */
        !           276:       array->indices = (struct sindex**)
        !           277:        __objc_xrealloc(array->indices, 
        !           278:                        (new_max_index+1)*sizeof(struct sindex*));
        !           279: #else /* OBJC_SPARSE2 */
        !           280:       array->buckets = (struct sbucket**)
        !           281:        __objc_xrealloc(array->buckets, 
        !           282:                        (new_max_index+1)*sizeof(struct sbucket*));
        !           283: #endif      
        !           284:       idxsize -= (old_max_index-new_max_index);
        !           285: 
        !           286:       return;
        !           287:     }
        !           288: 
        !           289:   /* We are asked to extend the array -- reallocate the bucket table, */
        !           290:   /* and insert empty_bucket in newly allocated places. */
        !           291:   if(rounded_size > array->capacity) 
        !           292:     {
        !           293:       /* update capacity */
        !           294:       array->capacity = rounded_size;
        !           295: 
        !           296: #ifdef OBJC_SPARSE3
        !           297:       /* realloc to make room in table above old_max_index */
        !           298:       array->indices = (struct sindex**)
        !           299:        __objc_xrealloc(array->indices, 
        !           300:                        (new_max_index+1)*sizeof(struct sindex*));
        !           301: 
        !           302:       /* reset entries above old_max_index to empty_bucket */
        !           303:       for(counter = old_max_index+1; counter <= new_max_index; counter++)
        !           304:        array->indices[counter] = array->empty_index;
        !           305: 
        !           306: #else /* OBJC_SPARSE2 */
        !           307: 
        !           308:       /* realloc to make room in table above old_max_index */
        !           309:       array->buckets = (struct sbucket**)
        !           310:        __objc_xrealloc(array->buckets, 
        !           311:                        (new_max_index+1)*sizeof(struct sbucket*));
        !           312: 
        !           313:       /* reset entries above old_max_index to empty_bucket */
        !           314:       for(counter = old_max_index+1; counter <= new_max_index; counter++)
        !           315:        array->buckets[counter] = array->empty_bucket;
        !           316: 
        !           317: #endif
        !           318:       idxsize += (new_max_index-old_max_index);
        !           319:       return;
        !           320:     }
        !           321: }
        !           322: 
        !           323: 
        !           324: /* Free a sparse array allocated with sarray_new */
        !           325: 
        !           326: void 
        !           327: sarray_free(struct sarray* array) {
        !           328: #ifdef OBJC_SPARSE3
        !           329:   size_t old_max_index = (array->capacity-1)/INDEX_CAPACITY;
        !           330: #else
        !           331:   size_t old_max_index = (array->capacity-1)/BUCKET_SIZE;
        !           332: #endif
        !           333:   int counter = 0;
        !           334: 
        !           335:   assert(array->ref_count != 0);       /* Freed multiple times!!! */
        !           336: 
        !           337:   if(--(array->ref_count) != 0)        /* There exists copies of me */
        !           338:     return;
        !           339: 
        !           340:   if((array->is_copy_of) && ((array->is_copy_of->ref_count - 1) == 0))
        !           341:     sarray_free(array->is_copy_of);
        !           342: 
        !           343:   /* Free all entries that do not point to empty_bucket */
        !           344:   for(counter = 0; counter <= old_max_index; counter++ ) {
        !           345: #ifdef OBJC_SPARSE3
        !           346:     struct sindex* idx = array->indices[counter];
        !           347:     if((idx != array->empty_index) && (idx->version == array->version)) {
        !           348:       int c2; 
        !           349:       for(c2=0; c2<INDEX_SIZE; c2++) {
        !           350:        struct sbucket* bkt = idx->buckets[c2];
        !           351:        if((bkt != array->empty_bucket) && (bkt->version == array->version))
        !           352:          {
        !           353:            free(bkt);
        !           354:            nbuckets -= 1;
        !           355:          }
        !           356:       }
        !           357:       free(idx);
        !           358:       nindices -= 1;
        !           359:     }
        !           360: #else /* OBJC_SPARSE2 */
        !           361:     struct sbucket* bkt = array->buckets[counter];
        !           362:     if ((bkt != array->empty_bucket) && (bkt->version == array->version))
        !           363:       {
        !           364:        free(bkt);
        !           365:        nbuckets -= 1;
        !           366:       }
        !           367: #endif
        !           368:   }
        !           369:        
        !           370: #ifdef OBJC_SPARSE3  
        !           371:   /* free empty_index */
        !           372:   if(array->empty_index->version == array->version) {
        !           373:     free(array->empty_index);
        !           374:     nindices -= 1;
        !           375:   }
        !           376: #endif
        !           377: 
        !           378:   /* free empty_bucket */
        !           379:   if(array->empty_bucket->version == array->version) {
        !           380:     free(array->empty_bucket);
        !           381:     nbuckets -= 1;
        !           382:   }
        !           383: 
        !           384: #ifdef OBJC_SPARSE3
        !           385:   /* free bucket table */
        !           386:   free(array->indices);
        !           387:   idxsize -= (old_max_index+1);
        !           388: 
        !           389: #else
        !           390:   /* free bucket table */
        !           391:   free(array->buckets);
        !           392:   idxsize -= (old_max_index+1);
        !           393: 
        !           394: #endif
        !           395: 
        !           396:   /* free array */
        !           397:   free(array);
        !           398:   narrays -= 1;
        !           399: }
        !           400: 
        !           401: /* This is a lazy copy.  Only the core of the structure is actually */
        !           402: /* copied.   */
        !           403: 
        !           404: struct sarray* 
        !           405: sarray_lazy_copy(struct sarray* oarr)
        !           406: {
        !           407: #ifdef OBJC_SPARSE3
        !           408:   size_t num_indices = ((oarr->capacity-1)/INDEX_CAPACITY)+1;
        !           409: #else /* OBJC_SPARSE2 */
        !           410:   size_t num_indices = ((oarr->capacity-1)/BUCKET_SIZE)+1;
        !           411: #endif
        !           412:   struct sarray* arr;
        !           413: 
        !           414:   /* Allocate core array */
        !           415:   arr = (struct sarray*) __objc_xmalloc(sizeof(struct sarray));
        !           416:   memcpy( arr,oarr, sizeof(struct sarray));
        !           417:   arr->version = oarr->version + 1;
        !           418:   arr->is_copy_of = oarr;
        !           419:   oarr->ref_count += 1;
        !           420:   arr->ref_count = 1;
        !           421:   
        !           422: #ifdef OBJC_SPARSE3
        !           423:   /* Copy bucket table */
        !           424:   arr->indices = (struct sindex**) 
        !           425:     __objc_xmalloc(sizeof(struct sindex*)*num_indices);
        !           426:   memcpy( arr->indices,oarr->indices, 
        !           427:        sizeof(struct sindex*)*num_indices);
        !           428: #else 
        !           429:   /* Copy bucket table */
        !           430:   arr->buckets = (struct sbucket**) 
        !           431:     __objc_xmalloc(sizeof(struct sbucket*)*num_indices);
        !           432:   memcpy( arr->buckets,oarr->buckets, 
        !           433:        sizeof(struct sbucket*)*num_indices);
        !           434: #endif
        !           435: 
        !           436:   idxsize += num_indices;
        !           437:   narrays += 1;
        !           438: 
        !           439:   return arr;
        !           440: }

unix.superglobalmegacorp.com

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