step.o:
storage.o:
- 000002a0 D Abort_Names
- 00000015 B Bignum_Debug
- 00000004 D CONT_PRINT_EXPR_MESSAGE
- 00000000 D CONT_PRINT_RETURN_MESSAGE
- 0000000d B Cont_Debug
- 00000010 B Define_Debug
- 00000013 B Dump_Debug
- 000002e0 D Error_Names
- 00000008 B Eval_Debug
- 0000000a B File_Load_Debug
- 00000000 B Free_primitive
- 00000011 B GC_Debug
- 00000009 B Hex_Input_Debug
- 0000000c B Intern_Debug
- 0000000f B Lookup_Debug
- 00000010 D MAX_RETURN
- 00000016 B Per_File
- 0000000e B Primitive_Debug
- 0000000c D RESTORE_CONT_EXPR_MESSAGE
- 00000008 D RESTORE_CONT_RETURN_MESSAGE
- 0000000b B Reloc_Debug
- 00000020 D Return_Names
- 000003e0 D Term_Names
- 00000014 B Trace_On_Error
- 00000012 B Upgrade_Debug
- 00000004 C constant_alloc_next
- 00000004 C constant_end
- 00000004 C constant_start
- 00000190 C debug_circle
- 0000001c B debug_nslots
- 00000018 B debug_slotno
- 00000000 B ephemeron_array
- 00000004 B ephemeron_count
- 00000001 C ephemeron_request_hard_p
+ 000002a0 D Abort_Names read-only
+ 00000015 B Bignum_Debug gdb
+ 00000004 D CONT_PRINT_EXPR_MESSAGE read-only
+ 00000000 D CONT_PRINT_RETURN_MESSAGE read-only
+ 0000000d B Cont_Debug gdb
+ 00000010 B Define_Debug gdb
+ 00000013 B Dump_Debug gdb
+ 000002e0 D Error_Names read-only
+ 00000008 B Eval_Debug gdb
+ 0000000a B File_Load_Debug gdb
+ 00000000 B Free_primitive __thread
+ 00000011 B GC_Debug gdb
+ 00000009 B Hex_Input_Debug gdb
+ 0000000c B Intern_Debug gdb
+ 0000000f B Lookup_Debug gdb
+ 00000010 D MAX_RETURN read-only
+ 00000016 B Per_File gdb
+ 0000000e B Primitive_Debug gdb
+ 0000000c D RESTORE_CONT_EXPR_MESSAGE read-only
+ 00000008 D RESTORE_CONT_RETURN_MESSAGE read-only
+ 0000000b B Reloc_Debug gdb
+ 00000020 D Return_Names read-only
+ 000003e0 D Term_Names read-only
+ 00000014 B Trace_On_Error gdb
+ 00000012 B Upgrade_Debug gdb
+ 00000004 C constant_alloc_next gc-wait
+ 00000004 C constant_end gc-wait
+ 00000004 C constant_start gc-wait
+ 00000190 C debug_circle __thread
+ 0000001c B debug_nslots __thread
+ 00000018 B debug_slotno __thread
+ 00000000 B ephemeron_array gc-wait, locked
+ 00000004 B ephemeron_count gc-wait, locked
+ 00000001 C ephemeron_request_hard_p gc-wait, locked
00000004 C fixed_objects
- 000004e0 D fixed_objects_names
- 00000018 B gc_shared_space_needed
- 00000014 B gc_space_needed
- 00000004 C heap_reserved
- 00000004 B heap_start
- 0000000c B last_return_code
- 00000190 C local_circle
- 00000024 B local_nslots
- 00000020 B local_slotno
- 00000004 C memory_block_end
- 00000004 C memory_block_start
- 00000004 C n_ephemerons_requested
- 00000004 C p0_heap_start
- 00000004 C shared_heap_end
- 00000004 C shared_heap_free
- 00000004 C shared_heap_start
- 00000008 B stack_end
- 00000460 D term_messages
- 00000010 B trapping
- 000001a0 D type_names
+ 000004e0 D fixed_objects_names read-only
+ 00000018 B gc_shared_space_needed __thread
+ 00000014 B gc_space_needed __thread
+ 00000004 C heap_reserved gc-wait
+ 00000004 B heap_start __thread
+ 0000000c B last_return_code __thread
+ 00000190 C local_circle __thread
+ 00000024 B local_nslots __thread
+ 00000020 B local_slotno __thread
+ 00000004 C memory_block_end read-only, init by ALLOCATE_HEAP_SPACE
+ 00000004 C memory_block_start read-only, init by ALLOCATE_HEAP_SPACE
+ 00000004 C n_ephemerons_requested gc-wait, locked
+ 00000004 C p0_heap_start read-only, reset_allocator_parameters
+ 00000004 C shared_heap_end read-only, reset_allocator_parameters
+ 00000004 C shared_heap_free gc-wait, locked
+ 00000004 C shared_heap_start read-only, reset_allocator_parameters
+ 00000008 B stack_end __thread
+ 00000460 D term_messages read-only
+ 00000010 B trapping __thread
+ 000001a0 D type_names read-only
+
+ OK. Debugging flags apply to all processors(?). local_circle
+ is modified by EVAL_UCODE_HOOK and debug_circle is a copy made
+ in Do_Micro_Error. They and their associates are now thread-
+ local.
+
+ Prim_make_ephemeron reserves space in the shared heap. It is
+ already guaranteed not to conflict with GC, but multiple
+ threads allocating ephemerons simultaneously could be a
+ problem. Thus use of Prim_make_ephemeron is serialized by a
+ mutex.
+
+ fixed_objects is modified only by Prim_band_load or
+ Prim_set_fixed_objects_vector. Multiple threads hacking
+ fixed_objects should not happen.
+
+ heap_reserved is initialized (reset) by reset_allocator_
+ parameters and is modified only by Prim_garbage_collect and
+ Prim_primitive_purify.
string.o:
00000000 b external_strings
#ifdef ENABLE_SMP
int saved_processor_count;
unsigned long saved_processor_heap_size;
+static pthread_mutex_t mutex = MUTEX_INITIALIZER;
+# ifdef ENABLE_DEBUGGING_TOOLS
+static bool locked_p = false;
+# endif
#endif
static gc_tospace_allocator_t allocate_tospace;
SAFETY-MARGIN, which must be a non-negative integer. Finally, runs\n\
the primitive GC daemons before returning.")
{
+ unsigned long safety_margin = 0;
PRIMITIVE_HEADER (1);
canonicalize_primitive_context ();
if ((ARG_HEAP_RESERVED (1)) < (heap_end - heap_start))
{
- heap_reserved = (ARG_HEAP_RESERVED (1));
- heap_alloc_limit = (heap_end - heap_reserved);
+ safety_margin = (ARG_HEAP_RESERVED (1));
}
POP_PRIMITIVE_FRAME (1);
#endif /* ENABLE_SMP */
+ if (safety_margin)
+ {
+ heap_reserved = safety_margin;
+ heap_alloc_limit = (heap_end - heap_reserved);
+ }
+
initialize_weak_chain ();
ephemeron_count = 0;
void
guarantee_extra_ephemeron_space (unsigned long n)
{
+ LOCK();
ephemeron_count = n;
if (!ephemeron_array_big_enough_p (n))
{
assert (SHARED_HEAP_AVAILABLE_P (VECTOR_DATA + length));
ephemeron_array = make_ephemeron_vector (length);
}
+ UNLOCK();
}
\f
static void
{
n_ephemerons_requested = 1;
ephemeron_request_hard_p = true;
+ UNLOCK();
Primitive_GC (EPHEMERON_SIZE);
}
#endif
DEFINE_PRIMITIVE ("MAKE-EPHEMERON", Prim_make_ephemeron, 2, 2, 0)
{
PRIMITIVE_HEADER (2);
+ LOCK();
ephemeron_count += 1;
if (ephemeron_array_big_enough_p (ephemeron_count))
gc_if_needed_for_ephemeron (0);
(*Free++) = SHARP_F; /* list */
(*Free++) = SHARP_F; /* queue */
assert ((Free - addr) == EPHEMERON_SIZE);
+ UNLOCK();
PRIMITIVE_RETURN (MAKE_POINTER_OBJECT (TC_EPHEMERON, addr));
}
}