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

Annotation of OpenXM_contrib/gc/reclaim.c, Revision 1.1.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, February 15, 1996 2:41 pm PST */
                     15:
                     16: #include <stdio.h>
                     17: #include "gc_priv.h"
                     18:
                     19: signed_word GC_mem_found = 0;
                     20:                        /* Number of words of memory reclaimed     */
                     21:
                     22: # ifdef FIND_LEAK
                     23: static void report_leak(p, sz)
                     24: ptr_t p;
                     25: word sz;
                     26: {
                     27:     if (HDR(p) -> hb_obj_kind == PTRFREE) {
                     28:         GC_err_printf0("Leaked atomic object at ");
                     29:     } else {
                     30:         GC_err_printf0("Leaked composite object at ");
                     31:     }
                     32:     if (GC_debugging_started && GC_has_debug_info(p)) {
                     33:         GC_print_obj(p);
                     34:     } else {
                     35:         GC_err_printf2("0x%lx (appr. size = %ld)\n",
                     36:                              (unsigned long)p,
                     37:                              (unsigned long)WORDS_TO_BYTES(sz));
                     38:     }
                     39: }
                     40:
                     41: #   define FOUND_FREE(hblk, word_no) \
                     42:       if (abort_if_found) { \
                     43:          report_leak((ptr_t)hblk + WORDS_TO_BYTES(word_no), \
                     44:                     HDR(hblk) -> hb_sz); \
                     45:       }
                     46: # else
                     47: #   define FOUND_FREE(hblk, word_no)
                     48: # endif
                     49:
                     50: /*
                     51:  * reclaim phase
                     52:  *
                     53:  */
                     54:
                     55:
                     56: /*
                     57:  * Test whether a block is completely empty, i.e. contains no marked
                     58:  * objects.  This does not require the block to be in physical
                     59:  * memory.
                     60:  */
                     61:
                     62: GC_bool GC_block_empty(hhdr)
                     63: register hdr * hhdr;
                     64: {
                     65:     register word *p = (word *)(&(hhdr -> hb_marks[0]));
                     66:     register word * plim =
                     67:                        (word *)(&(hhdr -> hb_marks[MARK_BITS_SZ]));
                     68:     while (p < plim) {
                     69:        if (*p++) return(FALSE);
                     70:     }
                     71:     return(TRUE);
                     72: }
                     73:
                     74: # ifdef GATHERSTATS
                     75: #   define INCR_WORDS(sz) n_words_found += (sz)
                     76: # else
                     77: #   define INCR_WORDS(sz)
                     78: # endif
                     79: /*
                     80:  * Restore unmarked small objects in h of size sz to the object
                     81:  * free list.  Returns the new list.
                     82:  * Clears unmarked objects.
                     83:  */
                     84: /*ARGSUSED*/
                     85: ptr_t GC_reclaim_clear(hbp, hhdr, sz, list, abort_if_found)
                     86: register struct hblk *hbp;     /* ptr to current heap block            */
                     87: register hdr * hhdr;
                     88: GC_bool abort_if_found;                /* Abort if a reclaimable object is found */
                     89: register ptr_t list;
                     90: register word sz;
                     91: {
                     92:     register int word_no;
                     93:     register word *p, *q, *plim;
                     94: #   ifdef GATHERSTATS
                     95:         register int n_words_found = 0;
                     96: #   endif
                     97:
                     98:     p = (word *)(hbp->hb_body);
                     99:     word_no = HDR_WORDS;
                    100:     plim = (word *)((((word)hbp) + HBLKSIZE)
                    101:                   - WORDS_TO_BYTES(sz));
                    102:
                    103:     /* go through all words in block */
                    104:        while( p <= plim )  {
                    105:            if( mark_bit_from_hdr(hhdr, word_no) ) {
                    106:                p += sz;
                    107:            } else {
                    108:                FOUND_FREE(hbp, word_no);
                    109:                INCR_WORDS(sz);
                    110:                /* object is available - put on list */
                    111:                    obj_link(p) = list;
                    112:                    list = ((ptr_t)p);
                    113:                /* Clear object, advance p to next object in the process */
                    114:                    q = p + sz;
                    115:                     p++; /* Skip link field */
                    116:                     while (p < q) {
                    117:                        *p++ = 0;
                    118:                    }
                    119:            }
                    120:            word_no += sz;
                    121:        }
                    122: #   ifdef GATHERSTATS
                    123:        GC_mem_found += n_words_found;
                    124: #   endif
                    125:     return(list);
                    126: }
                    127:
                    128: #ifndef SMALL_CONFIG
                    129:
                    130: /*
                    131:  * A special case for 2 word composite objects (e.g. cons cells):
                    132:  */
                    133: /*ARGSUSED*/
                    134: ptr_t GC_reclaim_clear2(hbp, hhdr, list, abort_if_found)
                    135: register struct hblk *hbp;     /* ptr to current heap block            */
                    136: hdr * hhdr;
                    137: GC_bool abort_if_found;                /* Abort if a reclaimable object is found */
                    138: register ptr_t list;
                    139: {
                    140:     register word * mark_word_addr = &(hhdr->hb_marks[divWORDSZ(HDR_WORDS)]);
                    141:     register word *p, *plim;
                    142: #   ifdef GATHERSTATS
                    143:         register int n_words_found = 0;
                    144: #   endif
                    145:     register word mark_word;
                    146:     register int i;
                    147: #   define DO_OBJ(start_displ) \
                    148:        if (!(mark_word & ((word)1 << start_displ))) { \
                    149:            FOUND_FREE(hbp, p - (word *)hbp + start_displ); \
                    150:            p[start_displ] = (word)list; \
                    151:            list = (ptr_t)(p+start_displ); \
                    152:            p[start_displ+1] = 0; \
                    153:            INCR_WORDS(2); \
                    154:        }
                    155:
                    156:     p = (word *)(hbp->hb_body);
                    157:     plim = (word *)(((word)hbp) + HBLKSIZE);
                    158:
                    159:     /* go through all words in block */
                    160:        while( p < plim )  {
                    161:            mark_word = *mark_word_addr++;
                    162:            for (i = 0; i < WORDSZ; i += 8) {
                    163:                DO_OBJ(0);
                    164:                DO_OBJ(2);
                    165:                DO_OBJ(4);
                    166:                DO_OBJ(6);
                    167:                p += 8;
                    168:                mark_word >>= 8;
                    169:            }
                    170:        }
                    171: #   ifdef GATHERSTATS
                    172:        GC_mem_found += n_words_found;
                    173: #   endif
                    174:     return(list);
                    175: #   undef DO_OBJ
                    176: }
                    177:
                    178: /*
                    179:  * Another special case for 4 word composite objects:
                    180:  */
                    181: /*ARGSUSED*/
                    182: ptr_t GC_reclaim_clear4(hbp, hhdr, list, abort_if_found)
                    183: register struct hblk *hbp;     /* ptr to current heap block            */
                    184: hdr * hhdr;
                    185: GC_bool abort_if_found;                /* Abort if a reclaimable object is found */
                    186: register ptr_t list;
                    187: {
                    188:     register word * mark_word_addr = &(hhdr->hb_marks[divWORDSZ(HDR_WORDS)]);
                    189:     register word *p, *plim;
                    190: #   ifdef GATHERSTATS
                    191:         register int n_words_found = 0;
                    192: #   endif
                    193:     register word mark_word;
                    194: #   define DO_OBJ(start_displ) \
                    195:        if (!(mark_word & ((word)1 << start_displ))) { \
                    196:            FOUND_FREE(hbp, p - (word *)hbp + start_displ); \
                    197:            p[start_displ] = (word)list; \
                    198:            list = (ptr_t)(p+start_displ); \
                    199:            p[start_displ+1] = 0; \
                    200:            p[start_displ+2] = 0; \
                    201:            p[start_displ+3] = 0; \
                    202:            INCR_WORDS(4); \
                    203:        }
                    204:
                    205:     p = (word *)(hbp->hb_body);
                    206:     plim = (word *)(((word)hbp) + HBLKSIZE);
                    207:
                    208:     /* go through all words in block */
                    209:        while( p < plim )  {
                    210:            mark_word = *mark_word_addr++;
                    211:            DO_OBJ(0);
                    212:            DO_OBJ(4);
                    213:            DO_OBJ(8);
                    214:            DO_OBJ(12);
                    215:            DO_OBJ(16);
                    216:            DO_OBJ(20);
                    217:            DO_OBJ(24);
                    218:            DO_OBJ(28);
                    219: #          if CPP_WORDSZ == 64
                    220:              DO_OBJ(32);
                    221:              DO_OBJ(36);
                    222:              DO_OBJ(40);
                    223:              DO_OBJ(44);
                    224:              DO_OBJ(48);
                    225:              DO_OBJ(52);
                    226:              DO_OBJ(56);
                    227:              DO_OBJ(60);
                    228: #          endif
                    229:            p += WORDSZ;
                    230:        }
                    231: #   ifdef GATHERSTATS
                    232:        GC_mem_found += n_words_found;
                    233: #   endif
                    234:     return(list);
                    235: #   undef DO_OBJ
                    236: }
                    237:
                    238: #endif /* !SMALL_CONFIG */
                    239:
                    240: /* The same thing, but don't clear objects: */
                    241: /*ARGSUSED*/
                    242: ptr_t GC_reclaim_uninit(hbp, hhdr, sz, list, abort_if_found)
                    243: register struct hblk *hbp;     /* ptr to current heap block            */
                    244: register hdr * hhdr;
                    245: GC_bool abort_if_found;                /* Abort if a reclaimable object is found */
                    246: register ptr_t list;
                    247: register word sz;
                    248: {
                    249:     register int word_no;
                    250:     register word *p, *plim;
                    251: #   ifdef GATHERSTATS
                    252:         register int n_words_found = 0;
                    253: #   endif
                    254:
                    255:     p = (word *)(hbp->hb_body);
                    256:     word_no = HDR_WORDS;
                    257:     plim = (word *)((((word)hbp) + HBLKSIZE)
                    258:                   - WORDS_TO_BYTES(sz));
                    259:
                    260:     /* go through all words in block */
                    261:        while( p <= plim )  {
                    262:            if( !mark_bit_from_hdr(hhdr, word_no) ) {
                    263:                FOUND_FREE(hbp, word_no);
                    264:                INCR_WORDS(sz);
                    265:                /* object is available - put on list */
                    266:                    obj_link(p) = list;
                    267:                    list = ((ptr_t)p);
                    268:            }
                    269:            p += sz;
                    270:            word_no += sz;
                    271:        }
                    272: #   ifdef GATHERSTATS
                    273:        GC_mem_found += n_words_found;
                    274: #   endif
                    275:     return(list);
                    276: }
                    277:
                    278: #ifndef SMALL_CONFIG
                    279: /*
                    280:  * Another special case for 2 word atomic objects:
                    281:  */
                    282: /*ARGSUSED*/
                    283: ptr_t GC_reclaim_uninit2(hbp, hhdr, list, abort_if_found)
                    284: register struct hblk *hbp;     /* ptr to current heap block            */
                    285: hdr * hhdr;
                    286: GC_bool abort_if_found;                /* Abort if a reclaimable object is found */
                    287: register ptr_t list;
                    288: {
                    289:     register word * mark_word_addr = &(hhdr->hb_marks[divWORDSZ(HDR_WORDS)]);
                    290:     register word *p, *plim;
                    291: #   ifdef GATHERSTATS
                    292:         register int n_words_found = 0;
                    293: #   endif
                    294:     register word mark_word;
                    295:     register int i;
                    296: #   define DO_OBJ(start_displ) \
                    297:        if (!(mark_word & ((word)1 << start_displ))) { \
                    298:            FOUND_FREE(hbp, p - (word *)hbp + start_displ); \
                    299:            p[start_displ] = (word)list; \
                    300:            list = (ptr_t)(p+start_displ); \
                    301:            INCR_WORDS(2); \
                    302:        }
                    303:
                    304:     p = (word *)(hbp->hb_body);
                    305:     plim = (word *)(((word)hbp) + HBLKSIZE);
                    306:
                    307:     /* go through all words in block */
                    308:        while( p < plim )  {
                    309:            mark_word = *mark_word_addr++;
                    310:            for (i = 0; i < WORDSZ; i += 8) {
                    311:                DO_OBJ(0);
                    312:                DO_OBJ(2);
                    313:                DO_OBJ(4);
                    314:                DO_OBJ(6);
                    315:                p += 8;
                    316:                mark_word >>= 8;
                    317:            }
                    318:        }
                    319: #   ifdef GATHERSTATS
                    320:        GC_mem_found += n_words_found;
                    321: #   endif
                    322:     return(list);
                    323: #   undef DO_OBJ
                    324: }
                    325:
                    326: /*
                    327:  * Another special case for 4 word atomic objects:
                    328:  */
                    329: /*ARGSUSED*/
                    330: ptr_t GC_reclaim_uninit4(hbp, hhdr, list, abort_if_found)
                    331: register struct hblk *hbp;     /* ptr to current heap block            */
                    332: hdr * hhdr;
                    333: GC_bool abort_if_found;                /* Abort if a reclaimable object is found */
                    334: register ptr_t list;
                    335: {
                    336:     register word * mark_word_addr = &(hhdr->hb_marks[divWORDSZ(HDR_WORDS)]);
                    337:     register word *p, *plim;
                    338: #   ifdef GATHERSTATS
                    339:         register int n_words_found = 0;
                    340: #   endif
                    341:     register word mark_word;
                    342: #   define DO_OBJ(start_displ) \
                    343:        if (!(mark_word & ((word)1 << start_displ))) { \
                    344:            FOUND_FREE(hbp, p - (word *)hbp + start_displ); \
                    345:            p[start_displ] = (word)list; \
                    346:            list = (ptr_t)(p+start_displ); \
                    347:            INCR_WORDS(4); \
                    348:        }
                    349:
                    350:     p = (word *)(hbp->hb_body);
                    351:     plim = (word *)(((word)hbp) + HBLKSIZE);
                    352:
                    353:     /* go through all words in block */
                    354:        while( p < plim )  {
                    355:            mark_word = *mark_word_addr++;
                    356:            DO_OBJ(0);
                    357:            DO_OBJ(4);
                    358:            DO_OBJ(8);
                    359:            DO_OBJ(12);
                    360:            DO_OBJ(16);
                    361:            DO_OBJ(20);
                    362:            DO_OBJ(24);
                    363:            DO_OBJ(28);
                    364: #          if CPP_WORDSZ == 64
                    365:              DO_OBJ(32);
                    366:              DO_OBJ(36);
                    367:              DO_OBJ(40);
                    368:              DO_OBJ(44);
                    369:              DO_OBJ(48);
                    370:              DO_OBJ(52);
                    371:              DO_OBJ(56);
                    372:              DO_OBJ(60);
                    373: #          endif
                    374:            p += WORDSZ;
                    375:        }
                    376: #   ifdef GATHERSTATS
                    377:        GC_mem_found += n_words_found;
                    378: #   endif
                    379:     return(list);
                    380: #   undef DO_OBJ
                    381: }
                    382:
                    383: /* Finally the one word case, which never requires any clearing: */
                    384: /*ARGSUSED*/
                    385: ptr_t GC_reclaim1(hbp, hhdr, list, abort_if_found)
                    386: register struct hblk *hbp;     /* ptr to current heap block            */
                    387: hdr * hhdr;
                    388: GC_bool abort_if_found;                /* Abort if a reclaimable object is found */
                    389: register ptr_t list;
                    390: {
                    391:     register word * mark_word_addr = &(hhdr->hb_marks[divWORDSZ(HDR_WORDS)]);
                    392:     register word *p, *plim;
                    393: #   ifdef GATHERSTATS
                    394:         register int n_words_found = 0;
                    395: #   endif
                    396:     register word mark_word;
                    397:     register int i;
                    398: #   define DO_OBJ(start_displ) \
                    399:        if (!(mark_word & ((word)1 << start_displ))) { \
                    400:            FOUND_FREE(hbp, p - (word *)hbp + start_displ); \
                    401:            p[start_displ] = (word)list; \
                    402:            list = (ptr_t)(p+start_displ); \
                    403:            INCR_WORDS(1); \
                    404:        }
                    405:
                    406:     p = (word *)(hbp->hb_body);
                    407:     plim = (word *)(((word)hbp) + HBLKSIZE);
                    408:
                    409:     /* go through all words in block */
                    410:        while( p < plim )  {
                    411:            mark_word = *mark_word_addr++;
                    412:            for (i = 0; i < WORDSZ; i += 4) {
                    413:                DO_OBJ(0);
                    414:                DO_OBJ(1);
                    415:                DO_OBJ(2);
                    416:                DO_OBJ(3);
                    417:                p += 4;
                    418:                mark_word >>= 4;
                    419:            }
                    420:        }
                    421: #   ifdef GATHERSTATS
                    422:        GC_mem_found += n_words_found;
                    423: #   endif
                    424:     return(list);
                    425: #   undef DO_OBJ
                    426: }
                    427:
                    428: #endif /* !SMALL_CONFIG */
                    429:
                    430: /*
                    431:  * Restore unmarked small objects in the block pointed to by hbp
                    432:  * to the appropriate object free list.
                    433:  * If entirely empty blocks are to be completely deallocated, then
                    434:  * caller should perform that check.
                    435:  */
                    436: void GC_reclaim_small_nonempty_block(hbp, abort_if_found)
                    437: register struct hblk *hbp;     /* ptr to current heap block            */
                    438: int abort_if_found;            /* Abort if a reclaimable object is found */
                    439: {
                    440:     hdr * hhdr;
                    441:     register word sz;          /* size of objects in current block     */
                    442:     register struct obj_kind * ok;
                    443:     register ptr_t * flh;
                    444:     register int kind;
                    445:
                    446:     hhdr = HDR(hbp);
                    447:     sz = hhdr -> hb_sz;
                    448:     hhdr -> hb_last_reclaimed = (unsigned short) GC_gc_no;
                    449:     kind = hhdr -> hb_obj_kind;
                    450:     ok = &GC_obj_kinds[kind];
                    451:     flh = &(ok -> ok_freelist[sz]);
                    452:     GC_write_hint(hbp);
                    453:
                    454:     if (ok -> ok_init) {
                    455:       switch(sz) {
                    456: #      ifndef SMALL_CONFIG
                    457:         case 1:
                    458:             *flh = GC_reclaim1(hbp, hhdr, *flh, abort_if_found);
                    459:             break;
                    460:         case 2:
                    461:             *flh = GC_reclaim_clear2(hbp, hhdr, *flh, abort_if_found);
                    462:             break;
                    463:         case 4:
                    464:             *flh = GC_reclaim_clear4(hbp, hhdr, *flh, abort_if_found);
                    465:             break;
                    466: #      endif
                    467:         default:
                    468:             *flh = GC_reclaim_clear(hbp, hhdr, sz, *flh, abort_if_found);
                    469:             break;
                    470:       }
                    471:     } else {
                    472:       switch(sz) {
                    473: #      ifndef SMALL_CONFIG
                    474:         case 1:
                    475:             *flh = GC_reclaim1(hbp, hhdr, *flh, abort_if_found);
                    476:             break;
                    477:         case 2:
                    478:             *flh = GC_reclaim_uninit2(hbp, hhdr, *flh, abort_if_found);
                    479:             break;
                    480:         case 4:
                    481:             *flh = GC_reclaim_uninit4(hbp, hhdr, *flh, abort_if_found);
                    482:             break;
                    483: #      endif
                    484:         default:
                    485:             *flh = GC_reclaim_uninit(hbp, hhdr, sz, *flh, abort_if_found);
                    486:             break;
                    487:       }
                    488:     }
                    489:     if (IS_UNCOLLECTABLE(kind)) GC_set_hdr_marks(hhdr);
                    490: }
                    491:
                    492: /*
                    493:  * Restore an unmarked large object or an entirely empty blocks of small objects
                    494:  * to the heap block free list.
                    495:  * Otherwise enqueue the block for later processing
                    496:  * by GC_reclaim_small_nonempty_block.
                    497:  * If abort_if_found is TRUE, then process any block immediately.
                    498:  */
                    499: void GC_reclaim_block(hbp, abort_if_found)
                    500: register struct hblk *hbp;     /* ptr to current heap block            */
                    501: word abort_if_found;           /* Abort if a reclaimable object is found */
                    502: {
                    503:     register hdr * hhdr;
                    504:     register word sz;          /* size of objects in current block     */
                    505:     register struct obj_kind * ok;
                    506:     struct hblk ** rlh;
                    507:
                    508:     hhdr = HDR(hbp);
                    509:     sz = hhdr -> hb_sz;
                    510:     ok = &GC_obj_kinds[hhdr -> hb_obj_kind];
                    511:
                    512:     if( sz > MAXOBJSZ ) {  /* 1 big object */
                    513:         if( !mark_bit_from_hdr(hhdr, HDR_WORDS) ) {
                    514:            FOUND_FREE(hbp, HDR_WORDS);
                    515: #          ifdef GATHERSTATS
                    516:                GC_mem_found += sz;
                    517: #          endif
                    518:            GC_freehblk(hbp);
                    519:        }
                    520:     } else {
                    521:         GC_bool empty = GC_block_empty(hhdr);
                    522:         if (abort_if_found) {
                    523:          GC_reclaim_small_nonempty_block(hbp, (int)abort_if_found);
                    524:         } else if (empty) {
                    525: #        ifdef GATHERSTATS
                    526:             GC_mem_found += BYTES_TO_WORDS(HBLKSIZE);
                    527: #        endif
                    528:           GC_freehblk(hbp);
                    529:         } else {
                    530:           /* group of smaller objects, enqueue the real work */
                    531:           rlh = &(ok -> ok_reclaim_list[sz]);
                    532:           hhdr -> hb_next = *rlh;
                    533:           *rlh = hbp;
                    534:         }
                    535:     }
                    536: }
                    537:
                    538: #if !defined(NO_DEBUGGING)
                    539: /* Routines to gather and print heap block info        */
                    540: /* intended for debugging.  Otherwise should be called */
                    541: /* with lock.                                          */
                    542: static size_t number_of_blocks;
                    543: static size_t total_bytes;
                    544:
                    545: /* Number of set bits in a word.  Not performance critical.    */
                    546: static int set_bits(n)
                    547: word n;
                    548: {
                    549:     register word m = n;
                    550:     register int result = 0;
                    551:
                    552:     while (m > 0) {
                    553:        if (m & 1) result++;
                    554:        m >>= 1;
                    555:     }
                    556:     return(result);
                    557: }
                    558:
                    559: /* Return the number of set mark bits in the given header      */
                    560: int GC_n_set_marks(hhdr)
                    561: hdr * hhdr;
                    562: {
                    563:     register int result = 0;
                    564:     register int i;
                    565:
                    566:     for (i = 0; i < MARK_BITS_SZ; i++) {
                    567:         result += set_bits(hhdr -> hb_marks[i]);
                    568:     }
                    569:     return(result);
                    570: }
                    571:
                    572: /*ARGSUSED*/
                    573: void GC_print_block_descr(h, dummy)
                    574: struct hblk *h;
                    575: word dummy;
                    576: {
                    577:     register hdr * hhdr = HDR(h);
                    578:     register size_t bytes = WORDS_TO_BYTES(hhdr -> hb_sz);
                    579:
                    580:     GC_printf3("(%lu:%lu,%lu)", (unsigned long)(hhdr -> hb_obj_kind),
                    581:                                (unsigned long)bytes,
                    582:                                (unsigned long)(GC_n_set_marks(hhdr)));
                    583:     bytes += HDR_BYTES + HBLKSIZE-1;
                    584:     bytes &= ~(HBLKSIZE-1);
                    585:     total_bytes += bytes;
                    586:     number_of_blocks++;
                    587: }
                    588:
                    589: void GC_print_block_list()
                    590: {
                    591:     GC_printf0("(kind(0=ptrfree,1=normal,2=unc.,3=stubborn):size_in_bytes, #_marks_set)\n");
                    592:     number_of_blocks = 0;
                    593:     total_bytes = 0;
                    594:     GC_apply_to_all_blocks(GC_print_block_descr, (word)0);
                    595:     GC_printf2("\nblocks = %lu, bytes = %lu\n",
                    596:               (unsigned long)number_of_blocks,
                    597:               (unsigned long)total_bytes);
                    598: }
                    599:
                    600: #endif /* NO_DEBUGGING */
                    601:
                    602: /*
                    603:  * Do the same thing on the entire heap, after first clearing small object
                    604:  * free lists (if we are not just looking for leaks).
                    605:  */
                    606: void GC_start_reclaim(abort_if_found)
                    607: int abort_if_found;            /* Abort if a GC_reclaimable object is found */
                    608: {
                    609:     int kind;
                    610:
                    611:     /* Clear reclaim- and free-lists */
                    612:       for (kind = 0; kind < GC_n_kinds; kind++) {
                    613:         register ptr_t *fop;
                    614:         register ptr_t *lim;
                    615:         register struct hblk ** rlp;
                    616:         register struct hblk ** rlim;
                    617:         register struct hblk ** rlist = GC_obj_kinds[kind].ok_reclaim_list;
                    618:
                    619:         if (rlist == 0) continue;      /* This kind not used.  */
                    620:         if (!abort_if_found) {
                    621:             lim = &(GC_obj_kinds[kind].ok_freelist[MAXOBJSZ+1]);
                    622:            for( fop = GC_obj_kinds[kind].ok_freelist; fop < lim; fop++ ) {
                    623:              *fop = 0;
                    624:            }
                    625:        } /* otherwise free list objects are marked,    */
                    626:          /* and its safe to leave them                 */
                    627:        rlim = rlist + MAXOBJSZ+1;
                    628:        for( rlp = rlist; rlp < rlim; rlp++ ) {
                    629:            *rlp = 0;
                    630:        }
                    631:       }
                    632:
                    633: #   ifdef PRINTBLOCKS
                    634:         GC_printf0("GC_reclaim: current block sizes:\n");
                    635:         GC_print_block_list();
                    636: #   endif
                    637:
                    638:   /* Go through all heap blocks (in hblklist) and reclaim unmarked objects */
                    639:   /* or enqueue the block for later processing.                                   */
                    640:     GC_apply_to_all_blocks(GC_reclaim_block, (word)abort_if_found);
                    641:
                    642: }
                    643:
                    644: /*
                    645:  * Sweep blocks of the indicated object size and kind until either the
                    646:  * appropriate free list is nonempty, or there are no more blocks to
                    647:  * sweep.
                    648:  */
                    649: void GC_continue_reclaim(sz, kind)
                    650: word sz;       /* words */
                    651: int kind;
                    652: {
                    653:     register hdr * hhdr;
                    654:     register struct hblk * hbp;
                    655:     register struct obj_kind * ok = &(GC_obj_kinds[kind]);
                    656:     struct hblk ** rlh = ok -> ok_reclaim_list;
                    657:     ptr_t *flh = &(ok -> ok_freelist[sz]);
                    658:
                    659:     if (rlh == 0) return;      /* No blocks of this kind.      */
                    660:     rlh += sz;
                    661:     while ((hbp = *rlh) != 0) {
                    662:         hhdr = HDR(hbp);
                    663:         *rlh = hhdr -> hb_next;
                    664:         GC_reclaim_small_nonempty_block(hbp, FALSE);
                    665:         if (*flh != 0) break;
                    666:     }
                    667: }
                    668:
                    669: /*
                    670:  * Reclaim all small blocks waiting to be reclaimed.
                    671:  * Abort and return FALSE when/if (*stop_func)() returns TRUE.
                    672:  * If this returns TRUE, then it's safe to restart the world
                    673:  * with incorrectly cleared mark bits.
                    674:  * If ignore_old is TRUE, then reclain only blocks that have been
                    675:  * recently reclaimed, and discard the rest.
                    676:  * Stop_func may be 0.
                    677:  */
                    678: GC_bool GC_reclaim_all(stop_func, ignore_old)
                    679: GC_stop_func stop_func;
                    680: GC_bool ignore_old;
                    681: {
                    682:     register word sz;
                    683:     register int kind;
                    684:     register hdr * hhdr;
                    685:     register struct hblk * hbp;
                    686:     register struct obj_kind * ok;
                    687:     struct hblk ** rlp;
                    688:     struct hblk ** rlh;
                    689: #   ifdef PRINTTIMES
                    690:        CLOCK_TYPE start_time;
                    691:        CLOCK_TYPE done_time;
                    692:
                    693:        GET_TIME(start_time);
                    694: #   endif
                    695:
                    696:     for (kind = 0; kind < GC_n_kinds; kind++) {
                    697:        ok = &(GC_obj_kinds[kind]);
                    698:        rlp = ok -> ok_reclaim_list;
                    699:        if (rlp == 0) continue;
                    700:        for (sz = 1; sz <= MAXOBJSZ; sz++) {
                    701:            rlh = rlp + sz;
                    702:            while ((hbp = *rlh) != 0) {
                    703:                if (stop_func != (GC_stop_func)0 && (*stop_func)()) {
                    704:                    return(FALSE);
                    705:                }
                    706:                hhdr = HDR(hbp);
                    707:                *rlh = hhdr -> hb_next;
                    708:                if (!ignore_old || hhdr -> hb_last_reclaimed == GC_gc_no - 1) {
                    709:                    /* It's likely we'll need it this time, too */
                    710:                    /* It's been touched recently, so this      */
                    711:                    /* shouldn't trigger paging.                */
                    712:                    GC_reclaim_small_nonempty_block(hbp, FALSE);
                    713:                }
                    714:             }
                    715:         }
                    716:     }
                    717: #   ifdef PRINTTIMES
                    718:        GET_TIME(done_time);
                    719:        GC_printf1("Disposing of reclaim lists took %lu msecs\n",
                    720:                   MS_TIME_DIFF(done_time,start_time));
                    721: #   endif
                    722:     return(TRUE);
                    723: }

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