home *** CD-ROM | disk | FTP | other *** search
/ OS/2 Shareware BBS: 10 Tools / 10-Tools.zip / mitsch75.zip / scheme-7_5_17-src.zip / scheme-7.5.17 / src / microcode / daemon.c < prev    next >
C/C++ Source or Header  |  1999-01-02  |  5KB  |  171 lines

  1. /* -*-C-*-
  2.  
  3. $Id: daemon.c,v 9.31 1999/01/02 06:11:34 cph Exp $
  4.  
  5. Copyright (c) 1987-1999 Massachusetts Institute of Technology
  6.  
  7. This program is free software; you can redistribute it and/or modify
  8. it under the terms of the GNU General Public License as published by
  9. the Free Software Foundation; either version 2 of the License, or (at
  10. your option) any later version.
  11.  
  12. This program is distributed in the hope that it will be useful, but
  13. WITHOUT ANY WARRANTY; without even the implied warranty of
  14. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
  15. General Public License for more details.
  16.  
  17. You should have received a copy of the GNU General Public License
  18. along with this program; if not, write to the Free Software
  19. Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  20. */
  21.  
  22. /* This file contains code for the Garbage Collection daemons.
  23.    There are currently two daemons, one for closing files which
  24.    have disappeared due to GC, the other for supporting object
  25.    hash tables where entries disappear when the corresponding
  26.    object is released due to GC.
  27.  
  28.    Both of these daemons should be written in Scheme, but since the
  29.    interpreter conses while executing Scheme programs, they are
  30.    unsafe.  The Scheme versions actually exist, but are commented out
  31.    of the appropriate runtime system sources. */
  32.  
  33. #include "scheme.h"
  34. #include "prims.h"
  35. #include "osio.h"
  36.  
  37. /* (CLOSE-LOST-OPEN-FILES file-list)
  38.    file-list is an assq-like list where the associations are weak
  39.    pairs rather than normal pairs.  This primitive destructively
  40.    removes those weak pairs whose cars are #F, and closes the
  41.    corresponding file descriptor contained in the cdrs. See io.scm in
  42.    the runtime system for a longer description. */
  43.  
  44. DEFINE_PRIMITIVE ("CLOSE-LOST-OPEN-FILES", Prim_close_lost_open_files, 1, 1, 0)
  45. {
  46.   PRIMITIVE_HEADER (1);
  47.   {
  48.     SCHEME_OBJECT file_list = (ARG_REF (1));
  49.     SCHEME_OBJECT * smash = (PAIR_CDR_LOC (file_list));
  50.     SCHEME_OBJECT cell = (*smash);
  51.     while (cell != EMPTY_LIST)
  52.       {
  53.     SCHEME_OBJECT weak_cell = (FAST_PAIR_CAR (cell));
  54.     if ((FAST_PAIR_CAR (weak_cell)) == SHARP_F)
  55.       {
  56.         OS_channel_close_noerror
  57.           (UNSIGNED_FIXNUM_TO_LONG (FAST_PAIR_CDR (weak_cell)));
  58.         cell = (FAST_PAIR_CDR (cell));
  59.         (*smash) = cell;
  60.       }
  61.     else
  62.       {
  63.         smash = (PAIR_CDR_LOC (cell));
  64.         cell = (*smash);
  65.       }
  66.       }
  67.   }
  68.   PRIMITIVE_RETURN (UNSPECIFIC);
  69. }
  70.  
  71. /* Utilities for the rehash daemon below */
  72.  
  73. /* This runs with GC locked, being part of a GC daemon.
  74.    It is also the case that the storage needed by this daemon is
  75.    available, since it was all reclaimed by the immediately preceeding
  76.    garbage collection, and at most that much is allocated now.
  77.    Therefore, there is no gc check here. */
  78.  
  79. static void
  80. DEFUN (rehash_pair, (pair, hash_table, table_size),
  81.        SCHEME_OBJECT pair AND SCHEME_OBJECT hash_table
  82.        AND long table_size)
  83. {
  84.   long object_datum, hash_address;
  85.   SCHEME_OBJECT * new_pair;
  86.  
  87.   object_datum = (OBJECT_DATUM (FAST_PAIR_CAR (pair)));
  88.   hash_address = (2 + (object_datum % table_size));
  89.   new_pair = Free;
  90.   *Free++ = (OBJECT_NEW_TYPE (TC_LIST, pair));
  91.   *Free++ = (FAST_MEMORY_REF (hash_table, hash_address));
  92.   FAST_MEMORY_SET (hash_table,
  93.            hash_address,
  94.            (MAKE_POINTER_OBJECT (TC_LIST, new_pair)));
  95.   return;
  96. }
  97.  
  98. static void
  99. DEFUN (rehash_bucket, (bucket, hash_table, table_size),
  100.        SCHEME_OBJECT * bucket AND SCHEME_OBJECT hash_table
  101.        AND long table_size)
  102. {
  103.   fast SCHEME_OBJECT weak_pair;
  104.  
  105.   while (*bucket != EMPTY_LIST)
  106.   {
  107.     weak_pair = (FAST_PAIR_CAR (*bucket));
  108.     if ((FAST_PAIR_CAR (weak_pair)) != SHARP_F)
  109.     {
  110.       rehash_pair (weak_pair, hash_table, table_size);
  111.     }
  112.     bucket = (PAIR_CDR_LOC (*bucket));
  113.   }
  114.   return;
  115. }
  116.  
  117. static void
  118. DEFUN (splice_and_rehash_bucket, (bucket, hash_table, table_size),
  119.        SCHEME_OBJECT * bucket AND SCHEME_OBJECT hash_table
  120.        AND long table_size)
  121. {
  122.   fast SCHEME_OBJECT weak_pair;
  123.  
  124.   while ((*bucket) != EMPTY_LIST)
  125.   {
  126.     weak_pair = (FAST_PAIR_CAR (*bucket));
  127.     if ((FAST_PAIR_CAR (weak_pair)) != SHARP_F)
  128.     {
  129.       rehash_pair (weak_pair, hash_table, table_size);
  130.       bucket = (PAIR_CDR_LOC (*bucket));
  131.     }
  132.     else
  133.       *bucket = (FAST_PAIR_CDR (*bucket));
  134.   }
  135.   return;
  136. }
  137.  
  138. /* (REHASH unhash-table hash-table)
  139.    Cleans up and recomputes hash-table from the valid information in
  140.    unhash-table after a garbage collection.
  141.    See hash.scm in the runtime system for a description. */
  142.  
  143. DEFINE_PRIMITIVE ("REHASH", Prim_rehash, 2, 2, 0)
  144. {
  145.   long table_size, counter;
  146.   SCHEME_OBJECT *bucket;
  147.   PRIMITIVE_HEADER (2);
  148.   table_size = (VECTOR_LENGTH (ARG_REF (1)));
  149.  
  150.   /* First cleanup the hash table */
  151.   counter = table_size;
  152.   bucket = (MEMORY_LOC ((ARG_REF (2)), 2));
  153.   while ((counter--) > 0)
  154.     (*bucket++) = EMPTY_LIST;
  155.  
  156.   /* Now rehash all the entries from the unhash table and maybe splice
  157.      the buckets. */
  158.   counter = table_size;
  159.   bucket = (MEMORY_LOC ((ARG_REF (1)), 1));
  160.   while ((counter--) > 0)
  161.     {
  162.       if ((FAST_PAIR_CAR (*bucket)) == SHARP_T)
  163.     splice_and_rehash_bucket
  164.       ((PAIR_CDR_LOC (*bucket)), (ARG_REF (2)), table_size);
  165.       else
  166.     rehash_bucket ((PAIR_CDR_LOC (*bucket)), (ARG_REF (2)), table_size);
  167.       bucket += 1;
  168.     }
  169.   PRIMITIVE_RETURN (UNSPECIFIC);
  170. }
  171.