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

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

1.1     ! maekawa     1: /*
        !             2:  * Copyright (c) 1992-1994 by Xerox Corporation.  All rights reserved.
        !             3:  *
        !             4:  * THIS MATERIAL IS PROVIDED AS IS, WITH ABSOLUTELY NO WARRANTY EXPRESSED
        !             5:  * OR IMPLIED.  ANY USE IS AT YOUR OWN RISK.
        !             6:  *
        !             7:  * Permission is hereby granted to use or copy this program
        !             8:  * for any purpose,  provided the above notices are retained on all copies.
        !             9:  * Permission to modify the code and to distribute modified code is granted,
        !            10:  * provided the above notices are retained, and a notice that the code was
        !            11:  * modified is included with the above copyright notice.
        !            12:  */
        !            13: /* Boehm, March 29, 1995 12:51 pm PST */
        !            14: # ifdef CHECKSUMS
        !            15:
        !            16: # include "gc_priv.h"
        !            17:
        !            18: /* This is debugging code intended to verify the results of dirty bit  */
        !            19: /* computations. Works only in a single threaded environment.          */
        !            20: /* We assume that stubborn objects are changed only when they are      */
        !            21: /* enabled for writing.  (Certain kinds of writing are actually                */
        !            22: /* safe under other conditions.)                                       */
        !            23: # define NSUMS 2000
        !            24:
        !            25: # define OFFSET 0x10000
        !            26:
        !            27: typedef struct {
        !            28:        GC_bool new_valid;
        !            29:        word old_sum;
        !            30:        word new_sum;
        !            31:        struct hblk * block;    /* Block to which this refers + OFFSET  */
        !            32:                                /* to hide it from colector.            */
        !            33: } page_entry;
        !            34:
        !            35: page_entry GC_sums [NSUMS];
        !            36:
        !            37: word GC_checksum(h)
        !            38: struct hblk *h;
        !            39: {
        !            40:     register word *p = (word *)h;
        !            41:     register word *lim = (word *)(h+1);
        !            42:     register word result = 0;
        !            43:
        !            44:     while (p < lim) {
        !            45:         result += *p++;
        !            46:     }
        !            47:     return(result | 0x80000000 /* doesn't look like pointer */);
        !            48: }
        !            49:
        !            50: # ifdef STUBBORN_ALLOC
        !            51: /* Check whether a stubborn object from the given block appears on     */
        !            52: /* the appropriate free list.                                          */
        !            53: GC_bool GC_on_free_list(h)
        !            54: struct hblk *h;
        !            55: {
        !            56:     register hdr * hhdr = HDR(h);
        !            57:     register int sz = hhdr -> hb_sz;
        !            58:     ptr_t p;
        !            59:
        !            60:     if (sz > MAXOBJSZ) return(FALSE);
        !            61:     for (p = GC_sobjfreelist[sz]; p != 0; p = obj_link(p)) {
        !            62:         if (HBLKPTR(p) == h) return(TRUE);
        !            63:     }
        !            64:     return(FALSE);
        !            65: }
        !            66: # endif
        !            67:
        !            68: int GC_n_dirty_errors;
        !            69: int GC_n_changed_errors;
        !            70: int GC_n_clean;
        !            71: int GC_n_dirty;
        !            72:
        !            73: void GC_update_check_page(h, index)
        !            74: struct hblk *h;
        !            75: int index;
        !            76: {
        !            77:     page_entry *pe = GC_sums + index;
        !            78:     register hdr * hhdr = HDR(h);
        !            79:
        !            80:     if (pe -> block != 0 && pe -> block != h + OFFSET) ABORT("goofed");
        !            81:     pe -> old_sum = pe -> new_sum;
        !            82:     pe -> new_sum = GC_checksum(h);
        !            83: #   ifndef MSWIN32
        !            84:         if (pe -> new_sum != 0 && !GC_page_was_ever_dirty(h)) {
        !            85:             GC_printf1("GC_page_was_ever_dirty(0x%lx) is wrong\n",
        !            86:                       (unsigned long)h);
        !            87:         }
        !            88: #   endif
        !            89:     if (GC_page_was_dirty(h)) {
        !            90:        GC_n_dirty++;
        !            91:     } else {
        !            92:        GC_n_clean++;
        !            93:     }
        !            94:     if (pe -> new_valid && pe -> old_sum != pe -> new_sum) {
        !            95:        if (!GC_page_was_dirty(h) || !GC_page_was_ever_dirty(h)) {
        !            96:            /* Set breakpoint here */GC_n_dirty_errors++;
        !            97:        }
        !            98: #      ifdef STUBBORN_ALLOC
        !            99:          if (!IS_FORWARDING_ADDR_OR_NIL(hhdr)
        !           100:            && hhdr -> hb_map != GC_invalid_map
        !           101:            && hhdr -> hb_obj_kind == STUBBORN
        !           102:            && !GC_page_was_changed(h)
        !           103:            && !GC_on_free_list(h)) {
        !           104:            /* if GC_on_free_list(h) then reclaim may have touched it   */
        !           105:            /* without any allocations taking place.                    */
        !           106:            /* Set breakpoint here */GC_n_changed_errors++;
        !           107:          }
        !           108: #      endif
        !           109:     }
        !           110:     pe -> new_valid = TRUE;
        !           111:     pe -> block = h + OFFSET;
        !           112: }
        !           113:
        !           114: word GC_bytes_in_used_blocks;
        !           115:
        !           116: void GC_add_block(h, dummy)
        !           117: struct hblk *h;
        !           118: word dummy;
        !           119: {
        !           120:    register hdr * hhdr = HDR(h);
        !           121:    register bytes = WORDS_TO_BYTES(hhdr -> hb_sz);
        !           122:
        !           123:    bytes += HDR_BYTES + HBLKSIZE-1;
        !           124:    bytes &= ~(HBLKSIZE-1);
        !           125:    GC_bytes_in_used_blocks += bytes;
        !           126: }
        !           127:
        !           128: void GC_check_blocks()
        !           129: {
        !           130:     word bytes_in_free_blocks = 0;
        !           131:     struct hblk * h = GC_hblkfreelist;
        !           132:     hdr * hhdr = HDR(h);
        !           133:     word sz;
        !           134:
        !           135:     GC_bytes_in_used_blocks = 0;
        !           136:     GC_apply_to_all_blocks(GC_add_block, (word)0);
        !           137:     while (h != 0) {
        !           138:         sz = hhdr -> hb_sz;
        !           139:         bytes_in_free_blocks += sz;
        !           140:         h = hhdr -> hb_next;
        !           141:         hhdr = HDR(h);
        !           142:     }
        !           143:     GC_printf2("GC_bytes_in_used_blocks = %ld, bytes_in_free_blocks = %ld ",
        !           144:                GC_bytes_in_used_blocks, bytes_in_free_blocks);
        !           145:     GC_printf1("GC_heapsize = %ld\n", GC_heapsize);
        !           146:     if (GC_bytes_in_used_blocks + bytes_in_free_blocks != GC_heapsize) {
        !           147:        GC_printf0("LOST SOME BLOCKS!!\n");
        !           148:     }
        !           149: }
        !           150:
        !           151: /* Should be called immediately after GC_read_dirty and GC_read_changed. */
        !           152: void GC_check_dirty()
        !           153: {
        !           154:     register int index;
        !           155:     register unsigned i;
        !           156:     register struct hblk *h;
        !           157:     register ptr_t start;
        !           158:
        !           159:     GC_check_blocks();
        !           160:
        !           161:     GC_n_dirty_errors = 0;
        !           162:     GC_n_changed_errors = 0;
        !           163:     GC_n_clean = 0;
        !           164:     GC_n_dirty = 0;
        !           165:
        !           166:     index = 0;
        !           167:     for (i = 0; i < GC_n_heap_sects; i++) {
        !           168:        start = GC_heap_sects[i].hs_start;
        !           169:         for (h = (struct hblk *)start;
        !           170:              h < (struct hblk *)(start + GC_heap_sects[i].hs_bytes);
        !           171:              h++) {
        !           172:              GC_update_check_page(h, index);
        !           173:              index++;
        !           174:              if (index >= NSUMS) goto out;
        !           175:         }
        !           176:     }
        !           177: out:
        !           178:     GC_printf2("Checked %lu clean and %lu dirty pages\n",
        !           179:              (unsigned long) GC_n_clean, (unsigned long) GC_n_dirty);
        !           180:     if (GC_n_dirty_errors > 0) {
        !           181:         GC_printf1("Found %lu dirty bit errors\n",
        !           182:                   (unsigned long)GC_n_dirty_errors);
        !           183:     }
        !           184:     if (GC_n_changed_errors > 0) {
        !           185:        GC_printf1("Found %lu changed bit errors\n",
        !           186:                   (unsigned long)GC_n_changed_errors);
        !           187:        GC_printf0("These may be benign (provoked by nonpointer changes)\n");
        !           188: #      ifdef THREADS
        !           189:            GC_printf0(
        !           190:            "Also expect 1 per thread currently allocating a stubborn obj.\n");
        !           191: #      endif
        !           192:     }
        !           193: }
        !           194:
        !           195: # else
        !           196:
        !           197: extern int GC_quiet;
        !           198:        /* ANSI C doesn't allow translation units to be empty.  */
        !           199:        /* So we guarantee this one is nonempty.                */
        !           200:
        !           201: # endif /* CHECKSUMS */

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