Blame reclaim.c

Packit d28291
/*
Packit d28291
 * Copyright 1988, 1989 Hans-J. Boehm, Alan J. Demers
Packit d28291
 * Copyright (c) 1991-1996 by Xerox Corporation.  All rights reserved.
Packit d28291
 * Copyright (c) 1996-1999 by Silicon Graphics.  All rights reserved.
Packit d28291
 * Copyright (c) 1999-2004 Hewlett-Packard Development Company, L.P.
Packit d28291
 *
Packit d28291
 * THIS MATERIAL IS PROVIDED AS IS, WITH ABSOLUTELY NO WARRANTY EXPRESSED
Packit d28291
 * OR IMPLIED.  ANY USE IS AT YOUR OWN RISK.
Packit d28291
 *
Packit d28291
 * Permission is hereby granted to use or copy this program
Packit d28291
 * for any purpose,  provided the above notices are retained on all copies.
Packit d28291
 * Permission to modify the code and to distribute modified code is granted,
Packit d28291
 * provided the above notices are retained, and a notice that the code was
Packit d28291
 * modified is included with the above copyright notice.
Packit d28291
 */
Packit d28291
Packit d28291
#include "private/gc_priv.h"
Packit d28291
Packit d28291
#ifdef ENABLE_DISCLAIM
Packit d28291
#  include "gc_disclaim.h"
Packit d28291
#endif
Packit d28291
Packit d28291
#include <stdio.h>
Packit d28291
Packit d28291
GC_INNER signed_word GC_bytes_found = 0;
Packit d28291
                        /* Number of bytes of memory reclaimed     */
Packit d28291
                        /* minus the number of bytes originally    */
Packit d28291
                        /* on free lists which we had to drop.     */
Packit d28291
Packit d28291
#if defined(PARALLEL_MARK)
Packit d28291
  GC_INNER signed_word GC_fl_builder_count = 0;
Packit d28291
        /* Number of threads currently building free lists without      */
Packit d28291
        /* holding GC lock.  It is not safe to collect if this is       */
Packit d28291
        /* nonzero.  Also, together with the mark lock, it is used as   */
Packit d28291
        /* a semaphore during marker threads startup.                   */
Packit d28291
#endif /* PARALLEL_MARK */
Packit d28291
Packit d28291
/* We defer printing of leaked objects until we're done with the GC     */
Packit d28291
/* cycle, since the routine for printing objects needs to run outside   */
Packit d28291
/* the collector, e.g. without the allocation lock.                     */
Packit d28291
#ifndef MAX_LEAKED
Packit d28291
# define MAX_LEAKED 40
Packit d28291
#endif
Packit d28291
STATIC ptr_t GC_leaked[MAX_LEAKED] = { NULL };
Packit d28291
STATIC unsigned GC_n_leaked = 0;
Packit d28291
Packit d28291
GC_INNER GC_bool GC_have_errors = FALSE;
Packit d28291
Packit d28291
#if !defined(EAGER_SWEEP) && defined(ENABLE_DISCLAIM)
Packit d28291
  STATIC void GC_reclaim_unconditionally_marked(void);
Packit d28291
#endif
Packit d28291
Packit d28291
GC_INLINE void GC_add_leaked(ptr_t leaked)
Packit d28291
{
Packit d28291
#  ifndef SHORT_DBG_HDRS
Packit d28291
     if (GC_findleak_delay_free && !GC_check_leaked(leaked))
Packit d28291
       return;
Packit d28291
#  endif
Packit d28291
Packit d28291
    GC_have_errors = TRUE;
Packit d28291
    if (GC_n_leaked < MAX_LEAKED) {
Packit d28291
      GC_leaked[GC_n_leaked++] = leaked;
Packit d28291
      /* Make sure it's not reclaimed this cycle */
Packit d28291
      GC_set_mark_bit(leaked);
Packit d28291
    }
Packit d28291
}
Packit d28291
Packit d28291
/* Print all objects on the list after printing any smashed objects.    */
Packit d28291
/* Clear both lists.  Called without the allocation lock held.          */
Packit d28291
GC_INNER void GC_print_all_errors(void)
Packit d28291
{
Packit d28291
    static GC_bool printing_errors = FALSE;
Packit d28291
    GC_bool have_errors;
Packit d28291
    unsigned i, n_leaked;
Packit d28291
    ptr_t leaked[MAX_LEAKED];
Packit d28291
    DCL_LOCK_STATE;
Packit d28291
Packit d28291
    LOCK();
Packit d28291
    if (printing_errors) {
Packit d28291
        UNLOCK();
Packit d28291
        return;
Packit d28291
    }
Packit d28291
    have_errors = GC_have_errors;
Packit d28291
    printing_errors = TRUE;
Packit d28291
    n_leaked = GC_n_leaked;
Packit d28291
    GC_ASSERT(n_leaked <= MAX_LEAKED);
Packit d28291
    BCOPY(GC_leaked, leaked, n_leaked * sizeof(ptr_t));
Packit d28291
    GC_n_leaked = 0;
Packit d28291
    BZERO(GC_leaked, n_leaked * sizeof(ptr_t));
Packit d28291
    UNLOCK();
Packit d28291
Packit d28291
    if (GC_debugging_started) {
Packit d28291
      GC_print_all_smashed();
Packit d28291
    } else {
Packit d28291
      have_errors = FALSE;
Packit d28291
    }
Packit d28291
Packit d28291
    if (n_leaked > 0) {
Packit d28291
        GC_err_printf("Found %u leaked objects:\n", n_leaked);
Packit d28291
        have_errors = TRUE;
Packit d28291
    }
Packit d28291
    for (i = 0; i < n_leaked; i++) {
Packit d28291
        ptr_t p = leaked[i];
Packit d28291
        GC_print_heap_obj(p);
Packit d28291
        GC_free(p);
Packit d28291
    }
Packit d28291
Packit d28291
    if (have_errors
Packit d28291
#       ifndef GC_ABORT_ON_LEAK
Packit d28291
          && GETENV("GC_ABORT_ON_LEAK") != NULL
Packit d28291
#       endif
Packit d28291
        ) {
Packit d28291
      ABORT("Leaked or smashed objects encountered");
Packit d28291
    }
Packit d28291
Packit d28291
    LOCK();
Packit d28291
    printing_errors = FALSE;
Packit d28291
    UNLOCK();
Packit d28291
}
Packit d28291
Packit d28291
Packit d28291
/*
Packit d28291
 * reclaim phase
Packit d28291
 *
Packit d28291
 */
Packit d28291
Packit d28291
/* Test whether a block is completely empty, i.e. contains no marked    */
Packit d28291
/* objects.  This does not require the block to be in physical memory.  */
Packit d28291
GC_INNER GC_bool GC_block_empty(hdr *hhdr)
Packit d28291
{
Packit d28291
    return (hhdr -> hb_n_marks == 0);
Packit d28291
}
Packit d28291
Packit d28291
STATIC GC_bool GC_block_nearly_full(hdr *hhdr)
Packit d28291
{
Packit d28291
    return (hhdr -> hb_n_marks > 7 * HBLK_OBJS(hhdr -> hb_sz)/8);
Packit d28291
}
Packit d28291
Packit d28291
/* FIXME: This should perhaps again be specialized for USE_MARK_BYTES   */
Packit d28291
/* and USE_MARK_BITS cases.                                             */
Packit d28291
Packit d28291
/*
Packit d28291
 * Restore unmarked small objects in h of size sz to the object
Packit d28291
 * free list.  Returns the new list.
Packit d28291
 * Clears unmarked objects.  Sz is in bytes.
Packit d28291
 */
Packit d28291
STATIC ptr_t GC_reclaim_clear(struct hblk *hbp, hdr *hhdr, size_t sz,
Packit d28291
                              ptr_t list, signed_word *count)
Packit d28291
{
Packit d28291
    word bit_no = 0;
Packit d28291
    word *p, *q, *plim;
Packit d28291
    signed_word n_bytes_found = 0;
Packit d28291
Packit d28291
    GC_ASSERT(hhdr == GC_find_header((ptr_t)hbp));
Packit d28291
    GC_ASSERT(sz == hhdr -> hb_sz);
Packit d28291
    GC_ASSERT((sz & (BYTES_PER_WORD-1)) == 0);
Packit d28291
    p = (word *)(hbp->hb_body);
Packit d28291
    plim = (word *)(hbp->hb_body + HBLKSIZE - sz);
Packit d28291
Packit d28291
    /* go through all words in block */
Packit d28291
        while ((word)p <= (word)plim) {
Packit d28291
            if (mark_bit_from_hdr(hhdr, bit_no)) {
Packit d28291
                p = (word *)((ptr_t)p + sz);
Packit d28291
            } else {
Packit d28291
                n_bytes_found += sz;
Packit d28291
                /* object is available - put on list */
Packit d28291
                    obj_link(p) = list;
Packit d28291
                    list = ((ptr_t)p);
Packit d28291
                /* Clear object, advance p to next object in the process */
Packit d28291
                    q = (word *)((ptr_t)p + sz);
Packit d28291
#                   ifdef USE_MARK_BYTES
Packit d28291
                      GC_ASSERT(!(sz & 1)
Packit d28291
                                && !((word)p & (2 * sizeof(word) - 1)));
Packit d28291
                      p[1] = 0;
Packit d28291
                      p += 2;
Packit d28291
                      while ((word)p < (word)q) {
Packit d28291
                        CLEAR_DOUBLE(p);
Packit d28291
                        p += 2;
Packit d28291
                      }
Packit d28291
#                   else
Packit d28291
                      p++; /* Skip link field */
Packit d28291
                      while ((word)p < (word)q) {
Packit d28291
                        *p++ = 0;
Packit d28291
                      }
Packit d28291
#                   endif
Packit d28291
            }
Packit d28291
            bit_no += MARK_BIT_OFFSET(sz);
Packit d28291
        }
Packit d28291
    *count += n_bytes_found;
Packit d28291
    return(list);
Packit d28291
}
Packit d28291
Packit d28291
/* The same thing, but don't clear objects: */
Packit d28291
STATIC ptr_t GC_reclaim_uninit(struct hblk *hbp, hdr *hhdr, size_t sz,
Packit d28291
                               ptr_t list, signed_word *count)
Packit d28291
{
Packit d28291
    word bit_no = 0;
Packit d28291
    word *p, *plim;
Packit d28291
    signed_word n_bytes_found = 0;
Packit d28291
Packit d28291
    GC_ASSERT(sz == hhdr -> hb_sz);
Packit d28291
    p = (word *)(hbp->hb_body);
Packit d28291
    plim = (word *)((ptr_t)hbp + HBLKSIZE - sz);
Packit d28291
Packit d28291
    /* go through all words in block */
Packit d28291
        while ((word)p <= (word)plim) {
Packit d28291
            if (!mark_bit_from_hdr(hhdr, bit_no)) {
Packit d28291
                n_bytes_found += sz;
Packit d28291
                /* object is available - put on list */
Packit d28291
                    obj_link(p) = list;
Packit d28291
                    list = ((ptr_t)p);
Packit d28291
            }
Packit d28291
            p = (word *)((ptr_t)p + sz);
Packit d28291
            bit_no += MARK_BIT_OFFSET(sz);
Packit d28291
        }
Packit d28291
    *count += n_bytes_found;
Packit d28291
    return(list);
Packit d28291
}
Packit d28291
Packit d28291
#ifdef ENABLE_DISCLAIM
Packit d28291
  /* Call reclaim notifier for block's kind on each unmarked object in  */
Packit d28291
  /* block, all within a pair of corresponding enter/leave callbacks.   */
Packit d28291
  STATIC ptr_t GC_disclaim_and_reclaim(struct hblk *hbp, hdr *hhdr, size_t sz,
Packit d28291
                                       ptr_t list, signed_word *count)
Packit d28291
  {
Packit d28291
    word bit_no = 0;
Packit d28291
    word *p, *q, *plim;
Packit d28291
    signed_word n_bytes_found = 0;
Packit d28291
    struct obj_kind *ok = &GC_obj_kinds[hhdr->hb_obj_kind];
Packit d28291
    int (GC_CALLBACK *disclaim)(void *) = ok->ok_disclaim_proc;
Packit d28291
Packit d28291
    GC_ASSERT(sz == hhdr -> hb_sz);
Packit d28291
    p = (word *)(hbp -> hb_body);
Packit d28291
    plim = (word *)((ptr_t)p + HBLKSIZE - sz);
Packit d28291
Packit d28291
    while ((word)p <= (word)plim) {
Packit d28291
        int marked = mark_bit_from_hdr(hhdr, bit_no);
Packit d28291
        if (!marked && (*disclaim)(p)) {
Packit d28291
            hhdr -> hb_n_marks++;
Packit d28291
            marked = 1;
Packit d28291
        }
Packit d28291
        if (marked)
Packit d28291
            p = (word *)((ptr_t)p + sz);
Packit d28291
        else {
Packit d28291
                n_bytes_found += sz;
Packit d28291
                /* object is available - put on list */
Packit d28291
                    obj_link(p) = list;
Packit d28291
                    list = ((ptr_t)p);
Packit d28291
                /* Clear object, advance p to next object in the process */
Packit d28291
                    q = (word *)((ptr_t)p + sz);
Packit d28291
#                   ifdef USE_MARK_BYTES
Packit d28291
                      GC_ASSERT((sz & 1) == 0);
Packit d28291
                      GC_ASSERT(((word)p & (2 * sizeof(word) - 1)) == 0);
Packit d28291
                      p[1] = 0;
Packit d28291
                      p += 2;
Packit d28291
                      while ((word)p < (word)q) {
Packit d28291
                        CLEAR_DOUBLE(p);
Packit d28291
                        p += 2;
Packit d28291
                      }
Packit d28291
#                   else
Packit d28291
                      p++; /* Skip link field */
Packit d28291
                      while ((word)p < (word)q) {
Packit d28291
                        *p++ = 0;
Packit d28291
                      }
Packit d28291
#                   endif
Packit d28291
        }
Packit d28291
        bit_no += MARK_BIT_OFFSET(sz);
Packit d28291
    }
Packit d28291
    *count += n_bytes_found;
Packit d28291
    return list;
Packit d28291
  }
Packit d28291
#endif /* ENABLE_DISCLAIM */
Packit d28291
Packit d28291
/* Don't really reclaim objects, just check for unmarked ones: */
Packit d28291
STATIC void GC_reclaim_check(struct hblk *hbp, hdr *hhdr, word sz)
Packit d28291
{
Packit d28291
    word bit_no;
Packit d28291
    ptr_t p, plim;
Packit d28291
    GC_ASSERT(sz == hhdr -> hb_sz);
Packit d28291
Packit d28291
    /* go through all words in block */
Packit d28291
    p = hbp->hb_body;
Packit d28291
    plim = p + HBLKSIZE - sz;
Packit d28291
    for (bit_no = 0; (word)p <= (word)plim;
Packit d28291
         p += sz, bit_no += MARK_BIT_OFFSET(sz)) {
Packit d28291
      if (!mark_bit_from_hdr(hhdr, bit_no)) {
Packit d28291
        GC_add_leaked(p);
Packit d28291
      }
Packit d28291
    }
Packit d28291
}
Packit d28291
Packit d28291
/*
Packit d28291
 * Generic procedure to rebuild a free list in hbp.
Packit d28291
 * Also called directly from GC_malloc_many.
Packit d28291
 * Sz is now in bytes.
Packit d28291
 */
Packit d28291
GC_INNER ptr_t GC_reclaim_generic(struct hblk * hbp, hdr *hhdr, size_t sz,
Packit d28291
                                  GC_bool init, ptr_t list,
Packit d28291
                                  signed_word *count)
Packit d28291
{
Packit d28291
    ptr_t result;
Packit d28291
Packit d28291
    GC_ASSERT(GC_find_header((ptr_t)hbp) == hhdr);
Packit d28291
#   ifndef GC_DISABLE_INCREMENTAL
Packit d28291
      GC_remove_protection(hbp, 1, (hhdr)->hb_descr == 0 /* Pointer-free? */);
Packit d28291
#   endif
Packit d28291
#   ifdef ENABLE_DISCLAIM
Packit d28291
      if ((hhdr -> hb_flags & HAS_DISCLAIM) != 0) {
Packit d28291
        result = GC_disclaim_and_reclaim(hbp, hhdr, sz, list, count);
Packit d28291
      } else
Packit d28291
#   endif
Packit d28291
    /* else */ if (init || GC_debugging_started) {
Packit d28291
      result = GC_reclaim_clear(hbp, hhdr, sz, list, count);
Packit d28291
    } else {
Packit d28291
      GC_ASSERT((hhdr)->hb_descr == 0 /* Pointer-free block */);
Packit d28291
      result = GC_reclaim_uninit(hbp, hhdr, sz, list, count);
Packit d28291
    }
Packit d28291
    if (IS_UNCOLLECTABLE(hhdr -> hb_obj_kind)) GC_set_hdr_marks(hhdr);
Packit d28291
    return result;
Packit d28291
}
Packit d28291
Packit d28291
/*
Packit d28291
 * Restore unmarked small objects in the block pointed to by hbp
Packit d28291
 * to the appropriate object free list.
Packit d28291
 * If entirely empty blocks are to be completely deallocated, then
Packit d28291
 * caller should perform that check.
Packit d28291
 */
Packit d28291
STATIC void GC_reclaim_small_nonempty_block(struct hblk *hbp,
Packit d28291
                                            GC_bool report_if_found)
Packit d28291
{
Packit d28291
    hdr *hhdr = HDR(hbp);
Packit d28291
    size_t sz = hhdr -> hb_sz;
Packit d28291
    struct obj_kind * ok = &GC_obj_kinds[hhdr -> hb_obj_kind];
Packit d28291
    void **flh = &(ok -> ok_freelist[BYTES_TO_GRANULES(sz)]);
Packit d28291
Packit d28291
    hhdr -> hb_last_reclaimed = (unsigned short) GC_gc_no;
Packit d28291
Packit d28291
    if (report_if_found) {
Packit d28291
        GC_reclaim_check(hbp, hhdr, sz);
Packit d28291
    } else {
Packit d28291
        *flh = GC_reclaim_generic(hbp, hhdr, sz, ok -> ok_init,
Packit d28291
                                  *flh, &GC_bytes_found);
Packit d28291
    }
Packit d28291
}
Packit d28291
Packit d28291
#ifdef ENABLE_DISCLAIM
Packit d28291
  STATIC void GC_disclaim_and_reclaim_or_free_small_block(struct hblk *hbp)
Packit d28291
  {
Packit d28291
    hdr *hhdr = HDR(hbp);
Packit d28291
    size_t sz = hhdr -> hb_sz;
Packit d28291
    struct obj_kind * ok = &GC_obj_kinds[hhdr -> hb_obj_kind];
Packit d28291
    void **flh = &(ok -> ok_freelist[BYTES_TO_GRANULES(sz)]);
Packit d28291
    void *flh_next;
Packit d28291
Packit d28291
    hhdr -> hb_last_reclaimed = (unsigned short) GC_gc_no;
Packit d28291
    flh_next = GC_reclaim_generic(hbp, hhdr, sz, ok -> ok_init,
Packit d28291
                                  *flh, &GC_bytes_found);
Packit d28291
    if (hhdr -> hb_n_marks)
Packit d28291
        *flh = flh_next;
Packit d28291
    else {
Packit d28291
        GC_bytes_found += HBLKSIZE;
Packit d28291
        GC_freehblk(hbp);
Packit d28291
    }
Packit d28291
  }
Packit d28291
#endif /* ENABLE_DISCLAIM */
Packit d28291
Packit d28291
/*
Packit d28291
 * Restore an unmarked large object or an entirely empty blocks of small objects
Packit d28291
 * to the heap block free list.
Packit d28291
 * Otherwise enqueue the block for later processing
Packit d28291
 * by GC_reclaim_small_nonempty_block.
Packit d28291
 * If report_if_found is TRUE, then process any block immediately, and
Packit d28291
 * simply report free objects; do not actually reclaim them.
Packit d28291
 */
Packit d28291
STATIC void GC_reclaim_block(struct hblk *hbp, word report_if_found)
Packit d28291
{
Packit d28291
    hdr * hhdr = HDR(hbp);
Packit d28291
    size_t sz = hhdr -> hb_sz;  /* size of objects in current block     */
Packit d28291
    struct obj_kind * ok = &GC_obj_kinds[hhdr -> hb_obj_kind];
Packit d28291
Packit d28291
    if( sz > MAXOBJBYTES ) {  /* 1 big object */
Packit d28291
        if( !mark_bit_from_hdr(hhdr, 0) ) {
Packit d28291
            if (report_if_found) {
Packit d28291
              GC_add_leaked((ptr_t)hbp);
Packit d28291
            } else {
Packit d28291
              size_t blocks;
Packit d28291
Packit d28291
#             ifdef ENABLE_DISCLAIM
Packit d28291
                if (EXPECT(hhdr->hb_flags & HAS_DISCLAIM, 0)) {
Packit d28291
                  struct obj_kind *ok = &GC_obj_kinds[hhdr->hb_obj_kind];
Packit d28291
                  if ((*ok->ok_disclaim_proc)(hbp)) {
Packit d28291
                    /* Not disclaimed => resurrect the object. */
Packit d28291
                    set_mark_bit_from_hdr(hhdr, 0);
Packit d28291
                    goto in_use;
Packit d28291
                  }
Packit d28291
                }
Packit d28291
#             endif
Packit d28291
              blocks = OBJ_SZ_TO_BLOCKS(sz);
Packit d28291
              if (blocks > 1) {
Packit d28291
                GC_large_allocd_bytes -= blocks * HBLKSIZE;
Packit d28291
              }
Packit d28291
              GC_bytes_found += sz;
Packit d28291
              GC_freehblk(hbp);
Packit d28291
            }
Packit d28291
        } else {
Packit d28291
#        ifdef ENABLE_DISCLAIM
Packit d28291
           in_use:
Packit d28291
#        endif
Packit d28291
            if (hhdr -> hb_descr != 0) {
Packit d28291
              GC_composite_in_use += sz;
Packit d28291
            } else {
Packit d28291
              GC_atomic_in_use += sz;
Packit d28291
            }
Packit d28291
        }
Packit d28291
    } else {
Packit d28291
        GC_bool empty = GC_block_empty(hhdr);
Packit d28291
#       ifdef PARALLEL_MARK
Packit d28291
          /* Count can be low or one too high because we sometimes      */
Packit d28291
          /* have to ignore decrements.  Objects can also potentially   */
Packit d28291
          /* be repeatedly marked by each marker.                       */
Packit d28291
          /* Here we assume two markers, but this is extremely          */
Packit d28291
          /* unlikely to fail spuriously with more.  And if it does, it */
Packit d28291
          /* should be looked at.                                       */
Packit d28291
          GC_ASSERT(hhdr -> hb_n_marks <= 2 * (HBLKSIZE/sz + 1) + 16);
Packit d28291
#       else
Packit d28291
          GC_ASSERT(sz * hhdr -> hb_n_marks <= HBLKSIZE);
Packit d28291
#       endif
Packit d28291
        if (report_if_found) {
Packit d28291
          GC_reclaim_small_nonempty_block(hbp, TRUE /* report_if_found */);
Packit d28291
        } else if (empty) {
Packit d28291
#       ifdef ENABLE_DISCLAIM
Packit d28291
          if ((hhdr -> hb_flags & HAS_DISCLAIM) != 0) {
Packit d28291
            GC_disclaim_and_reclaim_or_free_small_block(hbp);
Packit d28291
          } else
Packit d28291
#       endif
Packit d28291
          /* else */ {
Packit d28291
            GC_bytes_found += HBLKSIZE;
Packit d28291
            GC_freehblk(hbp);
Packit d28291
          }
Packit d28291
        } else if (GC_find_leak || !GC_block_nearly_full(hhdr)) {
Packit d28291
          /* group of smaller objects, enqueue the real work */
Packit d28291
          struct hblk **rlh = ok -> ok_reclaim_list;
Packit d28291
Packit d28291
          if (rlh != NULL) {
Packit d28291
            rlh += BYTES_TO_GRANULES(sz);
Packit d28291
            hhdr -> hb_next = *rlh;
Packit d28291
            *rlh = hbp;
Packit d28291
          }
Packit d28291
        } /* else not worth salvaging. */
Packit d28291
        /* We used to do the nearly_full check later, but we    */
Packit d28291
        /* already have the right cache context here.  Also     */
Packit d28291
        /* doing it here avoids some silly lock contention in   */
Packit d28291
        /* GC_malloc_many.                                      */
Packit d28291
Packit d28291
        if (hhdr -> hb_descr != 0) {
Packit d28291
          GC_composite_in_use += sz * hhdr -> hb_n_marks;
Packit d28291
        } else {
Packit d28291
          GC_atomic_in_use += sz * hhdr -> hb_n_marks;
Packit d28291
        }
Packit d28291
    }
Packit d28291
}
Packit d28291
Packit d28291
#if !defined(NO_DEBUGGING)
Packit d28291
/* Routines to gather and print heap block info         */
Packit d28291
/* intended for debugging.  Otherwise should be called  */
Packit d28291
/* with lock.                                           */
Packit d28291
Packit d28291
struct Print_stats
Packit d28291
{
Packit d28291
        size_t number_of_blocks;
Packit d28291
        size_t total_bytes;
Packit d28291
};
Packit d28291
Packit d28291
#ifdef USE_MARK_BYTES
Packit d28291
Packit d28291
/* Return the number of set mark bits in the given header.      */
Packit d28291
/* Remains externally visible as used by GNU GCJ currently.     */
Packit d28291
int GC_n_set_marks(hdr *hhdr)
Packit d28291
{
Packit d28291
    int result = 0;
Packit d28291
    int i;
Packit d28291
    size_t sz = hhdr -> hb_sz;
Packit d28291
    int offset = (int)MARK_BIT_OFFSET(sz);
Packit d28291
    int limit = (int)FINAL_MARK_BIT(sz);
Packit d28291
Packit d28291
    for (i = 0; i < limit; i += offset) {
Packit d28291
        result += hhdr -> hb_marks[i];
Packit d28291
    }
Packit d28291
    GC_ASSERT(hhdr -> hb_marks[limit]);
Packit d28291
    return(result);
Packit d28291
}
Packit d28291
Packit d28291
#else
Packit d28291
Packit d28291
/* Number of set bits in a word.  Not performance critical.     */
Packit d28291
static int set_bits(word n)
Packit d28291
{
Packit d28291
    word m = n;
Packit d28291
    int result = 0;
Packit d28291
Packit d28291
    while (m > 0) {
Packit d28291
        if (m & 1) result++;
Packit d28291
        m >>= 1;
Packit d28291
    }
Packit d28291
    return(result);
Packit d28291
}
Packit d28291
Packit d28291
int GC_n_set_marks(hdr *hhdr)
Packit d28291
{
Packit d28291
    int result = 0;
Packit d28291
    int i;
Packit d28291
    int n_mark_words;
Packit d28291
#   ifdef MARK_BIT_PER_OBJ
Packit d28291
      int n_objs = (int)HBLK_OBJS(hhdr -> hb_sz);
Packit d28291
Packit d28291
      if (0 == n_objs) n_objs = 1;
Packit d28291
      n_mark_words = divWORDSZ(n_objs + WORDSZ - 1);
Packit d28291
#   else /* MARK_BIT_PER_GRANULE */
Packit d28291
      n_mark_words = MARK_BITS_SZ;
Packit d28291
#   endif
Packit d28291
    for (i = 0; i < n_mark_words - 1; i++) {
Packit d28291
        result += set_bits(hhdr -> hb_marks[i]);
Packit d28291
    }
Packit d28291
#   ifdef MARK_BIT_PER_OBJ
Packit d28291
      result += set_bits((hhdr -> hb_marks[n_mark_words - 1])
Packit d28291
                         << (n_mark_words * WORDSZ - n_objs));
Packit d28291
#   else
Packit d28291
      result += set_bits(hhdr -> hb_marks[n_mark_words - 1]);
Packit d28291
#   endif
Packit d28291
    return(result - 1);
Packit d28291
}
Packit d28291
Packit d28291
#endif /* !USE_MARK_BYTES  */
Packit d28291
Packit d28291
STATIC void GC_print_block_descr(struct hblk *h,
Packit d28291
                                 word /* struct PrintStats */ raw_ps)
Packit d28291
{
Packit d28291
    hdr * hhdr = HDR(h);
Packit d28291
    size_t bytes = hhdr -> hb_sz;
Packit d28291
    struct Print_stats *ps;
Packit d28291
    unsigned n_marks = GC_n_set_marks(hhdr);
Packit d28291
    unsigned n_objs = (unsigned)HBLK_OBJS(bytes);
Packit d28291
Packit d28291
    if (0 == n_objs) n_objs = 1;
Packit d28291
    if (hhdr -> hb_n_marks != n_marks) {
Packit d28291
      GC_printf("%u,%u,%u!=%u,%u\n", hhdr->hb_obj_kind, (unsigned)bytes,
Packit d28291
                (unsigned)hhdr->hb_n_marks, n_marks, n_objs);
Packit d28291
    } else {
Packit d28291
      GC_printf("%u,%u,%u,%u\n", hhdr->hb_obj_kind, (unsigned)bytes,
Packit d28291
                n_marks, n_objs);
Packit d28291
    }
Packit d28291
Packit d28291
    ps = (struct Print_stats *)raw_ps;
Packit d28291
    ps->total_bytes += (bytes + (HBLKSIZE-1)) & ~(HBLKSIZE-1); /* round up */
Packit d28291
    ps->number_of_blocks++;
Packit d28291
}
Packit d28291
Packit d28291
void GC_print_block_list(void)
Packit d28291
{
Packit d28291
    struct Print_stats pstats;
Packit d28291
Packit d28291
    GC_printf("kind(0=ptrfree,1=normal,2=unc.),"
Packit d28291
              "size_in_bytes,#_marks_set,#objs\n");
Packit d28291
    pstats.number_of_blocks = 0;
Packit d28291
    pstats.total_bytes = 0;
Packit d28291
    GC_apply_to_all_blocks(GC_print_block_descr, (word)&pstats);
Packit d28291
    GC_printf("blocks= %lu, bytes= %lu\n",
Packit d28291
              (unsigned long)pstats.number_of_blocks,
Packit d28291
              (unsigned long)pstats.total_bytes);
Packit d28291
}
Packit d28291
Packit d28291
#include "gc_inline.h" /* for GC_print_free_list prototype */
Packit d28291
Packit d28291
/* Currently for debugger use only: */
Packit d28291
GC_API void GC_CALL GC_print_free_list(int kind, size_t sz_in_granules)
Packit d28291
{
Packit d28291
    ptr_t flh;
Packit d28291
    int n;
Packit d28291
Packit d28291
    GC_ASSERT(kind < MAXOBJKINDS);
Packit d28291
    GC_ASSERT(sz_in_granules <= MAXOBJGRANULES);
Packit d28291
    flh = GC_obj_kinds[kind].ok_freelist[sz_in_granules];
Packit d28291
    for (n = 0; flh; n++) {
Packit d28291
        struct hblk *block = HBLKPTR(flh);
Packit d28291
        GC_printf("Free object in heap block %p [%d]: %p\n",
Packit d28291
                  (void *)block, n, (void *)flh);
Packit d28291
        flh = obj_link(flh);
Packit d28291
    }
Packit d28291
}
Packit d28291
Packit d28291
#endif /* !NO_DEBUGGING */
Packit d28291
Packit d28291
/*
Packit d28291
 * Clear all obj_link pointers in the list of free objects *flp.
Packit d28291
 * Clear *flp.
Packit d28291
 * This must be done before dropping a list of free gcj-style objects,
Packit d28291
 * since may otherwise end up with dangling "descriptor" pointers.
Packit d28291
 * It may help for other pointer-containing objects.
Packit d28291
 */
Packit d28291
STATIC void GC_clear_fl_links(void **flp)
Packit d28291
{
Packit d28291
    void *next = *flp;
Packit d28291
Packit d28291
    while (0 != next) {
Packit d28291
       *flp = 0;
Packit d28291
       flp = &(obj_link(next));
Packit d28291
       next = *flp;
Packit d28291
    }
Packit d28291
}
Packit d28291
Packit d28291
/*
Packit d28291
 * Perform GC_reclaim_block on the entire heap, after first clearing
Packit d28291
 * small object free lists (if we are not just looking for leaks).
Packit d28291
 */
Packit d28291
GC_INNER void GC_start_reclaim(GC_bool report_if_found)
Packit d28291
{
Packit d28291
    unsigned kind;
Packit d28291
Packit d28291
#   if defined(PARALLEL_MARK)
Packit d28291
      GC_ASSERT(0 == GC_fl_builder_count);
Packit d28291
#   endif
Packit d28291
    /* Reset in use counters.  GC_reclaim_block recomputes them. */
Packit d28291
      GC_composite_in_use = 0;
Packit d28291
      GC_atomic_in_use = 0;
Packit d28291
    /* Clear reclaim- and free-lists */
Packit d28291
      for (kind = 0; kind < GC_n_kinds; kind++) {
Packit d28291
        struct hblk ** rlist = GC_obj_kinds[kind].ok_reclaim_list;
Packit d28291
        GC_bool should_clobber = (GC_obj_kinds[kind].ok_descriptor != 0);
Packit d28291
Packit d28291
        if (rlist == 0) continue;       /* This kind not used.  */
Packit d28291
        if (!report_if_found) {
Packit d28291
            void **fop;
Packit d28291
            void **lim = &(GC_obj_kinds[kind].ok_freelist[MAXOBJGRANULES+1]);
Packit d28291
Packit d28291
            for (fop = GC_obj_kinds[kind].ok_freelist;
Packit d28291
                 (word)fop < (word)lim; (*(word **)&fop)++) {
Packit d28291
              if (*fop != 0) {
Packit d28291
                if (should_clobber) {
Packit d28291
                  GC_clear_fl_links(fop);
Packit d28291
                } else {
Packit d28291
                  *fop = 0;
Packit d28291
                }
Packit d28291
              }
Packit d28291
            }
Packit d28291
        } /* otherwise free list objects are marked,    */
Packit d28291
          /* and its safe to leave them                 */
Packit d28291
        BZERO(rlist, (MAXOBJGRANULES + 1) * sizeof(void *));
Packit d28291
      }
Packit d28291
Packit d28291
Packit d28291
  /* Go through all heap blocks (in hblklist) and reclaim unmarked objects */
Packit d28291
  /* or enqueue the block for later processing.                            */
Packit d28291
    GC_apply_to_all_blocks(GC_reclaim_block, (word)report_if_found);
Packit d28291
Packit d28291
# ifdef EAGER_SWEEP
Packit d28291
    /* This is a very stupid thing to do.  We make it possible anyway,  */
Packit d28291
    /* so that you can convince yourself that it really is very stupid. */
Packit d28291
    GC_reclaim_all((GC_stop_func)0, FALSE);
Packit d28291
# elif defined(ENABLE_DISCLAIM)
Packit d28291
    /* However, make sure to clear reclaimable objects of kinds with    */
Packit d28291
    /* unconditional marking enabled before we do any significant       */
Packit d28291
    /* marking work.                                                    */
Packit d28291
    GC_reclaim_unconditionally_marked();
Packit d28291
# endif
Packit d28291
# if defined(PARALLEL_MARK)
Packit d28291
    GC_ASSERT(0 == GC_fl_builder_count);
Packit d28291
# endif
Packit d28291
Packit d28291
}
Packit d28291
Packit d28291
/*
Packit d28291
 * Sweep blocks of the indicated object size and kind until either the
Packit d28291
 * appropriate free list is nonempty, or there are no more blocks to
Packit d28291
 * sweep.
Packit d28291
 */
Packit d28291
GC_INNER void GC_continue_reclaim(size_t sz /* granules */, int kind)
Packit d28291
{
Packit d28291
    hdr * hhdr;
Packit d28291
    struct hblk * hbp;
Packit d28291
    struct obj_kind * ok = &(GC_obj_kinds[kind]);
Packit d28291
    struct hblk ** rlh = ok -> ok_reclaim_list;
Packit d28291
    void **flh = &(ok -> ok_freelist[sz]);
Packit d28291
Packit d28291
    if (rlh == 0) return;       /* No blocks of this kind.      */
Packit d28291
    rlh += sz;
Packit d28291
    while ((hbp = *rlh) != 0) {
Packit d28291
        hhdr = HDR(hbp);
Packit d28291
        *rlh = hhdr -> hb_next;
Packit d28291
        GC_reclaim_small_nonempty_block(hbp, FALSE);
Packit d28291
        if (*flh != 0) break;
Packit d28291
    }
Packit d28291
}
Packit d28291
Packit d28291
/*
Packit d28291
 * Reclaim all small blocks waiting to be reclaimed.
Packit d28291
 * Abort and return FALSE when/if (*stop_func)() returns TRUE.
Packit d28291
 * If this returns TRUE, then it's safe to restart the world
Packit d28291
 * with incorrectly cleared mark bits.
Packit d28291
 * If ignore_old is TRUE, then reclaim only blocks that have been
Packit d28291
 * recently reclaimed, and discard the rest.
Packit d28291
 * Stop_func may be 0.
Packit d28291
 */
Packit d28291
GC_INNER GC_bool GC_reclaim_all(GC_stop_func stop_func, GC_bool ignore_old)
Packit d28291
{
Packit d28291
    word sz;
Packit d28291
    unsigned kind;
Packit d28291
    hdr * hhdr;
Packit d28291
    struct hblk * hbp;
Packit d28291
    struct obj_kind * ok;
Packit d28291
    struct hblk ** rlp;
Packit d28291
    struct hblk ** rlh;
Packit d28291
#   ifndef SMALL_CONFIG
Packit d28291
      CLOCK_TYPE start_time = 0; /* initialized to prevent warning. */
Packit d28291
Packit d28291
      if (GC_print_stats == VERBOSE)
Packit d28291
        GET_TIME(start_time);
Packit d28291
#   endif
Packit d28291
Packit d28291
    for (kind = 0; kind < GC_n_kinds; kind++) {
Packit d28291
        ok = &(GC_obj_kinds[kind]);
Packit d28291
        rlp = ok -> ok_reclaim_list;
Packit d28291
        if (rlp == 0) continue;
Packit d28291
        for (sz = 1; sz <= MAXOBJGRANULES; sz++) {
Packit d28291
            rlh = rlp + sz;
Packit d28291
            while ((hbp = *rlh) != 0) {
Packit d28291
                if (stop_func != (GC_stop_func)0 && (*stop_func)()) {
Packit d28291
                    return(FALSE);
Packit d28291
                }
Packit d28291
                hhdr = HDR(hbp);
Packit d28291
                *rlh = hhdr -> hb_next;
Packit d28291
                if (!ignore_old || hhdr -> hb_last_reclaimed == GC_gc_no - 1) {
Packit d28291
                    /* It's likely we'll need it this time, too */
Packit d28291
                    /* It's been touched recently, so this      */
Packit d28291
                    /* shouldn't trigger paging.                */
Packit d28291
                    GC_reclaim_small_nonempty_block(hbp, FALSE);
Packit d28291
                }
Packit d28291
            }
Packit d28291
        }
Packit d28291
    }
Packit d28291
#   ifndef SMALL_CONFIG
Packit d28291
      if (GC_print_stats == VERBOSE) {
Packit d28291
        CLOCK_TYPE done_time;
Packit d28291
Packit d28291
        GET_TIME(done_time);
Packit d28291
        GC_verbose_log_printf("Disposing of reclaim lists took %lu msecs\n",
Packit d28291
                              MS_TIME_DIFF(done_time,start_time));
Packit d28291
      }
Packit d28291
#   endif
Packit d28291
    return(TRUE);
Packit d28291
}
Packit d28291
Packit d28291
#if !defined(EAGER_SWEEP) && defined(ENABLE_DISCLAIM)
Packit d28291
/* We do an eager sweep on heap blocks where unconditional marking has  */
Packit d28291
/* been enabled, so that any reclaimable objects have been reclaimed    */
Packit d28291
/* before we start marking.  This is a simplified GC_reclaim_all        */
Packit d28291
/* restricted to kinds where ok_mark_unconditionally is true.           */
Packit d28291
  STATIC void GC_reclaim_unconditionally_marked(void)
Packit d28291
  {
Packit d28291
    word sz;
Packit d28291
    unsigned kind;
Packit d28291
    hdr * hhdr;
Packit d28291
    struct hblk * hbp;
Packit d28291
    struct obj_kind * ok;
Packit d28291
    struct hblk ** rlp;
Packit d28291
    struct hblk ** rlh;
Packit d28291
Packit d28291
    for (kind = 0; kind < GC_n_kinds; kind++) {
Packit d28291
        ok = &(GC_obj_kinds[kind]);
Packit d28291
        if (!ok->ok_mark_unconditionally)
Packit d28291
          continue;
Packit d28291
        rlp = ok->ok_reclaim_list;
Packit d28291
        if (rlp == 0)
Packit d28291
          continue;
Packit d28291
        for (sz = 1; sz <= MAXOBJGRANULES; sz++) {
Packit d28291
            rlh = rlp + sz;
Packit d28291
            while ((hbp = *rlh) != 0) {
Packit d28291
                hhdr = HDR(hbp);
Packit d28291
                *rlh = hhdr->hb_next;
Packit d28291
                GC_reclaim_small_nonempty_block(hbp, FALSE);
Packit d28291
            }
Packit d28291
        }
Packit d28291
    }
Packit d28291
  }
Packit d28291
#endif /* !EAGER_SWEEP && ENABLE_DISCLAIM */
Packit d28291
Packit d28291
struct enumerate_reachable_s {
Packit d28291
  GC_reachable_object_proc proc;
Packit d28291
  void *client_data;
Packit d28291
};
Packit d28291
Packit d28291
STATIC void GC_do_enumerate_reachable_objects(struct hblk *hbp, word ped)
Packit d28291
{
Packit d28291
  struct hblkhdr *hhdr = HDR(hbp);
Packit d28291
  size_t sz = hhdr -> hb_sz;
Packit d28291
  size_t bit_no;
Packit d28291
  char *p, *plim;
Packit d28291
Packit d28291
  if (GC_block_empty(hhdr)) {
Packit d28291
    return;
Packit d28291
  }
Packit d28291
Packit d28291
  p = hbp->hb_body;
Packit d28291
  if (sz > MAXOBJBYTES) { /* one big object */
Packit d28291
    plim = p;
Packit d28291
  } else {
Packit d28291
    plim = hbp->hb_body + HBLKSIZE - sz;
Packit d28291
  }
Packit d28291
  /* Go through all words in block. */
Packit d28291
  for (bit_no = 0; p <= plim; bit_no += MARK_BIT_OFFSET(sz), p += sz) {
Packit d28291
    if (mark_bit_from_hdr(hhdr, bit_no)) {
Packit d28291
      ((struct enumerate_reachable_s *)ped)->proc(p, sz,
Packit d28291
                        ((struct enumerate_reachable_s *)ped)->client_data);
Packit d28291
    }
Packit d28291
  }
Packit d28291
}
Packit d28291
Packit d28291
GC_API void GC_CALL GC_enumerate_reachable_objects_inner(
Packit d28291
                                                GC_reachable_object_proc proc,
Packit d28291
                                                void *client_data)
Packit d28291
{
Packit d28291
  struct enumerate_reachable_s ed;
Packit d28291
Packit d28291
  GC_ASSERT(I_HOLD_LOCK());
Packit d28291
  ed.proc = proc;
Packit d28291
  ed.client_data = client_data;
Packit d28291
  GC_apply_to_all_blocks(GC_do_enumerate_reachable_objects, (word)&ed);
Packit d28291
}