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

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

1.1     ! maekawa     1: /*
        !             2:  * Copyright 1988, 1989 Hans-J. Boehm, Alan J. Demers
        !             3:  * Copyright (c) 1991-1996 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, February 1, 1996 1:19 pm PST */
        !            15: # define I_HIDE_POINTERS
        !            16: # include "gc_priv.h"
        !            17: # include "gc_mark.h"
        !            18:
        !            19: /* Type of mark procedure used for marking from finalizable object.    */
        !            20: /* This procedure normally does not mark the object, only its          */
        !            21: /* descendents.                                                                */
        !            22: typedef void finalization_mark_proc(/* ptr_t finalizable_obj_ptr */);
        !            23:
        !            24: # define HASH3(addr,size,log_size) \
        !            25:     ((((word)(addr) >> 3) ^ ((word)(addr) >> (3+(log_size)))) \
        !            26:     & ((size) - 1))
        !            27: #define HASH2(addr,log_size) HASH3(addr, 1 << log_size, log_size)
        !            28:
        !            29: struct hash_chain_entry {
        !            30:     word hidden_key;
        !            31:     struct hash_chain_entry * next;
        !            32: };
        !            33:
        !            34: unsigned GC_finalization_failures = 0;
        !            35:        /* Number of finalization requests that failed for lack of memory. */
        !            36:
        !            37: static struct disappearing_link {
        !            38:     struct hash_chain_entry prolog;
        !            39: #   define dl_hidden_link prolog.hidden_key
        !            40:                                /* Field to be cleared.         */
        !            41: #   define dl_next(x) (struct disappearing_link *)((x) -> prolog.next)
        !            42: #   define dl_set_next(x,y) (x) -> prolog.next = (struct hash_chain_entry *)(y)
        !            43:
        !            44:     word dl_hidden_obj;                /* Pointer to object base       */
        !            45: } **dl_head = 0;
        !            46:
        !            47: static signed_word log_dl_table_size = -1;
        !            48:                        /* Binary log of                                */
        !            49:                        /* current size of array pointed to by dl_head. */
        !            50:                        /* -1 ==> size is 0.                            */
        !            51:
        !            52: word GC_dl_entries = 0;        /* Number of entries currently in disappearing  */
        !            53:                        /* link table.                                  */
        !            54:
        !            55: static struct finalizable_object {
        !            56:     struct hash_chain_entry prolog;
        !            57: #   define fo_hidden_base prolog.hidden_key
        !            58:                                /* Pointer to object base.      */
        !            59:                                /* No longer hidden once object */
        !            60:                                /* is on finalize_now queue.    */
        !            61: #   define fo_next(x) (struct finalizable_object *)((x) -> prolog.next)
        !            62: #   define fo_set_next(x,y) (x) -> prolog.next = (struct hash_chain_entry *)(y)
        !            63:     GC_finalization_proc fo_fn;        /* Finalizer.                   */
        !            64:     ptr_t fo_client_data;
        !            65:     word fo_object_size;       /* In bytes.                    */
        !            66:     finalization_mark_proc * fo_mark_proc;     /* Mark-through procedure */
        !            67: } **fo_head = 0;
        !            68:
        !            69: struct finalizable_object * GC_finalize_now = 0;
        !            70:        /* LIst of objects that should be finalized now.        */
        !            71:
        !            72: static signed_word log_fo_table_size = -1;
        !            73:
        !            74: word GC_fo_entries = 0;
        !            75:
        !            76: # ifdef SRC_M3
        !            77: void GC_push_finalizer_structures()
        !            78: {
        !            79:     GC_push_all((ptr_t)(&dl_head), (ptr_t)(&dl_head) + sizeof(word));
        !            80:     GC_push_all((ptr_t)(&fo_head), (ptr_t)(&fo_head) + sizeof(word));
        !            81: }
        !            82: # endif
        !            83:
        !            84: /* Double the size of a hash table. *size_ptr is the log of its current        */
        !            85: /* size.  May be a noop.                                               */
        !            86: /* *table is a pointer to an array of hash headers.  If we succeed, we */
        !            87: /* update both *table and *log_size_ptr.                               */
        !            88: /* Lock is held.  Signals are disabled.                                        */
        !            89: void GC_grow_table(table, log_size_ptr)
        !            90: struct hash_chain_entry ***table;
        !            91: signed_word * log_size_ptr;
        !            92: {
        !            93:     register word i;
        !            94:     register struct hash_chain_entry *p;
        !            95:     int log_old_size = *log_size_ptr;
        !            96:     register int log_new_size = log_old_size + 1;
        !            97:     word old_size = ((log_old_size == -1)? 0: (1 << log_old_size));
        !            98:     register word new_size = 1 << log_new_size;
        !            99:     struct hash_chain_entry **new_table = (struct hash_chain_entry **)
        !           100:        GC_generic_malloc_inner_ignore_off_page(
        !           101:                (size_t)new_size * sizeof(struct hash_chain_entry *), NORMAL);
        !           102:
        !           103:     if (new_table == 0) {
        !           104:        if (table == 0) {
        !           105:            ABORT("Insufficient space for initial table allocation");
        !           106:        } else {
        !           107:            return;
        !           108:        }
        !           109:     }
        !           110:     for (i = 0; i < old_size; i++) {
        !           111:       p = (*table)[i];
        !           112:       while (p != 0) {
        !           113:         register ptr_t real_key = (ptr_t)REVEAL_POINTER(p -> hidden_key);
        !           114:         register struct hash_chain_entry *next = p -> next;
        !           115:         register int new_hash = HASH3(real_key, new_size, log_new_size);
        !           116:
        !           117:         p -> next = new_table[new_hash];
        !           118:         new_table[new_hash] = p;
        !           119:         p = next;
        !           120:       }
        !           121:     }
        !           122:     *log_size_ptr = log_new_size;
        !           123:     *table = new_table;
        !           124: }
        !           125:
        !           126: # if defined(__STDC__) || defined(__cplusplus)
        !           127:     int GC_register_disappearing_link(GC_PTR * link)
        !           128: # else
        !           129:     int GC_register_disappearing_link(link)
        !           130:     GC_PTR * link;
        !           131: # endif
        !           132: {
        !           133:     ptr_t base;
        !           134:
        !           135:     base = (ptr_t)GC_base((GC_PTR)link);
        !           136:     if (base == 0)
        !           137:        ABORT("Bad arg to GC_register_disappearing_link");
        !           138:     return(GC_general_register_disappearing_link(link, base));
        !           139: }
        !           140:
        !           141: # if defined(__STDC__) || defined(__cplusplus)
        !           142:     int GC_general_register_disappearing_link(GC_PTR * link,
        !           143:                                              GC_PTR obj)
        !           144: # else
        !           145:     int GC_general_register_disappearing_link(link, obj)
        !           146:     GC_PTR * link;
        !           147:     GC_PTR obj;
        !           148: # endif
        !           149:
        !           150: {
        !           151:     struct disappearing_link *curr_dl;
        !           152:     int index;
        !           153:     struct disappearing_link * new_dl;
        !           154:     DCL_LOCK_STATE;
        !           155:
        !           156:     if ((word)link & (ALIGNMENT-1))
        !           157:        ABORT("Bad arg to GC_general_register_disappearing_link");
        !           158: #   ifdef THREADS
        !           159:        DISABLE_SIGNALS();
        !           160:        LOCK();
        !           161: #   endif
        !           162:     if (log_dl_table_size == -1
        !           163:         || GC_dl_entries > ((word)1 << log_dl_table_size)) {
        !           164: #      ifndef THREADS
        !           165:            DISABLE_SIGNALS();
        !           166: #      endif
        !           167:        GC_grow_table((struct hash_chain_entry ***)(&dl_head),
        !           168:                      &log_dl_table_size);
        !           169: #      ifdef PRINTSTATS
        !           170:            GC_printf1("Grew dl table to %lu entries\n",
        !           171:                        (unsigned long)(1 << log_dl_table_size));
        !           172: #      endif
        !           173: #      ifndef THREADS
        !           174:            ENABLE_SIGNALS();
        !           175: #      endif
        !           176:     }
        !           177:     index = HASH2(link, log_dl_table_size);
        !           178:     curr_dl = dl_head[index];
        !           179:     for (curr_dl = dl_head[index]; curr_dl != 0; curr_dl = dl_next(curr_dl)) {
        !           180:         if (curr_dl -> dl_hidden_link == HIDE_POINTER(link)) {
        !           181:             curr_dl -> dl_hidden_obj = HIDE_POINTER(obj);
        !           182: #          ifdef THREADS
        !           183:                 UNLOCK();
        !           184:                ENABLE_SIGNALS();
        !           185: #          endif
        !           186:             return(1);
        !           187:         }
        !           188:     }
        !           189: #   ifdef THREADS
        !           190:       new_dl = (struct disappearing_link *)
        !           191:        GC_generic_malloc_inner(sizeof(struct disappearing_link),NORMAL);
        !           192: #   else
        !           193:       new_dl = (struct disappearing_link *)
        !           194:        GC_malloc(sizeof(struct disappearing_link));
        !           195: #   endif
        !           196:     if (new_dl != 0) {
        !           197:         new_dl -> dl_hidden_obj = HIDE_POINTER(obj);
        !           198:         new_dl -> dl_hidden_link = HIDE_POINTER(link);
        !           199:         dl_set_next(new_dl, dl_head[index]);
        !           200:         dl_head[index] = new_dl;
        !           201:         GC_dl_entries++;
        !           202:     } else {
        !           203:         GC_finalization_failures++;
        !           204:     }
        !           205: #   ifdef THREADS
        !           206:         UNLOCK();
        !           207:         ENABLE_SIGNALS();
        !           208: #   endif
        !           209:     return(0);
        !           210: }
        !           211:
        !           212: # if defined(__STDC__) || defined(__cplusplus)
        !           213:     int GC_unregister_disappearing_link(GC_PTR * link)
        !           214: # else
        !           215:     int GC_unregister_disappearing_link(link)
        !           216:     GC_PTR * link;
        !           217: # endif
        !           218: {
        !           219:     struct disappearing_link *curr_dl, *prev_dl;
        !           220:     int index;
        !           221:     DCL_LOCK_STATE;
        !           222:
        !           223:     DISABLE_SIGNALS();
        !           224:     LOCK();
        !           225:     index = HASH2(link, log_dl_table_size);
        !           226:     if (((unsigned long)link & (ALIGNMENT-1))) goto out;
        !           227:     prev_dl = 0; curr_dl = dl_head[index];
        !           228:     while (curr_dl != 0) {
        !           229:         if (curr_dl -> dl_hidden_link == HIDE_POINTER(link)) {
        !           230:             if (prev_dl == 0) {
        !           231:                 dl_head[index] = dl_next(curr_dl);
        !           232:             } else {
        !           233:                 dl_set_next(prev_dl, dl_next(curr_dl));
        !           234:             }
        !           235:             GC_dl_entries--;
        !           236:             UNLOCK();
        !           237:            ENABLE_SIGNALS();
        !           238:             GC_free((GC_PTR)curr_dl);
        !           239:             return(1);
        !           240:         }
        !           241:         prev_dl = curr_dl;
        !           242:         curr_dl = dl_next(curr_dl);
        !           243:     }
        !           244: out:
        !           245:     UNLOCK();
        !           246:     ENABLE_SIGNALS();
        !           247:     return(0);
        !           248: }
        !           249:
        !           250: /* Possible finalization_marker procedures.  Note that mark stack      */
        !           251: /* overflow is handled by the caller, and is not a disaster.           */
        !           252: void GC_normal_finalize_mark_proc(p)
        !           253: ptr_t p;
        !           254: {
        !           255:     hdr * hhdr = HDR(p);
        !           256:
        !           257:     PUSH_OBJ((word *)p, hhdr, GC_mark_stack_top,
        !           258:             &(GC_mark_stack[GC_mark_stack_size]));
        !           259: }
        !           260:
        !           261: /* This only pays very partial attention to the mark descriptor.       */
        !           262: /* It does the right thing for normal and atomic objects, and treats   */
        !           263: /* most others as normal.                                              */
        !           264: void GC_ignore_self_finalize_mark_proc(p)
        !           265: ptr_t p;
        !           266: {
        !           267:     hdr * hhdr = HDR(p);
        !           268:     word descr = hhdr -> hb_descr;
        !           269:     ptr_t q, r;
        !           270:     ptr_t scan_limit;
        !           271:     ptr_t target_limit = p + WORDS_TO_BYTES(hhdr -> hb_sz) - 1;
        !           272:
        !           273:     if ((descr & DS_TAGS) == DS_LENGTH) {
        !           274:        scan_limit = p + descr - sizeof(word);
        !           275:     } else {
        !           276:        scan_limit = target_limit + 1 - sizeof(word);
        !           277:     }
        !           278:     for (q = p; q <= scan_limit; q += ALIGNMENT) {
        !           279:        r = *(ptr_t *)q;
        !           280:        if (r < p || r > target_limit) {
        !           281:            GC_PUSH_ONE_HEAP((word)r, q);
        !           282:        }
        !           283:     }
        !           284: }
        !           285:
        !           286: /*ARGSUSED*/
        !           287: void GC_null_finalize_mark_proc(p)
        !           288: ptr_t p;
        !           289: {
        !           290: }
        !           291:
        !           292:
        !           293:
        !           294: /* Register a finalization function.  See gc.h for details.    */
        !           295: /* in the nonthreads case, we try to avoid disabling signals,  */
        !           296: /* since it can be expensive.  Threads packages typically      */
        !           297: /* make it cheaper.                                            */
        !           298: void GC_register_finalizer_inner(obj, fn, cd, ofn, ocd, mp)
        !           299: GC_PTR obj;
        !           300: GC_finalization_proc fn;
        !           301: GC_PTR cd;
        !           302: GC_finalization_proc * ofn;
        !           303: GC_PTR * ocd;
        !           304: finalization_mark_proc * mp;
        !           305: {
        !           306:     ptr_t base;
        !           307:     struct finalizable_object * curr_fo, * prev_fo;
        !           308:     int index;
        !           309:     struct finalizable_object *new_fo;
        !           310:     DCL_LOCK_STATE;
        !           311:
        !           312: #   ifdef THREADS
        !           313:        DISABLE_SIGNALS();
        !           314:        LOCK();
        !           315: #   endif
        !           316:     if (log_fo_table_size == -1
        !           317:         || GC_fo_entries > ((word)1 << log_fo_table_size)) {
        !           318: #      ifndef THREADS
        !           319:            DISABLE_SIGNALS();
        !           320: #      endif
        !           321:        GC_grow_table((struct hash_chain_entry ***)(&fo_head),
        !           322:                      &log_fo_table_size);
        !           323: #      ifdef PRINTSTATS
        !           324:            GC_printf1("Grew fo table to %lu entries\n",
        !           325:                        (unsigned long)(1 << log_fo_table_size));
        !           326: #      endif
        !           327: #      ifndef THREADS
        !           328:            ENABLE_SIGNALS();
        !           329: #      endif
        !           330:     }
        !           331:     /* in the THREADS case signals are disabled and we hold allocation */
        !           332:     /* lock; otherwise neither is true.  Proceed carefully.            */
        !           333:     base = (ptr_t)obj;
        !           334:     index = HASH2(base, log_fo_table_size);
        !           335:     prev_fo = 0; curr_fo = fo_head[index];
        !           336:     while (curr_fo != 0) {
        !           337:         if (curr_fo -> fo_hidden_base == HIDE_POINTER(base)) {
        !           338:             /* Interruption by a signal in the middle of this  */
        !           339:             /* should be safe.  The client may see only *ocd   */
        !           340:             /* updated, but we'll declare that to be his       */
        !           341:             /* problem.                                                */
        !           342:             if (ocd) *ocd = (GC_PTR) curr_fo -> fo_client_data;
        !           343:             if (ofn) *ofn = curr_fo -> fo_fn;
        !           344:             /* Delete the structure for base. */
        !           345:                 if (prev_fo == 0) {
        !           346:                   fo_head[index] = fo_next(curr_fo);
        !           347:                 } else {
        !           348:                   fo_set_next(prev_fo, fo_next(curr_fo));
        !           349:                 }
        !           350:             if (fn == 0) {
        !           351:                 GC_fo_entries--;
        !           352:                   /* May not happen if we get a signal.  But a high    */
        !           353:                   /* estimate will only make the table larger than     */
        !           354:                   /* necessary.                                                */
        !           355: #              ifndef THREADS
        !           356:                   GC_free((GC_PTR)curr_fo);
        !           357: #              endif
        !           358:             } else {
        !           359:                 curr_fo -> fo_fn = fn;
        !           360:                 curr_fo -> fo_client_data = (ptr_t)cd;
        !           361:                 curr_fo -> fo_mark_proc = mp;
        !           362:                /* Reinsert it.  We deleted it first to maintain        */
        !           363:                /* consistency in the event of a signal.                */
        !           364:                if (prev_fo == 0) {
        !           365:                   fo_head[index] = curr_fo;
        !           366:                 } else {
        !           367:                   fo_set_next(prev_fo, curr_fo);
        !           368:                 }
        !           369:             }
        !           370: #          ifdef THREADS
        !           371:                 UNLOCK();
        !           372:                ENABLE_SIGNALS();
        !           373: #          endif
        !           374:             return;
        !           375:         }
        !           376:         prev_fo = curr_fo;
        !           377:         curr_fo = fo_next(curr_fo);
        !           378:     }
        !           379:     if (ofn) *ofn = 0;
        !           380:     if (ocd) *ocd = 0;
        !           381:     if (fn == 0) {
        !           382: #      ifdef THREADS
        !           383:             UNLOCK();
        !           384:            ENABLE_SIGNALS();
        !           385: #      endif
        !           386:         return;
        !           387:     }
        !           388: #   ifdef THREADS
        !           389:       new_fo = (struct finalizable_object *)
        !           390:        GC_generic_malloc_inner(sizeof(struct finalizable_object),NORMAL);
        !           391: #   else
        !           392:       new_fo = (struct finalizable_object *)
        !           393:        GC_malloc(sizeof(struct finalizable_object));
        !           394: #   endif
        !           395:     if (new_fo != 0) {
        !           396:         new_fo -> fo_hidden_base = (word)HIDE_POINTER(base);
        !           397:        new_fo -> fo_fn = fn;
        !           398:        new_fo -> fo_client_data = (ptr_t)cd;
        !           399:        new_fo -> fo_object_size = GC_size(base);
        !           400:        new_fo -> fo_mark_proc = mp;
        !           401:        fo_set_next(new_fo, fo_head[index]);
        !           402:        GC_fo_entries++;
        !           403:        fo_head[index] = new_fo;
        !           404:     } else {
        !           405:        GC_finalization_failures++;
        !           406:     }
        !           407: #   ifdef THREADS
        !           408:         UNLOCK();
        !           409:        ENABLE_SIGNALS();
        !           410: #   endif
        !           411: }
        !           412:
        !           413: # if defined(__STDC__)
        !           414:     void GC_register_finalizer(void * obj,
        !           415:                               GC_finalization_proc fn, void * cd,
        !           416:                               GC_finalization_proc *ofn, void ** ocd)
        !           417: # else
        !           418:     void GC_register_finalizer(obj, fn, cd, ofn, ocd)
        !           419:     GC_PTR obj;
        !           420:     GC_finalization_proc fn;
        !           421:     GC_PTR cd;
        !           422:     GC_finalization_proc * ofn;
        !           423:     GC_PTR * ocd;
        !           424: # endif
        !           425: {
        !           426:     GC_register_finalizer_inner(obj, fn, cd, ofn,
        !           427:                                ocd, GC_normal_finalize_mark_proc);
        !           428: }
        !           429:
        !           430: # if defined(__STDC__)
        !           431:     void GC_register_finalizer_ignore_self(void * obj,
        !           432:                               GC_finalization_proc fn, void * cd,
        !           433:                               GC_finalization_proc *ofn, void ** ocd)
        !           434: # else
        !           435:     void GC_register_finalizer_ignore_self(obj, fn, cd, ofn, ocd)
        !           436:     GC_PTR obj;
        !           437:     GC_finalization_proc fn;
        !           438:     GC_PTR cd;
        !           439:     GC_finalization_proc * ofn;
        !           440:     GC_PTR * ocd;
        !           441: # endif
        !           442: {
        !           443:     GC_register_finalizer_inner(obj, fn, cd, ofn,
        !           444:                                ocd, GC_ignore_self_finalize_mark_proc);
        !           445: }
        !           446:
        !           447: # if defined(__STDC__)
        !           448:     void GC_register_finalizer_no_order(void * obj,
        !           449:                               GC_finalization_proc fn, void * cd,
        !           450:                               GC_finalization_proc *ofn, void ** ocd)
        !           451: # else
        !           452:     void GC_register_finalizer_no_order(obj, fn, cd, ofn, ocd)
        !           453:     GC_PTR obj;
        !           454:     GC_finalization_proc fn;
        !           455:     GC_PTR cd;
        !           456:     GC_finalization_proc * ofn;
        !           457:     GC_PTR * ocd;
        !           458: # endif
        !           459: {
        !           460:     GC_register_finalizer_inner(obj, fn, cd, ofn,
        !           461:                                ocd, GC_null_finalize_mark_proc);
        !           462: }
        !           463:
        !           464: /* Called with world stopped.  Cause disappearing links to disappear,  */
        !           465: /* and invoke finalizers.                                              */
        !           466: void GC_finalize()
        !           467: {
        !           468:     struct disappearing_link * curr_dl, * prev_dl, * next_dl;
        !           469:     struct finalizable_object * curr_fo, * prev_fo, * next_fo;
        !           470:     ptr_t real_ptr, real_link;
        !           471:     register int i;
        !           472:     int dl_size = (log_dl_table_size == -1 ) ? 0 : (1 << log_dl_table_size);
        !           473:     int fo_size = (log_fo_table_size == -1 ) ? 0 : (1 << log_fo_table_size);
        !           474:
        !           475:   /* Make disappearing links disappear */
        !           476:     for (i = 0; i < dl_size; i++) {
        !           477:       curr_dl = dl_head[i];
        !           478:       prev_dl = 0;
        !           479:       while (curr_dl != 0) {
        !           480:         real_ptr = (ptr_t)REVEAL_POINTER(curr_dl -> dl_hidden_obj);
        !           481:         real_link = (ptr_t)REVEAL_POINTER(curr_dl -> dl_hidden_link);
        !           482:         if (!GC_is_marked(real_ptr)) {
        !           483:             *(word *)real_link = 0;
        !           484:             next_dl = dl_next(curr_dl);
        !           485:             if (prev_dl == 0) {
        !           486:                 dl_head[i] = next_dl;
        !           487:             } else {
        !           488:                 dl_set_next(prev_dl, next_dl);
        !           489:             }
        !           490:             GC_clear_mark_bit((ptr_t)curr_dl);
        !           491:             GC_dl_entries--;
        !           492:             curr_dl = next_dl;
        !           493:         } else {
        !           494:             prev_dl = curr_dl;
        !           495:             curr_dl = dl_next(curr_dl);
        !           496:         }
        !           497:       }
        !           498:     }
        !           499:   /* Mark all objects reachable via chains of 1 or more pointers       */
        !           500:   /* from finalizable objects.                                         */
        !           501: #   ifdef PRINTSTATS
        !           502:         if (GC_mark_state != MS_NONE) ABORT("Bad mark state");
        !           503: #   endif
        !           504:     for (i = 0; i < fo_size; i++) {
        !           505:       for (curr_fo = fo_head[i]; curr_fo != 0; curr_fo = fo_next(curr_fo)) {
        !           506:         real_ptr = (ptr_t)REVEAL_POINTER(curr_fo -> fo_hidden_base);
        !           507:         if (!GC_is_marked(real_ptr)) {
        !           508:             GC_MARK_FO(real_ptr, curr_fo -> fo_mark_proc);
        !           509:             if (GC_is_marked(real_ptr)) {
        !           510:                 WARN("Finalization cycle involving %lx\n", real_ptr);
        !           511:             }
        !           512:         }
        !           513:       }
        !           514:     }
        !           515:   /* Enqueue for finalization all objects that are still               */
        !           516:   /* unreachable.                                                      */
        !           517:     GC_words_finalized = 0;
        !           518:     for (i = 0; i < fo_size; i++) {
        !           519:       curr_fo = fo_head[i];
        !           520:       prev_fo = 0;
        !           521:       while (curr_fo != 0) {
        !           522:         real_ptr = (ptr_t)REVEAL_POINTER(curr_fo -> fo_hidden_base);
        !           523:         if (!GC_is_marked(real_ptr)) {
        !           524: #         ifndef JAVA_FINALIZATION
        !           525:             GC_set_mark_bit(real_ptr);
        !           526: #         endif
        !           527:             /* Delete from hash table */
        !           528:               next_fo = fo_next(curr_fo);
        !           529:               if (prev_fo == 0) {
        !           530:                 fo_head[i] = next_fo;
        !           531:               } else {
        !           532:                 fo_set_next(prev_fo, next_fo);
        !           533:               }
        !           534:               GC_fo_entries--;
        !           535:             /* Add to list of objects awaiting finalization.   */
        !           536:               fo_set_next(curr_fo, GC_finalize_now);
        !           537:               GC_finalize_now = curr_fo;
        !           538:               /* unhide object pointer so any future collections will  */
        !           539:               /* see it.                                               */
        !           540:               curr_fo -> fo_hidden_base =
        !           541:                        (word) REVEAL_POINTER(curr_fo -> fo_hidden_base);
        !           542:               GC_words_finalized +=
        !           543:                        ALIGNED_WORDS(curr_fo -> fo_object_size)
        !           544:                        + ALIGNED_WORDS(sizeof(struct finalizable_object));
        !           545: #          ifdef PRINTSTATS
        !           546:               if (!GC_is_marked((ptr_t)curr_fo)) {
        !           547:                 ABORT("GC_finalize: found accessible unmarked object\n");
        !           548:               }
        !           549: #          endif
        !           550:             curr_fo = next_fo;
        !           551:         } else {
        !           552:             prev_fo = curr_fo;
        !           553:             curr_fo = fo_next(curr_fo);
        !           554:         }
        !           555:       }
        !           556:     }
        !           557:
        !           558: # ifdef JAVA_FINALIZATION
        !           559:   /* make sure we mark everything reachable from objects finalized
        !           560:      using the no_order mark_proc */
        !           561:     for (curr_fo = GC_finalize_now;
        !           562:         curr_fo != NULL; curr_fo = fo_next(curr_fo)) {
        !           563:        real_ptr = (ptr_t)curr_fo -> fo_hidden_base;
        !           564:        if (!GC_is_marked(real_ptr)) {
        !           565:            if (curr_fo -> fo_mark_proc == GC_null_finalize_mark_proc) {
        !           566:                GC_MARK_FO(real_ptr, GC_normal_finalize_mark_proc);
        !           567:            }
        !           568:            GC_set_mark_bit(real_ptr);
        !           569:        }
        !           570:     }
        !           571: # endif
        !           572:
        !           573:   /* Remove dangling disappearing links. */
        !           574:     for (i = 0; i < dl_size; i++) {
        !           575:       curr_dl = dl_head[i];
        !           576:       prev_dl = 0;
        !           577:       while (curr_dl != 0) {
        !           578:         real_link = GC_base((ptr_t)REVEAL_POINTER(curr_dl -> dl_hidden_link));
        !           579:         if (real_link != 0 && !GC_is_marked(real_link)) {
        !           580:             next_dl = dl_next(curr_dl);
        !           581:             if (prev_dl == 0) {
        !           582:                 dl_head[i] = next_dl;
        !           583:             } else {
        !           584:                 dl_set_next(prev_dl, next_dl);
        !           585:             }
        !           586:             GC_clear_mark_bit((ptr_t)curr_dl);
        !           587:             GC_dl_entries--;
        !           588:             curr_dl = next_dl;
        !           589:         } else {
        !           590:             prev_dl = curr_dl;
        !           591:             curr_dl = dl_next(curr_dl);
        !           592:         }
        !           593:       }
        !           594:     }
        !           595: }
        !           596:
        !           597: #ifdef JAVA_FINALIZATION
        !           598:
        !           599: /* Enqueue all remaining finalizers to be run - Assumes lock is
        !           600:  * held, and signals are disabled */
        !           601: void GC_enqueue_all_finalizers()
        !           602: {
        !           603:     struct finalizable_object * curr_fo, * prev_fo, * next_fo;
        !           604:     ptr_t real_ptr, real_link;
        !           605:     register int i;
        !           606:     int fo_size;
        !           607:
        !           608:     fo_size = (log_fo_table_size == -1 ) ? 0 : (1 << log_fo_table_size);
        !           609:     GC_words_finalized = 0;
        !           610:     for (i = 0; i < fo_size; i++) {
        !           611:         curr_fo = fo_head[i];
        !           612:         prev_fo = 0;
        !           613:       while (curr_fo != 0) {
        !           614:           real_ptr = (ptr_t)REVEAL_POINTER(curr_fo -> fo_hidden_base);
        !           615:           GC_MARK_FO(real_ptr, GC_normal_finalize_mark_proc);
        !           616:           GC_set_mark_bit(real_ptr);
        !           617:
        !           618:           /* Delete from hash table */
        !           619:           next_fo = fo_next(curr_fo);
        !           620:           if (prev_fo == 0) {
        !           621:               fo_head[i] = next_fo;
        !           622:           } else {
        !           623:               fo_set_next(prev_fo, next_fo);
        !           624:           }
        !           625:           GC_fo_entries--;
        !           626:
        !           627:           /* Add to list of objects awaiting finalization.     */
        !           628:           fo_set_next(curr_fo, GC_finalize_now);
        !           629:           GC_finalize_now = curr_fo;
        !           630:
        !           631:           /* unhide object pointer so any future collections will      */
        !           632:           /* see it.                                           */
        !           633:           curr_fo -> fo_hidden_base =
        !           634:                        (word) REVEAL_POINTER(curr_fo -> fo_hidden_base);
        !           635:
        !           636:           GC_words_finalized +=
        !           637:                ALIGNED_WORDS(curr_fo -> fo_object_size)
        !           638:                        + ALIGNED_WORDS(sizeof(struct finalizable_object));
        !           639:           curr_fo = next_fo;
        !           640:         }
        !           641:     }
        !           642:
        !           643:     return;
        !           644: }
        !           645:
        !           646: /* Invoke all remaining finalizers that haven't yet been run.
        !           647:  * This is needed for strict compliance with the Java standard,
        !           648:  * which can make the runtime guarantee that all finalizers are run.
        !           649:  * Unfortunately, the Java standard implies we have to keep running
        !           650:  * finalizers until there are no more left, a potential infinite loop.
        !           651:  * YUCK.
        !           652:  * This routine is externally callable, so is called without
        !           653:  * the allocation lock.
        !           654:  */
        !           655: void GC_finalize_all()
        !           656: {
        !           657:     DCL_LOCK_STATE;
        !           658:
        !           659:     DISABLE_SIGNALS();
        !           660:     LOCK();
        !           661:     while (GC_fo_entries > 0) {
        !           662:       GC_enqueue_all_finalizers();
        !           663:       UNLOCK();
        !           664:       ENABLE_SIGNALS();
        !           665:       GC_INVOKE_FINALIZERS();
        !           666:       DISABLE_SIGNALS();
        !           667:       LOCK();
        !           668:     }
        !           669:     UNLOCK();
        !           670:     ENABLE_SIGNALS();
        !           671: }
        !           672: #endif
        !           673:
        !           674: /* Invoke finalizers for all objects that are ready to be finalized.   */
        !           675: /* Should be called without allocation lock.                           */
        !           676: int GC_invoke_finalizers()
        !           677: {
        !           678:     register struct finalizable_object * curr_fo;
        !           679:     register int count = 0;
        !           680:     DCL_LOCK_STATE;
        !           681:
        !           682:     while (GC_finalize_now != 0) {
        !           683: #      ifdef THREADS
        !           684:            DISABLE_SIGNALS();
        !           685:            LOCK();
        !           686: #      endif
        !           687:        curr_fo = GC_finalize_now;
        !           688: #      ifdef THREADS
        !           689:            if (curr_fo != 0) GC_finalize_now = fo_next(curr_fo);
        !           690:            UNLOCK();
        !           691:            ENABLE_SIGNALS();
        !           692:            if (curr_fo == 0) break;
        !           693: #      else
        !           694:            GC_finalize_now = fo_next(curr_fo);
        !           695: #      endif
        !           696:        fo_set_next(curr_fo, 0);
        !           697:        (*(curr_fo -> fo_fn))((ptr_t)(curr_fo -> fo_hidden_base),
        !           698:                              curr_fo -> fo_client_data);
        !           699:        curr_fo -> fo_client_data = 0;
        !           700:        ++count;
        !           701: #      ifdef UNDEFINED
        !           702:            /* This is probably a bad idea.  It throws off accounting if */
        !           703:            /* nearly all objects are finalizable.  O.w. it shouldn't    */
        !           704:            /* matter.                                                   */
        !           705:            GC_free((GC_PTR)curr_fo);
        !           706: #      endif
        !           707:     }
        !           708:     return count;
        !           709: }
        !           710:
        !           711: # ifdef __STDC__
        !           712:     GC_PTR GC_call_with_alloc_lock(GC_fn_type fn,
        !           713:                                         GC_PTR client_data)
        !           714: # else
        !           715:     GC_PTR GC_call_with_alloc_lock(fn, client_data)
        !           716:     GC_fn_type fn;
        !           717:     GC_PTR client_data;
        !           718: # endif
        !           719: {
        !           720:     GC_PTR result;
        !           721:     DCL_LOCK_STATE;
        !           722:
        !           723: #   ifdef THREADS
        !           724:       DISABLE_SIGNALS();
        !           725:       LOCK();
        !           726:       SET_LOCK_HOLDER();
        !           727: #   endif
        !           728:     result = (*fn)(client_data);
        !           729: #   ifdef THREADS
        !           730:       UNSET_LOCK_HOLDER();
        !           731:       UNLOCK();
        !           732:       ENABLE_SIGNALS();
        !           733: #   endif
        !           734:     return(result);
        !           735: }

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