Annotation of OpenXM_contrib/gc/blacklst.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, August 9, 1995 6:09 pm PDT */
15: # include "gc_priv.h"
16:
17: /*
18: * We maintain several hash tables of hblks that have had false hits.
19: * Each contains one bit per hash bucket; If any page in the bucket
20: * has had a false hit, we assume that all of them have.
21: * See the definition of page_hash_table in gc_private.h.
22: * False hits from the stack(s) are much more dangerous than false hits
23: * from elsewhere, since the former can pin a large object that spans the
24: * block, eventhough it does not start on the dangerous block.
25: */
26:
27: /*
28: * Externally callable routines are:
29:
30: * GC_add_to_black_list_normal
31: * GC_add_to_black_list_stack
32: * GC_promote_black_lists
33: * GC_is_black_listed
34: *
35: * All require that the allocator lock is held.
36: */
37:
38: /* Pointers to individual tables. We replace one table by another by */
39: /* switching these pointers. */
40: word * GC_old_normal_bl;
41: /* Nonstack false references seen at last full */
42: /* collection. */
43: word * GC_incomplete_normal_bl;
44: /* Nonstack false references seen since last */
45: /* full collection. */
46: word * GC_old_stack_bl;
47: word * GC_incomplete_stack_bl;
48:
49: word GC_total_stack_black_listed;
50:
51: word GC_black_list_spacing = MINHINCR*HBLKSIZE; /* Initial rough guess */
52:
53: void GC_clear_bl();
54:
55: void GC_default_print_heap_obj_proc(p)
56: ptr_t p;
57: {
58: ptr_t base = GC_base(p);
59:
60: GC_err_printf2("start: 0x%lx, appr. length: %ld", base, GC_size(base));
61: }
62:
63: void (*GC_print_heap_obj)(/* char * s, ptr_t p */) =
64: GC_default_print_heap_obj_proc;
65:
66: void GC_print_source_ptr(p)
67: ptr_t p;
68: {
69: ptr_t base = GC_base(p);
70: if (0 == base) {
71: if (0 == p) {
72: GC_err_printf0("in register");
73: } else {
74: GC_err_printf0("in root set");
75: }
76: } else {
77: GC_err_printf0("in object at ");
78: (*GC_print_heap_obj)(base);
79: }
80: }
81:
82: void GC_bl_init()
83: {
84: # ifndef ALL_INTERIOR_POINTERS
85: GC_old_normal_bl = (word *)
86: GC_scratch_alloc((word)(sizeof (page_hash_table)));
87: GC_incomplete_normal_bl = (word *)GC_scratch_alloc
88: ((word)(sizeof(page_hash_table)));
89: if (GC_old_normal_bl == 0 || GC_incomplete_normal_bl == 0) {
90: GC_err_printf0("Insufficient memory for black list\n");
91: EXIT();
92: }
93: GC_clear_bl(GC_old_normal_bl);
94: GC_clear_bl(GC_incomplete_normal_bl);
95: # endif
96: GC_old_stack_bl = (word *)GC_scratch_alloc((word)(sizeof(page_hash_table)));
97: GC_incomplete_stack_bl = (word *)GC_scratch_alloc
98: ((word)(sizeof(page_hash_table)));
99: if (GC_old_stack_bl == 0 || GC_incomplete_stack_bl == 0) {
100: GC_err_printf0("Insufficient memory for black list\n");
101: EXIT();
102: }
103: GC_clear_bl(GC_old_stack_bl);
104: GC_clear_bl(GC_incomplete_stack_bl);
105: }
106:
107: void GC_clear_bl(doomed)
108: word *doomed;
109: {
110: BZERO(doomed, sizeof(page_hash_table));
111: }
112:
113: void GC_copy_bl(old, new)
114: word *new, *old;
115: {
116: BCOPY(old, new, sizeof(page_hash_table));
117: }
118:
119: static word total_stack_black_listed();
120:
121: /* Signal the completion of a collection. Turn the incomplete black */
122: /* lists into new black lists, etc. */
123: void GC_promote_black_lists()
124: {
125: word * very_old_normal_bl = GC_old_normal_bl;
126: word * very_old_stack_bl = GC_old_stack_bl;
127:
128: GC_old_normal_bl = GC_incomplete_normal_bl;
129: GC_old_stack_bl = GC_incomplete_stack_bl;
130: # ifndef ALL_INTERIOR_POINTERS
131: GC_clear_bl(very_old_normal_bl);
132: # endif
133: GC_clear_bl(very_old_stack_bl);
134: GC_incomplete_normal_bl = very_old_normal_bl;
135: GC_incomplete_stack_bl = very_old_stack_bl;
136: GC_total_stack_black_listed = total_stack_black_listed();
137: # ifdef PRINTSTATS
138: GC_printf1("%ld bytes in heap blacklisted for interior pointers\n",
139: (unsigned long)GC_total_stack_black_listed);
140: # endif
141: if (GC_total_stack_black_listed != 0) {
142: GC_black_list_spacing =
143: HBLKSIZE*(GC_heapsize/GC_total_stack_black_listed);
144: }
145: if (GC_black_list_spacing < 3 * HBLKSIZE) {
146: GC_black_list_spacing = 3 * HBLKSIZE;
147: }
148: }
149:
150: void GC_unpromote_black_lists()
151: {
152: # ifndef ALL_INTERIOR_POINTERS
153: GC_copy_bl(GC_old_normal_bl, GC_incomplete_normal_bl);
154: # endif
155: GC_copy_bl(GC_old_stack_bl, GC_incomplete_stack_bl);
156: }
157:
158: # ifndef ALL_INTERIOR_POINTERS
159: /* P is not a valid pointer reference, but it falls inside */
160: /* the plausible heap bounds. */
161: /* Add it to the normal incomplete black list if appropriate. */
162: #ifdef PRINT_BLACK_LIST
163: void GC_add_to_black_list_normal(p, source)
164: ptr_t source;
165: #else
166: void GC_add_to_black_list_normal(p)
167: #endif
168: word p;
169: {
170: if (!(GC_modws_valid_offsets[p & (sizeof(word)-1)])) return;
171: {
172: register int index = PHT_HASH(p);
173:
174: if (HDR(p) == 0 || get_pht_entry_from_index(GC_old_normal_bl, index)) {
175: # ifdef PRINT_BLACK_LIST
176: if (!get_pht_entry_from_index(GC_incomplete_normal_bl, index)) {
177: GC_err_printf2(
178: "Black listing (normal) 0x%lx referenced from 0x%lx ",
179: (unsigned long) p, (unsigned long) source);
180: GC_print_source_ptr(source);
181: GC_err_puts("\n");
182: }
183: # endif
184: set_pht_entry_from_index(GC_incomplete_normal_bl, index);
185: } /* else this is probably just an interior pointer to an allocated */
186: /* object, and isn't worth black listing. */
187: }
188: }
189: # endif
190:
191: /* And the same for false pointers from the stack. */
192: #ifdef PRINT_BLACK_LIST
193: void GC_add_to_black_list_stack(p, source)
194: ptr_t source;
195: #else
196: void GC_add_to_black_list_stack(p)
197: #endif
198: word p;
199: {
200: register int index = PHT_HASH(p);
201:
202: if (HDR(p) == 0 || get_pht_entry_from_index(GC_old_stack_bl, index)) {
203: # ifdef PRINT_BLACK_LIST
204: if (!get_pht_entry_from_index(GC_incomplete_stack_bl, index)) {
205: GC_err_printf2(
206: "Black listing (stack) 0x%lx referenced from 0x%lx ",
207: (unsigned long)p, (unsigned long)source);
208: GC_print_source_ptr(source);
209: GC_err_puts("\n");
210: }
211: # endif
212: set_pht_entry_from_index(GC_incomplete_stack_bl, index);
213: }
214: }
215:
216: /*
217: * Is the block starting at h of size len bytes black listed? If so,
218: * return the address of the next plausible r such that (r, len) might not
219: * be black listed. (R may not actually be in the heap. We guarantee only
220: * that every smaller value of r after h is also black listed.)
221: * If (h,len) is not black listed, return 0.
222: * Knows about the structure of the black list hash tables.
223: */
224: struct hblk * GC_is_black_listed(h, len)
225: struct hblk * h;
226: word len;
227: {
228: register int index = PHT_HASH((word)h);
229: register word i;
230: word nblocks = divHBLKSZ(len);
231:
232: # ifndef ALL_INTERIOR_POINTERS
233: if (get_pht_entry_from_index(GC_old_normal_bl, index)
234: || get_pht_entry_from_index(GC_incomplete_normal_bl, index)) {
235: return(h+1);
236: }
237: # endif
238:
239: for (i = 0; ; ) {
240: if (GC_old_stack_bl[divWORDSZ(index)] == 0
241: && GC_incomplete_stack_bl[divWORDSZ(index)] == 0) {
242: /* An easy case */
243: i += WORDSZ - modWORDSZ(index);
244: } else {
245: if (get_pht_entry_from_index(GC_old_stack_bl, index)
246: || get_pht_entry_from_index(GC_incomplete_stack_bl, index)) {
247: return(h+i+1);
248: }
249: i++;
250: }
251: if (i >= nblocks) break;
252: index = PHT_HASH((word)(h+i));
253: }
254: return(0);
255: }
256:
257:
258: /* Return the number of blacklisted blocks in a given range. */
259: /* Used only for statistical purposes. */
260: /* Looks only at the GC_incomplete_stack_bl. */
261: word GC_number_stack_black_listed(start, endp1)
262: struct hblk *start, *endp1;
263: {
264: register struct hblk * h;
265: word result = 0;
266:
267: for (h = start; h < endp1; h++) {
268: register int index = PHT_HASH((word)h);
269:
270: if (get_pht_entry_from_index(GC_old_stack_bl, index)) result++;
271: }
272: return(result);
273: }
274:
275:
276: /* Return the total number of (stack) black-listed bytes. */
277: static word total_stack_black_listed()
278: {
279: register unsigned i;
280: word total = 0;
281:
282: for (i = 0; i < GC_n_heap_sects; i++) {
283: struct hblk * start = (struct hblk *) GC_heap_sects[i].hs_start;
284: word len = (word) GC_heap_sects[i].hs_bytes;
285: struct hblk * endp1 = start + len/HBLKSIZE;
286:
287: total += GC_number_stack_black_listed(start, endp1);
288: }
289: return(total * HBLKSIZE);
290: }
291:
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>