Blame win32_threads.c

Packit d28291
/*
Packit d28291
 * Copyright (c) 1994 by Xerox Corporation.  All rights reserved.
Packit d28291
 * Copyright (c) 1996 by Silicon Graphics.  All rights reserved.
Packit d28291
 * Copyright (c) 1998 by Fergus Henderson.  All rights reserved.
Packit d28291
 * Copyright (c) 2000-2008 by Hewlett-Packard Development Company.
Packit d28291
 * All rights reserved.
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
#if defined(GC_WIN32_THREADS)
Packit d28291
Packit d28291
#ifndef WIN32_LEAN_AND_MEAN
Packit d28291
# define WIN32_LEAN_AND_MEAN 1
Packit d28291
#endif
Packit d28291
#define NOSERVICE
Packit d28291
#include <windows.h>
Packit d28291
Packit d28291
#ifdef THREAD_LOCAL_ALLOC
Packit d28291
# include "private/thread_local_alloc.h"
Packit d28291
#endif /* THREAD_LOCAL_ALLOC */
Packit d28291
Packit d28291
/* Allocation lock declarations.        */
Packit d28291
#if !defined(USE_PTHREAD_LOCKS)
Packit d28291
  GC_INNER CRITICAL_SECTION GC_allocate_ml;
Packit d28291
# ifdef GC_ASSERTIONS
Packit d28291
    GC_INNER DWORD GC_lock_holder = NO_THREAD;
Packit d28291
        /* Thread id for current holder of allocation lock */
Packit d28291
# endif
Packit d28291
#else
Packit d28291
  GC_INNER pthread_mutex_t GC_allocate_ml = PTHREAD_MUTEX_INITIALIZER;
Packit d28291
# ifdef GC_ASSERTIONS
Packit d28291
    GC_INNER unsigned long GC_lock_holder = NO_THREAD;
Packit d28291
# endif
Packit d28291
#endif
Packit d28291
Packit d28291
#undef CreateThread
Packit d28291
#undef ExitThread
Packit d28291
#undef _beginthreadex
Packit d28291
#undef _endthreadex
Packit d28291
Packit d28291
#ifdef GC_PTHREADS
Packit d28291
# include <errno.h> /* for EAGAIN */
Packit d28291
Packit d28291
 /* Cygwin-specific forward decls */
Packit d28291
# undef pthread_create
Packit d28291
# undef pthread_join
Packit d28291
# undef pthread_detach
Packit d28291
Packit d28291
# ifndef GC_NO_PTHREAD_SIGMASK
Packit d28291
#   undef pthread_sigmask
Packit d28291
# endif
Packit d28291
Packit d28291
  STATIC void * GC_pthread_start(void * arg);
Packit d28291
  STATIC void GC_thread_exit_proc(void *arg);
Packit d28291
Packit d28291
# include <pthread.h>
Packit d28291
# ifdef CAN_CALL_ATFORK
Packit d28291
#   include <unistd.h>
Packit d28291
# endif
Packit d28291
Packit d28291
#elif !defined(MSWINCE)
Packit d28291
# include <process.h>  /* For _beginthreadex, _endthreadex */
Packit d28291
# include <errno.h> /* for errno, EAGAIN */
Packit d28291
Packit d28291
#endif /* !GC_PTHREADS && !MSWINCE */
Packit d28291
Packit d28291
/* DllMain-based thread registration is currently incompatible  */
Packit d28291
/* with thread-local allocation, pthreads and WinCE.            */
Packit d28291
#if (defined(GC_DLL) || defined(GC_INSIDE_DLL)) \
Packit d28291
        && !defined(GC_NO_THREADS_DISCOVERY) && !defined(MSWINCE) \
Packit d28291
        && !defined(THREAD_LOCAL_ALLOC) && !defined(GC_PTHREADS)
Packit d28291
# include "atomic_ops.h"
Packit d28291
Packit d28291
  /* This code operates in two distinct modes, depending on     */
Packit d28291
  /* the setting of GC_win32_dll_threads.                       */
Packit d28291
  /* If GC_win32_dll_threads is set, all threads in the process */
Packit d28291
  /* are implicitly registered with the GC by DllMain.          */
Packit d28291
  /* No explicit registration is required, and attempts at      */
Packit d28291
  /* explicit registration are ignored.  This mode is           */
Packit d28291
  /* very different from the Posix operation of the collector.  */
Packit d28291
  /* In this mode access to the thread table is lock-free.      */
Packit d28291
  /* Hence there is a static limit on the number of threads.    */
Packit d28291
Packit d28291
# ifdef GC_DISCOVER_TASK_THREADS
Packit d28291
    /* GC_DISCOVER_TASK_THREADS should be used if DllMain-based */
Packit d28291
    /* thread registration is required but it is impossible to  */
Packit d28291
    /* call GC_use_threads_discovery before other GC routines.  */
Packit d28291
#   define GC_win32_dll_threads TRUE
Packit d28291
# else
Packit d28291
    STATIC GC_bool GC_win32_dll_threads = FALSE;
Packit d28291
    /* GC_win32_dll_threads must be set (if needed) at the      */
Packit d28291
    /* application initialization time, i.e. before any         */
Packit d28291
    /* collector or thread calls.  We make it a "dynamic"       */
Packit d28291
    /* option only to avoid multiple library versions.          */
Packit d28291
# endif
Packit d28291
Packit d28291
#else
Packit d28291
  /* If GC_win32_dll_threads is FALSE (or the collector is      */
Packit d28291
  /* built without GC_DLL defined), things operate in a way     */
Packit d28291
  /* that is very similar to Posix platforms, and new threads   */
Packit d28291
  /* must be registered with the collector, e.g. by using       */
Packit d28291
  /* preprocessor-based interception of the thread primitives.  */
Packit d28291
  /* In this case, we use a real data structure for the thread  */
Packit d28291
  /* table.  Note that there is no equivalent of linker-based   */
Packit d28291
  /* call interception, since we don't have ELF-like            */
Packit d28291
  /* facilities.  The Windows analog appears to be "API         */
Packit d28291
  /* hooking", which really seems to be a standard way to       */
Packit d28291
  /* do minor binary rewriting (?).  I'd prefer not to have     */
Packit d28291
  /* the basic collector rely on such facilities, but an        */
Packit d28291
  /* optional package that intercepts thread calls this way     */
Packit d28291
  /* would probably be nice.                                    */
Packit d28291
# ifndef GC_NO_THREADS_DISCOVERY
Packit d28291
#   define GC_NO_THREADS_DISCOVERY
Packit d28291
# endif
Packit d28291
# define GC_win32_dll_threads FALSE
Packit d28291
# undef MAX_THREADS
Packit d28291
# define MAX_THREADS 1 /* dll_thread_table[] is always empty.   */
Packit d28291
#endif /* GC_NO_THREADS_DISCOVERY */
Packit d28291
Packit d28291
/* We have two versions of the thread table.  Which one */
Packit d28291
/* we us depends on whether or not GC_win32_dll_threads */
Packit d28291
/* is set.  Note that before initialization, we don't   */
Packit d28291
/* add any entries to either table, even if DllMain is  */
Packit d28291
/* called.  The main thread will be added on            */
Packit d28291
/* initialization.                                      */
Packit d28291
Packit d28291
/* The type of the first argument to InterlockedExchange.       */
Packit d28291
/* Documented to be LONG volatile *, but at least gcc likes     */
Packit d28291
/* this better.                                                 */
Packit d28291
typedef LONG * IE_t;
Packit d28291
Packit d28291
STATIC GC_bool GC_thr_initialized = FALSE;
Packit d28291
Packit d28291
#ifndef GC_ALWAYS_MULTITHREADED
Packit d28291
  GC_INNER GC_bool GC_need_to_lock = FALSE;
Packit d28291
#endif
Packit d28291
Packit d28291
static GC_bool parallel_initialized = FALSE;
Packit d28291
Packit d28291
/* GC_use_threads_discovery() is currently incompatible with pthreads   */
Packit d28291
/* and WinCE.  It might be possible to get DllMain-based thread         */
Packit d28291
/* registration to work with Cygwin, but if you try it then you are on  */
Packit d28291
/* your own.                                                            */
Packit d28291
GC_API void GC_CALL GC_use_threads_discovery(void)
Packit d28291
{
Packit d28291
# ifdef GC_NO_THREADS_DISCOVERY
Packit d28291
    ABORT("GC DllMain-based thread registration unsupported");
Packit d28291
# else
Packit d28291
    /* Turn on GC_win32_dll_threads. */
Packit d28291
    GC_ASSERT(!parallel_initialized);
Packit d28291
    /* Note that GC_use_threads_discovery is expected to be called by   */
Packit d28291
    /* the client application (not from DllMain) at start-up.           */
Packit d28291
#   ifndef GC_DISCOVER_TASK_THREADS
Packit d28291
      GC_win32_dll_threads = TRUE;
Packit d28291
#   endif
Packit d28291
    GC_init_parallel();
Packit d28291
# endif
Packit d28291
}
Packit d28291
Packit d28291
STATIC DWORD GC_main_thread = 0;
Packit d28291
Packit d28291
#define ADDR_LIMIT ((ptr_t)(word)-1)
Packit d28291
Packit d28291
struct GC_Thread_Rep {
Packit d28291
  union {
Packit d28291
#   ifndef GC_NO_THREADS_DISCOVERY
Packit d28291
      volatile AO_t in_use;
Packit d28291
                        /* Updated without lock.                */
Packit d28291
                        /* We assert that unused                */
Packit d28291
                        /* entries have invalid ids of          */
Packit d28291
                        /* zero and zero stack fields.          */
Packit d28291
                        /* Used only with GC_win32_dll_threads. */
Packit d28291
#   endif
Packit d28291
    struct GC_Thread_Rep * next;
Packit d28291
                        /* Hash table link without              */
Packit d28291
                        /* GC_win32_dll_threads.                */
Packit d28291
                        /* More recently allocated threads      */
Packit d28291
                        /* with a given pthread id come         */
Packit d28291
                        /* first.  (All but the first are       */
Packit d28291
                        /* guaranteed to be dead, but we may    */
Packit d28291
                        /* not yet have registered the join.)   */
Packit d28291
  } tm; /* table_management */
Packit d28291
  DWORD id;
Packit d28291
Packit d28291
# ifdef MSWINCE
Packit d28291
    /* According to MSDN specs for WinCE targets:                       */
Packit d28291
    /* - DuplicateHandle() is not applicable to thread handles; and     */
Packit d28291
    /* - the value returned by GetCurrentThreadId() could be used as    */
Packit d28291
    /* a "real" thread handle (for SuspendThread(), ResumeThread() and  */
Packit d28291
    /* GetThreadContext()).                                             */
Packit d28291
#   define THREAD_HANDLE(t) (HANDLE)(word)(t)->id
Packit d28291
# else
Packit d28291
    HANDLE handle;
Packit d28291
#   define THREAD_HANDLE(t) (t)->handle
Packit d28291
# endif
Packit d28291
Packit d28291
  ptr_t stack_base;     /* The cold end of the stack.   */
Packit d28291
                        /* 0 ==> entry not valid.       */
Packit d28291
                        /* !in_use ==> stack_base == 0  */
Packit d28291
  ptr_t last_stack_min; /* Last known minimum (hottest) address */
Packit d28291
                        /* in stack or ADDR_LIMIT if unset      */
Packit d28291
# ifdef IA64
Packit d28291
    ptr_t backing_store_end;
Packit d28291
    ptr_t backing_store_ptr;
Packit d28291
# endif
Packit d28291
Packit d28291
  ptr_t thread_blocked_sp;      /* Protected by GC lock.                */
Packit d28291
                                /* NULL value means thread unblocked.   */
Packit d28291
                                /* If set to non-NULL, thread will      */
Packit d28291
                                /* acquire GC lock before doing any     */
Packit d28291
                                /* pointer manipulations.  Thus it does */
Packit d28291
                                /* not need to stop this thread.        */
Packit d28291
Packit d28291
  struct GC_traced_stack_sect_s *traced_stack_sect;
Packit d28291
                                /* Points to the "stack section" data   */
Packit d28291
                                /* held in stack by the innermost       */
Packit d28291
                                /* GC_call_with_gc_active() of this     */
Packit d28291
                                /* thread.  May be NULL.                */
Packit d28291
Packit d28291
  unsigned short finalizer_skipped;
Packit d28291
  unsigned char finalizer_nested;
Packit d28291
                                /* Used by GC_check_finalizer_nested()  */
Packit d28291
                                /* to minimize the level of recursion   */
Packit d28291
                                /* when a client finalizer allocates    */
Packit d28291
                                /* memory (initially both are 0).       */
Packit d28291
Packit d28291
  unsigned char suspended; /* really of GC_bool type */
Packit d28291
Packit d28291
# ifdef GC_PTHREADS
Packit d28291
    unsigned char flags;        /* Protected by GC lock.                */
Packit d28291
#   define FINISHED 1           /* Thread has exited.                   */
Packit d28291
#   define DETACHED 2           /* Thread is intended to be detached.   */
Packit d28291
#   define KNOWN_FINISHED(t) (((t) -> flags) & FINISHED)
Packit d28291
    pthread_t pthread_id;
Packit d28291
    void *status;  /* hold exit value until join in case it's a pointer */
Packit d28291
# else
Packit d28291
#   define KNOWN_FINISHED(t) 0
Packit d28291
# endif
Packit d28291
Packit d28291
# ifdef THREAD_LOCAL_ALLOC
Packit d28291
    struct thread_local_freelists tlfs;
Packit d28291
# endif
Packit d28291
};
Packit d28291
Packit d28291
typedef struct GC_Thread_Rep * GC_thread;
Packit d28291
typedef volatile struct GC_Thread_Rep * GC_vthread;
Packit d28291
Packit d28291
#ifndef GC_NO_THREADS_DISCOVERY
Packit d28291
  /* We assumed that volatile ==> memory ordering, at least among       */
Packit d28291
  /* volatiles.  This code should consistently use atomic_ops.          */
Packit d28291
  STATIC volatile GC_bool GC_please_stop = FALSE;
Packit d28291
#elif defined(GC_ASSERTIONS)
Packit d28291
  STATIC GC_bool GC_please_stop = FALSE;
Packit d28291
#endif
Packit d28291
Packit d28291
/*
Packit d28291
 * We track thread attachments while the world is supposed to be stopped.
Packit d28291
 * Unfortunately, we can't stop them from starting, since blocking in
Packit d28291
 * DllMain seems to cause the world to deadlock.  Thus we have to recover
Packit d28291
 * If we notice this in the middle of marking.
Packit d28291
 */
Packit d28291
Packit d28291
#ifndef GC_NO_THREADS_DISCOVERY
Packit d28291
  STATIC volatile AO_t GC_attached_thread = FALSE;
Packit d28291
#endif
Packit d28291
Packit d28291
#if defined(WRAP_MARK_SOME) && !defined(GC_PTHREADS)
Packit d28291
  /* Return TRUE if an thread was attached since we last asked or */
Packit d28291
  /* since GC_attached_thread was explicitly reset.               */
Packit d28291
  GC_INNER GC_bool GC_started_thread_while_stopped(void)
Packit d28291
  {
Packit d28291
#   ifndef GC_NO_THREADS_DISCOVERY
Packit d28291
      if (GC_win32_dll_threads) {
Packit d28291
#       ifdef AO_HAVE_compare_and_swap_release
Packit d28291
          if (AO_compare_and_swap_release(&GC_attached_thread, TRUE,
Packit d28291
                                          FALSE /* stored */))
Packit d28291
            return TRUE;
Packit d28291
#       else
Packit d28291
          AO_nop_full(); /* Prior heap reads need to complete earlier. */
Packit d28291
          if (AO_load(&GC_attached_thread)) {
Packit d28291
            AO_store(&GC_attached_thread, FALSE);
Packit d28291
            return TRUE;
Packit d28291
          }
Packit d28291
#       endif
Packit d28291
      }
Packit d28291
#   endif
Packit d28291
    return FALSE;
Packit d28291
  }
Packit d28291
#endif /* WRAP_MARK_SOME */
Packit d28291
Packit d28291
/* Thread table used if GC_win32_dll_threads is set.    */
Packit d28291
/* This is a fixed size array.                          */
Packit d28291
/* Since we use runtime conditionals, both versions     */
Packit d28291
/* are always defined.                                  */
Packit d28291
# ifndef MAX_THREADS
Packit d28291
#   define MAX_THREADS 512
Packit d28291
# endif
Packit d28291
Packit d28291
/* Things may get quite slow for large numbers of threads,      */
Packit d28291
/* since we look them up with sequential search.                */
Packit d28291
volatile struct GC_Thread_Rep dll_thread_table[MAX_THREADS];
Packit d28291
Packit d28291
STATIC volatile LONG GC_max_thread_index = 0;
Packit d28291
                        /* Largest index in dll_thread_table    */
Packit d28291
                        /* that was ever used.                  */
Packit d28291
Packit d28291
/* And now the version used if GC_win32_dll_threads is not set. */
Packit d28291
/* This is a chained hash table, with much of the code borrowed */
Packit d28291
/* From the Posix implementation.                               */
Packit d28291
#ifndef THREAD_TABLE_SZ
Packit d28291
# define THREAD_TABLE_SZ 256    /* Power of 2 (for speed). */
Packit d28291
#endif
Packit d28291
#define THREAD_TABLE_INDEX(id) /* id is of DWORD type */ \
Packit d28291
                (int)((((id) >> 8) ^ (id)) % THREAD_TABLE_SZ)
Packit d28291
STATIC GC_thread GC_threads[THREAD_TABLE_SZ];
Packit d28291
Packit d28291
/* It may not be safe to allocate when we register the first thread.    */
Packit d28291
/* Thus we allocated one statically.  It does not contain any field we  */
Packit d28291
/* need to push ("next" and "status" fields are unused).                */
Packit d28291
static struct GC_Thread_Rep first_thread;
Packit d28291
static GC_bool first_thread_used = FALSE;
Packit d28291
Packit d28291
/* Add a thread to GC_threads.  We assume it wasn't already there.      */
Packit d28291
/* Caller holds allocation lock.                                        */
Packit d28291
/* Unlike the pthreads version, the id field is set by the caller.      */
Packit d28291
STATIC GC_thread GC_new_thread(DWORD id)
Packit d28291
{
Packit d28291
  int hv = THREAD_TABLE_INDEX(id);
Packit d28291
  GC_thread result;
Packit d28291
Packit d28291
  GC_ASSERT(I_HOLD_LOCK());
Packit d28291
  if (!EXPECT(first_thread_used, TRUE)) {
Packit d28291
    result = &first_thread;
Packit d28291
    first_thread_used = TRUE;
Packit d28291
  } else {
Packit d28291
    GC_ASSERT(!GC_win32_dll_threads);
Packit d28291
    result = (struct GC_Thread_Rep *)
Packit d28291
                GC_INTERNAL_MALLOC(sizeof(struct GC_Thread_Rep), NORMAL);
Packit d28291
    /* result can be NULL */
Packit d28291
    if (result == 0) return(0);
Packit d28291
  }
Packit d28291
  /* result -> id = id; Done by caller.       */
Packit d28291
  result -> tm.next = GC_threads[hv];
Packit d28291
  GC_threads[hv] = result;
Packit d28291
# ifdef GC_PTHREADS
Packit d28291
    GC_ASSERT(result -> flags == 0);
Packit d28291
# endif
Packit d28291
  GC_ASSERT(result -> thread_blocked_sp == NULL);
Packit d28291
  return(result);
Packit d28291
}
Packit d28291
Packit d28291
STATIC GC_bool GC_in_thread_creation = FALSE;
Packit d28291
                                /* Protected by allocation lock. */
Packit d28291
Packit d28291
GC_INLINE void GC_record_stack_base(GC_vthread me,
Packit d28291
                                    const struct GC_stack_base *sb)
Packit d28291
{
Packit d28291
  me -> stack_base = sb -> mem_base;
Packit d28291
# ifdef IA64
Packit d28291
    me -> backing_store_end = sb -> reg_base;
Packit d28291
# endif
Packit d28291
  if (me -> stack_base == NULL)
Packit d28291
    ABORT("Bad stack base in GC_register_my_thread");
Packit d28291
}
Packit d28291
Packit d28291
/* This may be called from DllMain, and hence operates under unusual    */
Packit d28291
/* constraints.  In particular, it must be lock-free if                 */
Packit d28291
/* GC_win32_dll_threads is set.  Always called from the thread being    */
Packit d28291
/* added.  If GC_win32_dll_threads is not set, we already hold the      */
Packit d28291
/* allocation lock except possibly during single-threaded startup code. */
Packit d28291
STATIC GC_thread GC_register_my_thread_inner(const struct GC_stack_base *sb,
Packit d28291
                                             DWORD thread_id)
Packit d28291
{
Packit d28291
  GC_vthread me;
Packit d28291
Packit d28291
  /* The following should be a no-op according to the win32     */
Packit d28291
  /* documentation.  There is empirical evidence that it        */
Packit d28291
  /* isn't.             - HB                                    */
Packit d28291
# if defined(MPROTECT_VDB)
Packit d28291
    if (GC_incremental
Packit d28291
#       ifdef GWW_VDB
Packit d28291
          && !GC_gww_dirty_init()
Packit d28291
#       endif
Packit d28291
        )
Packit d28291
      GC_set_write_fault_handler();
Packit d28291
# endif
Packit d28291
Packit d28291
# ifndef GC_NO_THREADS_DISCOVERY
Packit d28291
    if (GC_win32_dll_threads) {
Packit d28291
      int i;
Packit d28291
      /* It appears to be unsafe to acquire a lock here, since this     */
Packit d28291
      /* code is apparently not preemptible on some systems.            */
Packit d28291
      /* (This is based on complaints, not on Microsoft's official      */
Packit d28291
      /* documentation, which says this should perform "only simple     */
Packit d28291
      /* initialization tasks".)                                        */
Packit d28291
      /* Hence we make do with nonblocking synchronization.             */
Packit d28291
      /* It has been claimed that DllMain is really only executed with  */
Packit d28291
      /* a particular system lock held, and thus careful use of locking */
Packit d28291
      /* around code that doesn't call back into the system libraries   */
Packit d28291
      /* might be OK.  But this hasn't been tested across all win32     */
Packit d28291
      /* variants.                                                      */
Packit d28291
                  /* cast away volatile qualifier */
Packit d28291
      for (i = 0;
Packit d28291
           InterlockedExchange((void*)&dll_thread_table[i].tm.in_use, 1) != 0;
Packit d28291
           i++) {
Packit d28291
        /* Compare-and-swap would make this cleaner, but that's not     */
Packit d28291
        /* supported before Windows 98 and NT 4.0.  In Windows 2000,    */
Packit d28291
        /* InterlockedExchange is supposed to be replaced by            */
Packit d28291
        /* InterlockedExchangePointer, but that's not really what I     */
Packit d28291
        /* want here.                                                   */
Packit d28291
        /* FIXME: We should eventually declare Win95 dead and use AO_   */
Packit d28291
        /* primitives here.                                             */
Packit d28291
        if (i == MAX_THREADS - 1)
Packit d28291
          ABORT("Too many threads");
Packit d28291
      }
Packit d28291
      /* Update GC_max_thread_index if necessary.  The following is     */
Packit d28291
      /* safe, and unlike CompareExchange-based solutions seems to work */
Packit d28291
      /* on all Windows95 and later platforms.                          */
Packit d28291
      /* Unfortunately, GC_max_thread_index may be temporarily out of   */
Packit d28291
      /* bounds, so readers have to compensate.                         */
Packit d28291
      while (i > GC_max_thread_index) {
Packit d28291
        InterlockedIncrement((IE_t)&GC_max_thread_index);
Packit d28291
      }
Packit d28291
      if (GC_max_thread_index >= MAX_THREADS) {
Packit d28291
        /* We overshot due to simultaneous increments.  */
Packit d28291
        /* Setting it to MAX_THREADS-1 is always safe.  */
Packit d28291
        GC_max_thread_index = MAX_THREADS - 1;
Packit d28291
      }
Packit d28291
      me = dll_thread_table + i;
Packit d28291
    } else
Packit d28291
# endif
Packit d28291
  /* else */ /* Not using DllMain */ {
Packit d28291
    GC_ASSERT(I_HOLD_LOCK());
Packit d28291
    GC_in_thread_creation = TRUE; /* OK to collect from unknown thread. */
Packit d28291
    me = GC_new_thread(thread_id);
Packit d28291
    GC_in_thread_creation = FALSE;
Packit d28291
    if (me == 0)
Packit d28291
      ABORT("Failed to allocate memory for thread registering");
Packit d28291
  }
Packit d28291
# ifdef GC_PTHREADS
Packit d28291
    /* me can be NULL -> segfault */
Packit d28291
    me -> pthread_id = pthread_self();
Packit d28291
# endif
Packit d28291
# ifndef MSWINCE
Packit d28291
    /* GetCurrentThread() returns a pseudohandle (a const value).       */
Packit d28291
    if (!DuplicateHandle(GetCurrentProcess(), GetCurrentThread(),
Packit d28291
                        GetCurrentProcess(),
Packit d28291
                        (HANDLE*)&(me -> handle),
Packit d28291
                        0 /* dwDesiredAccess */, FALSE /* bInheritHandle */,
Packit d28291
                        DUPLICATE_SAME_ACCESS)) {
Packit d28291
        ABORT_ARG1("DuplicateHandle failed",
Packit d28291
                   ": errcode= 0x%X", (unsigned)GetLastError());
Packit d28291
    }
Packit d28291
# endif
Packit d28291
  me -> last_stack_min = ADDR_LIMIT;
Packit d28291
  GC_record_stack_base(me, sb);
Packit d28291
  /* Up until this point, GC_push_all_stacks considers this thread      */
Packit d28291
  /* invalid.                                                           */
Packit d28291
  /* Up until this point, this entry is viewed as reserved but invalid  */
Packit d28291
  /* by GC_delete_thread.                                               */
Packit d28291
  me -> id = thread_id;
Packit d28291
# if defined(THREAD_LOCAL_ALLOC)
Packit d28291
    GC_init_thread_local((GC_tlfs)(&(me->tlfs)));
Packit d28291
# endif
Packit d28291
# ifndef GC_NO_THREADS_DISCOVERY
Packit d28291
    if (GC_win32_dll_threads) {
Packit d28291
      if (GC_please_stop) {
Packit d28291
        AO_store(&GC_attached_thread, TRUE);
Packit d28291
        AO_nop_full(); /* Later updates must become visible after this. */
Packit d28291
      }
Packit d28291
      /* We'd like to wait here, but can't, since waiting in DllMain    */
Packit d28291
      /* provokes deadlocks.                                            */
Packit d28291
      /* Thus we force marking to be restarted instead.                 */
Packit d28291
    } else
Packit d28291
# endif
Packit d28291
  /* else */ {
Packit d28291
    GC_ASSERT(!GC_please_stop);
Packit d28291
        /* Otherwise both we and the thread stopping code would be      */
Packit d28291
        /* holding the allocation lock.                                 */
Packit d28291
  }
Packit d28291
  return (GC_thread)(me);
Packit d28291
}
Packit d28291
Packit d28291
/*
Packit d28291
 * GC_max_thread_index may temporarily be larger than MAX_THREADS.
Packit d28291
 * To avoid subscript errors, we check on access.
Packit d28291
 */
Packit d28291
GC_INLINE LONG GC_get_max_thread_index(void)
Packit d28291
{
Packit d28291
  LONG my_max = GC_max_thread_index;
Packit d28291
  if (my_max >= MAX_THREADS) return MAX_THREADS - 1;
Packit d28291
  return my_max;
Packit d28291
}
Packit d28291
Packit d28291
/* Return the GC_thread corresponding to a thread id.  May be called    */
Packit d28291
/* without a lock, but should be called in contexts in which the        */
Packit d28291
/* requested thread cannot be asynchronously deleted, e.g. from the     */
Packit d28291
/* thread itself.                                                       */
Packit d28291
/* This version assumes that either GC_win32_dll_threads is set, or     */
Packit d28291
/* we hold the allocator lock.                                          */
Packit d28291
/* Also used (for assertion checking only) from thread_local_alloc.c.   */
Packit d28291
STATIC GC_thread GC_lookup_thread_inner(DWORD thread_id)
Packit d28291
{
Packit d28291
# ifndef GC_NO_THREADS_DISCOVERY
Packit d28291
    if (GC_win32_dll_threads) {
Packit d28291
      int i;
Packit d28291
      LONG my_max = GC_get_max_thread_index();
Packit d28291
      for (i = 0; i <= my_max &&
Packit d28291
                  (!AO_load_acquire(&dll_thread_table[i].tm.in_use)
Packit d28291
                  || dll_thread_table[i].id != thread_id);
Packit d28291
           /* Must still be in_use, since nobody else can store our     */
Packit d28291
           /* thread_id.                                                */
Packit d28291
           i++) {
Packit d28291
        /* empty */
Packit d28291
      }
Packit d28291
      return i <= my_max ? (GC_thread)(dll_thread_table + i) : NULL;
Packit d28291
    } else
Packit d28291
# endif
Packit d28291
  /* else */ {
Packit d28291
    GC_thread p = GC_threads[THREAD_TABLE_INDEX(thread_id)];
Packit d28291
Packit d28291
    GC_ASSERT(I_HOLD_LOCK());
Packit d28291
    while (p != 0 && p -> id != thread_id) p = p -> tm.next;
Packit d28291
    return(p);
Packit d28291
  }
Packit d28291
}
Packit d28291
Packit d28291
#ifdef LINT2
Packit d28291
# define CHECK_LOOKUP_MY_THREAD(me) \
Packit d28291
        if (!(me)) ABORT("GC_lookup_thread_inner(GetCurrentThreadId) failed")
Packit d28291
#else
Packit d28291
# define CHECK_LOOKUP_MY_THREAD(me) /* empty */
Packit d28291
#endif
Packit d28291
Packit d28291
/* Called by GC_finalize() (in case of an allocation failure observed). */
Packit d28291
/* GC_reset_finalizer_nested() is the same as in pthread_support.c.     */
Packit d28291
GC_INNER void GC_reset_finalizer_nested(void)
Packit d28291
{
Packit d28291
  GC_thread me = GC_lookup_thread_inner(GetCurrentThreadId());
Packit d28291
  CHECK_LOOKUP_MY_THREAD(me);
Packit d28291
  me->finalizer_nested = 0;
Packit d28291
}
Packit d28291
Packit d28291
/* Checks and updates the thread-local level of finalizers recursion.   */
Packit d28291
/* Returns NULL if GC_invoke_finalizers() should not be called by the   */
Packit d28291
/* collector (to minimize the risk of a deep finalizers recursion),     */
Packit d28291
/* otherwise returns a pointer to the thread-local finalizer_nested.    */
Packit d28291
/* Called by GC_notify_or_invoke_finalizers() only (the lock is held).  */
Packit d28291
/* GC_check_finalizer_nested() is the same as in pthread_support.c.     */
Packit d28291
GC_INNER unsigned char *GC_check_finalizer_nested(void)
Packit d28291
{
Packit d28291
  GC_thread me = GC_lookup_thread_inner(GetCurrentThreadId());
Packit d28291
  unsigned nesting_level;
Packit d28291
  CHECK_LOOKUP_MY_THREAD(me);
Packit d28291
  nesting_level = me->finalizer_nested;
Packit d28291
  if (nesting_level) {
Packit d28291
    /* We are inside another GC_invoke_finalizers().            */
Packit d28291
    /* Skip some implicitly-called GC_invoke_finalizers()       */
Packit d28291
    /* depending on the nesting (recursion) level.              */
Packit d28291
    if (++me->finalizer_skipped < (1U << nesting_level)) return NULL;
Packit d28291
    me->finalizer_skipped = 0;
Packit d28291
  }
Packit d28291
  me->finalizer_nested = (unsigned char)(nesting_level + 1);
Packit d28291
  return &me->finalizer_nested;
Packit d28291
}
Packit d28291
Packit d28291
#if defined(GC_ASSERTIONS) && defined(THREAD_LOCAL_ALLOC)
Packit d28291
  /* This is called from thread-local GC_malloc(). */
Packit d28291
  GC_bool GC_is_thread_tsd_valid(void *tsd)
Packit d28291
  {
Packit d28291
    GC_thread me;
Packit d28291
    DCL_LOCK_STATE;
Packit d28291
Packit d28291
    LOCK();
Packit d28291
    me = GC_lookup_thread_inner(GetCurrentThreadId());
Packit d28291
    UNLOCK();
Packit d28291
    return (word)tsd >= (word)(&me->tlfs)
Packit d28291
            && (word)tsd < (word)(&me->tlfs) + sizeof(me->tlfs);
Packit d28291
  }
Packit d28291
#endif /* GC_ASSERTIONS && THREAD_LOCAL_ALLOC */
Packit d28291
Packit d28291
GC_API int GC_CALL GC_thread_is_registered(void)
Packit d28291
{
Packit d28291
    DWORD thread_id = GetCurrentThreadId();
Packit d28291
    GC_thread me;
Packit d28291
    DCL_LOCK_STATE;
Packit d28291
Packit d28291
    LOCK();
Packit d28291
    me = GC_lookup_thread_inner(thread_id);
Packit d28291
    UNLOCK();
Packit d28291
    return me != NULL;
Packit d28291
}
Packit d28291
Packit d28291
GC_API void GC_CALL GC_register_altstack(void *stack GC_ATTR_UNUSED,
Packit d28291
                                         GC_word stack_size GC_ATTR_UNUSED,
Packit d28291
                                         void *altstack GC_ATTR_UNUSED,
Packit d28291
                                         GC_word altstack_size GC_ATTR_UNUSED)
Packit d28291
{
Packit d28291
  /* TODO: Implement */
Packit d28291
}
Packit d28291
Packit d28291
/* Make sure thread descriptor t is not protected by the VDB            */
Packit d28291
/* implementation.                                                      */
Packit d28291
/* Used to prevent write faults when the world is (partially) stopped,  */
Packit d28291
/* since it may have been stopped with a system lock held, and that     */
Packit d28291
/* lock may be required for fault handling.                             */
Packit d28291
#if defined(MPROTECT_VDB)
Packit d28291
# define UNPROTECT_THREAD(t) \
Packit d28291
    if (!GC_win32_dll_threads && GC_incremental && t != &first_thread) { \
Packit d28291
      GC_ASSERT(SMALL_OBJ(GC_size(t))); \
Packit d28291
      GC_remove_protection(HBLKPTR(t), 1, FALSE); \
Packit d28291
    } else (void)0
Packit d28291
#else
Packit d28291
# define UNPROTECT_THREAD(t) (void)0
Packit d28291
#endif
Packit d28291
Packit d28291
#ifdef CYGWIN32
Packit d28291
# define GC_PTHREAD_PTRVAL(pthread_id) pthread_id
Packit d28291
#elif defined(GC_WIN32_PTHREADS) || defined(GC_PTHREADS_PARAMARK)
Packit d28291
# include <pthread.h> /* to check for winpthreads */
Packit d28291
# if defined(__WINPTHREADS_VERSION_MAJOR)
Packit d28291
#   define GC_PTHREAD_PTRVAL(pthread_id) pthread_id
Packit d28291
# else
Packit d28291
#   define GC_PTHREAD_PTRVAL(pthread_id) pthread_id.p
Packit d28291
# endif
Packit d28291
#endif
Packit d28291
Packit d28291
/* If a thread has been joined, but we have not yet             */
Packit d28291
/* been notified, then there may be more than one thread        */
Packit d28291
/* in the table with the same win32 id.                         */
Packit d28291
/* This is OK, but we need a way to delete a specific one.      */
Packit d28291
/* Assumes we hold the allocation lock unless                   */
Packit d28291
/* GC_win32_dll_threads is set.  Does not actually free         */
Packit d28291
/* GC_thread entry (only unlinks it).                           */
Packit d28291
/* If GC_win32_dll_threads is set it should be called from the  */
Packit d28291
/* thread being deleted.                                        */
Packit d28291
STATIC void GC_delete_gc_thread_no_free(GC_vthread t)
Packit d28291
{
Packit d28291
# ifndef MSWINCE
Packit d28291
    CloseHandle(t->handle);
Packit d28291
# endif
Packit d28291
# ifndef GC_NO_THREADS_DISCOVERY
Packit d28291
    if (GC_win32_dll_threads) {
Packit d28291
      /* This is intended to be lock-free.                              */
Packit d28291
      /* It is either called synchronously from the thread being        */
Packit d28291
      /* deleted, or by the joining thread.                             */
Packit d28291
      /* In this branch asynchronous changes to (*t) are possible.      */
Packit d28291
      /* It's not allowed to call GC_printf (and the friends) here,     */
Packit d28291
      /* see GC_stop_world() for the information.                       */
Packit d28291
      t -> stack_base = 0;
Packit d28291
      t -> id = 0;
Packit d28291
      AO_store_release(&t->tm.in_use, FALSE);
Packit d28291
    } else
Packit d28291
# endif
Packit d28291
  /* else */ {
Packit d28291
    DWORD id = ((GC_thread)t) -> id;
Packit d28291
                /* Cast away volatile qualifier, since we have lock.    */
Packit d28291
    int hv = THREAD_TABLE_INDEX(id);
Packit d28291
    register GC_thread p = GC_threads[hv];
Packit d28291
    register GC_thread prev = 0;
Packit d28291
Packit d28291
    GC_ASSERT(I_HOLD_LOCK());
Packit d28291
    while (p != (GC_thread)t) {
Packit d28291
      prev = p;
Packit d28291
      p = p -> tm.next;
Packit d28291
    }
Packit d28291
    if (prev == 0) {
Packit d28291
      GC_threads[hv] = p -> tm.next;
Packit d28291
    } else {
Packit d28291
      prev -> tm.next = p -> tm.next;
Packit d28291
    }
Packit d28291
  }
Packit d28291
}
Packit d28291
Packit d28291
/* Delete a thread from GC_threads.  We assume it is there.     */
Packit d28291
/* (The code intentionally traps if it wasn't.)  Assumes we     */
Packit d28291
/* hold the allocation lock unless GC_win32_dll_threads is set. */
Packit d28291
/* If GC_win32_dll_threads is set then it should be called from */
Packit d28291
/* the thread being deleted.  It is also safe to delete the     */
Packit d28291
/* main thread (unless GC_win32_dll_threads).                   */
Packit d28291
STATIC void GC_delete_thread(DWORD id)
Packit d28291
{
Packit d28291
  if (GC_win32_dll_threads) {
Packit d28291
    GC_vthread t = GC_lookup_thread_inner(id);
Packit d28291
Packit d28291
    if (0 == t) {
Packit d28291
      WARN("Removing nonexistent thread, id = %" WARN_PRIdPTR "\n", id);
Packit d28291
    } else {
Packit d28291
      GC_delete_gc_thread_no_free(t);
Packit d28291
    }
Packit d28291
  } else {
Packit d28291
    int hv = THREAD_TABLE_INDEX(id);
Packit d28291
    register GC_thread p = GC_threads[hv];
Packit d28291
    register GC_thread prev = 0;
Packit d28291
Packit d28291
    GC_ASSERT(I_HOLD_LOCK());
Packit d28291
    while (p -> id != id) {
Packit d28291
      prev = p;
Packit d28291
      p = p -> tm.next;
Packit d28291
    }
Packit d28291
#   ifndef MSWINCE
Packit d28291
      CloseHandle(p->handle);
Packit d28291
#   endif
Packit d28291
    if (prev == 0) {
Packit d28291
      GC_threads[hv] = p -> tm.next;
Packit d28291
    } else {
Packit d28291
      prev -> tm.next = p -> tm.next;
Packit d28291
    }
Packit d28291
    if (p != &first_thread) {
Packit d28291
      GC_INTERNAL_FREE(p);
Packit d28291
    }
Packit d28291
  }
Packit d28291
}
Packit d28291
Packit d28291
GC_API void GC_CALL GC_allow_register_threads(void)
Packit d28291
{
Packit d28291
  /* Check GC is initialized and the current thread is registered. */
Packit d28291
  GC_ASSERT(GC_lookup_thread_inner(GetCurrentThreadId()) != 0);
Packit d28291
# if !defined(GC_ALWAYS_MULTITHREADED) && !defined(PARALLEL_MARK) \
Packit d28291
     && !defined(GC_NO_THREADS_DISCOVERY)
Packit d28291
      /* GC_init() does not call GC_init_parallel() in this case.   */
Packit d28291
    parallel_initialized = TRUE;
Packit d28291
# endif
Packit d28291
  set_need_to_lock();
Packit d28291
}
Packit d28291
Packit d28291
GC_API int GC_CALL GC_register_my_thread(const struct GC_stack_base *sb)
Packit d28291
{
Packit d28291
  GC_thread me;
Packit d28291
  DWORD thread_id = GetCurrentThreadId();
Packit d28291
  DCL_LOCK_STATE;
Packit d28291
Packit d28291
  if (GC_need_to_lock == FALSE)
Packit d28291
    ABORT("Threads explicit registering is not previously enabled");
Packit d28291
Packit d28291
  /* We lock here, since we want to wait for an ongoing GC.     */
Packit d28291
  LOCK();
Packit d28291
  me = GC_lookup_thread_inner(thread_id);
Packit d28291
  if (me == 0) {
Packit d28291
#   ifdef GC_PTHREADS
Packit d28291
      me = GC_register_my_thread_inner(sb, thread_id);
Packit d28291
      me -> flags |= DETACHED;
Packit d28291
          /* Treat as detached, since we do not need to worry about     */
Packit d28291
          /* pointer results.                                           */
Packit d28291
#   else
Packit d28291
      GC_register_my_thread_inner(sb, thread_id);
Packit d28291
#   endif
Packit d28291
    UNLOCK();
Packit d28291
    return GC_SUCCESS;
Packit d28291
  } else
Packit d28291
#   ifdef GC_PTHREADS
Packit d28291
      /* else */ if ((me -> flags & FINISHED) != 0) {
Packit d28291
        GC_record_stack_base(me, sb);
Packit d28291
        me -> flags &= ~FINISHED; /* but not DETACHED */
Packit d28291
#       ifdef THREAD_LOCAL_ALLOC
Packit d28291
          GC_init_thread_local((GC_tlfs)(&me->tlfs));
Packit d28291
#       endif
Packit d28291
        UNLOCK();
Packit d28291
        return GC_SUCCESS;
Packit d28291
      } else
Packit d28291
#   endif
Packit d28291
  /* else */ {
Packit d28291
    UNLOCK();
Packit d28291
    return GC_DUPLICATE;
Packit d28291
  }
Packit d28291
}
Packit d28291
Packit d28291
/* Similar to that in pthread_support.c.        */
Packit d28291
STATIC void GC_wait_for_gc_completion(GC_bool wait_for_all)
Packit d28291
{
Packit d28291
  GC_ASSERT(I_HOLD_LOCK());
Packit d28291
  if (GC_incremental && GC_collection_in_progress()) {
Packit d28291
    word old_gc_no = GC_gc_no;
Packit d28291
Packit d28291
    /* Make sure that no part of our stack is still on the mark stack,  */
Packit d28291
    /* since it's about to be unmapped.                                 */
Packit d28291
    do {
Packit d28291
      ENTER_GC();
Packit d28291
      GC_in_thread_creation = TRUE;
Packit d28291
      GC_collect_a_little_inner(1);
Packit d28291
      GC_in_thread_creation = FALSE;
Packit d28291
      EXIT_GC();
Packit d28291
Packit d28291
      UNLOCK();
Packit d28291
      Sleep(0); /* yield */
Packit d28291
      LOCK();
Packit d28291
    } while (GC_incremental && GC_collection_in_progress()
Packit d28291
             && (wait_for_all || old_gc_no == GC_gc_no));
Packit d28291
  }
Packit d28291
}
Packit d28291
Packit d28291
GC_API int GC_CALL GC_unregister_my_thread(void)
Packit d28291
{
Packit d28291
  DCL_LOCK_STATE;
Packit d28291
Packit d28291
# ifdef DEBUG_THREADS
Packit d28291
    GC_log_printf("Unregistering thread 0x%lx\n", (long)GetCurrentThreadId());
Packit d28291
# endif
Packit d28291
Packit d28291
  if (GC_win32_dll_threads) {
Packit d28291
#   if defined(THREAD_LOCAL_ALLOC)
Packit d28291
      /* Can't happen: see GC_use_threads_discovery(). */
Packit d28291
      GC_ASSERT(FALSE);
Packit d28291
#   else
Packit d28291
      /* FIXME: Should we just ignore this? */
Packit d28291
      GC_delete_thread(GetCurrentThreadId());
Packit d28291
#   endif
Packit d28291
  } else {
Packit d28291
#   if defined(THREAD_LOCAL_ALLOC) || defined(GC_PTHREADS)
Packit d28291
      GC_thread me;
Packit d28291
#   endif
Packit d28291
    DWORD thread_id = GetCurrentThreadId();
Packit d28291
Packit d28291
    LOCK();
Packit d28291
    GC_wait_for_gc_completion(FALSE);
Packit d28291
#   if defined(THREAD_LOCAL_ALLOC) || defined(GC_PTHREADS)
Packit d28291
      me = GC_lookup_thread_inner(thread_id);
Packit d28291
      CHECK_LOOKUP_MY_THREAD(me);
Packit d28291
      GC_ASSERT(!KNOWN_FINISHED(me));
Packit d28291
#   endif
Packit d28291
#   if defined(THREAD_LOCAL_ALLOC)
Packit d28291
      GC_ASSERT(GC_getspecific(GC_thread_key) == &me->tlfs);
Packit d28291
      GC_destroy_thread_local(&(me->tlfs));
Packit d28291
#   endif
Packit d28291
#   ifdef GC_PTHREADS
Packit d28291
      if ((me -> flags & DETACHED) == 0) {
Packit d28291
        me -> flags |= FINISHED;
Packit d28291
      } else
Packit d28291
#   endif
Packit d28291
    /* else */ {
Packit d28291
      GC_delete_thread(thread_id);
Packit d28291
    }
Packit d28291
#   if defined(THREAD_LOCAL_ALLOC)
Packit d28291
      /* It is required to call remove_specific defined in specific.c. */
Packit d28291
      GC_remove_specific(GC_thread_key);
Packit d28291
#   endif
Packit d28291
    UNLOCK();
Packit d28291
  }
Packit d28291
  return GC_SUCCESS;
Packit d28291
}
Packit d28291
Packit d28291
/* Wrapper for functions that are likely to block for an appreciable    */
Packit d28291
/* length of time.                                                      */
Packit d28291
Packit d28291
/* GC_do_blocking_inner() is nearly the same as in pthread_support.c    */
Packit d28291
GC_INNER void GC_do_blocking_inner(ptr_t data, void * context GC_ATTR_UNUSED)
Packit d28291
{
Packit d28291
  struct blocking_data * d = (struct blocking_data *) data;
Packit d28291
  DWORD thread_id = GetCurrentThreadId();
Packit d28291
  GC_thread me;
Packit d28291
# ifdef IA64
Packit d28291
    ptr_t stack_ptr = GC_save_regs_in_stack();
Packit d28291
# endif
Packit d28291
  DCL_LOCK_STATE;
Packit d28291
Packit d28291
  LOCK();
Packit d28291
  me = GC_lookup_thread_inner(thread_id);
Packit d28291
  CHECK_LOOKUP_MY_THREAD(me);
Packit d28291
  GC_ASSERT(me -> thread_blocked_sp == NULL);
Packit d28291
# ifdef IA64
Packit d28291
    me -> backing_store_ptr = stack_ptr;
Packit d28291
# endif
Packit d28291
  me -> thread_blocked_sp = (ptr_t) &d; /* save approx. sp */
Packit d28291
  /* Save context here if we want to support precise stack marking */
Packit d28291
  UNLOCK();
Packit d28291
  d -> client_data = (d -> fn)(d -> client_data);
Packit d28291
  LOCK();   /* This will block if the world is stopped. */
Packit d28291
# if defined(CPPCHECK)
Packit d28291
    GC_noop1((word)me->thread_blocked_sp);
Packit d28291
# endif
Packit d28291
  me -> thread_blocked_sp = NULL;
Packit d28291
  UNLOCK();
Packit d28291
}
Packit d28291
Packit d28291
/* GC_call_with_gc_active() has the opposite to GC_do_blocking()        */
Packit d28291
/* functionality.  It might be called from a user function invoked by   */
Packit d28291
/* GC_do_blocking() to temporarily back allow calling any GC function   */
Packit d28291
/* and/or manipulating pointers to the garbage collected heap.          */
Packit d28291
GC_API void * GC_CALL GC_call_with_gc_active(GC_fn_type fn,
Packit d28291
                                             void * client_data)
Packit d28291
{
Packit d28291
  struct GC_traced_stack_sect_s stacksect;
Packit d28291
  DWORD thread_id = GetCurrentThreadId();
Packit d28291
  GC_thread me;
Packit d28291
  DCL_LOCK_STATE;
Packit d28291
Packit d28291
  LOCK();   /* This will block if the world is stopped.         */
Packit d28291
  me = GC_lookup_thread_inner(thread_id);
Packit d28291
  CHECK_LOOKUP_MY_THREAD(me);
Packit d28291
  /* Adjust our stack base value (this could happen unless      */
Packit d28291
  /* GC_get_stack_base() was used which returned GC_SUCCESS).   */
Packit d28291
  GC_ASSERT(me -> stack_base != NULL);
Packit d28291
  if ((word)me->stack_base < (word)(&stacksect))
Packit d28291
    me -> stack_base = (ptr_t)(&stacksect);
Packit d28291
Packit d28291
  if (me -> thread_blocked_sp == NULL) {
Packit d28291
    /* We are not inside GC_do_blocking() - do nothing more.    */
Packit d28291
    UNLOCK();
Packit d28291
    client_data = fn(client_data);
Packit d28291
    /* Prevent treating the above as a tail call.       */
Packit d28291
    GC_noop1((word)(&stacksect));
Packit d28291
    return client_data; /* result */
Packit d28291
  }
Packit d28291
Packit d28291
  /* Setup new "stack section". */
Packit d28291
  stacksect.saved_stack_ptr = me -> thread_blocked_sp;
Packit d28291
# ifdef IA64
Packit d28291
    /* This is the same as in GC_call_with_stack_base().        */
Packit d28291
    stacksect.backing_store_end = GC_save_regs_in_stack();
Packit d28291
    /* Unnecessarily flushes register stack,    */
Packit d28291
    /* but that probably doesn't hurt.          */
Packit d28291
    stacksect.saved_backing_store_ptr = me -> backing_store_ptr;
Packit d28291
# endif
Packit d28291
  stacksect.prev = me -> traced_stack_sect;
Packit d28291
  me -> thread_blocked_sp = NULL;
Packit d28291
  me -> traced_stack_sect = &stacksect;
Packit d28291
Packit d28291
  UNLOCK();
Packit d28291
  client_data = fn(client_data);
Packit d28291
  GC_ASSERT(me -> thread_blocked_sp == NULL);
Packit d28291
  GC_ASSERT(me -> traced_stack_sect == &stacksect);
Packit d28291
Packit d28291
  /* Restore original "stack section".  */
Packit d28291
  LOCK();
Packit d28291
# if defined(CPPCHECK)
Packit d28291
    GC_noop1((word)me->traced_stack_sect);
Packit d28291
# endif
Packit d28291
  me -> traced_stack_sect = stacksect.prev;
Packit d28291
# ifdef IA64
Packit d28291
    me -> backing_store_ptr = stacksect.saved_backing_store_ptr;
Packit d28291
# endif
Packit d28291
  me -> thread_blocked_sp = stacksect.saved_stack_ptr;
Packit d28291
  UNLOCK();
Packit d28291
Packit d28291
  return client_data; /* result */
Packit d28291
}
Packit d28291
Packit d28291
#ifdef GC_PTHREADS
Packit d28291
Packit d28291
  /* A quick-and-dirty cache of the mapping between pthread_t   */
Packit d28291
  /* and win32 thread id.                                       */
Packit d28291
# define PTHREAD_MAP_SIZE 512
Packit d28291
  DWORD GC_pthread_map_cache[PTHREAD_MAP_SIZE] = {0};
Packit d28291
# define PTHREAD_MAP_INDEX(pthread_id) \
Packit d28291
                ((NUMERIC_THREAD_ID(pthread_id) >> 5) % PTHREAD_MAP_SIZE)
Packit d28291
        /* It appears pthread_t is really a pointer type ... */
Packit d28291
# define SET_PTHREAD_MAP_CACHE(pthread_id, win32_id) \
Packit d28291
      (void)(GC_pthread_map_cache[PTHREAD_MAP_INDEX(pthread_id)] = (win32_id))
Packit d28291
# define GET_PTHREAD_MAP_CACHE(pthread_id) \
Packit d28291
          GC_pthread_map_cache[PTHREAD_MAP_INDEX(pthread_id)]
Packit d28291
Packit d28291
  /* Return a GC_thread corresponding to a given pthread_t.     */
Packit d28291
  /* Returns 0 if it's not there.                               */
Packit d28291
  /* We assume that this is only called for pthread ids that    */
Packit d28291
  /* have not yet terminated or are still joinable, and         */
Packit d28291
  /* cannot be concurrently terminated.                         */
Packit d28291
  /* Assumes we do NOT hold the allocation lock.                */
Packit d28291
  STATIC GC_thread GC_lookup_pthread(pthread_t id)
Packit d28291
  {
Packit d28291
#   ifndef GC_NO_THREADS_DISCOVERY
Packit d28291
      if (GC_win32_dll_threads) {
Packit d28291
        int i;
Packit d28291
        LONG my_max = GC_get_max_thread_index();
Packit d28291
Packit d28291
        for (i = 0; i <= my_max &&
Packit d28291
                    (!AO_load_acquire(&dll_thread_table[i].tm.in_use)
Packit d28291
                    || THREAD_EQUAL(dll_thread_table[i].pthread_id, id));
Packit d28291
                    /* Must still be in_use, since nobody else can      */
Packit d28291
                    /* store our thread_id.                             */
Packit d28291
             i++) {
Packit d28291
          /* empty */
Packit d28291
        }
Packit d28291
        return i <= my_max ? (GC_thread)(dll_thread_table + i) : NULL;
Packit d28291
      } else
Packit d28291
#   endif
Packit d28291
    /* else */ {
Packit d28291
      /* We first try the cache.  If that fails, we use a very slow     */
Packit d28291
      /* approach.                                                      */
Packit d28291
      DWORD win32_id = GET_PTHREAD_MAP_CACHE(id);
Packit d28291
      int hv_guess = THREAD_TABLE_INDEX(win32_id);
Packit d28291
      int hv;
Packit d28291
      GC_thread p;
Packit d28291
      DCL_LOCK_STATE;
Packit d28291
Packit d28291
      LOCK();
Packit d28291
      for (p = GC_threads[hv_guess]; 0 != p; p = p -> tm.next) {
Packit d28291
        if (THREAD_EQUAL(p -> pthread_id, id))
Packit d28291
          goto foundit;
Packit d28291
      }
Packit d28291
      for (hv = 0; hv < THREAD_TABLE_SZ; ++hv) {
Packit d28291
        for (p = GC_threads[hv]; 0 != p; p = p -> tm.next) {
Packit d28291
          if (THREAD_EQUAL(p -> pthread_id, id))
Packit d28291
            goto foundit;
Packit d28291
        }
Packit d28291
      }
Packit d28291
      p = 0;
Packit d28291
     foundit:
Packit d28291
      UNLOCK();
Packit d28291
      return p;
Packit d28291
    }
Packit d28291
  }
Packit d28291
Packit d28291
#endif /* GC_PTHREADS */
Packit d28291
Packit d28291
#ifdef CAN_HANDLE_FORK
Packit d28291
    /* Similar to that in pthread_support.c but also rehashes the table */
Packit d28291
    /* since hash map key (thread_id) differs from that in the parent.  */
Packit d28291
    STATIC void GC_remove_all_threads_but_me(void)
Packit d28291
    {
Packit d28291
      int hv;
Packit d28291
      GC_thread p, next, me = NULL;
Packit d28291
      DWORD thread_id;
Packit d28291
      pthread_t pthread_id = pthread_self(); /* same as in parent */
Packit d28291
Packit d28291
      GC_ASSERT(!GC_win32_dll_threads);
Packit d28291
      for (hv = 0; hv < THREAD_TABLE_SZ; ++hv) {
Packit d28291
        for (p = GC_threads[hv]; 0 != p; p = next) {
Packit d28291
          next = p -> tm.next;
Packit d28291
          if (THREAD_EQUAL(p -> pthread_id, pthread_id)
Packit d28291
              && me == NULL) { /* ignore dead threads with the same id */
Packit d28291
            me = p;
Packit d28291
            p -> tm.next = 0;
Packit d28291
          } else {
Packit d28291
#           ifdef THREAD_LOCAL_ALLOC
Packit d28291
              if ((p -> flags & FINISHED) == 0) {
Packit d28291
                /* Cannot call GC_destroy_thread_local here (see the    */
Packit d28291
                /* corresponding comment in pthread_support.c).         */
Packit d28291
                GC_remove_specific_after_fork(GC_thread_key, p -> pthread_id);
Packit d28291
              }
Packit d28291
#           endif
Packit d28291
            if (&first_thread != p)
Packit d28291
              GC_INTERNAL_FREE(p);
Packit d28291
          }
Packit d28291
        }
Packit d28291
        GC_threads[hv] = NULL;
Packit d28291
      }
Packit d28291
Packit d28291
      /* Put "me" back to GC_threads.   */
Packit d28291
      GC_ASSERT(me != NULL);
Packit d28291
      thread_id = GetCurrentThreadId(); /* differs from that in parent */
Packit d28291
      GC_threads[THREAD_TABLE_INDEX(thread_id)] = me;
Packit d28291
Packit d28291
      /* Update Win32 thread Id and handle.     */
Packit d28291
      me -> id = thread_id;
Packit d28291
#     ifndef MSWINCE
Packit d28291
        if (!DuplicateHandle(GetCurrentProcess(), GetCurrentThread(),
Packit d28291
                        GetCurrentProcess(), (HANDLE *)&me->handle,
Packit d28291
                        0 /* dwDesiredAccess */, FALSE /* bInheritHandle */,
Packit d28291
                        DUPLICATE_SAME_ACCESS))
Packit d28291
          ABORT("DuplicateHandle failed");
Packit d28291
#     endif
Packit d28291
Packit d28291
#     if defined(THREAD_LOCAL_ALLOC) && !defined(USE_CUSTOM_SPECIFIC)
Packit d28291
        /* For Cygwin, we need to re-assign thread-local pointer to     */
Packit d28291
        /* 'tlfs' (it is OK to call GC_destroy_thread_local and         */
Packit d28291
        /* GC_free_internal before this action).                        */
Packit d28291
        if (GC_setspecific(GC_thread_key, &me->tlfs) != 0)
Packit d28291
          ABORT("GC_setspecific failed (in child)");
Packit d28291
#     endif
Packit d28291
    }
Packit d28291
Packit d28291
    static void fork_prepare_proc(void)
Packit d28291
    {
Packit d28291
      LOCK();
Packit d28291
#     ifdef PARALLEL_MARK
Packit d28291
        if (GC_parallel)
Packit d28291
          GC_wait_for_reclaim();
Packit d28291
#     endif
Packit d28291
      GC_wait_for_gc_completion(TRUE);
Packit d28291
#     ifdef PARALLEL_MARK
Packit d28291
        if (GC_parallel)
Packit d28291
          GC_acquire_mark_lock();
Packit d28291
#     endif
Packit d28291
    }
Packit d28291
Packit d28291
    static void fork_parent_proc(void)
Packit d28291
    {
Packit d28291
#     ifdef PARALLEL_MARK
Packit d28291
        if (GC_parallel)
Packit d28291
          GC_release_mark_lock();
Packit d28291
#     endif
Packit d28291
      UNLOCK();
Packit d28291
    }
Packit d28291
Packit d28291
    static void fork_child_proc(void)
Packit d28291
    {
Packit d28291
#     ifdef PARALLEL_MARK
Packit d28291
        if (GC_parallel) {
Packit d28291
          GC_release_mark_lock();
Packit d28291
          GC_parallel = FALSE; /* or GC_markers_m1 = 0 */
Packit d28291
                /* Turn off parallel marking in the child, since we are */
Packit d28291
                /* probably just going to exec, and we would have to    */
Packit d28291
                /* restart mark threads.                                */
Packit d28291
        }
Packit d28291
#     endif
Packit d28291
      GC_remove_all_threads_but_me();
Packit d28291
      UNLOCK();
Packit d28291
    }
Packit d28291
Packit d28291
  /* Routines for fork handling by client (no-op if pthread_atfork works). */
Packit d28291
  GC_API void GC_CALL GC_atfork_prepare(void)
Packit d28291
  {
Packit d28291
    if (!EXPECT(GC_is_initialized, TRUE)) GC_init();
Packit d28291
    if (GC_handle_fork <= 0)
Packit d28291
      fork_prepare_proc();
Packit d28291
  }
Packit d28291
Packit d28291
  GC_API void GC_CALL GC_atfork_parent(void)
Packit d28291
  {
Packit d28291
    if (GC_handle_fork <= 0)
Packit d28291
      fork_parent_proc();
Packit d28291
  }
Packit d28291
Packit d28291
  GC_API void GC_CALL GC_atfork_child(void)
Packit d28291
  {
Packit d28291
    if (GC_handle_fork <= 0)
Packit d28291
      fork_child_proc();
Packit d28291
  }
Packit d28291
#endif /* CAN_HANDLE_FORK */
Packit d28291
Packit d28291
void GC_push_thread_structures(void)
Packit d28291
{
Packit d28291
  GC_ASSERT(I_HOLD_LOCK());
Packit d28291
# ifndef GC_NO_THREADS_DISCOVERY
Packit d28291
    if (GC_win32_dll_threads) {
Packit d28291
      /* Unlike the other threads implementations, the thread table     */
Packit d28291
      /* here contains no pointers to the collectible heap (note also   */
Packit d28291
      /* that GC_PTHREADS is incompatible with DllMain-based thread     */
Packit d28291
      /* registration).  Thus we have no private structures we need     */
Packit d28291
      /* to preserve.                                                   */
Packit d28291
    } else
Packit d28291
# endif
Packit d28291
  /* else */ {
Packit d28291
    GC_PUSH_ALL_SYM(GC_threads);
Packit d28291
  }
Packit d28291
# if defined(THREAD_LOCAL_ALLOC)
Packit d28291
    GC_PUSH_ALL_SYM(GC_thread_key);
Packit d28291
    /* Just in case we ever use our own TLS implementation.     */
Packit d28291
# endif
Packit d28291
}
Packit d28291
Packit d28291
/* Suspend the given thread, if it's still active.      */
Packit d28291
STATIC void GC_suspend(GC_thread t)
Packit d28291
{
Packit d28291
# ifndef MSWINCE
Packit d28291
    /* Apparently the Windows 95 GetOpenFileName call creates           */
Packit d28291
    /* a thread that does not properly get cleaned up, and              */
Packit d28291
    /* SuspendThread on its descriptor may provoke a crash.             */
Packit d28291
    /* This reduces the probability of that event, though it still      */
Packit d28291
    /* appears there's a race here.                                     */
Packit d28291
    DWORD exitCode;
Packit d28291
# endif
Packit d28291
  UNPROTECT_THREAD(t);
Packit d28291
# ifndef MSWINCE
Packit d28291
    if (GetExitCodeThread(t -> handle, &exitCode) &&
Packit d28291
        exitCode != STILL_ACTIVE) {
Packit d28291
#     ifdef GC_PTHREADS
Packit d28291
        t -> stack_base = 0; /* prevent stack from being pushed */
Packit d28291
#     else
Packit d28291
        /* this breaks pthread_join on Cygwin, which is guaranteed to  */
Packit d28291
        /* only see user pthreads                                      */
Packit d28291
        GC_ASSERT(GC_win32_dll_threads);
Packit d28291
        GC_delete_gc_thread_no_free(t);
Packit d28291
#     endif
Packit d28291
      return;
Packit d28291
    }
Packit d28291
# endif
Packit d28291
# if defined(MPROTECT_VDB)
Packit d28291
    /* Acquire the spin lock we use to update dirty bits.       */
Packit d28291
    /* Threads shouldn't get stopped holding it.  But we may    */
Packit d28291
    /* acquire and release it in the UNPROTECT_THREAD call.     */
Packit d28291
    while (AO_test_and_set_acquire(&GC_fault_handler_lock) == AO_TS_SET) {
Packit d28291
      /* empty */
Packit d28291
    }
Packit d28291
# endif
Packit d28291
Packit d28291
# ifdef MSWINCE
Packit d28291
    /* SuspendThread() will fail if thread is running kernel code.      */
Packit d28291
    while (SuspendThread(THREAD_HANDLE(t)) == (DWORD)-1)
Packit d28291
      Sleep(10); /* in millis */
Packit d28291
# else
Packit d28291
    if (SuspendThread(t -> handle) == (DWORD)-1)
Packit d28291
      ABORT("SuspendThread failed");
Packit d28291
# endif /* !MSWINCE */
Packit d28291
  t -> suspended = (unsigned char)TRUE;
Packit d28291
# if defined(MPROTECT_VDB)
Packit d28291
    AO_CLEAR(&GC_fault_handler_lock);
Packit d28291
# endif
Packit d28291
  if (GC_on_thread_event)
Packit d28291
    GC_on_thread_event(GC_EVENT_THREAD_SUSPENDED, THREAD_HANDLE(t));
Packit d28291
}
Packit d28291
Packit d28291
#if defined(GC_ASSERTIONS) && !defined(CYGWIN32)
Packit d28291
  GC_INNER GC_bool GC_write_disabled = FALSE;
Packit d28291
                /* TRUE only if GC_stop_world() acquired GC_write_cs.   */
Packit d28291
#endif
Packit d28291
Packit d28291
GC_INNER void GC_stop_world(void)
Packit d28291
{
Packit d28291
  DWORD thread_id = GetCurrentThreadId();
Packit d28291
Packit d28291
  if (!GC_thr_initialized)
Packit d28291
    ABORT("GC_stop_world() called before GC_thr_init()");
Packit d28291
  GC_ASSERT(I_HOLD_LOCK());
Packit d28291
Packit d28291
  /* This code is the same as in pthread_stop_world.c */
Packit d28291
# ifdef PARALLEL_MARK
Packit d28291
    if (GC_parallel) {
Packit d28291
      GC_acquire_mark_lock();
Packit d28291
      GC_ASSERT(GC_fl_builder_count == 0);
Packit d28291
      /* We should have previously waited for it to become zero. */
Packit d28291
    }
Packit d28291
# endif /* PARALLEL_MARK */
Packit d28291
Packit d28291
# if !defined(GC_NO_THREADS_DISCOVERY) || defined(GC_ASSERTIONS)
Packit d28291
    GC_please_stop = TRUE;
Packit d28291
# endif
Packit d28291
# ifndef CYGWIN32
Packit d28291
    GC_ASSERT(!GC_write_disabled);
Packit d28291
    EnterCriticalSection(&GC_write_cs);
Packit d28291
    /* It's not allowed to call GC_printf() (and friends) here down to  */
Packit d28291
    /* LeaveCriticalSection (same applies recursively to GC_suspend,    */
Packit d28291
    /* GC_delete_gc_thread_no_free, GC_get_max_thread_index, GC_size    */
Packit d28291
    /* and GC_remove_protection).                                       */
Packit d28291
#   ifdef GC_ASSERTIONS
Packit d28291
      GC_write_disabled = TRUE;
Packit d28291
#   endif
Packit d28291
# endif
Packit d28291
# ifndef GC_NO_THREADS_DISCOVERY
Packit d28291
    if (GC_win32_dll_threads) {
Packit d28291
      int i;
Packit d28291
      int my_max;
Packit d28291
      /* Any threads being created during this loop will end up setting */
Packit d28291
      /* GC_attached_thread when they start.  This will force marking   */
Packit d28291
      /* to restart.  This is not ideal, but hopefully correct.         */
Packit d28291
      AO_store(&GC_attached_thread, FALSE);
Packit d28291
      my_max = (int)GC_get_max_thread_index();
Packit d28291
      for (i = 0; i <= my_max; i++) {
Packit d28291
        GC_vthread t = dll_thread_table + i;
Packit d28291
        if (t -> stack_base != 0 && t -> thread_blocked_sp == NULL
Packit d28291
            && t -> id != thread_id) {
Packit d28291
          GC_suspend((GC_thread)t);
Packit d28291
        }
Packit d28291
      }
Packit d28291
    } else
Packit d28291
# endif
Packit d28291
  /* else */ {
Packit d28291
    GC_thread t;
Packit d28291
    int i;
Packit d28291
Packit d28291
    for (i = 0; i < THREAD_TABLE_SZ; i++) {
Packit d28291
      for (t = GC_threads[i]; t != 0; t = t -> tm.next) {
Packit d28291
        if (t -> stack_base != 0 && t -> thread_blocked_sp == NULL
Packit d28291
            && !KNOWN_FINISHED(t) && t -> id != thread_id) {
Packit d28291
          GC_suspend(t);
Packit d28291
        }
Packit d28291
      }
Packit d28291
    }
Packit d28291
  }
Packit d28291
# ifndef CYGWIN32
Packit d28291
#   ifdef GC_ASSERTIONS
Packit d28291
      GC_write_disabled = FALSE;
Packit d28291
#   endif
Packit d28291
    LeaveCriticalSection(&GC_write_cs);
Packit d28291
# endif
Packit d28291
# ifdef PARALLEL_MARK
Packit d28291
    if (GC_parallel)
Packit d28291
      GC_release_mark_lock();
Packit d28291
# endif
Packit d28291
}
Packit d28291
Packit d28291
GC_INNER void GC_start_world(void)
Packit d28291
{
Packit d28291
# ifdef GC_ASSERTIONS
Packit d28291
    DWORD thread_id = GetCurrentThreadId();
Packit d28291
# endif
Packit d28291
Packit d28291
  GC_ASSERT(I_HOLD_LOCK());
Packit d28291
  if (GC_win32_dll_threads) {
Packit d28291
    LONG my_max = GC_get_max_thread_index();
Packit d28291
    int i;
Packit d28291
Packit d28291
    for (i = 0; i <= my_max; i++) {
Packit d28291
      GC_thread t = (GC_thread)(dll_thread_table + i);
Packit d28291
      if (t -> suspended) {
Packit d28291
        GC_ASSERT(t -> stack_base != 0 && t -> id != thread_id);
Packit d28291
        if (ResumeThread(THREAD_HANDLE(t)) == (DWORD)-1)
Packit d28291
          ABORT("ResumeThread failed");
Packit d28291
        t -> suspended = FALSE;
Packit d28291
        if (GC_on_thread_event)
Packit d28291
          GC_on_thread_event(GC_EVENT_THREAD_UNSUSPENDED, THREAD_HANDLE(t));
Packit d28291
      }
Packit d28291
    }
Packit d28291
  } else {
Packit d28291
    GC_thread t;
Packit d28291
    int i;
Packit d28291
Packit d28291
    for (i = 0; i < THREAD_TABLE_SZ; i++) {
Packit d28291
      for (t = GC_threads[i]; t != 0; t = t -> tm.next) {
Packit d28291
        if (t -> suspended) {
Packit d28291
          GC_ASSERT(t -> stack_base != 0 && t -> id != thread_id);
Packit d28291
          if (ResumeThread(THREAD_HANDLE(t)) == (DWORD)-1)
Packit d28291
            ABORT("ResumeThread failed");
Packit d28291
          UNPROTECT_THREAD(t);
Packit d28291
          t -> suspended = FALSE;
Packit d28291
          if (GC_on_thread_event)
Packit d28291
            GC_on_thread_event(GC_EVENT_THREAD_UNSUSPENDED, THREAD_HANDLE(t));
Packit d28291
        }
Packit d28291
      }
Packit d28291
    }
Packit d28291
  }
Packit d28291
# if !defined(GC_NO_THREADS_DISCOVERY) || defined(GC_ASSERTIONS)
Packit d28291
    GC_please_stop = FALSE;
Packit d28291
# endif
Packit d28291
}
Packit d28291
Packit d28291
#ifdef MSWINCE
Packit d28291
  /* The VirtualQuery calls below won't work properly on some old WinCE */
Packit d28291
  /* versions, but since each stack is restricted to an aligned 64 KiB  */
Packit d28291
  /* region of virtual memory we can just take the next lowest multiple */
Packit d28291
  /* of 64 KiB.  The result of this macro must not be used as its       */
Packit d28291
  /* argument later and must not be used as the lower bound for sp      */
Packit d28291
  /* check (since the stack may be bigger than 64 KiB).                 */
Packit d28291
# define GC_wince_evaluate_stack_min(s) \
Packit d28291
                        (ptr_t)(((word)(s) - 1) & ~(word)0xFFFF)
Packit d28291
#elif defined(GC_ASSERTIONS)
Packit d28291
# define GC_dont_query_stack_min FALSE
Packit d28291
#endif
Packit d28291
Packit d28291
/* A cache holding the results of the recent VirtualQuery call. */
Packit d28291
/* Protected by the allocation lock.                            */
Packit d28291
static ptr_t last_address = 0;
Packit d28291
static MEMORY_BASIC_INFORMATION last_info;
Packit d28291
Packit d28291
/* Probe stack memory region (starting at "s") to find out its  */
Packit d28291
/* lowest address (i.e. stack top).                             */
Packit d28291
/* S must be a mapped address inside the region, NOT the first  */
Packit d28291
/* unmapped address.                                            */
Packit d28291
STATIC ptr_t GC_get_stack_min(ptr_t s)
Packit d28291
{
Packit d28291
  ptr_t bottom;
Packit d28291
Packit d28291
  GC_ASSERT(I_HOLD_LOCK());
Packit d28291
  if (s != last_address) {
Packit d28291
    VirtualQuery(s, &last_info, sizeof(last_info));
Packit d28291
    last_address = s;
Packit d28291
  }
Packit d28291
  do {
Packit d28291
    bottom = last_info.BaseAddress;
Packit d28291
    VirtualQuery(bottom - 1, &last_info, sizeof(last_info));
Packit d28291
    last_address = bottom - 1;
Packit d28291
  } while ((last_info.Protect & PAGE_READWRITE)
Packit d28291
           && !(last_info.Protect & PAGE_GUARD));
Packit d28291
  return(bottom);
Packit d28291
}
Packit d28291
Packit d28291
/* Return true if the page at s has protections appropriate     */
Packit d28291
/* for a stack page.                                            */
Packit d28291
static GC_bool may_be_in_stack(ptr_t s)
Packit d28291
{
Packit d28291
  GC_ASSERT(I_HOLD_LOCK());
Packit d28291
  if (s != last_address) {
Packit d28291
    VirtualQuery(s, &last_info, sizeof(last_info));
Packit d28291
    last_address = s;
Packit d28291
  }
Packit d28291
  return (last_info.Protect & PAGE_READWRITE)
Packit d28291
          && !(last_info.Protect & PAGE_GUARD);
Packit d28291
}
Packit d28291
Packit d28291
STATIC word GC_push_stack_for(GC_thread thread, DWORD me)
Packit d28291
{
Packit d28291
  ptr_t sp, stack_min;
Packit d28291
Packit d28291
  struct GC_traced_stack_sect_s *traced_stack_sect =
Packit d28291
                                      thread -> traced_stack_sect;
Packit d28291
  if (thread -> id == me) {
Packit d28291
    GC_ASSERT(thread -> thread_blocked_sp == NULL);
Packit d28291
    sp = GC_approx_sp();
Packit d28291
  } else if ((sp = thread -> thread_blocked_sp) == NULL) {
Packit d28291
              /* Use saved sp value for blocked threads. */
Packit d28291
    /* For unblocked threads call GetThreadContext().   */
Packit d28291
    CONTEXT context;
Packit d28291
    context.ContextFlags = CONTEXT_INTEGER|CONTEXT_CONTROL;
Packit d28291
    if (!GetThreadContext(THREAD_HANDLE(thread), &context))
Packit d28291
      ABORT("GetThreadContext failed");
Packit d28291
Packit d28291
    /* Push all registers that might point into the heap.  Frame        */
Packit d28291
    /* pointer registers are included in case client code was           */
Packit d28291
    /* compiled with the 'omit frame pointer' optimization.             */
Packit d28291
#   define PUSH1(reg) GC_push_one((word)context.reg)
Packit d28291
#   define PUSH2(r1,r2) (PUSH1(r1), PUSH1(r2))
Packit d28291
#   define PUSH4(r1,r2,r3,r4) (PUSH2(r1,r2), PUSH2(r3,r4))
Packit d28291
#   if defined(I386)
Packit d28291
      PUSH4(Edi,Esi,Ebx,Edx), PUSH2(Ecx,Eax), PUSH1(Ebp);
Packit d28291
      sp = (ptr_t)context.Esp;
Packit d28291
#   elif defined(X86_64)
Packit d28291
      PUSH4(Rax,Rcx,Rdx,Rbx); PUSH2(Rbp, Rsi); PUSH1(Rdi);
Packit d28291
      PUSH4(R8, R9, R10, R11); PUSH4(R12, R13, R14, R15);
Packit d28291
      sp = (ptr_t)context.Rsp;
Packit d28291
#   elif defined(ARM32)
Packit d28291
      PUSH4(R0,R1,R2,R3),PUSH4(R4,R5,R6,R7),PUSH4(R8,R9,R10,R11);
Packit d28291
      PUSH1(R12);
Packit d28291
      sp = (ptr_t)context.Sp;
Packit d28291
#   elif defined(SHx)
Packit d28291
      PUSH4(R0,R1,R2,R3), PUSH4(R4,R5,R6,R7), PUSH4(R8,R9,R10,R11);
Packit d28291
      PUSH2(R12,R13), PUSH1(R14);
Packit d28291
      sp = (ptr_t)context.R15;
Packit d28291
#   elif defined(MIPS)
Packit d28291
      PUSH4(IntAt,IntV0,IntV1,IntA0), PUSH4(IntA1,IntA2,IntA3,IntT0);
Packit d28291
      PUSH4(IntT1,IntT2,IntT3,IntT4), PUSH4(IntT5,IntT6,IntT7,IntS0);
Packit d28291
      PUSH4(IntS1,IntS2,IntS3,IntS4), PUSH4(IntS5,IntS6,IntS7,IntT8);
Packit d28291
      PUSH4(IntT9,IntK0,IntK1,IntS8);
Packit d28291
      sp = (ptr_t)context.IntSp;
Packit d28291
#   elif defined(PPC)
Packit d28291
      PUSH4(Gpr0, Gpr3, Gpr4, Gpr5),  PUSH4(Gpr6, Gpr7, Gpr8, Gpr9);
Packit d28291
      PUSH4(Gpr10,Gpr11,Gpr12,Gpr14), PUSH4(Gpr15,Gpr16,Gpr17,Gpr18);
Packit d28291
      PUSH4(Gpr19,Gpr20,Gpr21,Gpr22), PUSH4(Gpr23,Gpr24,Gpr25,Gpr26);
Packit d28291
      PUSH4(Gpr27,Gpr28,Gpr29,Gpr30), PUSH1(Gpr31);
Packit d28291
      sp = (ptr_t)context.Gpr1;
Packit d28291
#   elif defined(ALPHA)
Packit d28291
      PUSH4(IntV0,IntT0,IntT1,IntT2), PUSH4(IntT3,IntT4,IntT5,IntT6);
Packit d28291
      PUSH4(IntT7,IntS0,IntS1,IntS2), PUSH4(IntS3,IntS4,IntS5,IntFp);
Packit d28291
      PUSH4(IntA0,IntA1,IntA2,IntA3), PUSH4(IntA4,IntA5,IntT8,IntT9);
Packit d28291
      PUSH4(IntT10,IntT11,IntT12,IntAt);
Packit d28291
      sp = (ptr_t)context.IntSp;
Packit d28291
#   elif !defined(CPPCHECK)
Packit d28291
#     error "architecture is not supported"
Packit d28291
#   endif
Packit d28291
  } /* ! current thread */
Packit d28291
Packit d28291
  /* Set stack_min to the lowest address in the thread stack,   */
Packit d28291
  /* or to an address in the thread stack no larger than sp,    */
Packit d28291
  /* taking advantage of the old value to avoid slow traversals */
Packit d28291
  /* of large stacks.                                           */
Packit d28291
  if (thread -> last_stack_min == ADDR_LIMIT) {
Packit d28291
#   ifdef MSWINCE
Packit d28291
      if (GC_dont_query_stack_min) {
Packit d28291
        stack_min = GC_wince_evaluate_stack_min(traced_stack_sect != NULL ?
Packit d28291
                      (ptr_t)traced_stack_sect : thread -> stack_base);
Packit d28291
        /* Keep last_stack_min value unmodified. */
Packit d28291
      } else
Packit d28291
#   endif
Packit d28291
    /* else */ {
Packit d28291
      stack_min = GC_get_stack_min(traced_stack_sect != NULL ?
Packit d28291
                      (ptr_t)traced_stack_sect : thread -> stack_base);
Packit d28291
      UNPROTECT_THREAD(thread);
Packit d28291
      thread -> last_stack_min = stack_min;
Packit d28291
    }
Packit d28291
  } else {
Packit d28291
    /* First, adjust the latest known minimum stack address if we       */
Packit d28291
    /* are inside GC_call_with_gc_active().                             */
Packit d28291
    if (traced_stack_sect != NULL &&
Packit d28291
        (word)thread->last_stack_min > (word)traced_stack_sect) {
Packit d28291
      UNPROTECT_THREAD(thread);
Packit d28291
      thread -> last_stack_min = (ptr_t)traced_stack_sect;
Packit d28291
    }
Packit d28291
Packit d28291
    if ((word)sp < (word)thread->stack_base
Packit d28291
        && (word)sp >= (word)thread->last_stack_min) {
Packit d28291
      stack_min = sp;
Packit d28291
    } else {
Packit d28291
      /* In the current thread it is always safe to use sp value.       */
Packit d28291
      if (may_be_in_stack(thread -> id == me &&
Packit d28291
                          (word)sp < (word)thread->last_stack_min ?
Packit d28291
                          sp : thread -> last_stack_min)) {
Packit d28291
        stack_min = last_info.BaseAddress;
Packit d28291
        /* Do not probe rest of the stack if sp is correct. */
Packit d28291
        if ((word)sp < (word)stack_min
Packit d28291
            || (word)sp >= (word)thread->stack_base)
Packit d28291
          stack_min = GC_get_stack_min(thread -> last_stack_min);
Packit d28291
      } else {
Packit d28291
        /* Stack shrunk?  Is this possible? */
Packit d28291
        stack_min = GC_get_stack_min(thread -> stack_base);
Packit d28291
      }
Packit d28291
      UNPROTECT_THREAD(thread);
Packit d28291
      thread -> last_stack_min = stack_min;
Packit d28291
    }
Packit d28291
  }
Packit d28291
Packit d28291
  GC_ASSERT(GC_dont_query_stack_min
Packit d28291
            || stack_min == GC_get_stack_min(thread -> stack_base)
Packit d28291
            || ((word)sp >= (word)stack_min
Packit d28291
                && (word)stack_min < (word)thread->stack_base
Packit d28291
                && (word)stack_min
Packit d28291
                        > (word)GC_get_stack_min(thread -> stack_base)));
Packit d28291
Packit d28291
  if ((word)sp >= (word)stack_min && (word)sp < (word)thread->stack_base) {
Packit d28291
#   ifdef DEBUG_THREADS
Packit d28291
      GC_log_printf("Pushing stack for 0x%x from sp %p to %p from 0x%x\n",
Packit d28291
                    (int)thread->id, (void *)sp, (void *)thread->stack_base,
Packit d28291
                    (int)me);
Packit d28291
#   endif
Packit d28291
    GC_push_all_stack_sections(sp, thread->stack_base, traced_stack_sect);
Packit d28291
  } else {
Packit d28291
    /* If not current thread then it is possible for sp to point to     */
Packit d28291
    /* the guarded (untouched yet) page just below the current          */
Packit d28291
    /* stack_min of the thread.                                         */
Packit d28291
    if (thread -> id == me || (word)sp >= (word)thread->stack_base
Packit d28291
        || (word)(sp + GC_page_size) < (word)stack_min)
Packit d28291
      WARN("Thread stack pointer %p out of range, pushing everything\n",
Packit d28291
           sp);
Packit d28291
#   ifdef DEBUG_THREADS
Packit d28291
      GC_log_printf("Pushing stack for 0x%x from (min) %p to %p from 0x%x\n",
Packit d28291
                    (int)thread->id, (void *)stack_min,
Packit d28291
                    (void *)thread->stack_base, (int)me);
Packit d28291
#   endif
Packit d28291
    /* Push everything - ignore "traced stack section" data.            */
Packit d28291
    GC_push_all_stack(stack_min, thread->stack_base);
Packit d28291
  }
Packit d28291
  return thread->stack_base - sp; /* stack grows down */
Packit d28291
}
Packit d28291
Packit d28291
GC_INNER void GC_push_all_stacks(void)
Packit d28291
{
Packit d28291
  DWORD thread_id = GetCurrentThreadId();
Packit d28291
  GC_bool found_me = FALSE;
Packit d28291
# ifndef SMALL_CONFIG
Packit d28291
    unsigned nthreads = 0;
Packit d28291
# endif
Packit d28291
  word total_size = 0;
Packit d28291
# ifndef GC_NO_THREADS_DISCOVERY
Packit d28291
    if (GC_win32_dll_threads) {
Packit d28291
      int i;
Packit d28291
      LONG my_max = GC_get_max_thread_index();
Packit d28291
Packit d28291
      for (i = 0; i <= my_max; i++) {
Packit d28291
        GC_thread t = (GC_thread)(dll_thread_table + i);
Packit d28291
        if (t -> tm.in_use && t -> stack_base) {
Packit d28291
#         ifndef SMALL_CONFIG
Packit d28291
            ++nthreads;
Packit d28291
#         endif
Packit d28291
          total_size += GC_push_stack_for(t, thread_id);
Packit d28291
          if (t -> id == thread_id) found_me = TRUE;
Packit d28291
        }
Packit d28291
      }
Packit d28291
    } else
Packit d28291
# endif
Packit d28291
  /* else */ {
Packit d28291
    int i;
Packit d28291
    for (i = 0; i < THREAD_TABLE_SZ; i++) {
Packit d28291
      GC_thread t;
Packit d28291
      for (t = GC_threads[i]; t != 0; t = t -> tm.next) {
Packit d28291
        if (!KNOWN_FINISHED(t) && t -> stack_base) {
Packit d28291
#         ifndef SMALL_CONFIG
Packit d28291
            ++nthreads;
Packit d28291
#         endif
Packit d28291
          total_size += GC_push_stack_for(t, thread_id);
Packit d28291
          if (t -> id == thread_id) found_me = TRUE;
Packit d28291
        }
Packit d28291
      }
Packit d28291
    }
Packit d28291
  }
Packit d28291
# ifndef SMALL_CONFIG
Packit d28291
    GC_VERBOSE_LOG_PRINTF("Pushed %d thread stacks%s\n", nthreads,
Packit d28291
                          GC_win32_dll_threads ?
Packit d28291
                                " based on DllMain thread tracking" : "");
Packit d28291
# endif
Packit d28291
  if (!found_me && !GC_in_thread_creation)
Packit d28291
    ABORT("Collecting from unknown thread");
Packit d28291
  GC_total_stacksize = total_size;
Packit d28291
}
Packit d28291
Packit d28291
#ifdef PARALLEL_MARK
Packit d28291
Packit d28291
# ifndef MAX_MARKERS
Packit d28291
#   define MAX_MARKERS 16
Packit d28291
# endif
Packit d28291
Packit d28291
  static ptr_t marker_sp[MAX_MARKERS - 1]; /* The cold end of the stack */
Packit d28291
                                           /* for markers.              */
Packit d28291
# ifdef IA64
Packit d28291
    static ptr_t marker_bsp[MAX_MARKERS - 1];
Packit d28291
# endif
Packit d28291
Packit d28291
  static ptr_t marker_last_stack_min[MAX_MARKERS - 1];
Packit d28291
                                /* Last known minimum (hottest) address */
Packit d28291
                                /* in stack (or ADDR_LIMIT if unset)    */
Packit d28291
                                /* for markers.                         */
Packit d28291
Packit d28291
#endif /* PARALLEL_MARK */
Packit d28291
Packit d28291
/* Find stack with the lowest address which overlaps the        */
Packit d28291
/* interval [start, limit).                                     */
Packit d28291
/* Return stack bounds in *lo and *hi.  If no such stack        */
Packit d28291
/* is found, both *hi and *lo will be set to an address         */
Packit d28291
/* higher than limit.                                           */
Packit d28291
GC_INNER void GC_get_next_stack(char *start, char *limit,
Packit d28291
                                char **lo, char **hi)
Packit d28291
{
Packit d28291
  int i;
Packit d28291
  char * current_min = ADDR_LIMIT;  /* Least in-range stack base      */
Packit d28291
  ptr_t *plast_stack_min = NULL;    /* Address of last_stack_min      */
Packit d28291
                                    /* field for thread corresponding */
Packit d28291
                                    /* to current_min.                */
Packit d28291
  GC_thread thread = NULL;          /* Either NULL or points to the   */
Packit d28291
                                    /* thread's hash table entry      */
Packit d28291
                                    /* containing *plast_stack_min.   */
Packit d28291
Packit d28291
  /* First set current_min, ignoring limit. */
Packit d28291
  if (GC_win32_dll_threads) {
Packit d28291
    LONG my_max = GC_get_max_thread_index();
Packit d28291
Packit d28291
    for (i = 0; i <= my_max; i++) {
Packit d28291
      ptr_t s = (ptr_t)(dll_thread_table[i].stack_base);
Packit d28291
Packit d28291
      if ((word)s > (word)start && (word)s < (word)current_min) {
Packit d28291
        /* Update address of last_stack_min. */
Packit d28291
        plast_stack_min = (ptr_t * /* no volatile */)
Packit d28291
                            &dll_thread_table[i].last_stack_min;
Packit d28291
        current_min = s;
Packit d28291
#       if defined(CPPCHECK)
Packit d28291
          /* To avoid a warning that thread is always null.     */
Packit d28291
          thread = (GC_thread)&dll_thread_table[i];
Packit d28291
#       endif
Packit d28291
      }
Packit d28291
    }
Packit d28291
  } else {
Packit d28291
    for (i = 0; i < THREAD_TABLE_SZ; i++) {
Packit d28291
      GC_thread t;
Packit d28291
Packit d28291
      for (t = GC_threads[i]; t != 0; t = t -> tm.next) {
Packit d28291
        ptr_t s = t -> stack_base;
Packit d28291
Packit d28291
        if ((word)s > (word)start && (word)s < (word)current_min) {
Packit d28291
          /* Update address of last_stack_min. */
Packit d28291
          plast_stack_min = &t -> last_stack_min;
Packit d28291
          thread = t; /* Remember current thread to unprotect. */
Packit d28291
          current_min = s;
Packit d28291
        }
Packit d28291
      }
Packit d28291
    }
Packit d28291
#   ifdef PARALLEL_MARK
Packit d28291
      for (i = 0; i < GC_markers_m1; ++i) {
Packit d28291
        ptr_t s = marker_sp[i];
Packit d28291
#       ifdef IA64
Packit d28291
          /* FIXME: not implemented */
Packit d28291
#       endif
Packit d28291
        if ((word)s > (word)start && (word)s < (word)current_min) {
Packit d28291
          GC_ASSERT(marker_last_stack_min[i] != NULL);
Packit d28291
          plast_stack_min = &marker_last_stack_min[i];
Packit d28291
          current_min = s;
Packit d28291
          thread = NULL; /* Not a thread's hash table entry. */
Packit d28291
        }
Packit d28291
      }
Packit d28291
#   endif
Packit d28291
  }
Packit d28291
Packit d28291
  *hi = current_min;
Packit d28291
  if (current_min == ADDR_LIMIT) {
Packit d28291
      *lo = ADDR_LIMIT;
Packit d28291
      return;
Packit d28291
  }
Packit d28291
Packit d28291
  GC_ASSERT((word)current_min > (word)start && plast_stack_min != NULL);
Packit d28291
# ifdef MSWINCE
Packit d28291
    if (GC_dont_query_stack_min) {
Packit d28291
      *lo = GC_wince_evaluate_stack_min(current_min);
Packit d28291
      /* Keep last_stack_min value unmodified. */
Packit d28291
      return;
Packit d28291
    }
Packit d28291
# endif
Packit d28291
Packit d28291
  if ((word)current_min > (word)limit && !may_be_in_stack(limit)) {
Packit d28291
    /* Skip the rest since the memory region at limit address is        */
Packit d28291
    /* not a stack (so the lowest address of the found stack would      */
Packit d28291
    /* be above the limit value anyway).                                */
Packit d28291
    *lo = ADDR_LIMIT;
Packit d28291
    return;
Packit d28291
  }
Packit d28291
Packit d28291
  /* Get the minimum address of the found stack by probing its memory   */
Packit d28291
  /* region starting from the recent known minimum (if set).            */
Packit d28291
  if (*plast_stack_min == ADDR_LIMIT
Packit d28291
      || !may_be_in_stack(*plast_stack_min)) {
Packit d28291
    /* Unsafe to start from last_stack_min value. */
Packit d28291
    *lo = GC_get_stack_min(current_min);
Packit d28291
  } else {
Packit d28291
    /* Use the recent value to optimize search for min address. */
Packit d28291
    *lo = GC_get_stack_min(*plast_stack_min);
Packit d28291
  }
Packit d28291
Packit d28291
  /* Remember current stack_min value. */
Packit d28291
  if (thread != NULL) {
Packit d28291
    UNPROTECT_THREAD(thread);
Packit d28291
  }
Packit d28291
  *plast_stack_min = *lo;
Packit d28291
}
Packit d28291
Packit d28291
#ifdef PARALLEL_MARK
Packit d28291
Packit d28291
# if defined(GC_PTHREADS) && !defined(GC_PTHREADS_PARAMARK)
Packit d28291
    /* Use pthread-based parallel mark implementation.    */
Packit d28291
Packit d28291
    /* Workaround a deadlock in winpthreads-3.0b internals (observed    */
Packit d28291
    /* with MinGW 32/64).                                               */
Packit d28291
#   if !defined(__MINGW32__)
Packit d28291
#     define GC_PTHREADS_PARAMARK
Packit d28291
#   endif
Packit d28291
# endif
Packit d28291
Packit d28291
# if !defined(GC_PTHREADS_PARAMARK)
Packit d28291
    STATIC HANDLE GC_marker_cv[MAX_MARKERS - 1] = {0};
Packit d28291
                        /* Events with manual reset (one for each       */
Packit d28291
                        /* mark helper).                                */
Packit d28291
Packit d28291
    STATIC DWORD GC_marker_Id[MAX_MARKERS - 1] = {0};
Packit d28291
                        /* This table is used for mapping helper        */
Packit d28291
                        /* threads ID to mark helper index (linear      */
Packit d28291
                        /* search is used since the mapping contains    */
Packit d28291
                        /* only a few entries).                         */
Packit d28291
# endif
Packit d28291
Packit d28291
  /* GC_mark_thread() is the same as in pthread_support.c */
Packit d28291
# ifdef GC_PTHREADS_PARAMARK
Packit d28291
    STATIC void * GC_mark_thread(void * id)
Packit d28291
# elif defined(MSWINCE)
Packit d28291
    STATIC DWORD WINAPI GC_mark_thread(LPVOID id)
Packit d28291
# else
Packit d28291
    STATIC unsigned __stdcall GC_mark_thread(void * id)
Packit d28291
# endif
Packit d28291
  {
Packit d28291
    word my_mark_no = 0;
Packit d28291
Packit d28291
    if ((word)id == (word)-1) return 0; /* to make compiler happy */
Packit d28291
    marker_sp[(word)id] = GC_approx_sp();
Packit d28291
#   ifdef IA64
Packit d28291
      marker_bsp[(word)id] = GC_save_regs_in_stack();
Packit d28291
#   endif
Packit d28291
#   if !defined(GC_PTHREADS_PARAMARK)
Packit d28291
      GC_marker_Id[(word)id] = GetCurrentThreadId();
Packit d28291
#   endif
Packit d28291
Packit d28291
    /* Inform GC_start_mark_threads about completion of marker data init. */
Packit d28291
    GC_acquire_mark_lock();
Packit d28291
    if (0 == --GC_fl_builder_count) /* count may have a negative value */
Packit d28291
      GC_notify_all_builder();
Packit d28291
Packit d28291
    for (;; ++my_mark_no) {
Packit d28291
      if (my_mark_no - GC_mark_no > (word)2) {
Packit d28291
        /* resynchronize if we get far off, e.g. because GC_mark_no     */
Packit d28291
        /* wrapped.                                                     */
Packit d28291
        my_mark_no = GC_mark_no;
Packit d28291
      }
Packit d28291
#     ifdef DEBUG_THREADS
Packit d28291
        GC_log_printf("Starting mark helper for mark number %lu\n",
Packit d28291
                      (unsigned long)my_mark_no);
Packit d28291
#     endif
Packit d28291
      GC_help_marker(my_mark_no);
Packit d28291
    }
Packit d28291
  }
Packit d28291
Packit d28291
# ifndef GC_ASSERTIONS
Packit d28291
#   define SET_MARK_LOCK_HOLDER (void)0
Packit d28291
#   define UNSET_MARK_LOCK_HOLDER (void)0
Packit d28291
# endif
Packit d28291
Packit d28291
  /* GC_mark_threads[] is unused here unlike that in pthread_support.c  */
Packit d28291
Packit d28291
# ifdef CAN_HANDLE_FORK
Packit d28291
    static int available_markers_m1 = 0;
Packit d28291
# else
Packit d28291
#   define available_markers_m1 GC_markers_m1
Packit d28291
# endif
Packit d28291
Packit d28291
# ifdef GC_PTHREADS_PARAMARK
Packit d28291
#   include <pthread.h>
Packit d28291
Packit d28291
#   if defined(GC_ASSERTIONS) && !defined(NUMERIC_THREAD_ID)
Packit d28291
#     define NUMERIC_THREAD_ID(id) (unsigned long)(word)GC_PTHREAD_PTRVAL(id)
Packit d28291
      /* Id not guaranteed to be unique. */
Packit d28291
#   endif
Packit d28291
Packit d28291
#   ifdef CAN_HANDLE_FORK
Packit d28291
      static pthread_cond_t mark_cv;
Packit d28291
                        /* initialized by GC_start_mark_threads_inner   */
Packit d28291
#   else
Packit d28291
      static pthread_cond_t mark_cv = PTHREAD_COND_INITIALIZER;
Packit d28291
#   endif
Packit d28291
Packit d28291
    /* GC_start_mark_threads is the same as in pthread_support.c except */
Packit d28291
    /* for thread stack that is assumed to be large enough.             */
Packit d28291
Packit d28291
    GC_INNER void GC_start_mark_threads_inner(void)
Packit d28291
    {
Packit d28291
      int i;
Packit d28291
      pthread_attr_t attr;
Packit d28291
      pthread_t new_thread;
Packit d28291
#     ifndef NO_MARKER_SPECIAL_SIGMASK
Packit d28291
        sigset_t set, oldset;
Packit d28291
#     endif
Packit d28291
Packit d28291
      GC_ASSERT(I_DONT_HOLD_LOCK());
Packit d28291
      if (available_markers_m1 <= 0) return;
Packit d28291
                /* Skip if parallel markers disabled or already started. */
Packit d28291
#     ifdef CAN_HANDLE_FORK
Packit d28291
        if (GC_parallel) return;
Packit d28291
Packit d28291
        /* Reset mark_cv state after forking (as in pthread_support.c). */
Packit d28291
        {
Packit d28291
          pthread_cond_t mark_cv_local = PTHREAD_COND_INITIALIZER;
Packit d28291
          BCOPY(&mark_cv_local, &mark_cv, sizeof(mark_cv));
Packit d28291
        }
Packit d28291
#     endif
Packit d28291
Packit d28291
      GC_ASSERT(GC_fl_builder_count == 0);
Packit d28291
      if (0 != pthread_attr_init(&attr)) ABORT("pthread_attr_init failed");
Packit d28291
      if (0 != pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED))
Packit d28291
        ABORT("pthread_attr_setdetachstate failed");
Packit d28291
Packit d28291
#     ifndef NO_MARKER_SPECIAL_SIGMASK
Packit d28291
        /* Apply special signal mask to GC marker threads, and don't drop */
Packit d28291
        /* user defined signals by GC marker threads.                     */
Packit d28291
        if (sigfillset(&set) != 0)
Packit d28291
          ABORT("sigfillset failed");
Packit d28291
        if (pthread_sigmask(SIG_BLOCK, &set, &oldset) < 0) {
Packit d28291
          WARN("pthread_sigmask set failed, no markers started,"
Packit d28291
               " errno = %" WARN_PRIdPTR "\n", errno);
Packit d28291
          GC_markers_m1 = 0;
Packit d28291
          (void)pthread_attr_destroy(&attr);
Packit d28291
          return;
Packit d28291
        }
Packit d28291
#     endif /* !NO_MARKER_SPECIAL_SIGMASK */
Packit d28291
Packit d28291
#     ifdef CAN_HANDLE_FORK
Packit d28291
        /* To have proper GC_parallel value in GC_help_marker.  */
Packit d28291
        GC_markers_m1 = available_markers_m1;
Packit d28291
#     endif
Packit d28291
      for (i = 0; i < available_markers_m1; ++i) {
Packit d28291
        marker_last_stack_min[i] = ADDR_LIMIT;
Packit d28291
        if (0 != pthread_create(&new_thread, &attr,
Packit d28291
                                GC_mark_thread, (void *)(word)i)) {
Packit d28291
          WARN("Marker thread creation failed\n", 0);
Packit d28291
          /* Don't try to create other marker threads.    */
Packit d28291
          GC_markers_m1 = i;
Packit d28291
          break;
Packit d28291
        }
Packit d28291
      }
Packit d28291
Packit d28291
#     ifndef NO_MARKER_SPECIAL_SIGMASK
Packit d28291
        /* Restore previous signal mask.        */
Packit d28291
        if (pthread_sigmask(SIG_SETMASK, &oldset, NULL) < 0) {
Packit d28291
          WARN("pthread_sigmask restore failed, errno = %" WARN_PRIdPTR "\n",
Packit d28291
               errno);
Packit d28291
        }
Packit d28291
#     endif
Packit d28291
Packit d28291
      (void)pthread_attr_destroy(&attr);
Packit d28291
      GC_wait_for_markers_init();
Packit d28291
      GC_COND_LOG_PRINTF("Started %d mark helper threads\n", GC_markers_m1);
Packit d28291
    }
Packit d28291
Packit d28291
#   ifdef GC_ASSERTIONS
Packit d28291
      STATIC unsigned long GC_mark_lock_holder = NO_THREAD;
Packit d28291
#     define SET_MARK_LOCK_HOLDER \
Packit d28291
                (void)(GC_mark_lock_holder = NUMERIC_THREAD_ID(pthread_self()))
Packit d28291
#     define UNSET_MARK_LOCK_HOLDER \
Packit d28291
                do { \
Packit d28291
                  GC_ASSERT(GC_mark_lock_holder \
Packit d28291
                                == NUMERIC_THREAD_ID(pthread_self())); \
Packit d28291
                  GC_mark_lock_holder = NO_THREAD; \
Packit d28291
                } while (0)
Packit d28291
#   endif /* GC_ASSERTIONS */
Packit d28291
Packit d28291
    static pthread_mutex_t mark_mutex = PTHREAD_MUTEX_INITIALIZER;
Packit d28291
Packit d28291
    static pthread_cond_t builder_cv = PTHREAD_COND_INITIALIZER;
Packit d28291
Packit d28291
    /* GC_acquire/release_mark_lock(), GC_wait_builder/marker(),          */
Packit d28291
    /* GC_wait_for_reclaim(), GC_notify_all_builder/marker() are the same */
Packit d28291
    /* as in pthread_support.c except that GC_generic_lock() is not used. */
Packit d28291
Packit d28291
#   ifdef LOCK_STATS
Packit d28291
      volatile AO_t GC_block_count = 0;
Packit d28291
#   endif
Packit d28291
Packit d28291
    GC_INNER void GC_acquire_mark_lock(void)
Packit d28291
    {
Packit d28291
#     ifdef NUMERIC_THREAD_ID_UNIQUE
Packit d28291
        GC_ASSERT(GC_mark_lock_holder != NUMERIC_THREAD_ID(pthread_self()));
Packit d28291
#     endif
Packit d28291
      if (pthread_mutex_lock(&mark_mutex) != 0) {
Packit d28291
        ABORT("pthread_mutex_lock failed");
Packit d28291
      }
Packit d28291
#     ifdef LOCK_STATS
Packit d28291
        (void)AO_fetch_and_add1(&GC_block_count);
Packit d28291
#     endif
Packit d28291
      /* GC_generic_lock(&mark_mutex); */
Packit d28291
      SET_MARK_LOCK_HOLDER;
Packit d28291
    }
Packit d28291
Packit d28291
    GC_INNER void GC_release_mark_lock(void)
Packit d28291
    {
Packit d28291
      UNSET_MARK_LOCK_HOLDER;
Packit d28291
      if (pthread_mutex_unlock(&mark_mutex) != 0) {
Packit d28291
        ABORT("pthread_mutex_unlock failed");
Packit d28291
      }
Packit d28291
    }
Packit d28291
Packit d28291
    /* Collector must wait for a freelist builders for 2 reasons:       */
Packit d28291
    /* 1) Mark bits may still be getting examined without lock.         */
Packit d28291
    /* 2) Partial free lists referenced only by locals may not be       */
Packit d28291
    /* scanned correctly, e.g. if they contain "pointer-free" objects,  */
Packit d28291
    /* since the free-list link may be ignored.                         */
Packit d28291
    STATIC void GC_wait_builder(void)
Packit d28291
    {
Packit d28291
      UNSET_MARK_LOCK_HOLDER;
Packit d28291
      if (pthread_cond_wait(&builder_cv, &mark_mutex) != 0) {
Packit d28291
        ABORT("pthread_cond_wait failed");
Packit d28291
      }
Packit d28291
      GC_ASSERT(GC_mark_lock_holder == NO_THREAD);
Packit d28291
      SET_MARK_LOCK_HOLDER;
Packit d28291
    }
Packit d28291
Packit d28291
    GC_INNER void GC_wait_for_reclaim(void)
Packit d28291
    {
Packit d28291
      GC_acquire_mark_lock();
Packit d28291
      while (GC_fl_builder_count > 0) {
Packit d28291
        GC_wait_builder();
Packit d28291
      }
Packit d28291
      GC_release_mark_lock();
Packit d28291
    }
Packit d28291
Packit d28291
    GC_INNER void GC_notify_all_builder(void)
Packit d28291
    {
Packit d28291
      GC_ASSERT(GC_mark_lock_holder == NUMERIC_THREAD_ID(pthread_self()));
Packit d28291
      if (pthread_cond_broadcast(&builder_cv) != 0) {
Packit d28291
        ABORT("pthread_cond_broadcast failed");
Packit d28291
      }
Packit d28291
    }
Packit d28291
Packit d28291
    GC_INNER void GC_wait_marker(void)
Packit d28291
    {
Packit d28291
      GC_ASSERT(GC_parallel);
Packit d28291
      UNSET_MARK_LOCK_HOLDER;
Packit d28291
      if (pthread_cond_wait(&mark_cv, &mark_mutex) != 0) {
Packit d28291
        ABORT("pthread_cond_wait failed");
Packit d28291
      }
Packit d28291
      GC_ASSERT(GC_mark_lock_holder == NO_THREAD);
Packit d28291
      SET_MARK_LOCK_HOLDER;
Packit d28291
    }
Packit d28291
Packit d28291
    GC_INNER void GC_notify_all_marker(void)
Packit d28291
    {
Packit d28291
      GC_ASSERT(GC_parallel);
Packit d28291
      if (pthread_cond_broadcast(&mark_cv) != 0) {
Packit d28291
        ABORT("pthread_cond_broadcast failed");
Packit d28291
      }
Packit d28291
    }
Packit d28291
Packit d28291
# else /* ! GC_PTHREADS_PARAMARK */
Packit d28291
Packit d28291
#   ifndef MARK_THREAD_STACK_SIZE
Packit d28291
#     define MARK_THREAD_STACK_SIZE 0   /* default value */
Packit d28291
#   endif
Packit d28291
Packit d28291
    /* mark_mutex_event, builder_cv, mark_cv are initialized in GC_thr_init */
Packit d28291
    static HANDLE mark_mutex_event = (HANDLE)0; /* Event with auto-reset.   */
Packit d28291
    static HANDLE builder_cv = (HANDLE)0; /* Event with manual reset.       */
Packit d28291
    static HANDLE mark_cv = (HANDLE)0; /* Event with manual reset.          */
Packit d28291
Packit d28291
    GC_INNER void GC_start_mark_threads_inner(void)
Packit d28291
    {
Packit d28291
      int i;
Packit d28291
Packit d28291
      GC_ASSERT(I_DONT_HOLD_LOCK());
Packit d28291
      if (available_markers_m1 <= 0) return;
Packit d28291
Packit d28291
      GC_ASSERT(GC_fl_builder_count == 0);
Packit d28291
      /* Initialize GC_marker_cv[] fully before starting the    */
Packit d28291
      /* first helper thread.                                   */
Packit d28291
      for (i = 0; i < GC_markers_m1; ++i) {
Packit d28291
        if ((GC_marker_cv[i] = CreateEvent(NULL /* attrs */,
Packit d28291
                                        TRUE /* isManualReset */,
Packit d28291
                                        FALSE /* initialState */,
Packit d28291
                                        NULL /* name (A/W) */)) == (HANDLE)0)
Packit d28291
          ABORT("CreateEvent failed");
Packit d28291
      }
Packit d28291
Packit d28291
      for (i = 0; i < GC_markers_m1; ++i) {
Packit d28291
#       ifdef MSWINCE
Packit d28291
          HANDLE handle;
Packit d28291
          DWORD thread_id;
Packit d28291
Packit d28291
          marker_last_stack_min[i] = ADDR_LIMIT;
Packit d28291
          /* There is no _beginthreadex() in WinCE. */
Packit d28291
          handle = CreateThread(NULL /* lpsa */,
Packit d28291
                                MARK_THREAD_STACK_SIZE /* ignored */,
Packit d28291
                                GC_mark_thread, (LPVOID)(word)i,
Packit d28291
                                0 /* fdwCreate */, &thread_id);
Packit d28291
          if (handle == NULL) {
Packit d28291
            WARN("Marker thread creation failed\n", 0);
Packit d28291
            /* The most probable failure reason is "not enough memory". */
Packit d28291
            /* Don't try to create other marker threads.                */
Packit d28291
            break;
Packit d28291
          } else {
Packit d28291
            /* It's safe to detach the thread.  */
Packit d28291
            CloseHandle(handle);
Packit d28291
          }
Packit d28291
#       else
Packit d28291
          GC_uintptr_t handle;
Packit d28291
          unsigned thread_id;
Packit d28291
Packit d28291
          marker_last_stack_min[i] = ADDR_LIMIT;
Packit d28291
          handle = _beginthreadex(NULL /* security_attr */,
Packit d28291
                                MARK_THREAD_STACK_SIZE, GC_mark_thread,
Packit d28291
                                (void *)(word)i, 0 /* flags */, &thread_id);
Packit d28291
          if (!handle || handle == (GC_uintptr_t)-1L) {
Packit d28291
            WARN("Marker thread creation failed\n", 0);
Packit d28291
            /* Don't try to create other marker threads.                */
Packit d28291
            break;
Packit d28291
          } else {/* We may detach the thread (if handle is of HANDLE type) */
Packit d28291
            /* CloseHandle((HANDLE)handle); */
Packit d28291
          }
Packit d28291
#       endif
Packit d28291
      }
Packit d28291
Packit d28291
      /* Adjust GC_markers_m1 (and free unused resources) if failed.    */
Packit d28291
      while (GC_markers_m1 > i) {
Packit d28291
        GC_markers_m1--;
Packit d28291
        CloseHandle(GC_marker_cv[GC_markers_m1]);
Packit d28291
      }
Packit d28291
      GC_wait_for_markers_init();
Packit d28291
      GC_COND_LOG_PRINTF("Started %d mark helper threads\n", GC_markers_m1);
Packit d28291
      if (i == 0) {
Packit d28291
        CloseHandle(mark_cv);
Packit d28291
        CloseHandle(builder_cv);
Packit d28291
        CloseHandle(mark_mutex_event);
Packit d28291
      }
Packit d28291
    }
Packit d28291
Packit d28291
#   ifdef GC_ASSERTIONS
Packit d28291
      STATIC DWORD GC_mark_lock_holder = NO_THREAD;
Packit d28291
#     define SET_MARK_LOCK_HOLDER \
Packit d28291
                (void)(GC_mark_lock_holder = GetCurrentThreadId())
Packit d28291
#     define UNSET_MARK_LOCK_HOLDER \
Packit d28291
                do { \
Packit d28291
                  GC_ASSERT(GC_mark_lock_holder == GetCurrentThreadId()); \
Packit d28291
                  GC_mark_lock_holder = NO_THREAD; \
Packit d28291
                } while (0)
Packit d28291
#   endif /* GC_ASSERTIONS */
Packit d28291
Packit d28291
    STATIC /* volatile */ LONG GC_mark_mutex_state = 0;
Packit d28291
                                /* Mutex state: 0 - unlocked,           */
Packit d28291
                                /* 1 - locked and no other waiters,     */
Packit d28291
                                /* -1 - locked and waiters may exist.   */
Packit d28291
                                /* Accessed by InterlockedExchange().   */
Packit d28291
Packit d28291
    /* #define LOCK_STATS */
Packit d28291
#   ifdef LOCK_STATS
Packit d28291
      volatile AO_t GC_block_count = 0;
Packit d28291
      volatile AO_t GC_unlocked_count = 0;
Packit d28291
#   endif
Packit d28291
Packit d28291
    GC_INNER void GC_acquire_mark_lock(void)
Packit d28291
    {
Packit d28291
      GC_ASSERT(GC_mark_lock_holder != GetCurrentThreadId());
Packit d28291
      if (InterlockedExchange(&GC_mark_mutex_state, 1 /* locked */) != 0) {
Packit d28291
#       ifdef LOCK_STATS
Packit d28291
          (void)AO_fetch_and_add1(&GC_block_count);
Packit d28291
#       endif
Packit d28291
        /* Repeatedly reset the state and wait until acquire the lock.  */
Packit d28291
        while (InterlockedExchange(&GC_mark_mutex_state,
Packit d28291
                                   -1 /* locked_and_has_waiters */) != 0) {
Packit d28291
          if (WaitForSingleObject(mark_mutex_event, INFINITE) == WAIT_FAILED)
Packit d28291
            ABORT("WaitForSingleObject failed");
Packit d28291
        }
Packit d28291
      }
Packit d28291
#     ifdef LOCK_STATS
Packit d28291
        else {
Packit d28291
          (void)AO_fetch_and_add1(&GC_unlocked_count);
Packit d28291
        }
Packit d28291
#     endif
Packit d28291
Packit d28291
      GC_ASSERT(GC_mark_lock_holder == NO_THREAD);
Packit d28291
      SET_MARK_LOCK_HOLDER;
Packit d28291
    }
Packit d28291
Packit d28291
    GC_INNER void GC_release_mark_lock(void)
Packit d28291
    {
Packit d28291
      UNSET_MARK_LOCK_HOLDER;
Packit d28291
      if (InterlockedExchange(&GC_mark_mutex_state, 0 /* unlocked */) < 0) {
Packit d28291
        /* wake a waiter */
Packit d28291
        if (SetEvent(mark_mutex_event) == FALSE)
Packit d28291
          ABORT("SetEvent failed");
Packit d28291
      }
Packit d28291
    }
Packit d28291
Packit d28291
    /* In GC_wait_for_reclaim/GC_notify_all_builder() we emulate POSIX    */
Packit d28291
    /* cond_wait/cond_broadcast() primitives with WinAPI Event object     */
Packit d28291
    /* (working in "manual reset" mode).  This works here because         */
Packit d28291
    /* GC_notify_all_builder() is always called holding lock on           */
Packit d28291
    /* mark_mutex and the checked condition (GC_fl_builder_count == 0)    */
Packit d28291
    /* is the only one for which broadcasting on builder_cv is performed. */
Packit d28291
Packit d28291
    GC_INNER void GC_wait_for_reclaim(void)
Packit d28291
    {
Packit d28291
      GC_ASSERT(builder_cv != 0);
Packit d28291
      for (;;) {
Packit d28291
        GC_acquire_mark_lock();
Packit d28291
        if (GC_fl_builder_count == 0)
Packit d28291
          break;
Packit d28291
        if (ResetEvent(builder_cv) == FALSE)
Packit d28291
          ABORT("ResetEvent failed");
Packit d28291
        GC_release_mark_lock();
Packit d28291
        if (WaitForSingleObject(builder_cv, INFINITE) == WAIT_FAILED)
Packit d28291
          ABORT("WaitForSingleObject failed");
Packit d28291
      }
Packit d28291
      GC_release_mark_lock();
Packit d28291
    }
Packit d28291
Packit d28291
    GC_INNER void GC_notify_all_builder(void)
Packit d28291
    {
Packit d28291
      GC_ASSERT(GC_mark_lock_holder == GetCurrentThreadId());
Packit d28291
      GC_ASSERT(builder_cv != 0);
Packit d28291
      GC_ASSERT(GC_fl_builder_count == 0);
Packit d28291
      if (SetEvent(builder_cv) == FALSE)
Packit d28291
        ABORT("SetEvent failed");
Packit d28291
    }
Packit d28291
Packit d28291
    /* mark_cv is used (for waiting) by a non-helper thread.    */
Packit d28291
Packit d28291
    GC_INNER void GC_wait_marker(void)
Packit d28291
    {
Packit d28291
      HANDLE event = mark_cv;
Packit d28291
      DWORD thread_id = GetCurrentThreadId();
Packit d28291
      int i = GC_markers_m1;
Packit d28291
Packit d28291
      while (i-- > 0) {
Packit d28291
        if (GC_marker_Id[i] == thread_id) {
Packit d28291
          event = GC_marker_cv[i];
Packit d28291
          break;
Packit d28291
        }
Packit d28291
      }
Packit d28291
Packit d28291
      if (ResetEvent(event) == FALSE)
Packit d28291
        ABORT("ResetEvent failed");
Packit d28291
      GC_release_mark_lock();
Packit d28291
      if (WaitForSingleObject(event, INFINITE) == WAIT_FAILED)
Packit d28291
        ABORT("WaitForSingleObject failed");
Packit d28291
      GC_acquire_mark_lock();
Packit d28291
    }
Packit d28291
Packit d28291
    GC_INNER void GC_notify_all_marker(void)
Packit d28291
    {
Packit d28291
      DWORD thread_id = GetCurrentThreadId();
Packit d28291
      int i = GC_markers_m1;
Packit d28291
Packit d28291
      while (i-- > 0) {
Packit d28291
        /* Notify every marker ignoring self (for efficiency).  */
Packit d28291
        if (SetEvent(GC_marker_Id[i] != thread_id ? GC_marker_cv[i] :
Packit d28291
                     mark_cv) == FALSE)
Packit d28291
          ABORT("SetEvent failed");
Packit d28291
      }
Packit d28291
    }
Packit d28291
Packit d28291
# endif /* ! GC_PTHREADS_PARAMARK */
Packit d28291
Packit d28291
#endif /* PARALLEL_MARK */
Packit d28291
Packit d28291
  /* We have no DllMain to take care of new threads.  Thus we   */
Packit d28291
  /* must properly intercept thread creation.                   */
Packit d28291
Packit d28291
  typedef struct {
Packit d28291
    LPTHREAD_START_ROUTINE start;
Packit d28291
    LPVOID param;
Packit d28291
  } thread_args;
Packit d28291
Packit d28291
  STATIC void * GC_CALLBACK GC_win32_start_inner(struct GC_stack_base *sb,
Packit d28291
                                                 void *arg)
Packit d28291
  {
Packit d28291
    void * ret;
Packit d28291
    LPTHREAD_START_ROUTINE start = ((thread_args *)arg)->start;
Packit d28291
    LPVOID param = ((thread_args *)arg)->param;
Packit d28291
Packit d28291
    GC_register_my_thread(sb); /* This waits for an in-progress GC.     */
Packit d28291
Packit d28291
#   ifdef DEBUG_THREADS
Packit d28291
      GC_log_printf("thread 0x%lx starting...\n", (long)GetCurrentThreadId());
Packit d28291
#   endif
Packit d28291
Packit d28291
    GC_free(arg);
Packit d28291
Packit d28291
    /* Clear the thread entry even if we exit with an exception.        */
Packit d28291
    /* This is probably pointless, since an uncaught exception is       */
Packit d28291
    /* supposed to result in the process being killed.                  */
Packit d28291
#   ifndef __GNUC__
Packit d28291
      __try
Packit d28291
#   endif
Packit d28291
    {
Packit d28291
      ret = (void *)(word)(*start)(param);
Packit d28291
    }
Packit d28291
#   ifndef __GNUC__
Packit d28291
      __finally
Packit d28291
#   endif
Packit d28291
    {
Packit d28291
      GC_unregister_my_thread();
Packit d28291
    }
Packit d28291
Packit d28291
#   ifdef DEBUG_THREADS
Packit d28291
      GC_log_printf("thread 0x%lx returned from start routine\n",
Packit d28291
                    (long)GetCurrentThreadId());
Packit d28291
#   endif
Packit d28291
    return ret;
Packit d28291
  }
Packit d28291
Packit d28291
  STATIC DWORD WINAPI GC_win32_start(LPVOID arg)
Packit d28291
  {
Packit d28291
    return (DWORD)(word)GC_call_with_stack_base(GC_win32_start_inner, arg);
Packit d28291
  }
Packit d28291
Packit d28291
  GC_API HANDLE WINAPI GC_CreateThread(
Packit d28291
                        LPSECURITY_ATTRIBUTES lpThreadAttributes,
Packit d28291
                        GC_WIN32_SIZE_T dwStackSize,
Packit d28291
                        LPTHREAD_START_ROUTINE lpStartAddress,
Packit d28291
                        LPVOID lpParameter, DWORD dwCreationFlags,
Packit d28291
                        LPDWORD lpThreadId)
Packit d28291
  {
Packit d28291
    if (!EXPECT(parallel_initialized, TRUE))
Packit d28291
      GC_init_parallel();
Packit d28291
                /* make sure GC is initialized (i.e. main thread is     */
Packit d28291
                /* attached, tls initialized).                          */
Packit d28291
Packit d28291
#   ifdef DEBUG_THREADS
Packit d28291
      GC_log_printf("About to create a thread from 0x%lx\n",
Packit d28291
                    (long)GetCurrentThreadId());
Packit d28291
#   endif
Packit d28291
    if (GC_win32_dll_threads) {
Packit d28291
      return CreateThread(lpThreadAttributes, dwStackSize, lpStartAddress,
Packit d28291
                          lpParameter, dwCreationFlags, lpThreadId);
Packit d28291
    } else {
Packit d28291
      thread_args *args = GC_malloc_uncollectable(sizeof(thread_args));
Packit d28291
                /* Handed off to and deallocated by child thread.       */
Packit d28291
      HANDLE thread_h;
Packit d28291
Packit d28291
      if (0 == args) {
Packit d28291
        SetLastError(ERROR_NOT_ENOUGH_MEMORY);
Packit d28291
        return NULL;
Packit d28291
      }
Packit d28291
Packit d28291
      /* set up thread arguments */
Packit d28291
      args -> start = lpStartAddress;
Packit d28291
      args -> param = lpParameter;
Packit d28291
Packit d28291
      set_need_to_lock();
Packit d28291
      thread_h = CreateThread(lpThreadAttributes, dwStackSize, GC_win32_start,
Packit d28291
                              args, dwCreationFlags, lpThreadId);
Packit d28291
      if (thread_h == 0) GC_free(args);
Packit d28291
      return thread_h;
Packit d28291
    }
Packit d28291
  }
Packit d28291
Packit d28291
  GC_API DECLSPEC_NORETURN void WINAPI GC_ExitThread(DWORD dwExitCode)
Packit d28291
  {
Packit d28291
    GC_unregister_my_thread();
Packit d28291
    ExitThread(dwExitCode);
Packit d28291
  }
Packit d28291
Packit d28291
# if !defined(MSWINCE) && !defined(CYGWIN32)
Packit d28291
Packit d28291
    GC_API GC_uintptr_t GC_CALL GC_beginthreadex(
Packit d28291
                                  void *security, unsigned stack_size,
Packit d28291
                                  unsigned (__stdcall *start_address)(void *),
Packit d28291
                                  void *arglist, unsigned initflag,
Packit d28291
                                  unsigned *thrdaddr)
Packit d28291
    {
Packit d28291
      if (!EXPECT(parallel_initialized, TRUE))
Packit d28291
        GC_init_parallel();
Packit d28291
                /* make sure GC is initialized (i.e. main thread is     */
Packit d28291
                /* attached, tls initialized).                          */
Packit d28291
#     ifdef DEBUG_THREADS
Packit d28291
        GC_log_printf("About to create a thread from 0x%lx\n",
Packit d28291
                      (long)GetCurrentThreadId());
Packit d28291
#     endif
Packit d28291
Packit d28291
      if (GC_win32_dll_threads) {
Packit d28291
        return _beginthreadex(security, stack_size, start_address,
Packit d28291
                              arglist, initflag, thrdaddr);
Packit d28291
      } else {
Packit d28291
        GC_uintptr_t thread_h;
Packit d28291
        thread_args *args = GC_malloc_uncollectable(sizeof(thread_args));
Packit d28291
                /* Handed off to and deallocated by child thread.       */
Packit d28291
Packit d28291
        if (0 == args) {
Packit d28291
          /* MSDN docs say _beginthreadex() returns 0 on error and sets */
Packit d28291
          /* errno to either EAGAIN (too many threads) or EINVAL (the   */
Packit d28291
          /* argument is invalid or the stack size is incorrect), so we */
Packit d28291
          /* set errno to EAGAIN on "not enough memory".                */
Packit d28291
          errno = EAGAIN;
Packit d28291
          return 0;
Packit d28291
        }
Packit d28291
Packit d28291
        /* set up thread arguments */
Packit d28291
        args -> start = (LPTHREAD_START_ROUTINE)start_address;
Packit d28291
        args -> param = arglist;
Packit d28291
Packit d28291
        set_need_to_lock();
Packit d28291
        thread_h = _beginthreadex(security, stack_size,
Packit d28291
                        (unsigned (__stdcall *)(void *))GC_win32_start,
Packit d28291
                        args, initflag, thrdaddr);
Packit d28291
        if (thread_h == 0) GC_free(args);
Packit d28291
        return thread_h;
Packit d28291
      }
Packit d28291
    }
Packit d28291
Packit d28291
    GC_API void GC_CALL GC_endthreadex(unsigned retval)
Packit d28291
    {
Packit d28291
      GC_unregister_my_thread();
Packit d28291
      _endthreadex(retval);
Packit d28291
    }
Packit d28291
Packit d28291
# endif /* !MSWINCE && !CYGWIN32 */
Packit d28291
Packit d28291
#ifdef GC_WINMAIN_REDIRECT
Packit d28291
  /* This might be useful on WinCE.  Shouldn't be used with GC_DLL.     */
Packit d28291
Packit d28291
# if defined(MSWINCE) && defined(UNDER_CE)
Packit d28291
#   define WINMAIN_LPTSTR LPWSTR
Packit d28291
# else
Packit d28291
#   define WINMAIN_LPTSTR LPSTR
Packit d28291
# endif
Packit d28291
Packit d28291
  /* This is defined in gc.h.   */
Packit d28291
# undef WinMain
Packit d28291
Packit d28291
  /* Defined outside GC by an application.      */
Packit d28291
  int WINAPI GC_WinMain(HINSTANCE, HINSTANCE, WINMAIN_LPTSTR, int);
Packit d28291
Packit d28291
  typedef struct {
Packit d28291
    HINSTANCE hInstance;
Packit d28291
    HINSTANCE hPrevInstance;
Packit d28291
    WINMAIN_LPTSTR lpCmdLine;
Packit d28291
    int nShowCmd;
Packit d28291
  } main_thread_args;
Packit d28291
Packit d28291
  static DWORD WINAPI main_thread_start(LPVOID arg)
Packit d28291
  {
Packit d28291
    main_thread_args * args = (main_thread_args *) arg;
Packit d28291
    return (DWORD)GC_WinMain(args->hInstance, args->hPrevInstance,
Packit d28291
                             args->lpCmdLine, args->nShowCmd);
Packit d28291
  }
Packit d28291
Packit d28291
  STATIC void * GC_waitForSingleObjectInfinite(void * handle)
Packit d28291
  {
Packit d28291
    return (void *)(word)WaitForSingleObject((HANDLE)handle, INFINITE);
Packit d28291
  }
Packit d28291
Packit d28291
# ifndef WINMAIN_THREAD_STACK_SIZE
Packit d28291
#   define WINMAIN_THREAD_STACK_SIZE 0  /* default value */
Packit d28291
# endif
Packit d28291
Packit d28291
  int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance,
Packit d28291
                     WINMAIN_LPTSTR lpCmdLine, int nShowCmd)
Packit d28291
  {
Packit d28291
    DWORD exit_code = 1;
Packit d28291
Packit d28291
    main_thread_args args = {
Packit d28291
                hInstance, hPrevInstance, lpCmdLine, nShowCmd
Packit d28291
    };
Packit d28291
    HANDLE thread_h;
Packit d28291
    DWORD thread_id;
Packit d28291
Packit d28291
    /* initialize everything */
Packit d28291
    GC_INIT();
Packit d28291
Packit d28291
    /* start the main thread */
Packit d28291
    thread_h = GC_CreateThread(NULL /* lpsa */,
Packit d28291
                        WINMAIN_THREAD_STACK_SIZE /* ignored on WinCE */,
Packit d28291
                        main_thread_start, &args, 0 /* fdwCreate */,
Packit d28291
                        &thread_id);
Packit d28291
Packit d28291
    if (thread_h != NULL) {
Packit d28291
      if ((DWORD)(word)GC_do_blocking(GC_waitForSingleObjectInfinite,
Packit d28291
                                      (void *)thread_h) == WAIT_FAILED)
Packit d28291
        ABORT("WaitForSingleObject(main_thread) failed");
Packit d28291
      GetExitCodeThread (thread_h, &exit_code);
Packit d28291
      CloseHandle (thread_h);
Packit d28291
    } else {
Packit d28291
      ABORT("GC_CreateThread(main_thread) failed");
Packit d28291
    }
Packit d28291
Packit d28291
#   ifdef MSWINCE
Packit d28291
      GC_deinit();
Packit d28291
      DeleteCriticalSection(&GC_allocate_ml);
Packit d28291
#   endif
Packit d28291
    return (int) exit_code;
Packit d28291
  }
Packit d28291
Packit d28291
#endif /* GC_WINMAIN_REDIRECT */
Packit d28291
Packit d28291
GC_INNER void GC_thr_init(void)
Packit d28291
{
Packit d28291
  struct GC_stack_base sb;
Packit d28291
# ifdef GC_ASSERTIONS
Packit d28291
    int sb_result;
Packit d28291
# endif
Packit d28291
Packit d28291
  GC_ASSERT(I_HOLD_LOCK());
Packit d28291
  if (GC_thr_initialized) return;
Packit d28291
Packit d28291
  GC_ASSERT((word)&GC_threads % sizeof(word) == 0);
Packit d28291
  GC_main_thread = GetCurrentThreadId();
Packit d28291
  GC_thr_initialized = TRUE;
Packit d28291
Packit d28291
# ifdef CAN_HANDLE_FORK
Packit d28291
    /* Prepare for forks if requested.  */
Packit d28291
    if (GC_handle_fork) {
Packit d28291
#     ifdef CAN_CALL_ATFORK
Packit d28291
        if (pthread_atfork(fork_prepare_proc, fork_parent_proc,
Packit d28291
                           fork_child_proc) == 0) {
Packit d28291
          /* Handlers successfully registered.  */
Packit d28291
          GC_handle_fork = 1;
Packit d28291
        } else
Packit d28291
#     endif
Packit d28291
      /* else */ if (GC_handle_fork != -1)
Packit d28291
        ABORT("pthread_atfork failed");
Packit d28291
    }
Packit d28291
# endif
Packit d28291
Packit d28291
  /* Add the initial thread, so we can stop it. */
Packit d28291
# ifdef GC_ASSERTIONS
Packit d28291
    sb_result =
Packit d28291
# endif
Packit d28291
        GC_get_stack_base(&sb);
Packit d28291
  GC_ASSERT(sb_result == GC_SUCCESS);
Packit d28291
Packit d28291
# if defined(PARALLEL_MARK)
Packit d28291
    {
Packit d28291
      char * markers_string = GETENV("GC_MARKERS");
Packit d28291
      int markers;
Packit d28291
Packit d28291
      if (markers_string != NULL) {
Packit d28291
        markers = atoi(markers_string);
Packit d28291
        if (markers <= 0 || markers > MAX_MARKERS) {
Packit d28291
          WARN("Too big or invalid number of mark threads: %" WARN_PRIdPTR
Packit d28291
               "; using maximum threads\n", (signed_word)markers);
Packit d28291
          markers = MAX_MARKERS;
Packit d28291
        }
Packit d28291
      } else {
Packit d28291
#       ifdef MSWINCE
Packit d28291
          /* There is no GetProcessAffinityMask() in WinCE.     */
Packit d28291
          /* GC_sysinfo is already initialized.                 */
Packit d28291
          markers = (int)GC_sysinfo.dwNumberOfProcessors;
Packit d28291
#       else
Packit d28291
#         ifdef _WIN64
Packit d28291
            DWORD_PTR procMask = 0;
Packit d28291
            DWORD_PTR sysMask;
Packit d28291
#         else
Packit d28291
            DWORD procMask = 0;
Packit d28291
            DWORD sysMask;
Packit d28291
#         endif
Packit d28291
          int ncpu = 0;
Packit d28291
          if (GetProcessAffinityMask(GetCurrentProcess(),
Packit d28291
                                     (void *)&procMask, (void *)&sysMask)
Packit d28291
              && procMask) {
Packit d28291
            do {
Packit d28291
              ncpu++;
Packit d28291
            } while ((procMask &= procMask - 1) != 0);
Packit d28291
          }
Packit d28291
          markers = ncpu;
Packit d28291
#       endif
Packit d28291
#       if defined(GC_MIN_MARKERS) && !defined(CPPCHECK)
Packit d28291
          /* This is primarily for testing on systems without getenv(). */
Packit d28291
          if (markers < GC_MIN_MARKERS)
Packit d28291
            markers = GC_MIN_MARKERS;
Packit d28291
#       endif
Packit d28291
        if (markers > MAX_MARKERS)
Packit d28291
          markers = MAX_MARKERS; /* silently limit the value */
Packit d28291
      }
Packit d28291
      available_markers_m1 = markers - 1;
Packit d28291
    }
Packit d28291
Packit d28291
    /* Check whether parallel mode could be enabled.    */
Packit d28291
    {
Packit d28291
      if (GC_win32_dll_threads || available_markers_m1 <= 0) {
Packit d28291
        /* Disable parallel marking. */
Packit d28291
        GC_parallel = FALSE;
Packit d28291
        GC_COND_LOG_PRINTF(
Packit d28291
                "Single marker thread, turning off parallel marking\n");
Packit d28291
      } else {
Packit d28291
#       ifndef GC_PTHREADS_PARAMARK
Packit d28291
          /* Initialize Win32 event objects for parallel marking.       */
Packit d28291
          mark_mutex_event = CreateEvent(NULL /* attrs */,
Packit d28291
                                FALSE /* isManualReset */,
Packit d28291
                                FALSE /* initialState */, NULL /* name */);
Packit d28291
          builder_cv = CreateEvent(NULL /* attrs */,
Packit d28291
                                TRUE /* isManualReset */,
Packit d28291
                                FALSE /* initialState */, NULL /* name */);
Packit d28291
          mark_cv = CreateEvent(NULL /* attrs */, TRUE /* isManualReset */,
Packit d28291
                                FALSE /* initialState */, NULL /* name */);
Packit d28291
          if (mark_mutex_event == (HANDLE)0 || builder_cv == (HANDLE)0
Packit d28291
              || mark_cv == (HANDLE)0)
Packit d28291
            ABORT("CreateEvent failed");
Packit d28291
#       endif
Packit d28291
        /* Disable true incremental collection, but generational is OK. */
Packit d28291
        GC_time_limit = GC_TIME_UNLIMITED;
Packit d28291
      }
Packit d28291
    }
Packit d28291
# endif /* PARALLEL_MARK */
Packit d28291
Packit d28291
  GC_ASSERT(0 == GC_lookup_thread_inner(GC_main_thread));
Packit d28291
  GC_register_my_thread_inner(&sb, GC_main_thread);
Packit d28291
}
Packit d28291
Packit d28291
#ifdef GC_PTHREADS
Packit d28291
Packit d28291
  struct start_info {
Packit d28291
    void *(*start_routine)(void *);
Packit d28291
    void *arg;
Packit d28291
    GC_bool detached;
Packit d28291
  };
Packit d28291
Packit d28291
  GC_API int GC_pthread_join(pthread_t pthread_id, void **retval)
Packit d28291
  {
Packit d28291
    int result;
Packit d28291
    GC_thread t;
Packit d28291
    DCL_LOCK_STATE;
Packit d28291
Packit d28291
    GC_ASSERT(!GC_win32_dll_threads);
Packit d28291
#   ifdef DEBUG_THREADS
Packit d28291
      GC_log_printf("thread %p(0x%lx) is joining thread %p\n",
Packit d28291
                    (void *)GC_PTHREAD_PTRVAL(pthread_self()),
Packit d28291
                    (long)GetCurrentThreadId(),
Packit d28291
                    (void *)GC_PTHREAD_PTRVAL(pthread_id));
Packit d28291
#   endif
Packit d28291
Packit d28291
    /* Thread being joined might not have registered itself yet. */
Packit d28291
    /* After the join, thread id may have been recycled.         */
Packit d28291
    /* FIXME: It would be better if this worked more like        */
Packit d28291
    /* pthread_support.c.                                        */
Packit d28291
#   ifndef GC_WIN32_PTHREADS
Packit d28291
      while ((t = GC_lookup_pthread(pthread_id)) == 0)
Packit d28291
        Sleep(10);
Packit d28291
#   endif
Packit d28291
    result = pthread_join(pthread_id, retval);
Packit d28291
    if (0 == result) {
Packit d28291
#     ifdef GC_WIN32_PTHREADS
Packit d28291
        /* pthreads-win32 and winpthreads id are unique (not recycled). */
Packit d28291
        t = GC_lookup_pthread(pthread_id);
Packit d28291
        if (NULL == t) ABORT("Thread not registered");
Packit d28291
#     endif
Packit d28291
Packit d28291
      LOCK();
Packit d28291
      if ((t -> flags & FINISHED) != 0) {
Packit d28291
        GC_delete_gc_thread_no_free(t);
Packit d28291
        GC_INTERNAL_FREE(t);
Packit d28291
      }
Packit d28291
      UNLOCK();
Packit d28291
    }
Packit d28291
Packit d28291
#   ifdef DEBUG_THREADS
Packit d28291
      GC_log_printf("thread %p(0x%lx) join with thread %p %s\n",
Packit d28291
                    (void *)GC_PTHREAD_PTRVAL(pthread_self()),
Packit d28291
                    (long)GetCurrentThreadId(),
Packit d28291
                    (void *)GC_PTHREAD_PTRVAL(pthread_id),
Packit d28291
                    result != 0 ? "failed" : "succeeded");
Packit d28291
#   endif
Packit d28291
    return result;
Packit d28291
  }
Packit d28291
Packit d28291
  /* Cygwin-pthreads calls CreateThread internally, but it's not easily */
Packit d28291
  /* interceptable by us..., so intercept pthread_create instead.       */
Packit d28291
  GC_API int GC_pthread_create(pthread_t *new_thread,
Packit d28291
                               GC_PTHREAD_CREATE_CONST pthread_attr_t *attr,
Packit d28291
                               void *(*start_routine)(void *), void *arg)
Packit d28291
  {
Packit d28291
    int result;
Packit d28291
    struct start_info * si;
Packit d28291
Packit d28291
    if (!EXPECT(parallel_initialized, TRUE))
Packit d28291
      GC_init_parallel();
Packit d28291
             /* make sure GC is initialized (i.e. main thread is attached) */
Packit d28291
    GC_ASSERT(!GC_win32_dll_threads);
Packit d28291
Packit d28291
      /* This is otherwise saved only in an area mmapped by the thread  */
Packit d28291
      /* library, which isn't visible to the collector.                 */
Packit d28291
      si = GC_malloc_uncollectable(sizeof(struct start_info));
Packit d28291
      if (0 == si) return(EAGAIN);
Packit d28291
Packit d28291
      si -> start_routine = start_routine;
Packit d28291
      si -> arg = arg;
Packit d28291
      if (attr != 0 &&
Packit d28291
          pthread_attr_getdetachstate(attr, &si->detached)
Packit d28291
          == PTHREAD_CREATE_DETACHED) {
Packit d28291
        si->detached = TRUE;
Packit d28291
      }
Packit d28291
Packit d28291
#     ifdef DEBUG_THREADS
Packit d28291
        GC_log_printf("About to create a thread from %p(0x%lx)\n",
Packit d28291
                      (void *)GC_PTHREAD_PTRVAL(pthread_self()),
Packit d28291
                      (long)GetCurrentThreadId());
Packit d28291
#     endif
Packit d28291
      set_need_to_lock();
Packit d28291
      result = pthread_create(new_thread, attr, GC_pthread_start, si);
Packit d28291
Packit d28291
      if (result) { /* failure */
Packit d28291
          GC_free(si);
Packit d28291
      }
Packit d28291
      return(result);
Packit d28291
  }
Packit d28291
Packit d28291
  STATIC void * GC_CALLBACK GC_pthread_start_inner(struct GC_stack_base *sb,
Packit d28291
                                                   void * arg)
Packit d28291
  {
Packit d28291
    struct start_info * si = arg;
Packit d28291
    void * result;
Packit d28291
    void *(*start)(void *);
Packit d28291
    void *start_arg;
Packit d28291
    DWORD thread_id = GetCurrentThreadId();
Packit d28291
    pthread_t pthread_id = pthread_self();
Packit d28291
    GC_thread me;
Packit d28291
    DCL_LOCK_STATE;
Packit d28291
Packit d28291
#   ifdef DEBUG_THREADS
Packit d28291
      GC_log_printf("thread %p(0x%x) starting...\n",
Packit d28291
                    (void *)GC_PTHREAD_PTRVAL(pthread_id), (int)thread_id);
Packit d28291
#   endif
Packit d28291
Packit d28291
    GC_ASSERT(!GC_win32_dll_threads);
Packit d28291
    /* If a GC occurs before the thread is registered, that GC will     */
Packit d28291
    /* ignore this thread.  That's fine, since it will block trying to  */
Packit d28291
    /* acquire the allocation lock, and won't yet hold interesting      */
Packit d28291
    /* pointers.                                                        */
Packit d28291
    LOCK();
Packit d28291
    /* We register the thread here instead of in the parent, so that    */
Packit d28291
    /* we don't need to hold the allocation lock during pthread_create. */
Packit d28291
    me = GC_register_my_thread_inner(sb, thread_id);
Packit d28291
    SET_PTHREAD_MAP_CACHE(pthread_id, thread_id);
Packit d28291
    me -> pthread_id = pthread_id;
Packit d28291
    if (si->detached) me -> flags |= DETACHED;
Packit d28291
    UNLOCK();
Packit d28291
Packit d28291
    start = si -> start_routine;
Packit d28291
    start_arg = si -> arg;
Packit d28291
Packit d28291
    GC_free(si); /* was allocated uncollectible */
Packit d28291
Packit d28291
    pthread_cleanup_push(GC_thread_exit_proc, (void *)me);
Packit d28291
    result = (*start)(start_arg);
Packit d28291
    me -> status = result;
Packit d28291
    pthread_cleanup_pop(1);
Packit d28291
Packit d28291
#   ifdef DEBUG_THREADS
Packit d28291
      GC_log_printf("thread %p(0x%x) returned from start routine\n",
Packit d28291
                    (void *)GC_PTHREAD_PTRVAL(pthread_id), (int)thread_id);
Packit d28291
#   endif
Packit d28291
    return(result);
Packit d28291
  }
Packit d28291
Packit d28291
  STATIC void * GC_pthread_start(void * arg)
Packit d28291
  {
Packit d28291
    return GC_call_with_stack_base(GC_pthread_start_inner, arg);
Packit d28291
  }
Packit d28291
Packit d28291
  STATIC void GC_thread_exit_proc(void *arg)
Packit d28291
  {
Packit d28291
    GC_thread me = (GC_thread)arg;
Packit d28291
    DCL_LOCK_STATE;
Packit d28291
Packit d28291
    GC_ASSERT(!GC_win32_dll_threads);
Packit d28291
#   ifdef DEBUG_THREADS
Packit d28291
      GC_log_printf("thread %p(0x%lx) called pthread_exit()\n",
Packit d28291
                    (void *)GC_PTHREAD_PTRVAL(pthread_self()),
Packit d28291
                    (long)GetCurrentThreadId());
Packit d28291
#   endif
Packit d28291
Packit d28291
    LOCK();
Packit d28291
    GC_wait_for_gc_completion(FALSE);
Packit d28291
#   if defined(THREAD_LOCAL_ALLOC)
Packit d28291
      GC_ASSERT(GC_getspecific(GC_thread_key) == &me->tlfs);
Packit d28291
      GC_destroy_thread_local(&(me->tlfs));
Packit d28291
#   endif
Packit d28291
    if (me -> flags & DETACHED) {
Packit d28291
      GC_delete_thread(GetCurrentThreadId());
Packit d28291
    } else {
Packit d28291
      /* deallocate it as part of join */
Packit d28291
      me -> flags |= FINISHED;
Packit d28291
    }
Packit d28291
#   if defined(THREAD_LOCAL_ALLOC)
Packit d28291
      /* It is required to call remove_specific defined in specific.c. */
Packit d28291
      GC_remove_specific(GC_thread_key);
Packit d28291
#   endif
Packit d28291
    UNLOCK();
Packit d28291
  }
Packit d28291
Packit d28291
# ifndef GC_NO_PTHREAD_SIGMASK
Packit d28291
    /* Win32 pthread does not support sigmask.  */
Packit d28291
    /* So, nothing required here...             */
Packit d28291
    GC_API int GC_pthread_sigmask(int how, const sigset_t *set,
Packit d28291
                                  sigset_t *oset)
Packit d28291
    {
Packit d28291
      return pthread_sigmask(how, set, oset);
Packit d28291
    }
Packit d28291
# endif /* !GC_NO_PTHREAD_SIGMASK */
Packit d28291
Packit d28291
  GC_API int GC_pthread_detach(pthread_t thread)
Packit d28291
  {
Packit d28291
    int result;
Packit d28291
    GC_thread t;
Packit d28291
    DCL_LOCK_STATE;
Packit d28291
Packit d28291
    GC_ASSERT(!GC_win32_dll_threads);
Packit d28291
    /* The thread might not have registered itself yet. */
Packit d28291
    /* TODO: Wait for registration of the created thread in pthread_create. */
Packit d28291
    while ((t = GC_lookup_pthread(thread)) == NULL)
Packit d28291
      Sleep(10);
Packit d28291
    result = pthread_detach(thread);
Packit d28291
    if (result == 0) {
Packit d28291
      LOCK();
Packit d28291
      t -> flags |= DETACHED;
Packit d28291
      /* Here the pthread thread id may have been recycled. */
Packit d28291
      if ((t -> flags & FINISHED) != 0) {
Packit d28291
        GC_delete_gc_thread_no_free(t);
Packit d28291
        GC_INTERNAL_FREE(t);
Packit d28291
      }
Packit d28291
      UNLOCK();
Packit d28291
    }
Packit d28291
    return result;
Packit d28291
  }
Packit d28291
Packit d28291
#elif !defined(GC_NO_THREADS_DISCOVERY)
Packit d28291
    /* We avoid acquiring locks here, since this doesn't seem to be     */
Packit d28291
    /* preemptible.  This may run with an uninitialized collector, in   */
Packit d28291
    /* which case we don't do much.  This implies that no threads other */
Packit d28291
    /* than the main one should be created with an uninitialized        */
Packit d28291
    /* collector.  (The alternative of initializing the collector here  */
Packit d28291
    /* seems dangerous, since DllMain is limited in what it can do.)    */
Packit d28291
Packit d28291
# ifdef GC_INSIDE_DLL
Packit d28291
    /* Export only if needed by client. */
Packit d28291
    GC_API
Packit d28291
# else
Packit d28291
#   define GC_DllMain DllMain
Packit d28291
# endif
Packit d28291
  BOOL WINAPI GC_DllMain(HINSTANCE inst GC_ATTR_UNUSED, ULONG reason,
Packit d28291
                         LPVOID reserved GC_ATTR_UNUSED)
Packit d28291
  {
Packit d28291
      DWORD thread_id;
Packit d28291
      static int entry_count = 0;
Packit d28291
Packit d28291
      /* Note that GC_use_threads_discovery should be called by the     */
Packit d28291
      /* client application at start-up to activate automatic thread    */
Packit d28291
      /* registration (it is the default GC behavior since v7.0alpha7); */
Packit d28291
      /* to always have automatic thread registration turned on, the GC */
Packit d28291
      /* should be compiled with -D GC_DISCOVER_TASK_THREADS.           */
Packit d28291
      if (!GC_win32_dll_threads && parallel_initialized) return TRUE;
Packit d28291
Packit d28291
      switch (reason) {
Packit d28291
       case DLL_THREAD_ATTACH:
Packit d28291
#       ifdef PARALLEL_MARK
Packit d28291
          /* Don't register marker threads. */
Packit d28291
          if (GC_parallel) {
Packit d28291
            /* We could reach here only if parallel_initialized == FALSE. */
Packit d28291
            break;
Packit d28291
          }
Packit d28291
#       endif
Packit d28291
        GC_ASSERT(entry_count == 0 || parallel_initialized);
Packit d28291
        ++entry_count;
Packit d28291
        /* FALLTHRU */
Packit d28291
       case DLL_PROCESS_ATTACH:
Packit d28291
        /* This may run with the collector uninitialized. */
Packit d28291
        thread_id = GetCurrentThreadId();
Packit d28291
        if (parallel_initialized && GC_main_thread != thread_id) {
Packit d28291
#         ifdef PARALLEL_MARK
Packit d28291
            ABORT("Cannot initialize parallel marker from DllMain");
Packit d28291
#         else
Packit d28291
            struct GC_stack_base sb;
Packit d28291
            /* Don't lock here. */
Packit d28291
#           ifdef GC_ASSERTIONS
Packit d28291
              int sb_result =
Packit d28291
#           endif
Packit d28291
                        GC_get_stack_base(&sb);
Packit d28291
            GC_ASSERT(sb_result == GC_SUCCESS);
Packit d28291
            GC_register_my_thread_inner(&sb, thread_id);
Packit d28291
#         endif
Packit d28291
        } /* o.w. we already did it during GC_thr_init, called by GC_init */
Packit d28291
        break;
Packit d28291
Packit d28291
       case DLL_THREAD_DETACH:
Packit d28291
        /* We are hopefully running in the context of the exiting thread. */
Packit d28291
        GC_ASSERT(parallel_initialized);
Packit d28291
        if (GC_win32_dll_threads) {
Packit d28291
          GC_delete_thread(GetCurrentThreadId());
Packit d28291
        }
Packit d28291
        break;
Packit d28291
Packit d28291
       case DLL_PROCESS_DETACH:
Packit d28291
        if (GC_win32_dll_threads) {
Packit d28291
          int i;
Packit d28291
          int my_max = (int)GC_get_max_thread_index();
Packit d28291
Packit d28291
          for (i = 0; i <= my_max; ++i) {
Packit d28291
           if (AO_load(&(dll_thread_table[i].tm.in_use)))
Packit d28291
             GC_delete_gc_thread_no_free(&dll_thread_table[i]);
Packit d28291
          }
Packit d28291
          GC_deinit();
Packit d28291
          DeleteCriticalSection(&GC_allocate_ml);
Packit d28291
        }
Packit d28291
        break;
Packit d28291
      }
Packit d28291
      return TRUE;
Packit d28291
  }
Packit d28291
#endif /* !GC_NO_THREADS_DISCOVERY && !GC_PTHREADS */
Packit d28291
Packit d28291
/* Perform all initializations, including those that    */
Packit d28291
/* may require allocation.                              */
Packit d28291
/* Called without allocation lock.                      */
Packit d28291
/* Must be called before a second thread is created.    */
Packit d28291
GC_INNER void GC_init_parallel(void)
Packit d28291
{
Packit d28291
# if defined(THREAD_LOCAL_ALLOC)
Packit d28291
    GC_thread me;
Packit d28291
    DCL_LOCK_STATE;
Packit d28291
# endif
Packit d28291
Packit d28291
  if (parallel_initialized) return;
Packit d28291
  parallel_initialized = TRUE;
Packit d28291
  /* GC_init() calls us back, so set flag first.      */
Packit d28291
Packit d28291
  if (!GC_is_initialized) GC_init();
Packit d28291
# if defined(CPPCHECK) && !defined(GC_NO_THREADS_DISCOVERY)
Packit d28291
    GC_noop1((word)&GC_DllMain);
Packit d28291
# endif
Packit d28291
  if (GC_win32_dll_threads) {
Packit d28291
    set_need_to_lock();
Packit d28291
        /* Cannot intercept thread creation.  Hence we don't know if    */
Packit d28291
        /* other threads exist.  However, client is not allowed to      */
Packit d28291
        /* create other threads before collector initialization.        */
Packit d28291
        /* Thus it's OK not to lock before this.                        */
Packit d28291
  }
Packit d28291
  /* Initialize thread local free lists if used.        */
Packit d28291
# if defined(THREAD_LOCAL_ALLOC)
Packit d28291
    LOCK();
Packit d28291
    me = GC_lookup_thread_inner(GetCurrentThreadId());
Packit d28291
    CHECK_LOOKUP_MY_THREAD(me);
Packit d28291
    GC_init_thread_local(&me->tlfs);
Packit d28291
    UNLOCK();
Packit d28291
# endif
Packit d28291
}
Packit d28291
Packit d28291
#if defined(USE_PTHREAD_LOCKS)
Packit d28291
  /* Support for pthread locking code.          */
Packit d28291
  /* pthread_mutex_trylock may not win here,    */
Packit d28291
  /* due to builtin support for spinning first? */
Packit d28291
Packit d28291
  GC_INNER void GC_lock(void)
Packit d28291
  {
Packit d28291
    pthread_mutex_lock(&GC_allocate_ml);
Packit d28291
  }
Packit d28291
#endif /* USE_PTHREAD_LOCKS */
Packit d28291
Packit d28291
#if defined(THREAD_LOCAL_ALLOC)
Packit d28291
Packit d28291
  /* Add thread-local allocation support.  VC++ uses __declspec(thread).  */
Packit d28291
Packit d28291
  /* We must explicitly mark ptrfree and gcj free lists, since the free   */
Packit d28291
  /* list links wouldn't otherwise be found.  We also set them in the     */
Packit d28291
  /* normal free lists, since that involves touching less memory than if  */
Packit d28291
  /* we scanned them normally.                                            */
Packit d28291
  GC_INNER void GC_mark_thread_local_free_lists(void)
Packit d28291
  {
Packit d28291
    int i;
Packit d28291
    GC_thread p;
Packit d28291
Packit d28291
    for (i = 0; i < THREAD_TABLE_SZ; ++i) {
Packit d28291
      for (p = GC_threads[i]; 0 != p; p = p -> tm.next) {
Packit d28291
        if (!KNOWN_FINISHED(p)) {
Packit d28291
#         ifdef DEBUG_THREADS
Packit d28291
            GC_log_printf("Marking thread locals for 0x%x\n", (int)p -> id);
Packit d28291
#         endif
Packit d28291
          GC_mark_thread_local_fls_for(&(p->tlfs));
Packit d28291
        }
Packit d28291
      }
Packit d28291
    }
Packit d28291
  }
Packit d28291
Packit d28291
# if defined(GC_ASSERTIONS)
Packit d28291
    void GC_check_tls_for(GC_tlfs p);
Packit d28291
#   if defined(USE_CUSTOM_SPECIFIC)
Packit d28291
      void GC_check_tsd_marks(tsd *key);
Packit d28291
#   endif
Packit d28291
    /* Check that all thread-local free-lists are completely marked.    */
Packit d28291
    /* also check that thread-specific-data structures are marked.      */
Packit d28291
    void GC_check_tls(void)
Packit d28291
    {
Packit d28291
        int i;
Packit d28291
        GC_thread p;
Packit d28291
Packit d28291
        for (i = 0; i < THREAD_TABLE_SZ; ++i) {
Packit d28291
          for (p = GC_threads[i]; 0 != p; p = p -> tm.next) {
Packit d28291
            if (!KNOWN_FINISHED(p))
Packit d28291
              GC_check_tls_for(&(p->tlfs));
Packit d28291
          }
Packit d28291
        }
Packit d28291
#       if defined(USE_CUSTOM_SPECIFIC)
Packit d28291
          if (GC_thread_key != 0)
Packit d28291
            GC_check_tsd_marks(GC_thread_key);
Packit d28291
#       endif
Packit d28291
    }
Packit d28291
# endif /* GC_ASSERTIONS */
Packit d28291
Packit d28291
#endif /* THREAD_LOCAL_ALLOC ... */
Packit d28291
Packit d28291
# ifndef GC_NO_THREAD_REDIRECTS
Packit d28291
    /* Restore thread calls redirection.        */
Packit d28291
#   define CreateThread GC_CreateThread
Packit d28291
#   define ExitThread GC_ExitThread
Packit d28291
#   undef _beginthreadex
Packit d28291
#   define _beginthreadex GC_beginthreadex
Packit d28291
#   undef _endthreadex
Packit d28291
#   define _endthreadex GC_endthreadex
Packit d28291
# endif /* !GC_NO_THREAD_REDIRECTS */
Packit d28291
Packit d28291
#endif /* GC_WIN32_THREADS */