[BACK]Return to stubborn.c CVS log [TXT][DIR] Up to [local] / OpenXM_contrib / gc

Annotation of OpenXM_contrib/gc/stubborn.c, Revision 1.1

1.1     ! maekawa     1: /*
        !             2:  * Copyright 1988, 1989 Hans-J. Boehm, Alan J. Demers
        !             3:  * Copyright (c) 1991-1994 by Xerox Corporation.  All rights reserved.
        !             4:  *
        !             5:  * THIS MATERIAL IS PROVIDED AS IS, WITH ABSOLUTELY NO WARRANTY EXPRESSED
        !             6:  * OR IMPLIED.  ANY USE IS AT YOUR OWN RISK.
        !             7:  *
        !             8:  * Permission is hereby granted to use or copy this program
        !             9:  * for any purpose,  provided the above notices are retained on all copies.
        !            10:  * Permission to modify the code and to distribute modified code is granted,
        !            11:  * provided the above notices are retained, and a notice that the code was
        !            12:  * modified is included with the above copyright notice.
        !            13:  */
        !            14: /* Boehm, July 31, 1995 5:02 pm PDT */
        !            15:
        !            16:
        !            17: #include "gc_priv.h"
        !            18:
        !            19: # ifdef STUBBORN_ALLOC
        !            20: /* Stubborn object (hard to change, nearly immutable) allocation. */
        !            21:
        !            22: extern ptr_t GC_clear_stack(); /* in misc.c, behaves like identity */
        !            23:
        !            24: #define GENERAL_MALLOC(lb,k) \
        !            25:     (GC_PTR)GC_clear_stack(GC_generic_malloc((word)lb, k))
        !            26:
        !            27: /* Data structure representing immutable objects that  */
        !            28: /* are still being initialized.                                */
        !            29: /* This is a bit baroque in order to avoid acquiring   */
        !            30: /* the lock twice for a typical allocation.            */
        !            31:
        !            32: GC_PTR * GC_changing_list_start;
        !            33:
        !            34: # ifdef THREADS
        !            35:   VOLATILE GC_PTR * VOLATILE GC_changing_list_current;
        !            36: # else
        !            37:   GC_PTR * GC_changing_list_current;
        !            38: # endif
        !            39:        /* Points at last added element.  Also (ab)used for             */
        !            40:        /* synchronization.  Updates and reads are assumed atomic.      */
        !            41:
        !            42: GC_PTR * GC_changing_list_limit;
        !            43:        /* Points at the last word of the buffer, which is always 0     */
        !            44:        /* All entries in (GC_changing_list_current,                    */
        !            45:        /* GC_changing_list_limit] are 0                                */
        !            46:
        !            47:
        !            48: void GC_stubborn_init()
        !            49: {
        !            50: #   define INIT_SIZE 10
        !            51:
        !            52:     GC_changing_list_start = (GC_PTR *)
        !            53:                        GC_generic_malloc_inner(
        !            54:                                (word)(INIT_SIZE * sizeof(GC_PTR)),
        !            55:                                PTRFREE);
        !            56:     BZERO(GC_changing_list_start,
        !            57:          INIT_SIZE * sizeof(GC_PTR));
        !            58:     if (GC_changing_list_start == 0) {
        !            59:         GC_err_printf0("Insufficient space to start up\n");
        !            60:         ABORT("GC_stubborn_init: put of space");
        !            61:     }
        !            62:     GC_changing_list_current = GC_changing_list_start;
        !            63:     GC_changing_list_limit = GC_changing_list_start + INIT_SIZE - 1;
        !            64:     * GC_changing_list_limit = 0;
        !            65: }
        !            66:
        !            67: /* Compact and possibly grow GC_uninit_list.  The old copy is          */
        !            68: /* left alone. Lock must be held.                                      */
        !            69: /* When called GC_changing_list_current == GC_changing_list_limit      */
        !            70: /* which is one past the current element.                              */
        !            71: /* When we finish GC_changing_list_current again points one past last  */
        !            72: /* element.                                                            */
        !            73: /* Invariant while this is running: GC_changing_list_current           */
        !            74: /* points at a word containing 0.                                      */
        !            75: /* Returns FALSE on failure.                                           */
        !            76: GC_bool GC_compact_changing_list()
        !            77: {
        !            78:     register GC_PTR *p, *q;
        !            79:     register word count = 0;
        !            80:     word old_size = (char **)GC_changing_list_limit
        !            81:                    - (char **)GC_changing_list_start+1;
        !            82:                    /* The casts are needed as a workaround for an Amiga bug */
        !            83:     register word new_size = old_size;
        !            84:     GC_PTR * new_list;
        !            85:
        !            86:     for (p = GC_changing_list_start; p < GC_changing_list_limit; p++) {
        !            87:         if (*p != 0) count++;
        !            88:     }
        !            89:     if (2 * count > old_size) new_size = 2 * count;
        !            90:     new_list = (GC_PTR *)
        !            91:                GC_generic_malloc_inner(
        !            92:                                new_size * sizeof(GC_PTR), PTRFREE);
        !            93:                /* PTRFREE is a lie.  But we don't want the collector to  */
        !            94:                /* consider these.  We do want the list itself to be      */
        !            95:                /* collectable.                                           */
        !            96:     if (new_list == 0) return(FALSE);
        !            97:     BZERO(new_list, new_size * sizeof(GC_PTR));
        !            98:     q = new_list;
        !            99:     for (p = GC_changing_list_start; p < GC_changing_list_limit; p++) {
        !           100:         if (*p != 0) *q++ = *p;
        !           101:     }
        !           102:     GC_changing_list_start = new_list;
        !           103:     GC_changing_list_limit = new_list + new_size - 1;
        !           104:     GC_changing_list_current = q;
        !           105:     return(TRUE);
        !           106: }
        !           107:
        !           108: /* Add p to changing list.  Clear p on failure.        */
        !           109: # define ADD_CHANGING(p) \
        !           110:        {       \
        !           111:            register struct hblk * h = HBLKPTR(p);      \
        !           112:            register word index = PHT_HASH(h);  \
        !           113:            \
        !           114:            set_pht_entry_from_index(GC_changed_pages, index);  \
        !           115:        }       \
        !           116:        if (*GC_changing_list_current != 0 \
        !           117:            && ++GC_changing_list_current == GC_changing_list_limit) { \
        !           118:            if (!GC_compact_changing_list()) (p) = 0; \
        !           119:        } \
        !           120:        *GC_changing_list_current = p;
        !           121:
        !           122: void GC_change_stubborn(p)
        !           123: GC_PTR p;
        !           124: {
        !           125:     DCL_LOCK_STATE;
        !           126:
        !           127:     DISABLE_SIGNALS();
        !           128:     LOCK();
        !           129:     ADD_CHANGING(p);
        !           130:     UNLOCK();
        !           131:     ENABLE_SIGNALS();
        !           132: }
        !           133:
        !           134: void GC_end_stubborn_change(p)
        !           135: GC_PTR p;
        !           136: {
        !           137: #   ifdef THREADS
        !           138:       register VOLATILE GC_PTR * my_current = GC_changing_list_current;
        !           139: #   else
        !           140:       register GC_PTR * my_current = GC_changing_list_current;
        !           141: #   endif
        !           142:     register GC_bool tried_quick;
        !           143:     DCL_LOCK_STATE;
        !           144:
        !           145:     if (*my_current == p) {
        !           146:         /* Hopefully the normal case.                                  */
        !           147:         /* Compaction could not have been running when we started.     */
        !           148:         *my_current = 0;
        !           149: #      ifdef THREADS
        !           150:           if (my_current == GC_changing_list_current) {
        !           151:             /* Compaction can't have run in the interim.       */
        !           152:             /* We got away with the quick and dirty approach.   */
        !           153:             return;
        !           154:           }
        !           155:           tried_quick = TRUE;
        !           156: #      else
        !           157:          return;
        !           158: #      endif
        !           159:     } else {
        !           160:         tried_quick = FALSE;
        !           161:     }
        !           162:     DISABLE_SIGNALS();
        !           163:     LOCK();
        !           164:     my_current = GC_changing_list_current;
        !           165:     for (; my_current >= GC_changing_list_start; my_current--) {
        !           166:         if (*my_current == p) {
        !           167:             *my_current = 0;
        !           168:             UNLOCK();
        !           169:             ENABLE_SIGNALS();
        !           170:             return;
        !           171:         }
        !           172:     }
        !           173:     if (!tried_quick) {
        !           174:         GC_err_printf1("Bad arg to GC_end_stubborn_change: 0x%lx\n",
        !           175:                       (unsigned long)p);
        !           176:         ABORT("Bad arg to GC_end_stubborn_change");
        !           177:     }
        !           178:     UNLOCK();
        !           179:     ENABLE_SIGNALS();
        !           180: }
        !           181:
        !           182: /* Allocate lb bytes of composite (pointerful) data    */
        !           183: /* No pointer fields may be changed after a call to    */
        !           184: /* GC_end_stubborn_change(p) where p is the value      */
        !           185: /* returned by GC_malloc_stubborn.                     */
        !           186: # ifdef __STDC__
        !           187:     GC_PTR GC_malloc_stubborn(size_t lb)
        !           188: # else
        !           189:     GC_PTR GC_malloc_stubborn(lb)
        !           190:     size_t lb;
        !           191: # endif
        !           192: {
        !           193: register ptr_t op;
        !           194: register ptr_t *opp;
        !           195: register word lw;
        !           196: ptr_t result;
        !           197: DCL_LOCK_STATE;
        !           198:
        !           199:     if( SMALL_OBJ(lb) ) {
        !           200: #       ifdef MERGE_SIZES
        !           201:          lw = GC_size_map[lb];
        !           202: #      else
        !           203:          lw = ALIGNED_WORDS(lb);
        !           204: #       endif
        !           205:        opp = &(GC_sobjfreelist[lw]);
        !           206:        FASTLOCK();
        !           207:         if( !FASTLOCK_SUCCEEDED() || (op = *opp) == 0 ) {
        !           208:             FASTUNLOCK();
        !           209:             result = GC_generic_malloc((word)lb, STUBBORN);
        !           210:             goto record;
        !           211:         }
        !           212:         *opp = obj_link(op);
        !           213:         obj_link(op) = 0;
        !           214:         GC_words_allocd += lw;
        !           215:         result = (GC_PTR) op;
        !           216:         ADD_CHANGING(result);
        !           217:         FASTUNLOCK();
        !           218:         return((GC_PTR)result);
        !           219:    } else {
        !           220:        result = (GC_PTR)
        !           221:                GC_generic_malloc((word)lb, STUBBORN);
        !           222:    }
        !           223: record:
        !           224:    DISABLE_SIGNALS();
        !           225:    LOCK();
        !           226:    ADD_CHANGING(result);
        !           227:    UNLOCK();
        !           228:    ENABLE_SIGNALS();
        !           229:    return((GC_PTR)GC_clear_stack(result));
        !           230: }
        !           231:
        !           232:
        !           233: /* Functions analogous to GC_read_dirty and GC_page_was_dirty. */
        !           234: /* Report pages on which stubborn objects were changed.                */
        !           235: void GC_read_changed()
        !           236: {
        !           237:     register GC_PTR * p = GC_changing_list_start;
        !           238:     register GC_PTR q;
        !           239:     register struct hblk * h;
        !           240:     register word index;
        !           241:
        !           242:     if (p == 0) /* initializing */ return;
        !           243:     BCOPY(GC_changed_pages, GC_prev_changed_pages,
        !           244:           (sizeof GC_changed_pages));
        !           245:     BZERO(GC_changed_pages, (sizeof GC_changed_pages));
        !           246:     for (; p <= GC_changing_list_current; p++) {
        !           247:         if ((q = *p) != 0) {
        !           248:             h = HBLKPTR(q);
        !           249:             index = PHT_HASH(h);
        !           250:             set_pht_entry_from_index(GC_changed_pages, index);
        !           251:         }
        !           252:     }
        !           253: }
        !           254:
        !           255: GC_bool GC_page_was_changed(h)
        !           256: struct hblk * h;
        !           257: {
        !           258:     register word index = PHT_HASH(h);
        !           259:
        !           260:     return(get_pht_entry_from_index(GC_prev_changed_pages, index));
        !           261: }
        !           262:
        !           263: /* Remove unreachable entries from changed list. Should only be        */
        !           264: /* called with mark bits consistent and lock held.             */
        !           265: void GC_clean_changing_list()
        !           266: {
        !           267:     register GC_PTR * p = GC_changing_list_start;
        !           268:     register GC_PTR q;
        !           269:     register ptr_t r;
        !           270:     register unsigned long count = 0;
        !           271:     register unsigned long dropped_count = 0;
        !           272:
        !           273:     if (p == 0) /* initializing */ return;
        !           274:     for (; p <= GC_changing_list_current; p++) {
        !           275:         if ((q = *p) != 0) {
        !           276:             count++;
        !           277:             r = (ptr_t)GC_base(q);
        !           278:             if (r == 0 || !GC_is_marked(r)) {
        !           279:                 *p = 0;
        !           280:                 dropped_count++;
        !           281:            }
        !           282:         }
        !           283:     }
        !           284: #   ifdef PRINTSTATS
        !           285:       if (count > 0) {
        !           286:         GC_printf2("%lu entries in changing list: reclaimed %lu\n",
        !           287:                   (unsigned long)count, (unsigned long)dropped_count);
        !           288:       }
        !           289: #   endif
        !           290: }
        !           291:
        !           292: #else /* !STUBBORN_ALLOC */
        !           293:
        !           294: # ifdef __STDC__
        !           295:     GC_PTR GC_malloc_stubborn(size_t lb)
        !           296: # else
        !           297:     GC_PTR GC_malloc_stubborn(lb)
        !           298:     size_t lb;
        !           299: # endif
        !           300: {
        !           301:     return(GC_malloc(lb));
        !           302: }
        !           303:
        !           304: /*ARGSUSED*/
        !           305: void GC_end_stubborn_change(p)
        !           306: GC_PTR p;
        !           307: {
        !           308: }
        !           309:
        !           310: /*ARGSUSED*/
        !           311: void GC_change_stubborn(p)
        !           312: GC_PTR p;
        !           313: {
        !           314: }
        !           315:
        !           316:
        !           317: #endif

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>