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>