1
0
mirror of https://github.com/vlang/v.git synced 2023-08-10 21:13:21 +03:00
v/thirdparty/libgc/gc.c

30267 lines
772 KiB
C

/*
* Copyright (c) 1994 by Xerox Corporation. All rights reserved.
* Copyright (c) 1996 by Silicon Graphics. All rights reserved.
* Copyright (c) 1998 by Fergus Henderson. All rights reserved.
* Copyright (c) 2000-2009 by Hewlett-Packard Development Company.
* All rights reserved.
* Copyright (c) 2009-2018 Ivan Maidanski
*
* THIS MATERIAL IS PROVIDED AS IS, WITH ABSOLUTELY NO WARRANTY EXPRESSED
* OR IMPLIED. ANY USE IS AT YOUR OWN RISK.
*
* Permission is hereby granted to use or copy this program
* for any purpose, provided the above notices are retained on all copies.
* Permission to modify the code and to distribute modified code is granted,
* provided the above notices are retained, and a notice that the code was
* modified is included with the above copyright notice.
*/
#ifndef __cplusplus
#define GC_INNER STATIC
#define GC_EXTERN GC_INNER
#endif
#ifndef GC_DBG_MLC_H
#define GC_DBG_MLC_H
#ifndef GC_PRIVATE_H
#define GC_PRIVATE_H
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
#if!defined(GC_BUILD)&&!defined(NOT_GCBUILD)
#define GC_BUILD
#endif
#if (defined(__linux__)||defined(__GLIBC__)||defined(__GNU__)||(defined(__CYGWIN__)&&(defined(GC_THREADS)||!defined(USE_MMAP))))&&!defined(_GNU_SOURCE)
#define _GNU_SOURCE 1
#endif
#if defined(__INTERIX)&&!defined(_ALL_SOURCE)
#define _ALL_SOURCE 1
#endif
#if (defined(DGUX)&&defined(GC_THREADS)||defined(DGUX386_THREADS)||defined(GC_DGUX386_THREADS))&&!defined(_USING_POSIX4A_DRAFT10)
#define _USING_POSIX4A_DRAFT10 1
#endif
#if defined(__MINGW32__)&&!defined(__MINGW_EXCPT_DEFINE_PSDK)&&defined(__i386__)&&defined(GC_EXTERN)
#define __MINGW_EXCPT_DEFINE_PSDK 1
#endif
#if defined(NO_DEBUGGING)&&!defined(GC_ASSERTIONS)&&!defined(NDEBUG)
#define NDEBUG 1
#endif
#ifndef GC_H
#ifndef GC_H
#define GC_H
#if (defined(WIN64)&&!defined(_WIN64))&&defined(_MSC_VER)
#pragma message("Warning:Expecting _WIN64 for x64 targets!Notice the leading underscore!")
#endif
#if defined(GC_H)
#define GC_TMP_VERSION_MAJOR 8
#define GC_TMP_VERSION_MINOR 1
#define GC_TMP_VERSION_MICRO 0
#ifdef GC_VERSION_MAJOR
#if GC_TMP_VERSION_MAJOR!=GC_VERSION_MAJOR||GC_TMP_VERSION_MINOR!=GC_VERSION_MINOR||GC_TMP_VERSION_MICRO!=GC_VERSION_MICRO
#error Inconsistent version info. Check README.md,include/gc_version.h and configure.ac.
#endif
#else
#define GC_VERSION_MAJOR GC_TMP_VERSION_MAJOR
#define GC_VERSION_MINOR GC_TMP_VERSION_MINOR
#define GC_VERSION_MICRO GC_TMP_VERSION_MICRO
#endif
#endif
#if defined(GC_H)
#if defined(__GNUC__)&&defined(__GNUC_MINOR__)
#define GC_GNUC_PREREQ(major,minor)((__GNUC__<<16)+__GNUC_MINOR__>=((major)<<16)+(minor))
#else
#define GC_GNUC_PREREQ(major,minor)0
#endif
#if defined(SOLARIS_THREADS)||defined(_SOLARIS_THREADS)||defined(_SOLARIS_PTHREADS)||defined(GC_SOLARIS_PTHREADS)
#ifndef GC_SOLARIS_THREADS
#define GC_SOLARIS_THREADS
#endif
#endif
#if defined(IRIX_THREADS)
#define GC_IRIX_THREADS
#endif
#if defined(DGUX_THREADS)&&!defined(GC_DGUX386_THREADS)
#define GC_DGUX386_THREADS
#endif
#if defined(AIX_THREADS)
#define GC_AIX_THREADS
#endif
#if defined(HPUX_THREADS)
#define GC_HPUX_THREADS
#endif
#if defined(OSF1_THREADS)
#define GC_OSF1_THREADS
#endif
#if defined(LINUX_THREADS)
#define GC_LINUX_THREADS
#endif
#if defined(WIN32_THREADS)
#define GC_WIN32_THREADS
#endif
#if defined(RTEMS_THREADS)
#define GC_RTEMS_PTHREADS
#endif
#if defined(USE_LD_WRAP)
#define GC_USE_LD_WRAP
#endif
#if defined(GC_WIN32_PTHREADS)&&!defined(GC_WIN32_THREADS)
#define GC_WIN32_THREADS
#endif
#if defined(GC_AIX_THREADS)||defined(GC_DARWIN_THREADS)||defined(GC_DGUX386_THREADS)||defined(GC_FREEBSD_THREADS)||defined(GC_HPUX_THREADS)||defined(GC_IRIX_THREADS)||defined(GC_LINUX_THREADS)||defined(GC_NETBSD_THREADS)||defined(GC_OPENBSD_THREADS)||defined(GC_OSF1_THREADS)||defined(GC_SOLARIS_THREADS)||defined(GC_WIN32_THREADS)||defined(GC_RTEMS_PTHREADS)
#ifndef GC_THREADS
#define GC_THREADS
#endif
#elif defined(GC_THREADS)
#if defined(__linux__)
#define GC_LINUX_THREADS
#elif defined(__OpenBSD__)
#define GC_OPENBSD_THREADS
#elif defined(_PA_RISC1_1)||defined(_PA_RISC2_0)||defined(hppa)||defined(__HPPA)||(defined(__ia64)&&defined(_HPUX_SOURCE))
#define GC_HPUX_THREADS
#elif defined(__HAIKU__)
#define GC_HAIKU_THREADS
#elif defined(__DragonFly__)||defined(__FreeBSD_kernel__)||(defined(__FreeBSD__)&&!defined(SN_TARGET_ORBIS))
#define GC_FREEBSD_THREADS
#elif defined(__NetBSD__)
#define GC_NETBSD_THREADS
#elif defined(__alpha)||defined(__alpha__)
#define GC_OSF1_THREADS
#elif (defined(mips)||defined(__mips)||defined(_mips))&&!(defined(nec_ews)||defined(_nec_ews)||defined(ultrix)||defined(__ultrix))
#define GC_IRIX_THREADS
#elif defined(__sparc)||((defined(sun)||defined(__sun))&&(defined(i386)||defined(__i386__)||defined(__amd64)||defined(__amd64__)))
#define GC_SOLARIS_THREADS
#elif defined(__APPLE__)&&defined(__MACH__)
#define GC_DARWIN_THREADS
#endif
#if defined(DGUX)&&(defined(i386)||defined(__i386__))
#define GC_DGUX386_THREADS
#endif
#if defined(_AIX)
#define GC_AIX_THREADS
#endif
#if (defined(_WIN32)||defined(_MSC_VER)||defined(__BORLANDC__)||defined(__CYGWIN32__)||defined(__CYGWIN__)||defined(__CEGCC__)||defined(_WIN32_WCE)||defined(__MINGW32__))&&!defined(GC_WIN32_THREADS)
#define GC_WIN32_THREADS
#endif
#if defined(__rtems__)&&(defined(i386)||defined(__i386__))
#define GC_RTEMS_PTHREADS
#endif
#endif
#undef GC_PTHREADS
#if (!defined(GC_WIN32_THREADS)||defined(GC_WIN32_PTHREADS)||defined(__CYGWIN32__)||defined(__CYGWIN__))&&defined(GC_THREADS)&&!defined(NN_PLATFORM_CTR)&&!defined(NN_BUILD_TARGET_PLATFORM_NX)
#define GC_PTHREADS
#endif
#if!defined(_PTHREADS)&&defined(GC_NETBSD_THREADS)
#define _PTHREADS
#endif
#if defined(GC_DGUX386_THREADS)&&!defined(_POSIX4A_DRAFT10_SOURCE)
#define _POSIX4A_DRAFT10_SOURCE 1
#endif
#if!defined(_REENTRANT)&&defined(GC_PTHREADS)&&!defined(GC_WIN32_THREADS)
#define _REENTRANT 1
#endif
#define __GC
#if!defined(_WIN32_WCE)||defined(__GNUC__)
#include <stddef.h>
#if defined(__MINGW32__)&&!defined(_WIN32_WCE)
#include <stdint.h>
#endif
#else
#include <stdlib.h>
#ifndef _PTRDIFF_T_DEFINED
#define _PTRDIFF_T_DEFINED
typedef long ptrdiff_t;
#endif
#endif
#if!defined(GC_NOT_DLL)&&!defined(GC_DLL)&&((defined(_DLL)&&!defined(__GNUC__))||(defined(DLL_EXPORT)&&defined(GC_BUILD)))
#define GC_DLL
#endif
#if defined(GC_DLL)&&!defined(GC_API)
#if defined(__CEGCC__)
#if defined(GC_BUILD)
#define GC_API __declspec(dllexport)
#else
#define GC_API __declspec(dllimport)
#endif
#elif defined(__MINGW32__)
#if defined(__cplusplus)&&defined(GC_BUILD)
#define GC_API extern __declspec(dllexport)
#elif defined(GC_BUILD)||defined(__MINGW32_DELAY_LOAD__)
#define GC_API __declspec(dllexport)
#else
#define GC_API extern __declspec(dllimport)
#endif
#elif defined(_MSC_VER)||defined(__DMC__)||defined(__BORLANDC__)||defined(__CYGWIN__)
#ifdef GC_BUILD
#define GC_API extern __declspec(dllexport)
#else
#define GC_API __declspec(dllimport)
#endif
#elif defined(__WATCOMC__)
#ifdef GC_BUILD
#define GC_API extern __declspec(dllexport)
#else
#define GC_API extern __declspec(dllimport)
#endif
#elif defined(__SYMBIAN32__)
#ifdef GC_BUILD
#define GC_API extern EXPORT_C
#else
#define GC_API extern IMPORT_C
#endif
#elif defined(__GNUC__)
#if defined(GC_BUILD)&&!defined(GC_NO_VISIBILITY)&&(GC_GNUC_PREREQ(4,0)||defined(GC_VISIBILITY_HIDDEN_SET))
#define GC_API extern __attribute__((__visibility__("default")))
#endif
#endif
#endif
#ifndef GC_API
#define GC_API extern
#endif
#ifndef GC_CALL
#define GC_CALL
#endif
#ifndef GC_CALLBACK
#define GC_CALLBACK GC_CALL
#endif
#ifndef GC_ATTR_MALLOC
#ifdef GC_OOM_FUNC_RETURNS_ALIAS
#define GC_ATTR_MALLOC
#elif GC_GNUC_PREREQ(3,1)
#define GC_ATTR_MALLOC __attribute__((__malloc__))
#elif defined(_MSC_VER)&&(_MSC_VER>=1900)&&!defined(__EDG__)
#define GC_ATTR_MALLOC __declspec(allocator)__declspec(noalias)__declspec(restrict)
#elif defined(_MSC_VER)&&_MSC_VER>=1400
#define GC_ATTR_MALLOC __declspec(noalias)__declspec(restrict)
#else
#define GC_ATTR_MALLOC
#endif
#endif
#ifndef GC_ATTR_ALLOC_SIZE
#undef GC_ATTR_CALLOC_SIZE
#ifdef __clang__
#if __has_attribute(__alloc_size__)
#define GC_ATTR_ALLOC_SIZE(argnum)__attribute__((__alloc_size__(argnum)))
#define GC_ATTR_CALLOC_SIZE(n,s)__attribute__((__alloc_size__(n,s)))
#else
#define GC_ATTR_ALLOC_SIZE(argnum)
#endif
#elif GC_GNUC_PREREQ(4,3)&&!defined(__ICC)
#define GC_ATTR_ALLOC_SIZE(argnum)__attribute__((__alloc_size__(argnum)))
#define GC_ATTR_CALLOC_SIZE(n,s)__attribute__((__alloc_size__(n,s)))
#else
#define GC_ATTR_ALLOC_SIZE(argnum)
#endif
#endif
#ifndef GC_ATTR_CALLOC_SIZE
#define GC_ATTR_CALLOC_SIZE(n,s)
#endif
#ifndef GC_ATTR_NONNULL
#if GC_GNUC_PREREQ(4,0)
#define GC_ATTR_NONNULL(argnum)__attribute__((__nonnull__(argnum)))
#else
#define GC_ATTR_NONNULL(argnum)
#endif
#endif
#ifndef GC_ATTR_CONST
#if GC_GNUC_PREREQ(4,0)
#define GC_ATTR_CONST __attribute__((__const__))
#else
#define GC_ATTR_CONST
#endif
#endif
#ifndef GC_ATTR_DEPRECATED
#ifdef GC_BUILD
#undef GC_ATTR_DEPRECATED
#define GC_ATTR_DEPRECATED
#elif GC_GNUC_PREREQ(4,0)
#define GC_ATTR_DEPRECATED __attribute__((__deprecated__))
#elif defined(_MSC_VER)&&_MSC_VER>=1200
#define GC_ATTR_DEPRECATED __declspec(deprecated)
#else
#define GC_ATTR_DEPRECATED
#endif
#endif
#if defined(__sgi)&&!defined(__GNUC__)&&_COMPILER_VERSION>=720
#define GC_ADD_CALLER
#define GC_RETURN_ADDR (GC_word)__return_address
#endif
#if defined(__linux__)||defined(__GLIBC__)
#if!defined(__native_client__)
#include <features.h>
#endif
#if (__GLIBC__==2&&__GLIBC_MINOR__>=1||__GLIBC__ > 2)&&!defined(__ia64__)&&!defined(GC_MISSING_EXECINFO_H)&&!defined(GC_HAVE_BUILTIN_BACKTRACE)
#define GC_HAVE_BUILTIN_BACKTRACE
#endif
#if defined(__i386__)||defined(__amd64__)||defined(__x86_64__)
#define GC_CAN_SAVE_CALL_STACKS
#endif
#endif
#if defined(_MSC_VER)&&_MSC_VER>=1200&&!defined(_AMD64_)&&!defined(_M_X64)&&!defined(_WIN32_WCE)&&!defined(GC_HAVE_NO_BUILTIN_BACKTRACE)&&!defined(GC_HAVE_BUILTIN_BACKTRACE)
#define GC_HAVE_BUILTIN_BACKTRACE
#endif
#if defined(GC_HAVE_BUILTIN_BACKTRACE)&&!defined(GC_CAN_SAVE_CALL_STACKS)
#define GC_CAN_SAVE_CALL_STACKS
#endif
#if defined(__sparc__)
#define GC_CAN_SAVE_CALL_STACKS
#endif
#if (defined(__linux__)||defined(__DragonFly__)||defined(__FreeBSD__)||defined(__FreeBSD_kernel__)||defined(__HAIKU__)||defined(__NetBSD__)||defined(__OpenBSD__)||defined(HOST_ANDROID)||defined(__ANDROID__))&&!defined(GC_CAN_SAVE_CALL_STACKS)
#define GC_ADD_CALLER
#if GC_GNUC_PREREQ(2,95)
#define GC_RETURN_ADDR (GC_word)__builtin_return_address(0)
#if GC_GNUC_PREREQ(4,0)&&(defined(__i386__)||defined(__amd64__)||defined(__x86_64__))
#define GC_HAVE_RETURN_ADDR_PARENT
#define GC_RETURN_ADDR_PARENT (GC_word)__builtin_extract_return_addr(__builtin_return_address(1))
#endif
#else
#define GC_RETURN_ADDR 0
#endif
#endif
#ifdef GC_PTHREADS
#if (defined(GC_DARWIN_THREADS)||defined(GC_WIN32_PTHREADS)||defined(__native_client__)||defined(GC_RTEMS_PTHREADS))&&!defined(GC_NO_DLOPEN)
#define GC_NO_DLOPEN
#endif
#if (defined(GC_DARWIN_THREADS)||defined(GC_WIN32_PTHREADS)||defined(GC_OPENBSD_THREADS)||defined(__native_client__))&&!defined(GC_NO_PTHREAD_SIGMASK)
#define GC_NO_PTHREAD_SIGMASK
#endif
#if defined(__native_client__)
#ifndef GC_PTHREAD_CREATE_CONST
#define GC_PTHREAD_CREATE_CONST
#endif
#ifndef GC_HAVE_PTHREAD_EXIT
#define GC_HAVE_PTHREAD_EXIT
#define GC_PTHREAD_EXIT_ATTRIBUTE
#endif
#endif
#if!defined(GC_HAVE_PTHREAD_EXIT)&&!defined(HOST_ANDROID)&&!defined(__ANDROID__)&&(defined(GC_LINUX_THREADS)||defined(GC_SOLARIS_THREADS))
#define GC_HAVE_PTHREAD_EXIT
#if GC_GNUC_PREREQ(2,7)
#define GC_PTHREAD_EXIT_ATTRIBUTE __attribute__((__noreturn__))
#elif defined(__NORETURN)
#define GC_PTHREAD_EXIT_ATTRIBUTE __NORETURN
#else
#define GC_PTHREAD_EXIT_ATTRIBUTE
#endif
#endif
#if (!defined(GC_HAVE_PTHREAD_EXIT)||defined(__native_client__))&&!defined(GC_NO_PTHREAD_CANCEL)
#define GC_NO_PTHREAD_CANCEL
#endif
#endif
#ifdef __cplusplus
#ifndef GC_ATTR_EXPLICIT
#if __cplusplus>=201103L&&!defined(__clang__)||_MSVC_LANG>=201103L||defined(CPPCHECK)
#define GC_ATTR_EXPLICIT explicit
#else
#define GC_ATTR_EXPLICIT
#endif
#endif
#ifndef GC_NOEXCEPT
#if defined(__DMC__)||(defined(__BORLANDC__)&&(defined(_RWSTD_NO_EXCEPTIONS)||defined(_RWSTD_NO_EX_SPEC)))||(defined(_MSC_VER)&&defined(_HAS_EXCEPTIONS)&&!_HAS_EXCEPTIONS)||(defined(__WATCOMC__)&&!defined(_CPPUNWIND))
#define GC_NOEXCEPT
#ifndef GC_NEW_ABORTS_ON_OOM
#define GC_NEW_ABORTS_ON_OOM
#endif
#elif __cplusplus>=201103L||_MSVC_LANG>=201103L
#define GC_NOEXCEPT noexcept
#else
#define GC_NOEXCEPT throw()
#endif
#endif
#endif
#endif
#ifdef __cplusplus
extern "C" {
#endif
typedef void*GC_PTR;
#ifdef _WIN64
#if defined(__int64)&&!defined(CPPCHECK)
typedef unsigned __int64 GC_word;
typedef __int64 GC_signed_word;
#else
typedef unsigned long long GC_word;
typedef long long GC_signed_word;
#endif
#else
typedef unsigned long GC_word;
typedef long GC_signed_word;
#endif
GC_API unsigned GC_CALL GC_get_version(void);
GC_API GC_ATTR_DEPRECATED GC_word GC_gc_no;
GC_API GC_word GC_CALL GC_get_gc_no(void);
#ifdef GC_THREADS
GC_API GC_ATTR_DEPRECATED int GC_parallel;
GC_API int GC_CALL GC_get_parallel(void);
GC_API void GC_CALL GC_set_markers_count(unsigned);
#endif
typedef void*(GC_CALLBACK*GC_oom_func)(size_t);
GC_API GC_ATTR_DEPRECATED GC_oom_func GC_oom_fn;
GC_API void GC_CALL GC_set_oom_fn(GC_oom_func)GC_ATTR_NONNULL(1);
GC_API GC_oom_func GC_CALL GC_get_oom_fn(void);
typedef void (GC_CALLBACK*GC_on_heap_resize_proc)(GC_word);
GC_API GC_ATTR_DEPRECATED GC_on_heap_resize_proc GC_on_heap_resize;
GC_API void GC_CALL GC_set_on_heap_resize(GC_on_heap_resize_proc);
GC_API GC_on_heap_resize_proc GC_CALL GC_get_on_heap_resize(void);
typedef enum {
GC_EVENT_START,
GC_EVENT_MARK_START,
GC_EVENT_MARK_END,
GC_EVENT_RECLAIM_START,
GC_EVENT_RECLAIM_END,
GC_EVENT_END,
GC_EVENT_PRE_STOP_WORLD,
GC_EVENT_POST_STOP_WORLD,
GC_EVENT_PRE_START_WORLD,
GC_EVENT_POST_START_WORLD,
GC_EVENT_THREAD_SUSPENDED,
GC_EVENT_THREAD_UNSUSPENDED
} GC_EventType;
typedef void (GC_CALLBACK*GC_on_collection_event_proc)(GC_EventType);
GC_API void GC_CALL GC_set_on_collection_event(GC_on_collection_event_proc);
GC_API GC_on_collection_event_proc GC_CALL GC_get_on_collection_event(void);
#if defined(GC_THREADS)||(defined(GC_BUILD)&&defined(NN_PLATFORM_CTR))
typedef void (GC_CALLBACK*GC_on_thread_event_proc)(GC_EventType,
void*);
GC_API void GC_CALL GC_set_on_thread_event(GC_on_thread_event_proc);
GC_API GC_on_thread_event_proc GC_CALL GC_get_on_thread_event(void);
#endif
GC_API GC_ATTR_DEPRECATED int GC_find_leak;
GC_API void GC_CALL GC_set_find_leak(int);
GC_API int GC_CALL GC_get_find_leak(void);
GC_API GC_ATTR_DEPRECATED int GC_all_interior_pointers;
GC_API void GC_CALL GC_set_all_interior_pointers(int);
GC_API int GC_CALL GC_get_all_interior_pointers(void);
GC_API GC_ATTR_DEPRECATED int GC_finalize_on_demand;
GC_API void GC_CALL GC_set_finalize_on_demand(int);
GC_API int GC_CALL GC_get_finalize_on_demand(void);
GC_API GC_ATTR_DEPRECATED int GC_java_finalization;
GC_API void GC_CALL GC_set_java_finalization(int);
GC_API int GC_CALL GC_get_java_finalization(void);
typedef void (GC_CALLBACK*GC_finalizer_notifier_proc)(void);
GC_API GC_ATTR_DEPRECATED GC_finalizer_notifier_proc GC_finalizer_notifier;
GC_API void GC_CALL GC_set_finalizer_notifier(GC_finalizer_notifier_proc);
GC_API GC_finalizer_notifier_proc GC_CALL GC_get_finalizer_notifier(void);
GC_API
#ifndef GC_DONT_GC
GC_ATTR_DEPRECATED
#endif
int GC_dont_gc;
GC_API GC_ATTR_DEPRECATED int GC_dont_expand;
GC_API void GC_CALL GC_set_dont_expand(int);
GC_API int GC_CALL GC_get_dont_expand(void);
GC_API GC_ATTR_DEPRECATED int GC_use_entire_heap;
GC_API GC_ATTR_DEPRECATED int GC_full_freq;
GC_API void GC_CALL GC_set_full_freq(int);
GC_API int GC_CALL GC_get_full_freq(void);
GC_API GC_ATTR_DEPRECATED GC_word GC_non_gc_bytes;
GC_API void GC_CALL GC_set_non_gc_bytes(GC_word);
GC_API GC_word GC_CALL GC_get_non_gc_bytes(void);
GC_API GC_ATTR_DEPRECATED int GC_no_dls;
GC_API void GC_CALL GC_set_no_dls(int);
GC_API int GC_CALL GC_get_no_dls(void);
GC_API GC_ATTR_DEPRECATED GC_word GC_free_space_divisor;
GC_API void GC_CALL GC_set_free_space_divisor(GC_word);
GC_API GC_word GC_CALL GC_get_free_space_divisor(void);
GC_API GC_ATTR_DEPRECATED GC_word GC_max_retries;
GC_API void GC_CALL GC_set_max_retries(GC_word);
GC_API GC_word GC_CALL GC_get_max_retries(void);
GC_API GC_ATTR_DEPRECATED char*GC_stackbottom;
GC_API GC_ATTR_DEPRECATED int GC_dont_precollect;
GC_API void GC_CALL GC_set_dont_precollect(int);
GC_API int GC_CALL GC_get_dont_precollect(void);
GC_API GC_ATTR_DEPRECATED unsigned long GC_time_limit;
#define GC_TIME_UNLIMITED 999999
GC_API void GC_CALL GC_set_time_limit(unsigned long);
GC_API unsigned long GC_CALL GC_get_time_limit(void);
struct GC_timeval_s {
unsigned long tv_ms;
unsigned long tv_nsec;
};
GC_API void GC_CALL GC_set_time_limit_tv(struct GC_timeval_s);
GC_API struct GC_timeval_s GC_CALL GC_get_time_limit_tv(void);
GC_API void GC_CALL GC_set_allocd_bytes_per_finalizer(GC_word);
GC_API GC_word GC_CALL GC_get_allocd_bytes_per_finalizer(void);
GC_API void GC_CALL GC_start_performance_measurement(void);
GC_API unsigned long GC_CALL GC_get_full_gc_total_time(void);
GC_API void GC_CALL GC_set_pages_executable(int);
GC_API int GC_CALL GC_get_pages_executable(void);
GC_API void GC_CALL GC_set_min_bytes_allocd(size_t);
GC_API size_t GC_CALL GC_get_min_bytes_allocd(void);
GC_API void GC_CALL GC_set_rate(int);
GC_API int GC_CALL GC_get_rate(void);
GC_API void GC_CALL GC_set_max_prior_attempts(int);
GC_API int GC_CALL GC_get_max_prior_attempts(void);
GC_API void GC_CALL GC_set_handle_fork(int);
GC_API void GC_CALL GC_atfork_prepare(void);
GC_API void GC_CALL GC_atfork_parent(void);
GC_API void GC_CALL GC_atfork_child(void);
GC_API void GC_CALL GC_init(void);
GC_API int GC_CALL GC_is_init_called(void);
GC_API void GC_CALL GC_deinit(void);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_malloc(size_t);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_malloc_atomic(size_t);
GC_API GC_ATTR_MALLOC char*GC_CALL GC_strdup(const char*);
GC_API GC_ATTR_MALLOC char*GC_CALL
GC_strndup(const char*,size_t)GC_ATTR_NONNULL(1);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_malloc_uncollectable(size_t);
GC_API GC_ATTR_DEPRECATED void*GC_CALL GC_malloc_stubborn(size_t);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(2)void*GC_CALL
GC_memalign(size_t,size_t);
GC_API int GC_CALL GC_posix_memalign(void**,size_t,
size_t)GC_ATTR_NONNULL(1);
GC_API void GC_CALL GC_free(void*);
#define GC_MALLOC_STUBBORN(sz)GC_MALLOC(sz)
#define GC_NEW_STUBBORN(t)GC_NEW(t)
#define GC_CHANGE_STUBBORN(p)GC_change_stubborn(p)
GC_API GC_ATTR_DEPRECATED void GC_CALL GC_change_stubborn(const void*);
GC_API void GC_CALL GC_end_stubborn_change(const void*)GC_ATTR_NONNULL(1);
GC_API void*GC_CALL GC_base(void*);
GC_API int GC_CALL GC_is_heap_ptr(const void*);
GC_API size_t GC_CALL GC_size(const void*)GC_ATTR_NONNULL(1);
GC_API void*GC_CALL GC_realloc(void*,
size_t)
GC_ATTR_ALLOC_SIZE(2);
GC_API int GC_CALL GC_expand_hp(size_t);
GC_API void GC_CALL GC_set_max_heap_size(GC_word);
GC_API void GC_CALL GC_exclude_static_roots(void*,
void*);
GC_API void GC_CALL GC_clear_exclusion_table(void);
GC_API void GC_CALL GC_clear_roots(void);
GC_API void GC_CALL GC_add_roots(void*,
void*);
GC_API void GC_CALL GC_remove_roots(void*,
void*);
GC_API void GC_CALL GC_register_displacement(size_t);
GC_API void GC_CALL GC_debug_register_displacement(size_t);
GC_API void GC_CALL GC_gcollect(void);
GC_API void GC_CALL GC_gcollect_and_unmap(void);
typedef int (GC_CALLBACK*GC_stop_func)(void);
GC_API int GC_CALL GC_try_to_collect(GC_stop_func)
GC_ATTR_NONNULL(1);
GC_API void GC_CALL GC_set_stop_func(GC_stop_func)
GC_ATTR_NONNULL(1);
GC_API GC_stop_func GC_CALL GC_get_stop_func(void);
GC_API size_t GC_CALL GC_get_heap_size(void);
GC_API size_t GC_CALL GC_get_free_bytes(void);
GC_API size_t GC_CALL GC_get_unmapped_bytes(void);
GC_API size_t GC_CALL GC_get_bytes_since_gc(void);
GC_API size_t GC_CALL GC_get_expl_freed_bytes_since_gc(void);
GC_API size_t GC_CALL GC_get_total_bytes(void);
GC_API void GC_CALL GC_get_heap_usage_safe(GC_word*,
GC_word*,
GC_word*,
GC_word*,
GC_word*);
struct GC_prof_stats_s {
GC_word heapsize_full;
GC_word free_bytes_full;
GC_word unmapped_bytes;
GC_word bytes_allocd_since_gc;
GC_word allocd_bytes_before_gc;
GC_word non_gc_bytes;
GC_word gc_no;
GC_word markers_m1;
GC_word bytes_reclaimed_since_gc;
GC_word reclaimed_bytes_before_gc;
GC_word expl_freed_bytes_since_gc;
};
GC_API size_t GC_CALL GC_get_prof_stats(struct GC_prof_stats_s*,
size_t);
#ifdef GC_THREADS
GC_API size_t GC_CALL GC_get_prof_stats_unsafe(struct GC_prof_stats_s*,
size_t);
#endif
GC_API size_t GC_CALL GC_get_size_map_at(int i);
GC_API size_t GC_CALL GC_get_memory_use(void);
GC_API void GC_CALL GC_disable(void);
GC_API int GC_CALL GC_is_disabled(void);
GC_API void GC_CALL GC_enable(void);
GC_API void GC_CALL GC_set_manual_vdb_allowed(int);
GC_API int GC_CALL GC_get_manual_vdb_allowed(void);
GC_API void GC_CALL GC_enable_incremental(void);
GC_API int GC_CALL GC_is_incremental_mode(void);
#define GC_PROTECTS_POINTER_HEAP 1
#define GC_PROTECTS_PTRFREE_HEAP 2
#define GC_PROTECTS_STATIC_DATA 4
#define GC_PROTECTS_STACK 8
#define GC_PROTECTS_NONE 0
GC_API int GC_CALL GC_incremental_protection_needs(void);
GC_API int GC_CALL GC_collect_a_little(void);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_malloc_ignore_off_page(size_t);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_malloc_atomic_ignore_off_page(size_t);
#ifdef GC_ADD_CALLER
#define GC_EXTRAS GC_RETURN_ADDR,__FILE__,__LINE__
#define GC_EXTRA_PARAMS GC_word ra,const char*s,int i
#else
#define GC_EXTRAS __FILE__,__LINE__
#define GC_EXTRA_PARAMS const char*s,int i
#endif
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_malloc_atomic_uncollectable(size_t);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_debug_malloc_atomic_uncollectable(size_t,GC_EXTRA_PARAMS);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_debug_malloc(size_t,GC_EXTRA_PARAMS);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_debug_malloc_atomic(size_t,GC_EXTRA_PARAMS);
GC_API GC_ATTR_MALLOC char*GC_CALL
GC_debug_strdup(const char*,GC_EXTRA_PARAMS);
GC_API GC_ATTR_MALLOC char*GC_CALL
GC_debug_strndup(const char*,size_t,GC_EXTRA_PARAMS)
GC_ATTR_NONNULL(1);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_debug_malloc_uncollectable(size_t,
GC_EXTRA_PARAMS);
GC_API GC_ATTR_DEPRECATED void*GC_CALL
GC_debug_malloc_stubborn(size_t,GC_EXTRA_PARAMS);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_debug_malloc_ignore_off_page(size_t,
GC_EXTRA_PARAMS);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_debug_malloc_atomic_ignore_off_page(size_t,
GC_EXTRA_PARAMS);
GC_API void GC_CALL GC_debug_free(void*);
GC_API void*GC_CALL GC_debug_realloc(void*,
size_t,GC_EXTRA_PARAMS)
GC_ATTR_ALLOC_SIZE(2);
GC_API GC_ATTR_DEPRECATED void GC_CALL GC_debug_change_stubborn(const void*);
GC_API void GC_CALL GC_debug_end_stubborn_change(const void*)
GC_ATTR_NONNULL(1);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_debug_malloc_replacement(size_t);
GC_API GC_ATTR_ALLOC_SIZE(2)void*GC_CALL
GC_debug_realloc_replacement(void*,
size_t);
#ifdef GC_DEBUG_REPLACEMENT
#define GC_MALLOC(sz)GC_debug_malloc_replacement(sz)
#define GC_REALLOC(old,sz)GC_debug_realloc_replacement(old,sz)
#elif defined(GC_DEBUG)
#define GC_MALLOC(sz)GC_debug_malloc(sz,GC_EXTRAS)
#define GC_REALLOC(old,sz)GC_debug_realloc(old,sz,GC_EXTRAS)
#else
#define GC_MALLOC(sz)GC_malloc(sz)
#define GC_REALLOC(old,sz)GC_realloc(old,sz)
#endif
#ifdef GC_DEBUG
#define GC_MALLOC_ATOMIC(sz)GC_debug_malloc_atomic(sz,GC_EXTRAS)
#define GC_STRDUP(s)GC_debug_strdup(s,GC_EXTRAS)
#define GC_STRNDUP(s,sz)GC_debug_strndup(s,sz,GC_EXTRAS)
#define GC_MALLOC_ATOMIC_UNCOLLECTABLE(sz)GC_debug_malloc_atomic_uncollectable(sz,GC_EXTRAS)
#define GC_MALLOC_UNCOLLECTABLE(sz)GC_debug_malloc_uncollectable(sz,GC_EXTRAS)
#define GC_MALLOC_IGNORE_OFF_PAGE(sz)GC_debug_malloc_ignore_off_page(sz,GC_EXTRAS)
#define GC_MALLOC_ATOMIC_IGNORE_OFF_PAGE(sz)GC_debug_malloc_atomic_ignore_off_page(sz,GC_EXTRAS)
#define GC_FREE(p)GC_debug_free(p)
#define GC_REGISTER_FINALIZER(p,f,d,of,od)GC_debug_register_finalizer(p,f,d,of,od)
#define GC_REGISTER_FINALIZER_IGNORE_SELF(p,f,d,of,od)GC_debug_register_finalizer_ignore_self(p,f,d,of,od)
#define GC_REGISTER_FINALIZER_NO_ORDER(p,f,d,of,od)GC_debug_register_finalizer_no_order(p,f,d,of,od)
#define GC_REGISTER_FINALIZER_UNREACHABLE(p,f,d,of,od)GC_debug_register_finalizer_unreachable(p,f,d,of,od)
#define GC_END_STUBBORN_CHANGE(p)GC_debug_end_stubborn_change(p)
#define GC_PTR_STORE_AND_DIRTY(p,q)GC_debug_ptr_store_and_dirty(p,q)
#define GC_GENERAL_REGISTER_DISAPPEARING_LINK(link,obj)GC_general_register_disappearing_link(link,GC_base(( void*)(obj)))
#define GC_REGISTER_LONG_LINK(link,obj)GC_register_long_link(link,GC_base(( void*)(obj)))
#define GC_REGISTER_DISPLACEMENT(n)GC_debug_register_displacement(n)
#else
#define GC_MALLOC_ATOMIC(sz)GC_malloc_atomic(sz)
#define GC_STRDUP(s)GC_strdup(s)
#define GC_STRNDUP(s,sz)GC_strndup(s,sz)
#define GC_MALLOC_ATOMIC_UNCOLLECTABLE(sz)GC_malloc_atomic_uncollectable(sz)
#define GC_MALLOC_UNCOLLECTABLE(sz)GC_malloc_uncollectable(sz)
#define GC_MALLOC_IGNORE_OFF_PAGE(sz)GC_malloc_ignore_off_page(sz)
#define GC_MALLOC_ATOMIC_IGNORE_OFF_PAGE(sz)GC_malloc_atomic_ignore_off_page(sz)
#define GC_FREE(p)GC_free(p)
#define GC_REGISTER_FINALIZER(p,f,d,of,od)GC_register_finalizer(p,f,d,of,od)
#define GC_REGISTER_FINALIZER_IGNORE_SELF(p,f,d,of,od)GC_register_finalizer_ignore_self(p,f,d,of,od)
#define GC_REGISTER_FINALIZER_NO_ORDER(p,f,d,of,od)GC_register_finalizer_no_order(p,f,d,of,od)
#define GC_REGISTER_FINALIZER_UNREACHABLE(p,f,d,of,od)GC_register_finalizer_unreachable(p,f,d,of,od)
#define GC_END_STUBBORN_CHANGE(p)GC_end_stubborn_change(p)
#define GC_PTR_STORE_AND_DIRTY(p,q)GC_ptr_store_and_dirty(p,q)
#define GC_GENERAL_REGISTER_DISAPPEARING_LINK(link,obj)GC_general_register_disappearing_link(link,obj)
#define GC_REGISTER_LONG_LINK(link,obj)GC_register_long_link(link,obj)
#define GC_REGISTER_DISPLACEMENT(n)GC_register_displacement(n)
#endif
#define GC_NEW(t)((t*)GC_MALLOC(sizeof(t)))
#define GC_NEW_ATOMIC(t)((t*)GC_MALLOC_ATOMIC(sizeof(t)))
#define GC_NEW_UNCOLLECTABLE(t)((t*)GC_MALLOC_UNCOLLECTABLE(sizeof(t)))
#ifdef GC_REQUIRE_WCSDUP
GC_API GC_ATTR_MALLOC wchar_t*GC_CALL
GC_wcsdup(const wchar_t*)GC_ATTR_NONNULL(1);
GC_API GC_ATTR_MALLOC wchar_t*GC_CALL
GC_debug_wcsdup(const wchar_t*,GC_EXTRA_PARAMS)GC_ATTR_NONNULL(1);
#ifdef GC_DEBUG
#define GC_WCSDUP(s)GC_debug_wcsdup(s,GC_EXTRAS)
#else
#define GC_WCSDUP(s)GC_wcsdup(s)
#endif
#endif
typedef void (GC_CALLBACK*GC_finalization_proc)(void*,
void*);
GC_API void GC_CALL GC_register_finalizer(void*,
GC_finalization_proc,void*,
GC_finalization_proc*,void**)
GC_ATTR_NONNULL(1);
GC_API void GC_CALL GC_debug_register_finalizer(void*,
GC_finalization_proc,void*,
GC_finalization_proc*,void**)
GC_ATTR_NONNULL(1);
GC_API void GC_CALL GC_register_finalizer_ignore_self(void*,
GC_finalization_proc,void*,
GC_finalization_proc*,void**)
GC_ATTR_NONNULL(1);
GC_API void GC_CALL GC_debug_register_finalizer_ignore_self(void*,
GC_finalization_proc,void*,
GC_finalization_proc*,void**)
GC_ATTR_NONNULL(1);
GC_API void GC_CALL GC_register_finalizer_no_order(void*,
GC_finalization_proc,void*,
GC_finalization_proc*,void**)
GC_ATTR_NONNULL(1);
GC_API void GC_CALL GC_debug_register_finalizer_no_order(void*,
GC_finalization_proc,void*,
GC_finalization_proc*,void**)
GC_ATTR_NONNULL(1);
GC_API void GC_CALL GC_register_finalizer_unreachable(void*,
GC_finalization_proc,void*,
GC_finalization_proc*,void**)
GC_ATTR_NONNULL(1);
GC_API void GC_CALL GC_debug_register_finalizer_unreachable(void*,
GC_finalization_proc,void*,
GC_finalization_proc*,void**)
GC_ATTR_NONNULL(1);
#define GC_NO_MEMORY 2
GC_API int GC_CALL GC_register_disappearing_link(void**)
GC_ATTR_NONNULL(1);
GC_API int GC_CALL GC_general_register_disappearing_link(void**,
const void*)
GC_ATTR_NONNULL(1)GC_ATTR_NONNULL(2);
GC_API int GC_CALL GC_move_disappearing_link(void**,
void**)
GC_ATTR_NONNULL(2);
GC_API int GC_CALL GC_unregister_disappearing_link(void**);
GC_API int GC_CALL GC_register_long_link(void**,
const void*)
GC_ATTR_NONNULL(1)GC_ATTR_NONNULL(2);
GC_API int GC_CALL GC_move_long_link(void**,
void**)
GC_ATTR_NONNULL(2);
GC_API int GC_CALL GC_unregister_long_link(void**);
typedef enum {
GC_TOGGLE_REF_DROP,
GC_TOGGLE_REF_STRONG,
GC_TOGGLE_REF_WEAK
} GC_ToggleRefStatus;
typedef GC_ToggleRefStatus (GC_CALLBACK*GC_toggleref_func)(void*);
GC_API void GC_CALL GC_set_toggleref_func(GC_toggleref_func);
GC_API GC_toggleref_func GC_CALL GC_get_toggleref_func(void);
GC_API int GC_CALL GC_toggleref_add(void*,int)
GC_ATTR_NONNULL(1);
typedef void (GC_CALLBACK*GC_await_finalize_proc)(void*);
GC_API void GC_CALL GC_set_await_finalize_proc(GC_await_finalize_proc);
GC_API GC_await_finalize_proc GC_CALL GC_get_await_finalize_proc(void);
GC_API int GC_CALL GC_should_invoke_finalizers(void);
GC_API int GC_CALL GC_invoke_finalizers(void);
#if defined(__GNUC__)&&!defined(__INTEL_COMPILER)
#define GC_reachable_here(ptr)__asm__ __volatile__(" "::"X"(ptr):"memory")
#else
GC_API void GC_CALL GC_noop1(GC_word);
#ifdef LINT2
#define GC_reachable_here(ptr)GC_noop1(~(GC_word)(ptr)^(~(GC_word)0))
#else
#define GC_reachable_here(ptr)GC_noop1((GC_word)(ptr))
#endif
#endif
typedef void (GC_CALLBACK*GC_warn_proc)(char*,
GC_word);
GC_API void GC_CALL GC_set_warn_proc(GC_warn_proc)GC_ATTR_NONNULL(1);
GC_API GC_warn_proc GC_CALL GC_get_warn_proc(void);
GC_API void GC_CALLBACK GC_ignore_warn_proc(char*,GC_word);
GC_API void GC_CALL GC_set_log_fd(int);
typedef void (GC_CALLBACK*GC_abort_func)(const char*);
GC_API void GC_CALL GC_set_abort_func(GC_abort_func)GC_ATTR_NONNULL(1);
GC_API GC_abort_func GC_CALL GC_get_abort_func(void);
GC_API void GC_CALL GC_abort_on_oom(void);
typedef GC_word GC_hidden_pointer;
#define GC_HIDE_POINTER(p)(~(GC_hidden_pointer)(p))
#define GC_REVEAL_POINTER(p)((void*)GC_HIDE_POINTER(p))
#if defined(I_HIDE_POINTERS)||defined(GC_I_HIDE_POINTERS)
#define HIDE_POINTER(p)GC_HIDE_POINTER(p)
#define REVEAL_POINTER(p)GC_REVEAL_POINTER(p)
#endif
#ifdef GC_THREADS
GC_API void GC_CALL GC_alloc_lock(void);
GC_API void GC_CALL GC_alloc_unlock(void);
#else
#define GC_alloc_lock()(void)0
#define GC_alloc_unlock()(void)0
#endif
typedef void*(GC_CALLBACK*GC_fn_type)(void*);
GC_API void*GC_CALL GC_call_with_alloc_lock(GC_fn_type,
void*)GC_ATTR_NONNULL(1);
struct GC_stack_base {
void*mem_base;
#if defined(__ia64)||defined(__ia64__)||defined(_M_IA64)
void*reg_base;
#endif
};
typedef void*(GC_CALLBACK*GC_stack_base_func)(
struct GC_stack_base*,void*);
GC_API void*GC_CALL GC_call_with_stack_base(GC_stack_base_func,
void*)GC_ATTR_NONNULL(1);
#define GC_SUCCESS 0
#define GC_DUPLICATE 1
#define GC_NO_THREADS 2
#define GC_UNIMPLEMENTED 3
#define GC_NOT_FOUND 4
#if defined(GC_DARWIN_THREADS)||defined(GC_WIN32_THREADS)
GC_API void GC_CALL GC_use_threads_discovery(void);
#endif
#ifdef GC_THREADS
GC_API void GC_CALL GC_set_suspend_signal(int);
GC_API void GC_CALL GC_set_thr_restart_signal(int);
GC_API int GC_CALL GC_get_suspend_signal(void);
GC_API int GC_CALL GC_get_thr_restart_signal(void);
GC_API void GC_CALL GC_start_mark_threads(void);
GC_API void GC_CALL GC_allow_register_threads(void);
GC_API int GC_CALL GC_register_my_thread(const struct GC_stack_base*)
GC_ATTR_NONNULL(1);
GC_API int GC_CALL GC_thread_is_registered(void);
GC_API void GC_CALL GC_register_altstack(void*,
GC_word,
void*,
GC_word);
GC_API int GC_CALL GC_unregister_my_thread(void);
GC_API void GC_CALL GC_stop_world_external(void);
GC_API void GC_CALL GC_start_world_external(void);
#endif
GC_API void*GC_CALL GC_do_blocking(GC_fn_type,
void*)GC_ATTR_NONNULL(1);
GC_API void*GC_CALL GC_call_with_gc_active(GC_fn_type,
void*)GC_ATTR_NONNULL(1);
GC_API int GC_CALL GC_get_stack_base(struct GC_stack_base*)
GC_ATTR_NONNULL(1);
GC_API void*GC_CALL GC_get_my_stackbottom(struct GC_stack_base*)
GC_ATTR_NONNULL(1);
GC_API void GC_CALL GC_set_stackbottom(void*,
const struct GC_stack_base*)
GC_ATTR_NONNULL(2);
GC_API void*GC_CALL GC_same_obj(void*,void*);
GC_API void*GC_CALL GC_pre_incr(void**,ptrdiff_t)
GC_ATTR_NONNULL(1);
GC_API void*GC_CALL GC_post_incr(void**,ptrdiff_t)
GC_ATTR_NONNULL(1);
GC_API void*GC_CALL GC_is_visible(void*);
GC_API void*GC_CALL GC_is_valid_displacement(void*);
GC_API void GC_CALL GC_dump(void);
GC_API void GC_CALL GC_dump_named(const char*);
GC_API void GC_CALL GC_dump_regions(void);
GC_API void GC_CALL GC_dump_finalization(void);
#if defined(GC_DEBUG)&&defined(__GNUC__)
#define GC_PTR_ADD3(x,n,type_of_result)((type_of_result)GC_same_obj((x)+(n),(x)))
#define GC_PRE_INCR3(x,n,type_of_result)((type_of_result)GC_pre_incr((void**)(&(x)),(n)*sizeof(*x)))
#define GC_POST_INCR3(x,n,type_of_result)((type_of_result)GC_post_incr((void**)(&(x)),(n)*sizeof(*x)))
#define GC_PTR_ADD(x,n)GC_PTR_ADD3(x,n,__typeof__(x))
#define GC_PRE_INCR(x,n)GC_PRE_INCR3(x,n,__typeof__(x))
#define GC_POST_INCR(x)GC_POST_INCR3(x,1,__typeof__(x))
#define GC_POST_DECR(x)GC_POST_INCR3(x,-1,__typeof__(x))
#else
#define GC_PTR_ADD(x,n)((x)+(n))
#define GC_PRE_INCR(x,n)((x)+=(n))
#define GC_POST_INCR(x)((x)++)
#define GC_POST_DECR(x)((x)--)
#endif
#ifdef GC_DEBUG
#define GC_PTR_STORE(p,q)(*(void**)GC_is_visible((void*)(p))=GC_is_valid_displacement((void*)(q)))
#else
#define GC_PTR_STORE(p,q)(*(void**)(p)=(void*)(q))
#endif
GC_API void GC_CALL GC_ptr_store_and_dirty(void*,
const void*);
GC_API void GC_CALL GC_debug_ptr_store_and_dirty(void*,
const void*);
GC_API void (GC_CALLBACK*GC_same_obj_print_proc)(void*,
void*);
GC_API void (GC_CALLBACK*GC_is_valid_displacement_print_proc)(void*);
GC_API void (GC_CALLBACK*GC_is_visible_print_proc)(void*);
#ifdef GC_PTHREADS
#ifdef __cplusplus
}
#endif
#ifndef GC_PTHREAD_REDIRECTS_H
#define GC_PTHREAD_REDIRECTS_H
#if defined(GC_H)&&defined(GC_PTHREADS)
#ifndef GC_PTHREAD_REDIRECTS_ONLY
#include <pthread.h>
#ifndef GC_NO_DLOPEN
#include <dlfcn.h>
#endif
#ifndef GC_NO_PTHREAD_SIGMASK
#include <signal.h>
#endif
#ifdef __cplusplus
extern "C" {
#endif
#ifndef GC_SUSPEND_THREAD_ID
#define GC_SUSPEND_THREAD_ID pthread_t
#endif
#ifndef GC_NO_DLOPEN
GC_API void*GC_dlopen(const char*,int);
#endif
#ifndef GC_NO_PTHREAD_SIGMASK
#if defined(GC_PTHREAD_SIGMASK_NEEDED)||defined(_BSD_SOURCE)||defined(_GNU_SOURCE)||(_POSIX_C_SOURCE>=199506L)||(_XOPEN_SOURCE>=500)
GC_API int GC_pthread_sigmask(int,const sigset_t*,
sigset_t*);
#endif
#endif
#ifndef GC_PTHREAD_CREATE_CONST
#define GC_PTHREAD_CREATE_CONST const
#endif
GC_API int GC_pthread_create(pthread_t*,
GC_PTHREAD_CREATE_CONST pthread_attr_t*,
void*(*)(void*),void*);
GC_API int GC_pthread_join(pthread_t,void**);
GC_API int GC_pthread_detach(pthread_t);
#ifndef GC_NO_PTHREAD_CANCEL
GC_API int GC_pthread_cancel(pthread_t);
#endif
#if defined(GC_HAVE_PTHREAD_EXIT)&&!defined(GC_PTHREAD_EXIT_DECLARED)
#define GC_PTHREAD_EXIT_DECLARED
GC_API void GC_pthread_exit(void*)GC_PTHREAD_EXIT_ATTRIBUTE;
#endif
#ifdef __cplusplus
}
#endif
#endif
#if!defined(GC_NO_THREAD_REDIRECTS)&&!defined(GC_USE_LD_WRAP)
#undef pthread_create
#undef pthread_join
#undef pthread_detach
#define pthread_create GC_pthread_create
#define pthread_join GC_pthread_join
#define pthread_detach GC_pthread_detach
#ifndef GC_NO_PTHREAD_SIGMASK
#undef pthread_sigmask
#define pthread_sigmask GC_pthread_sigmask
#endif
#ifndef GC_NO_DLOPEN
#undef dlopen
#define dlopen GC_dlopen
#endif
#ifndef GC_NO_PTHREAD_CANCEL
#undef pthread_cancel
#define pthread_cancel GC_pthread_cancel
#endif
#ifdef GC_HAVE_PTHREAD_EXIT
#undef pthread_exit
#define pthread_exit GC_pthread_exit
#endif
#endif
#endif
#endif
#ifdef __cplusplus
extern "C" {
#endif
#endif
GC_API GC_ATTR_MALLOC void*GC_CALL GC_malloc_many(size_t);
#define GC_NEXT(p)(*(void**)(p))
typedef int (GC_CALLBACK*GC_has_static_roots_func)(
const char*,
void*,
size_t);
GC_API void GC_CALL GC_register_has_static_roots_callback(
GC_has_static_roots_func);
#if!defined(CPPCHECK)&&!defined(GC_WINDOWS_H_INCLUDED)&&defined(WINAPI)
#define GC_WINDOWS_H_INCLUDED
#endif
#if defined(GC_WIN32_THREADS)&&(!defined(GC_PTHREADS)||defined(GC_BUILD)||defined(GC_WINDOWS_H_INCLUDED))
#if (!defined(GC_NO_THREAD_DECLS)||defined(GC_BUILD))&&!defined(GC_DONT_INCL_WINDOWS_H)
#ifdef __cplusplus
}
#endif
#if!defined(_WIN32_WCE)&&!defined(__CEGCC__)
#include <process.h>
#endif
#if defined(GC_BUILD)||!defined(GC_DONT_INCLUDE_WINDOWS_H)
#include <windows.h>
#define GC_WINDOWS_H_INCLUDED
#endif
#ifdef __cplusplus
extern "C" {
#endif
#ifdef GC_UNDERSCORE_STDCALL
#define GC_CreateThread _GC_CreateThread
#define GC_ExitThread _GC_ExitThread
#endif
#ifndef DECLSPEC_NORETURN
#ifdef GC_WINDOWS_H_INCLUDED
#define DECLSPEC_NORETURN
#else
#define DECLSPEC_NORETURN __declspec(noreturn)
#endif
#endif
#if!defined(_UINTPTR_T)&&!defined(_UINTPTR_T_DEFINED)&&!defined(UINTPTR_MAX)
typedef GC_word GC_uintptr_t;
#else
typedef uintptr_t GC_uintptr_t;
#endif
#ifdef _WIN64
#define GC_WIN32_SIZE_T GC_uintptr_t
#elif defined(GC_WINDOWS_H_INCLUDED)
#define GC_WIN32_SIZE_T DWORD
#else
#define GC_WIN32_SIZE_T unsigned long
#endif
#ifdef GC_INSIDE_DLL
#ifdef GC_UNDERSCORE_STDCALL
#define GC_DllMain _GC_DllMain
#endif
#ifdef GC_WINDOWS_H_INCLUDED
GC_API BOOL WINAPI GC_DllMain(HINSTANCE,
ULONG,
LPVOID);
#else
GC_API int __stdcall GC_DllMain(void*,unsigned long,void*);
#endif
#endif
#ifdef GC_WINDOWS_H_INCLUDED
GC_API HANDLE WINAPI GC_CreateThread(
LPSECURITY_ATTRIBUTES,
GC_WIN32_SIZE_T,
LPTHREAD_START_ROUTINE,
LPVOID,DWORD,
LPDWORD);
GC_API DECLSPEC_NORETURN void WINAPI GC_ExitThread(
DWORD);
#else
struct _SECURITY_ATTRIBUTES;
GC_API void*__stdcall GC_CreateThread(struct _SECURITY_ATTRIBUTES*,
GC_WIN32_SIZE_T,
unsigned long (__stdcall*)(void*),
void*,unsigned long,unsigned long*);
GC_API DECLSPEC_NORETURN void __stdcall GC_ExitThread(unsigned long);
#endif
#if!defined(_WIN32_WCE)&&!defined(__CEGCC__)
GC_API GC_uintptr_t GC_CALL GC_beginthreadex(
void*,unsigned,
unsigned (__stdcall*)(void*),
void*,unsigned,
unsigned*);
GC_API void GC_CALL GC_endthreadex(unsigned);
#endif
#endif
#ifdef GC_WINMAIN_REDIRECT
#define WinMain GC_WinMain
#endif
#define GC_use_DllMain GC_use_threads_discovery
#ifndef GC_NO_THREAD_REDIRECTS
#define CreateThread GC_CreateThread
#define ExitThread GC_ExitThread
#undef _beginthreadex
#define _beginthreadex GC_beginthreadex
#undef _endthreadex
#define _endthreadex GC_endthreadex
#endif
#endif
GC_API void GC_CALL GC_set_force_unmap_on_gcollect(int);
GC_API int GC_CALL GC_get_force_unmap_on_gcollect(void);
#if defined(__CYGWIN32__)||defined(__CYGWIN__)
#ifdef __x86_64__
extern int __data_start__[],__data_end__[];
extern int __bss_start__[],__bss_end__[];
#define GC_DATASTART ((GC_word)__data_start__ < (GC_word)__bss_start__?(void*)__data_start__:(void*)__bss_start__)
#define GC_DATAEND ((GC_word)__data_end__ > (GC_word)__bss_end__?(void*)__data_end__:(void*)__bss_end__)
#else
extern int _data_start__[],_data_end__[],_bss_start__[],_bss_end__[];
#define GC_DATASTART ((GC_word)_data_start__ < (GC_word)_bss_start__?(void*)_data_start__:(void*)_bss_start__)
#define GC_DATAEND ((GC_word)_data_end__ > (GC_word)_bss_end__?(void*)_data_end__:(void*)_bss_end__)
#endif
#define GC_INIT_CONF_ROOTS GC_add_roots(GC_DATASTART,GC_DATAEND);GC_gcollect()
#elif defined(_AIX)
extern int _data[],_end[];
#define GC_DATASTART ((void*)_data)
#define GC_DATAEND ((void*)_end)
#define GC_INIT_CONF_ROOTS GC_add_roots(GC_DATASTART,GC_DATAEND)
#elif (defined(HOST_ANDROID)||defined(__ANDROID__))&&defined(IGNORE_DYNAMIC_LOADING)
#pragma weak __dso_handle
extern int __dso_handle[];
GC_API void*GC_CALL GC_find_limit(void*,int);
#define GC_INIT_CONF_ROOTS (void)(__dso_handle!=0?(GC_add_roots(__dso_handle,GC_find_limit(__dso_handle,1)),0):0)
#else
#define GC_INIT_CONF_ROOTS
#endif
#ifdef GC_DONT_EXPAND
#define GC_INIT_CONF_DONT_EXPAND GC_set_dont_expand(1)
#else
#define GC_INIT_CONF_DONT_EXPAND
#endif
#ifdef GC_FORCE_UNMAP_ON_GCOLLECT
#define GC_INIT_CONF_FORCE_UNMAP_ON_GCOLLECT GC_set_force_unmap_on_gcollect(1)
#else
#define GC_INIT_CONF_FORCE_UNMAP_ON_GCOLLECT
#endif
#ifdef GC_DONT_GC
#define GC_INIT_CONF_MAX_RETRIES (void)(GC_dont_gc=1)
#elif defined(GC_MAX_RETRIES)&&!defined(CPPCHECK)
#define GC_INIT_CONF_MAX_RETRIES GC_set_max_retries(GC_MAX_RETRIES)
#else
#define GC_INIT_CONF_MAX_RETRIES
#endif
#if defined(GC_ALLOCD_BYTES_PER_FINALIZER)&&!defined(CPPCHECK)
#define GC_INIT_CONF_ALLOCD_BYTES_PER_FINALIZER GC_set_allocd_bytes_per_finalizer(GC_ALLOCD_BYTES_PER_FINALIZER)
#else
#define GC_INIT_CONF_ALLOCD_BYTES_PER_FINALIZER
#endif
#if defined(GC_FREE_SPACE_DIVISOR)&&!defined(CPPCHECK)
#define GC_INIT_CONF_FREE_SPACE_DIVISOR GC_set_free_space_divisor(GC_FREE_SPACE_DIVISOR)
#else
#define GC_INIT_CONF_FREE_SPACE_DIVISOR
#endif
#if defined(GC_FULL_FREQ)&&!defined(CPPCHECK)
#define GC_INIT_CONF_FULL_FREQ GC_set_full_freq(GC_FULL_FREQ)
#else
#define GC_INIT_CONF_FULL_FREQ
#endif
#if defined(GC_TIME_LIMIT)&&!defined(CPPCHECK)
#define GC_INIT_CONF_TIME_LIMIT GC_set_time_limit(GC_TIME_LIMIT)
#else
#define GC_INIT_CONF_TIME_LIMIT
#endif
#if defined(GC_MARKERS)&&defined(GC_THREADS)&&!defined(CPPCHECK)
#define GC_INIT_CONF_MARKERS GC_set_markers_count(GC_MARKERS)
#else
#define GC_INIT_CONF_MARKERS
#endif
#if defined(GC_SIG_SUSPEND)&&defined(GC_THREADS)&&!defined(CPPCHECK)
#define GC_INIT_CONF_SUSPEND_SIGNAL GC_set_suspend_signal(GC_SIG_SUSPEND)
#else
#define GC_INIT_CONF_SUSPEND_SIGNAL
#endif
#if defined(GC_SIG_THR_RESTART)&&defined(GC_THREADS)&&!defined(CPPCHECK)
#define GC_INIT_CONF_THR_RESTART_SIGNAL GC_set_thr_restart_signal(GC_SIG_THR_RESTART)
#else
#define GC_INIT_CONF_THR_RESTART_SIGNAL
#endif
#if defined(GC_MAXIMUM_HEAP_SIZE)&&!defined(CPPCHECK)
#define GC_INIT_CONF_MAXIMUM_HEAP_SIZE GC_set_max_heap_size(GC_MAXIMUM_HEAP_SIZE)
#else
#define GC_INIT_CONF_MAXIMUM_HEAP_SIZE
#endif
#ifdef GC_IGNORE_WARN
#define GC_INIT_CONF_IGNORE_WARN GC_set_warn_proc(GC_ignore_warn_proc)
#else
#define GC_INIT_CONF_IGNORE_WARN
#endif
#if defined(GC_INITIAL_HEAP_SIZE)&&!defined(CPPCHECK)
#define GC_INIT_CONF_INITIAL_HEAP_SIZE { size_t heap_size=GC_get_heap_size();if (heap_size < (GC_INITIAL_HEAP_SIZE))(void)GC_expand_hp((GC_INITIAL_HEAP_SIZE)- heap_size);}
#else
#define GC_INIT_CONF_INITIAL_HEAP_SIZE
#endif
#define GC_INIT(){ GC_INIT_CONF_DONT_EXPAND;GC_INIT_CONF_FORCE_UNMAP_ON_GCOLLECT;GC_INIT_CONF_MAX_RETRIES;GC_INIT_CONF_ALLOCD_BYTES_PER_FINALIZER;GC_INIT_CONF_FREE_SPACE_DIVISOR;GC_INIT_CONF_FULL_FREQ;GC_INIT_CONF_TIME_LIMIT;GC_INIT_CONF_MARKERS;GC_INIT_CONF_SUSPEND_SIGNAL;GC_INIT_CONF_THR_RESTART_SIGNAL;GC_INIT_CONF_MAXIMUM_HEAP_SIZE;GC_init();GC_INIT_CONF_ROOTS;GC_INIT_CONF_IGNORE_WARN;GC_INIT_CONF_INITIAL_HEAP_SIZE;}
GC_API void GC_CALL GC_win32_free_heap(void);
#if defined(__SYMBIAN32__)
void GC_init_global_static_roots(void);
#endif
#if defined(_AMIGA)&&!defined(GC_AMIGA_MAKINGLIB)
void*GC_amiga_realloc(void*,size_t);
#define GC_realloc(a,b)GC_amiga_realloc(a,b)
void GC_amiga_set_toany(void (*)(void));
extern int GC_amiga_free_space_divisor_inc;
extern void*(*GC_amiga_allocwrapper_do)(size_t,void*(GC_CALL*)(size_t));
#define GC_malloc(a)(*GC_amiga_allocwrapper_do)(a,GC_malloc)
#define GC_malloc_atomic(a)(*GC_amiga_allocwrapper_do)(a,GC_malloc_atomic)
#define GC_malloc_uncollectable(a)(*GC_amiga_allocwrapper_do)(a,GC_malloc_uncollectable)
#define GC_malloc_atomic_uncollectable(a)(*GC_amiga_allocwrapper_do)(a,GC_malloc_atomic_uncollectable)
#define GC_malloc_ignore_off_page(a)(*GC_amiga_allocwrapper_do)(a,GC_malloc_ignore_off_page)
#define GC_malloc_atomic_ignore_off_page(a)(*GC_amiga_allocwrapper_do)(a,GC_malloc_atomic_ignore_off_page)
#endif
#ifdef __cplusplus
}
#endif
#endif
#endif
#include <stdlib.h>
#if!defined(sony_news)
#include <stddef.h>
#endif
#ifdef DGUX
#include <sys/types.h>
#include <sys/time.h>
#include <sys/resource.h>
#endif
#ifdef BSD_TIME
#include <sys/types.h>
#include <sys/time.h>
#include <sys/resource.h>
#endif
#ifdef PARALLEL_MARK
#define AO_REQUIRE_CAS
#if!defined(__GNUC__)&&!defined(AO_ASSUME_WINDOWS98)
#define AO_ASSUME_WINDOWS98
#endif
#endif
#ifndef GC_TINY_FL_H
#define GC_TINY_FL_H
#ifndef GC_GRANULE_BYTES
#if defined(__LP64__)||defined (_LP64)||defined(_WIN64)||defined(__s390x__)||(defined(__x86_64__)&&!defined(__ILP32__))||defined(__alpha__)||defined(__powerpc64__)||defined(__arch64__)
#define GC_GRANULE_BYTES 16
#define GC_GRANULE_WORDS 2
#else
#define GC_GRANULE_BYTES 8
#define GC_GRANULE_WORDS 2
#endif
#endif
#if GC_GRANULE_WORDS==2
#define GC_WORDS_TO_GRANULES(n)((n)>>1)
#else
#define GC_WORDS_TO_GRANULES(n)((n)*sizeof(void*)/GC_GRANULE_BYTES)
#endif
#ifndef GC_TINY_FREELISTS
#if GC_GRANULE_BYTES==16
#define GC_TINY_FREELISTS 25
#else
#define GC_TINY_FREELISTS 33
#endif
#endif
#define GC_RAW_BYTES_FROM_INDEX(i)((i)*GC_GRANULE_BYTES)
#endif
#ifndef GC_MARK_H
#define GC_MARK_H
#ifndef GC_H
#endif
#ifdef __cplusplus
extern "C" {
#endif
#define GC_PROC_BYTES 100
#if defined(GC_BUILD)||defined(NOT_GCBUILD)
struct GC_ms_entry;
#else
struct GC_ms_entry { void*opaque;};
#endif
typedef struct GC_ms_entry*(*GC_mark_proc)(GC_word*,
struct GC_ms_entry*,
struct GC_ms_entry*,
GC_word);
#define GC_LOG_MAX_MARK_PROCS 6
#define GC_MAX_MARK_PROCS (1<<GC_LOG_MAX_MARK_PROCS)
#define GC_RESERVED_MARK_PROCS 8
#define GC_GCJ_RESERVED_MARK_PROC_INDEX 0
#define GC_DS_TAG_BITS 2
#define GC_DS_TAGS ((1<<GC_DS_TAG_BITS)- 1)
#define GC_DS_LENGTH 0
#define GC_DS_BITMAP 1
#define GC_DS_PROC 2
#define GC_MAKE_PROC(proc_index,env)(((((env)<<GC_LOG_MAX_MARK_PROCS)|(proc_index))<<GC_DS_TAG_BITS)|GC_DS_PROC)
#define GC_DS_PER_OBJECT 3
#define GC_INDIR_PER_OBJ_BIAS 0x10
GC_API void*GC_least_plausible_heap_addr;
GC_API void*GC_greatest_plausible_heap_addr;
GC_API struct GC_ms_entry*GC_CALL GC_mark_and_push(void*,
struct GC_ms_entry*,
struct GC_ms_entry*,
void**);
#define GC_MARK_AND_PUSH(obj,msp,lim,src)((GC_word)(obj)>=(GC_word)GC_least_plausible_heap_addr&&(GC_word)(obj)<=(GC_word)GC_greatest_plausible_heap_addr?GC_mark_and_push(obj,msp,lim,src):(msp))
GC_API GC_ATTR_CONST size_t GC_CALL GC_get_debug_header_size(void);
#define GC_USR_PTR_FROM_BASE(p)((void*)((char*)(p)+GC_get_debug_header_size()))
GC_API GC_ATTR_DEPRECATED
#ifdef GC_BUILD
const
#endif
size_t GC_debug_header_size;
GC_API void**GC_CALL GC_new_free_list(void);
GC_API void**GC_CALL GC_new_free_list_inner(void);
GC_API unsigned GC_CALL GC_new_kind(void**,
GC_word,
int,
int)GC_ATTR_NONNULL(1);
GC_API unsigned GC_CALL GC_new_kind_inner(void**,
GC_word,
int,
int)GC_ATTR_NONNULL(1);
GC_API unsigned GC_CALL GC_new_proc(GC_mark_proc);
GC_API unsigned GC_CALL GC_new_proc_inner(GC_mark_proc);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL GC_generic_malloc(
size_t,
int);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_generic_malloc_ignore_off_page(
size_t,int);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_generic_malloc_uncollectable(
size_t,int);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_generic_or_special_malloc(
size_t,int);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_debug_generic_or_special_malloc(
size_t,int,
GC_EXTRA_PARAMS);
#ifdef GC_DEBUG
#define GC_GENERIC_OR_SPECIAL_MALLOC(sz,knd)GC_debug_generic_or_special_malloc(sz,knd,GC_EXTRAS)
#else
#define GC_GENERIC_OR_SPECIAL_MALLOC(sz,knd)GC_generic_or_special_malloc(sz,knd)
#endif
GC_API int GC_CALL GC_get_kind_and_size(const void*,size_t*)
GC_ATTR_NONNULL(1);
typedef void (GC_CALLBACK*GC_describe_type_fn)(void*,
char*);
#define GC_TYPE_DESCR_LEN 40
GC_API void GC_CALL GC_register_describe_type_fn(int,
GC_describe_type_fn);
GC_API void*GC_CALL GC_clear_stack(void*);
typedef void (GC_CALLBACK*GC_start_callback_proc)(void);
GC_API void GC_CALL GC_set_start_callback(GC_start_callback_proc);
GC_API GC_start_callback_proc GC_CALL GC_get_start_callback(void);
GC_API int GC_CALL GC_is_marked(const void*)GC_ATTR_NONNULL(1);
GC_API void GC_CALL GC_clear_mark_bit(const void*)GC_ATTR_NONNULL(1);
GC_API void GC_CALL GC_set_mark_bit(const void*)GC_ATTR_NONNULL(1);
GC_API void GC_CALL GC_push_all(void*,void*);
GC_API void GC_CALL GC_push_all_eager(void*,void*);
GC_API void GC_CALL GC_push_conditional(void*,void*,
int);
GC_API void GC_CALL GC_push_finalizer_structures(void);
typedef void (GC_CALLBACK*GC_push_other_roots_proc)(void);
GC_API void GC_CALL GC_set_push_other_roots(GC_push_other_roots_proc);
GC_API GC_push_other_roots_proc GC_CALL GC_get_push_other_roots(void);
typedef void (GC_CALLBACK*GC_reachable_object_proc)(void*,
size_t,
void*);
GC_API void GC_CALL GC_enumerate_reachable_objects_inner(
GC_reachable_object_proc,
void*)GC_ATTR_NONNULL(1);
GC_API int GC_CALL GC_is_tmp_root(void*);
GC_API void GC_CALL GC_print_trace(GC_word);
GC_API void GC_CALL GC_print_trace_inner(GC_word);
#ifdef __cplusplus
}
#endif
#endif
typedef GC_word word;
typedef GC_signed_word signed_word;
typedef unsigned int unsigned32;
typedef int GC_bool;
#define TRUE 1
#define FALSE 0
#ifndef PTR_T_DEFINED
typedef char*ptr_t;
#define PTR_T_DEFINED
#endif
#ifndef SIZE_MAX
#include <limits.h>
#endif
#if defined(SIZE_MAX)&&!defined(CPPCHECK)
#define GC_SIZE_MAX ((size_t)SIZE_MAX)
#else
#define GC_SIZE_MAX (~(size_t)0)
#endif
#if GC_GNUC_PREREQ(3,0)&&!defined(LINT2)
#define EXPECT(expr,outcome)__builtin_expect(expr,outcome)
#else
#define EXPECT(expr,outcome)(expr)
#endif
#define SIZET_SAT_ADD(a,b)(EXPECT((a)< GC_SIZE_MAX - (b),TRUE)?(a)+(b):GC_SIZE_MAX)
#ifndef GCCONFIG_H
#define GCCONFIG_H
#ifdef CPPCHECK
#undef CLOCKS_PER_SEC
#undef FIXUP_POINTER
#undef POINTER_MASK
#undef POINTER_SHIFT
#undef REDIRECT_REALLOC
#undef _MAX_PATH
#endif
#ifndef PTR_T_DEFINED
typedef char*ptr_t;
#define PTR_T_DEFINED
#endif
#if!defined(sony_news)
#include <stddef.h>
#endif
#ifdef __cplusplus
#define EXTERN_C_BEGIN extern "C" {
#define EXTERN_C_END }
#else
#define EXTERN_C_BEGIN
#define EXTERN_C_END
#endif
EXTERN_C_BEGIN
#if defined(__clang__)&&defined(__clang_major__)
#define GC_CLANG_PREREQ(major,minor)((__clang_major__<<16)+__clang_minor__>=((major)<<16)+(minor))
#define GC_CLANG_PREREQ_FULL(major,minor,patchlevel)(GC_CLANG_PREREQ(major,(minor)+1)||(__clang_major__==(major)&&__clang_minor__==(minor)&&__clang_patchlevel__>=(patchlevel)))
#else
#define GC_CLANG_PREREQ(major,minor)0
#define GC_CLANG_PREREQ_FULL(major,minor,patchlevel)0
#endif
#ifdef LINT2
#define COVERT_DATAFLOW(w)(~(GC_word)(w)^(~(GC_word)0))
#else
#define COVERT_DATAFLOW(w)((GC_word)(w))
#endif
#if defined(__ANDROID__)&&!defined(HOST_ANDROID)
#define HOST_ANDROID 1
#endif
#if defined(TIZEN)&&!defined(HOST_TIZEN)
#define HOST_TIZEN 1
#endif
#if defined(__SYMBIAN32__)&&!defined(SYMBIAN)
#define SYMBIAN
#ifdef __WINS__
#pragma data_seg(".data2")
#endif
#endif
#if (defined(linux)||defined(__linux__)||defined(HOST_ANDROID))&&!defined(LINUX)&&!defined(__native_client__)
#define LINUX
#endif
#if defined(__NetBSD__)
#define NETBSD
#endif
#if defined(__OpenBSD__)
#define OPENBSD
#endif
#if (defined(__FreeBSD__)||defined(__DragonFly__)||defined(__FreeBSD_kernel__))&&!defined(FREEBSD)&&!defined(SN_TARGET_ORBIS)
#define FREEBSD
#endif
#if defined(macosx)||(defined(__APPLE__)&&defined(__MACH__))
#define DARWIN
EXTERN_C_END
#include <TargetConditionals.h>
EXTERN_C_BEGIN
#endif
#if defined(__native_client__)
#define NACL
#if!defined(__portable_native_client__)&&!defined(__arm__)
#define I386
#define mach_type_known
#else
#endif
#endif
#if defined(__aarch64__)
#define AARCH64
#if!defined(LINUX)&&!defined(DARWIN)&&!defined(FREEBSD)&&!defined(NETBSD)&&!defined(NN_BUILD_TARGET_PLATFORM_NX)&&!defined(OPENBSD)
#define NOSYS
#define mach_type_known
#endif
#endif
#if defined(__arm)||defined(__arm__)||defined(__thumb__)
#define ARM32
#if defined(NACL)
#define mach_type_known
#elif!defined(LINUX)&&!defined(NETBSD)&&!defined(FREEBSD)&&!defined(OPENBSD)&&!defined(DARWIN)&&!defined(_WIN32)&&!defined(__CEGCC__)&&!defined(NN_PLATFORM_CTR)&&!defined(SN_TARGET_ORBIS)&&!defined(SN_TARGET_PSP2)&&!defined(SYMBIAN)
#define NOSYS
#define mach_type_known
#endif
#endif
#if defined(sun)&&defined(mc68000)&&!defined(CPPCHECK)
#error SUNOS4 no longer supported
#endif
#if defined(hp9000s300)&&!defined(CPPCHECK)
#error M68K based HP machines no longer supported
#endif
#if defined(OPENBSD)&&defined(m68k)
#define M68K
#define mach_type_known
#endif
#if defined(OPENBSD)&&defined(__sparc__)
#define SPARC
#define mach_type_known
#endif
#if defined(OPENBSD)&&defined(__arm__)
#define ARM32
#define mach_type_known
#endif
#if defined(OPENBSD)&&defined(__aarch64__)
#define AARCH64
#define mach_type_known
#endif
#if defined(OPENBSD)&&defined(__sh__)
#define SH
#define mach_type_known
#endif
#if defined(NETBSD)&&(defined(m68k)||defined(__m68k__))
#define M68K
#define mach_type_known
#endif
#if defined(NETBSD)&&defined(__powerpc__)
#define POWERPC
#define mach_type_known
#endif
#if defined(NETBSD)&&(defined(__arm32__)||defined(__arm__))
#define ARM32
#define mach_type_known
#endif
#if defined(NETBSD)&&defined(__aarch64__)
#define AARCH64
#define mach_type_known
#endif
#if defined(NETBSD)&&defined(__sh__)
#define SH
#define mach_type_known
#endif
#if defined(vax)||defined(__vax__)
#define VAX
#ifdef ultrix
#define ULTRIX
#else
#define BSD
#endif
#define mach_type_known
#endif
#if defined(NETBSD)&&defined(__vax__)
#define VAX
#define mach_type_known
#endif
#if defined(mips)||defined(__mips)||defined(_mips)
#define MIPS
#if defined(nec_ews)||defined(_nec_ews)
#define EWS4800
#endif
#if!defined(LINUX)&&!defined(EWS4800)&&!defined(NETBSD)&&!defined(OPENBSD)
#if defined(ultrix)||defined(__ultrix)
#define ULTRIX
#else
#define IRIX5
#endif
#endif
#if defined(NETBSD)&&defined(__MIPSEL__)
#undef ULTRIX
#endif
#define mach_type_known
#endif
#if defined(__QNX__)
#define I386
#define mach_type_known
#endif
#if defined(__NIOS2__)||defined(__NIOS2)||defined(__nios2__)
#define NIOS2
#define mach_type_known
#endif
#if defined(__or1k__)
#define OR1K
#define mach_type_known
#endif
#if defined(DGUX)&&(defined(i386)||defined(__i386__))
#define I386
#ifndef _USING_DGUX
#define _USING_DGUX
#endif
#define mach_type_known
#endif
#if defined(sequent)&&(defined(i386)||defined(__i386__))
#define I386
#define SEQUENT
#define mach_type_known
#endif
#if (defined(sun)||defined(__sun))&&(defined(i386)||defined(__i386__))
#define I386
#define SOLARIS
#define mach_type_known
#endif
#if (defined(sun)||defined(__sun))&&defined(__amd64)
#define X86_64
#define SOLARIS
#define mach_type_known
#endif
#if (defined(__OS2__)||defined(__EMX__))&&defined(__32BIT__)
#define I386
#define OS2
#define mach_type_known
#endif
#if defined(ibm032)&&!defined(CPPCHECK)
#error IBM PC/RT no longer supported
#endif
#if (defined(sun)||defined(__sun))&&(defined(sparc)||defined(__sparc))
EXTERN_C_END
#include <errno.h>
EXTERN_C_BEGIN
#define SPARC
#define SOLARIS
#define mach_type_known
#elif defined(sparc)&&defined(unix)&&!defined(sun)&&!defined(linux)&&!defined(FREEBSD)&&!defined(NETBSD)&&!defined(OPENBSD)
#define SPARC
#define DRSNX
#define mach_type_known
#endif
#if defined(_IBMR2)
#define POWERPC
#define AIX
#define mach_type_known
#endif
#if defined(NETBSD)&&defined(__sparc__)
#define SPARC
#define mach_type_known
#endif
#if defined(_M_XENIX)&&defined(_M_SYSV)&&defined(_M_I386)
#define I386
#if defined(_SCO_ELF)
#define SCO_ELF
#else
#define SCO
#endif
#define mach_type_known
#endif
#if defined(_AUX_SOURCE)&&!defined(CPPCHECK)
#error A/UX no longer supported
#endif
#if defined(_PA_RISC1_0)||defined(_PA_RISC1_1)||defined(_PA_RISC2_0)||defined(hppa)||defined(__hppa__)
#define HP_PA
#if!defined(LINUX)&&!defined(HPUX)&&!defined(OPENBSD)
#define HPUX
#endif
#define mach_type_known
#endif
#if defined(__ia64)&&(defined(_HPUX_SOURCE)||defined(__HP_aCC))
#define IA64
#ifndef HPUX
#define HPUX
#endif
#define mach_type_known
#endif
#if (defined(__BEOS__)||defined(__HAIKU__))&&defined(_X86_)
#define I386
#define HAIKU
#define mach_type_known
#endif
#if defined(__HAIKU__)&&(defined(__amd64__)||defined(__x86_64__))
#define X86_64
#define HAIKU
#define mach_type_known
#endif
#if defined(OPENBSD)&&defined(__amd64__)
#define X86_64
#define mach_type_known
#endif
#if defined(LINUX)&&(defined(i386)||defined(__i386__))
#define I386
#define mach_type_known
#endif
#if defined(LINUX)&&defined(__x86_64__)
#define X86_64
#define mach_type_known
#endif
#if defined(LINUX)&&(defined(__ia64__)||defined(__ia64))
#define IA64
#define mach_type_known
#endif
#if defined(LINUX)&&defined(__aarch64__)
#define AARCH64
#define mach_type_known
#endif
#if defined(LINUX)&&(defined(__arm)||defined(__arm__))
#define ARM32
#define mach_type_known
#endif
#if defined(LINUX)&&defined(__cris__)
#ifndef CRIS
#define CRIS
#endif
#define mach_type_known
#endif
#if defined(LINUX)&&(defined(powerpc)||defined(__powerpc__)||defined(powerpc64)||defined(__powerpc64__))
#define POWERPC
#define mach_type_known
#endif
#if defined(LINUX)&&defined(__mc68000__)
#define M68K
#define mach_type_known
#endif
#if defined(LINUX)&&(defined(sparc)||defined(__sparc__))
#define SPARC
#define mach_type_known
#endif
#if defined(LINUX)&&defined(__sh__)
#define SH
#define mach_type_known
#endif
#if defined(LINUX)&&defined(__avr32__)
#define AVR32
#define mach_type_known
#endif
#if defined(LINUX)&&defined(__m32r__)
#define M32R
#define mach_type_known
#endif
#if defined(__alpha)||defined(__alpha__)
#define ALPHA
#if!defined(LINUX)&&!defined(NETBSD)&&!defined(OPENBSD)&&!defined(FREEBSD)
#define OSF1
#endif
#define mach_type_known
#endif
#if defined(_AMIGA)&&!defined(AMIGA)
#define AMIGA
#endif
#ifdef AMIGA
#define M68K
#define mach_type_known
#endif
#if defined(THINK_C)||(defined(__MWERKS__)&&!defined(__powerc)&&!defined(SYMBIAN))
#define M68K
#define MACOS
#define mach_type_known
#endif
#if defined(__MWERKS__)&&defined(__powerc)&&!defined(__MACH__)&&!defined(SYMBIAN)
#define POWERPC
#define MACOS
#define mach_type_known
#endif
#if defined(OPENBSD)&&defined(__powerpc__)
#define POWERPC
#define mach_type_known
#endif
#if defined(DARWIN)
#if defined(__ppc__)||defined(__ppc64__)
#define POWERPC
#define mach_type_known
#elif defined(__x86_64__)||defined(__x86_64)
#define X86_64
#define mach_type_known
#elif defined(__i386__)
#define I386
#define mach_type_known
#elif defined(__arm__)
#define ARM32
#define mach_type_known
#elif defined(__aarch64__)
#define AARCH64
#define mach_type_known
#endif
#endif
#if defined(__rtems__)&&(defined(i386)||defined(__i386__))
#define I386
#define RTEMS
#define mach_type_known
#endif
#if defined(NeXT)&&defined(mc68000)
#define M68K
#define NEXT
#define mach_type_known
#endif
#if defined(NeXT)&&(defined(i386)||defined(__i386__))
#define I386
#define NEXT
#define mach_type_known
#endif
#if defined(OPENBSD)&&(defined(i386)||defined(__i386__))
#define I386
#define mach_type_known
#endif
#if defined(NETBSD)&&(defined(i386)||defined(__i386__))
#define I386
#define mach_type_known
#endif
#if defined(NETBSD)&&defined(__x86_64__)
#define X86_64
#define mach_type_known
#endif
#if defined(FREEBSD)&&(defined(i386)||defined(__i386__))
#define I386
#define mach_type_known
#endif
#if (defined(FREEBSD)||defined(SN_TARGET_ORBIS))&&(defined(__amd64__)||defined(__x86_64__))
#define X86_64
#define mach_type_known
#endif
#if defined(FREEBSD)&&defined(__sparc__)
#define SPARC
#define mach_type_known
#endif
#if defined(FREEBSD)&&(defined(powerpc)||defined(__powerpc__))
#define POWERPC
#define mach_type_known
#endif
#if defined(FREEBSD)&&defined(__arm__)
#define ARM32
#define mach_type_known
#endif
#if defined(FREEBSD)&&defined(__aarch64__)
#define AARCH64
#define mach_type_known
#endif
#if defined(FREEBSD)&&(defined(mips)||defined(__mips)||defined(_mips))
#define MIPS
#define mach_type_known
#endif
#if defined(bsdi)&&(defined(i386)||defined(__i386__))
#define I386
#define BSDI
#define mach_type_known
#endif
#if!defined(mach_type_known)&&defined(__386BSD__)
#define I386
#define THREE86BSD
#define mach_type_known
#endif
#if defined(_CX_UX)&&defined(_M88K)
#define M88K
#define CX_UX
#define mach_type_known
#endif
#if defined(DGUX)&&defined(m88k)
#define M88K
#define mach_type_known
#endif
#if defined(_WIN32_WCE)||defined(__CEGCC__)||defined(__MINGW32CE__)
#if defined(SH3)||defined(SH4)
#define SH
#endif
#if defined(x86)||defined(__i386__)
#define I386
#endif
#if defined(_M_ARM)||defined(ARM)||defined(_ARM_)
#define ARM32
#endif
#define MSWINCE
#define mach_type_known
#else
#if ((defined(_MSDOS)||defined(_MSC_VER))&&(_M_IX86>=300))||(defined(_WIN32)&&!defined(__CYGWIN32__)&&!defined(__CYGWIN__)&&!defined(__INTERIX)&&!defined(SYMBIAN))
#if defined(__LP64__)||defined(_M_X64)
#define X86_64
#elif defined(_M_ARM)
#define ARM32
#elif defined(_M_ARM64)
#define AARCH64
#else
#define I386
#endif
#ifdef _XBOX_ONE
#define MSWIN_XBOX1
#else
#ifndef MSWIN32
#define MSWIN32
#endif
#if defined(WINAPI_FAMILY)&&(WINAPI_FAMILY==WINAPI_FAMILY_APP)
#define MSWINRT_FLAVOR
#endif
#endif
#define mach_type_known
#endif
#if defined(_MSC_VER)&&defined(_M_IA64)
#define IA64
#define MSWIN32
#endif
#endif
#if defined(__DJGPP__)
#define I386
#ifndef DJGPP
#define DJGPP
#endif
#define mach_type_known
#endif
#if defined(__CYGWIN32__)||defined(__CYGWIN__)
#if defined(__LP64__)
#define X86_64
#else
#define I386
#endif
#define CYGWIN32
#define mach_type_known
#endif
#if defined(__INTERIX)
#define I386
#define INTERIX
#define mach_type_known
#endif
#if defined(__MINGW32__)&&!defined(mach_type_known)
#define I386
#define MSWIN32
#define mach_type_known
#endif
#if defined(__BORLANDC__)
#define I386
#define MSWIN32
#define mach_type_known
#endif
#if defined(_UTS)&&!defined(mach_type_known)
#define S370
#define UTS4
#define mach_type_known
#endif
#if defined(__pj__)&&!defined(CPPCHECK)
#error PicoJava no longer supported
#endif
#if defined(__embedded__)&&defined(PPC)
#define POWERPC
#define NOSYS
#define mach_type_known
#endif
#if defined(__WATCOMC__)&&defined(__386__)
#define I386
#if!defined(OS2)&&!defined(MSWIN32)&&!defined(DOS4GW)
#if defined(__OS2__)
#define OS2
#else
#if defined(__WINDOWS_386__)||defined(__NT__)
#define MSWIN32
#else
#define DOS4GW
#endif
#endif
#endif
#define mach_type_known
#endif
#if defined(__s390__)&&defined(LINUX)
#define S390
#define mach_type_known
#endif
#if defined(__GNU__)
#if defined(__i386__)
#define HURD
#define I386
#define mach_type_known
#endif
#endif
#if defined(__TANDEM)
#define MIPS
#define NONSTOP
#define mach_type_known
#endif
#if defined(__arc__)&&defined(LINUX)
#define ARC
#define mach_type_known
#endif
#if defined(__hexagon__)&&defined(LINUX)
#define HEXAGON
#define mach_type_known
#endif
#if defined(__tile__)&&defined(LINUX)
#ifdef __tilegx__
#define TILEGX
#else
#define TILEPRO
#endif
#define mach_type_known
#endif
#if defined(__riscv)&&(defined(FREEBSD)||defined(LINUX))
#define RISCV
#define mach_type_known
#endif
#if defined(SN_TARGET_PSP2)
#define mach_type_known
#endif
#if defined(NN_PLATFORM_CTR)
#define mach_type_known
#endif
#if defined(NN_BUILD_TARGET_PLATFORM_NX)
#define NINTENDO_SWITCH
#define mach_type_known
#endif
#if defined(SYMBIAN)
#define mach_type_known
#endif
#if defined(__EMSCRIPTEN__)
#define I386
#define mach_type_known
#endif
#if!defined(mach_type_known)&&!defined(CPPCHECK)
#error The collector has not been ported to this machine/OS combination
#endif
#if GC_GNUC_PREREQ(2,8)&&!defined(__INTEL_COMPILER)&&!defined(__PATHCC__)&&!defined(__FUJITSU)&&!(defined(POWERPC)&&defined(DARWIN))&&!defined(RTEMS)&&!defined(__ARMCC_VERSION)&&!defined(__clang__)
#define HAVE_BUILTIN_UNWIND_INIT
#endif
#ifdef SYMBIAN
#define MACH_TYPE "SYMBIAN"
#define OS_TYPE "SYMBIAN"
#define CPP_WORDSZ 32
#define ALIGNMENT 4
#define DATASTART (ptr_t)ALIGNMENT
#define DATAEND (ptr_t)ALIGNMENT
#endif
#define STACK_GRAN 0x1000000
#ifdef M68K
#define MACH_TYPE "M68K"
#define ALIGNMENT 2
#ifdef OPENBSD
#define OS_TYPE "OPENBSD"
#define HEURISTIC2
#ifdef __ELF__
extern ptr_t GC_data_start;
#define DATASTART GC_data_start
#define DYNAMIC_LOADING
#else
extern char etext[];
#define DATASTART ((ptr_t)(etext))
#endif
#endif
#ifdef NETBSD
#define OS_TYPE "NETBSD"
#define HEURISTIC2
#ifdef __ELF__
extern ptr_t GC_data_start;
#define DATASTART GC_data_start
#define DYNAMIC_LOADING
#else
extern char etext[];
#define DATASTART ((ptr_t)(etext))
#endif
#endif
#ifdef LINUX
#define OS_TYPE "LINUX"
#define LINUX_STACKBOTTOM
#define COUNT_UNMAPPED_REGIONS
#if!defined(REDIRECT_MALLOC)
#define MPROTECT_VDB
#endif
#ifdef __ELF__
#define DYNAMIC_LOADING
EXTERN_C_END
#include <features.h>
EXTERN_C_BEGIN
#if defined(__GLIBC__)&&__GLIBC__>=2
#define SEARCH_FOR_DATA_START
#else
extern char**__environ;
#define DATASTART ((ptr_t)(&__environ))
#endif
extern int _end[];
#define DATAEND ((ptr_t)(_end))
#else
extern int etext[];
#define DATASTART ((ptr_t)((((word)(etext))+0xfff)&~0xfff))
#endif
#endif
#ifdef AMIGA
#define OS_TYPE "AMIGA"
#define DATAEND
#define GETPAGESIZE()4096
#endif
#ifdef MACOS
#ifndef __LOWMEM__
EXTERN_C_END
#include <LowMem.h>
EXTERN_C_BEGIN
#endif
#define OS_TYPE "MACOS"
#define STACKBOTTOM ((ptr_t)LMGetCurStackBase())
#define DATAEND
#define GETPAGESIZE()4096
#endif
#ifdef NEXT
#define OS_TYPE "NEXT"
#define DATASTART ((ptr_t)get_etext())
#define DATASTART_IS_FUNC
#define STACKBOTTOM ((ptr_t)0x4000000)
#define DATAEND
#endif
#endif
#ifdef POWERPC
#define MACH_TYPE "POWERPC"
#ifdef MACOS
#define ALIGNMENT 2
#ifndef __LOWMEM__
EXTERN_C_END
#include <LowMem.h>
EXTERN_C_BEGIN
#endif
#define OS_TYPE "MACOS"
#define STACKBOTTOM ((ptr_t)LMGetCurStackBase())
#define DATAEND
#endif
#ifdef LINUX
#if defined(__powerpc64__)
#define ALIGNMENT 8
#define CPP_WORDSZ 64
#ifndef HBLKSIZE
#define HBLKSIZE 4096
#endif
#else
#define ALIGNMENT 4
#endif
#define OS_TYPE "LINUX"
#if defined(__bg__)
#define HEURISTIC2
#define NO_PTHREAD_GETATTR_NP
#else
#define LINUX_STACKBOTTOM
#endif
#define COUNT_UNMAPPED_REGIONS
#define DYNAMIC_LOADING
#define SEARCH_FOR_DATA_START
extern int _end[];
#define DATAEND ((ptr_t)(_end))
#endif
#ifdef DARWIN
#define OS_TYPE "DARWIN"
#define DYNAMIC_LOADING
#if defined(__ppc64__)
#define ALIGNMENT 8
#define CPP_WORDSZ 64
#define STACKBOTTOM ((ptr_t)0x7fff5fc00000)
#define CACHE_LINE_SIZE 64
#ifndef HBLKSIZE
#define HBLKSIZE 4096
#endif
#else
#define ALIGNMENT 4
#define STACKBOTTOM ((ptr_t)0xc0000000)
#endif
#define DATASTART ((ptr_t)get_etext())
#define DATAEND ((ptr_t)get_end())
#define USE_MMAP_ANON
#define MPROTECT_VDB
EXTERN_C_END
#include <unistd.h>
EXTERN_C_BEGIN
#define GETPAGESIZE()(unsigned)getpagesize()
#if defined(USE_PPC_PREFETCH)&&defined(__GNUC__)
#define PREFETCH(x)__asm__ __volatile__ ("dcbt 0,%0"::"r" ((const void*)(x)))
#define GC_PREFETCH_FOR_WRITE(x)__asm__ __volatile__ ("dcbtst 0,%0"::"r" ((const void*)(x)))
#endif
#define NO_PTHREAD_TRYLOCK
#endif
#ifdef OPENBSD
#define OS_TYPE "OPENBSD"
#if defined(__powerpc64__)
#define ALIGNMENT 8
#define CPP_WORDSZ 64
#else
#define ALIGNMENT 4
#endif
#ifndef GC_OPENBSD_THREADS
#define HEURISTIC2
#endif
extern int __data_start[];
#define DATASTART ((ptr_t)__data_start)
extern int _end[];
#define DATAEND ((ptr_t)(&_end))
#define DYNAMIC_LOADING
#endif
#ifdef FREEBSD
#if defined(__powerpc64__)
#define ALIGNMENT 8
#define CPP_WORDSZ 64
#ifndef HBLKSIZE
#define HBLKSIZE 4096
#endif
#else
#define ALIGNMENT 4
#endif
#define OS_TYPE "FREEBSD"
#ifndef GC_FREEBSD_THREADS
#define MPROTECT_VDB
#endif
#define SIG_SUSPEND SIGUSR1
#define SIG_THR_RESTART SIGUSR2
#define FREEBSD_STACKBOTTOM
#define DYNAMIC_LOADING
extern char etext[];
#define DATASTART GC_FreeBSDGetDataStart(0x1000,(ptr_t)etext)
#define DATASTART_USES_BSDGETDATASTART
#endif
#ifdef NETBSD
#define ALIGNMENT 4
#define OS_TYPE "NETBSD"
#define HEURISTIC2
extern ptr_t GC_data_start;
#define DATASTART GC_data_start
#define DYNAMIC_LOADING
#endif
#ifdef SN_TARGET_PS3
#define OS_TYPE "SN_TARGET_PS3"
#define NO_GETENV
#define CPP_WORDSZ 32
#define ALIGNMENT 4
extern int _end[];
extern int __bss_start;
#define DATASTART ((ptr_t)(__bss_start))
#define DATAEND ((ptr_t)(_end))
#define STACKBOTTOM ((ptr_t)ps3_get_stack_bottom())
#define NO_PTHREAD_TRYLOCK
#endif
#ifdef AIX
#define OS_TYPE "AIX"
#undef ALIGNMENT
#undef IA64
#ifdef __64BIT__
#define ALIGNMENT 8
#define CPP_WORDSZ 64
#define STACKBOTTOM ((ptr_t)0x1000000000000000)
#else
#define ALIGNMENT 4
#define CPP_WORDSZ 32
#define STACKBOTTOM ((ptr_t)((ulong)&errno))
#endif
#define USE_MMAP_ANON
extern int _data[],_end[];
#define DATASTART ((ptr_t)((ulong)_data))
#define DATAEND ((ptr_t)((ulong)_end))
extern int errno;
#define DYNAMIC_LOADING
#endif
#ifdef NOSYS
#define ALIGNMENT 4
#define OS_TYPE "NOSYS"
extern void __end[],__dso_handle[];
#define DATASTART ((ptr_t)__dso_handle)
#define DATAEND ((ptr_t)(__end))
#undef STACK_GRAN
#define STACK_GRAN 0x10000000
#define HEURISTIC1
#endif
#endif
#ifdef NACL
#define OS_TYPE "NACL"
#if defined(__GLIBC__)
#define DYNAMIC_LOADING
#endif
#define DATASTART ((ptr_t)0x10020000)
extern int _end[];
#define DATAEND ((ptr_t)_end)
#undef STACK_GRAN
#define STACK_GRAN 0x10000
#define HEURISTIC1
#define NO_PTHREAD_GETATTR_NP
#define USE_MMAP_ANON
#define GETPAGESIZE()65536
#define MAX_NACL_GC_THREADS 1024
#endif
#ifdef VAX
#define MACH_TYPE "VAX"
#define ALIGNMENT 4
extern char etext[];
#define DATASTART ((ptr_t)(etext))
#ifdef BSD
#define OS_TYPE "BSD"
#define HEURISTIC1
#endif
#ifdef ULTRIX
#define OS_TYPE "ULTRIX"
#define STACKBOTTOM ((ptr_t)0x7fffc800)
#endif
#endif
#ifdef SPARC
#define MACH_TYPE "SPARC"
#if defined(__arch64__)||defined(__sparcv9)
#define ALIGNMENT 8
#define CPP_WORDSZ 64
#define ELF_CLASS ELFCLASS64
#else
#define ALIGNMENT 4
#define CPP_WORDSZ 32
#endif
#ifdef SOLARIS
#define OS_TYPE "SOLARIS"
extern int _etext[];
extern int _end[];
ptr_t GC_SysVGetDataStart(size_t,ptr_t);
#define DATASTART GC_SysVGetDataStart(0x10000,(ptr_t)_etext)
#define DATASTART_IS_FUNC
#define DATAEND ((ptr_t)(_end))
#if!defined(USE_MMAP)&&defined(REDIRECT_MALLOC)
#define USE_MMAP 1
#endif
#ifdef USE_MMAP
#define HEAP_START (ptr_t)0x40000000
#else
#define HEAP_START DATAEND
#endif
#define PROC_VDB
EXTERN_C_END
#include <sys/vmparam.h>
#include <unistd.h>
EXTERN_C_BEGIN
#ifdef USERLIMIT
#define STACKBOTTOM ((ptr_t)USRSTACK)
#else
#define HEURISTIC2
#endif
#define GETPAGESIZE()(unsigned)sysconf(_SC_PAGESIZE)
#define DYNAMIC_LOADING
#endif
#ifdef DRSNX
#define OS_TYPE "DRSNX"
extern int etext[];
ptr_t GC_SysVGetDataStart(size_t,ptr_t);
#define DATASTART GC_SysVGetDataStart(0x10000,(ptr_t)etext)
#define DATASTART_IS_FUNC
#define MPROTECT_VDB
#define STACKBOTTOM ((ptr_t)0xdfff0000)
#define DYNAMIC_LOADING
#endif
#ifdef LINUX
#define OS_TYPE "LINUX"
#ifdef __ELF__
#define DYNAMIC_LOADING
#elif!defined(CPPCHECK)
#error Linux SPARC a.out not supported
#endif
#define COUNT_UNMAPPED_REGIONS
extern int _end[];
extern int _etext[];
#define DATAEND ((ptr_t)(_end))
#define SVR4
ptr_t GC_SysVGetDataStart(size_t,ptr_t);
#ifdef __arch64__
#define DATASTART GC_SysVGetDataStart(0x100000,(ptr_t)_etext)
#else
#define DATASTART GC_SysVGetDataStart(0x10000,(ptr_t)_etext)
#endif
#define DATASTART_IS_FUNC
#define LINUX_STACKBOTTOM
#endif
#ifdef OPENBSD
#define OS_TYPE "OPENBSD"
#ifndef GC_OPENBSD_THREADS
#define HEURISTIC2
#endif
extern int __data_start[];
#define DATASTART ((ptr_t)__data_start)
extern int _end[];
#define DATAEND ((ptr_t)(&_end))
#define DYNAMIC_LOADING
#endif
#ifdef NETBSD
#define OS_TYPE "NETBSD"
#define HEURISTIC2
#ifdef __ELF__
extern ptr_t GC_data_start;
#define DATASTART GC_data_start
#define DYNAMIC_LOADING
#else
extern char etext[];
#define DATASTART ((ptr_t)(etext))
#endif
#endif
#ifdef FREEBSD
#define OS_TYPE "FREEBSD"
#define SIG_SUSPEND SIGUSR1
#define SIG_THR_RESTART SIGUSR2
#define FREEBSD_STACKBOTTOM
#define DYNAMIC_LOADING
extern char etext[];
extern char edata[];
#if!defined(CPPCHECK)
extern char end[];
#endif
#define NEED_FIND_LIMIT
#define DATASTART ((ptr_t)(&etext))
void*GC_find_limit(void*,int);
#define DATAEND (ptr_t)GC_find_limit(DATASTART,TRUE)
#define DATAEND_IS_FUNC
#define GC_HAVE_DATAREGION2
#define DATASTART2 ((ptr_t)(&edata))
#define DATAEND2 ((ptr_t)(&end))
#endif
#endif
#ifdef I386
#define MACH_TYPE "I386"
#if (defined(__LP64__)||defined(_WIN64))&&!defined(CPPCHECK)
#error This should be handled as X86_64
#else
#define CPP_WORDSZ 32
#define ALIGNMENT 4
#endif
#ifdef SEQUENT
#define OS_TYPE "SEQUENT"
extern int etext[];
#define DATASTART ((ptr_t)((((word)(etext))+0xfff)&~0xfff))
#define STACKBOTTOM ((ptr_t)0x3ffff000)
#endif
#if defined(__EMSCRIPTEN__)
#define OS_TYPE "EMSCRIPTEN"
#define DATASTART (ptr_t)ALIGNMENT
#define DATAEND (ptr_t)ALIGNMENT
#define USE_MMAP_ANON
#define STACK_GROWS_DOWN
#endif
#if defined(__QNX__)
#define OS_TYPE "QNX"
#define SA_RESTART 0
#define HEURISTIC1
extern char etext[];
extern int _end[];
#define DATASTART ((ptr_t)etext)
#define DATAEND ((ptr_t)_end)
#endif
#ifdef HAIKU
#define OS_TYPE "HAIKU"
EXTERN_C_END
#include <OS.h>
EXTERN_C_BEGIN
#define GETPAGESIZE()(unsigned)B_PAGE_SIZE
extern int etext[];
#define DATASTART ((ptr_t)((((word)(etext))+0xfff)&~0xfff))
#define DYNAMIC_LOADING
#define MPROTECT_VDB
#endif
#ifdef SOLARIS
#define OS_TYPE "SOLARIS"
extern int _etext[],_end[];
ptr_t GC_SysVGetDataStart(size_t,ptr_t);
#define DATASTART GC_SysVGetDataStart(0x1000,(ptr_t)_etext)
#define DATASTART_IS_FUNC
#define DATAEND ((ptr_t)(_end))
EXTERN_C_END
#include <sys/vmparam.h>
EXTERN_C_BEGIN
#ifdef USERLIMIT
#define STACKBOTTOM ((ptr_t)USRSTACK)
#else
#define HEURISTIC2
#endif
#ifdef SOLARIS25_PROC_VDB_BUG_FIXED
#define PROC_VDB
#endif
#ifndef GC_THREADS
#define MPROTECT_VDB
#endif
#define DYNAMIC_LOADING
#if!defined(USE_MMAP)&&defined(REDIRECT_MALLOC)
#define USE_MMAP 1
#endif
#ifdef USE_MMAP
#define HEAP_START (ptr_t)0x40000000
#else
#define HEAP_START DATAEND
#endif
#endif
#ifdef SCO
#define OS_TYPE "SCO"
extern int etext[];
#define DATASTART ((ptr_t)((((word)(etext))+0x3fffff)&~0x3fffff)+((word)(etext)&0xfff))
#define STACKBOTTOM ((ptr_t)0x7ffffffc)
#endif
#ifdef SCO_ELF
#define OS_TYPE "SCO_ELF"
extern int etext[];
#define DATASTART ((ptr_t)(etext))
#define STACKBOTTOM ((ptr_t)0x08048000)
#define DYNAMIC_LOADING
#define ELF_CLASS ELFCLASS32
#endif
#ifdef DGUX
#define OS_TYPE "DGUX"
extern int _etext,_end;
ptr_t GC_SysVGetDataStart(size_t,ptr_t);
#define DATASTART GC_SysVGetDataStart(0x1000,(ptr_t)(&_etext))
#define DATASTART_IS_FUNC
#define DATAEND ((ptr_t)(&_end))
#define STACK_GROWS_DOWN
#define HEURISTIC2
EXTERN_C_END
#include <unistd.h>
EXTERN_C_BEGIN
#define GETPAGESIZE()(unsigned)sysconf(_SC_PAGESIZE)
#define DYNAMIC_LOADING
#ifndef USE_MMAP
#define USE_MMAP 1
#endif
#define MAP_FAILED (void*)((word)-1)
#define HEAP_START (ptr_t)0x40000000
#endif
#ifdef LINUX
#define OS_TYPE "LINUX"
#define LINUX_STACKBOTTOM
#define COUNT_UNMAPPED_REGIONS
#if!defined(REDIRECT_MALLOC)
#define MPROTECT_VDB
#else
#endif
#define HEAP_START (ptr_t)0x1000
#ifdef __ELF__
#define DYNAMIC_LOADING
EXTERN_C_END
#include <features.h>
EXTERN_C_BEGIN
#if defined(__GLIBC__)&&__GLIBC__>=2||defined(HOST_ANDROID)||defined(HOST_TIZEN)
#define SEARCH_FOR_DATA_START
#else
extern char**__environ;
#define DATASTART ((ptr_t)(&__environ))
#endif
extern int _end[];
#define DATAEND ((ptr_t)(_end))
#if!defined(GC_NO_SIGSETJMP)&&(defined(HOST_TIZEN)||(defined(HOST_ANDROID)&&!(GC_GNUC_PREREQ(4,8)||GC_CLANG_PREREQ(3,2)||__ANDROID_API__>=18)))
#define GC_NO_SIGSETJMP 1
#endif
#else
extern int etext[];
#define DATASTART ((ptr_t)((((word)(etext))+0xfff)&~0xfff))
#endif
#ifdef USE_I686_PREFETCH
#define PREFETCH(x)__asm__ __volatile__ ("prefetchnta %0"::"m"(*(char*)(x)))
#ifdef FORCE_WRITE_PREFETCH
#define GC_PREFETCH_FOR_WRITE(x)__asm__ __volatile__ ("prefetcht0 %0"::"m"(*(char*)(x)))
#else
#define GC_NO_PREFETCH_FOR_WRITE
#endif
#elif defined(USE_3DNOW_PREFETCH)
#define PREFETCH(x)__asm__ __volatile__ ("prefetch %0"::"m"(*(char*)(x)))
#define GC_PREFETCH_FOR_WRITE(x)__asm__ __volatile__ ("prefetchw %0"::"m"(*(char*)(x)))
#endif
#if defined(__GLIBC__)&&!defined(__UCLIBC__)
#define GLIBC_2_19_TSX_BUG
EXTERN_C_END
#include <gnu/libc-version.h>
EXTERN_C_BEGIN
#endif
#endif
#ifdef CYGWIN32
#define OS_TYPE "CYGWIN32"
#define WOW64_THREAD_CONTEXT_WORKAROUND
#define RETRY_GET_THREAD_CONTEXT
#define DATASTART ((ptr_t)GC_DATASTART)
#define DATAEND ((ptr_t)GC_DATAEND)
#ifdef USE_WINALLOC
#define GWW_VDB
#else
#
#ifdef USE_MMAP
#define NEED_FIND_LIMIT
#define USE_MMAP_ANON
#endif
#endif
#endif
#ifdef INTERIX
#define OS_TYPE "INTERIX"
extern int _data_start__[];
extern int _bss_end__[];
#define DATASTART ((ptr_t)_data_start__)
#define DATAEND ((ptr_t)_bss_end__)
#define STACKBOTTOM ({ ptr_t rv;__asm__ __volatile__ ("movl %%fs:4,%%eax":"=a" (rv));rv;})
#define USE_MMAP_ANON
#endif
#ifdef OS2
#define OS_TYPE "OS2"
#define DATAEND
#endif
#ifdef MSWIN32
#define OS_TYPE "MSWIN32"
#define WOW64_THREAD_CONTEXT_WORKAROUND
#define RETRY_GET_THREAD_CONTEXT
#define MPROTECT_VDB
#define GWW_VDB
#define DATAEND
#endif
#ifdef MSWINCE
#define OS_TYPE "MSWINCE"
#define DATAEND
#endif
#ifdef DJGPP
#define OS_TYPE "DJGPP"
EXTERN_C_END
#include "stubinfo.h"
EXTERN_C_BEGIN
extern int etext[];
extern int _stklen;
extern int __djgpp_stack_limit;
#define DATASTART ((ptr_t)((((word)(etext))+0x1ff)&~0x1ff))
#define STACKBOTTOM ((ptr_t)((word)__djgpp_stack_limit+_stklen))
#endif
#ifdef OPENBSD
#define OS_TYPE "OPENBSD"
#ifndef GC_OPENBSD_THREADS
#define HEURISTIC2
#endif
extern int __data_start[];
#define DATASTART ((ptr_t)__data_start)
extern int _end[];
#define DATAEND ((ptr_t)(&_end))
#define DYNAMIC_LOADING
#endif
#ifdef FREEBSD
#define OS_TYPE "FREEBSD"
#ifndef GC_FREEBSD_THREADS
#define MPROTECT_VDB
#endif
#ifdef __GLIBC__
#define SIG_SUSPEND (32+6)
#define SIG_THR_RESTART (32+5)
extern int _end[];
#define DATAEND ((ptr_t)(_end))
#else
#define SIG_SUSPEND SIGUSR1
#define SIG_THR_RESTART SIGUSR2
#endif
#define FREEBSD_STACKBOTTOM
#ifdef __ELF__
#define DYNAMIC_LOADING
#endif
extern char etext[];
#define DATASTART GC_FreeBSDGetDataStart(0x1000,(ptr_t)etext)
#define DATASTART_USES_BSDGETDATASTART
#endif
#ifdef NETBSD
#define OS_TYPE "NETBSD"
#ifdef __ELF__
#define DYNAMIC_LOADING
#endif
#endif
#ifdef THREE86BSD
#define OS_TYPE "THREE86BSD"
#endif
#ifdef BSDI
#define OS_TYPE "BSDI"
#endif
#if defined(NETBSD)||defined(THREE86BSD)||defined(BSDI)
#define HEURISTIC2
extern char etext[];
#define DATASTART ((ptr_t)(etext))
#endif
#ifdef NEXT
#define OS_TYPE "NEXT"
#define DATASTART ((ptr_t)get_etext())
#define DATASTART_IS_FUNC
#define STACKBOTTOM ((ptr_t)0xc0000000)
#define DATAEND
#endif
#ifdef RTEMS
#define OS_TYPE "RTEMS"
EXTERN_C_END
#include <sys/unistd.h>
EXTERN_C_BEGIN
extern int etext[];
void*rtems_get_stack_bottom(void);
#define InitStackBottom rtems_get_stack_bottom()
#define DATASTART ((ptr_t)etext)
#define STACKBOTTOM ((ptr_t)InitStackBottom)
#define SIG_SUSPEND SIGUSR1
#define SIG_THR_RESTART SIGUSR2
#endif
#ifdef DOS4GW
#define OS_TYPE "DOS4GW"
extern long __nullarea;
extern char _end;
extern char*_STACKTOP;
#pragma aux __nullarea "*";
#pragma aux _end "*";
#define STACKBOTTOM ((ptr_t)_STACKTOP)
#define DATASTART ((ptr_t)(&__nullarea))
#define DATAEND ((ptr_t)(&_end))
#endif
#ifdef HURD
#define OS_TYPE "HURD"
#define STACK_GROWS_DOWN
#define HEURISTIC2
#define SIG_SUSPEND SIGUSR1
#define SIG_THR_RESTART SIGUSR2
#define SEARCH_FOR_DATA_START
extern int _end[];
#define DATAEND ((ptr_t)(_end))
#define DYNAMIC_LOADING
#define USE_MMAP_ANON
#endif
#ifdef DARWIN
#define OS_TYPE "DARWIN"
#define DARWIN_DONT_PARSE_STACK 1
#define DYNAMIC_LOADING
#define DATASTART ((ptr_t)get_etext())
#define DATAEND ((ptr_t)get_end())
#define STACKBOTTOM ((ptr_t)0xc0000000)
#define USE_MMAP_ANON
#define MPROTECT_VDB
EXTERN_C_END
#include <unistd.h>
EXTERN_C_BEGIN
#define GETPAGESIZE()(unsigned)getpagesize()
#define NO_PTHREAD_TRYLOCK
#if TARGET_OS_IPHONE&&!defined(NO_DYLD_BIND_FULLY_IMAGE)
#define NO_DYLD_BIND_FULLY_IMAGE
#endif
#endif
#endif
#ifdef NS32K
#define MACH_TYPE "NS32K"
#define ALIGNMENT 4
extern char**environ;
#define DATASTART ((ptr_t)(&environ))
#define STACKBOTTOM ((ptr_t)0xfffff000)
#endif
#ifdef MIPS
#define MACH_TYPE "MIPS"
#ifdef LINUX
#define OS_TYPE "LINUX"
#define DYNAMIC_LOADING
#define COUNT_UNMAPPED_REGIONS
extern int _end[];
#pragma weak __data_start
extern int __data_start[];
#define DATASTART ((ptr_t)(__data_start))
#define DATAEND ((ptr_t)(_end))
#ifdef _MIPS_SZPTR
#define CPP_WORDSZ _MIPS_SZPTR
#define ALIGNMENT (_MIPS_SZPTR/8)
#else
#define ALIGNMENT 4
#endif
#ifndef HBLKSIZE
#define HBLKSIZE 4096
#endif
#if __GLIBC__==2&&__GLIBC_MINOR__>=2||__GLIBC__ > 2
#define LINUX_STACKBOTTOM
#else
#define STACKBOTTOM ((ptr_t)0x7fff8000)
#endif
#endif
#ifdef EWS4800
#define HEURISTIC2
#if defined(_MIPS_SZPTR)&&(_MIPS_SZPTR==64)
extern int _fdata[],_end[];
#define DATASTART ((ptr_t)_fdata)
#define DATAEND ((ptr_t)_end)
#define CPP_WORDSZ _MIPS_SZPTR
#define ALIGNMENT (_MIPS_SZPTR/8)
#else
extern int etext[],edata[];
#if!defined(CPPCHECK)
extern int end[];
#endif
extern int _DYNAMIC_LINKING[],_gp[];
#define DATASTART ((ptr_t)((((word)(etext)+0x3ffff)&~0x3ffff)+((word)(etext)&0xffff)))
#define DATAEND ((ptr_t)(edata))
#define GC_HAVE_DATAREGION2
#define DATASTART2 (_DYNAMIC_LINKING?(ptr_t)(((word)_gp+0x8000+0x3ffff)&~0x3ffff):(ptr_t)edata)
#define DATAEND2 ((ptr_t)(end))
#define ALIGNMENT 4
#endif
#define OS_TYPE "EWS4800"
#endif
#ifdef ULTRIX
#define HEURISTIC2
#define DATASTART ((ptr_t)0x10000000)
#define OS_TYPE "ULTRIX"
#define ALIGNMENT 4
#endif
#ifdef IRIX5
#define HEURISTIC2
extern int _fdata[];
#define DATASTART ((ptr_t)(_fdata))
#ifdef USE_MMAP
#define HEAP_START (ptr_t)0x30000000
#else
#define HEAP_START DATASTART
#endif
#define OS_TYPE "IRIX5"
#ifdef _MIPS_SZPTR
#define CPP_WORDSZ _MIPS_SZPTR
#define ALIGNMENT (_MIPS_SZPTR/8)
#else
#define ALIGNMENT 4
#endif
#define DYNAMIC_LOADING
#endif
#ifdef MSWINCE
#define OS_TYPE "MSWINCE"
#define ALIGNMENT 4
#define DATAEND
#endif
#ifdef NETBSD
#define OS_TYPE "NETBSD"
#define ALIGNMENT 4
#define HEURISTIC2
#ifdef __ELF__
extern ptr_t GC_data_start;
#define DATASTART GC_data_start
#define NEED_FIND_LIMIT
#define DYNAMIC_LOADING
#else
#define DATASTART ((ptr_t)0x10000000)
#define STACKBOTTOM ((ptr_t)0x7ffff000)
#endif
#endif
#ifdef OPENBSD
#define OS_TYPE "OPENBSD"
#define CPP_WORDSZ 64
#define ALIGNMENT 8
#ifndef GC_OPENBSD_THREADS
#define HEURISTIC2
#endif
extern int __data_start[];
#define DATASTART ((ptr_t)__data_start)
extern int _end[];
#define DATAEND ((ptr_t)(&_end))
#define DYNAMIC_LOADING
#endif
#ifdef FREEBSD
#define OS_TYPE "FREEBSD"
#define ALIGNMENT 4
#ifndef GC_FREEBSD_THREADS
#define MPROTECT_VDB
#endif
#define SIG_SUSPEND SIGUSR1
#define SIG_THR_RESTART SIGUSR2
#define FREEBSD_STACKBOTTOM
#define DYNAMIC_LOADING
extern char etext[];
#define DATASTART GC_FreeBSDGetDataStart(0x1000,(ptr_t)etext)
#define DATASTART_USES_BSDGETDATASTART
#endif
#if defined(NONSTOP)
#define CPP_WORDSZ 32
#define OS_TYPE "NONSTOP"
#define ALIGNMENT 4
#define DATASTART ((ptr_t)0x08000000)
extern char**environ;
#define DATAEND ((ptr_t)(environ - 0x10))
#define STACKBOTTOM ((ptr_t)0x4fffffff)
#endif
#endif
#ifdef NIOS2
#define CPP_WORDSZ 32
#define MACH_TYPE "NIOS2"
#ifdef LINUX
#define OS_TYPE "LINUX"
#define DYNAMIC_LOADING
#define COUNT_UNMAPPED_REGIONS
extern int _end[];
extern int __data_start[];
#define DATASTART ((ptr_t)(__data_start))
#define DATAEND ((ptr_t)(_end))
#define ALIGNMENT 4
#ifndef HBLKSIZE
#define HBLKSIZE 4096
#endif
#define LINUX_STACKBOTTOM
#endif
#endif
#ifdef OR1K
#define CPP_WORDSZ 32
#define MACH_TYPE "OR1K"
#ifdef LINUX
#define OS_TYPE "LINUX"
#define DYNAMIC_LOADING
#define COUNT_UNMAPPED_REGIONS
extern int _end[];
extern int __data_start[];
#define DATASTART ((ptr_t)(__data_start))
#define DATAEND ((ptr_t)(_end))
#define ALIGNMENT 4
#ifndef HBLKSIZE
#define HBLKSIZE 4096
#endif
#define LINUX_STACKBOTTOM
#endif
#endif
#ifdef HP_PA
#define MACH_TYPE "HP_PA"
#ifdef __LP64__
#define CPP_WORDSZ 64
#define ALIGNMENT 8
#else
#define CPP_WORDSZ 32
#define ALIGNMENT 4
#endif
#if!defined(GC_HPUX_THREADS)&&!defined(GC_LINUX_THREADS)&&!defined(OPENBSD)&&!defined(LINUX)
#define MPROTECT_VDB
#endif
#define STACK_GROWS_UP
#ifdef HPUX
#define OS_TYPE "HPUX"
extern int __data_start[];
#define DATASTART ((ptr_t)(__data_start))
#ifdef USE_MMAP
#define USE_MMAP_ANON
#endif
#ifdef USE_HPUX_FIXED_STACKBOTTOM
#define STACKBOTTOM ((ptr_t)0x7b033000)
#elif defined(USE_ENVIRON_POINTER)
extern char**environ;
#define STACKBOTTOM ((ptr_t)environ)
#elif!defined(HEURISTIC2)
#define HPUX_MAIN_STACKBOTTOM
#define NEED_FIND_LIMIT
#endif
#define DYNAMIC_LOADING
EXTERN_C_END
#include <unistd.h>
EXTERN_C_BEGIN
#define GETPAGESIZE()(unsigned)sysconf(_SC_PAGE_SIZE)
#ifndef __GNUC__
#define PREFETCH(x)do { register long addr=(long)(x);(void)_asm ("LDW",0,0,addr,0);} while (0)
#endif
#endif
#ifdef LINUX
#define OS_TYPE "LINUX"
#define LINUX_STACKBOTTOM
#define COUNT_UNMAPPED_REGIONS
#define DYNAMIC_LOADING
#define SEARCH_FOR_DATA_START
extern int _end[];
#define DATAEND ((ptr_t)(&_end))
#endif
#ifdef OPENBSD
#define OS_TYPE "OPENBSD"
#ifndef GC_OPENBSD_THREADS
#define HEURISTIC2
#endif
extern int __data_start[];
#define DATASTART ((ptr_t)__data_start)
extern int _end[];
#define DATAEND ((ptr_t)(&_end))
#define DYNAMIC_LOADING
#endif
#endif
#ifdef ALPHA
#define MACH_TYPE "ALPHA"
#define ALIGNMENT 8
#define CPP_WORDSZ 64
#ifdef NETBSD
#define OS_TYPE "NETBSD"
#define HEURISTIC2
extern ptr_t GC_data_start;
#define DATASTART GC_data_start
#define ELFCLASS32 32
#define ELFCLASS64 64
#define ELF_CLASS ELFCLASS64
#define DYNAMIC_LOADING
#endif
#ifdef OPENBSD
#define OS_TYPE "OPENBSD"
#ifndef GC_OPENBSD_THREADS
#define HEURISTIC2
#endif
extern int __data_start[];
#define DATASTART ((ptr_t)__data_start)
extern int _end[];
#define DATAEND ((ptr_t)(&_end))
#define DYNAMIC_LOADING
#endif
#ifdef FREEBSD
#define OS_TYPE "FREEBSD"
#define SIG_SUSPEND SIGUSR1
#define SIG_THR_RESTART SIGUSR2
#define FREEBSD_STACKBOTTOM
#define DYNAMIC_LOADING
extern char etext[];
extern char edata[];
#if!defined(CPPCHECK)
extern char end[];
#endif
#define NEED_FIND_LIMIT
#define DATASTART ((ptr_t)(&etext))
void*GC_find_limit(void*,int);
#define DATAEND (ptr_t)GC_find_limit(DATASTART,TRUE)
#define DATAEND_IS_FUNC
#define GC_HAVE_DATAREGION2
#define DATASTART2 ((ptr_t)(&edata))
#define DATAEND2 ((ptr_t)(&end))
#endif
#ifdef OSF1
#define OS_TYPE "OSF1"
#define DATASTART ((ptr_t)0x140000000)
extern int _end[];
#define DATAEND ((ptr_t)(&_end))
extern char**environ;
#define STACKBOTTOM ((ptr_t)(((word)(environ)|(getpagesize()-1))+1))
extern int __start[];
#define HEURISTIC2_LIMIT ((ptr_t)((word)(__start)&~(getpagesize()-1)))
#ifndef GC_OSF1_THREADS
#define MPROTECT_VDB
#endif
#define DYNAMIC_LOADING
#endif
#ifdef LINUX
#define OS_TYPE "LINUX"
#define LINUX_STACKBOTTOM
#define COUNT_UNMAPPED_REGIONS
#ifdef __ELF__
#define SEARCH_FOR_DATA_START
#define DYNAMIC_LOADING
#else
#define DATASTART ((ptr_t)0x140000000)
#endif
extern int _end[];
#define DATAEND ((ptr_t)(_end))
#if!defined(REDIRECT_MALLOC)
#define MPROTECT_VDB
#endif
#endif
#endif
#ifdef IA64
#define MACH_TYPE "IA64"
#ifdef HPUX
#ifdef _ILP32
#define CPP_WORDSZ 32
#define ALIGNMENT 4
#else
#if!defined(_LP64)&&!defined(CPPCHECK)
#error Unknown ABI
#endif
#define CPP_WORDSZ 64
#define ALIGNMENT 8
#endif
#define OS_TYPE "HPUX"
extern int __data_start[];
#define DATASTART ((ptr_t)(__data_start))
#ifdef USE_MMAP
#define USE_MMAP_ANON
#endif
extern char**environ;
#define STACKBOTTOM ((ptr_t)environ)
#define HPUX_STACKBOTTOM
#define DYNAMIC_LOADING
EXTERN_C_END
#include <unistd.h>
EXTERN_C_BEGIN
#define GETPAGESIZE()(unsigned)sysconf(_SC_PAGE_SIZE)
#define BACKING_STORE_DISPLACEMENT 0x1000000
#define BACKING_STORE_ALIGNMENT 0x1000
extern ptr_t GC_register_stackbottom;
#define BACKING_STORE_BASE GC_register_stackbottom
#endif
#ifdef LINUX
#define CPP_WORDSZ 64
#define ALIGNMENT 8
#define OS_TYPE "LINUX"
#define LINUX_STACKBOTTOM
extern ptr_t GC_register_stackbottom;
#define BACKING_STORE_BASE GC_register_stackbottom
#define COUNT_UNMAPPED_REGIONS
#define SEARCH_FOR_DATA_START
#ifdef __GNUC__
#define DYNAMIC_LOADING
#else
#endif
#if!defined(REDIRECT_MALLOC)
#define MPROTECT_VDB
#endif
extern int _end[];
#define DATAEND ((ptr_t)(_end))
#ifdef __GNUC__
#ifndef __INTEL_COMPILER
#define PREFETCH(x)__asm__ (" lfetch [%0]"::"r"(x))
#define GC_PREFETCH_FOR_WRITE(x)__asm__ (" lfetch.excl [%0]"::"r"(x))
#define CLEAR_DOUBLE(x)__asm__ (" stf.spill [%0]=f0"::"r"((void*)(x)))
#else
EXTERN_C_END
#include <ia64intrin.h>
EXTERN_C_BEGIN
#define PREFETCH(x)__lfetch(__lfhint_none,(x))
#define GC_PREFETCH_FOR_WRITE(x)__lfetch(__lfhint_nta,(x))
#define CLEAR_DOUBLE(x)__stf_spill((void*)(x),0)
#endif
#endif
#endif
#ifdef MSWIN32
#define OS_TYPE "MSWIN32"
#define DATAEND
#if defined(_WIN64)
#define CPP_WORDSZ 64
#else
#define CPP_WORDSZ 32
#endif
#define ALIGNMENT 8
#endif
#endif
#ifdef M88K
#define MACH_TYPE "M88K"
#define ALIGNMENT 4
extern int etext[];
#ifdef CX_UX
#define OS_TYPE "CX_UX"
#define DATASTART ((ptr_t)((((word)(etext)+0x3fffff)&~0x3fffff)+0x10000))
#endif
#ifdef DGUX
#define OS_TYPE "DGUX"
ptr_t GC_SysVGetDataStart(size_t,ptr_t);
#define DATASTART GC_SysVGetDataStart(0x10000,(ptr_t)etext)
#define DATASTART_IS_FUNC
#endif
#define STACKBOTTOM ((char*)0xf0000000)
#endif
#ifdef S370
#define MACH_TYPE "S370"
#define ALIGNMENT 4
#ifdef UTS4
#define OS_TYPE "UTS4"
extern int etext[];
extern int _etext[];
extern int _end[];
ptr_t GC_SysVGetDataStart(size_t,ptr_t);
#define DATASTART GC_SysVGetDataStart(0x10000,(ptr_t)_etext)
#define DATASTART_IS_FUNC
#define DATAEND ((ptr_t)(_end))
#define HEURISTIC2
#endif
#endif
#ifdef S390
#define MACH_TYPE "S390"
#ifndef __s390x__
#define ALIGNMENT 4
#define CPP_WORDSZ 32
#else
#define ALIGNMENT 8
#define CPP_WORDSZ 64
#ifndef HBLKSIZE
#define HBLKSIZE 4096
#endif
#endif
#ifdef LINUX
#define OS_TYPE "LINUX"
#define LINUX_STACKBOTTOM
#define DYNAMIC_LOADING
#define COUNT_UNMAPPED_REGIONS
extern int __data_start[] __attribute__((__weak__));
#define DATASTART ((ptr_t)(__data_start))
extern int _end[] __attribute__((__weak__));
#define DATAEND ((ptr_t)(_end))
#define CACHE_LINE_SIZE 256
#define GETPAGESIZE()4096
#endif
#endif
#ifdef AARCH64
#define MACH_TYPE "AARCH64"
#ifdef __ILP32__
#define CPP_WORDSZ 32
#define ALIGNMENT 4
#else
#define CPP_WORDSZ 64
#define ALIGNMENT 8
#endif
#ifndef HBLKSIZE
#define HBLKSIZE 4096
#endif
#ifdef LINUX
#define OS_TYPE "LINUX"
#define LINUX_STACKBOTTOM
#define COUNT_UNMAPPED_REGIONS
#if!defined(REDIRECT_MALLOC)
#define MPROTECT_VDB
#endif
#define DYNAMIC_LOADING
#if defined(HOST_ANDROID)
#define SEARCH_FOR_DATA_START
#else
extern int __data_start[];
#define DATASTART ((ptr_t)__data_start)
#endif
extern int _end[];
#define DATAEND ((ptr_t)(&_end))
#endif
#ifdef DARWIN
#define OS_TYPE "DARWIN"
#define DARWIN_DONT_PARSE_STACK 1
#define DYNAMIC_LOADING
#define DATASTART ((ptr_t)get_etext())
#define DATAEND ((ptr_t)get_end())
#define STACKBOTTOM ((ptr_t)0x16fdfffff)
#define USE_MMAP_ANON
EXTERN_C_END
#include <unistd.h>
EXTERN_C_BEGIN
#define GETPAGESIZE()(unsigned)getpagesize()
#define NO_PTHREAD_TRYLOCK
#if TARGET_OS_IPHONE&&!defined(NO_DYLD_BIND_FULLY_IMAGE)
#define NO_DYLD_BIND_FULLY_IMAGE
#endif
#endif
#ifdef FREEBSD
#define OS_TYPE "FREEBSD"
#ifndef GC_FREEBSD_THREADS
#define MPROTECT_VDB
#endif
#define FREEBSD_STACKBOTTOM
#define DYNAMIC_LOADING
extern char etext[];
#define DATASTART GC_FreeBSDGetDataStart(0x1000,(ptr_t)etext)
#define DATASTART_USES_BSDGETDATASTART
#endif
#ifdef NETBSD
#define OS_TYPE "NETBSD"
#define HEURISTIC2
extern ptr_t GC_data_start;
#define DATASTART GC_data_start
#define ELF_CLASS ELFCLASS64
#define DYNAMIC_LOADING
#endif
#ifdef OPENBSD
#define OS_TYPE "OPENBSD"
#ifndef GC_OPENBSD_THREADS
#define HEURISTIC2
#endif
extern int __data_start[];
#define DATASTART ((ptr_t)__data_start)
extern int _end[];
#define DATAEND ((ptr_t)(&_end))
#define DYNAMIC_LOADING
#endif
#ifdef NINTENDO_SWITCH
#define OS_TYPE "NINTENDO_SWITCH"
extern int __bss_end[];
#define NO_HANDLE_FORK 1
#define DATASTART (ptr_t)ALIGNMENT
#define DATAEND (ptr_t)(&__bss_end)
void*switch_get_stack_bottom(void);
#define STACKBOTTOM ((ptr_t)switch_get_stack_bottom())
#endif
#ifdef MSWIN32
#define OS_TYPE "MSWIN32"
#ifndef DATAEND
#define DATAEND
#endif
#endif
#ifdef NOSYS
extern int __data_start[];
#define DATASTART ((ptr_t)__data_start)
extern void*__stack_base__;
#define STACKBOTTOM ((ptr_t)__stack_base__)
#endif
#endif
#ifdef ARM32
#if defined(NACL)
#define MACH_TYPE "NACL"
#else
#define MACH_TYPE "ARM32"
#endif
#define CPP_WORDSZ 32
#define ALIGNMENT 4
#ifdef NETBSD
#define OS_TYPE "NETBSD"
#define HEURISTIC2
#ifdef __ELF__
extern ptr_t GC_data_start;
#define DATASTART GC_data_start
#define DYNAMIC_LOADING
#else
extern char etext[];
#define DATASTART ((ptr_t)(etext))
#endif
#endif
#ifdef LINUX
#define OS_TYPE "LINUX"
#define LINUX_STACKBOTTOM
#define COUNT_UNMAPPED_REGIONS
#if!defined(REDIRECT_MALLOC)
#define MPROTECT_VDB
#endif
#define DYNAMIC_LOADING
EXTERN_C_END
#include <features.h>
EXTERN_C_BEGIN
#if defined(__GLIBC__)&&__GLIBC__>=2||defined(HOST_ANDROID)||defined(HOST_TIZEN)
#define SEARCH_FOR_DATA_START
#else
extern char**__environ;
#define DATASTART ((ptr_t)(&__environ))
#endif
extern int _end[];
#define DATAEND ((ptr_t)(_end))
#endif
#ifdef MSWINCE
#define OS_TYPE "MSWINCE"
#define DATAEND
#endif
#ifdef FREEBSD
#define OS_TYPE "FREEBSD"
#ifndef GC_FREEBSD_THREADS
#define MPROTECT_VDB
#endif
#define SIG_SUSPEND SIGUSR1
#define SIG_THR_RESTART SIGUSR2
#define FREEBSD_STACKBOTTOM
#define DYNAMIC_LOADING
extern char etext[];
#define DATASTART GC_FreeBSDGetDataStart(0x1000,(ptr_t)etext)
#define DATASTART_USES_BSDGETDATASTART
#endif
#ifdef DARWIN
#define OS_TYPE "DARWIN"
#define DARWIN_DONT_PARSE_STACK 1
#define DYNAMIC_LOADING
#define DATASTART ((ptr_t)get_etext())
#define DATAEND ((ptr_t)get_end())
#define STACKBOTTOM ((ptr_t)0x30000000)
#define USE_MMAP_ANON
EXTERN_C_END
#include <unistd.h>
EXTERN_C_BEGIN
#define GETPAGESIZE()(unsigned)getpagesize()
#define NO_PTHREAD_TRYLOCK
#if TARGET_OS_IPHONE&&!defined(NO_DYLD_BIND_FULLY_IMAGE)
#define NO_DYLD_BIND_FULLY_IMAGE
#endif
#endif
#ifdef OPENBSD
#define OS_TYPE "OPENBSD"
#ifndef GC_OPENBSD_THREADS
#define HEURISTIC2
#endif
extern int __data_start[];
#define DATASTART ((ptr_t)__data_start)
extern int _end[];
#define DATAEND ((ptr_t)(&_end))
#define DYNAMIC_LOADING
#endif
#ifdef SN_TARGET_PSP2
#define OS_TYPE "SN_TARGET_PSP2"
#define NO_HANDLE_FORK 1
#define DATASTART (ptr_t)ALIGNMENT
#define DATAEND (ptr_t)ALIGNMENT
void*psp2_get_stack_bottom(void);
#define STACKBOTTOM ((ptr_t)psp2_get_stack_bottom())
#endif
#ifdef NN_PLATFORM_CTR
#define OS_TYPE "NN_PLATFORM_CTR"
extern unsigned char Image$$ZI$$ZI$$Base[];
#define DATASTART (ptr_t)(Image$$ZI$$ZI$$Base)
extern unsigned char Image$$ZI$$ZI$$Limit[];
#define DATAEND (ptr_t)(Image$$ZI$$ZI$$Limit)
void*n3ds_get_stack_bottom(void);
#define STACKBOTTOM ((ptr_t)n3ds_get_stack_bottom())
#endif
#ifdef MSWIN32
#define OS_TYPE "MSWIN32"
#ifndef DATAEND
#define DATAEND
#endif
#endif
#ifdef NOSYS
extern int __data_start[];
#define DATASTART ((ptr_t)(__data_start))
extern void*__stack_base__;
#define STACKBOTTOM ((ptr_t)(__stack_base__))
#endif
#endif
#ifdef CRIS
#define MACH_TYPE "CRIS"
#define CPP_WORDSZ 32
#define ALIGNMENT 1
#ifdef LINUX
#define OS_TYPE "LINUX"
#define DYNAMIC_LOADING
#define LINUX_STACKBOTTOM
#define COUNT_UNMAPPED_REGIONS
#define SEARCH_FOR_DATA_START
extern int _end[];
#define DATAEND ((ptr_t)(_end))
#endif
#endif
#if defined(SH)&&!defined(SH4)
#define MACH_TYPE "SH"
#define ALIGNMENT 4
#ifdef MSWINCE
#define OS_TYPE "MSWINCE"
#define DATAEND
#endif
#ifdef LINUX
#define OS_TYPE "LINUX"
#define LINUX_STACKBOTTOM
#define COUNT_UNMAPPED_REGIONS
#define DYNAMIC_LOADING
#define SEARCH_FOR_DATA_START
extern int _end[];
#define DATAEND ((ptr_t)(_end))
#endif
#ifdef NETBSD
#define OS_TYPE "NETBSD"
#define HEURISTIC2
extern ptr_t GC_data_start;
#define DATASTART GC_data_start
#define DYNAMIC_LOADING
#endif
#ifdef OPENBSD
#define OS_TYPE "OPENBSD"
#ifndef GC_OPENBSD_THREADS
#define HEURISTIC2
#endif
extern int __data_start[];
#define DATASTART ((ptr_t)__data_start)
extern int _end[];
#define DATAEND ((ptr_t)(&_end))
#define DYNAMIC_LOADING
#endif
#endif
#ifdef SH4
#define MACH_TYPE "SH4"
#define ALIGNMENT 4
#ifdef MSWINCE
#define OS_TYPE "MSWINCE"
#define DATAEND
#endif
#endif
#ifdef AVR32
#define MACH_TYPE "AVR32"
#define CPP_WORDSZ 32
#define ALIGNMENT 4
#ifdef LINUX
#define OS_TYPE "LINUX"
#define DYNAMIC_LOADING
#define LINUX_STACKBOTTOM
#define COUNT_UNMAPPED_REGIONS
#define SEARCH_FOR_DATA_START
extern int _end[];
#define DATAEND ((ptr_t)(_end))
#endif
#endif
#ifdef M32R
#define CPP_WORDSZ 32
#define MACH_TYPE "M32R"
#define ALIGNMENT 4
#ifdef LINUX
#define OS_TYPE "LINUX"
#define LINUX_STACKBOTTOM
#define COUNT_UNMAPPED_REGIONS
#define DYNAMIC_LOADING
#define SEARCH_FOR_DATA_START
extern int _end[];
#define DATAEND ((ptr_t)(_end))
#endif
#endif
#ifdef X86_64
#define MACH_TYPE "X86_64"
#ifdef __ILP32__
#define ALIGNMENT 4
#define CPP_WORDSZ 32
#else
#define ALIGNMENT 8
#define CPP_WORDSZ 64
#endif
#ifndef HBLKSIZE
#define HBLKSIZE 4096
#endif
#ifndef CACHE_LINE_SIZE
#define CACHE_LINE_SIZE 64
#endif
#ifdef SN_TARGET_ORBIS
#define OS_TYPE "SN_TARGET_ORBIS"
#define DATASTART (ptr_t)ALIGNMENT
#define DATAEND (ptr_t)ALIGNMENT
void*ps4_get_stack_bottom(void);
#define STACKBOTTOM ((ptr_t)ps4_get_stack_bottom())
#endif
#ifdef OPENBSD
#define OS_TYPE "OPENBSD"
#ifndef GC_OPENBSD_THREADS
#define HEURISTIC2
#endif
extern int __data_start[];
extern int _end[];
#define DATASTART ((ptr_t)__data_start)
#define DATAEND ((ptr_t)(&_end))
#define DYNAMIC_LOADING
#endif
#ifdef LINUX
#define OS_TYPE "LINUX"
#define LINUX_STACKBOTTOM
#if!defined(REDIRECT_MALLOC)
#define MPROTECT_VDB
#else
#endif
#define COUNT_UNMAPPED_REGIONS
#define DYNAMIC_LOADING
EXTERN_C_END
#include <features.h>
EXTERN_C_BEGIN
#define SEARCH_FOR_DATA_START
extern int _end[];
#define DATAEND ((ptr_t)(_end))
#if defined(__GLIBC__)&&!defined(__UCLIBC__)
#define USE_MMAP_ANON
#define GETCONTEXT_FPU_EXCMASK_BUG
#define GLIBC_2_19_TSX_BUG
EXTERN_C_END
#include <gnu/libc-version.h>
EXTERN_C_BEGIN
#endif
#endif
#ifdef DARWIN
#define OS_TYPE "DARWIN"
#define DARWIN_DONT_PARSE_STACK 1
#define DYNAMIC_LOADING
#define DATASTART ((ptr_t)get_etext())
#define DATAEND ((ptr_t)get_end())
#define STACKBOTTOM ((ptr_t)0x7fff5fc00000)
#define USE_MMAP_ANON
#define MPROTECT_VDB
EXTERN_C_END
#include <unistd.h>
EXTERN_C_BEGIN
#define GETPAGESIZE()(unsigned)getpagesize()
#define NO_PTHREAD_TRYLOCK
#if TARGET_OS_IPHONE&&!defined(NO_DYLD_BIND_FULLY_IMAGE)
#define NO_DYLD_BIND_FULLY_IMAGE
#endif
#endif
#ifdef FREEBSD
#define OS_TYPE "FREEBSD"
#ifndef GC_FREEBSD_THREADS
#define MPROTECT_VDB
#endif
#ifdef __GLIBC__
#define SIG_SUSPEND (32+6)
#define SIG_THR_RESTART (32+5)
extern int _end[];
#define DATAEND ((ptr_t)(_end))
#else
#define SIG_SUSPEND SIGUSR1
#define SIG_THR_RESTART SIGUSR2
#endif
#define FREEBSD_STACKBOTTOM
#if defined(__DragonFly__)
#define COUNT_UNMAPPED_REGIONS
#endif
#define DYNAMIC_LOADING
extern char etext[];
#define DATASTART GC_FreeBSDGetDataStart(0x1000,(ptr_t)etext)
#define DATASTART_USES_BSDGETDATASTART
#endif
#ifdef NETBSD
#define OS_TYPE "NETBSD"
#define HEURISTIC2
extern ptr_t GC_data_start;
#define DATASTART GC_data_start
#define DYNAMIC_LOADING
#endif
#ifdef HAIKU
#define OS_TYPE "HAIKU"
EXTERN_C_END
#include <OS.h>
EXTERN_C_BEGIN
#define GETPAGESIZE()(unsigned)B_PAGE_SIZE
#define HEURISTIC2
#define SEARCH_FOR_DATA_START
#define DYNAMIC_LOADING
#define MPROTECT_VDB
#endif
#ifdef SOLARIS
#define OS_TYPE "SOLARIS"
#define ELF_CLASS ELFCLASS64
extern int _etext[],_end[];
ptr_t GC_SysVGetDataStart(size_t,ptr_t);
#define DATASTART GC_SysVGetDataStart(0x1000,(ptr_t)_etext)
#define DATASTART_IS_FUNC
#define DATAEND ((ptr_t)(_end))
EXTERN_C_END
#include <sys/vmparam.h>
EXTERN_C_BEGIN
#ifdef USERLIMIT
#define STACKBOTTOM ((ptr_t)USRSTACK)
#else
#define HEURISTIC2
#endif
#ifdef SOLARIS25_PROC_VDB_BUG_FIXED
#define PROC_VDB
#endif
#ifndef GC_THREADS
#define MPROTECT_VDB
#endif
#define DYNAMIC_LOADING
#if!defined(USE_MMAP)&&defined(REDIRECT_MALLOC)
#define USE_MMAP 1
#endif
#ifdef USE_MMAP
#define HEAP_START (ptr_t)0x40000000
#else
#define HEAP_START DATAEND
#endif
#endif
#ifdef CYGWIN32
#define OS_TYPE "CYGWIN32"
#define RETRY_GET_THREAD_CONTEXT
#ifdef USE_WINALLOC
#define GWW_VDB
#else
#if defined(THREAD_LOCAL_ALLOC)
#else
#define MPROTECT_VDB
#endif
#ifdef USE_MMAP
#define USE_MMAP_ANON
#endif
#endif
#endif
#ifdef MSWIN_XBOX1
#define OS_TYPE "MSWIN_XBOX1"
#define NO_GETENV
#define DATASTART (ptr_t)ALIGNMENT
#define DATAEND (ptr_t)ALIGNMENT
LONG64 durango_get_stack_bottom(void);
#define STACKBOTTOM ((ptr_t)durango_get_stack_bottom())
#define GETPAGESIZE()4096
#ifndef USE_MMAP
#define USE_MMAP 1
#endif
#define PROT_NONE 0
#define PROT_READ 1
#define PROT_WRITE 2
#define PROT_EXEC 4
#define MAP_PRIVATE 2
#define MAP_FIXED 0x10
#define MAP_FAILED ((void*)-1)
#endif
#ifdef MSWIN32
#define OS_TYPE "MSWIN32"
#define RETRY_GET_THREAD_CONTEXT
#if!defined(__GNUC__)||defined(__INTEL_COMPILER)||GC_GNUC_PREREQ(4,7)
#define MPROTECT_VDB
#endif
#define GWW_VDB
#ifndef DATAEND
#define DATAEND
#endif
#endif
#endif
#ifdef ARC
#define CPP_WORDSZ 32
#define MACH_TYPE "ARC"
#define ALIGNMENT 4
#define CACHE_LINE_SIZE 64
#ifdef LINUX
#define OS_TYPE "LINUX"
#define LINUX_STACKBOTTOM
#define COUNT_UNMAPPED_REGIONS
#define DYNAMIC_LOADING
extern int __data_start[] __attribute__((__weak__));
#define DATASTART ((ptr_t)__data_start)
#endif
#endif
#ifdef HEXAGON
#define CPP_WORDSZ 32
#define MACH_TYPE "HEXAGON"
#define ALIGNMENT 4
#ifdef LINUX
#define OS_TYPE "LINUX"
#define LINUX_STACKBOTTOM
#if!defined(REDIRECT_MALLOC)
#define MPROTECT_VDB
#endif
#define COUNT_UNMAPPED_REGIONS
#define DYNAMIC_LOADING
EXTERN_C_END
#include <features.h>
EXTERN_C_BEGIN
#if defined(__GLIBC__)
#define SEARCH_FOR_DATA_START
#elif!defined(CPPCHECK)
#error Unknown Hexagon libc configuration
#endif
extern int _end[];
#define DATAEND ((ptr_t)(_end))
#endif
#endif
#ifdef TILEPRO
#define CPP_WORDSZ 32
#define MACH_TYPE "TILEPro"
#define ALIGNMENT 4
#define PREFETCH(x)__insn_prefetch(x)
#define CACHE_LINE_SIZE 64
#ifdef LINUX
#define OS_TYPE "LINUX"
extern int __data_start[];
#define DATASTART ((ptr_t)__data_start)
#define LINUX_STACKBOTTOM
#define COUNT_UNMAPPED_REGIONS
#define DYNAMIC_LOADING
#endif
#endif
#ifdef TILEGX
#define CPP_WORDSZ (__SIZEOF_POINTER__*8)
#define MACH_TYPE "TILE-Gx"
#define ALIGNMENT __SIZEOF_POINTER__
#if CPP_WORDSZ < 64
#define CLEAR_DOUBLE(x)(*(long long*)(x)=0)
#endif
#define PREFETCH(x)__insn_prefetch_l1(x)
#define CACHE_LINE_SIZE 64
#ifdef LINUX
#define OS_TYPE "LINUX"
extern int __data_start[];
#define DATASTART ((ptr_t)__data_start)
#define LINUX_STACKBOTTOM
#define COUNT_UNMAPPED_REGIONS
#define DYNAMIC_LOADING
#endif
#endif
#ifdef RISCV
#define MACH_TYPE "RISC-V"
#define CPP_WORDSZ __riscv_xlen
#define ALIGNMENT (CPP_WORDSZ/8)
#ifdef FREEBSD
#define OS_TYPE "FREEBSD"
#ifndef GC_FREEBSD_THREADS
#define MPROTECT_VDB
#endif
#define SIG_SUSPEND SIGUSR1
#define SIG_THR_RESTART SIGUSR2
#define FREEBSD_STACKBOTTOM
#define DYNAMIC_LOADING
extern char etext[];
#define DATASTART GC_FreeBSDGetDataStart(0x1000,(ptr_t)etext)
#define DATASTART_USES_BSDGETDATASTART
#endif
#ifdef LINUX
#define OS_TYPE "LINUX"
extern int __data_start[] __attribute__((__weak__));
#define DATASTART ((ptr_t)__data_start)
#define LINUX_STACKBOTTOM
#define COUNT_UNMAPPED_REGIONS
#define DYNAMIC_LOADING
#endif
#endif
#if defined(__GLIBC__)&&!defined(DONT_USE_LIBC_PRIVATES)
#define USE_LIBC_PRIVATES
#endif
#ifdef NO_RETRY_GET_THREAD_CONTEXT
#undef RETRY_GET_THREAD_CONTEXT
#endif
#if defined(LINUX_STACKBOTTOM)&&defined(NO_PROC_STAT)&&!defined(USE_LIBC_PRIVATES)
#undef LINUX_STACKBOTTOM
#define HEURISTIC2
#endif
#if defined(USE_MMAP_ANON)&&!defined(USE_MMAP)
#define USE_MMAP 1
#elif (defined(LINUX)||defined(OPENBSD))&&defined(USE_MMAP)
#define USE_MMAP_ANON
#endif
#if defined(GC_LINUX_THREADS)&&defined(REDIRECT_MALLOC)&&!defined(USE_PROC_FOR_LIBRARIES)
#define USE_PROC_FOR_LIBRARIES
#endif
#ifndef STACK_GROWS_UP
#define STACK_GROWS_DOWN
#endif
#ifndef CPP_WORDSZ
#define CPP_WORDSZ 32
#endif
#ifndef OS_TYPE
#define OS_TYPE ""
#endif
#ifndef DATAEND
#if!defined(CPPCHECK)
extern int end[];
#endif
#define DATAEND ((ptr_t)(end))
#endif
#if defined(HOST_ANDROID)&&defined(__clang__)&&!defined(BROKEN_UUENDUU_SYM)
#undef DATAEND
#pragma weak __end__
extern int __end__[];
#define DATAEND (__end__!=0?(ptr_t)__end__:(ptr_t)_end)
#endif
#if (defined(SVR4)||defined(HOST_ANDROID)||defined(HOST_TIZEN))&&!defined(GETPAGESIZE)
EXTERN_C_END
#include <unistd.h>
EXTERN_C_BEGIN
#define GETPAGESIZE()(unsigned)sysconf(_SC_PAGESIZE)
#endif
#ifndef GETPAGESIZE
#if defined(SOLARIS)||defined(IRIX5)||defined(LINUX)||defined(NETBSD)||defined(FREEBSD)||defined(HPUX)
EXTERN_C_END
#include <unistd.h>
EXTERN_C_BEGIN
#endif
#define GETPAGESIZE()(unsigned)getpagesize()
#endif
#if defined(HOST_ANDROID)&&!(__ANDROID_API__>=23)&&((defined(MIPS)&&(CPP_WORDSZ==32))||defined(ARM32)||defined(I386))
#define USE_TKILL_ON_ANDROID
#endif
#if defined(SOLARIS)||defined(DRSNX)||defined(UTS4)
#define SVR4
#endif
#if defined(SOLARIS)||defined(DRSNX)
#define SOLARISDL
#define SUNOS5SIGS
#endif
#if defined(HPUX)
#define SUNOS5SIGS
#endif
#if defined(FREEBSD)&&(defined(__DragonFly__)||__FreeBSD__>=4||(__FreeBSD_kernel__>=4))
#define SUNOS5SIGS
#endif
#if!defined(GC_EXPLICIT_SIGNALS_UNBLOCK)&&defined(SUNOS5SIGS)&&!defined(GC_NO_PTHREAD_SIGMASK)
#define GC_EXPLICIT_SIGNALS_UNBLOCK
#endif
#if!defined(NO_SIGNALS_UNBLOCK_IN_MAIN)&&defined(GC_NO_PTHREAD_SIGMASK)
#define NO_SIGNALS_UNBLOCK_IN_MAIN
#endif
#if!defined(NO_MARKER_SPECIAL_SIGMASK)&&(defined(NACL)||defined(GC_WIN32_PTHREADS))
#define NO_MARKER_SPECIAL_SIGMASK
#endif
#ifdef GC_NETBSD_THREADS
#define SIGRTMIN 33
#define SIGRTMAX 63
#define GC_NETBSD_THREADS_WORKAROUND
#endif
#ifdef GC_OPENBSD_THREADS
EXTERN_C_END
#include <sys/param.h>
EXTERN_C_BEGIN
#if OpenBSD < 201211
#define GC_OPENBSD_UTHREADS 1
#endif
#endif
#if defined(SVR4)||defined(LINUX)||defined(IRIX5)||defined(HPUX)||defined(OPENBSD)||defined(NETBSD)||defined(FREEBSD)||defined(DGUX)||defined(BSD)||defined(HAIKU)||defined(HURD)||defined(AIX)||defined(DARWIN)||defined(OSF1)
#define UNIX_LIKE
#endif
#if defined(CPPCHECK)
#undef CPP_WORDSZ
#define CPP_WORDSZ (__SIZEOF_POINTER__*8)
#elif CPP_WORDSZ!=32&&CPP_WORDSZ!=64
#error Bad word size
#endif
#if!defined(ALIGNMENT)&&!defined(CPPCHECK)
#error Undefined ALIGNMENT
#endif
#ifdef PCR
#undef DYNAMIC_LOADING
#undef STACKBOTTOM
#undef HEURISTIC1
#undef HEURISTIC2
#undef PROC_VDB
#undef MPROTECT_VDB
#define PCR_VDB
#endif
#if!defined(STACKBOTTOM)&&(defined(ECOS)||defined(NOSYS))&&!defined(CPPCHECK)
#error Undefined STACKBOTTOM
#endif
#ifdef IGNORE_DYNAMIC_LOADING
#undef DYNAMIC_LOADING
#endif
#if defined(SMALL_CONFIG)&&!defined(GC_DISABLE_INCREMENTAL)
#define GC_DISABLE_INCREMENTAL
#endif
#if (defined(MSWIN32)||defined(MSWINCE))&&!defined(USE_WINALLOC)
#define USE_WINALLOC 1
#endif
#ifdef USE_WINALLOC
#undef USE_MMAP
#endif
#if defined(DARWIN)||defined(FREEBSD)||defined(HAIKU)||defined(IRIX5)||defined(LINUX)||defined(NETBSD)||defined(OPENBSD)||defined(SOLARIS)||((defined(CYGWIN32)||defined(USE_MMAP)||defined(USE_MUNMAP))&&!defined(USE_WINALLOC))
#define MMAP_SUPPORTED
#endif
#if defined(USE_MUNMAP)&&!defined(MUNMAP_THRESHOLD)&&(defined(SN_TARGET_ORBIS)||defined(SN_TARGET_PS3)||defined(SN_TARGET_PSP2)||defined(MSWIN_XBOX1))
#define MUNMAP_THRESHOLD 2
#endif
#if defined(USE_MUNMAP)&&defined(COUNT_UNMAPPED_REGIONS)&&!defined(GC_UNMAPPED_REGIONS_SOFT_LIMIT)
#if defined(__DragonFly__)
#define GC_UNMAPPED_REGIONS_SOFT_LIMIT (1000000/4)
#else
#define GC_UNMAPPED_REGIONS_SOFT_LIMIT 16384
#endif
#endif
#if defined(GC_DISABLE_INCREMENTAL)||defined(DEFAULT_VDB)
#undef GWW_VDB
#undef MPROTECT_VDB
#undef PCR_VDB
#undef PROC_VDB
#endif
#ifdef GC_DISABLE_INCREMENTAL
#undef CHECKSUMS
#endif
#ifdef USE_GLOBAL_ALLOC
#undef GWW_VDB
#endif
#if defined(BASE_ATOMIC_OPS_EMULATED)
#undef MPROTECT_VDB
#endif
#if defined(USE_PROC_FOR_LIBRARIES)&&defined(GC_LINUX_THREADS)
#undef MPROTECT_VDB
#endif
#if defined(MPROTECT_VDB)&&defined(GC_PREFER_MPROTECT_VDB)
#undef PCR_VDB
#undef PROC_VDB
#endif
#ifdef PROC_VDB
#undef MPROTECT_VDB
#endif
#if defined(MPROTECT_VDB)&&!defined(MSWIN32)&&!defined(MSWINCE)
#include <signal.h>
#endif
#if defined(SIGBUS)&&!defined(HAVE_SIGBUS)&&!defined(CPPCHECK)
#define HAVE_SIGBUS
#endif
#ifndef SA_SIGINFO
#define NO_SA_SIGACTION
#endif
#if (defined(NO_SA_SIGACTION)||defined(GC_NO_SIGSETJMP))&&defined(MPROTECT_VDB)&&!defined(DARWIN)&&!defined(MSWIN32)&&!defined(MSWINCE)
#undef MPROTECT_VDB
#endif
#if!defined(PCR_VDB)&&!defined(PROC_VDB)&&!defined(MPROTECT_VDB)&&!defined(GWW_VDB)&&!defined(DEFAULT_VDB)&&!defined(GC_DISABLE_INCREMENTAL)
#define DEFAULT_VDB
#endif
#if ((defined(UNIX_LIKE)&&(defined(DARWIN)||defined(HAIKU)||defined(HURD)||defined(OPENBSD)||defined(ARM32)||defined(AVR32)||defined(MIPS)||defined(NIOS2)||defined(OR1K)))||(defined(LINUX)&&!defined(__gnu_linux__))||(defined(RTEMS)&&defined(I386))||defined(HOST_ANDROID))&&!defined(NO_GETCONTEXT)
#define NO_GETCONTEXT 1
#endif
#ifndef PREFETCH
#if GC_GNUC_PREREQ(3,0)&&!defined(NO_PREFETCH)
#define PREFETCH(x)__builtin_prefetch((x),0,0)
#else
#define PREFETCH(x)(void)0
#endif
#endif
#ifndef GC_PREFETCH_FOR_WRITE
#if GC_GNUC_PREREQ(3,0)&&!defined(GC_NO_PREFETCH_FOR_WRITE)
#define GC_PREFETCH_FOR_WRITE(x)__builtin_prefetch((x),1)
#else
#define GC_PREFETCH_FOR_WRITE(x)(void)0
#endif
#endif
#ifndef CACHE_LINE_SIZE
#define CACHE_LINE_SIZE 32
#endif
#ifndef STATIC
#ifdef GC_ASSERTIONS
#define STATIC
#else
#define STATIC static
#endif
#endif
#if defined(LINUX)&&(defined(USE_PROC_FOR_LIBRARIES)||defined(IA64)||!defined(SMALL_CONFIG))
#define NEED_PROC_MAPS
#endif
#if defined(LINUX)||defined(HURD)||defined(__GLIBC__)
#define REGISTER_LIBRARIES_EARLY
#endif
#if defined(SEARCH_FOR_DATA_START)
extern ptr_t GC_data_start;
#define DATASTART GC_data_start
#endif
#ifndef HEAP_START
#define HEAP_START ((ptr_t)0)
#endif
#ifndef CLEAR_DOUBLE
#define CLEAR_DOUBLE(x)(((word*)(x))[0]=0,((word*)(x))[1]=0)
#endif
#if defined(GC_LINUX_THREADS)&&defined(REDIRECT_MALLOC)&&!defined(INCLUDE_LINUX_THREAD_DESCR)
#define INCLUDE_LINUX_THREAD_DESCR
#endif
#if!defined(CPPCHECK)
#if defined(GC_IRIX_THREADS)&&!defined(IRIX5)
#error Inconsistent configuration
#endif
#if defined(GC_LINUX_THREADS)&&!defined(LINUX)&&!defined(NACL)
#error Inconsistent configuration
#endif
#if defined(GC_NETBSD_THREADS)&&!defined(NETBSD)
#error Inconsistent configuration
#endif
#if defined(GC_FREEBSD_THREADS)&&!defined(FREEBSD)
#error Inconsistent configuration
#endif
#if defined(GC_SOLARIS_THREADS)&&!defined(SOLARIS)
#error Inconsistent configuration
#endif
#if defined(GC_HPUX_THREADS)&&!defined(HPUX)
#error Inconsistent configuration
#endif
#if defined(GC_AIX_THREADS)&&!defined(_AIX)
#error Inconsistent configuration
#endif
#if defined(GC_WIN32_THREADS)&&!defined(CYGWIN32)&&!defined(MSWIN32)&&!defined(MSWINCE)&&!defined(MSWIN_XBOX1)
#error Inconsistent configuration
#endif
#if defined(GC_WIN32_PTHREADS)&&defined(CYGWIN32)
#error Inconsistent configuration
#endif
#endif
#if defined(PCR)||defined(GC_WIN32_THREADS)||defined(GC_PTHREADS)||((defined(NN_PLATFORM_CTR)||defined(NINTENDO_SWITCH)||defined(SN_TARGET_ORBIS)||defined(SN_TARGET_PS3)||defined(SN_TARGET_PSP2))&&defined(GC_THREADS))
#define THREADS
#endif
#if defined(PARALLEL_MARK)&&!defined(THREADS)&&!defined(CPPCHECK)
#error Invalid config:PARALLEL_MARK requires GC_THREADS
#endif
#if defined(GWW_VDB)&&!defined(USE_WINALLOC)&&!defined(CPPCHECK)
#error Invalid config:GWW_VDB requires USE_WINALLOC
#endif
#if (((defined(MSWIN32)||defined(MSWINCE))&&!defined(__GNUC__))||(defined(MSWIN32)&&defined(I386))||(defined(USE_PROC_FOR_LIBRARIES)&&defined(THREADS)))&&!defined(NO_CRT)&&!defined(NO_WRAP_MARK_SOME)
#define WRAP_MARK_SOME
#endif
#if defined(PARALLEL_MARK)&&!defined(DEFAULT_STACK_MAYBE_SMALL)&&(defined(HPUX)||defined(GC_DGUX386_THREADS)||defined(NO_GETCONTEXT))
#define DEFAULT_STACK_MAYBE_SMALL
#endif
#ifdef PARALLEL_MARK
#define MIN_STACK_SIZE (8*HBLKSIZE*sizeof(word))
#endif
#if defined(HOST_ANDROID)&&!defined(THREADS)&&!defined(USE_GET_STACKBASE_FOR_MAIN)
#define USE_GET_STACKBASE_FOR_MAIN
#endif
#if ((defined(FREEBSD)&&defined(__GLIBC__))||defined(LINUX)||defined(NETBSD)||defined(HOST_ANDROID))&&!defined(NO_PTHREAD_GETATTR_NP)
#define HAVE_PTHREAD_GETATTR_NP 1
#elif defined(FREEBSD)&&!defined(__GLIBC__)&&!defined(NO_PTHREAD_ATTR_GET_NP)
#define HAVE_PTHREAD_NP_H 1
#define HAVE_PTHREAD_ATTR_GET_NP 1
#endif
#if defined(UNIX_LIKE)&&defined(THREADS)&&!defined(NO_CANCEL_SAFE)&&!defined(HOST_ANDROID)
#define CANCEL_SAFE
#endif
#ifdef CANCEL_SAFE
#define IF_CANCEL(x)x
#else
#define IF_CANCEL(x)
#endif
#if!defined(CAN_HANDLE_FORK)&&!defined(NO_HANDLE_FORK)&&!defined(HAVE_NO_FORK)&&((defined(GC_PTHREADS)&&!defined(NACL)&&!defined(GC_WIN32_PTHREADS)&&!defined(USE_WINALLOC))||(defined(DARWIN)&&defined(MPROTECT_VDB))||defined(HANDLE_FORK))
#define CAN_HANDLE_FORK
#endif
#if defined(CAN_HANDLE_FORK)&&!defined(CAN_CALL_ATFORK)&&!defined(HURD)&&!defined(SN_TARGET_ORBIS)&&!defined(HOST_TIZEN)&&(!defined(HOST_ANDROID)||__ANDROID_API__>=21)
#define CAN_CALL_ATFORK
#endif
#if!defined(CAN_HANDLE_FORK)&&!defined(HAVE_NO_FORK)&&(defined(MSWIN32)||defined(MSWINCE)||defined(DOS4GW)||defined(OS2)||defined(SYMBIAN))
#define HAVE_NO_FORK
#endif
#if!defined(USE_MARK_BITS)&&!defined(USE_MARK_BYTES)&&defined(PARALLEL_MARK)
#define USE_MARK_BYTES
#endif
#if (defined(MSWINCE)&&!defined(__CEGCC__)||defined(MSWINRT_FLAVOR))&&!defined(NO_GETENV)
#define NO_GETENV
#endif
#if (defined(NO_GETENV)||defined(MSWINCE))&&!defined(NO_GETENV_WIN32)
#define NO_GETENV_WIN32
#endif
#if!defined(MSGBOX_ON_ERROR)&&!defined(NO_MSGBOX_ON_ERROR)&&!defined(SMALL_CONFIG)&&defined(MSWIN32)&&!defined(MSWINRT_FLAVOR)&&!defined(MSWIN_XBOX1)
#define MSGBOX_ON_ERROR
#endif
#ifndef STRTOULL
#if defined(_WIN64)&&!defined(__GNUC__)
#define STRTOULL _strtoui64
#elif defined(_LLP64)||defined(__LLP64__)||defined(_WIN64)
#define STRTOULL strtoull
#else
#define STRTOULL strtoul
#endif
#endif
#ifndef GC_WORD_C
#if defined(_WIN64)&&!defined(__GNUC__)
#define GC_WORD_C(val)val##ui64
#elif defined(_LLP64)||defined(__LLP64__)||defined(_WIN64)
#define GC_WORD_C(val)val##ULL
#else
#define GC_WORD_C(val)((word)val##UL)
#endif
#endif
#if defined(__has_feature)
#if __has_feature(address_sanitizer)&&!defined(ADDRESS_SANITIZER)
#define ADDRESS_SANITIZER
#endif
#if __has_feature(memory_sanitizer)&&!defined(MEMORY_SANITIZER)
#define MEMORY_SANITIZER
#endif
#if __has_feature(thread_sanitizer)&&!defined(THREAD_SANITIZER)
#define THREAD_SANITIZER
#endif
#else
#ifdef __SANITIZE_ADDRESS__
#define ADDRESS_SANITIZER
#endif
#endif
#if defined(SPARC)
#define ASM_CLEAR_CODE
#endif
#if defined(SPARC)
#define CAN_SAVE_CALL_ARGS
#endif
#if (defined(I386)||defined(X86_64))&&(defined(LINUX)||defined(__GLIBC__))
#define CAN_SAVE_CALL_ARGS
#endif
#if defined(SAVE_CALL_COUNT)&&!defined(GC_ADD_CALLER)&&defined(GC_CAN_SAVE_CALL_STACKS)
#define SAVE_CALL_CHAIN
#endif
#ifdef SAVE_CALL_CHAIN
#if defined(SAVE_CALL_NARGS)&&defined(CAN_SAVE_CALL_ARGS)
#define NARGS SAVE_CALL_NARGS
#else
#define NARGS 0
#endif
#endif
#ifdef SAVE_CALL_CHAIN
#if!defined(SAVE_CALL_COUNT)||defined(CPPCHECK)
#define NFRAMES 6
#else
#define NFRAMES ((SAVE_CALL_COUNT+1)&~1)
#endif
#define NEED_CALLINFO
#endif
#ifdef GC_ADD_CALLER
#define NFRAMES 1
#define NARGS 0
#define NEED_CALLINFO
#endif
#if (defined(FREEBSD)||(defined(DARWIN)&&!defined(_POSIX_C_SOURCE))||(defined(SOLARIS)&&(!defined(_XOPEN_SOURCE)||defined(__EXTENSIONS__)))||defined(LINUX))&&!defined(HAVE_DLADDR)
#define HAVE_DLADDR 1
#endif
#if defined(MAKE_BACK_GRAPH)&&!defined(DBG_HDRS_ALL)
#define DBG_HDRS_ALL 1
#endif
#if defined(POINTER_MASK)&&!defined(POINTER_SHIFT)
#define POINTER_SHIFT 0
#endif
#if defined(POINTER_SHIFT)&&!defined(POINTER_MASK)
#define POINTER_MASK ((word)(-1))
#endif
#if!defined(FIXUP_POINTER)&&defined(POINTER_MASK)
#define FIXUP_POINTER(p)(p=((p)&POINTER_MASK)<<POINTER_SHIFT)
#endif
#if defined(FIXUP_POINTER)
#define NEED_FIXUP_POINTER
#else
#define FIXUP_POINTER(p)
#endif
#if!defined(MARK_BIT_PER_GRANULE)&&!defined(MARK_BIT_PER_OBJ)
#define MARK_BIT_PER_GRANULE
#endif
#if!defined(CPPCHECK)
#if defined(MARK_BIT_PER_GRANULE)&&defined(MARK_BIT_PER_OBJ)
#error Define only one of MARK_BIT_PER_GRANULE and MARK_BIT_PER_OBJ
#endif
#if defined(STACK_GROWS_UP)&&defined(STACK_GROWS_DOWN)
#error Only one of STACK_GROWS_UP and STACK_GROWS_DOWN should be defined
#endif
#if!defined(STACK_GROWS_UP)&&!defined(STACK_GROWS_DOWN)
#error One of STACK_GROWS_UP and STACK_GROWS_DOWN should be defined
#endif
#if defined(REDIRECT_MALLOC)&&defined(THREADS)&&!defined(LINUX)&&!defined(REDIRECT_MALLOC_IN_HEADER)
#error REDIRECT_MALLOC with THREADS works at most on Linux
#endif
#endif
#ifdef GC_PRIVATE_H
struct hblk;
#if defined(PCR)
char*real_malloc(size_t bytes);
#define GET_MEM(bytes)HBLKPTR(real_malloc(SIZET_SAT_ADD(bytes,GC_page_size))+GC_page_size-1)
#elif defined(OS2)
void*os2_alloc(size_t bytes);
#define GET_MEM(bytes)HBLKPTR((ptr_t)os2_alloc( SIZET_SAT_ADD(bytes,GC_page_size))+GC_page_size-1)
#elif defined(NEXT)||defined(DOS4GW)||defined(NONSTOP)||(defined(AMIGA)&&!defined(GC_AMIGA_FASTALLOC))||(defined(SOLARIS)&&!defined(USE_MMAP))||defined(RTEMS)||defined(__CC_ARM)
#define GET_MEM(bytes)HBLKPTR((size_t)calloc(1,SIZET_SAT_ADD(bytes,GC_page_size))+GC_page_size - 1)
#elif defined(MSWIN_XBOX1)
ptr_t GC_durango_get_mem(size_t bytes);
#define GET_MEM(bytes)(struct hblk*)GC_durango_get_mem(bytes)
#elif defined(MSWIN32)||defined(CYGWIN32)
ptr_t GC_win32_get_mem(size_t bytes);
#define GET_MEM(bytes)(struct hblk*)GC_win32_get_mem(bytes)
#elif defined(MACOS)
#if defined(USE_TEMPORARY_MEMORY)
Ptr GC_MacTemporaryNewPtr(size_t size,Boolean clearMemory);
#define GET_MEM(bytes)HBLKPTR(GC_MacTemporaryNewPtr( SIZET_SAT_ADD(bytes,GC_page_size),true)+GC_page_size-1)
#else
#define GET_MEM(bytes)HBLKPTR(NewPtrClear(SIZET_SAT_ADD(bytes,GC_page_size))+GC_page_size-1)
#endif
#elif defined(MSWINCE)
ptr_t GC_wince_get_mem(size_t bytes);
#define GET_MEM(bytes)(struct hblk*)GC_wince_get_mem(bytes)
#elif defined(AMIGA)&&defined(GC_AMIGA_FASTALLOC)
void*GC_amiga_get_mem(size_t bytes);
#define GET_MEM(bytes)HBLKPTR((size_t)GC_amiga_get_mem( SIZET_SAT_ADD(bytes,GC_page_size))+GC_page_size-1)
#elif defined(SN_TARGET_ORBIS)
void*ps4_get_mem(size_t bytes);
#define GET_MEM(bytes)(struct hblk*)ps4_get_mem(bytes)
#elif defined(SN_TARGET_PS3)
void*ps3_get_mem(size_t bytes);
#define GET_MEM(bytes)(struct hblk*)ps3_get_mem(bytes)
#elif defined(SN_TARGET_PSP2)
void*psp2_get_mem(size_t bytes);
#define GET_MEM(bytes)(struct hblk*)psp2_get_mem(bytes)
#elif defined(NINTENDO_SWITCH)
void*switch_get_mem(size_t bytes);
#define GET_MEM(bytes)(struct hblk*)switch_get_mem(bytes)
#elif defined(HAIKU)
ptr_t GC_haiku_get_mem(size_t bytes);
#define GET_MEM(bytes)(struct hblk*)GC_haiku_get_mem(bytes)
#else
ptr_t GC_unix_get_mem(size_t bytes);
#define GET_MEM(bytes)(struct hblk*)GC_unix_get_mem(bytes)
#endif
#endif
EXTERN_C_END
#endif
#if!defined(GC_ATOMIC_UNCOLLECTABLE)&&defined(ATOMIC_UNCOLLECTABLE)
#define GC_ATOMIC_UNCOLLECTABLE
#endif
#ifndef GC_INNER
#if defined(GC_DLL)&&defined(__GNUC__)&&!defined(MSWIN32)&&!defined(MSWINCE)&&!defined(CYGWIN32)
#if GC_GNUC_PREREQ(4,0)&&!defined(GC_NO_VISIBILITY)
#define GC_INNER __attribute__((__visibility__("hidden")))
#else
#define GC_INNER
#endif
#else
#define GC_INNER
#endif
#define GC_EXTERN extern GC_INNER
#endif
#ifdef __cplusplus
#define REGISTER
#else
#define REGISTER register
#endif
#if defined(CPPCHECK)
#define MACRO_BLKSTMT_BEGIN {
#define MACRO_BLKSTMT_END }
#else
#define MACRO_BLKSTMT_BEGIN do {
#define MACRO_BLKSTMT_END } while (0)
#endif
#if defined(M68K)&&defined(__GNUC__)
#define GC_ATTR_WORD_ALIGNED __attribute__((__aligned__(sizeof(word))))
#else
#define GC_ATTR_WORD_ALIGNED
#endif
#ifndef HEADERS_H
#ifndef GC_HEADERS_H
#define GC_HEADERS_H
#if CPP_WORDSZ!=32&&CPP_WORDSZ < 36&&!defined(CPPCHECK)
#error Get a real machine
#endif
EXTERN_C_BEGIN
typedef struct hblkhdr hdr;
#if CPP_WORDSZ > 32
#define HASH_TL
#endif
#if defined(LARGE_CONFIG)||!defined(SMALL_CONFIG)
#define LOG_BOTTOM_SZ 10
#else
#define LOG_BOTTOM_SZ 11
#endif
#define BOTTOM_SZ (1<<LOG_BOTTOM_SZ)
#ifndef HASH_TL
#define LOG_TOP_SZ (WORDSZ - LOG_BOTTOM_SZ - LOG_HBLKSIZE)
#else
#define LOG_TOP_SZ 11
#endif
#define TOP_SZ (1<<LOG_TOP_SZ)
#ifdef COUNT_HDR_CACHE_HITS
extern word GC_hdr_cache_hits;
extern word GC_hdr_cache_misses;
#define HC_HIT()(void)(++GC_hdr_cache_hits)
#define HC_MISS()(void)(++GC_hdr_cache_misses)
#else
#define HC_HIT()
#define HC_MISS()
#endif
typedef struct hce {
word block_addr;
hdr*hce_hdr;
} hdr_cache_entry;
#define HDR_CACHE_SIZE 8
#define DECLARE_HDR_CACHE hdr_cache_entry hdr_cache[HDR_CACHE_SIZE]
#define INIT_HDR_CACHE BZERO(hdr_cache,sizeof(hdr_cache))
#define HCE(h)(hdr_cache+(((word)(h)>>LOG_HBLKSIZE)&(HDR_CACHE_SIZE-1)))
#define HCE_VALID_FOR(hce,h)((hce)->block_addr==((word)(h)>>LOG_HBLKSIZE))
#define HCE_HDR(h)((hce)->hce_hdr)
#ifdef PRINT_BLACK_LIST
GC_INNER hdr*GC_header_cache_miss(ptr_t p,hdr_cache_entry*hce,
ptr_t source);
#define HEADER_CACHE_MISS(p,hce,source)GC_header_cache_miss(p,hce,source)
#else
GC_INNER hdr*GC_header_cache_miss(ptr_t p,hdr_cache_entry*hce);
#define HEADER_CACHE_MISS(p,hce,source)GC_header_cache_miss(p,hce)
#endif
#define HC_GET_HDR(p,hhdr,source){ hdr_cache_entry*hce=HCE(p);if (EXPECT(HCE_VALID_FOR(hce,p),TRUE)){ HC_HIT();hhdr=hce->hce_hdr;} else { hhdr=HEADER_CACHE_MISS(p,hce,source);if (NULL==hhdr)break;} }
typedef struct bi {
hdr*index[BOTTOM_SZ];
struct bi*asc_link;
struct bi*desc_link;
word key;
#ifdef HASH_TL
struct bi*hash_link;
#endif
} bottom_index;
#define MAX_JUMP (HBLKSIZE - 1)
#define HDR_FROM_BI(bi,p)((bi)->index[((word)(p)>>LOG_HBLKSIZE)&(BOTTOM_SZ - 1)])
#ifndef HASH_TL
#define BI(p)(GC_top_index [(word)(p)>>(LOG_BOTTOM_SZ+LOG_HBLKSIZE)])
#define HDR_INNER(p)HDR_FROM_BI(BI(p),p)
#ifdef SMALL_CONFIG
#define HDR(p)GC_find_header((ptr_t)(p))
#else
#define HDR(p)HDR_INNER(p)
#endif
#define GET_BI(p,bottom_indx)(void)((bottom_indx)=BI(p))
#define GET_HDR(p,hhdr)(void)((hhdr)=HDR(p))
#define SET_HDR(p,hhdr)(void)(HDR_INNER(p)=(hhdr))
#define GET_HDR_ADDR(p,ha)(void)((ha)=&HDR_INNER(p))
#else
#define TL_HASH(hi)((hi)&(TOP_SZ - 1))
#define GET_BI(p,bottom_indx)do { REGISTER word hi=(word)(p)>>(LOG_BOTTOM_SZ+LOG_HBLKSIZE);REGISTER bottom_index*_bi=GC_top_index[TL_HASH(hi)];while (_bi->key!=hi&&_bi!=GC_all_nils)_bi=_bi->hash_link;(bottom_indx)=_bi;} while (0)
#define GET_HDR_ADDR(p,ha)do { REGISTER bottom_index*bi;GET_BI(p,bi);(ha)=&HDR_FROM_BI(bi,p);} while (0)
#define GET_HDR(p,hhdr)do { REGISTER hdr**_ha;GET_HDR_ADDR(p,_ha);(hhdr)=*_ha;} while (0)
#define SET_HDR(p,hhdr)do { REGISTER hdr**_ha;GET_HDR_ADDR(p,_ha);*_ha=(hhdr);} while (0)
#define HDR(p)GC_find_header((ptr_t)(p))
#endif
#define IS_FORWARDING_ADDR_OR_NIL(hhdr)((size_t)(hhdr)<=MAX_JUMP)
#define FORWARDED_ADDR(h,hhdr)((struct hblk*)(h)- (size_t)(hhdr))
EXTERN_C_END
#endif
#endif
#ifndef GC_ATTR_NO_SANITIZE_ADDR
#ifndef ADDRESS_SANITIZER
#define GC_ATTR_NO_SANITIZE_ADDR
#elif GC_CLANG_PREREQ(3,8)
#define GC_ATTR_NO_SANITIZE_ADDR __attribute__((no_sanitize("address")))
#else
#define GC_ATTR_NO_SANITIZE_ADDR __attribute__((no_sanitize_address))
#endif
#endif
#ifndef GC_ATTR_NO_SANITIZE_MEMORY
#ifndef MEMORY_SANITIZER
#define GC_ATTR_NO_SANITIZE_MEMORY
#elif GC_CLANG_PREREQ(3,8)
#define GC_ATTR_NO_SANITIZE_MEMORY __attribute__((no_sanitize("memory")))
#else
#define GC_ATTR_NO_SANITIZE_MEMORY __attribute__((no_sanitize_memory))
#endif
#endif
#ifndef GC_ATTR_NO_SANITIZE_THREAD
#ifndef THREAD_SANITIZER
#define GC_ATTR_NO_SANITIZE_THREAD
#elif GC_CLANG_PREREQ(3,8)
#define GC_ATTR_NO_SANITIZE_THREAD __attribute__((no_sanitize("thread")))
#else
#define GC_ATTR_NO_SANITIZE_THREAD __attribute__((no_sanitize_thread))
#endif
#endif
#ifndef GC_ATTR_UNUSED
#if GC_GNUC_PREREQ(3,4)
#define GC_ATTR_UNUSED __attribute__((__unused__))
#else
#define GC_ATTR_UNUSED
#endif
#endif
#ifdef HAVE_CONFIG_H
#define GC_INLINE static inline
#elif defined(_MSC_VER)||defined(__INTEL_COMPILER)||defined(__DMC__)||(GC_GNUC_PREREQ(3,0)&&defined(__STRICT_ANSI__))||defined(__WATCOMC__)
#define GC_INLINE static __inline
#elif GC_GNUC_PREREQ(3,0)||defined(__sun)
#define GC_INLINE static inline
#else
#define GC_INLINE static
#endif
#ifndef GC_ATTR_NOINLINE
#if GC_GNUC_PREREQ(4,0)
#define GC_ATTR_NOINLINE __attribute__((__noinline__))
#elif _MSC_VER>=1400
#define GC_ATTR_NOINLINE __declspec(noinline)
#else
#define GC_ATTR_NOINLINE
#endif
#endif
#ifndef GC_API_OSCALL
#if defined(__GNUC__)
#if GC_GNUC_PREREQ(4,0)&&!defined(GC_NO_VISIBILITY)
#define GC_API_OSCALL extern __attribute__((__visibility__("default")))
#else
#define GC_API_OSCALL extern
#endif
#else
#define GC_API_OSCALL GC_API
#endif
#endif
#ifndef GC_API_PRIV
#define GC_API_PRIV GC_API
#endif
#if defined(THREADS)&&!defined(NN_PLATFORM_CTR)
#ifndef GC_ATOMIC_OPS_H
#define GC_ATOMIC_OPS_H
#ifdef GC_BUILTIN_ATOMIC
#ifdef __cplusplus
extern "C" {
#endif
typedef GC_word AO_t;
#ifdef GC_PRIVATE_H
#define AO_INLINE GC_INLINE
#else
#define AO_INLINE static __inline
#endif
typedef unsigned char AO_TS_t;
#define AO_TS_CLEAR 0
#define AO_TS_INITIALIZER (AO_TS_t)AO_TS_CLEAR
#if defined(__GCC_ATOMIC_TEST_AND_SET_TRUEVAL)&&!defined(CPPCHECK)
#define AO_TS_SET __GCC_ATOMIC_TEST_AND_SET_TRUEVAL
#else
#define AO_TS_SET (AO_TS_t)1
#endif
#define AO_CLEAR(p)__atomic_clear(p,__ATOMIC_RELEASE)
#define AO_test_and_set_acquire(p)__atomic_test_and_set(p,__ATOMIC_ACQUIRE)
#define AO_HAVE_test_and_set_acquire
#define AO_compiler_barrier()__atomic_signal_fence(__ATOMIC_SEQ_CST)
#define AO_nop_full()__atomic_thread_fence(__ATOMIC_SEQ_CST)
#define AO_HAVE_nop_full
#define AO_fetch_and_add(p,v)__atomic_fetch_add(p,v,__ATOMIC_RELAXED)
#define AO_HAVE_fetch_and_add
#define AO_fetch_and_add1(p)AO_fetch_and_add(p,1)
#define AO_HAVE_fetch_and_add1
#define AO_or(p,v)(void)__atomic_or_fetch(p,v,__ATOMIC_RELAXED)
#define AO_HAVE_or
#define AO_load(p)__atomic_load_n(p,__ATOMIC_RELAXED)
#define AO_HAVE_load
#define AO_load_acquire(p)__atomic_load_n(p,__ATOMIC_ACQUIRE)
#define AO_HAVE_load_acquire
#define AO_load_acquire_read(p)AO_load_acquire(p)
#define AO_HAVE_load_acquire_read
#define AO_store(p,v)__atomic_store_n(p,v,__ATOMIC_RELAXED)
#define AO_HAVE_store
#define AO_store_release(p,v)__atomic_store_n(p,v,__ATOMIC_RELEASE)
#define AO_HAVE_store_release
#define AO_store_release_write(p,v)AO_store_release(p,v)
#define AO_HAVE_store_release_write
#define AO_char_load(p)__atomic_load_n(p,__ATOMIC_RELAXED)
#define AO_HAVE_char_load
#define AO_char_store(p,v)__atomic_store_n(p,v,__ATOMIC_RELAXED)
#define AO_HAVE_char_store
#ifdef AO_REQUIRE_CAS
AO_INLINE int
AO_compare_and_swap(volatile AO_t*p,AO_t ov,AO_t nv)
{
return (int)__atomic_compare_exchange_n(p,&ov,nv,0,
__ATOMIC_RELAXED,__ATOMIC_RELAXED);
}
AO_INLINE int
AO_compare_and_swap_release(volatile AO_t*p,AO_t ov,AO_t nv)
{
return (int)__atomic_compare_exchange_n(p,&ov,nv,0,
__ATOMIC_RELEASE,__ATOMIC_RELAXED);
}
#define AO_HAVE_compare_and_swap_release
#endif
#ifdef __cplusplus
}
#endif
#ifndef NO_LOCKFREE_AO_OR
#define HAVE_LOCKFREE_AO_OR 1
#endif
#else
#include "atomic_ops.h"
#if (!defined(AO_HAVE_load)||!defined(AO_HAVE_store))&&!defined(CPPCHECK)
#error AO_load or AO_store is missing;probably old version of atomic_ops
#endif
#endif
#endif
#ifndef AO_HAVE_compiler_barrier
#define AO_HAVE_compiler_barrier 1
#endif
#endif
#if defined(MSWIN32)||defined(MSWINCE)||defined(CYGWIN32)
#ifndef WIN32_LEAN_AND_MEAN
#define WIN32_LEAN_AND_MEAN 1
#endif
#define NOSERVICE
#include <windows.h>
#include <winbase.h>
#endif
#ifndef GC_LOCKS_H
#define GC_LOCKS_H
#ifdef THREADS
#ifdef PCR
#include <base/PCR_Base.h>
#include <th/PCR_Th.h>
#endif
EXTERN_C_BEGIN
#ifdef PCR
GC_EXTERN PCR_Th_ML GC_allocate_ml;
#if defined(CPPCHECK)
#define DCL_LOCK_STATE
#else
#define DCL_LOCK_STATE PCR_ERes GC_fastLockRes;PCR_sigset_t GC_old_sig_mask
#endif
#define UNCOND_LOCK()PCR_Th_ML_Acquire(&GC_allocate_ml)
#define UNCOND_UNLOCK()PCR_Th_ML_Release(&GC_allocate_ml)
#elif defined(NN_PLATFORM_CTR)||defined(NINTENDO_SWITCH)
extern void GC_lock(void);
extern void GC_unlock(void);
#define UNCOND_LOCK()GC_lock()
#define UNCOND_UNLOCK()GC_unlock()
#endif
#if (!defined(AO_HAVE_test_and_set_acquire)||defined(GC_RTEMS_PTHREADS)||defined(SN_TARGET_ORBIS)||defined(SN_TARGET_PS3)||defined(GC_WIN32_THREADS)||defined(BASE_ATOMIC_OPS_EMULATED)||defined(LINT2))&&defined(GC_PTHREADS)
#define USE_PTHREAD_LOCKS
#undef USE_SPIN_LOCK
#endif
#if defined(GC_WIN32_THREADS)&&!defined(USE_PTHREAD_LOCKS)
#define NO_THREAD (DWORD)(-1)
GC_EXTERN CRITICAL_SECTION GC_allocate_ml;
#ifdef GC_ASSERTIONS
GC_EXTERN DWORD GC_lock_holder;
#define SET_LOCK_HOLDER()GC_lock_holder=GetCurrentThreadId()
#define UNSET_LOCK_HOLDER()GC_lock_holder=NO_THREAD
#define I_HOLD_LOCK()(!GC_need_to_lock||GC_lock_holder==GetCurrentThreadId())
#ifdef THREAD_SANITIZER
#define I_DONT_HOLD_LOCK()TRUE
#else
#define I_DONT_HOLD_LOCK()(!GC_need_to_lock||GC_lock_holder!=GetCurrentThreadId())
#endif
#define UNCOND_LOCK(){ GC_ASSERT(I_DONT_HOLD_LOCK());EnterCriticalSection(&GC_allocate_ml);SET_LOCK_HOLDER();}
#define UNCOND_UNLOCK(){ GC_ASSERT(I_HOLD_LOCK());UNSET_LOCK_HOLDER();LeaveCriticalSection(&GC_allocate_ml);}
#else
#define UNCOND_LOCK()EnterCriticalSection(&GC_allocate_ml)
#define UNCOND_UNLOCK()LeaveCriticalSection(&GC_allocate_ml)
#endif
#elif defined(GC_PTHREADS)
EXTERN_C_END
#include <pthread.h>
EXTERN_C_BEGIN
#if!defined(GC_WIN32_PTHREADS)
#define NUMERIC_THREAD_ID(id)((unsigned long)(id))
#define THREAD_EQUAL(id1,id2)((id1)==(id2))
#define NUMERIC_THREAD_ID_UNIQUE
#elif defined(__WINPTHREADS_VERSION_MAJOR)
#define NUMERIC_THREAD_ID(id)((unsigned long)(id))
#define THREAD_EQUAL(id1,id2)((id1)==(id2))
#ifndef _WIN64
#define NUMERIC_THREAD_ID_UNIQUE
#endif
#else
#define NUMERIC_THREAD_ID(id)((unsigned long)(word)(id.p))
#define THREAD_EQUAL(id1,id2)((id1.p==id2.p)&&(id1.x==id2.x))
#undef NUMERIC_THREAD_ID_UNIQUE
#endif
#define NO_THREAD ((unsigned long)(-1l))
#ifdef SN_TARGET_PSP2
EXTERN_C_END
#include "psp2-support.h"
EXTERN_C_BEGIN
GC_EXTERN WapiMutex GC_allocate_ml_PSP2;
#define UNCOND_LOCK(){ int res;GC_ASSERT(I_DONT_HOLD_LOCK());res=PSP2_MutexLock(&GC_allocate_ml_PSP2);GC_ASSERT(0==res);(void)res;SET_LOCK_HOLDER();}
#define UNCOND_UNLOCK(){ int res;GC_ASSERT(I_HOLD_LOCK());UNSET_LOCK_HOLDER();res=PSP2_MutexUnlock(&GC_allocate_ml_PSP2);GC_ASSERT(0==res);(void)res;}
#elif (!defined(THREAD_LOCAL_ALLOC)||defined(USE_SPIN_LOCK))&&!defined(USE_PTHREAD_LOCKS)
#undef USE_SPIN_LOCK
#define USE_SPIN_LOCK
GC_EXTERN volatile AO_TS_t GC_allocate_lock;
GC_INNER void GC_lock(void);
#ifdef GC_ASSERTIONS
#define UNCOND_LOCK(){ GC_ASSERT(I_DONT_HOLD_LOCK());if (AO_test_and_set_acquire(&GC_allocate_lock)==AO_TS_SET)GC_lock();SET_LOCK_HOLDER();}
#define UNCOND_UNLOCK(){ GC_ASSERT(I_HOLD_LOCK());UNSET_LOCK_HOLDER();AO_CLEAR(&GC_allocate_lock);}
#else
#define UNCOND_LOCK(){ if (AO_test_and_set_acquire(&GC_allocate_lock)==AO_TS_SET)GC_lock();}
#define UNCOND_UNLOCK()AO_CLEAR(&GC_allocate_lock)
#endif
#else
#ifndef USE_PTHREAD_LOCKS
#define USE_PTHREAD_LOCKS
#endif
#endif
#ifdef USE_PTHREAD_LOCKS
EXTERN_C_END
#include <pthread.h>
EXTERN_C_BEGIN
GC_EXTERN pthread_mutex_t GC_allocate_ml;
#ifdef GC_ASSERTIONS
GC_INNER void GC_lock(void);
#define UNCOND_LOCK(){ GC_ASSERT(I_DONT_HOLD_LOCK());GC_lock();SET_LOCK_HOLDER();}
#define UNCOND_UNLOCK(){ GC_ASSERT(I_HOLD_LOCK());UNSET_LOCK_HOLDER();pthread_mutex_unlock(&GC_allocate_ml);}
#else
#if defined(NO_PTHREAD_TRYLOCK)
#define UNCOND_LOCK()pthread_mutex_lock(&GC_allocate_ml)
#else
GC_INNER void GC_lock(void);
#define UNCOND_LOCK(){ if (0!=pthread_mutex_trylock(&GC_allocate_ml))GC_lock();}
#endif
#define UNCOND_UNLOCK()pthread_mutex_unlock(&GC_allocate_ml)
#endif
#endif
#ifdef GC_ASSERTIONS
GC_EXTERN unsigned long GC_lock_holder;
#define SET_LOCK_HOLDER()GC_lock_holder=NUMERIC_THREAD_ID(pthread_self())
#define UNSET_LOCK_HOLDER()GC_lock_holder=NO_THREAD
#define I_HOLD_LOCK()(!GC_need_to_lock||GC_lock_holder==NUMERIC_THREAD_ID(pthread_self()))
#if!defined(NUMERIC_THREAD_ID_UNIQUE)||defined(THREAD_SANITIZER)
#define I_DONT_HOLD_LOCK()TRUE
#else
#define I_DONT_HOLD_LOCK()(!GC_need_to_lock||GC_lock_holder!=NUMERIC_THREAD_ID(pthread_self()))
#endif
#endif
#ifndef GC_WIN32_THREADS
GC_EXTERN volatile GC_bool GC_collecting;
#ifdef AO_HAVE_char_store
#define ENTER_GC()AO_char_store((unsigned char*)&GC_collecting,TRUE)
#define EXIT_GC()AO_char_store((unsigned char*)&GC_collecting,FALSE)
#else
#define ENTER_GC()(void)(GC_collecting=TRUE)
#define EXIT_GC()(void)(GC_collecting=FALSE)
#endif
#endif
#endif
#if defined(GC_ALWAYS_MULTITHREADED)&&(defined(USE_PTHREAD_LOCKS)||defined(USE_SPIN_LOCK))
#define GC_need_to_lock TRUE
#define set_need_to_lock()(void)0
#else
#if defined(GC_ALWAYS_MULTITHREADED)&&!defined(CPPCHECK)
#error Runtime initialization of GC lock is needed!
#endif
#undef GC_ALWAYS_MULTITHREADED
GC_EXTERN GC_bool GC_need_to_lock;
#ifdef THREAD_SANITIZER
#define set_need_to_lock()(void)(*(GC_bool volatile*)&GC_need_to_lock?FALSE:(GC_need_to_lock=TRUE))
#else
#define set_need_to_lock()(void)(GC_need_to_lock=TRUE)
#endif
#endif
EXTERN_C_END
#else
#define LOCK()(void)0
#define UNLOCK()(void)0
#ifdef GC_ASSERTIONS
#define I_HOLD_LOCK()TRUE
#define I_DONT_HOLD_LOCK()TRUE
#endif
#endif
#if defined(UNCOND_LOCK)&&!defined(LOCK)
#if (defined(LINT2)&&defined(USE_PTHREAD_LOCKS))||defined(GC_ALWAYS_MULTITHREADED)
#define LOCK()UNCOND_LOCK()
#define UNLOCK()UNCOND_UNLOCK()
#else
#define LOCK()do { if (GC_need_to_lock)UNCOND_LOCK();} while (0)
#define UNLOCK()do { if (GC_need_to_lock)UNCOND_UNLOCK();} while (0)
#endif
#endif
#ifndef ENTER_GC
#define ENTER_GC()
#define EXIT_GC()
#endif
#ifndef DCL_LOCK_STATE
#define DCL_LOCK_STATE
#endif
#endif
#define GC_WORD_MAX (~(word)0)
#ifdef STACK_GROWS_DOWN
#define COOLER_THAN >
#define HOTTER_THAN <
#define MAKE_COOLER(x,y)if ((word)((x)+(y))> (word)(x)){(x)+=(y);} else (x)=(ptr_t)GC_WORD_MAX
#define MAKE_HOTTER(x,y)(x)-=(y)
#else
#define COOLER_THAN <
#define HOTTER_THAN >
#define MAKE_COOLER(x,y)if ((word)((x)- (y))< (word)(x)){(x)-=(y);} else (x)=0
#define MAKE_HOTTER(x,y)(x)+=(y)
#endif
#if defined(AMIGA)&&defined(__SASC)
#define GC_FAR __far
#else
#define GC_FAR
#endif
EXTERN_C_BEGIN
#ifndef GC_NO_FINALIZATION
#define GC_INVOKE_FINALIZERS()GC_notify_or_invoke_finalizers()
GC_INNER void GC_notify_or_invoke_finalizers(void);
GC_INNER void GC_finalize(void);
#ifndef GC_TOGGLE_REFS_NOT_NEEDED
GC_INNER void GC_process_togglerefs(void);
#endif
#ifndef SMALL_CONFIG
GC_INNER void GC_print_finalization_stats(void);
#endif
#else
#define GC_INVOKE_FINALIZERS()(void)0
#endif
#if!defined(DONT_ADD_BYTE_AT_END)
#ifdef LINT2
#define EXTRA_BYTES ((size_t)(GC_all_interior_pointers?1:0))
#else
#define EXTRA_BYTES (size_t)GC_all_interior_pointers
#endif
#define MAX_EXTRA_BYTES 1
#else
#define EXTRA_BYTES 0
#define MAX_EXTRA_BYTES 0
#endif
#ifndef LARGE_CONFIG
#define MINHINCR 16
#define MAXHINCR 2048
#else
#define MINHINCR 64
#define MAXHINCR 4096
#endif
#define BL_LIMIT GC_black_list_spacing
#ifdef NEED_CALLINFO
struct callinfo {
word ci_pc;
#if NARGS > 0
word ci_arg[NARGS];
#endif
#if (NFRAMES*(NARGS+1))% 2==1
word ci_dummy;
#endif
};
#endif
#ifdef SAVE_CALL_CHAIN
GC_INNER void GC_save_callers(struct callinfo info[NFRAMES]);
GC_INNER void GC_print_callers(struct callinfo info[NFRAMES]);
#endif
EXTERN_C_END
#ifndef NO_CLOCK
#ifdef BSD_TIME
#undef CLOCK_TYPE
#undef GET_TIME
#undef MS_TIME_DIFF
#define CLOCK_TYPE struct timeval
#define CLOCK_TYPE_INITIALIZER { 0,0 }
#define GET_TIME(x)do { struct rusage rusage;getrusage(RUSAGE_SELF,&rusage);x=rusage.ru_utime;} while (0)
#define MS_TIME_DIFF(a,b)((unsigned long)((long)(a.tv_sec-b.tv_sec)*1000+(long)(a.tv_usec - b.tv_usec)/1000 - (a.tv_usec < b.tv_usec&&(long)(a.tv_usec - b.tv_usec)% 1000!=0?1:0)))
#define NS_FRAC_TIME_DIFF(a,b)((unsigned long)((a.tv_usec < b.tv_usec&&(long)(a.tv_usec - b.tv_usec)% 1000!=0?1000L:0)+(long)(a.tv_usec - b.tv_usec)% 1000)*1000)
#elif defined(MSWIN32)||defined(MSWINCE)||defined(WINXP_USE_PERF_COUNTER)
#if defined(MSWINRT_FLAVOR)||defined(WINXP_USE_PERF_COUNTER)
#define CLOCK_TYPE ULONGLONG
#define GET_TIME(x)do { LARGE_INTEGER freq,tc;if (!QueryPerformanceFrequency(&freq)||!QueryPerformanceCounter(&tc))ABORT("QueryPerformanceCounter requires WinXP+");x=(CLOCK_TYPE)((double)tc.QuadPart/freq.QuadPart*1e9);} while (0)
#define MS_TIME_DIFF(a,b)((unsigned long)(((a)- (b))/1000000UL))
#define NS_FRAC_TIME_DIFF(a,b)((unsigned long)(((a)- (b))% 1000000UL))
#else
#define CLOCK_TYPE DWORD
#define GET_TIME(x)(void)(x=GetTickCount())
#define MS_TIME_DIFF(a,b)((unsigned long)((a)- (b)))
#define NS_FRAC_TIME_DIFF(a,b)0UL
#endif
#elif defined(NN_PLATFORM_CTR)
#define CLOCK_TYPE long long
EXTERN_C_BEGIN
CLOCK_TYPE n3ds_get_system_tick(void);
CLOCK_TYPE n3ds_convert_tick_to_ms(CLOCK_TYPE tick);
EXTERN_C_END
#define GET_TIME(x)(void)(x=n3ds_get_system_tick())
#define MS_TIME_DIFF(a,b)((unsigned long)n3ds_convert_tick_to_ms((a)-(b)))
#define NS_FRAC_TIME_DIFF(a,b)0UL
#elif defined(NINTENDO_SWITCH)||(((defined(LINUX)&&defined(__USE_POSIX199309))||defined(CYGWIN32))&&defined(_POSIX_TIMERS))
#include <time.h>
#define CLOCK_TYPE struct timespec
#define CLOCK_TYPE_INITIALIZER { 0,0 }
#if defined(_POSIX_MONOTONIC_CLOCK)&&!defined(NINTENDO_SWITCH)
#define GET_TIME(x)do { if (clock_gettime(CLOCK_MONOTONIC,&x)==-1)ABORT("clock_gettime failed");} while (0)
#else
#define GET_TIME(x)do { if (clock_gettime(CLOCK_REALTIME,&x)==-1)ABORT("clock_gettime failed");} while (0)
#endif
#define MS_TIME_DIFF(a,b)((unsigned long)((a).tv_nsec+(1000000L*1000 - (b).tv_nsec))/1000000UL+((unsigned long)((a).tv_sec - (b).tv_sec)*1000UL)- 1000UL)
#define NS_FRAC_TIME_DIFF(a,b)((unsigned long)((a).tv_nsec+(1000000L*1000 - (b).tv_nsec))% 1000000UL)
#else
#include <time.h>
#if defined(FREEBSD)&&!defined(CLOCKS_PER_SEC)
#include <machine/limits.h>
#define CLOCKS_PER_SEC CLK_TCK
#endif
#if!defined(CLOCKS_PER_SEC)
#define CLOCKS_PER_SEC 1000000
#endif
#define CLOCK_TYPE clock_t
#define GET_TIME(x)(void)(x=clock())
#define MS_TIME_DIFF(a,b)(CLOCKS_PER_SEC % 1000==0?(unsigned long)((a)- (b))/(unsigned long)(CLOCKS_PER_SEC/1000):((unsigned long)((a)- (b))*1000)/(unsigned long)CLOCKS_PER_SEC)
#define NS_FRAC_TIME_DIFF(a,b)(CLOCKS_PER_SEC<=1000?0UL:(unsigned long)(CLOCKS_PER_SEC<=(clock_t)1000000UL?(((a)- (b))*((clock_t)1000000UL/CLOCKS_PER_SEC)% 1000)*1000:(CLOCKS_PER_SEC<=(clock_t)1000000UL*1000?((a)- (b))*((clock_t)1000000UL*1000/CLOCKS_PER_SEC):(((a)- (b))*(clock_t)1000000UL*1000)/CLOCKS_PER_SEC)% (clock_t)1000000UL))
#endif
#ifndef CLOCK_TYPE_INITIALIZER
#define CLOCK_TYPE_INITIALIZER 0
#endif
#endif
#if defined(SPARC)&&defined(SUNOS4)||(defined(M68K)&&defined(NEXT))||defined(VAX)
#define BCOPY_EXISTS
#elif defined(AMIGA)||defined(DARWIN)
#include <string.h>
#define BCOPY_EXISTS
#elif defined(MACOS)&&defined(POWERPC)
#include <MacMemory.h>
#define bcopy(x,y,n)BlockMoveData(x,y,n)
#define bzero(x,n)BlockZero(x,n)
#define BCOPY_EXISTS
#endif
#if!defined(BCOPY_EXISTS)||defined(CPPCHECK)
#include <string.h>
#define BCOPY(x,y,n)memcpy(y,x,(size_t)(n))
#define BZERO(x,n)memset(x,0,(size_t)(n))
#else
#define BCOPY(x,y,n)bcopy((void*)(x),(void*)(y),(size_t)(n))
#define BZERO(x,n)bzero((void*)(x),(size_t)(n))
#endif
#ifdef PCR
#include "th/PCR_ThCtl.h"
#endif
EXTERN_C_BEGIN
#ifdef PCR
#define STOP_WORLD()PCR_ThCtl_SetExclusiveMode(PCR_ThCtl_ExclusiveMode_stopNormal,PCR_allSigsBlocked,PCR_waitForever)
#define START_WORLD()PCR_ThCtl_SetExclusiveMode(PCR_ThCtl_ExclusiveMode_null,PCR_allSigsBlocked,PCR_waitForever)
#else
#if defined(NN_PLATFORM_CTR)||defined(NINTENDO_SWITCH)||defined(GC_WIN32_THREADS)||defined(GC_PTHREADS)
GC_INNER void GC_stop_world(void);
GC_INNER void GC_start_world(void);
#define STOP_WORLD()GC_stop_world()
#define START_WORLD()GC_start_world()
#else
#define STOP_WORLD()GC_ASSERT(GC_blocked_sp==NULL)
#define START_WORLD()
#endif
#endif
#ifdef THREADS
GC_EXTERN GC_on_thread_event_proc GC_on_thread_event;
#endif
#if defined(SMALL_CONFIG)||defined(PCR)
#define GC_on_abort(msg)(void)0
#else
GC_API_PRIV GC_abort_func GC_on_abort;
#endif
#if defined(CPPCHECK)
#define ABORT(msg){ GC_on_abort(msg);abort();}
#elif defined(PCR)
#define ABORT(s)PCR_Base_Panic(s)
#else
#if defined(MSWIN_XBOX1)&&!defined(DebugBreak)
#define DebugBreak()__debugbreak()
#elif defined(MSWINCE)&&!defined(DebugBreak)&&(!defined(UNDER_CE)||(defined(__MINGW32CE__)&&!defined(ARM32)))
#define DebugBreak()_exit(-1)
#endif
#if defined(MSWIN32)&&(defined(NO_DEBUGGING)||defined(LINT2))
#define ABORT(msg)(GC_on_abort(msg),_exit(-1))
#elif defined(MSWINCE)&&defined(NO_DEBUGGING)
#define ABORT(msg)(GC_on_abort(msg),ExitProcess(-1))
#elif defined(MSWIN32)||defined(MSWINCE)
#if defined(_CrtDbgBreak)&&defined(_DEBUG)&&defined(_MSC_VER)
#define ABORT(msg){ GC_on_abort(msg);_CrtDbgBreak();}
#else
#define ABORT(msg){ GC_on_abort(msg);DebugBreak();}
#endif
#else
#define ABORT(msg)(GC_on_abort(msg),abort())
#endif
#endif
#define ABORT_ARG1(C_msg,C_fmt,arg1)MACRO_BLKSTMT_BEGIN GC_INFOLOG_PRINTF(C_msg C_fmt "\n",arg1);ABORT(C_msg);MACRO_BLKSTMT_END
#define ABORT_ARG2(C_msg,C_fmt,arg1,arg2)MACRO_BLKSTMT_BEGIN GC_INFOLOG_PRINTF(C_msg C_fmt "\n",arg1,arg2);ABORT(C_msg);MACRO_BLKSTMT_END
#define ABORT_ARG3(C_msg,C_fmt,arg1,arg2,arg3)MACRO_BLKSTMT_BEGIN GC_INFOLOG_PRINTF(C_msg C_fmt "\n",arg1,arg2,arg3);ABORT(C_msg);MACRO_BLKSTMT_END
#define ABORT_RET(msg)if ((signed_word)GC_current_warn_proc==-1){} else ABORT(msg)
#ifdef PCR
#define EXIT()PCR_Base_Exit(1,PCR_waitForever)
#else
#define EXIT()(GC_on_abort(NULL),exit(1))
#endif
#define WARN(msg,arg)(*GC_current_warn_proc)(( char*)("GC Warning:" msg),(word)(arg))
GC_EXTERN GC_warn_proc GC_current_warn_proc;
#ifndef WARN_PRIdPTR
#define WARN_PRIdPTR "ld"
#endif
#define TRUSTED_STRING(s)(char*)COVERT_DATAFLOW(s)
#ifdef GC_READ_ENV_FILE
GC_INNER char*GC_envfile_getenv(const char*name);
#define GETENV(name)GC_envfile_getenv(name)
#elif defined(NO_GETENV)&&!defined(CPPCHECK)
#define GETENV(name)NULL
#elif defined(EMPTY_GETENV_RESULTS)
GC_INLINE char*fixed_getenv(const char*name)
{
char*value=getenv(name);
return value!=NULL&&*value!='\0'?value:NULL;
}
#define GETENV(name)fixed_getenv(name)
#else
#define GETENV(name)getenv(name)
#endif
EXTERN_C_END
#if defined(DARWIN)
#include <mach/thread_status.h>
#ifndef MAC_OS_X_VERSION_MAX_ALLOWED
#include <AvailabilityMacros.h>
#endif
#if defined(POWERPC)
#if CPP_WORDSZ==32
#define GC_THREAD_STATE_T ppc_thread_state_t
#else
#define GC_THREAD_STATE_T ppc_thread_state64_t
#define GC_MACH_THREAD_STATE PPC_THREAD_STATE64
#define GC_MACH_THREAD_STATE_COUNT PPC_THREAD_STATE64_COUNT
#endif
#elif defined(I386)||defined(X86_64)
#if CPP_WORDSZ==32
#if defined(i386_THREAD_STATE_COUNT)&&!defined(x86_THREAD_STATE32_COUNT)
#define GC_THREAD_STATE_T i386_thread_state_t
#define GC_MACH_THREAD_STATE i386_THREAD_STATE
#define GC_MACH_THREAD_STATE_COUNT i386_THREAD_STATE_COUNT
#else
#define GC_THREAD_STATE_T x86_thread_state32_t
#define GC_MACH_THREAD_STATE x86_THREAD_STATE32
#define GC_MACH_THREAD_STATE_COUNT x86_THREAD_STATE32_COUNT
#endif
#else
#define GC_THREAD_STATE_T x86_thread_state64_t
#define GC_MACH_THREAD_STATE x86_THREAD_STATE64
#define GC_MACH_THREAD_STATE_COUNT x86_THREAD_STATE64_COUNT
#endif
#elif defined(ARM32)&&defined(ARM_UNIFIED_THREAD_STATE)&&!defined(CPPCHECK)
#define GC_THREAD_STATE_T arm_unified_thread_state_t
#define GC_MACH_THREAD_STATE ARM_UNIFIED_THREAD_STATE
#define GC_MACH_THREAD_STATE_COUNT ARM_UNIFIED_THREAD_STATE_COUNT
#elif defined(ARM32)
#define GC_THREAD_STATE_T arm_thread_state_t
#ifdef ARM_MACHINE_THREAD_STATE_COUNT
#define GC_MACH_THREAD_STATE ARM_MACHINE_THREAD_STATE
#define GC_MACH_THREAD_STATE_COUNT ARM_MACHINE_THREAD_STATE_COUNT
#endif
#elif defined(AARCH64)
#define GC_THREAD_STATE_T arm_thread_state64_t
#define GC_MACH_THREAD_STATE ARM_THREAD_STATE64
#define GC_MACH_THREAD_STATE_COUNT ARM_THREAD_STATE64_COUNT
#elif!defined(CPPCHECK)
#error define GC_THREAD_STATE_T
#endif
#ifndef GC_MACH_THREAD_STATE
#define GC_MACH_THREAD_STATE MACHINE_THREAD_STATE
#define GC_MACH_THREAD_STATE_COUNT MACHINE_THREAD_STATE_COUNT
#endif
#if CPP_WORDSZ==32
#define GC_MACH_HEADER mach_header
#define GC_MACH_SECTION section
#define GC_GETSECTBYNAME getsectbynamefromheader
#else
#define GC_MACH_HEADER mach_header_64
#define GC_MACH_SECTION section_64
#define GC_GETSECTBYNAME getsectbynamefromheader_64
#endif
#if __DARWIN_UNIX03
#define THREAD_FLD_NAME(x)__ ## x
#else
#define THREAD_FLD_NAME(x)x
#endif
#if defined(ARM32)&&defined(ARM_UNIFIED_THREAD_STATE)
#define THREAD_FLD(x)ts_32.THREAD_FLD_NAME(x)
#else
#define THREAD_FLD(x)THREAD_FLD_NAME(x)
#endif
#endif
#include <setjmp.h>
#if __STDC_VERSION__>=201112L
#include <assert.h>
#endif
EXTERN_C_BEGIN
#if CPP_WORDSZ==32
#define WORDS_TO_BYTES(x)((x)<<2)
#define BYTES_TO_WORDS(x)((x)>>2)
#define LOGWL ((word)5)
#define modWORDSZ(n)((n)&0x1f)
#if ALIGNMENT!=4
#define UNALIGNED_PTRS
#endif
#endif
#if CPP_WORDSZ==64
#define WORDS_TO_BYTES(x)((x)<<3)
#define BYTES_TO_WORDS(x)((x)>>3)
#define LOGWL ((word)6)
#define modWORDSZ(n)((n)&0x3f)
#if ALIGNMENT!=8
#define UNALIGNED_PTRS
#endif
#endif
#define GRANULE_BYTES GC_GRANULE_BYTES
#define TINY_FREELISTS GC_TINY_FREELISTS
#define WORDSZ ((word)CPP_WORDSZ)
#define SIGNB ((word)1<<(WORDSZ-1))
#define BYTES_PER_WORD ((word)(sizeof (word)))
#define divWORDSZ(n)((n)>>LOGWL)
#if GRANULE_BYTES==8
#define BYTES_TO_GRANULES(n)((n)>>3)
#define GRANULES_TO_BYTES(n)((n)<<3)
#if CPP_WORDSZ==64
#define GRANULES_TO_WORDS(n)(n)
#elif CPP_WORDSZ==32
#define GRANULES_TO_WORDS(n)((n)<<1)
#else
#define GRANULES_TO_WORDS(n)BYTES_TO_WORDS(GRANULES_TO_BYTES(n))
#endif
#elif GRANULE_BYTES==16
#define BYTES_TO_GRANULES(n)((n)>>4)
#define GRANULES_TO_BYTES(n)((n)<<4)
#if CPP_WORDSZ==64
#define GRANULES_TO_WORDS(n)((n)<<1)
#elif CPP_WORDSZ==32
#define GRANULES_TO_WORDS(n)((n)<<2)
#else
#define GRANULES_TO_WORDS(n)BYTES_TO_WORDS(GRANULES_TO_BYTES(n))
#endif
#else
#error Bad GRANULE_BYTES value
#endif
#ifndef HBLKSIZE
#if defined(LARGE_CONFIG)||!defined(SMALL_CONFIG)
#ifdef ALPHA
#define CPP_LOG_HBLKSIZE 13
#elif defined(SN_TARGET_ORBIS)||defined(SN_TARGET_PSP2)
#define CPP_LOG_HBLKSIZE 16
#else
#define CPP_LOG_HBLKSIZE 12
#endif
#else
#define CPP_LOG_HBLKSIZE 10
#endif
#else
#if HBLKSIZE==512
#define CPP_LOG_HBLKSIZE 9
#elif HBLKSIZE==1024
#define CPP_LOG_HBLKSIZE 10
#elif HBLKSIZE==2048
#define CPP_LOG_HBLKSIZE 11
#elif HBLKSIZE==4096
#define CPP_LOG_HBLKSIZE 12
#elif HBLKSIZE==8192
#define CPP_LOG_HBLKSIZE 13
#elif HBLKSIZE==16384
#define CPP_LOG_HBLKSIZE 14
#elif!defined(CPPCHECK)
#error Bad HBLKSIZE value
#endif
#undef HBLKSIZE
#endif
#define CPP_HBLKSIZE (1<<CPP_LOG_HBLKSIZE)
#define LOG_HBLKSIZE ((size_t)CPP_LOG_HBLKSIZE)
#define HBLKSIZE ((size_t)CPP_HBLKSIZE)
#define GC_SQRT_SIZE_MAX ((((size_t)1)<<(WORDSZ/2))- 1)
#define CPP_MAXOBJBYTES (CPP_HBLKSIZE/2)
#define MAXOBJBYTES ((size_t)CPP_MAXOBJBYTES)
#define CPP_MAXOBJWORDS BYTES_TO_WORDS(CPP_MAXOBJBYTES)
#define MAXOBJWORDS ((size_t)CPP_MAXOBJWORDS)
#define CPP_MAXOBJGRANULES BYTES_TO_GRANULES(CPP_MAXOBJBYTES)
#define MAXOBJGRANULES ((size_t)CPP_MAXOBJGRANULES)
#define divHBLKSZ(n)((n)>>LOG_HBLKSIZE)
#define HBLK_PTR_DIFF(p,q)divHBLKSZ((ptr_t)p - (ptr_t)q)
#define modHBLKSZ(n)((n)&(HBLKSIZE-1))
#define HBLKPTR(objptr)((struct hblk*)(((word)(objptr))&~(word)(HBLKSIZE-1)))
#define HBLKDISPL(objptr)(((size_t)(objptr))&(HBLKSIZE-1))
#define ROUNDUP_GRANULE_SIZE(lb)(SIZET_SAT_ADD(lb,GRANULE_BYTES - 1)&~(GRANULE_BYTES - 1))
#define ROUNDED_UP_GRANULES(lb)BYTES_TO_GRANULES(SIZET_SAT_ADD(lb,GRANULE_BYTES - 1+EXTRA_BYTES))
#if MAX_EXTRA_BYTES==0
#define SMALL_OBJ(bytes)EXPECT((bytes)<=(MAXOBJBYTES),TRUE)
#else
#define SMALL_OBJ(bytes)(EXPECT((bytes)<=(MAXOBJBYTES - MAX_EXTRA_BYTES),TRUE)||(bytes)<=MAXOBJBYTES - EXTRA_BYTES)
#endif
#define ADD_SLOP(lb)SIZET_SAT_ADD(lb,EXTRA_BYTES)
#ifdef LARGE_CONFIG
#if CPP_WORDSZ==32
#define LOG_PHT_ENTRIES 20
#else
#define LOG_PHT_ENTRIES 21
#endif
#elif!defined(SMALL_CONFIG)
#define LOG_PHT_ENTRIES 18
#else
#define LOG_PHT_ENTRIES 15
#endif
#define PHT_ENTRIES ((word)1<<LOG_PHT_ENTRIES)
#define PHT_SIZE (PHT_ENTRIES>>LOGWL)
typedef word page_hash_table[PHT_SIZE];
#define PHT_HASH(addr)((((word)(addr))>>LOG_HBLKSIZE)&(PHT_ENTRIES - 1))
#define get_pht_entry_from_index(bl,index)(((bl)[divWORDSZ(index)]>>modWORDSZ(index))&1)
#define set_pht_entry_from_index(bl,index)(void)((bl)[divWORDSZ(index)]|=(word)1<<modWORDSZ(index))
#if defined(THREADS)&&defined(AO_HAVE_or)
#define set_pht_entry_from_index_concurrent(bl,index)AO_or((volatile AO_t*)&(bl)[divWORDSZ(index)],(AO_t)((word)1<<modWORDSZ(index)))
#else
#define set_pht_entry_from_index_concurrent(bl,index)set_pht_entry_from_index(bl,index)
#endif
#define HBLKMASK (HBLKSIZE-1)
#define MARK_BITS_PER_HBLK (HBLKSIZE/GRANULE_BYTES)
union word_ptr_ao_u {
word w;
signed_word sw;
void*vp;
#ifdef PARALLEL_MARK
volatile AO_t ao;
#endif
};
struct hblkhdr {
struct hblk*hb_next;
struct hblk*hb_prev;
struct hblk*hb_block;
unsigned char hb_obj_kind;
unsigned char hb_flags;
#define IGNORE_OFF_PAGE 1
#define WAS_UNMAPPED 2
#define FREE_BLK 4
#ifdef ENABLE_DISCLAIM
#define HAS_DISCLAIM 8
#define MARK_UNCONDITIONALLY 0x10
#endif
#ifdef MARK_BIT_PER_GRANULE
#define LARGE_BLOCK 0x20
#endif
unsigned short hb_last_reclaimed;
#ifdef MARK_BIT_PER_OBJ
unsigned32 hb_inv_sz;
#define LARGE_INV_SZ (1<<16)
#endif
word hb_sz;
word hb_descr;
#ifdef MARK_BIT_PER_GRANULE
unsigned short*hb_map;
#endif
#ifdef PARALLEL_MARK
volatile AO_t hb_n_marks;
#else
size_t hb_n_marks;
#endif
#ifdef USE_MARK_BYTES
#define MARK_BITS_SZ (MARK_BITS_PER_HBLK+1)
union {
char _hb_marks[MARK_BITS_SZ];
word dummy;
} _mark_byte_union;
#define hb_marks _mark_byte_union._hb_marks
#else
#define MARK_BITS_SZ (MARK_BITS_PER_HBLK/CPP_WORDSZ+1)
word hb_marks[MARK_BITS_SZ];
#endif
};
#define ANY_INDEX 23
#define HBLK_WORDS (HBLKSIZE/sizeof(word))
#define HBLK_GRANULES (HBLKSIZE/GRANULE_BYTES)
#define HBLK_OBJS(sz_in_bytes)(HBLKSIZE/(sz_in_bytes))
struct hblk {
char hb_body[HBLKSIZE];
};
#define HBLK_IS_FREE(hdr)(((hdr)->hb_flags&FREE_BLK)!=0)
#define OBJ_SZ_TO_BLOCKS(lb)divHBLKSZ((lb)+HBLKSIZE-1)
#define OBJ_SZ_TO_BLOCKS_CHECKED(lb)divHBLKSZ(SIZET_SAT_ADD(lb,HBLKSIZE - 1))
#define obj_link(p)(*(void**)(p))
#define LOG_MAX_MARK_PROCS 6
#define MAX_MARK_PROCS (1<<LOG_MAX_MARK_PROCS)
#ifdef LARGE_CONFIG
#define MAX_ROOT_SETS 8192
#elif!defined(SMALL_CONFIG)
#define MAX_ROOT_SETS 2048
#else
#define MAX_ROOT_SETS 512
#endif
#define MAX_EXCLUSIONS (MAX_ROOT_SETS/4)
struct exclusion {
ptr_t e_start;
ptr_t e_end;
};
struct roots {
ptr_t r_start;
ptr_t r_end;
#if!defined(MSWIN32)&&!defined(MSWINCE)&&!defined(CYGWIN32)
struct roots*r_next;
#endif
GC_bool r_tmp;
};
#if!defined(MSWIN32)&&!defined(MSWINCE)&&!defined(CYGWIN32)
#define LOG_RT_SIZE 6
#define RT_SIZE (1<<LOG_RT_SIZE)
#endif
#ifndef MAX_HEAP_SECTS
#ifdef LARGE_CONFIG
#if CPP_WORDSZ > 32
#define MAX_HEAP_SECTS 81920
#else
#define MAX_HEAP_SECTS 7680
#endif
#elif defined(SMALL_CONFIG)&&!defined(USE_PROC_FOR_LIBRARIES)
#if defined(PARALLEL_MARK)&&(defined(MSWIN32)||defined(CYGWIN32))
#define MAX_HEAP_SECTS 384
#else
#define MAX_HEAP_SECTS 128
#endif
#elif CPP_WORDSZ > 32
#define MAX_HEAP_SECTS 1024
#else
#define MAX_HEAP_SECTS 512
#endif
#endif
typedef struct GC_ms_entry {
ptr_t mse_start;
union word_ptr_ao_u mse_descr;
} mse;
typedef int mark_state_t;
struct disappearing_link;
struct finalizable_object;
struct dl_hashtbl_s {
struct disappearing_link**head;
word entries;
unsigned log_size;
};
struct fnlz_roots_s {
struct finalizable_object**fo_head;
struct finalizable_object*finalize_now;
};
union toggle_ref_u {
void*strong_ref;
GC_hidden_pointer weak_ref;
};
typedef struct {
word ed_bitmap;
GC_bool ed_continued;
} typed_ext_descr_t;
struct _GC_arrays {
word _heapsize;
word _requested_heapsize;
ptr_t _last_heap_addr;
ptr_t _prev_heap_addr;
word _large_free_bytes;
word _large_allocd_bytes;
word _max_large_allocd_bytes;
word _bytes_allocd_before_gc;
#ifndef SEPARATE_GLOBALS
#define GC_bytes_allocd GC_arrays._bytes_allocd
word _bytes_allocd;
#endif
word _bytes_dropped;
word _bytes_finalized;
word _bytes_freed;
word _finalizer_bytes_freed;
bottom_index*_all_bottom_indices;
bottom_index*_all_bottom_indices_end;
ptr_t _scratch_free_ptr;
hdr*_hdr_free_list;
ptr_t _scratch_end_ptr;
ptr_t _scratch_last_end_ptr;
mse*_mark_stack;
mse*_mark_stack_limit;
#ifdef PARALLEL_MARK
mse*volatile _mark_stack_top;
#else
mse*_mark_stack_top;
#endif
word _composite_in_use;
word _atomic_in_use;
#ifdef USE_MUNMAP
#define GC_unmapped_bytes GC_arrays._unmapped_bytes
word _unmapped_bytes;
#ifdef COUNT_UNMAPPED_REGIONS
#define GC_num_unmapped_regions GC_arrays._num_unmapped_regions
signed_word _num_unmapped_regions;
#endif
#else
#define GC_unmapped_bytes 0
#endif
bottom_index*_all_nils;
#define GC_scan_ptr GC_arrays._scan_ptr
struct hblk*_scan_ptr;
#ifdef PARALLEL_MARK
#define GC_main_local_mark_stack GC_arrays._main_local_mark_stack
mse*_main_local_mark_stack;
#define GC_first_nonempty GC_arrays._first_nonempty
volatile AO_t _first_nonempty;
#endif
#define GC_mark_stack_size GC_arrays._mark_stack_size
size_t _mark_stack_size;
#define GC_mark_state GC_arrays._mark_state
mark_state_t _mark_state;
#define GC_mark_stack_too_small GC_arrays._mark_stack_too_small
GC_bool _mark_stack_too_small;
#define GC_objects_are_marked GC_arrays._objects_are_marked
GC_bool _objects_are_marked;
#ifdef ENABLE_TRACE
#define GC_trace_addr GC_arrays._trace_addr
ptr_t _trace_addr;
#endif
#define GC_n_heap_sects GC_arrays._n_heap_sects
word _n_heap_sects;
#if defined(MSWIN32)||defined(MSWINCE)||defined(CYGWIN32)
#define GC_n_heap_bases GC_arrays._n_heap_bases
word _n_heap_bases;
#endif
#ifdef USE_PROC_FOR_LIBRARIES
#define GC_n_memory GC_arrays._n_memory
word _n_memory;
#endif
#ifdef GC_GCJ_SUPPORT
#define GC_gcjobjfreelist GC_arrays._gcjobjfreelist
ptr_t*_gcjobjfreelist;
#endif
#define GC_fo_entries GC_arrays._fo_entries
word _fo_entries;
#ifndef GC_NO_FINALIZATION
#define GC_dl_hashtbl GC_arrays._dl_hashtbl
#define GC_fnlz_roots GC_arrays._fnlz_roots
#define GC_log_fo_table_size GC_arrays._log_fo_table_size
#ifndef GC_LONG_REFS_NOT_NEEDED
#define GC_ll_hashtbl GC_arrays._ll_hashtbl
struct dl_hashtbl_s _ll_hashtbl;
#endif
struct dl_hashtbl_s _dl_hashtbl;
struct fnlz_roots_s _fnlz_roots;
unsigned _log_fo_table_size;
#ifndef GC_TOGGLE_REFS_NOT_NEEDED
#define GC_toggleref_arr GC_arrays._toggleref_arr
#define GC_toggleref_array_size GC_arrays._toggleref_array_size
#define GC_toggleref_array_capacity GC_arrays._toggleref_array_capacity
union toggle_ref_u*_toggleref_arr;
size_t _toggleref_array_size;
size_t _toggleref_array_capacity;
#endif
#endif
#ifdef TRACE_BUF
#define GC_trace_buf_ptr GC_arrays._trace_buf_ptr
int _trace_buf_ptr;
#endif
#ifdef ENABLE_DISCLAIM
#define GC_finalized_kind GC_arrays._finalized_kind
int _finalized_kind;
#endif
#define n_root_sets GC_arrays._n_root_sets
#define GC_excl_table_entries GC_arrays._excl_table_entries
int _n_root_sets;
size_t _excl_table_entries;
#ifdef THREADS
#define GC_roots_were_cleared GC_arrays._roots_were_cleared
GC_bool _roots_were_cleared;
#endif
#define GC_explicit_typing_initialized GC_arrays._explicit_typing_initialized
#define GC_ed_size GC_arrays._ed_size
#define GC_avail_descr GC_arrays._avail_descr
#define GC_ext_descriptors GC_arrays._ext_descriptors
#ifdef AO_HAVE_load_acquire
volatile AO_t _explicit_typing_initialized;
#else
GC_bool _explicit_typing_initialized;
#endif
size_t _ed_size;
size_t _avail_descr;
typed_ext_descr_t*_ext_descriptors;
GC_mark_proc _mark_procs[MAX_MARK_PROCS];
char _modws_valid_offsets[sizeof(word)];
#if!defined(MSWIN32)&&!defined(MSWINCE)&&!defined(CYGWIN32)
#define GC_root_index GC_arrays._root_index
struct roots*_root_index[RT_SIZE];
#endif
#ifdef SAVE_CALL_CHAIN
#define GC_last_stack GC_arrays._last_stack
struct callinfo _last_stack[NFRAMES];
#endif
#ifndef SEPARATE_GLOBALS
#define GC_objfreelist GC_arrays._objfreelist
void*_objfreelist[MAXOBJGRANULES+1];
#define GC_aobjfreelist GC_arrays._aobjfreelist
void*_aobjfreelist[MAXOBJGRANULES+1];
#endif
void*_uobjfreelist[MAXOBJGRANULES+1];
#ifdef GC_ATOMIC_UNCOLLECTABLE
#define GC_auobjfreelist GC_arrays._auobjfreelist
void*_auobjfreelist[MAXOBJGRANULES+1];
#endif
size_t _size_map[MAXOBJBYTES+1];
#ifdef MARK_BIT_PER_GRANULE
#define GC_obj_map GC_arrays._obj_map
unsigned short*_obj_map[MAXOBJGRANULES+1];
#define MAP_LEN BYTES_TO_GRANULES(HBLKSIZE)
#endif
#define VALID_OFFSET_SZ HBLKSIZE
char _valid_offsets[VALID_OFFSET_SZ];
#ifndef GC_DISABLE_INCREMENTAL
#define GC_grungy_pages GC_arrays._grungy_pages
page_hash_table _grungy_pages;
#define GC_dirty_pages GC_arrays._dirty_pages
volatile page_hash_table _dirty_pages;
#endif
#if (defined(CHECKSUMS)&&defined(GWW_VDB))||defined(PROC_VDB)
#define GC_written_pages GC_arrays._written_pages
page_hash_table _written_pages;
#endif
#define GC_heap_sects GC_arrays._heap_sects
struct HeapSect {
ptr_t hs_start;
size_t hs_bytes;
} _heap_sects[MAX_HEAP_SECTS];
#if defined(USE_PROC_FOR_LIBRARIES)
#define GC_our_memory GC_arrays._our_memory
struct HeapSect _our_memory[MAX_HEAP_SECTS];
#endif
#if defined(MSWIN32)||defined(MSWINCE)||defined(CYGWIN32)
#define GC_heap_bases GC_arrays._heap_bases
ptr_t _heap_bases[MAX_HEAP_SECTS];
#endif
#ifdef MSWINCE
#define GC_heap_lengths GC_arrays._heap_lengths
word _heap_lengths[MAX_HEAP_SECTS];
#endif
struct roots _static_roots[MAX_ROOT_SETS];
struct exclusion _excl_table[MAX_EXCLUSIONS];
bottom_index*_top_index[TOP_SZ];
};
GC_API_PRIV GC_FAR struct _GC_arrays GC_arrays;
#define GC_all_nils GC_arrays._all_nils
#define GC_atomic_in_use GC_arrays._atomic_in_use
#define GC_bytes_allocd_before_gc GC_arrays._bytes_allocd_before_gc
#define GC_bytes_dropped GC_arrays._bytes_dropped
#define GC_bytes_finalized GC_arrays._bytes_finalized
#define GC_bytes_freed GC_arrays._bytes_freed
#define GC_composite_in_use GC_arrays._composite_in_use
#define GC_excl_table GC_arrays._excl_table
#define GC_finalizer_bytes_freed GC_arrays._finalizer_bytes_freed
#define GC_heapsize GC_arrays._heapsize
#define GC_large_allocd_bytes GC_arrays._large_allocd_bytes
#define GC_large_free_bytes GC_arrays._large_free_bytes
#define GC_last_heap_addr GC_arrays._last_heap_addr
#define GC_mark_stack GC_arrays._mark_stack
#define GC_mark_stack_limit GC_arrays._mark_stack_limit
#define GC_mark_stack_top GC_arrays._mark_stack_top
#define GC_mark_procs GC_arrays._mark_procs
#define GC_max_large_allocd_bytes GC_arrays._max_large_allocd_bytes
#define GC_modws_valid_offsets GC_arrays._modws_valid_offsets
#define GC_prev_heap_addr GC_arrays._prev_heap_addr
#define GC_requested_heapsize GC_arrays._requested_heapsize
#define GC_all_bottom_indices GC_arrays._all_bottom_indices
#define GC_all_bottom_indices_end GC_arrays._all_bottom_indices_end
#define GC_scratch_free_ptr GC_arrays._scratch_free_ptr
#define GC_hdr_free_list GC_arrays._hdr_free_list
#define GC_scratch_end_ptr GC_arrays._scratch_end_ptr
#define GC_scratch_last_end_ptr GC_arrays._scratch_last_end_ptr
#define GC_size_map GC_arrays._size_map
#define GC_static_roots GC_arrays._static_roots
#define GC_top_index GC_arrays._top_index
#define GC_uobjfreelist GC_arrays._uobjfreelist
#define GC_valid_offsets GC_arrays._valid_offsets
#define beginGC_arrays ((ptr_t)(&GC_arrays))
#define endGC_arrays (((ptr_t)(&GC_arrays))+(sizeof GC_arrays))
#define USED_HEAP_SIZE (GC_heapsize - GC_large_free_bytes)
#ifndef MAXOBJKINDS
#define MAXOBJKINDS 16
#endif
GC_EXTERN struct obj_kind {
void**ok_freelist;
struct hblk**ok_reclaim_list;
word ok_descriptor;
GC_bool ok_relocate_descr;
GC_bool ok_init;
#ifdef ENABLE_DISCLAIM
GC_bool ok_mark_unconditionally;
int (GC_CALLBACK*ok_disclaim_proc)(void*);
#define OK_DISCLAIM_INITZ,FALSE,0
#else
#define OK_DISCLAIM_INITZ
#endif
} GC_obj_kinds[MAXOBJKINDS];
#define beginGC_obj_kinds ((ptr_t)(&GC_obj_kinds))
#define endGC_obj_kinds (beginGC_obj_kinds+(sizeof GC_obj_kinds))
#ifdef SEPARATE_GLOBALS
extern word GC_bytes_allocd;
extern ptr_t GC_objfreelist[MAXOBJGRANULES+1];
#define beginGC_objfreelist ((ptr_t)(&GC_objfreelist))
#define endGC_objfreelist (beginGC_objfreelist+sizeof(GC_objfreelist))
extern ptr_t GC_aobjfreelist[MAXOBJGRANULES+1];
#define beginGC_aobjfreelist ((ptr_t)(&GC_aobjfreelist))
#define endGC_aobjfreelist (beginGC_aobjfreelist+sizeof(GC_aobjfreelist))
#endif
#define PTRFREE 0
#define NORMAL 1
#define UNCOLLECTABLE 2
#ifdef GC_ATOMIC_UNCOLLECTABLE
#define AUNCOLLECTABLE 3
#define IS_UNCOLLECTABLE(k)(((k)&~1)==UNCOLLECTABLE)
#define GC_N_KINDS_INITIAL_VALUE 4
#else
#define IS_UNCOLLECTABLE(k)((k)==UNCOLLECTABLE)
#define GC_N_KINDS_INITIAL_VALUE 3
#endif
GC_EXTERN unsigned GC_n_kinds;
GC_EXTERN size_t GC_page_size;
#define ROUNDUP_PAGESIZE(lb)(SIZET_SAT_ADD(lb,GC_page_size - 1)&~(GC_page_size - 1))
#ifdef MMAP_SUPPORTED
#define ROUNDUP_PAGESIZE_IF_MMAP(lb)ROUNDUP_PAGESIZE(lb)
#else
#define ROUNDUP_PAGESIZE_IF_MMAP(lb)(lb)
#endif
#if defined(MSWIN32)||defined(MSWINCE)||defined(CYGWIN32)
GC_EXTERN SYSTEM_INFO GC_sysinfo;
GC_INNER GC_bool GC_is_heap_base(void*p);
#endif
GC_EXTERN word GC_black_list_spacing;
#ifdef GC_GCJ_SUPPORT
extern struct hblk*GC_hblkfreelist[];
extern word GC_free_bytes[];
#endif
GC_EXTERN word GC_root_size;
GC_EXTERN GC_bool GC_debugging_started;
struct blocking_data {
GC_fn_type fn;
void*client_data;
};
struct GC_traced_stack_sect_s {
ptr_t saved_stack_ptr;
#ifdef IA64
ptr_t saved_backing_store_ptr;
ptr_t backing_store_end;
#endif
struct GC_traced_stack_sect_s*prev;
};
#ifdef THREADS
GC_INNER void GC_push_all_stack_sections(ptr_t lo,ptr_t hi,
struct GC_traced_stack_sect_s*traced_stack_sect);
GC_EXTERN word GC_total_stacksize;
#else
GC_EXTERN ptr_t GC_blocked_sp;
GC_EXTERN struct GC_traced_stack_sect_s*GC_traced_stack_sect;
#endif
#ifdef IA64
GC_INNER void GC_push_all_register_sections(ptr_t bs_lo,ptr_t bs_hi,
int eager,struct GC_traced_stack_sect_s*traced_stack_sect);
#endif
#ifdef USE_MARK_BYTES
#define mark_bit_from_hdr(hhdr,n)((hhdr)->hb_marks[n])
#define set_mark_bit_from_hdr(hhdr,n)((hhdr)->hb_marks[n]=1)
#define clear_mark_bit_from_hdr(hhdr,n)((hhdr)->hb_marks[n]=0)
#else
#if defined(PARALLEL_MARK)||(defined(THREAD_SANITIZER)&&defined(THREADS))
#define OR_WORD(addr,bits)AO_or((volatile AO_t*)(addr),(AO_t)(bits))
#else
#define OR_WORD(addr,bits)(void)(*(addr)|=(bits))
#endif
#define mark_bit_from_hdr(hhdr,n)(((hhdr)->hb_marks[divWORDSZ(n)]>>modWORDSZ(n))&(word)1)
#define set_mark_bit_from_hdr(hhdr,n)OR_WORD((hhdr)->hb_marks+divWORDSZ(n),(word)1<<modWORDSZ(n))
#define clear_mark_bit_from_hdr(hhdr,n)((hhdr)->hb_marks[divWORDSZ(n)]&=~((word)1<<modWORDSZ(n)))
#endif
#ifdef MARK_BIT_PER_OBJ
#define MARK_BIT_NO(offset,sz)(((word)(offset))/(sz))
#define MARK_BIT_OFFSET(sz)1
#define IF_PER_OBJ(x)x
#define FINAL_MARK_BIT(sz)((sz)> MAXOBJBYTES?1:HBLK_OBJS(sz))
#else
#define MARK_BIT_NO(offset,sz)BYTES_TO_GRANULES((word)(offset))
#define MARK_BIT_OFFSET(sz)BYTES_TO_GRANULES(sz)
#define IF_PER_OBJ(x)
#define FINAL_MARK_BIT(sz)((sz)> MAXOBJBYTES?MARK_BITS_PER_HBLK:BYTES_TO_GRANULES((sz)*HBLK_OBJS(sz)))
#endif
GC_INNER ptr_t GC_approx_sp(void);
GC_INNER GC_bool GC_should_collect(void);
void GC_apply_to_all_blocks(void (*fn)(struct hblk*h,word client_data),
word client_data);
GC_INNER struct hblk*GC_next_block(struct hblk*h,GC_bool allow_free);
GC_INNER struct hblk*GC_prev_block(struct hblk*h);
GC_INNER void GC_mark_init(void);
GC_INNER void GC_clear_marks(void);
GC_INNER void GC_invalidate_mark_state(void);
GC_INNER GC_bool GC_mark_some(ptr_t cold_gc_frame);
GC_INNER void GC_initiate_gc(void);
GC_INNER GC_bool GC_collection_in_progress(void);
#define GC_PUSH_ALL_SYM(sym)GC_push_all(( void*)&(sym),( void*)(&(sym)+1))
GC_INNER void GC_push_all_stack(ptr_t b,ptr_t t);
#if defined(WRAP_MARK_SOME)&&defined(PARALLEL_MARK)
GC_INNER void GC_push_conditional_eager(void*bottom,void*top,
GC_bool all);
#endif
GC_INNER void GC_push_roots(GC_bool all,ptr_t cold_gc_frame);
GC_API_PRIV GC_push_other_roots_proc GC_push_other_roots;
#ifdef THREADS
void GC_push_thread_structures(void);
#endif
GC_EXTERN void (*GC_push_typed_structures)(void);
GC_INNER void GC_with_callee_saves_pushed(void (*fn)(ptr_t,void*),
volatile ptr_t arg);
#if defined(SPARC)||defined(IA64)
ptr_t GC_save_regs_in_stack(void);
#endif
#if defined(AMIGA)||defined(MACOS)||defined(GC_DARWIN_THREADS)
void GC_push_one(word p);
#endif
#ifdef GC_WIN32_THREADS
GC_INNER void GC_push_many_regs(const word*regs,unsigned count);
#endif
#if defined(PRINT_BLACK_LIST)||defined(KEEP_BACK_PTRS)
GC_INNER void GC_mark_and_push_stack(ptr_t p,ptr_t source);
#else
GC_INNER void GC_mark_and_push_stack(ptr_t p);
#endif
GC_INNER void GC_clear_hdr_marks(hdr*hhdr);
GC_INNER void GC_set_hdr_marks(hdr*hhdr);
GC_INNER void GC_set_fl_marks(ptr_t p);
#if defined(GC_ASSERTIONS)&&defined(THREAD_LOCAL_ALLOC)
void GC_check_fl_marks(void**);
#endif
void GC_add_roots_inner(ptr_t b,ptr_t e,GC_bool tmp);
#ifdef USE_PROC_FOR_LIBRARIES
GC_INNER void GC_remove_roots_subregion(ptr_t b,ptr_t e);
#endif
GC_INNER void GC_exclude_static_roots_inner(void*start,void*finish);
#if defined(DYNAMIC_LOADING)||defined(MSWIN32)||defined(MSWINCE)||defined(CYGWIN32)||defined(PCR)
GC_INNER void GC_register_dynamic_libraries(void);
#endif
GC_INNER void GC_cond_register_dynamic_libraries(void);
ptr_t GC_get_main_stack_base(void);
#ifdef IA64
GC_INNER ptr_t GC_get_register_stack_base(void);
#endif
void GC_register_data_segments(void);
#ifdef THREADS
GC_INNER void GC_thr_init(void);
GC_INNER void GC_init_parallel(void);
#else
GC_INNER GC_bool GC_is_static_root(void*p);
#ifdef TRACE_BUF
void GC_add_trace_entry(char*kind,word arg1,word arg2);
#endif
#endif
#ifdef PRINT_BLACK_LIST
GC_INNER void GC_add_to_black_list_normal(word p,ptr_t source);
#define GC_ADD_TO_BLACK_LIST_NORMAL(bits,source)if (GC_all_interior_pointers){ GC_add_to_black_list_stack((word)(bits),(source));} else GC_add_to_black_list_normal((word)(bits),(source))
GC_INNER void GC_add_to_black_list_stack(word p,ptr_t source);
#define GC_ADD_TO_BLACK_LIST_STACK(bits,source)GC_add_to_black_list_stack((word)(bits),(source))
#else
GC_INNER void GC_add_to_black_list_normal(word p);
#define GC_ADD_TO_BLACK_LIST_NORMAL(bits,source)if (GC_all_interior_pointers){ GC_add_to_black_list_stack((word)(bits));} else GC_add_to_black_list_normal((word)(bits))
GC_INNER void GC_add_to_black_list_stack(word p);
#define GC_ADD_TO_BLACK_LIST_STACK(bits,source)GC_add_to_black_list_stack((word)(bits))
#endif
struct hblk*GC_is_black_listed(struct hblk*h,word len);
GC_INNER void GC_promote_black_lists(void);
GC_INNER void GC_unpromote_black_lists(void);
GC_INNER ptr_t GC_scratch_alloc(size_t bytes);
#ifdef GWW_VDB
#else
#define GC_scratch_recycle_no_gww GC_scratch_recycle_inner
#endif
GC_INNER void GC_scratch_recycle_inner(void*ptr,size_t bytes);
#ifdef MARK_BIT_PER_GRANULE
GC_INNER GC_bool GC_add_map_entry(size_t sz);
#endif
GC_INNER void GC_register_displacement_inner(size_t offset);
GC_INNER void GC_new_hblk(size_t size_in_granules,int kind);
GC_INNER ptr_t GC_build_fl(struct hblk*h,size_t words,GC_bool clear,
ptr_t list);
GC_INNER struct hblk*GC_allochblk(size_t size_in_bytes,int kind,
unsigned flags);
GC_INNER ptr_t GC_alloc_large(size_t lb,int k,unsigned flags);
GC_INNER void GC_freehblk(struct hblk*p);
GC_INNER GC_bool GC_expand_hp_inner(word n);
GC_INNER void GC_start_reclaim(GC_bool abort_if_found);
GC_INNER void GC_continue_reclaim(word sz,int kind);
GC_INNER GC_bool GC_reclaim_all(GC_stop_func stop_func,GC_bool ignore_old);
GC_INNER ptr_t GC_reclaim_generic(struct hblk*hbp,hdr*hhdr,size_t sz,
GC_bool init,ptr_t list,
signed_word*count);
GC_INNER GC_bool GC_block_empty(hdr*hhdr);
GC_INNER int GC_CALLBACK GC_never_stop_func(void);
GC_INNER GC_bool GC_try_to_collect_inner(GC_stop_func f);
#define GC_gcollect_inner()(void)GC_try_to_collect_inner(GC_never_stop_func)
#ifdef THREADS
GC_EXTERN GC_bool GC_in_thread_creation;
#endif
GC_EXTERN GC_bool GC_is_initialized;
GC_INNER void GC_collect_a_little_inner(int n);
GC_INNER void*GC_generic_malloc_inner(size_t lb,int k);
#if defined(DBG_HDRS_ALL)||defined(GC_GCJ_SUPPORT)||!defined(GC_NO_FINALIZATION)
GC_INNER void*GC_generic_malloc_inner_ignore_off_page(size_t lb,int k);
#endif
GC_INNER GC_bool GC_collect_or_expand(word needed_blocks,
GC_bool ignore_off_page,GC_bool retry);
GC_INNER ptr_t GC_allocobj(size_t sz,int kind);
#ifdef GC_ADD_CALLER
#ifdef GC_HAVE_RETURN_ADDR_PARENT
#define GC_DBG_EXTRAS GC_RETURN_ADDR_PARENT,NULL,0
#else
#define GC_DBG_EXTRAS GC_RETURN_ADDR,NULL,0
#endif
#else
#define GC_DBG_EXTRAS "unknown",0
#endif
#ifdef GC_COLLECT_AT_MALLOC
extern size_t GC_dbg_collect_at_malloc_min_lb;
#define GC_DBG_COLLECT_AT_MALLOC(lb)(void)((lb)>=GC_dbg_collect_at_malloc_min_lb?(GC_gcollect(),0):0)
#else
#define GC_DBG_COLLECT_AT_MALLOC(lb)(void)0
#endif
#if defined(THREAD_LOCAL_ALLOC)&&defined(GC_GCJ_SUPPORT)
GC_INNER void*GC_core_gcj_malloc(size_t,void*);
#endif
GC_INNER void GC_init_headers(void);
GC_INNER struct hblkhdr*GC_install_header(struct hblk*h);
GC_INNER GC_bool GC_install_counts(struct hblk*h,size_t sz);
GC_INNER void GC_remove_header(struct hblk*h);
GC_INNER void GC_remove_counts(struct hblk*h,size_t sz);
GC_INNER hdr*GC_find_header(ptr_t h);
GC_INNER void GC_add_to_heap(struct hblk*p,size_t bytes);
#ifdef USE_PROC_FOR_LIBRARIES
GC_INNER void GC_add_to_our_memory(ptr_t p,size_t bytes);
#else
#define GC_add_to_our_memory(p,bytes)
#endif
GC_INNER void GC_print_all_errors(void);
GC_EXTERN void (*GC_check_heap)(void);
GC_EXTERN void (*GC_print_all_smashed)(void);
GC_EXTERN void (*GC_print_heap_obj)(ptr_t p);
#if defined(LINUX)&&defined(__ELF__)&&!defined(SMALL_CONFIG)
void GC_print_address_map(void);
#endif
#ifndef SHORT_DBG_HDRS
GC_EXTERN GC_bool GC_findleak_delay_free;
GC_INNER GC_bool GC_check_leaked(ptr_t base);
#endif
GC_EXTERN GC_bool GC_have_errors;
#define VERBOSE 2
#if!defined(NO_CLOCK)||!defined(SMALL_CONFIG)
extern int GC_print_stats;
#else
#define GC_print_stats 0
#endif
#ifdef KEEP_BACK_PTRS
GC_EXTERN long GC_backtraces;
GC_INNER void GC_generate_random_backtrace_no_gc(void);
#endif
#ifdef LINT2
#define GC_RAND_MAX (~0U>>1)
GC_API_PRIV long GC_random(void);
#endif
GC_EXTERN GC_bool GC_print_back_height;
#ifdef MAKE_BACK_GRAPH
void GC_print_back_graph_stats(void);
#endif
#ifdef THREADS
GC_INNER void GC_free_inner(void*p);
#endif
#ifdef DBG_HDRS_ALL
GC_INNER void*GC_debug_generic_malloc_inner(size_t lb,int k);
GC_INNER void*GC_debug_generic_malloc_inner_ignore_off_page(size_t lb,
int k);
#define GC_INTERNAL_MALLOC GC_debug_generic_malloc_inner
#define GC_INTERNAL_MALLOC_IGNORE_OFF_PAGE GC_debug_generic_malloc_inner_ignore_off_page
#ifdef THREADS
GC_INNER void GC_debug_free_inner(void*p);
#define GC_INTERNAL_FREE GC_debug_free_inner
#else
#define GC_INTERNAL_FREE GC_debug_free
#endif
#else
#define GC_INTERNAL_MALLOC GC_generic_malloc_inner
#define GC_INTERNAL_MALLOC_IGNORE_OFF_PAGE GC_generic_malloc_inner_ignore_off_page
#ifdef THREADS
#define GC_INTERNAL_FREE GC_free_inner
#else
#define GC_INTERNAL_FREE GC_free
#endif
#endif
#ifdef USE_MUNMAP
GC_INNER void GC_unmap_old(void);
GC_INNER void GC_merge_unmapped(void);
GC_INNER void GC_unmap(ptr_t start,size_t bytes);
GC_INNER void GC_remap(ptr_t start,size_t bytes);
GC_INNER void GC_unmap_gap(ptr_t start1,size_t bytes1,ptr_t start2,
size_t bytes2);
GC_INLINE ptr_t GC_unmap_end(ptr_t start,size_t bytes)
{
return (ptr_t)((word)(start+bytes)&~(GC_page_size - 1));
}
#endif
#ifdef CAN_HANDLE_FORK
GC_EXTERN int GC_handle_fork;
#endif
#ifdef GC_DISABLE_INCREMENTAL
#define GC_incremental FALSE
#define GC_auto_incremental FALSE
#define GC_manual_vdb FALSE
#define GC_dirty(p)(void)(p)
#define REACHABLE_AFTER_DIRTY(p)(void)(p)
#else
GC_EXTERN GC_bool GC_incremental;
GC_INNER void GC_read_dirty(GC_bool output_unneeded);
GC_INNER GC_bool GC_page_was_dirty(struct hblk*h);
GC_INNER void GC_remove_protection(struct hblk*h,word nblocks,
GC_bool pointerfree);
GC_INNER GC_bool GC_dirty_init(void);
GC_EXTERN GC_bool GC_manual_vdb;
#define GC_auto_incremental (GC_incremental&&!GC_manual_vdb)
GC_INNER void GC_dirty_inner(const void*p);
#define GC_dirty(p)(GC_manual_vdb?GC_dirty_inner(p):(void)0)
#define REACHABLE_AFTER_DIRTY(p)GC_reachable_here(p)
#endif
#define GC_base_C(p)((const void*)GC_base(( void*)(p)))
void GC_print_block_list(void);
void GC_print_hblkfreelist(void);
void GC_print_heap_sects(void);
void GC_print_static_roots(void);
#ifdef KEEP_BACK_PTRS
GC_INNER void GC_store_back_pointer(ptr_t source,ptr_t dest);
GC_INNER void GC_marked_for_finalization(ptr_t dest);
#define GC_STORE_BACK_PTR(source,dest)GC_store_back_pointer(source,dest)
#define GC_MARKED_FOR_FINALIZATION(dest)GC_marked_for_finalization(dest)
#else
#define GC_STORE_BACK_PTR(source,dest)(void)(source)
#define GC_MARKED_FOR_FINALIZATION(dest)
#endif
void GC_noop6(word,word,word,word,word,word);
GC_API void GC_CALL GC_noop1(word);
#ifndef GC_ATTR_FORMAT_PRINTF
#if GC_GNUC_PREREQ(3,0)
#define GC_ATTR_FORMAT_PRINTF(spec_argnum,first_checked)__attribute__((__format__(__printf__,spec_argnum,first_checked)))
#else
#define GC_ATTR_FORMAT_PRINTF(spec_argnum,first_checked)
#endif
#endif
GC_API_PRIV void GC_printf(const char*format,...)
GC_ATTR_FORMAT_PRINTF(1,2);
GC_API_PRIV void GC_err_printf(const char*format,...)
GC_ATTR_FORMAT_PRINTF(1,2);
GC_API_PRIV void GC_log_printf(const char*format,...)
GC_ATTR_FORMAT_PRINTF(1,2);
#ifndef GC_ANDROID_LOG
#define GC_PRINT_STATS_FLAG (GC_print_stats!=0)
#define GC_INFOLOG_PRINTF GC_COND_LOG_PRINTF
#define GC_verbose_log_printf GC_log_printf
#else
extern GC_bool GC_quiet;
#define GC_PRINT_STATS_FLAG (!GC_quiet)
#ifndef GC_INFOLOG_PRINTF
#define GC_INFOLOG_PRINTF if (GC_quiet){} else GC_info_log_printf
#endif
GC_INNER void GC_info_log_printf(const char*format,...)
GC_ATTR_FORMAT_PRINTF(1,2);
GC_INNER void GC_verbose_log_printf(const char*format,...)
GC_ATTR_FORMAT_PRINTF(1,2);
#endif
#define GC_COND_LOG_PRINTF if (EXPECT(!GC_print_stats,TRUE)){} else GC_log_printf
#define GC_VERBOSE_LOG_PRINTF if (EXPECT(GC_print_stats!=VERBOSE,TRUE)){} else GC_verbose_log_printf
#ifndef GC_DBGLOG_PRINTF
#define GC_DBGLOG_PRINTF if (!GC_PRINT_STATS_FLAG){} else GC_log_printf
#endif
void GC_err_puts(const char*s);
#define TO_KiB_UL(v)((unsigned long)(((v)+((1<<9)- 1))>>10))
GC_EXTERN unsigned GC_fail_count;
GC_EXTERN long GC_large_alloc_warn_interval;
GC_EXTERN signed_word GC_bytes_found;
#ifndef GC_GET_HEAP_USAGE_NOT_NEEDED
GC_EXTERN word GC_reclaimed_bytes_before_gc;
#endif
#ifdef USE_MUNMAP
GC_EXTERN int GC_unmap_threshold;
GC_EXTERN GC_bool GC_force_unmap_on_gcollect;
#endif
#ifdef MSWIN32
GC_EXTERN GC_bool GC_no_win32_dlls;
GC_EXTERN GC_bool GC_wnt;
#endif
#ifdef THREADS
#if (defined(MSWIN32)&&!defined(CONSOLE_LOG))||defined(MSWINCE)
GC_EXTERN CRITICAL_SECTION GC_write_cs;
#ifdef GC_ASSERTIONS
GC_EXTERN GC_bool GC_write_disabled;
#endif
#endif
#if defined(GC_DISABLE_INCREMENTAL)||defined(HAVE_LOCKFREE_AO_OR)
#define GC_acquire_dirty_lock()(void)0
#define GC_release_dirty_lock()(void)0
#else
#define GC_acquire_dirty_lock()do { } while (AO_test_and_set_acquire(&GC_fault_handler_lock)==AO_TS_SET)
#define GC_release_dirty_lock()AO_CLEAR(&GC_fault_handler_lock)
GC_EXTERN volatile AO_TS_t GC_fault_handler_lock;
#endif
#ifdef MSWINCE
GC_EXTERN GC_bool GC_dont_query_stack_min;
#endif
#elif defined(IA64)
GC_EXTERN ptr_t GC_save_regs_ret_val;
#endif
#ifdef THREAD_LOCAL_ALLOC
GC_EXTERN GC_bool GC_world_stopped;
GC_INNER void GC_mark_thread_local_free_lists(void);
#endif
#if defined(MPROTECT_VDB)&&defined(GWW_VDB)
GC_INNER GC_bool GC_gww_dirty_init(void);
#endif
#if defined(CHECKSUMS)||defined(PROC_VDB)
GC_INNER GC_bool GC_page_was_ever_dirty(struct hblk*h);
#endif
#ifdef CHECKSUMS
#if defined(MPROTECT_VDB)&&!defined(DARWIN)
void GC_record_fault(struct hblk*h);
#endif
void GC_check_dirty(void);
#endif
GC_INNER void GC_default_print_heap_obj_proc(ptr_t p);
GC_INNER void GC_setpagesize(void);
GC_INNER void GC_initialize_offsets(void);
GC_INNER void GC_bl_init(void);
GC_INNER void GC_bl_init_no_interiors(void);
GC_INNER void GC_start_debugging_inner(void);
GC_INNER void*GC_store_debug_info_inner(void*p,word sz,const char*str,
int linenum);
#ifdef REDIRECT_MALLOC
#ifdef GC_LINUX_THREADS
GC_INNER GC_bool GC_text_mapping(char*nm,ptr_t*startp,ptr_t*endp);
#endif
#elif defined(USE_WINALLOC)
GC_INNER void GC_add_current_malloc_heap(void);
#endif
#ifdef MAKE_BACK_GRAPH
GC_INNER void GC_build_back_graph(void);
GC_INNER void GC_traverse_back_graph(void);
#endif
#ifdef MSWIN32
GC_INNER void GC_init_win32(void);
#endif
#if!defined(MSWIN32)&&!defined(MSWINCE)&&!defined(CYGWIN32)
GC_INNER void*GC_roots_present(ptr_t);
#endif
#ifdef GC_WIN32_THREADS
GC_INNER void GC_get_next_stack(char*start,char*limit,char**lo,
char**hi);
#if defined(MPROTECT_VDB)&&!defined(CYGWIN32)
GC_INNER void GC_set_write_fault_handler(void);
#endif
#if defined(WRAP_MARK_SOME)&&!defined(GC_PTHREADS)
GC_INNER GC_bool GC_started_thread_while_stopped(void);
#endif
#endif
#ifdef THREADS
GC_INNER void GC_reset_finalizer_nested(void);
GC_INNER unsigned char*GC_check_finalizer_nested(void);
GC_INNER void GC_do_blocking_inner(ptr_t data,void*context);
GC_INNER void GC_push_all_stacks(void);
#ifdef USE_PROC_FOR_LIBRARIES
GC_INNER GC_bool GC_segment_is_thread_stack(ptr_t lo,ptr_t hi);
#endif
#ifdef IA64
GC_INNER ptr_t GC_greatest_stack_base_below(ptr_t bound);
#endif
#endif
#ifdef DYNAMIC_LOADING
GC_INNER GC_bool GC_register_main_static_data(void);
#ifdef DARWIN
GC_INNER void GC_init_dyld(void);
#endif
#endif
#ifdef SEARCH_FOR_DATA_START
GC_INNER void GC_init_linux_data_start(void);
void*GC_find_limit(void*,int);
#endif
#if defined(NETBSD)&&defined(__ELF__)
GC_INNER void GC_init_netbsd_elf(void);
void*GC_find_limit(void*,int);
#endif
#ifdef UNIX_LIKE
GC_INNER void GC_set_and_save_fault_handler(void (*handler)(int));
#endif
#ifdef NEED_PROC_MAPS
#if defined(DYNAMIC_LOADING)&&defined(USE_PROC_FOR_LIBRARIES)
GC_INNER char*GC_parse_map_entry(char*buf_ptr,ptr_t*start,ptr_t*end,
char**prot,unsigned int*maj_dev,
char**mapping_name);
#endif
#if defined(IA64)||defined(INCLUDE_LINUX_THREAD_DESCR)
GC_INNER GC_bool GC_enclosing_mapping(ptr_t addr,
ptr_t*startp,ptr_t*endp);
#endif
GC_INNER char*GC_get_maps(void);
#endif
#ifdef GC_ASSERTIONS
#define GC_ASSERT(expr)do { if (!(expr)){ GC_err_printf("Assertion failure:%s:%d\n",__FILE__,__LINE__);ABORT("assertion failure");} } while (0)
GC_INNER word GC_compute_large_free_bytes(void);
GC_INNER word GC_compute_root_size(void);
#else
#define GC_ASSERT(expr)
#endif
#if _MSC_VER>=1700
#define GC_STATIC_ASSERT(expr)static_assert(expr,"static assertion failed:" #expr)
#elif defined(static_assert)&&__STDC_VERSION__>=201112L
#define GC_STATIC_ASSERT(expr)static_assert(expr,#expr)
#elif defined(mips)&&!defined(__GNUC__)
#define GC_STATIC_ASSERT(expr)do { if (0){ char j[(expr)?1:-1];j[0]='\0';j[0]=j[0];} } while(0)
#else
#define GC_STATIC_ASSERT(expr)(void)sizeof(char[(expr)?1:-1])
#endif
#if GC_GNUC_PREREQ(4,0)
#define NONNULL_ARG_NOT_NULL(arg)(*(volatile void**)&(arg)!=NULL)
#else
#define NONNULL_ARG_NOT_NULL(arg)(NULL!=(arg))
#endif
#define COND_DUMP_CHECKS do { GC_ASSERT(GC_compute_large_free_bytes()==GC_large_free_bytes);GC_ASSERT(GC_compute_root_size()==GC_root_size);} while (0)
#ifndef NO_DEBUGGING
GC_EXTERN GC_bool GC_dump_regularly;
#define COND_DUMP if (EXPECT(GC_dump_regularly,FALSE)){ GC_dump_named(NULL);} else COND_DUMP_CHECKS
#else
#define COND_DUMP COND_DUMP_CHECKS
#endif
#if defined(PARALLEL_MARK)
#define GC_markers_m1 GC_parallel
GC_EXTERN GC_bool GC_parallel_mark_disabled;
GC_INNER void GC_wait_for_markers_init(void);
GC_INNER void GC_acquire_mark_lock(void);
GC_INNER void GC_release_mark_lock(void);
GC_INNER void GC_notify_all_builder(void);
GC_INNER void GC_wait_for_reclaim(void);
GC_EXTERN signed_word GC_fl_builder_count;
GC_INNER void GC_notify_all_marker(void);
GC_INNER void GC_wait_marker(void);
GC_EXTERN word GC_mark_no;
GC_INNER void GC_help_marker(word my_mark_no);
GC_INNER void GC_start_mark_threads_inner(void);
#endif
#if defined(GC_PTHREADS)&&!defined(GC_WIN32_THREADS)&&!defined(NACL)&&!defined(GC_DARWIN_THREADS)&&!defined(SIG_SUSPEND)
#if (defined(GC_LINUX_THREADS)||defined(GC_DGUX386_THREADS))&&!defined(GC_USESIGRT_SIGNALS)
#if defined(SPARC)&&!defined(SIGPWR)
#define SIG_SUSPEND SIGLOST
#else
#define SIG_SUSPEND SIGPWR
#endif
#elif defined(GC_OPENBSD_THREADS)
#ifndef GC_OPENBSD_UTHREADS
#define SIG_SUSPEND SIGXFSZ
#endif
#elif defined(_SIGRTMIN)&&!defined(CPPCHECK)
#define SIG_SUSPEND _SIGRTMIN+6
#else
#define SIG_SUSPEND SIGRTMIN+6
#endif
#endif
#if defined(GC_PTHREADS)&&!defined(GC_SEM_INIT_PSHARED)
#define GC_SEM_INIT_PSHARED 0
#endif
#if (defined(UNIX_LIKE)||(defined(NEED_FIND_LIMIT)&&defined(CYGWIN32)))&&!defined(GC_NO_SIGSETJMP)
#if defined(SUNOS5SIGS)&&!defined(FREEBSD)&&!defined(LINUX)
EXTERN_C_END
#include <sys/siginfo.h>
EXTERN_C_BEGIN
#endif
#define SETJMP(env)sigsetjmp(env,1)
#define LONGJMP(env,val)siglongjmp(env,val)
#define JMP_BUF sigjmp_buf
#else
#ifdef ECOS
#define SETJMP(env)hal_setjmp(env)
#else
#define SETJMP(env)setjmp(env)
#endif
#define LONGJMP(env,val)longjmp(env,val)
#define JMP_BUF jmp_buf
#endif
#if defined(HEURISTIC2)||defined(SEARCH_FOR_DATA_START)||((defined(SVR4)||defined(AIX)||defined(DGUX)||(defined(LINUX)&&defined(SPARC)))&&!defined(PCR))
#define NEED_FIND_LIMIT
#endif
#if defined(DATASTART_USES_BSDGETDATASTART)
EXTERN_C_END
#include <machine/trap.h>
EXTERN_C_BEGIN
#if!defined(PCR)
#define NEED_FIND_LIMIT
#endif
GC_INNER ptr_t GC_FreeBSDGetDataStart(size_t,ptr_t);
#define DATASTART_IS_FUNC
#endif
#if (defined(NETBSD)||defined(OPENBSD))&&defined(__ELF__)&&!defined(NEED_FIND_LIMIT)
#define NEED_FIND_LIMIT
#endif
#if defined(IA64)&&!defined(NEED_FIND_LIMIT)
#define NEED_FIND_LIMIT
#endif
#if defined(NEED_FIND_LIMIT)||(defined(USE_PROC_FOR_LIBRARIES)&&defined(THREADS))
GC_EXTERN JMP_BUF GC_jmp_buf;
GC_INNER void GC_setup_temporary_fault_handler(void);
GC_INNER void GC_reset_fault_handler(void);
#endif
#if defined(CANCEL_SAFE)
#if defined(GC_ASSERTIONS)&&(defined(USE_COMPILER_TLS)||(defined(LINUX)&&!defined(ARM32)&&GC_GNUC_PREREQ(3,3)||defined(HPUX)))
extern __thread unsigned char GC_cancel_disable_count;
#define NEED_CANCEL_DISABLE_COUNT
#define INCR_CANCEL_DISABLE()++GC_cancel_disable_count
#define DECR_CANCEL_DISABLE()--GC_cancel_disable_count
#define ASSERT_CANCEL_DISABLED()GC_ASSERT(GC_cancel_disable_count > 0)
#else
#define INCR_CANCEL_DISABLE()
#define DECR_CANCEL_DISABLE()
#define ASSERT_CANCEL_DISABLED()(void)0
#endif
#define DISABLE_CANCEL(state)do { pthread_setcancelstate(PTHREAD_CANCEL_DISABLE,&state);INCR_CANCEL_DISABLE();} while (0)
#define RESTORE_CANCEL(state)do { ASSERT_CANCEL_DISABLED();pthread_setcancelstate(state,NULL);DECR_CANCEL_DISABLE();} while (0)
#else
#define DISABLE_CANCEL(state)(void)0
#define RESTORE_CANCEL(state)(void)0
#define ASSERT_CANCEL_DISABLED()(void)0
#endif
EXTERN_C_END
#endif
#ifdef KEEP_BACK_PTRS
#ifndef GC_BACKPTR_H
#define GC_BACKPTR_H
#ifndef GC_H
#endif
#ifdef __cplusplus
extern "C" {
#endif
typedef enum {
GC_UNREFERENCED,
GC_NO_SPACE,
GC_REFD_FROM_ROOT,
GC_REFD_FROM_REG,
GC_REFD_FROM_HEAP,
GC_FINALIZER_REFD
} GC_ref_kind;
GC_API GC_ref_kind GC_CALL GC_get_back_ptr_info(void*,
void**,size_t*)
GC_ATTR_NONNULL(1);
GC_API void*GC_CALL GC_generate_random_heap_address(void);
GC_API void*GC_CALL GC_generate_random_valid_address(void);
GC_API void GC_CALL GC_generate_random_backtrace(void);
GC_API void GC_CALL GC_print_backtrace(void*)GC_ATTR_NONNULL(1);
#ifdef __cplusplus
}
#endif
#endif
#endif
EXTERN_C_BEGIN
#if CPP_WORDSZ==32
#define START_FLAG (word)0xfedcedcb
#define END_FLAG (word)0xbcdecdef
#else
#define START_FLAG GC_WORD_C(0xFEDCEDCBfedcedcb)
#define END_FLAG GC_WORD_C(0xBCDECDEFbcdecdef)
#endif
#if defined(KEEP_BACK_PTRS)||defined(PRINT_BLACK_LIST)||defined(MAKE_BACK_GRAPH)
#define NOT_MARKED (ptr_t)0
#define MARKED_FOR_FINALIZATION ((ptr_t)(word)2)
#define MARKED_FROM_REGISTER ((ptr_t)(word)4)
#endif
typedef struct {
#if defined(KEEP_BACK_PTRS)||defined(MAKE_BACK_GRAPH)
#if ALIGNMENT==1
#define HIDE_BACK_PTR(p)GC_HIDE_POINTER(~1&(word)(p))
#else
#define HIDE_BACK_PTR(p)GC_HIDE_POINTER(p)
#endif
#ifdef KEEP_BACK_PTRS
GC_hidden_pointer oh_back_ptr;
#endif
#ifdef MAKE_BACK_GRAPH
GC_hidden_pointer oh_bg_ptr;
#endif
#if defined(KEEP_BACK_PTRS)!=defined(MAKE_BACK_GRAPH)
word oh_dummy;
#endif
#endif
const char*oh_string;
signed_word oh_int;
#ifdef NEED_CALLINFO
struct callinfo oh_ci[NFRAMES];
#endif
#ifndef SHORT_DBG_HDRS
word oh_sz;
word oh_sf;
#endif
} oh;
#ifdef SHORT_DBG_HDRS
#define DEBUG_BYTES (sizeof (oh))
#define UNCOLLECTABLE_DEBUG_BYTES DEBUG_BYTES
#else
#define UNCOLLECTABLE_DEBUG_BYTES (sizeof (oh)+sizeof (word))
#define DEBUG_BYTES (UNCOLLECTABLE_DEBUG_BYTES - EXTRA_BYTES)
#endif
#define SIMPLE_ROUNDED_UP_WORDS(n)BYTES_TO_WORDS((n)+WORDS_TO_BYTES(1)- 1)
#if defined(SAVE_CALL_CHAIN)
struct callinfo;
GC_INNER void GC_save_callers(struct callinfo info[NFRAMES]);
GC_INNER void GC_print_callers(struct callinfo info[NFRAMES]);
#define ADD_CALL_CHAIN(base,ra)GC_save_callers(((oh*)(base))->oh_ci)
#define PRINT_CALL_CHAIN(base)GC_print_callers(((oh*)(base))->oh_ci)
#elif defined(GC_ADD_CALLER)
struct callinfo;
GC_INNER void GC_print_callers(struct callinfo info[NFRAMES]);
#define ADD_CALL_CHAIN(base,ra)((oh*)(base))->oh_ci[0].ci_pc=(ra)
#define PRINT_CALL_CHAIN(base)GC_print_callers(((oh*)(base))->oh_ci)
#else
#define ADD_CALL_CHAIN(base,ra)
#define PRINT_CALL_CHAIN(base)
#endif
#ifdef GC_ADD_CALLER
#define OPT_RA ra,
#else
#define OPT_RA
#endif
#ifdef SHORT_DBG_HDRS
#define GC_has_other_debug_info(p)1
#else
GC_INNER int GC_has_other_debug_info(ptr_t p);
#endif
#if defined(KEEP_BACK_PTRS)||defined(MAKE_BACK_GRAPH)
#if defined(SHORT_DBG_HDRS)&&!defined(CPPCHECK)
#error Non-ptr stored in object results in GC_HAS_DEBUG_INFO malfunction
#endif
#if defined(PARALLEL_MARK)&&defined(KEEP_BACK_PTRS)
#define GC_HAS_DEBUG_INFO(p)((AO_load((volatile AO_t*)(p))&1)!=0&&GC_has_other_debug_info(p)> 0)
#else
#define GC_HAS_DEBUG_INFO(p)((*(word*)(p)&1)&&GC_has_other_debug_info(p)> 0)
#endif
#else
#define GC_HAS_DEBUG_INFO(p)(GC_has_other_debug_info(p)> 0)
#endif
EXTERN_C_END
#endif
#ifdef MAKE_BACK_GRAPH
#define MAX_IN 10
#if (!defined(DBG_HDRS_ALL)||(ALIGNMENT!=CPP_WORDSZ/8))&&!defined(CPPCHECK)
#error The configuration does not support MAKE_BACK_GRAPH
#endif
#define FLAG_MANY 2
typedef struct back_edges_struct {
word n_edges;
unsigned short flags;
#define RETAIN 1
unsigned short height_gc_no;
signed_word height;
#define HEIGHT_UNKNOWN (-2)
#define HEIGHT_IN_PROGRESS (-1)
ptr_t edges[MAX_IN];
struct back_edges_struct*cont;
} back_edges;
#define MAX_BACK_EDGE_STRUCTS 100000
static back_edges*back_edge_space=0;
STATIC int GC_n_back_edge_structs=0;
static back_edges*avail_back_edges=0;
static back_edges*new_back_edges(void)
{
if (0==back_edge_space){
size_t bytes_to_get=ROUNDUP_PAGESIZE_IF_MMAP(MAX_BACK_EDGE_STRUCTS
*sizeof(back_edges));
GC_ASSERT(GC_page_size!=0);
back_edge_space=(back_edges*)GET_MEM(bytes_to_get);
if (NULL==back_edge_space)
ABORT("Insufficient memory for back edges");
GC_add_to_our_memory((ptr_t)back_edge_space,bytes_to_get);
}
if (0!=avail_back_edges){
back_edges*result=avail_back_edges;
avail_back_edges=result->cont;
result->cont=0;
return result;
}
if (GC_n_back_edge_structs>=MAX_BACK_EDGE_STRUCTS - 1){
ABORT("Needed too much space for back edges:adjust "
"MAX_BACK_EDGE_STRUCTS");
}
return back_edge_space+(GC_n_back_edge_structs++);
}
static void deallocate_back_edges(back_edges*p)
{
back_edges*last=p;
while (0!=last->cont)last=last->cont;
last->cont=avail_back_edges;
avail_back_edges=p;
}
#define INITIAL_IN_PROGRESS 10000
static ptr_t*in_progress_space=0;
static size_t in_progress_size=0;
static size_t n_in_progress=0;
static void push_in_progress(ptr_t p)
{
if (n_in_progress>=in_progress_size){
ptr_t*new_in_progress_space;
GC_ASSERT(GC_page_size!=0);
if (NULL==in_progress_space){
in_progress_size=ROUNDUP_PAGESIZE_IF_MMAP(INITIAL_IN_PROGRESS
*sizeof(ptr_t))
/sizeof(ptr_t);
new_in_progress_space=
(ptr_t*)GET_MEM(in_progress_size*sizeof(ptr_t));
} else {
in_progress_size*=2;
new_in_progress_space=(ptr_t*)
GET_MEM(in_progress_size*sizeof(ptr_t));
if (new_in_progress_space!=NULL)
BCOPY(in_progress_space,new_in_progress_space,
n_in_progress*sizeof(ptr_t));
}
GC_add_to_our_memory((ptr_t)new_in_progress_space,
in_progress_size*sizeof(ptr_t));
#ifndef GWW_VDB
GC_scratch_recycle_no_gww(in_progress_space,
n_in_progress*sizeof(ptr_t));
#elif defined(LINT2)
GC_noop1((word)in_progress_space);
#endif
in_progress_space=new_in_progress_space;
}
if (in_progress_space==0)
ABORT("MAKE_BACK_GRAPH:Out of in-progress space:"
"Huge linear data structure?");
in_progress_space[n_in_progress++]=p;
}
static GC_bool is_in_progress(ptr_t p)
{
size_t i;
for (i=0;i < n_in_progress;++i){
if (in_progress_space[i]==p)return TRUE;
}
return FALSE;
}
GC_INLINE void pop_in_progress(ptr_t p GC_ATTR_UNUSED)
{
--n_in_progress;
GC_ASSERT(in_progress_space[n_in_progress]==p);
}
#define GET_OH_BG_PTR(p)(ptr_t)GC_REVEAL_POINTER(((oh*)(p))->oh_bg_ptr)
#define SET_OH_BG_PTR(p,q)(((oh*)(p))->oh_bg_ptr=GC_HIDE_POINTER(q))
static void ensure_struct(ptr_t p)
{
ptr_t old_back_ptr=GET_OH_BG_PTR(p);
if (!((word)old_back_ptr&FLAG_MANY)){
back_edges*be=new_back_edges();
be->flags=0;
if (0==old_back_ptr){
be->n_edges=0;
} else {
be->n_edges=1;
be->edges[0]=old_back_ptr;
}
be->height=HEIGHT_UNKNOWN;
be->height_gc_no=(unsigned short)(GC_gc_no - 1);
GC_ASSERT((word)be>=(word)back_edge_space);
SET_OH_BG_PTR(p,(word)be|FLAG_MANY);
}
}
static void add_edge(ptr_t p,ptr_t q)
{
ptr_t pred=GET_OH_BG_PTR(q);
back_edges*be,*be_cont;
word i;
GC_ASSERT(p==GC_base(p)&&q==GC_base(q));
if (!GC_HAS_DEBUG_INFO(q)||!GC_HAS_DEBUG_INFO(p)){
return;
}
if (NULL==pred){
static unsigned random_number=13;
#define GOT_LUCKY_NUMBER (((++random_number)&0x7f)==0)
SET_OH_BG_PTR(q,p);
if (GOT_LUCKY_NUMBER)ensure_struct(q);
return;
}
{
back_edges*e=(back_edges*)((word)pred&~FLAG_MANY);
word n_edges;
word total;
int local=0;
if (((word)pred&FLAG_MANY)!=0){
n_edges=e->n_edges;
} else if (((word)COVERT_DATAFLOW(pred)&1)==0){
n_edges=1;
local=-1;
} else {
n_edges=0;
}
for (total=0;total < n_edges;++total){
if (local==MAX_IN){
e=e->cont;
local=0;
}
if (local>=0)
pred=e->edges[local++];
if (pred==p)
return;
}
}
ensure_struct(q);
be=(back_edges*)((word)GET_OH_BG_PTR(q)&~FLAG_MANY);
for (i=be->n_edges,be_cont=be;i > MAX_IN;i-=MAX_IN)
be_cont=be_cont->cont;
if (i==MAX_IN){
be_cont->cont=new_back_edges();
be_cont=be_cont->cont;
i=0;
}
be_cont->edges[i]=p;
be->n_edges++;
#ifdef DEBUG_PRINT_BIG_N_EDGES
if (GC_print_stats==VERBOSE&&be->n_edges==100){
GC_err_printf("The following object has big in-degree:\n");
GC_print_heap_obj(q);
}
#endif
}
typedef void (*per_object_func)(ptr_t p,size_t n_bytes,word gc_descr);
static void per_object_helper(struct hblk*h,word fn)
{
hdr*hhdr=HDR(h);
size_t sz=(size_t)hhdr->hb_sz;
word descr=hhdr->hb_descr;
per_object_func f=(per_object_func)fn;
size_t i=0;
do {
f((ptr_t)(h->hb_body+i),sz,descr);
i+=sz;
} while (i+sz<=BYTES_TO_WORDS(HBLKSIZE));
}
GC_INLINE void GC_apply_to_each_object(per_object_func f)
{
GC_apply_to_all_blocks(per_object_helper,(word)f);
}
static void reset_back_edge(ptr_t p,size_t n_bytes GC_ATTR_UNUSED,
word gc_descr GC_ATTR_UNUSED)
{
if (GC_HAS_DEBUG_INFO(p)){
ptr_t old_back_ptr=GET_OH_BG_PTR(p);
if ((word)old_back_ptr&FLAG_MANY){
back_edges*be=(back_edges*)((word)old_back_ptr&~FLAG_MANY);
if (!(be->flags&RETAIN)){
deallocate_back_edges(be);
SET_OH_BG_PTR(p,0);
} else {
GC_ASSERT(GC_is_marked(p));
be->n_edges=0;
if (0!=be->cont){
deallocate_back_edges(be->cont);
be->cont=0;
}
GC_ASSERT(GC_is_marked(p));
be->flags&=~RETAIN;
}
} else {
SET_OH_BG_PTR(p,0);
}
}
}
static void add_back_edges(ptr_t p,size_t n_bytes,word gc_descr)
{
word*currentp=(word*)(p+sizeof(oh));
if((gc_descr&GC_DS_TAGS)!=GC_DS_LENGTH){
gc_descr=n_bytes;
}
while ((word)currentp < (word)(p+gc_descr)){
word current=*currentp++;
FIXUP_POINTER(current);
if (current>=(word)GC_least_plausible_heap_addr&&
current<=(word)GC_greatest_plausible_heap_addr){
ptr_t target=(ptr_t)GC_base((void*)current);
if (0!=target){
add_edge(p,target);
}
}
}
}
GC_INNER void GC_build_back_graph(void)
{
GC_ASSERT(I_HOLD_LOCK());
GC_apply_to_each_object(add_back_edges);
}
static word backwards_height(ptr_t p)
{
word result;
ptr_t pred=GET_OH_BG_PTR(p);
back_edges*be;
if (NULL==pred)
return 1;
if (((word)pred&FLAG_MANY)==0){
if (is_in_progress(p))return 0;
push_in_progress(p);
result=backwards_height(pred)+1;
pop_in_progress(p);
return result;
}
be=(back_edges*)((word)pred&~FLAG_MANY);
if (be->height>=0&&be->height_gc_no==(unsigned short)GC_gc_no)
return be->height;
if (be->height==HEIGHT_IN_PROGRESS)return 0;
result=(be->height > 0?be->height:1);
be->height=HEIGHT_IN_PROGRESS;
{
back_edges*e=be;
word n_edges;
word total;
int local=0;
if (((word)pred&FLAG_MANY)!=0){
n_edges=e->n_edges;
} else if (((word)pred&1)==0){
n_edges=1;
local=-1;
} else {
n_edges=0;
}
for (total=0;total < n_edges;++total){
word this_height;
if (local==MAX_IN){
e=e->cont;
local=0;
}
if (local>=0)
pred=e->edges[local++];
if (GC_is_marked(pred)&&((word)GET_OH_BG_PTR(p)&FLAG_MANY)==0){
GC_COND_LOG_PRINTF("Found bogus pointer from %p to %p\n",
(void*)pred,(void*)p);
this_height=1;
} else {
this_height=backwards_height(pred);
}
if (this_height>=result)
result=this_height+1;
}
}
be->height=result;
be->height_gc_no=(unsigned short)GC_gc_no;
return result;
}
STATIC word GC_max_height=0;
STATIC ptr_t GC_deepest_obj=NULL;
static void update_max_height(ptr_t p,size_t n_bytes GC_ATTR_UNUSED,
word gc_descr GC_ATTR_UNUSED)
{
if (GC_is_marked(p)&&GC_HAS_DEBUG_INFO(p)){
word p_height=0;
ptr_t p_deepest_obj=0;
ptr_t back_ptr;
back_edges*be=0;
back_ptr=GET_OH_BG_PTR(p);
if (0!=back_ptr&&((word)back_ptr&FLAG_MANY)){
be=(back_edges*)((word)back_ptr&~FLAG_MANY);
if (be->height!=HEIGHT_UNKNOWN)p_height=be->height;
}
{
ptr_t pred=GET_OH_BG_PTR(p);
back_edges*e=(back_edges*)((word)pred&~FLAG_MANY);
word n_edges;
word total;
int local=0;
if (((word)pred&FLAG_MANY)!=0){
n_edges=e->n_edges;
} else if (pred!=NULL&&((word)pred&1)==0){
n_edges=1;
local=-1;
} else {
n_edges=0;
}
for (total=0;total < n_edges;++total){
if (local==MAX_IN){
e=e->cont;
local=0;
}
if (local>=0)
pred=e->edges[local++];
if (!GC_is_marked(pred)&&GC_HAS_DEBUG_INFO(pred)){
word this_height=backwards_height(pred);
if (this_height > p_height){
p_height=this_height;
p_deepest_obj=pred;
}
}
}
}
if (p_height > 0){
if (be==0){
ensure_struct(p);
back_ptr=GET_OH_BG_PTR(p);
be=(back_edges*)((word)back_ptr&~FLAG_MANY);
}
be->flags|=RETAIN;
be->height=p_height;
be->height_gc_no=(unsigned short)GC_gc_no;
}
if (p_height > GC_max_height){
GC_max_height=p_height;
GC_deepest_obj=p_deepest_obj;
}
}
}
STATIC word GC_max_max_height=0;
GC_INNER void GC_traverse_back_graph(void)
{
GC_ASSERT(I_HOLD_LOCK());
GC_max_height=0;
GC_apply_to_each_object(update_max_height);
if (0!=GC_deepest_obj)
GC_set_mark_bit(GC_deepest_obj);
}
void GC_print_back_graph_stats(void)
{
GC_ASSERT(I_HOLD_LOCK());
GC_printf("Maximum backwards height of reachable objects at GC %lu is %lu\n",
(unsigned long)GC_gc_no,(unsigned long)GC_max_height);
if (GC_max_height > GC_max_max_height){
ptr_t obj=GC_deepest_obj;
GC_max_max_height=GC_max_height;
UNLOCK();
GC_err_printf(
"The following unreachable object is last in a longest chain "
"of unreachable objects:\n");
GC_print_heap_obj(obj);
LOCK();
}
GC_COND_LOG_PRINTF("Needed max total of %d back-edge structs\n",
GC_n_back_edge_structs);
GC_apply_to_each_object(reset_back_edge);
GC_deepest_obj=0;
}
#endif
STATIC word*GC_old_normal_bl=NULL;
STATIC word*GC_incomplete_normal_bl=NULL;
STATIC word*GC_old_stack_bl=NULL;
STATIC word*GC_incomplete_stack_bl=NULL;
STATIC word GC_total_stack_black_listed=0;
GC_INNER word GC_black_list_spacing=MINHINCR*HBLKSIZE;
STATIC void GC_clear_bl(word*);
GC_INNER void GC_default_print_heap_obj_proc(ptr_t p)
{
ptr_t base=(ptr_t)GC_base(p);
int kind=HDR(base)->hb_obj_kind;
GC_err_printf("object at %p of appr. %lu bytes (%s)\n",
(void*)base,(unsigned long)GC_size(base),
kind==PTRFREE?"atomic":
IS_UNCOLLECTABLE(kind)?"uncollectable":"composite");
}
GC_INNER void (*GC_print_heap_obj)(ptr_t p)=GC_default_print_heap_obj_proc;
#ifdef PRINT_BLACK_LIST
STATIC void GC_print_blacklisted_ptr(word p,ptr_t source,
const char*kind_str)
{
ptr_t base=(ptr_t)GC_base(source);
if (0==base){
GC_err_printf("Black listing (%s)%p referenced from %p in %s\n",
kind_str,(void*)p,(void*)source,
NULL!=source?"root set":"register");
} else {
GC_err_printf("Black listing (%s)%p referenced from %p in"
" object at %p of appr. %lu bytes\n",
kind_str,(void*)p,(void*)source,
(void*)base,(unsigned long)GC_size(base));
}
}
#endif
GC_INNER void GC_bl_init_no_interiors(void)
{
if (GC_incomplete_normal_bl==0){
GC_old_normal_bl=(word*)GC_scratch_alloc(sizeof(page_hash_table));
GC_incomplete_normal_bl=(word*)GC_scratch_alloc(
sizeof(page_hash_table));
if (GC_old_normal_bl==0||GC_incomplete_normal_bl==0){
GC_err_printf("Insufficient memory for black list\n");
EXIT();
}
GC_clear_bl(GC_old_normal_bl);
GC_clear_bl(GC_incomplete_normal_bl);
}
}
GC_INNER void GC_bl_init(void)
{
if (!GC_all_interior_pointers){
GC_bl_init_no_interiors();
}
GC_ASSERT(NULL==GC_old_stack_bl&&NULL==GC_incomplete_stack_bl);
GC_old_stack_bl=(word*)GC_scratch_alloc(sizeof(page_hash_table));
GC_incomplete_stack_bl=(word*)GC_scratch_alloc(sizeof(page_hash_table));
if (GC_old_stack_bl==0||GC_incomplete_stack_bl==0){
GC_err_printf("Insufficient memory for black list\n");
EXIT();
}
GC_clear_bl(GC_old_stack_bl);
GC_clear_bl(GC_incomplete_stack_bl);
}
STATIC void GC_clear_bl(word*doomed)
{
BZERO(doomed,sizeof(page_hash_table));
}
STATIC void GC_copy_bl(word*old,word*dest)
{
BCOPY(old,dest,sizeof(page_hash_table));
}
static word total_stack_black_listed(void);
GC_INNER void GC_promote_black_lists(void)
{
word*very_old_normal_bl=GC_old_normal_bl;
word*very_old_stack_bl=GC_old_stack_bl;
GC_old_normal_bl=GC_incomplete_normal_bl;
GC_old_stack_bl=GC_incomplete_stack_bl;
if (!GC_all_interior_pointers){
GC_clear_bl(very_old_normal_bl);
}
GC_clear_bl(very_old_stack_bl);
GC_incomplete_normal_bl=very_old_normal_bl;
GC_incomplete_stack_bl=very_old_stack_bl;
GC_total_stack_black_listed=total_stack_black_listed();
GC_VERBOSE_LOG_PRINTF(
"%lu bytes in heap blacklisted for interior pointers\n",
(unsigned long)GC_total_stack_black_listed);
if (GC_total_stack_black_listed!=0){
GC_black_list_spacing=
HBLKSIZE*(GC_heapsize/GC_total_stack_black_listed);
}
if (GC_black_list_spacing < 3*HBLKSIZE){
GC_black_list_spacing=3*HBLKSIZE;
}
if (GC_black_list_spacing > MAXHINCR*HBLKSIZE){
GC_black_list_spacing=MAXHINCR*HBLKSIZE;
}
}
GC_INNER void GC_unpromote_black_lists(void)
{
if (!GC_all_interior_pointers){
GC_copy_bl(GC_old_normal_bl,GC_incomplete_normal_bl);
}
GC_copy_bl(GC_old_stack_bl,GC_incomplete_stack_bl);
}
#if defined(PARALLEL_MARK)&&defined(THREAD_SANITIZER)
#define backlist_set_pht_entry_from_index(db,index)set_pht_entry_from_index_concurrent(db,index)
#else
#define backlist_set_pht_entry_from_index(bl,index)set_pht_entry_from_index(bl,index)
#endif
#ifdef PRINT_BLACK_LIST
GC_INNER void GC_add_to_black_list_normal(word p,ptr_t source)
#else
GC_INNER void GC_add_to_black_list_normal(word p)
#endif
{
if (GC_modws_valid_offsets[p&(sizeof(word)-1)]){
word index=PHT_HASH((word)p);
if (HDR(p)==0||get_pht_entry_from_index(GC_old_normal_bl,index)){
#ifdef PRINT_BLACK_LIST
if (!get_pht_entry_from_index(GC_incomplete_normal_bl,index)){
GC_print_blacklisted_ptr(p,source,"normal");
}
#endif
backlist_set_pht_entry_from_index(GC_incomplete_normal_bl,index);
}
}
}
#ifdef PRINT_BLACK_LIST
GC_INNER void GC_add_to_black_list_stack(word p,ptr_t source)
#else
GC_INNER void GC_add_to_black_list_stack(word p)
#endif
{
word index=PHT_HASH((word)p);
if (HDR(p)==0||get_pht_entry_from_index(GC_old_stack_bl,index)){
#ifdef PRINT_BLACK_LIST
if (!get_pht_entry_from_index(GC_incomplete_stack_bl,index)){
GC_print_blacklisted_ptr(p,source,"stack");
}
#endif
backlist_set_pht_entry_from_index(GC_incomplete_stack_bl,index);
}
}
struct hblk*GC_is_black_listed(struct hblk*h,word len)
{
word index=PHT_HASH((word)h);
word i;
word nblocks;
if (!GC_all_interior_pointers
&&(get_pht_entry_from_index(GC_old_normal_bl,index)
||get_pht_entry_from_index(GC_incomplete_normal_bl,index))){
return (h+1);
}
nblocks=divHBLKSZ(len);
for (i=0;;){
if (GC_old_stack_bl[divWORDSZ(index)]==0
&&GC_incomplete_stack_bl[divWORDSZ(index)]==0){
i+=WORDSZ - modWORDSZ(index);
} else {
if (get_pht_entry_from_index(GC_old_stack_bl,index)
||get_pht_entry_from_index(GC_incomplete_stack_bl,index)){
return(h+i+1);
}
i++;
}
if (i>=nblocks)break;
index=PHT_HASH((word)(h+i));
}
return(0);
}
STATIC word GC_number_stack_black_listed(struct hblk*start,
struct hblk*endp1)
{
struct hblk*h;
word result=0;
for (h=start;(word)h < (word)endp1;h++){
word index=PHT_HASH((word)h);
if (get_pht_entry_from_index(GC_old_stack_bl,index))result++;
}
return(result);
}
static word total_stack_black_listed(void)
{
unsigned i;
word total=0;
for (i=0;i < GC_n_heap_sects;i++){
struct hblk*start=(struct hblk*)GC_heap_sects[i].hs_start;
struct hblk*endp1=start+divHBLKSZ(GC_heap_sects[i].hs_bytes);
total+=GC_number_stack_black_listed(start,endp1);
}
return(total*HBLKSIZE);
}
#ifdef CHECKSUMS
#define NSUMS 10000
#define OFFSET 0x10000
typedef struct {
GC_bool new_valid;
word old_sum;
word new_sum;
struct hblk*block;
} page_entry;
page_entry GC_sums[NSUMS];
STATIC word GC_faulted[NSUMS]={ 0 };
STATIC size_t GC_n_faulted=0;
#if defined(MPROTECT_VDB)&&!defined(DARWIN)
void GC_record_fault(struct hblk*h)
{
word page=(word)h&~(GC_page_size - 1);
GC_ASSERT(GC_page_size!=0);
if (GC_n_faulted>=NSUMS)ABORT("write fault log overflowed");
GC_faulted[GC_n_faulted++]=page;
}
#endif
STATIC GC_bool GC_was_faulted(struct hblk*h)
{
size_t i;
word page=(word)h&~(GC_page_size - 1);
for (i=0;i < GC_n_faulted;++i){
if (GC_faulted[i]==page)return TRUE;
}
return FALSE;
}
STATIC word GC_checksum(struct hblk*h)
{
word*p=(word*)h;
word*lim=(word*)(h+1);
word result=0;
while ((word)p < (word)lim){
result+=*p++;
}
return(result|0x80000000);
}
int GC_n_dirty_errors=0;
int GC_n_faulted_dirty_errors=0;
unsigned long GC_n_clean=0;
unsigned long GC_n_dirty=0;
STATIC void GC_update_check_page(struct hblk*h,int index)
{
page_entry*pe=GC_sums+index;
hdr*hhdr=HDR(h);
struct hblk*b;
if (pe->block!=0&&pe->block!=h+OFFSET)ABORT("goofed");
pe->old_sum=pe->new_sum;
pe->new_sum=GC_checksum(h);
#if!defined(MSWIN32)&&!defined(MSWINCE)
if (pe->new_sum!=0x80000000&&!GC_page_was_ever_dirty(h)){
GC_err_printf("GC_page_was_ever_dirty(%p)is wrong\n",(void*)h);
}
#endif
if (GC_page_was_dirty(h)){
GC_n_dirty++;
} else {
GC_n_clean++;
}
b=h;
while (IS_FORWARDING_ADDR_OR_NIL(hhdr)&&hhdr!=0){
b-=(word)hhdr;
hhdr=HDR(b);
}
if (pe->new_valid
&&hhdr!=0&&hhdr->hb_descr!=0
&&pe->old_sum!=pe->new_sum){
if (!GC_page_was_dirty(h)||!GC_page_was_ever_dirty(h)){
GC_bool was_faulted=GC_was_faulted(h);
GC_n_dirty_errors++;
if (was_faulted)GC_n_faulted_dirty_errors++;
}
}
pe->new_valid=TRUE;
pe->block=h+OFFSET;
}
word GC_bytes_in_used_blocks=0;
STATIC void GC_add_block(struct hblk*h,word dummy GC_ATTR_UNUSED)
{
hdr*hhdr=HDR(h);
GC_bytes_in_used_blocks+=(hhdr->hb_sz+HBLKSIZE-1)&~(HBLKSIZE-1);
}
STATIC void GC_check_blocks(void)
{
word bytes_in_free_blocks=GC_large_free_bytes;
GC_bytes_in_used_blocks=0;
GC_apply_to_all_blocks(GC_add_block,(word)0);
GC_COND_LOG_PRINTF("GC_bytes_in_used_blocks=%lu,"
" bytes_in_free_blocks=%lu,heapsize=%lu\n",
(unsigned long)GC_bytes_in_used_blocks,
(unsigned long)bytes_in_free_blocks,
(unsigned long)GC_heapsize);
if (GC_bytes_in_used_blocks+bytes_in_free_blocks!=GC_heapsize){
GC_err_printf("LOST SOME BLOCKS!!\n");
}
}
void GC_check_dirty(void)
{
int index;
unsigned i;
struct hblk*h;
ptr_t start;
GC_check_blocks();
GC_n_dirty_errors=0;
GC_n_faulted_dirty_errors=0;
GC_n_clean=0;
GC_n_dirty=0;
index=0;
for (i=0;i < GC_n_heap_sects;i++){
start=GC_heap_sects[i].hs_start;
for (h=(struct hblk*)start;
(word)h < (word)(start+GC_heap_sects[i].hs_bytes);h++){
GC_update_check_page(h,index);
index++;
if (index>=NSUMS)goto out;
}
}
out:
GC_COND_LOG_PRINTF("Checked %lu clean and %lu dirty pages\n",
GC_n_clean,GC_n_dirty);
if (GC_n_dirty_errors > 0){
GC_err_printf("Found %d dirty bit errors (%d were faulted)\n",
GC_n_dirty_errors,GC_n_faulted_dirty_errors);
}
for (i=0;i < GC_n_faulted;++i){
GC_faulted[i]=0;
}
GC_n_faulted=0;
}
#endif
#ifndef GC_PMARK_H
#define GC_PMARK_H
#if defined(HAVE_CONFIG_H)&&!defined(GC_PRIVATE_H)
#endif
#ifndef GC_BUILD
#define GC_BUILD
#endif
#if (defined(__linux__)||defined(__GLIBC__)||defined(__GNU__))&&!defined(_GNU_SOURCE)&&defined(GC_PTHREADS)&&!defined(GC_NO_PTHREAD_SIGMASK)
#define _GNU_SOURCE 1
#endif
#if defined(KEEP_BACK_PTRS)||defined(PRINT_BLACK_LIST)
#endif
EXTERN_C_BEGIN
#ifndef MARK_DESCR_OFFSET
#define MARK_DESCR_OFFSET sizeof(word)
#endif
#define BITMAP_BITS (WORDSZ - GC_DS_TAG_BITS)
#define PROC(descr)(GC_mark_procs[((descr)>>GC_DS_TAG_BITS)&(GC_MAX_MARK_PROCS-1)])
#define ENV(descr)((descr)>>(GC_DS_TAG_BITS+GC_LOG_MAX_MARK_PROCS))
#define MAX_ENV (((word)1<<(WORDSZ - GC_DS_TAG_BITS - GC_LOG_MAX_MARK_PROCS))- 1)
GC_EXTERN unsigned GC_n_mark_procs;
#define GC_MARK_STACK_DISCARDS (INITIAL_MARK_STACK_SIZE/8)
#ifdef PARALLEL_MARK
#endif
GC_INNER mse*GC_signal_mark_stack_overflow(mse*msp);
GC_INLINE mse*GC_push_obj(ptr_t obj,hdr*hhdr,mse*mark_stack_top,
mse*mark_stack_limit)
{
word descr=hhdr->hb_descr;
GC_ASSERT(!HBLK_IS_FREE(hhdr));
if (descr!=0){
mark_stack_top++;
if ((word)mark_stack_top>=(word)mark_stack_limit){
mark_stack_top=GC_signal_mark_stack_overflow(mark_stack_top);
}
mark_stack_top->mse_start=obj;
mark_stack_top->mse_descr.w=descr;
}
return mark_stack_top;
}
#define PUSH_CONTENTS(current,mark_stack_top,mark_stack_limit,source)do { hdr*my_hhdr;HC_GET_HDR(current,my_hhdr,source);mark_stack_top=GC_push_contents_hdr(current,mark_stack_top,mark_stack_limit,source,my_hhdr,TRUE);} while (0)
#ifdef USE_MARK_BYTES
#if defined(PARALLEL_MARK)&&defined(AO_HAVE_char_store)&&!defined(BASE_ATOMIC_OPS_EMULATED)
#define SET_MARK_BIT_EXIT_IF_SET(hhdr,bit_no){ volatile unsigned char*mark_byte_addr=(unsigned char*)(hhdr)->hb_marks+(bit_no);if (AO_char_load(mark_byte_addr)!=0)break;AO_char_store(mark_byte_addr,1);}
#else
#define SET_MARK_BIT_EXIT_IF_SET(hhdr,bit_no){ char*mark_byte_addr=(char*)(hhdr)->hb_marks+(bit_no);if (*mark_byte_addr!=0)break;*mark_byte_addr=1;}
#endif
#else
#ifdef PARALLEL_MARK
#ifdef THREAD_SANITIZER
#define OR_WORD_EXIT_IF_SET(addr,bits){ if (!((word)AO_load((volatile AO_t*)(addr))&(bits))){ AO_or((volatile AO_t*)(addr),(AO_t)(bits));} else { break;} }
#else
#define OR_WORD_EXIT_IF_SET(addr,bits){ if (!(*(addr)&(bits))){ AO_or((volatile AO_t*)(addr),(AO_t)(bits));} else { break;} }
#endif
#else
#define OR_WORD_EXIT_IF_SET(addr,bits){ word old=*(addr);word my_bits=(bits);if ((old&my_bits)!=0)break;*(addr)=old|my_bits;}
#endif
#define SET_MARK_BIT_EXIT_IF_SET(hhdr,bit_no){ word*mark_word_addr=(hhdr)->hb_marks+divWORDSZ(bit_no);OR_WORD_EXIT_IF_SET(mark_word_addr,(word)1<<modWORDSZ(bit_no));}
#endif
#ifdef PARALLEL_MARK
#define INCR_MARKS(hhdr)AO_store(&hhdr->hb_n_marks,AO_load(&hhdr->hb_n_marks)+1)
#else
#define INCR_MARKS(hhdr)(void)(++hhdr->hb_n_marks)
#endif
#ifdef ENABLE_TRACE
#define TRACE(source,cmd)if (GC_trace_addr!=0&&(ptr_t)(source)==GC_trace_addr)cmd
#define TRACE_TARGET(target,cmd)if (GC_trace_addr!=0&&(target)==*(ptr_t*)GC_trace_addr)cmd
#else
#define TRACE(source,cmd)
#define TRACE_TARGET(source,cmd)
#endif
#if defined(I386)&&defined(__GNUC__)&&!defined(NACL)
#define LONG_MULT(hprod,lprod,x,y)do { __asm__ __volatile__("mull %2":"=a"(lprod),"=d"(hprod):"g"(y),"0"(x));} while (0)
#else
#if defined(__int64)&&!defined(__GNUC__)&&!defined(CPPCHECK)
#define ULONG_MULT_T unsigned __int64
#else
#define ULONG_MULT_T unsigned long long
#endif
#define LONG_MULT(hprod,lprod,x,y)do { ULONG_MULT_T prod=(ULONG_MULT_T)(x)*(ULONG_MULT_T)(y);GC_STATIC_ASSERT(sizeof(x)+sizeof(y)<=sizeof(prod));hprod=prod>>32;lprod=(unsigned32)prod;} while (0)
#endif
GC_INLINE mse*GC_push_contents_hdr(ptr_t current,mse*mark_stack_top,
mse*mark_stack_limit,ptr_t source,
hdr*hhdr,GC_bool do_offset_check)
{
do {
size_t displ=HBLKDISPL(current);
ptr_t base=current;
#ifdef MARK_BIT_PER_GRANULE
size_t gran_displ=BYTES_TO_GRANULES(displ);
size_t gran_offset=hhdr->hb_map[gran_displ];
size_t byte_offset=displ&(GRANULE_BYTES - 1);
if (EXPECT((gran_offset|byte_offset)!=0,FALSE))
#else
unsigned32 gran_displ;
unsigned32 inv_sz=hhdr->hb_inv_sz;
#endif
{
#ifdef MARK_BIT_PER_GRANULE
if ((hhdr->hb_flags&LARGE_BLOCK)!=0)
#else
if (EXPECT(inv_sz==LARGE_INV_SZ,FALSE))
#endif
{
size_t obj_displ;
base=(ptr_t)hhdr->hb_block;
obj_displ=current - base;
if (obj_displ!=displ){
GC_ASSERT(obj_displ < hhdr->hb_sz);
} else if (do_offset_check&&!GC_valid_offsets[obj_displ]){
GC_ADD_TO_BLACK_LIST_NORMAL(current,source);
break;
}
GC_ASSERT(hhdr->hb_sz > HBLKSIZE
||hhdr->hb_block==HBLKPTR(current));
GC_ASSERT((word)hhdr->hb_block<=(word)current);
gran_displ=0;
} else {
#ifdef MARK_BIT_PER_GRANULE
size_t obj_displ=GRANULES_TO_BYTES(gran_offset)+byte_offset;
#else
unsigned32 low_prod;
LONG_MULT(gran_displ,low_prod,(unsigned32)displ,inv_sz);
if ((low_prod>>16)!=0)
#endif
{
#if defined(MARK_BIT_PER_OBJ)&&!defined(MARK_BIT_PER_GRANULE)
size_t obj_displ;
GC_STATIC_ASSERT(HBLKSIZE<=(1<<15));
obj_displ=(((low_prod>>16)+1)*(size_t)hhdr->hb_sz)>>16;
#endif
if (do_offset_check&&!GC_valid_offsets[obj_displ]){
GC_ADD_TO_BLACK_LIST_NORMAL(current,source);
break;
}
#ifdef MARK_BIT_PER_GRANULE
gran_displ-=gran_offset;
#endif
base-=obj_displ;
}
}
}
#ifdef MARK_BIT_PER_GRANULE
GC_ASSERT(hhdr==GC_find_header(base));
GC_ASSERT(gran_displ % BYTES_TO_GRANULES(hhdr->hb_sz)==0);
#else
GC_ASSERT(gran_displ<=HBLK_OBJS(hhdr->hb_sz));
#endif
TRACE(source,GC_log_printf("GC #%u:passed validity tests\n",
(unsigned)GC_gc_no));
SET_MARK_BIT_EXIT_IF_SET(hhdr,gran_displ);
TRACE(source,GC_log_printf("GC #%u:previously unmarked\n",
(unsigned)GC_gc_no));
TRACE_TARGET(base,GC_log_printf("GC #%u:marking %p from %p instead\n",
(unsigned)GC_gc_no,(void*)base,
(void*)source));
INCR_MARKS(hhdr);
GC_STORE_BACK_PTR(source,base);
mark_stack_top=GC_push_obj(base,hhdr,mark_stack_top,
mark_stack_limit);
} while (0);
return mark_stack_top;
}
#if defined(PRINT_BLACK_LIST)||defined(KEEP_BACK_PTRS)
#define PUSH_ONE_CHECKED_STACK(p,source)GC_mark_and_push_stack((ptr_t)(p),(ptr_t)(source))
#else
#define PUSH_ONE_CHECKED_STACK(p,source)GC_mark_and_push_stack((ptr_t)(p))
#endif
#ifdef NEED_FIXUP_POINTER
#define GC_PUSH_ONE_STACK(p,source)do { if ((word)(p)>=(word)GC_least_plausible_heap_addr&&(word)(p)< (word)GC_greatest_plausible_heap_addr){ PUSH_ONE_CHECKED_STACK(p,source);} FIXUP_POINTER(p);if ((word)(p)>=(word)GC_least_plausible_heap_addr&&(word)(p)< (word)GC_greatest_plausible_heap_addr){ PUSH_ONE_CHECKED_STACK(p,source);} } while (0)
#else
#define GC_PUSH_ONE_STACK(p,source)do { if ((word)(p)>=(word)GC_least_plausible_heap_addr&&(word)(p)< (word)GC_greatest_plausible_heap_addr){ PUSH_ONE_CHECKED_STACK(p,source);} } while (0)
#endif
#define GC_PUSH_ONE_HEAP(p,source,mark_stack_top)do { FIXUP_POINTER(p);if ((word)(p)>=(word)GC_least_plausible_heap_addr&&(word)(p)< (word)GC_greatest_plausible_heap_addr)mark_stack_top=GC_mark_and_push((void*)(p),mark_stack_top,GC_mark_stack_limit,(void**)(source));} while (0)
GC_INNER mse*GC_mark_from(mse*top,mse*bottom,mse*limit);
#define MARK_FROM_MARK_STACK()GC_mark_stack_top=GC_mark_from(GC_mark_stack_top,GC_mark_stack,GC_mark_stack+GC_mark_stack_size);
#define GC_mark_stack_empty()((word)GC_mark_stack_top < (word)GC_mark_stack)
#define GC_MARK_FO(real_ptr,mark_proc)do { (*(mark_proc))(real_ptr);while (!GC_mark_stack_empty())MARK_FROM_MARK_STACK();if (GC_mark_state!=MS_NONE){ GC_set_mark_bit(real_ptr);while (!GC_mark_some((ptr_t)0)){ } } } while (0)
#define MS_NONE 0
#define MS_PUSH_RESCUERS 1
#define MS_PUSH_UNCOLLECTABLE 2
#define MS_ROOTS_PUSHED 3
#define MS_PARTIALLY_INVALID 4
#define MS_INVALID 5
EXTERN_C_END
#endif
#ifdef GC_GCJ_SUPPORT
#ifndef GC_GCJ_H
#define GC_GCJ_H
#ifndef GC_H
#endif
#ifdef __cplusplus
extern "C" {
#endif
GC_API void GC_CALL GC_init_gcj_malloc(int,
void*);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_gcj_malloc(size_t,
void*);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_debug_gcj_malloc(size_t,
void*,
GC_EXTRA_PARAMS);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_gcj_malloc_ignore_off_page(size_t,
void*);
GC_API int GC_gcj_kind;
GC_API int GC_gcj_debug_kind;
#ifdef GC_DEBUG
#define GC_GCJ_MALLOC(s,d)GC_debug_gcj_malloc(s,d,GC_EXTRAS)
#define GC_GCJ_MALLOC_IGNORE_OFF_PAGE(s,d)GC_debug_gcj_malloc(s,d,GC_EXTRAS)
#else
#define GC_GCJ_MALLOC(s,d)GC_gcj_malloc(s,d)
#define GC_GCJ_MALLOC_IGNORE_OFF_PAGE(s,d)GC_gcj_malloc_ignore_off_page(s,d)
#endif
#ifdef __cplusplus
}
#endif
#endif
int GC_gcj_kind=0;
int GC_gcj_debug_kind=0;
STATIC struct GC_ms_entry*GC_gcj_fake_mark_proc(word*addr GC_ATTR_UNUSED,
struct GC_ms_entry*mark_stack_ptr,
struct GC_ms_entry*mark_stack_limit GC_ATTR_UNUSED,
word env GC_ATTR_UNUSED)
{
ABORT_RET("No client gcj mark proc is specified");
return mark_stack_ptr;
}
GC_API void GC_CALL GC_init_gcj_malloc(int mp_index,
void*mp)
{
#ifndef GC_IGNORE_GCJ_INFO
GC_bool ignore_gcj_info;
#endif
DCL_LOCK_STATE;
if (mp==0)
mp=(void*)(word)GC_gcj_fake_mark_proc;
GC_init();
LOCK();
if (GC_gcjobjfreelist!=NULL){
UNLOCK();
return;
}
#ifdef GC_IGNORE_GCJ_INFO
#define ignore_gcj_info TRUE
#else
ignore_gcj_info=(0!=GETENV("GC_IGNORE_GCJ_INFO"));
#endif
if (ignore_gcj_info){
GC_COND_LOG_PRINTF("Gcj-style type information is disabled!\n");
}
GC_ASSERT(GC_mark_procs[mp_index]==(GC_mark_proc)0);
GC_mark_procs[mp_index]=(GC_mark_proc)(word)mp;
if ((unsigned)mp_index>=GC_n_mark_procs)
ABORT("GC_init_gcj_malloc:bad index");
GC_gcjobjfreelist=(ptr_t*)GC_new_free_list_inner();
if (ignore_gcj_info){
GC_gcj_kind=GC_new_kind_inner((void**)GC_gcjobjfreelist,
GC_DS_LENGTH,
TRUE,TRUE);
GC_gcj_debug_kind=GC_gcj_kind;
} else {
GC_gcj_kind=GC_new_kind_inner(
(void**)GC_gcjobjfreelist,
(((word)(-(signed_word)MARK_DESCR_OFFSET
- GC_INDIR_PER_OBJ_BIAS))
|GC_DS_PER_OBJECT),
FALSE,TRUE);
GC_gcj_debug_kind=GC_new_kind_inner(GC_new_free_list_inner(),
GC_MAKE_PROC(mp_index,
1),
FALSE,TRUE);
}
UNLOCK();
#undef ignore_gcj_info
}
#define GENERAL_MALLOC_INNER(lb,k)GC_clear_stack(GC_generic_malloc_inner(lb,k))
#define GENERAL_MALLOC_INNER_IOP(lb,k)GC_clear_stack(GC_generic_malloc_inner_ignore_off_page(lb,k))
static void maybe_finalize(void)
{
static word last_finalized_no=0;
DCL_LOCK_STATE;
if (GC_gc_no==last_finalized_no||
!EXPECT(GC_is_initialized,TRUE))return;
UNLOCK();
GC_INVOKE_FINALIZERS();
LOCK();
last_finalized_no=GC_gc_no;
}
#ifdef THREAD_LOCAL_ALLOC
GC_INNER void*GC_core_gcj_malloc(size_t lb,
void*ptr_to_struct_containing_descr)
#else
GC_API GC_ATTR_MALLOC void*GC_CALL GC_gcj_malloc(size_t lb,
void*ptr_to_struct_containing_descr)
#endif
{
ptr_t op;
DCL_LOCK_STATE;
GC_DBG_COLLECT_AT_MALLOC(lb);
if(SMALL_OBJ(lb)){
word lg;
LOCK();
lg=GC_size_map[lb];
op=GC_gcjobjfreelist[lg];
if(EXPECT(0==op,FALSE)){
maybe_finalize();
op=(ptr_t)GENERAL_MALLOC_INNER((word)lb,GC_gcj_kind);
if (0==op){
GC_oom_func oom_fn=GC_oom_fn;
UNLOCK();
return((*oom_fn)(lb));
}
} else {
GC_gcjobjfreelist[lg]=(ptr_t)obj_link(op);
GC_bytes_allocd+=GRANULES_TO_BYTES((word)lg);
}
GC_ASSERT(((void**)op)[1]==0);
} else {
LOCK();
maybe_finalize();
op=(ptr_t)GENERAL_MALLOC_INNER((word)lb,GC_gcj_kind);
if (0==op){
GC_oom_func oom_fn=GC_oom_fn;
UNLOCK();
return((*oom_fn)(lb));
}
}
*(void**)op=ptr_to_struct_containing_descr;
UNLOCK();
GC_dirty(op);
REACHABLE_AFTER_DIRTY(ptr_to_struct_containing_descr);
return (void*)op;
}
GC_API GC_ATTR_MALLOC void*GC_CALL GC_debug_gcj_malloc(size_t lb,
void*ptr_to_struct_containing_descr,GC_EXTRA_PARAMS)
{
void*result;
DCL_LOCK_STATE;
LOCK();
maybe_finalize();
result=GC_generic_malloc_inner(SIZET_SAT_ADD(lb,DEBUG_BYTES),
GC_gcj_debug_kind);
if (result==0){
GC_oom_func oom_fn=GC_oom_fn;
UNLOCK();
GC_err_printf("GC_debug_gcj_malloc(%lu,%p)returning NULL (%s:%d)\n",
(unsigned long)lb,ptr_to_struct_containing_descr,s,i);
return((*oom_fn)(lb));
}
*((void**)((ptr_t)result+sizeof(oh)))=ptr_to_struct_containing_descr;
if (!GC_debugging_started){
GC_start_debugging_inner();
}
ADD_CALL_CHAIN(result,ra);
result=GC_store_debug_info_inner(result,(word)lb,s,i);
UNLOCK();
GC_dirty(result);
REACHABLE_AFTER_DIRTY(ptr_to_struct_containing_descr);
return result;
}
GC_API GC_ATTR_MALLOC void*GC_CALL GC_gcj_malloc_ignore_off_page(size_t lb,
void*ptr_to_struct_containing_descr)
{
ptr_t op;
DCL_LOCK_STATE;
GC_DBG_COLLECT_AT_MALLOC(lb);
if(SMALL_OBJ(lb)){
word lg;
LOCK();
lg=GC_size_map[lb];
op=GC_gcjobjfreelist[lg];
if (EXPECT(0==op,FALSE)){
maybe_finalize();
op=(ptr_t)GENERAL_MALLOC_INNER_IOP(lb,GC_gcj_kind);
if (0==op){
GC_oom_func oom_fn=GC_oom_fn;
UNLOCK();
return((*oom_fn)(lb));
}
} else {
GC_gcjobjfreelist[lg]=(ptr_t)obj_link(op);
GC_bytes_allocd+=GRANULES_TO_BYTES((word)lg);
}
} else {
LOCK();
maybe_finalize();
op=(ptr_t)GENERAL_MALLOC_INNER_IOP(lb,GC_gcj_kind);
if (0==op){
GC_oom_func oom_fn=GC_oom_fn;
UNLOCK();
return((*oom_fn)(lb));
}
}
*(void**)op=ptr_to_struct_containing_descr;
UNLOCK();
GC_dirty(op);
REACHABLE_AFTER_DIRTY(ptr_to_struct_containing_descr);
return (void*)op;
}
#endif
GC_INNER hdr*GC_find_header(ptr_t h)
{
#ifdef HASH_TL
hdr*result;
GET_HDR(h,result);
return(result);
#else
return(HDR_INNER(h));
#endif
}
GC_INNER hdr*
#ifdef PRINT_BLACK_LIST
GC_header_cache_miss(ptr_t p,hdr_cache_entry*hce,ptr_t source)
#else
GC_header_cache_miss(ptr_t p,hdr_cache_entry*hce)
#endif
{
hdr*hhdr;
HC_MISS();
GET_HDR(p,hhdr);
if (IS_FORWARDING_ADDR_OR_NIL(hhdr)){
if (GC_all_interior_pointers){
if (hhdr!=0){
ptr_t current=p;
current=(ptr_t)HBLKPTR(current);
do {
current=current - HBLKSIZE*(word)hhdr;
hhdr=HDR(current);
} while(IS_FORWARDING_ADDR_OR_NIL(hhdr));
if (hhdr->hb_flags&IGNORE_OFF_PAGE)
return 0;
if (HBLK_IS_FREE(hhdr)
||p - current>=(ptrdiff_t)(hhdr->hb_sz)){
GC_ADD_TO_BLACK_LIST_NORMAL(p,source);
return 0;
}
} else {
GC_ADD_TO_BLACK_LIST_NORMAL(p,source);
}
GC_ASSERT(hhdr==0||!HBLK_IS_FREE(hhdr));
return hhdr;
} else {
if (hhdr==0){
GC_ADD_TO_BLACK_LIST_NORMAL(p,source);
}
return 0;
}
} else {
if (HBLK_IS_FREE(hhdr)){
GC_ADD_TO_BLACK_LIST_NORMAL(p,source);
return 0;
} else {
hce->block_addr=(word)(p)>>LOG_HBLKSIZE;
hce->hce_hdr=hhdr;
return hhdr;
}
}
}
GC_INNER ptr_t GC_scratch_alloc(size_t bytes)
{
ptr_t result=GC_scratch_free_ptr;
size_t bytes_to_get;
bytes=ROUNDUP_GRANULE_SIZE(bytes);
for (;;){
GC_scratch_free_ptr+=bytes;
if ((word)GC_scratch_free_ptr<=(word)GC_scratch_end_ptr){
return result;
}
GC_ASSERT(GC_page_size!=0);
if (bytes>=MINHINCR*HBLKSIZE){
bytes_to_get=ROUNDUP_PAGESIZE_IF_MMAP(bytes);
result=(ptr_t)GET_MEM(bytes_to_get);
GC_add_to_our_memory(result,bytes_to_get);
GC_scratch_free_ptr-=bytes;
if (result!=NULL){
GC_scratch_last_end_ptr=result+bytes;
}
return result;
}
bytes_to_get=ROUNDUP_PAGESIZE_IF_MMAP(MINHINCR*HBLKSIZE);
result=(ptr_t)GET_MEM(bytes_to_get);
GC_add_to_our_memory(result,bytes_to_get);
if (NULL==result){
WARN("Out of memory - trying to allocate requested amount"
" (%" WARN_PRIdPTR " bytes)...\n",(word)bytes);
GC_scratch_free_ptr-=bytes;
bytes_to_get=ROUNDUP_PAGESIZE_IF_MMAP(bytes);
result=(ptr_t)GET_MEM(bytes_to_get);
GC_add_to_our_memory(result,bytes_to_get);
return result;
}
GC_scratch_free_ptr=result;
GC_scratch_end_ptr=GC_scratch_free_ptr+bytes_to_get;
GC_scratch_last_end_ptr=GC_scratch_end_ptr;
}
}
static hdr*alloc_hdr(void)
{
hdr*result;
if (NULL==GC_hdr_free_list){
result=(hdr*)GC_scratch_alloc(sizeof(hdr));
} else {
result=GC_hdr_free_list;
GC_hdr_free_list=(hdr*)result->hb_next;
}
return(result);
}
GC_INLINE void free_hdr(hdr*hhdr)
{
hhdr->hb_next=(struct hblk*)GC_hdr_free_list;
GC_hdr_free_list=hhdr;
}
#ifdef COUNT_HDR_CACHE_HITS
word GC_hdr_cache_hits=0;
word GC_hdr_cache_misses=0;
#endif
GC_INNER void GC_init_headers(void)
{
unsigned i;
GC_ASSERT(NULL==GC_all_nils);
GC_all_nils=(bottom_index*)GC_scratch_alloc(sizeof(bottom_index));
if (GC_all_nils==NULL){
GC_err_printf("Insufficient memory for GC_all_nils\n");
EXIT();
}
BZERO(GC_all_nils,sizeof(bottom_index));
for (i=0;i < TOP_SZ;i++){
GC_top_index[i]=GC_all_nils;
}
}
static GC_bool get_index(word addr)
{
word hi=(word)(addr)>>(LOG_BOTTOM_SZ+LOG_HBLKSIZE);
bottom_index*r;
bottom_index*p;
bottom_index**prev;
bottom_index*pi;
word i;
GC_ASSERT(I_HOLD_LOCK());
#ifdef HASH_TL
i=TL_HASH(hi);
pi=p=GC_top_index[i];
while(p!=GC_all_nils){
if (p->key==hi)return(TRUE);
p=p->hash_link;
}
#else
if (GC_top_index[hi]!=GC_all_nils)
return TRUE;
i=hi;
#endif
r=(bottom_index*)GC_scratch_alloc(sizeof(bottom_index));
if (EXPECT(NULL==r,FALSE))
return FALSE;
BZERO(r,sizeof(bottom_index));
r->key=hi;
#ifdef HASH_TL
r->hash_link=pi;
#endif
prev=&GC_all_bottom_indices;
pi=0;
while ((p=*prev)!=0&&p->key < hi){
pi=p;
prev=&(p->asc_link);
}
r->desc_link=pi;
if (0==p){
GC_all_bottom_indices_end=r;
} else {
p->desc_link=r;
}
r->asc_link=p;
*prev=r;
GC_top_index[i]=r;
return(TRUE);
}
GC_INNER struct hblkhdr*GC_install_header(struct hblk*h)
{
hdr*result;
if (!get_index((word)h))return(0);
result=alloc_hdr();
if (result){
SET_HDR(h,result);
#ifdef USE_MUNMAP
result->hb_last_reclaimed=(unsigned short)GC_gc_no;
#endif
}
return(result);
}
GC_INNER GC_bool GC_install_counts(struct hblk*h,size_t sz)
{
struct hblk*hbp;
for (hbp=h;(word)hbp < (word)h+sz;hbp+=BOTTOM_SZ){
if (!get_index((word)hbp))
return FALSE;
if ((word)hbp > GC_WORD_MAX - (word)BOTTOM_SZ*HBLKSIZE)
break;
}
if (!get_index((word)h+sz - 1))
return FALSE;
for (hbp=h+1;(word)hbp < (word)h+sz;hbp+=1){
word i=HBLK_PTR_DIFF(hbp,h);
SET_HDR(hbp,(hdr*)(i > MAX_JUMP?MAX_JUMP:i));
}
return TRUE;
}
GC_INNER void GC_remove_header(struct hblk*h)
{
hdr**ha;
GET_HDR_ADDR(h,ha);
free_hdr(*ha);
*ha=0;
}
GC_INNER void GC_remove_counts(struct hblk*h,size_t sz)
{
struct hblk*hbp;
for (hbp=h+1;(word)hbp < (word)h+sz;hbp+=1){
SET_HDR(hbp,0);
}
}
void GC_apply_to_all_blocks(void (*fn)(struct hblk*h,word client_data),
word client_data)
{
signed_word j;
bottom_index*index_p;
for (index_p=GC_all_bottom_indices;index_p!=0;
index_p=index_p->asc_link){
for (j=BOTTOM_SZ-1;j>=0;){
if (!IS_FORWARDING_ADDR_OR_NIL(index_p->index[j])){
if (!HBLK_IS_FREE(index_p->index[j])){
(*fn)(((struct hblk*)
(((index_p->key<<LOG_BOTTOM_SZ)+(word)j)
<<LOG_HBLKSIZE)),
client_data);
}
j--;
} else if (index_p->index[j]==0){
j--;
} else {
j-=(signed_word)(index_p->index[j]);
}
}
}
}
GC_INNER struct hblk*GC_next_block(struct hblk*h,GC_bool allow_free)
{
REGISTER bottom_index*bi;
REGISTER word j=((word)h>>LOG_HBLKSIZE)&(BOTTOM_SZ-1);
GC_ASSERT(I_HOLD_LOCK());
GET_BI(h,bi);
if (bi==GC_all_nils){
REGISTER word hi=(word)h>>(LOG_BOTTOM_SZ+LOG_HBLKSIZE);
bi=GC_all_bottom_indices;
while (bi!=0&&bi->key < hi)bi=bi->asc_link;
j=0;
}
while (bi!=0){
while (j < BOTTOM_SZ){
hdr*hhdr=bi->index[j];
if (IS_FORWARDING_ADDR_OR_NIL(hhdr)){
j++;
} else {
if (allow_free||!HBLK_IS_FREE(hhdr)){
return ((struct hblk*)
(((bi->key<<LOG_BOTTOM_SZ)+j)
<<LOG_HBLKSIZE));
} else {
j+=divHBLKSZ(hhdr->hb_sz);
}
}
}
j=0;
bi=bi->asc_link;
}
return(0);
}
GC_INNER struct hblk*GC_prev_block(struct hblk*h)
{
bottom_index*bi;
signed_word j=((word)h>>LOG_HBLKSIZE)&(BOTTOM_SZ-1);
GC_ASSERT(I_HOLD_LOCK());
GET_BI(h,bi);
if (bi==GC_all_nils){
word hi=(word)h>>(LOG_BOTTOM_SZ+LOG_HBLKSIZE);
bi=GC_all_bottom_indices_end;
while (bi!=0&&bi->key > hi)bi=bi->desc_link;
j=BOTTOM_SZ - 1;
}
while(bi!=0){
while (j>=0){
hdr*hhdr=bi->index[j];
if (0==hhdr){
--j;
} else if (IS_FORWARDING_ADDR_OR_NIL(hhdr)){
j-=(signed_word)hhdr;
} else {
return((struct hblk*)
(((bi->key<<LOG_BOTTOM_SZ)+j)
<<LOG_HBLKSIZE));
}
}
j=BOTTOM_SZ - 1;
bi=bi->desc_link;
}
return(0);
}
#include <stdio.h>
#ifndef SMALL_CONFIG
STATIC ptr_t GC_build_fl_clear2(struct hblk*h,ptr_t ofl)
{
word*p=(word*)(h->hb_body);
word*lim=(word*)(h+1);
p[0]=(word)ofl;
p[1]=0;
p[2]=(word)p;
p[3]=0;
p+=4;
for (;(word)p < (word)lim;p+=4){
p[0]=(word)(p-2);
p[1]=0;
p[2]=(word)p;
p[3]=0;
};
return((ptr_t)(p-2));
}
STATIC ptr_t GC_build_fl_clear4(struct hblk*h,ptr_t ofl)
{
word*p=(word*)(h->hb_body);
word*lim=(word*)(h+1);
p[0]=(word)ofl;
p[1]=0;
p[2]=0;
p[3]=0;
p+=4;
for (;(word)p < (word)lim;p+=4){
GC_PREFETCH_FOR_WRITE((ptr_t)(p+64));
p[0]=(word)(p-4);
p[1]=0;
CLEAR_DOUBLE(p+2);
};
return((ptr_t)(p-4));
}
STATIC ptr_t GC_build_fl2(struct hblk*h,ptr_t ofl)
{
word*p=(word*)(h->hb_body);
word*lim=(word*)(h+1);
p[0]=(word)ofl;
p[2]=(word)p;
p+=4;
for (;(word)p < (word)lim;p+=4){
p[0]=(word)(p-2);
p[2]=(word)p;
};
return((ptr_t)(p-2));
}
STATIC ptr_t GC_build_fl4(struct hblk*h,ptr_t ofl)
{
word*p=(word*)(h->hb_body);
word*lim=(word*)(h+1);
p[0]=(word)ofl;
p[4]=(word)p;
p+=8;
for (;(word)p < (word)lim;p+=8){
GC_PREFETCH_FOR_WRITE((ptr_t)(p+64));
p[0]=(word)(p-4);
p[4]=(word)p;
};
return((ptr_t)(p-4));
}
#endif
GC_INNER ptr_t GC_build_fl(struct hblk*h,size_t sz,GC_bool clear,
ptr_t list)
{
word*p,*prev;
word*last_object;
GC_PREFETCH_FOR_WRITE((ptr_t)h);
GC_PREFETCH_FOR_WRITE((ptr_t)h+128);
GC_PREFETCH_FOR_WRITE((ptr_t)h+256);
GC_PREFETCH_FOR_WRITE((ptr_t)h+378);
#ifndef SMALL_CONFIG
switch (sz){
case 2:if (clear){
return GC_build_fl_clear2(h,list);
} else {
return GC_build_fl2(h,list);
}
case 4:if (clear){
return GC_build_fl_clear4(h,list);
} else {
return GC_build_fl4(h,list);
}
default:
break;
}
#endif
if (clear)BZERO(h,HBLKSIZE);
p=(word*)(h->hb_body)+sz;
prev=(word*)(h->hb_body);
last_object=(word*)((char*)h+HBLKSIZE);
last_object-=sz;
while ((word)p<=(word)last_object){
obj_link(p)=(ptr_t)prev;
prev=p;
p+=sz;
}
p-=sz;
*(ptr_t*)h=list;
return ((ptr_t)p);
}
GC_INNER void GC_new_hblk(size_t gran,int kind)
{
struct hblk*h;
GC_bool clear=GC_obj_kinds[kind].ok_init;
GC_STATIC_ASSERT((sizeof (struct hblk))==HBLKSIZE);
if (GC_debugging_started)clear=TRUE;
h=GC_allochblk(GRANULES_TO_BYTES(gran),kind,0);
if (h==0)return;
if (IS_UNCOLLECTABLE(kind))GC_set_hdr_marks(HDR(h));
GC_obj_kinds[kind].ok_freelist[gran]=
GC_build_fl(h,GRANULES_TO_WORDS(gran),clear,
(ptr_t)GC_obj_kinds[kind].ok_freelist[gran]);
}
GC_API void GC_CALL GC_register_displacement(size_t offset)
{
DCL_LOCK_STATE;
LOCK();
GC_register_displacement_inner(offset);
UNLOCK();
}
GC_INNER void GC_register_displacement_inner(size_t offset)
{
GC_ASSERT(I_HOLD_LOCK());
if (offset>=VALID_OFFSET_SZ){
ABORT("Bad argument to GC_register_displacement");
}
if (!GC_valid_offsets[offset]){
GC_valid_offsets[offset]=TRUE;
GC_modws_valid_offsets[offset % sizeof(word)]=TRUE;
}
}
#ifdef MARK_BIT_PER_GRANULE
GC_INNER GC_bool GC_add_map_entry(size_t granules)
{
unsigned displ;
unsigned short*new_map;
if (granules > BYTES_TO_GRANULES(MAXOBJBYTES))granules=0;
if (GC_obj_map[granules]!=0){
return(TRUE);
}
new_map=(unsigned short*)GC_scratch_alloc(MAP_LEN*sizeof(short));
if (new_map==0)return(FALSE);
GC_COND_LOG_PRINTF(
"Adding block map for size of %u granules (%u bytes)\n",
(unsigned)granules,(unsigned)GRANULES_TO_BYTES(granules));
if (granules==0){
for (displ=0;displ < BYTES_TO_GRANULES(HBLKSIZE);displ++){
new_map[displ]=1;
}
} else {
for (displ=0;displ < BYTES_TO_GRANULES(HBLKSIZE);displ++){
new_map[displ]=(unsigned short)(displ % granules);
}
}
GC_obj_map[granules]=new_map;
return(TRUE);
}
#endif
GC_INNER void GC_initialize_offsets(void)
{
unsigned i;
if (GC_all_interior_pointers){
for (i=0;i < VALID_OFFSET_SZ;++i)
GC_valid_offsets[i]=TRUE;
} else {
BZERO(GC_valid_offsets,sizeof(GC_valid_offsets));
for (i=0;i < sizeof(word);++i)
GC_modws_valid_offsets[i]=FALSE;
}
}
STATIC void GC_CALLBACK GC_default_same_obj_print_proc(void*p,void*q)
{
ABORT_ARG2("GC_same_obj test failed",
":%p and %p are not in the same object",p,q);
}
void (GC_CALLBACK*GC_same_obj_print_proc)(void*,void*)
=GC_default_same_obj_print_proc;
GC_API void*GC_CALL GC_same_obj(void*p,void*q)
{
struct hblk*h;
hdr*hhdr;
ptr_t base,limit;
word sz;
if (!EXPECT(GC_is_initialized,TRUE))GC_init();
hhdr=HDR((word)p);
if (hhdr==0){
if (divHBLKSZ((word)p)!=divHBLKSZ((word)q)
&&HDR((word)q)!=0){
goto fail;
}
return(p);
}
if (IS_FORWARDING_ADDR_OR_NIL(hhdr)){
h=HBLKPTR(p)- (word)hhdr;
hhdr=HDR(h);
while (IS_FORWARDING_ADDR_OR_NIL(hhdr)){
h=FORWARDED_ADDR(h,hhdr);
hhdr=HDR(h);
}
limit=(ptr_t)h+hhdr->hb_sz;
if ((word)p>=(word)limit||(word)q>=(word)limit
||(word)q < (word)h){
goto fail;
}
return(p);
}
sz=hhdr->hb_sz;
if (sz > MAXOBJBYTES){
base=(ptr_t)HBLKPTR(p);
limit=base+sz;
if ((word)p>=(word)limit){
goto fail;
}
} else {
size_t offset;
size_t pdispl=HBLKDISPL(p);
offset=pdispl % sz;
if (HBLKPTR(p)!=HBLKPTR(q))goto fail;
base=(ptr_t)p - offset;
limit=base+sz;
}
if ((word)q>=(word)limit||(word)q < (word)base){
goto fail;
}
return(p);
fail:
(*GC_same_obj_print_proc)((ptr_t)p,(ptr_t)q);
return(p);
}
STATIC void GC_CALLBACK GC_default_is_valid_displacement_print_proc (void*p)
{
ABORT_ARG1("GC_is_valid_displacement test failed",":%p not valid",p);
}
void (GC_CALLBACK*GC_is_valid_displacement_print_proc)(void*)=
GC_default_is_valid_displacement_print_proc;
GC_API void*GC_CALL GC_is_valid_displacement(void*p)
{
hdr*hhdr;
word pdispl;
word offset;
struct hblk*h;
word sz;
if (!EXPECT(GC_is_initialized,TRUE))GC_init();
hhdr=HDR((word)p);
if (hhdr==0)return(p);
h=HBLKPTR(p);
if (GC_all_interior_pointers){
while (IS_FORWARDING_ADDR_OR_NIL(hhdr)){
h=FORWARDED_ADDR(h,hhdr);
hhdr=HDR(h);
}
} else if (IS_FORWARDING_ADDR_OR_NIL(hhdr)){
goto fail;
}
sz=hhdr->hb_sz;
pdispl=HBLKDISPL(p);
offset=pdispl % sz;
if ((sz > MAXOBJBYTES&&(word)p>=(word)h+sz)
||!GC_valid_offsets[offset]
||((word)p+(sz - offset)> (word)(h+1)
&&!IS_FORWARDING_ADDR_OR_NIL(HDR(h+1)))){
goto fail;
}
return(p);
fail:
(*GC_is_valid_displacement_print_proc)((ptr_t)p);
return(p);
}
STATIC void GC_CALLBACK GC_default_is_visible_print_proc(void*p)
{
ABORT_ARG1("GC_is_visible test failed",":%p not GC-visible",p);
}
void (GC_CALLBACK*GC_is_visible_print_proc)(void*p)=
GC_default_is_visible_print_proc;
#ifndef THREADS
STATIC GC_bool GC_on_stack(void*p)
{
#ifdef STACK_GROWS_DOWN
if ((word)p>=(word)GC_approx_sp()
&&(word)p < (word)GC_stackbottom){
return(TRUE);
}
#else
if ((word)p<=(word)GC_approx_sp()
&&(word)p > (word)GC_stackbottom){
return(TRUE);
}
#endif
return(FALSE);
}
#endif
GC_API void*GC_CALL GC_is_visible(void*p)
{
hdr*hhdr;
if ((word)p&(ALIGNMENT - 1))goto fail;
if (!EXPECT(GC_is_initialized,TRUE))GC_init();
#ifdef THREADS
hhdr=HDR((word)p);
if (hhdr!=0&&GC_base(p)==0){
goto fail;
} else {
return(p);
}
#else
if (GC_on_stack(p))return(p);
hhdr=HDR((word)p);
if (hhdr==0){
if (GC_is_static_root(p))return(p);
#if defined(DYNAMIC_LOADING)||defined(MSWIN32)||defined(MSWINCE)||defined(CYGWIN32)||defined(PCR)
GC_register_dynamic_libraries();
if (GC_is_static_root(p))
return(p);
#endif
goto fail;
} else {
word descr;
ptr_t base=(ptr_t)GC_base(p);
if (NULL==base)goto fail;
if (HBLKPTR(base)!=HBLKPTR(p))
hhdr=HDR(base);
descr=hhdr->hb_descr;
retry:
switch(descr&GC_DS_TAGS){
case GC_DS_LENGTH:
if ((word)p - (word)base > descr)goto fail;
break;
case GC_DS_BITMAP:
if ((word)p - (word)base>=WORDS_TO_BYTES(BITMAP_BITS)
||((word)p&(sizeof(word)- 1)))goto fail;
if (!(((word)1<<(WORDSZ - ((ptr_t)p - (ptr_t)base)- 1))
&descr))goto fail;
break;
case GC_DS_PROC:
break;
case GC_DS_PER_OBJECT:
if ((signed_word)descr>=0){
descr=*(word*)((ptr_t)base+(descr&~GC_DS_TAGS));
} else {
ptr_t type_descr=*(ptr_t*)base;
descr=*(word*)(type_descr
- (descr - (word)(GC_DS_PER_OBJECT
- GC_INDIR_PER_OBJ_BIAS)));
}
goto retry;
}
return(p);
}
#endif
fail:
(*GC_is_visible_print_proc)((ptr_t)p);
return(p);
}
GC_API void*GC_CALL GC_pre_incr (void**p,ptrdiff_t how_much)
{
void*initial=*p;
void*result=GC_same_obj((void*)((ptr_t)initial+how_much),initial);
if (!GC_all_interior_pointers){
(void)GC_is_valid_displacement(result);
}
return (*p=result);
}
GC_API void*GC_CALL GC_post_incr (void**p,ptrdiff_t how_much)
{
void*initial=*p;
void*result=GC_same_obj((void*)((ptr_t)initial+how_much),initial);
if (!GC_all_interior_pointers){
(void)GC_is_valid_displacement(result);
}
*p=result;
return(initial);
}
#ifndef GC_INLINE_H
#define GC_INLINE_H
#if GC_GNUC_PREREQ(3,0)
#define GC_EXPECT(expr,outcome)__builtin_expect(expr,outcome)
#else
#define GC_EXPECT(expr,outcome)(expr)
#endif
#ifndef GC_ASSERT
#ifdef NDEBUG
#define GC_ASSERT(expr)
#else
#include <assert.h>
#define GC_ASSERT(expr)assert(expr)
#endif
#endif
#ifdef __cplusplus
extern "C" {
#endif
#ifndef GC_PREFETCH_FOR_WRITE
#if GC_GNUC_PREREQ(3,0)&&!defined(GC_NO_PREFETCH_FOR_WRITE)
#define GC_PREFETCH_FOR_WRITE(x)__builtin_prefetch((x),1)
#else
#define GC_PREFETCH_FOR_WRITE(x)(void)0
#endif
#endif
#define GC_I_PTRFREE 0
#define GC_I_NORMAL 1
GC_API void GC_CALL GC_generic_malloc_many(size_t,int,
void**);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_malloc_kind(size_t,int);
#ifdef GC_THREADS
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_malloc_kind_global(size_t,int);
#else
#define GC_malloc_kind_global GC_malloc_kind
#endif
#if defined(GC_THREADS)&&defined(AO_HAVE_store)
#define GC_FAST_M_AO_STORE(my_fl,next)AO_store((volatile AO_t*)(my_fl),(AO_t)(next))
#else
#define GC_FAST_M_AO_STORE(my_fl,next)(void)(*(my_fl)=(next))
#endif
#define GC_FAST_MALLOC_GRANS(result,granules,tiny_fl,num_direct,kind,default_expr,init)do { if (GC_EXPECT((granules)>=GC_TINY_FREELISTS,0)){ result=(default_expr);} else { void**my_fl=(tiny_fl)+(granules);void*my_entry=*my_fl;void*next;for (;;){ if (GC_EXPECT((GC_word)my_entry > (num_direct)+GC_TINY_FREELISTS+1,1)){ next=*(void**)(my_entry);result=(void*)my_entry;GC_FAST_M_AO_STORE(my_fl,next);init;GC_PREFETCH_FOR_WRITE(next);if ((kind)!=GC_I_PTRFREE){ GC_end_stubborn_change(my_fl);GC_reachable_here(next);} GC_ASSERT(GC_size(result)>=(granules)*GC_GRANULE_BYTES);GC_ASSERT((kind)==GC_I_PTRFREE||((GC_word*)result)[1]==0);break;} if ((GC_signed_word)my_entry - (GC_signed_word)(num_direct)<=0&&my_entry!=0){ GC_FAST_M_AO_STORE(my_fl,(char*)my_entry+(granules)+1);result=(default_expr);break;} else { GC_generic_malloc_many(((granules)==0?GC_GRANULE_BYTES:GC_RAW_BYTES_FROM_INDEX(granules)),kind,my_fl);my_entry=*my_fl;if (my_entry==0){ result=(*GC_get_oom_fn())((granules)*GC_GRANULE_BYTES);break;} } } } } while (0)
#define GC_WORDS_TO_WHOLE_GRANULES(n)GC_WORDS_TO_GRANULES((n)+GC_GRANULE_WORDS - 1)
#define GC_MALLOC_WORDS_KIND(result,n,tiny_fl,kind,init)do { size_t granules=GC_WORDS_TO_WHOLE_GRANULES(n);GC_FAST_MALLOC_GRANS(result,granules,tiny_fl,0,kind,GC_malloc_kind(granules*GC_GRANULE_BYTES,kind),init);} while (0)
#define GC_MALLOC_WORDS(result,n,tiny_fl)GC_MALLOC_WORDS_KIND(result,n,tiny_fl,GC_I_NORMAL,*(void**)(result)=0)
#define GC_MALLOC_ATOMIC_WORDS(result,n,tiny_fl)GC_MALLOC_WORDS_KIND(result,n,tiny_fl,GC_I_PTRFREE,(void)0)
#define GC_CONS(result,first,second,tiny_fl)do { void*l=(void*)(first);void*r=(void*)(second);GC_MALLOC_WORDS_KIND(result,2,tiny_fl,GC_I_NORMAL,(void)0);if ((result)!=0){*(void**)(result)=l;GC_PTR_STORE_AND_DIRTY((void**)(result)+1,r);GC_reachable_here(l);} } while (0)
GC_API void GC_CALL GC_print_free_list(int,
size_t);
#ifdef __cplusplus
}
#endif
#endif
#include <stdio.h>
#ifdef GC_USE_ENTIRE_HEAP
int GC_use_entire_heap=TRUE;
#else
int GC_use_entire_heap=FALSE;
#endif
#define MAX_BLACK_LIST_ALLOC (2*HBLKSIZE)
#define UNIQUE_THRESHOLD 32
#define HUGE_THRESHOLD 256
#define FL_COMPRESSION 8
#define N_HBLK_FLS ((HUGE_THRESHOLD - UNIQUE_THRESHOLD)/FL_COMPRESSION+UNIQUE_THRESHOLD)
#ifndef GC_GCJ_SUPPORT
STATIC
#endif
struct hblk*GC_hblkfreelist[N_HBLK_FLS+1]={ 0 };
#ifndef GC_GCJ_SUPPORT
STATIC
#endif
word GC_free_bytes[N_HBLK_FLS+1]={ 0 };
GC_INLINE int GC_enough_large_bytes_left(void)
{
int n;
word bytes=GC_large_allocd_bytes;
GC_ASSERT(GC_max_large_allocd_bytes<=GC_heapsize);
for (n=N_HBLK_FLS;n>=0;--n){
bytes+=GC_free_bytes[n];
if (bytes>=GC_max_large_allocd_bytes)return n;
}
return 0;
}
STATIC int GC_hblk_fl_from_blocks(word blocks_needed)
{
if (blocks_needed<=UNIQUE_THRESHOLD)return (int)blocks_needed;
if (blocks_needed>=HUGE_THRESHOLD)return N_HBLK_FLS;
return (int)(blocks_needed - UNIQUE_THRESHOLD)/FL_COMPRESSION
+UNIQUE_THRESHOLD;
}
#define PHDR(hhdr)HDR((hhdr)->hb_prev)
#define NHDR(hhdr)HDR((hhdr)->hb_next)
#ifdef USE_MUNMAP
#define IS_MAPPED(hhdr)(((hhdr)->hb_flags&WAS_UNMAPPED)==0)
#else
#define IS_MAPPED(hhdr)TRUE
#endif
#if!defined(NO_DEBUGGING)||defined(GC_ASSERTIONS)
GC_INNER word GC_compute_large_free_bytes(void)
{
word total_free=0;
unsigned i;
for (i=0;i<=N_HBLK_FLS;++i){
struct hblk*h;
hdr*hhdr;
for (h=GC_hblkfreelist[i];h!=0;h=hhdr->hb_next){
hhdr=HDR(h);
total_free+=hhdr->hb_sz;
}
}
return total_free;
}
#endif
#if!defined(NO_DEBUGGING)
void GC_print_hblkfreelist(void)
{
unsigned i;
word total;
for (i=0;i<=N_HBLK_FLS;++i){
struct hblk*h=GC_hblkfreelist[i];
if (0!=h)GC_printf("Free list %u (total size %lu):\n",
i,(unsigned long)GC_free_bytes[i]);
while (h){
hdr*hhdr=HDR(h);
GC_printf("\t%p size %lu %s black listed\n",
(void*)h,(unsigned long)hhdr->hb_sz,
GC_is_black_listed(h,HBLKSIZE)!=0?"start":
GC_is_black_listed(h,hhdr->hb_sz)!=0?"partially":
"not");
h=hhdr->hb_next;
}
}
GC_printf("GC_large_free_bytes:%lu\n",
(unsigned long)GC_large_free_bytes);
if ((total=GC_compute_large_free_bytes())!=GC_large_free_bytes)
GC_err_printf("GC_large_free_bytes INCONSISTENT!!Should be:%lu\n",
(unsigned long)total);
}
static int free_list_index_of(hdr*wanted)
{
int i;
for (i=0;i<=N_HBLK_FLS;++i){
struct hblk*h;
hdr*hhdr;
for (h=GC_hblkfreelist[i];h!=0;h=hhdr->hb_next){
hhdr=HDR(h);
if (hhdr==wanted)return i;
}
}
return -1;
}
GC_API void GC_CALL GC_dump_regions(void)
{
unsigned i;
for (i=0;i < GC_n_heap_sects;++i){
ptr_t start=GC_heap_sects[i].hs_start;
size_t bytes=GC_heap_sects[i].hs_bytes;
ptr_t end=start+bytes;
ptr_t p;
while (i+1 < GC_n_heap_sects&&GC_heap_sects[i+1].hs_start==end){
++i;
end=GC_heap_sects[i].hs_start+GC_heap_sects[i].hs_bytes;
}
GC_printf("***Section from %p to %p\n",(void*)start,(void*)end);
for (p=start;(word)p < (word)end;){
hdr*hhdr=HDR(p);
if (IS_FORWARDING_ADDR_OR_NIL(hhdr)){
GC_printf("\t%p Missing header!!(%p)\n",
(void*)p,(void*)hhdr);
p+=HBLKSIZE;
continue;
}
if (HBLK_IS_FREE(hhdr)){
int correct_index=GC_hblk_fl_from_blocks(
divHBLKSZ(hhdr->hb_sz));
int actual_index;
GC_printf("\t%p\tfree block of size 0x%lx bytes%s\n",
(void*)p,(unsigned long)(hhdr->hb_sz),
IS_MAPPED(hhdr)?"":" (unmapped)");
actual_index=free_list_index_of(hhdr);
if (-1==actual_index){
GC_printf("\t\tBlock not on free list %d!!\n",
correct_index);
} else if (correct_index!=actual_index){
GC_printf("\t\tBlock on list %d,should be on %d!!\n",
actual_index,correct_index);
}
p+=hhdr->hb_sz;
} else {
GC_printf("\t%p\tused for blocks of size 0x%lx bytes\n",
(void*)p,(unsigned long)(hhdr->hb_sz));
p+=HBLKSIZE*OBJ_SZ_TO_BLOCKS(hhdr->hb_sz);
}
}
}
}
#endif
static GC_bool setup_header(hdr*hhdr,struct hblk*block,size_t byte_sz,
int kind,unsigned flags)
{
word descr;
#ifdef MARK_BIT_PER_GRANULE
if (byte_sz > MAXOBJBYTES)
flags|=LARGE_BLOCK;
#endif
#ifdef ENABLE_DISCLAIM
if (GC_obj_kinds[kind].ok_disclaim_proc)
flags|=HAS_DISCLAIM;
if (GC_obj_kinds[kind].ok_mark_unconditionally)
flags|=MARK_UNCONDITIONALLY;
#endif
hhdr->hb_sz=byte_sz;
hhdr->hb_obj_kind=(unsigned char)kind;
hhdr->hb_flags=(unsigned char)flags;
hhdr->hb_block=block;
descr=GC_obj_kinds[kind].ok_descriptor;
if (GC_obj_kinds[kind].ok_relocate_descr)descr+=byte_sz;
hhdr->hb_descr=descr;
#ifdef MARK_BIT_PER_OBJ
if (byte_sz > MAXOBJBYTES){
hhdr->hb_inv_sz=LARGE_INV_SZ;
} else {
word inv_sz;
#if CPP_WORDSZ==64
inv_sz=((word)1<<32)/byte_sz;
if (((inv_sz*byte_sz)>>32)==0)++inv_sz;
#else
GC_ASSERT(byte_sz>=4);
inv_sz=((unsigned)1<<31)/byte_sz;
inv_sz*=2;
while (inv_sz*byte_sz > byte_sz)++inv_sz;
#endif
#ifdef INV_SZ_COMPUTATION_CHECK
GC_ASSERT(((1ULL<<32)+byte_sz - 1)/byte_sz==inv_sz);
#endif
hhdr->hb_inv_sz=inv_sz;
}
#endif
#ifdef MARK_BIT_PER_GRANULE
{
size_t granules=BYTES_TO_GRANULES(byte_sz);
if (EXPECT(!GC_add_map_entry(granules),FALSE)){
hhdr->hb_sz=HBLKSIZE;
hhdr->hb_descr=0;
hhdr->hb_flags|=LARGE_BLOCK;
hhdr->hb_map=0;
return FALSE;
}
hhdr->hb_map=GC_obj_map[(hhdr->hb_flags&LARGE_BLOCK)!=0?
0:granules];
}
#endif
GC_clear_hdr_marks(hhdr);
hhdr->hb_last_reclaimed=(unsigned short)GC_gc_no;
return(TRUE);
}
STATIC void GC_remove_from_fl_at(hdr*hhdr,int index)
{
GC_ASSERT(((hhdr->hb_sz)&(HBLKSIZE-1))==0);
if (hhdr->hb_prev==0){
GC_ASSERT(HDR(GC_hblkfreelist[index])==hhdr);
GC_hblkfreelist[index]=hhdr->hb_next;
} else {
hdr*phdr;
GET_HDR(hhdr->hb_prev,phdr);
phdr->hb_next=hhdr->hb_next;
}
GC_ASSERT(GC_free_bytes[index]>=hhdr->hb_sz);
GC_free_bytes[index]-=hhdr->hb_sz;
if (0!=hhdr->hb_next){
hdr*nhdr;
GC_ASSERT(!IS_FORWARDING_ADDR_OR_NIL(NHDR(hhdr)));
GET_HDR(hhdr->hb_next,nhdr);
nhdr->hb_prev=hhdr->hb_prev;
}
}
GC_INLINE void GC_remove_from_fl(hdr*hhdr)
{
GC_remove_from_fl_at(hhdr,GC_hblk_fl_from_blocks(divHBLKSZ(hhdr->hb_sz)));
}
static struct hblk*get_block_ending_at(struct hblk*h)
{
struct hblk*p=h - 1;
hdr*phdr;
GET_HDR(p,phdr);
while (0!=phdr&&IS_FORWARDING_ADDR_OR_NIL(phdr)){
p=FORWARDED_ADDR(p,phdr);
phdr=HDR(p);
}
if (0!=phdr){
return p;
}
p=GC_prev_block(h - 1);
if (p){
phdr=HDR(p);
if ((ptr_t)p+phdr->hb_sz==(ptr_t)h){
return p;
}
}
return NULL;
}
STATIC struct hblk*GC_free_block_ending_at(struct hblk*h)
{
struct hblk*p=get_block_ending_at(h);
if (p){
hdr*phdr=HDR(p);
if (HBLK_IS_FREE(phdr)){
return p;
}
}
return 0;
}
STATIC void GC_add_to_fl(struct hblk*h,hdr*hhdr)
{
int index=GC_hblk_fl_from_blocks(divHBLKSZ(hhdr->hb_sz));
struct hblk*second=GC_hblkfreelist[index];
#if defined(GC_ASSERTIONS)&&!defined(USE_MUNMAP)
struct hblk*next=(struct hblk*)((word)h+hhdr->hb_sz);
hdr*nexthdr=HDR(next);
struct hblk*prev=GC_free_block_ending_at(h);
hdr*prevhdr=HDR(prev);
GC_ASSERT(nexthdr==0||!HBLK_IS_FREE(nexthdr)
||(GC_heapsize&SIGNB)!=0);
GC_ASSERT(prev==0||!HBLK_IS_FREE(prevhdr)
||(GC_heapsize&SIGNB)!=0);
#endif
GC_ASSERT(((hhdr->hb_sz)&(HBLKSIZE-1))==0);
GC_hblkfreelist[index]=h;
GC_free_bytes[index]+=hhdr->hb_sz;
GC_ASSERT(GC_free_bytes[index]<=GC_large_free_bytes);
hhdr->hb_next=second;
hhdr->hb_prev=0;
if (second){
hdr*second_hdr;
GET_HDR(second,second_hdr);
second_hdr->hb_prev=h;
}
hhdr->hb_flags|=FREE_BLK;
}
#ifdef USE_MUNMAP
#ifndef MUNMAP_THRESHOLD
#define MUNMAP_THRESHOLD 6
#endif
GC_INNER int GC_unmap_threshold=MUNMAP_THRESHOLD;
#ifdef COUNT_UNMAPPED_REGIONS
static int calc_num_unmapped_regions_delta(struct hblk*h,hdr*hhdr)
{
struct hblk*prev=get_block_ending_at(h);
struct hblk*next;
GC_bool prev_unmapped=FALSE;
GC_bool next_unmapped=FALSE;
next=GC_next_block((struct hblk*)((ptr_t)h+hhdr->hb_sz),TRUE);
if ((ptr_t)next!=GC_unmap_end((ptr_t)h,(size_t)hhdr->hb_sz)){
next=NULL;
}
if (prev!=NULL){
hdr*prevhdr=HDR(prev);
prev_unmapped=!IS_MAPPED(prevhdr);
}
if (next!=NULL){
hdr*nexthdr=HDR(next);
next_unmapped=!IS_MAPPED(nexthdr);
}
if (prev_unmapped&&next_unmapped){
return IS_MAPPED(hhdr)?-1:1;
}
if (!prev_unmapped&&!next_unmapped){
return IS_MAPPED(hhdr)?1:-1;
}
return 0;
}
#endif
GC_INLINE void GC_adjust_num_unmapped(struct hblk*h GC_ATTR_UNUSED,
hdr*hhdr GC_ATTR_UNUSED)
{
#ifdef COUNT_UNMAPPED_REGIONS
GC_num_unmapped_regions+=calc_num_unmapped_regions_delta(h,hhdr);
#endif
}
GC_INNER void GC_unmap_old(void)
{
int i;
if (GC_unmap_threshold==0)
return;
#ifdef COUNT_UNMAPPED_REGIONS
if (GC_num_unmapped_regions>=GC_UNMAPPED_REGIONS_SOFT_LIMIT)
return;
#endif
for (i=0;i<=N_HBLK_FLS;++i){
struct hblk*h;
hdr*hhdr;
for (h=GC_hblkfreelist[i];0!=h;h=hhdr->hb_next){
hhdr=HDR(h);
if (!IS_MAPPED(hhdr))continue;
if ((unsigned short)(GC_gc_no - hhdr->hb_last_reclaimed)>
(unsigned short)GC_unmap_threshold){
#ifdef COUNT_UNMAPPED_REGIONS
int delta=calc_num_unmapped_regions_delta(h,hhdr);
signed_word regions=GC_num_unmapped_regions+delta;
if (delta>=0&&regions>=GC_UNMAPPED_REGIONS_SOFT_LIMIT){
GC_COND_LOG_PRINTF("Unmapped regions limit reached!\n");
return;
}
GC_num_unmapped_regions=regions;
#endif
GC_unmap((ptr_t)h,(size_t)hhdr->hb_sz);
hhdr->hb_flags|=WAS_UNMAPPED;
}
}
}
}
GC_INNER void GC_merge_unmapped(void)
{
int i;
for (i=0;i<=N_HBLK_FLS;++i){
struct hblk*h=GC_hblkfreelist[i];
while (h!=0){
struct hblk*next;
hdr*hhdr,*nexthdr;
word size,nextsize;
GET_HDR(h,hhdr);
size=hhdr->hb_sz;
next=(struct hblk*)((word)h+size);
GET_HDR(next,nexthdr);
if (0!=nexthdr&&HBLK_IS_FREE(nexthdr)
&&(signed_word)(size+(nextsize=nexthdr->hb_sz))> 0
){
if (IS_MAPPED(hhdr)&&!IS_MAPPED(nexthdr)){
if (size > nextsize){
GC_adjust_num_unmapped(next,nexthdr);
GC_remap((ptr_t)next,nextsize);
} else {
GC_adjust_num_unmapped(h,hhdr);
GC_unmap((ptr_t)h,size);
GC_unmap_gap((ptr_t)h,size,(ptr_t)next,nextsize);
hhdr->hb_flags|=WAS_UNMAPPED;
}
} else if (IS_MAPPED(nexthdr)&&!IS_MAPPED(hhdr)){
if (size > nextsize){
GC_adjust_num_unmapped(next,nexthdr);
GC_unmap((ptr_t)next,nextsize);
GC_unmap_gap((ptr_t)h,size,(ptr_t)next,nextsize);
} else {
GC_adjust_num_unmapped(h,hhdr);
GC_remap((ptr_t)h,size);
hhdr->hb_flags&=~WAS_UNMAPPED;
hhdr->hb_last_reclaimed=nexthdr->hb_last_reclaimed;
}
} else if (!IS_MAPPED(hhdr)&&!IS_MAPPED(nexthdr)){
GC_unmap_gap((ptr_t)h,size,(ptr_t)next,nextsize);
}
GC_remove_from_fl_at(hhdr,i);
GC_remove_from_fl(nexthdr);
hhdr->hb_sz+=nexthdr->hb_sz;
GC_remove_header(next);
GC_add_to_fl(h,hhdr);
h=GC_hblkfreelist[i];
} else {
h=hhdr->hb_next;
}
}
}
}
#endif
STATIC struct hblk*GC_get_first_part(struct hblk*h,hdr*hhdr,
size_t bytes,int index)
{
word total_size=hhdr->hb_sz;
struct hblk*rest;
hdr*rest_hdr;
GC_ASSERT((total_size&(HBLKSIZE-1))==0);
GC_remove_from_fl_at(hhdr,index);
if (total_size==bytes)return h;
rest=(struct hblk*)((word)h+bytes);
rest_hdr=GC_install_header(rest);
if (0==rest_hdr){
WARN("Header allocation failed:dropping block\n",0);
return(0);
}
rest_hdr->hb_sz=total_size - bytes;
rest_hdr->hb_flags=0;
#ifdef GC_ASSERTIONS
hhdr->hb_flags&=~FREE_BLK;
#endif
GC_add_to_fl(rest,rest_hdr);
return h;
}
STATIC void GC_split_block(struct hblk*h,hdr*hhdr,struct hblk*n,
hdr*nhdr,int index)
{
word total_size=hhdr->hb_sz;
word h_size=(word)n - (word)h;
struct hblk*prev=hhdr->hb_prev;
struct hblk*next=hhdr->hb_next;
nhdr->hb_prev=prev;
nhdr->hb_next=next;
nhdr->hb_sz=total_size - h_size;
nhdr->hb_flags=0;
if (prev){
HDR(prev)->hb_next=n;
} else {
GC_hblkfreelist[index]=n;
}
if (next){
HDR(next)->hb_prev=n;
}
GC_ASSERT(GC_free_bytes[index] > h_size);
GC_free_bytes[index]-=h_size;
#ifdef USE_MUNMAP
hhdr->hb_last_reclaimed=(unsigned short)GC_gc_no;
#endif
hhdr->hb_sz=h_size;
GC_add_to_fl(h,hhdr);
nhdr->hb_flags|=FREE_BLK;
}
STATIC struct hblk*
GC_allochblk_nth(size_t sz,int kind,unsigned flags,int n,
int may_split);
#define AVOID_SPLIT_REMAPPED 2
GC_INNER struct hblk*
GC_allochblk(size_t sz,int kind,unsigned flags)
{
word blocks;
int start_list;
struct hblk*result;
int may_split;
int split_limit;
GC_ASSERT((sz&(GRANULE_BYTES - 1))==0);
blocks=OBJ_SZ_TO_BLOCKS_CHECKED(sz);
if ((signed_word)(blocks*HBLKSIZE)< 0){
return 0;
}
start_list=GC_hblk_fl_from_blocks(blocks);
result=GC_allochblk_nth(sz,kind,flags,start_list,FALSE);
if (0!=result)return result;
may_split=TRUE;
if (GC_use_entire_heap||GC_dont_gc
||USED_HEAP_SIZE < GC_requested_heapsize
||GC_incremental||!GC_should_collect()){
split_limit=N_HBLK_FLS;
} else if (GC_finalizer_bytes_freed > (GC_heapsize>>4)){
split_limit=0;
} else {
split_limit=GC_enough_large_bytes_left();
#ifdef USE_MUNMAP
if (split_limit > 0)
may_split=AVOID_SPLIT_REMAPPED;
#endif
}
if (start_list < UNIQUE_THRESHOLD){
++start_list;
}
for (;start_list<=split_limit;++start_list){
result=GC_allochblk_nth(sz,kind,flags,start_list,may_split);
if (0!=result)
break;
}
return result;
}
STATIC long GC_large_alloc_warn_suppressed=0;
STATIC struct hblk*
GC_allochblk_nth(size_t sz,int kind,unsigned flags,int n,int may_split)
{
struct hblk*hbp;
hdr*hhdr;
struct hblk*thishbp;
hdr*thishdr;
signed_word size_needed=HBLKSIZE*OBJ_SZ_TO_BLOCKS_CHECKED(sz);
for (hbp=GC_hblkfreelist[n];;hbp=hhdr->hb_next){
signed_word size_avail;
if (hbp){
} else {
return NULL;
}
GET_HDR(hbp,hhdr);
size_avail=(signed_word)hhdr->hb_sz;
if (size_avail < size_needed)continue;
if (size_avail!=size_needed){
if (!may_split)continue;
thishbp=hhdr->hb_next;
if (thishbp){
signed_word next_size;
GET_HDR(thishbp,thishdr);
next_size=(signed_word)(thishdr->hb_sz);
if (next_size < size_avail
&&next_size>=size_needed
&&!GC_is_black_listed(thishbp,(word)size_needed)){
continue;
}
}
}
if (!IS_UNCOLLECTABLE(kind)&&(kind!=PTRFREE
||size_needed > (signed_word)MAX_BLACK_LIST_ALLOC)){
struct hblk*lasthbp=hbp;
ptr_t search_end=(ptr_t)hbp+size_avail - size_needed;
signed_word orig_avail=size_avail;
signed_word eff_size_needed=(flags&IGNORE_OFF_PAGE)!=0?
(signed_word)HBLKSIZE
:size_needed;
while ((word)lasthbp<=(word)search_end
&&(thishbp=GC_is_black_listed(lasthbp,
(word)eff_size_needed))!=0){
lasthbp=thishbp;
}
size_avail-=(ptr_t)lasthbp - (ptr_t)hbp;
thishbp=lasthbp;
if (size_avail>=size_needed){
if (thishbp!=hbp){
#ifdef USE_MUNMAP
if (may_split==AVOID_SPLIT_REMAPPED&&!IS_MAPPED(hhdr))
continue;
#endif
thishdr=GC_install_header(thishbp);
if (0!=thishdr){
#ifdef USE_MUNMAP
if (!IS_MAPPED(hhdr)){
GC_adjust_num_unmapped(hbp,hhdr);
GC_remap((ptr_t)hbp,(size_t)hhdr->hb_sz);
hhdr->hb_flags&=~WAS_UNMAPPED;
}
#endif
GC_split_block(hbp,hhdr,thishbp,thishdr,n);
hbp=thishbp;
hhdr=thishdr;
}
}
} else if (size_needed > (signed_word)BL_LIMIT
&&orig_avail - size_needed
> (signed_word)BL_LIMIT){
if (++GC_large_alloc_warn_suppressed
>=GC_large_alloc_warn_interval){
WARN("Repeated allocation of very large block "
"(appr. size %" WARN_PRIdPTR "):\n"
"\tMay lead to memory leak and poor performance\n",
size_needed);
GC_large_alloc_warn_suppressed=0;
}
size_avail=orig_avail;
} else if (size_avail==0
&&size_needed==(signed_word)HBLKSIZE
&&IS_MAPPED(hhdr)){
if (!GC_find_leak){
static unsigned count=0;
if ((++count&3)==0){
word total_size=hhdr->hb_sz;
struct hblk*limit=hbp+divHBLKSZ(total_size);
struct hblk*h;
struct hblk*prev=hhdr->hb_prev;
GC_large_free_bytes-=total_size;
GC_bytes_dropped+=total_size;
GC_remove_from_fl_at(hhdr,n);
for (h=hbp;(word)h < (word)limit;h++){
if (h!=hbp){
hhdr=GC_install_header(h);
}
if (NULL!=hhdr){
(void)setup_header(hhdr,h,HBLKSIZE,PTRFREE,0);
if (GC_debugging_started){
BZERO(h,HBLKSIZE);
}
}
}
hbp=prev;
if (0==hbp){
return GC_allochblk_nth(sz,kind,flags,n,may_split);
}
hhdr=HDR(hbp);
}
}
}
}
if( size_avail>=size_needed){
#ifdef USE_MUNMAP
if (!IS_MAPPED(hhdr)){
GC_adjust_num_unmapped(hbp,hhdr);
GC_remap((ptr_t)hbp,(size_t)hhdr->hb_sz);
hhdr->hb_flags&=~WAS_UNMAPPED;
}
#endif
hbp=GC_get_first_part(hbp,hhdr,size_needed,n);
break;
}
}
if (0==hbp)return 0;
if (!GC_install_counts(hbp,(word)size_needed))return(0);
if (!setup_header(hhdr,hbp,sz,kind,flags)){
GC_remove_counts(hbp,(word)size_needed);
return(0);
}
#ifndef GC_DISABLE_INCREMENTAL
GC_ASSERT((size_needed&(HBLKSIZE-1))==0);
GC_remove_protection(hbp,divHBLKSZ(size_needed),
(hhdr->hb_descr==0));
#endif
GC_fail_count=0;
GC_large_free_bytes-=size_needed;
GC_ASSERT(IS_MAPPED(hhdr));
return( hbp);
}
GC_INNER void GC_freehblk(struct hblk*hbp)
{
struct hblk*next,*prev;
hdr*hhdr,*prevhdr,*nexthdr;
word size;
GET_HDR(hbp,hhdr);
size=HBLKSIZE*OBJ_SZ_TO_BLOCKS(hhdr->hb_sz);
if ((size&SIGNB)!=0)
ABORT("Deallocating excessively large block. Too large an allocation?");
GC_remove_counts(hbp,size);
hhdr->hb_sz=size;
#ifdef USE_MUNMAP
hhdr->hb_last_reclaimed=(unsigned short)GC_gc_no;
#endif
if (HBLK_IS_FREE(hhdr)){
ABORT_ARG1("Duplicate large block deallocation",
" of %p",(void*)hbp);
}
GC_ASSERT(IS_MAPPED(hhdr));
hhdr->hb_flags|=FREE_BLK;
next=(struct hblk*)((ptr_t)hbp+size);
GET_HDR(next,nexthdr);
prev=GC_free_block_ending_at(hbp);
if(0!=nexthdr&&HBLK_IS_FREE(nexthdr)&&IS_MAPPED(nexthdr)
&&(signed_word)(hhdr->hb_sz+nexthdr->hb_sz)> 0
){
GC_remove_from_fl(nexthdr);
hhdr->hb_sz+=nexthdr->hb_sz;
GC_remove_header(next);
}
if (prev){
prevhdr=HDR(prev);
if (IS_MAPPED(prevhdr)
&&(signed_word)(hhdr->hb_sz+prevhdr->hb_sz)> 0){
GC_remove_from_fl(prevhdr);
prevhdr->hb_sz+=hhdr->hb_sz;
#ifdef USE_MUNMAP
prevhdr->hb_last_reclaimed=(unsigned short)GC_gc_no;
#endif
GC_remove_header(hbp);
hbp=prev;
hhdr=prevhdr;
}
}
GC_large_free_bytes+=size;
GC_add_to_fl(hbp,hhdr);
}
#include <stdio.h>
#if!defined(MACOS)&&!defined(MSWINCE)
#include <signal.h>
#if!defined(SN_TARGET_ORBIS)&&!defined(SN_TARGET_PSP2)&&!defined(__CC_ARM)
#include <sys/types.h>
#endif
#endif
word GC_non_gc_bytes=0;
word GC_gc_no=0;
#ifndef NO_CLOCK
static unsigned long full_gc_total_time=0;
static unsigned full_gc_total_ns_frac=0;
static GC_bool measure_performance=FALSE;
GC_API void GC_CALL GC_start_performance_measurement(void)
{
measure_performance=TRUE;
}
GC_API unsigned long GC_CALL GC_get_full_gc_total_time(void)
{
return full_gc_total_time;
}
#endif
#ifndef GC_DISABLE_INCREMENTAL
GC_INNER GC_bool GC_incremental=FALSE;
#endif
GC_API int GC_CALL GC_is_incremental_mode(void)
{
return (int)GC_incremental;
}
#ifdef THREADS
int GC_parallel=FALSE;
#endif
#if defined(GC_FULL_FREQ)&&!defined(CPPCHECK)
int GC_full_freq=GC_FULL_FREQ;
#else
int GC_full_freq=19;
#endif
STATIC GC_bool GC_need_full_gc=FALSE;
#ifdef THREAD_LOCAL_ALLOC
GC_INNER GC_bool GC_world_stopped=FALSE;
#endif
STATIC word GC_used_heap_size_after_full=0;
EXTERN_C_BEGIN
extern const char*const GC_copyright[];
EXTERN_C_END
const char*const GC_copyright[]=
{"Copyright 1988,1989 Hans-J. Boehm and Alan J. Demers ",
"Copyright (c)1991-1995 by Xerox Corporation. All rights reserved. ",
"Copyright (c)1996-1998 by Silicon Graphics. All rights reserved. ",
"Copyright (c)1999-2009 by Hewlett-Packard Company. All rights reserved. ",
"Copyright (c)2008-2020 Ivan Maidanski ",
"THIS MATERIAL IS PROVIDED AS IS,WITH ABSOLUTELY NO WARRANTY",
" EXPRESSED OR IMPLIED. ANY USE IS AT YOUR OWN RISK.",
"See source code for details." };
#ifndef GC_NO_VERSION_VAR
EXTERN_C_BEGIN
extern const unsigned GC_version;
EXTERN_C_END
const unsigned GC_version=((GC_VERSION_MAJOR<<16)|
(GC_VERSION_MINOR<<8)|GC_VERSION_MICRO);
#endif
GC_API unsigned GC_CALL GC_get_version(void)
{
return (GC_VERSION_MAJOR<<16)|(GC_VERSION_MINOR<<8)|
GC_VERSION_MICRO;
}
#ifdef GC_DONT_EXPAND
int GC_dont_expand=TRUE;
#else
int GC_dont_expand=FALSE;
#endif
#if defined(GC_FREE_SPACE_DIVISOR)&&!defined(CPPCHECK)
word GC_free_space_divisor=GC_FREE_SPACE_DIVISOR;
#else
word GC_free_space_divisor=3;
#endif
GC_INNER int GC_CALLBACK GC_never_stop_func(void)
{
return(0);
}
#if defined(GC_TIME_LIMIT)&&!defined(CPPCHECK)
unsigned long GC_time_limit=GC_TIME_LIMIT;
#elif defined(PARALLEL_MARK)
unsigned long GC_time_limit=GC_TIME_UNLIMITED;
#else
unsigned long GC_time_limit=50;
#endif
#ifndef NO_CLOCK
STATIC unsigned long GC_time_lim_nsec=0;
#define TV_NSEC_LIMIT (1000UL*1000)
GC_API void GC_CALL GC_set_time_limit_tv(struct GC_timeval_s tv)
{
GC_ASSERT(tv.tv_ms<=GC_TIME_UNLIMITED);
GC_ASSERT(tv.tv_nsec < TV_NSEC_LIMIT);
GC_time_limit=tv.tv_ms;
GC_time_lim_nsec=tv.tv_nsec;
}
GC_API struct GC_timeval_s GC_CALL GC_get_time_limit_tv(void)
{
struct GC_timeval_s tv;
tv.tv_ms=GC_time_limit;
tv.tv_nsec=GC_time_lim_nsec;
return tv;
}
STATIC CLOCK_TYPE GC_start_time=CLOCK_TYPE_INITIALIZER;
#endif
STATIC int GC_n_attempts=0;
STATIC GC_stop_func GC_default_stop_func=GC_never_stop_func;
GC_API void GC_CALL GC_set_stop_func(GC_stop_func stop_func)
{
DCL_LOCK_STATE;
GC_ASSERT(NONNULL_ARG_NOT_NULL(stop_func));
LOCK();
GC_default_stop_func=stop_func;
UNLOCK();
}
GC_API GC_stop_func GC_CALL GC_get_stop_func(void)
{
GC_stop_func stop_func;
DCL_LOCK_STATE;
LOCK();
stop_func=GC_default_stop_func;
UNLOCK();
return stop_func;
}
#if defined(GC_DISABLE_INCREMENTAL)||defined(NO_CLOCK)
#define GC_timeout_stop_func GC_default_stop_func
#else
STATIC int GC_CALLBACK GC_timeout_stop_func (void)
{
CLOCK_TYPE current_time;
static unsigned count=0;
unsigned long time_diff,nsec_diff;
if ((*GC_default_stop_func)())
return(1);
if ((count++&3)!=0)return(0);
GET_TIME(current_time);
time_diff=MS_TIME_DIFF(current_time,GC_start_time);
nsec_diff=NS_FRAC_TIME_DIFF(current_time,GC_start_time);
#if defined(CPPCHECK)
GC_noop1((word)&nsec_diff);
#endif
if (time_diff>=GC_time_limit
&&(time_diff > GC_time_limit||nsec_diff>=GC_time_lim_nsec)){
GC_COND_LOG_PRINTF("Abandoning stopped marking after %lu ms %lu ns"
" (attempt %d)\n",
time_diff,nsec_diff,GC_n_attempts);
return 1;
}
return(0);
}
#endif
#ifdef THREADS
GC_INNER word GC_total_stacksize=0;
#endif
static size_t min_bytes_allocd_minimum=1;
GC_API void GC_CALL GC_set_min_bytes_allocd(size_t value)
{
GC_ASSERT(value > 0);
min_bytes_allocd_minimum=value;
}
GC_API size_t GC_CALL GC_get_min_bytes_allocd(void)
{
return min_bytes_allocd_minimum;
}
static word min_bytes_allocd(void)
{
word result;
word stack_size;
word total_root_size;
word scan_size;
#ifdef THREADS
if (GC_need_to_lock){
stack_size=GC_total_stacksize;
#ifdef DEBUG_THREADS
GC_log_printf("Total stacks size:%lu\n",
(unsigned long)stack_size);
#endif
} else
#endif
{
#ifdef STACK_NOT_SCANNED
stack_size=0;
#elif defined(STACK_GROWS_UP)
stack_size=GC_approx_sp()- GC_stackbottom;
#else
stack_size=GC_stackbottom - GC_approx_sp();
#endif
}
total_root_size=2*stack_size+GC_root_size;
scan_size=2*GC_composite_in_use+GC_atomic_in_use/4
+total_root_size;
result=scan_size/GC_free_space_divisor;
if (GC_incremental){
result/=2;
}
return result > min_bytes_allocd_minimum
?result:min_bytes_allocd_minimum;
}
STATIC word GC_non_gc_bytes_at_gc=0;
STATIC word GC_adj_bytes_allocd(void)
{
signed_word result;
signed_word expl_managed=(signed_word)GC_non_gc_bytes
- (signed_word)GC_non_gc_bytes_at_gc;
result=(signed_word)GC_bytes_allocd
+(signed_word)GC_bytes_dropped
- (signed_word)GC_bytes_freed
+(signed_word)GC_finalizer_bytes_freed
- expl_managed;
if (result > (signed_word)GC_bytes_allocd){
result=GC_bytes_allocd;
}
result+=GC_bytes_finalized;
if (result < (signed_word)(GC_bytes_allocd>>3)){
return(GC_bytes_allocd>>3);
} else {
return(result);
}
}
STATIC void GC_clear_a_few_frames(void)
{
#ifndef CLEAR_NWORDS
#define CLEAR_NWORDS 64
#endif
volatile word frames[CLEAR_NWORDS];
BZERO((word*)frames,CLEAR_NWORDS*sizeof(word));
}
STATIC word GC_collect_at_heapsize=GC_WORD_MAX;
GC_INNER GC_bool GC_should_collect(void)
{
static word last_min_bytes_allocd;
static word last_gc_no;
if (last_gc_no!=GC_gc_no){
last_min_bytes_allocd=min_bytes_allocd();
last_gc_no=GC_gc_no;
}
return(GC_adj_bytes_allocd()>=last_min_bytes_allocd
||GC_heapsize>=GC_collect_at_heapsize);
}
GC_start_callback_proc GC_start_call_back=0;
GC_API void GC_CALL GC_set_start_callback(GC_start_callback_proc fn)
{
DCL_LOCK_STATE;
LOCK();
GC_start_call_back=fn;
UNLOCK();
}
GC_API GC_start_callback_proc GC_CALL GC_get_start_callback(void)
{
GC_start_callback_proc fn;
DCL_LOCK_STATE;
LOCK();
fn=GC_start_call_back;
UNLOCK();
return fn;
}
GC_INLINE void GC_notify_full_gc(void)
{
if (GC_start_call_back!=0){
(*GC_start_call_back)();
}
}
STATIC GC_bool GC_is_full_gc=FALSE;
STATIC GC_bool GC_stopped_mark(GC_stop_func stop_func);
STATIC void GC_finish_collection(void);
STATIC void GC_maybe_gc(void)
{
GC_ASSERT(I_HOLD_LOCK());
ASSERT_CANCEL_DISABLED();
if (GC_should_collect()){
static int n_partial_gcs=0;
if (!GC_incremental){
GC_try_to_collect_inner(GC_never_stop_func);
n_partial_gcs=0;
return;
} else {
#ifdef PARALLEL_MARK
if (GC_parallel)
GC_wait_for_reclaim();
#endif
if (GC_need_full_gc||n_partial_gcs>=GC_full_freq){
GC_COND_LOG_PRINTF(
"***>Full mark for collection #%lu after %lu allocd bytes\n",
(unsigned long)GC_gc_no+1,(unsigned long)GC_bytes_allocd);
GC_promote_black_lists();
(void)GC_reclaim_all((GC_stop_func)0,TRUE);
GC_notify_full_gc();
GC_clear_marks();
n_partial_gcs=0;
GC_is_full_gc=TRUE;
} else {
n_partial_gcs++;
}
}
#ifndef NO_CLOCK
if (GC_time_limit!=GC_TIME_UNLIMITED){ GET_TIME(GC_start_time);}
#endif
if (GC_stopped_mark(GC_time_limit==GC_TIME_UNLIMITED?
GC_never_stop_func:GC_timeout_stop_func)){
#ifdef SAVE_CALL_CHAIN
GC_save_callers(GC_last_stack);
#endif
GC_finish_collection();
} else {
if (!GC_is_full_gc){
GC_n_attempts++;
}
}
}
}
STATIC GC_on_collection_event_proc GC_on_collection_event=0;
GC_API void GC_CALL GC_set_on_collection_event(GC_on_collection_event_proc fn)
{
DCL_LOCK_STATE;
LOCK();
GC_on_collection_event=fn;
UNLOCK();
}
GC_API GC_on_collection_event_proc GC_CALL GC_get_on_collection_event(void)
{
GC_on_collection_event_proc fn;
DCL_LOCK_STATE;
LOCK();
fn=GC_on_collection_event;
UNLOCK();
return fn;
}
GC_INNER GC_bool GC_try_to_collect_inner(GC_stop_func stop_func)
{
#ifndef NO_CLOCK
CLOCK_TYPE start_time=CLOCK_TYPE_INITIALIZER;
GC_bool start_time_valid;
#endif
ASSERT_CANCEL_DISABLED();
GC_ASSERT(I_HOLD_LOCK());
if (GC_dont_gc||(*stop_func)())return FALSE;
if (GC_on_collection_event)
GC_on_collection_event(GC_EVENT_START);
if (GC_incremental&&GC_collection_in_progress()){
GC_COND_LOG_PRINTF(
"GC_try_to_collect_inner:finishing collection in progress\n");
while(GC_collection_in_progress()){
if ((*stop_func)()){
return(FALSE);
}
ENTER_GC();
GC_collect_a_little_inner(1);
EXIT_GC();
}
}
GC_notify_full_gc();
#ifndef NO_CLOCK
start_time_valid=FALSE;
if ((GC_print_stats|(int)measure_performance)!=0){
if (GC_print_stats)
GC_log_printf("Initiating full world-stop collection!\n");
start_time_valid=TRUE;
GET_TIME(start_time);
}
#endif
GC_promote_black_lists();
#ifdef PARALLEL_MARK
if (GC_parallel)
GC_wait_for_reclaim();
#endif
if ((GC_find_leak||stop_func!=GC_never_stop_func)
&&!GC_reclaim_all(stop_func,FALSE)){
return(FALSE);
}
GC_invalidate_mark_state();
GC_clear_marks();
#ifdef SAVE_CALL_CHAIN
GC_save_callers(GC_last_stack);
#endif
GC_is_full_gc=TRUE;
if (!GC_stopped_mark(stop_func)){
if (!GC_incremental){
GC_invalidate_mark_state();
GC_unpromote_black_lists();
}
return(FALSE);
}
GC_finish_collection();
#ifndef NO_CLOCK
if (start_time_valid){
CLOCK_TYPE current_time;
unsigned long time_diff,ns_frac_diff;
GET_TIME(current_time);
time_diff=MS_TIME_DIFF(current_time,start_time);
ns_frac_diff=NS_FRAC_TIME_DIFF(current_time,start_time);
if (measure_performance){
full_gc_total_time+=time_diff;
full_gc_total_ns_frac+=(unsigned)ns_frac_diff;
if (full_gc_total_ns_frac>=1000000U){
full_gc_total_ns_frac-=1000000U;
full_gc_total_time++;
}
}
if (GC_print_stats)
GC_log_printf("Complete collection took %lu ms %lu ns\n",
time_diff,ns_frac_diff);
}
#endif
if (GC_on_collection_event)
GC_on_collection_event(GC_EVENT_END);
return(TRUE);
}
#ifndef GC_RATE
#define GC_RATE 10
#endif
#ifndef MAX_PRIOR_ATTEMPTS
#define MAX_PRIOR_ATTEMPTS 1
#endif
STATIC int GC_deficit=0;
STATIC int GC_rate=GC_RATE;
GC_API void GC_CALL GC_set_rate(int value)
{
GC_ASSERT(value > 0);
GC_rate=value;
}
GC_API int GC_CALL GC_get_rate(void)
{
return GC_rate;
}
static int max_prior_attempts=MAX_PRIOR_ATTEMPTS;
GC_API void GC_CALL GC_set_max_prior_attempts(int value)
{
GC_ASSERT(value>=0);
max_prior_attempts=value;
}
GC_API int GC_CALL GC_get_max_prior_attempts(void)
{
return max_prior_attempts;
}
GC_INNER void GC_collect_a_little_inner(int n)
{
IF_CANCEL(int cancel_state;)
GC_ASSERT(I_HOLD_LOCK());
if (GC_dont_gc)return;
DISABLE_CANCEL(cancel_state);
if (GC_incremental&&GC_collection_in_progress()){
int i;
int max_deficit=GC_rate*n;
#ifdef PARALLEL_MARK
if (GC_time_limit!=GC_TIME_UNLIMITED)
GC_parallel_mark_disabled=TRUE;
#endif
for (i=GC_deficit;i < max_deficit;i++){
if (GC_mark_some(NULL))
break;
}
#ifdef PARALLEL_MARK
GC_parallel_mark_disabled=FALSE;
#endif
if (i < max_deficit){
#ifdef SAVE_CALL_CHAIN
GC_save_callers(GC_last_stack);
#endif
#ifdef PARALLEL_MARK
if (GC_parallel)
GC_wait_for_reclaim();
#endif
if (GC_n_attempts < max_prior_attempts
&&GC_time_limit!=GC_TIME_UNLIMITED){
#ifndef NO_CLOCK
GET_TIME(GC_start_time);
#endif
if (GC_stopped_mark(GC_timeout_stop_func)){
GC_finish_collection();
} else {
GC_n_attempts++;
}
} else {
(void)GC_stopped_mark(GC_never_stop_func);
GC_finish_collection();
}
}
if (GC_deficit > 0){
GC_deficit-=max_deficit;
if (GC_deficit < 0)
GC_deficit=0;
}
} else {
GC_maybe_gc();
}
RESTORE_CANCEL(cancel_state);
}
GC_INNER void (*GC_check_heap)(void)=0;
GC_INNER void (*GC_print_all_smashed)(void)=0;
GC_API int GC_CALL GC_collect_a_little(void)
{
int result;
DCL_LOCK_STATE;
LOCK();
ENTER_GC();
GC_collect_a_little_inner(1);
EXIT_GC();
result=(int)GC_collection_in_progress();
UNLOCK();
if (!result&&GC_debugging_started)GC_print_all_smashed();
return(result);
}
#ifndef NO_CLOCK
static unsigned world_stopped_total_time=0;
static unsigned world_stopped_total_divisor=0;
#ifndef MAX_TOTAL_TIME_DIVISOR
#define MAX_TOTAL_TIME_DIVISOR 1000
#endif
#endif
#ifdef USE_MUNMAP
#define IF_USE_MUNMAP(x)x
#define COMMA_IF_USE_MUNMAP(x),x
#else
#define IF_USE_MUNMAP(x)
#define COMMA_IF_USE_MUNMAP(x)
#endif
STATIC GC_bool GC_stopped_mark(GC_stop_func stop_func)
{
int i;
#ifndef NO_CLOCK
CLOCK_TYPE start_time=CLOCK_TYPE_INITIALIZER;
#endif
GC_ASSERT(I_HOLD_LOCK());
#if!defined(REDIRECT_MALLOC)&&defined(USE_WINALLOC)
GC_add_current_malloc_heap();
#endif
#if defined(REGISTER_LIBRARIES_EARLY)
GC_cond_register_dynamic_libraries();
#endif
#ifndef NO_CLOCK
if (GC_PRINT_STATS_FLAG)
GET_TIME(start_time);
#endif
#if!defined(GC_NO_FINALIZATION)&&!defined(GC_TOGGLE_REFS_NOT_NEEDED)
GC_process_togglerefs();
#endif
#ifdef THREADS
if (GC_on_collection_event)
GC_on_collection_event(GC_EVENT_PRE_STOP_WORLD);
#endif
STOP_WORLD();
#ifdef THREADS
if (GC_on_collection_event)
GC_on_collection_event(GC_EVENT_POST_STOP_WORLD);
#endif
#ifdef THREAD_LOCAL_ALLOC
GC_world_stopped=TRUE;
#endif
GC_COND_LOG_PRINTF(
"\n--> Marking for collection #%lu after %lu allocated bytes\n",
(unsigned long)GC_gc_no+1,(unsigned long)GC_bytes_allocd);
#ifdef MAKE_BACK_GRAPH
if (GC_print_back_height){
GC_build_back_graph();
}
#endif
if (GC_on_collection_event)
GC_on_collection_event(GC_EVENT_MARK_START);
GC_clear_a_few_frames();
GC_noop6(0,0,0,0,0,0);
GC_initiate_gc();
#ifdef PARALLEL_MARK
if (stop_func!=GC_never_stop_func)
GC_parallel_mark_disabled=TRUE;
#endif
for (i=0;!(*stop_func)();i++){
if (GC_mark_some(GC_approx_sp())){
#ifdef PARALLEL_MARK
if (GC_parallel&&GC_parallel_mark_disabled){
GC_COND_LOG_PRINTF("Stopped marking done after %d iterations"
" with disabled parallel marker\n",i);
}
#endif
i=-1;
break;
}
}
#ifdef PARALLEL_MARK
GC_parallel_mark_disabled=FALSE;
#endif
if (i>=0){
GC_COND_LOG_PRINTF("Abandoned stopped marking after"
" %d iterations\n",i);
GC_deficit=i;
#ifdef THREAD_LOCAL_ALLOC
GC_world_stopped=FALSE;
#endif
#ifdef THREADS
if (GC_on_collection_event)
GC_on_collection_event(GC_EVENT_PRE_START_WORLD);
#endif
START_WORLD();
#ifdef THREADS
if (GC_on_collection_event)
GC_on_collection_event(GC_EVENT_POST_START_WORLD);
#endif
return FALSE;
}
GC_gc_no++;
GC_DBGLOG_PRINTF("GC #%lu freed %ld bytes,heap %lu KiB"
IF_USE_MUNMAP(" (+%lu KiB unmapped)")"\n",
(unsigned long)GC_gc_no,(long)GC_bytes_found,
TO_KiB_UL(GC_heapsize - GC_unmapped_bytes)
COMMA_IF_USE_MUNMAP(TO_KiB_UL(GC_unmapped_bytes)));
if (GC_debugging_started){
(*GC_check_heap)();
}
if (GC_on_collection_event){
GC_on_collection_event(GC_EVENT_MARK_END);
#ifdef THREADS
GC_on_collection_event(GC_EVENT_PRE_START_WORLD);
#endif
}
#ifdef THREAD_LOCAL_ALLOC
GC_world_stopped=FALSE;
#endif
START_WORLD();
#ifdef THREADS
if (GC_on_collection_event)
GC_on_collection_event(GC_EVENT_POST_START_WORLD);
#endif
#ifndef NO_CLOCK
if (GC_PRINT_STATS_FLAG){
unsigned long time_diff;
unsigned total_time,divisor;
CLOCK_TYPE current_time;
GET_TIME(current_time);
time_diff=MS_TIME_DIFF(current_time,start_time);
total_time=world_stopped_total_time;
divisor=world_stopped_total_divisor;
if ((int)total_time < 0||divisor>=MAX_TOTAL_TIME_DIVISOR){
total_time>>=1;
divisor>>=1;
}
total_time+=time_diff < (((unsigned)-1)>>1)?
(unsigned)time_diff:((unsigned)-1)>>1;
world_stopped_total_time=total_time;
world_stopped_total_divisor=++divisor;
GC_ASSERT(divisor!=0);
GC_log_printf("World-stopped marking took %lu ms %lu ns"
" (%u ms in average)\n",
time_diff,NS_FRAC_TIME_DIFF(current_time,start_time),
total_time/divisor);
}
#endif
return(TRUE);
}
GC_INNER void GC_set_fl_marks(ptr_t q)
{
if (q){
struct hblk*h=HBLKPTR(q);
struct hblk*last_h=h;
hdr*hhdr=HDR(h);
IF_PER_OBJ(word sz=hhdr->hb_sz;)
for (;;){
word bit_no=MARK_BIT_NO((ptr_t)q - (ptr_t)h,sz);
if (!mark_bit_from_hdr(hhdr,bit_no)){
set_mark_bit_from_hdr(hhdr,bit_no);
++hhdr->hb_n_marks;
}
q=(ptr_t)obj_link(q);
if (q==NULL)
break;
h=HBLKPTR(q);
if (h!=last_h){
last_h=h;
hhdr=HDR(h);
IF_PER_OBJ(sz=hhdr->hb_sz;)
}
}
}
}
#if defined(GC_ASSERTIONS)&&defined(THREAD_LOCAL_ALLOC)
void GC_check_fl_marks(void**pfreelist)
{
#if defined(AO_HAVE_load_acquire_read)&&!defined(THREAD_SANITIZER)
AO_t*list=(AO_t*)AO_load_acquire_read((AO_t*)pfreelist);
AO_t*prev;
AO_t*p;
if ((word)list<=HBLKSIZE)return;
prev=(AO_t*)pfreelist;
for (p=list;p!=NULL;){
AO_t*next;
if (!GC_is_marked(p)){
ABORT_ARG2("Unmarked local free list entry",
":object %p on list %p",(void*)p,(void*)list);
}
next=(AO_t*)AO_load_acquire_read(p);
if (AO_load(prev)!=(AO_t)p)
break;
prev=p;
p=next;
}
#else
(void)pfreelist;
#endif
}
#endif
STATIC void GC_clear_fl_marks(ptr_t q)
{
struct hblk*h=HBLKPTR(q);
struct hblk*last_h=h;
hdr*hhdr=HDR(h);
word sz=hhdr->hb_sz;
for (;;){
word bit_no=MARK_BIT_NO((ptr_t)q - (ptr_t)h,sz);
if (mark_bit_from_hdr(hhdr,bit_no)){
size_t n_marks=hhdr->hb_n_marks;
GC_ASSERT(n_marks!=0);
clear_mark_bit_from_hdr(hhdr,bit_no);
n_marks--;
#ifdef PARALLEL_MARK
if (0!=n_marks||!GC_parallel){
hhdr->hb_n_marks=n_marks;
}
#else
hhdr->hb_n_marks=n_marks;
#endif
}
GC_bytes_found-=sz;
q=(ptr_t)obj_link(q);
if (q==NULL)
break;
h=HBLKPTR(q);
if (h!=last_h){
last_h=h;
hhdr=HDR(h);
sz=hhdr->hb_sz;
}
}
}
#if defined(GC_ASSERTIONS)&&defined(THREAD_LOCAL_ALLOC)
void GC_check_tls(void);
#endif
GC_on_heap_resize_proc GC_on_heap_resize=0;
GC_INLINE int GC_compute_heap_usage_percent(void)
{
word used=GC_composite_in_use+GC_atomic_in_use;
word heap_sz=GC_heapsize - GC_unmapped_bytes;
#if defined(CPPCHECK)
word limit=(GC_WORD_MAX>>1)/50;
#else
const word limit=GC_WORD_MAX/100;
#endif
return used>=heap_sz?0:used < limit?
(int)((used*100)/heap_sz):(int)(used/(heap_sz/100));
}
STATIC void GC_finish_collection(void)
{
#ifndef NO_CLOCK
CLOCK_TYPE start_time=CLOCK_TYPE_INITIALIZER;
CLOCK_TYPE finalize_time=CLOCK_TYPE_INITIALIZER;
#endif
GC_ASSERT(I_HOLD_LOCK());
#if defined(GC_ASSERTIONS)&&defined(THREAD_LOCAL_ALLOC)&&!defined(DBG_HDRS_ALL)
GC_check_tls();
#endif
#ifndef NO_CLOCK
if (GC_print_stats)
GET_TIME(start_time);
#endif
if (GC_on_collection_event)
GC_on_collection_event(GC_EVENT_RECLAIM_START);
#ifndef GC_GET_HEAP_USAGE_NOT_NEEDED
if (GC_bytes_found > 0)
GC_reclaimed_bytes_before_gc+=(word)GC_bytes_found;
#endif
GC_bytes_found=0;
#if defined(LINUX)&&defined(__ELF__)&&!defined(SMALL_CONFIG)
if (GETENV("GC_PRINT_ADDRESS_MAP")!=0){
GC_print_address_map();
}
#endif
COND_DUMP;
if (GC_find_leak){
word size;
unsigned kind;
ptr_t q;
for (kind=0;kind < GC_n_kinds;kind++){
for (size=1;size<=MAXOBJGRANULES;size++){
q=(ptr_t)GC_obj_kinds[kind].ok_freelist[size];
if (q!=NULL)
GC_set_fl_marks(q);
}
}
GC_start_reclaim(TRUE);
}
#ifndef GC_NO_FINALIZATION
GC_finalize();
#endif
#ifndef NO_CLOCK
if (GC_print_stats)
GET_TIME(finalize_time);
#endif
if (GC_print_back_height){
#ifdef MAKE_BACK_GRAPH
GC_traverse_back_graph();
#elif!defined(SMALL_CONFIG)
GC_err_printf("Back height not available:"
"Rebuild collector with -DMAKE_BACK_GRAPH\n");
#endif
}
{
word size;
ptr_t q;
unsigned kind;
for (kind=0;kind < GC_n_kinds;kind++){
for (size=1;size<=MAXOBJGRANULES;size++){
q=(ptr_t)GC_obj_kinds[kind].ok_freelist[size];
if (q!=NULL)
GC_clear_fl_marks(q);
}
}
}
GC_VERBOSE_LOG_PRINTF("Bytes recovered before sweep - f.l. count=%ld\n",
(long)GC_bytes_found);
GC_start_reclaim(FALSE);
GC_DBGLOG_PRINTF("In-use heap:%d%% (%lu KiB pointers+%lu KiB other)\n",
GC_compute_heap_usage_percent(),
TO_KiB_UL(GC_composite_in_use),
TO_KiB_UL(GC_atomic_in_use));
if (GC_is_full_gc){
GC_used_heap_size_after_full=USED_HEAP_SIZE;
GC_need_full_gc=FALSE;
} else {
GC_need_full_gc=USED_HEAP_SIZE - GC_used_heap_size_after_full
> min_bytes_allocd();
}
GC_VERBOSE_LOG_PRINTF("Immediately reclaimed %ld bytes,heapsize:"
" %lu bytes" IF_USE_MUNMAP(" (%lu unmapped)")"\n",
(long)GC_bytes_found,
(unsigned long)GC_heapsize
COMMA_IF_USE_MUNMAP((unsigned long)
GC_unmapped_bytes));
GC_n_attempts=0;
GC_is_full_gc=FALSE;
GC_bytes_allocd_before_gc+=GC_bytes_allocd;
GC_non_gc_bytes_at_gc=GC_non_gc_bytes;
GC_bytes_allocd=0;
GC_bytes_dropped=0;
GC_bytes_freed=0;
GC_finalizer_bytes_freed=0;
IF_USE_MUNMAP(GC_unmap_old());
if (GC_on_collection_event)
GC_on_collection_event(GC_EVENT_RECLAIM_END);
#ifndef NO_CLOCK
if (GC_print_stats){
CLOCK_TYPE done_time;
GET_TIME(done_time);
#if!defined(SMALL_CONFIG)&&!defined(GC_NO_FINALIZATION)
GC_print_finalization_stats();
#endif
GC_log_printf("Finalize and initiate sweep took %lu ms %lu ns"
"+%lu ms %lu ns\n",
MS_TIME_DIFF(finalize_time,start_time),
NS_FRAC_TIME_DIFF(finalize_time,start_time),
MS_TIME_DIFF(done_time,finalize_time),
NS_FRAC_TIME_DIFF(done_time,finalize_time));
}
#elif!defined(SMALL_CONFIG)&&!defined(GC_NO_FINALIZATION)
if (GC_print_stats)
GC_print_finalization_stats();
#endif
}
STATIC GC_bool GC_try_to_collect_general(GC_stop_func stop_func,
GC_bool force_unmap GC_ATTR_UNUSED)
{
GC_bool result;
IF_USE_MUNMAP(int old_unmap_threshold;)
IF_CANCEL(int cancel_state;)
DCL_LOCK_STATE;
if (!EXPECT(GC_is_initialized,TRUE))GC_init();
if (GC_debugging_started)GC_print_all_smashed();
GC_INVOKE_FINALIZERS();
LOCK();
DISABLE_CANCEL(cancel_state);
#ifdef USE_MUNMAP
old_unmap_threshold=GC_unmap_threshold;
if (force_unmap||
(GC_force_unmap_on_gcollect&&old_unmap_threshold > 0))
GC_unmap_threshold=1;
#endif
ENTER_GC();
GC_noop6(0,0,0,0,0,0);
result=GC_try_to_collect_inner(stop_func!=0?stop_func:
GC_default_stop_func);
EXIT_GC();
IF_USE_MUNMAP(GC_unmap_threshold=old_unmap_threshold);
RESTORE_CANCEL(cancel_state);
UNLOCK();
if (result){
if (GC_debugging_started)GC_print_all_smashed();
GC_INVOKE_FINALIZERS();
}
return(result);
}
GC_API int GC_CALL GC_try_to_collect(GC_stop_func stop_func)
{
GC_ASSERT(NONNULL_ARG_NOT_NULL(stop_func));
return (int)GC_try_to_collect_general(stop_func,FALSE);
}
GC_API void GC_CALL GC_gcollect(void)
{
(void)GC_try_to_collect_general(0,FALSE);
if (GC_have_errors)GC_print_all_errors();
}
STATIC word GC_heapsize_at_forced_unmap=0;
GC_API void GC_CALL GC_gcollect_and_unmap(void)
{
GC_heapsize_at_forced_unmap=GC_heapsize;
(void)GC_try_to_collect_general(GC_never_stop_func,TRUE);
}
#ifdef USE_PROC_FOR_LIBRARIES
GC_INNER void GC_add_to_our_memory(ptr_t p,size_t bytes)
{
if (0==p)return;
if (GC_n_memory>=MAX_HEAP_SECTS)
ABORT("Too many GC-allocated memory sections:Increase MAX_HEAP_SECTS");
GC_our_memory[GC_n_memory].hs_start=p;
GC_our_memory[GC_n_memory].hs_bytes=bytes;
GC_n_memory++;
}
#endif
GC_INNER void GC_add_to_heap(struct hblk*p,size_t bytes)
{
hdr*phdr;
word endp;
if (GC_n_heap_sects>=MAX_HEAP_SECTS){
ABORT("Too many heap sections:Increase MAXHINCR or MAX_HEAP_SECTS");
}
while ((word)p<=HBLKSIZE){
++p;
bytes-=HBLKSIZE;
if (0==bytes)return;
}
endp=(word)p+bytes;
if (endp<=(word)p){
bytes-=HBLKSIZE;
if (0==bytes)return;
endp-=HBLKSIZE;
}
phdr=GC_install_header(p);
if (0==phdr){
return;
}
GC_ASSERT(endp > (word)p&&endp==(word)p+bytes);
GC_heap_sects[GC_n_heap_sects].hs_start=(ptr_t)p;
GC_heap_sects[GC_n_heap_sects].hs_bytes=bytes;
GC_n_heap_sects++;
phdr->hb_sz=bytes;
phdr->hb_flags=0;
GC_freehblk(p);
GC_heapsize+=bytes;
GC_collect_at_heapsize+=bytes;
if (GC_collect_at_heapsize < GC_heapsize)
GC_collect_at_heapsize=GC_WORD_MAX;
if ((word)p<=(word)GC_least_plausible_heap_addr
||GC_least_plausible_heap_addr==0){
GC_least_plausible_heap_addr=(void*)((ptr_t)p - sizeof(word));
}
if ((word)p+bytes>=(word)GC_greatest_plausible_heap_addr){
GC_greatest_plausible_heap_addr=(void*)endp;
}
}
#if!defined(NO_DEBUGGING)
void GC_print_heap_sects(void)
{
unsigned i;
GC_printf("Total heap size:%lu" IF_USE_MUNMAP(" (%lu unmapped)")"\n",
(unsigned long)GC_heapsize
COMMA_IF_USE_MUNMAP((unsigned long)GC_unmapped_bytes));
for (i=0;i < GC_n_heap_sects;i++){
ptr_t start=GC_heap_sects[i].hs_start;
size_t len=GC_heap_sects[i].hs_bytes;
struct hblk*h;
unsigned nbl=0;
for (h=(struct hblk*)start;(word)h < (word)(start+len);h++){
if (GC_is_black_listed(h,HBLKSIZE))nbl++;
}
GC_printf("Section %d from %p to %p %u/%lu blacklisted\n",
i,(void*)start,(void*)&start[len],
nbl,(unsigned long)divHBLKSZ(len));
}
}
#endif
void*GC_least_plausible_heap_addr=(void*)GC_WORD_MAX;
void*GC_greatest_plausible_heap_addr=0;
GC_INLINE word GC_max(word x,word y)
{
return(x > y?x:y);
}
GC_INLINE word GC_min(word x,word y)
{
return(x < y?x:y);
}
STATIC word GC_max_heapsize=0;
GC_API void GC_CALL GC_set_max_heap_size(GC_word n)
{
GC_max_heapsize=n;
}
GC_word GC_max_retries=0;
GC_INNER GC_bool GC_expand_hp_inner(word n)
{
size_t bytes;
struct hblk*space;
word expansion_slop;
GC_ASSERT(I_HOLD_LOCK());
GC_ASSERT(GC_page_size!=0);
if (n < MINHINCR)n=MINHINCR;
bytes=ROUNDUP_PAGESIZE((size_t)n*HBLKSIZE);
if (GC_max_heapsize!=0
&&(GC_max_heapsize < (word)bytes
||GC_heapsize > GC_max_heapsize - (word)bytes)){
return(FALSE);
}
space=GET_MEM(bytes);
GC_add_to_our_memory((ptr_t)space,bytes);
if (space==0){
WARN("Failed to expand heap by %" WARN_PRIdPTR " bytes\n",
(word)bytes);
return(FALSE);
}
GC_INFOLOG_PRINTF("Grow heap to %lu KiB after %lu bytes allocated\n",
TO_KiB_UL(GC_heapsize+(word)bytes),
(unsigned long)GC_bytes_allocd);
expansion_slop=min_bytes_allocd()+4*MAXHINCR*HBLKSIZE;
if ((GC_last_heap_addr==0&&!((word)space&SIGNB))
||(GC_last_heap_addr!=0
&&(word)GC_last_heap_addr < (word)space)){
word new_limit=(word)space+(word)bytes+expansion_slop;
if (new_limit > (word)space){
GC_greatest_plausible_heap_addr=
(void*)GC_max((word)GC_greatest_plausible_heap_addr,
(word)new_limit);
}
} else {
word new_limit=(word)space - expansion_slop;
if (new_limit < (word)space){
GC_least_plausible_heap_addr=
(void*)GC_min((word)GC_least_plausible_heap_addr,
(word)space - expansion_slop);
}
}
GC_prev_heap_addr=GC_last_heap_addr;
GC_last_heap_addr=(ptr_t)space;
GC_add_to_heap(space,bytes);
GC_collect_at_heapsize=
GC_heapsize+expansion_slop - 2*MAXHINCR*HBLKSIZE;
if (GC_collect_at_heapsize < GC_heapsize)
GC_collect_at_heapsize=GC_WORD_MAX;
if (GC_on_heap_resize)
(*GC_on_heap_resize)(GC_heapsize);
return(TRUE);
}
GC_API int GC_CALL GC_expand_hp(size_t bytes)
{
int result;
DCL_LOCK_STATE;
if (!EXPECT(GC_is_initialized,TRUE))GC_init();
LOCK();
result=(int)GC_expand_hp_inner(divHBLKSZ((word)bytes));
if (result)GC_requested_heapsize+=bytes;
UNLOCK();
return(result);
}
GC_INNER unsigned GC_fail_count=0;
#if defined(GC_ALLOCD_BYTES_PER_FINALIZER)&&!defined(CPPCHECK)
STATIC word GC_allocd_bytes_per_finalizer=GC_ALLOCD_BYTES_PER_FINALIZER;
#else
STATIC word GC_allocd_bytes_per_finalizer=10000;
#endif
GC_API void GC_CALL GC_set_allocd_bytes_per_finalizer(GC_word value)
{
GC_allocd_bytes_per_finalizer=value;
}
GC_API GC_word GC_CALL GC_get_allocd_bytes_per_finalizer(void)
{
return GC_allocd_bytes_per_finalizer;
}
static word last_fo_entries=0;
static word last_bytes_finalized=0;
GC_INNER GC_bool GC_collect_or_expand(word needed_blocks,
GC_bool ignore_off_page,
GC_bool retry)
{
GC_bool gc_not_stopped=TRUE;
word blocks_to_get;
IF_CANCEL(int cancel_state;)
GC_ASSERT(I_HOLD_LOCK());
DISABLE_CANCEL(cancel_state);
if (!GC_incremental&&!GC_dont_gc&&
((GC_dont_expand&&GC_bytes_allocd > 0)
||(GC_fo_entries > last_fo_entries
&&(last_bytes_finalized|GC_bytes_finalized)!=0
&&(GC_fo_entries - last_fo_entries)
*GC_allocd_bytes_per_finalizer > GC_bytes_allocd)
||GC_should_collect())){
gc_not_stopped=GC_try_to_collect_inner(
GC_bytes_allocd > 0&&(!GC_dont_expand||!retry)?
GC_default_stop_func:GC_never_stop_func);
if (gc_not_stopped==TRUE||!retry){
last_fo_entries=GC_fo_entries;
last_bytes_finalized=GC_bytes_finalized;
RESTORE_CANCEL(cancel_state);
return(TRUE);
}
}
blocks_to_get=(GC_heapsize - GC_heapsize_at_forced_unmap)
/(HBLKSIZE*GC_free_space_divisor)
+needed_blocks;
if (blocks_to_get > MAXHINCR){
word slop;
if (ignore_off_page){
slop=4;
} else {
slop=2*divHBLKSZ(BL_LIMIT);
if (slop > needed_blocks)slop=needed_blocks;
}
if (needed_blocks+slop > MAXHINCR){
blocks_to_get=needed_blocks+slop;
} else {
blocks_to_get=MAXHINCR;
}
if (blocks_to_get > divHBLKSZ(GC_WORD_MAX))
blocks_to_get=divHBLKSZ(GC_WORD_MAX);
}
if (!GC_expand_hp_inner(blocks_to_get)
&&(blocks_to_get==needed_blocks
||!GC_expand_hp_inner(needed_blocks))){
if (gc_not_stopped==FALSE){
GC_gcollect_inner();
GC_ASSERT(GC_bytes_allocd==0);
} else if (GC_fail_count++< GC_max_retries){
WARN("Out of Memory!Trying to continue...\n",0);
GC_gcollect_inner();
} else {
#if!defined(AMIGA)||!defined(GC_AMIGA_FASTALLOC)
WARN("Out of Memory!Heap size:%" WARN_PRIdPTR " MiB."
" Returning NULL!\n",(GC_heapsize - GC_unmapped_bytes)>>20);
#endif
RESTORE_CANCEL(cancel_state);
return(FALSE);
}
} else if (GC_fail_count){
GC_COND_LOG_PRINTF("Memory available again...\n");
}
RESTORE_CANCEL(cancel_state);
return(TRUE);
}
GC_INNER ptr_t GC_allocobj(size_t gran,int kind)
{
void**flh=&(GC_obj_kinds[kind].ok_freelist[gran]);
GC_bool tried_minor=FALSE;
GC_bool retry=FALSE;
GC_ASSERT(I_HOLD_LOCK());
if (gran==0)return(0);
while (*flh==0){
ENTER_GC();
#ifndef GC_DISABLE_INCREMENTAL
if (GC_incremental&&GC_time_limit!=GC_TIME_UNLIMITED){
GC_collect_a_little_inner(1);
}
#endif
GC_ASSERT(!GC_is_full_gc
||NULL==GC_obj_kinds[kind].ok_reclaim_list
||NULL==GC_obj_kinds[kind].ok_reclaim_list[gran]);
GC_continue_reclaim(gran,kind);
EXIT_GC();
#if defined(CPPCHECK)
GC_noop1((word)&flh);
#endif
if (NULL==*flh){
GC_new_hblk(gran,kind);
#if defined(CPPCHECK)
GC_noop1((word)&flh);
#endif
if (NULL==*flh){
ENTER_GC();
if (GC_incremental&&GC_time_limit==GC_TIME_UNLIMITED
&&!tried_minor){
GC_collect_a_little_inner(1);
tried_minor=TRUE;
} else {
if (!GC_collect_or_expand(1,FALSE,retry)){
EXIT_GC();
return(0);
}
retry=TRUE;
}
EXIT_GC();
}
}
}
GC_fail_count=0;
return (ptr_t)(*flh);
}
#ifndef MSWINCE
#include <errno.h>
#endif
#include <string.h>
#ifndef SHORT_DBG_HDRS
GC_INNER int GC_has_other_debug_info(ptr_t p)
{
ptr_t body=(ptr_t)((oh*)p+1);
word sz=GC_size(p);
if (HBLKPTR(p)!=HBLKPTR((ptr_t)body)
||sz < DEBUG_BYTES+EXTRA_BYTES){
return 0;
}
if (((oh*)p)->oh_sf!=(START_FLAG^(word)body)
&&((word*)p)[BYTES_TO_WORDS(sz)-1]!=(END_FLAG^(word)body)){
return 0;
}
if (((oh*)p)->oh_sz==sz){
return -1;
}
return 1;
}
#endif
#ifdef LINT2
long GC_random(void)
{
static unsigned seed=1;
seed=(seed*1103515245U+12345)&GC_RAND_MAX;
return (long)seed;
}
#endif
#ifdef KEEP_BACK_PTRS
#ifdef LINT2
#define RANDOM()GC_random()
#else
#include <stdlib.h>
#define GC_RAND_MAX RAND_MAX
#if defined(__GLIBC__)||defined(SOLARIS)||defined(HPUX)||defined(IRIX5)||defined(OSF1)
#define RANDOM()random()
#else
#define RANDOM()(long)rand()
#endif
#endif
GC_INNER void GC_store_back_pointer(ptr_t source,ptr_t dest)
{
if (GC_HAS_DEBUG_INFO(dest)){
#ifdef PARALLEL_MARK
AO_store((volatile AO_t*)&((oh*)dest)->oh_back_ptr,
(AO_t)HIDE_BACK_PTR(source));
#else
((oh*)dest)->oh_back_ptr=HIDE_BACK_PTR(source);
#endif
}
}
GC_INNER void GC_marked_for_finalization(ptr_t dest)
{
GC_store_back_pointer(MARKED_FOR_FINALIZATION,dest);
}
GC_API GC_ref_kind GC_CALL GC_get_back_ptr_info(void*dest,void**base_p,
size_t*offset_p)
{
oh*hdr=(oh*)GC_base(dest);
ptr_t bp;
ptr_t bp_base;
#ifdef LINT2
if (!hdr)ABORT("Invalid GC_get_back_ptr_info argument");
#endif
if (!GC_HAS_DEBUG_INFO((ptr_t)hdr))return GC_NO_SPACE;
bp=(ptr_t)GC_REVEAL_POINTER(hdr->oh_back_ptr);
if (MARKED_FOR_FINALIZATION==bp)return GC_FINALIZER_REFD;
if (MARKED_FROM_REGISTER==bp)return GC_REFD_FROM_REG;
if (NOT_MARKED==bp)return GC_UNREFERENCED;
#if ALIGNMENT==1
{
ptr_t alternate_ptr=bp+1;
ptr_t target=*(ptr_t*)bp;
ptr_t alternate_target=*(ptr_t*)alternate_ptr;
if ((word)alternate_target>=(word)GC_least_plausible_heap_addr
&&(word)alternate_target<=(word)GC_greatest_plausible_heap_addr
&&((word)target < (word)GC_least_plausible_heap_addr
||(word)target > (word)GC_greatest_plausible_heap_addr)){
bp=alternate_ptr;
}
}
#endif
bp_base=(ptr_t)GC_base(bp);
if (NULL==bp_base){
*base_p=bp;
*offset_p=0;
return GC_REFD_FROM_ROOT;
} else {
if (GC_HAS_DEBUG_INFO(bp_base))bp_base+=sizeof(oh);
*base_p=bp_base;
*offset_p=bp - bp_base;
return GC_REFD_FROM_HEAP;
}
}
GC_API void*GC_CALL GC_generate_random_heap_address(void)
{
size_t i;
word heap_offset=RANDOM();
if (GC_heapsize > GC_RAND_MAX){
heap_offset*=GC_RAND_MAX;
heap_offset+=RANDOM();
}
heap_offset%=GC_heapsize;
for (i=0;;++i){
size_t size;
if (i>=GC_n_heap_sects)
ABORT("GC_generate_random_heap_address:size inconsistency");
size=GC_heap_sects[i].hs_bytes;
if (heap_offset < size){
break;
} else {
heap_offset-=size;
}
}
return GC_heap_sects[i].hs_start+heap_offset;
}
GC_API void*GC_CALL GC_generate_random_valid_address(void)
{
ptr_t result;
ptr_t base;
do {
result=(ptr_t)GC_generate_random_heap_address();
base=(ptr_t)GC_base(result);
} while (NULL==base||!GC_is_marked(base));
return result;
}
GC_API void GC_CALL GC_print_backtrace(void*p)
{
void*current=p;
int i;
GC_ref_kind source;
size_t offset;
void*base;
GC_print_heap_obj((ptr_t)GC_base(current));
for (i=0;;++i){
source=GC_get_back_ptr_info(current,&base,&offset);
if (GC_UNREFERENCED==source){
GC_err_printf("Reference could not be found\n");
goto out;
}
if (GC_NO_SPACE==source){
GC_err_printf("No debug info in object:Can't find reference\n");
goto out;
}
GC_err_printf("Reachable via %d levels of pointers from ",i);
switch(source){
case GC_REFD_FROM_ROOT:
GC_err_printf("root at %p\n\n",base);
goto out;
case GC_REFD_FROM_REG:
GC_err_printf("root in register\n\n");
goto out;
case GC_FINALIZER_REFD:
GC_err_printf("list of finalizable objects\n\n");
goto out;
case GC_REFD_FROM_HEAP:
GC_err_printf("offset %ld in object:\n",(long)offset);
GC_print_heap_obj((ptr_t)GC_base(base));
break;
default:
GC_err_printf("INTERNAL ERROR:UNEXPECTED SOURCE!!!!\n");
goto out;
}
current=base;
}
out:;
}
GC_INNER void GC_generate_random_backtrace_no_gc(void)
{
void*current;
current=GC_generate_random_valid_address();
GC_printf("\n****Chosen address %p in object\n",current);
GC_print_backtrace(current);
}
GC_API void GC_CALL GC_generate_random_backtrace(void)
{
if (GC_try_to_collect(GC_never_stop_func)==0){
GC_err_printf("Cannot generate a backtrace:"
"garbage collection is disabled!\n");
return;
}
GC_generate_random_backtrace_no_gc();
}
#endif
#define CROSSES_HBLK(p,sz)(((word)((p)+sizeof(oh)+(sz)- 1)^(word)(p))>=HBLKSIZE)
GC_INNER void*GC_store_debug_info_inner(void*p,word sz GC_ATTR_UNUSED,
const char*string,int linenum)
{
word*result=(word*)((oh*)p+1);
GC_ASSERT(I_HOLD_LOCK());
GC_ASSERT(GC_size(p)>=sizeof(oh)+sz);
GC_ASSERT(!(SMALL_OBJ(sz)&&CROSSES_HBLK((ptr_t)p,sz)));
#ifdef KEEP_BACK_PTRS
((oh*)p)->oh_back_ptr=HIDE_BACK_PTR(NOT_MARKED);
#endif
#ifdef MAKE_BACK_GRAPH
((oh*)p)->oh_bg_ptr=HIDE_BACK_PTR((ptr_t)0);
#endif
((oh*)p)->oh_string=string;
((oh*)p)->oh_int=linenum;
#ifndef SHORT_DBG_HDRS
((oh*)p)->oh_sz=sz;
((oh*)p)->oh_sf=START_FLAG^(word)result;
((word*)p)[BYTES_TO_WORDS(GC_size(p))-1]=
result[SIMPLE_ROUNDED_UP_WORDS(sz)]=END_FLAG^(word)result;
#endif
return result;
}
static void*store_debug_info(void*p,size_t lb,
const char*fn,GC_EXTRA_PARAMS)
{
void*result;
DCL_LOCK_STATE;
if (NULL==p){
GC_err_printf("%s(%lu)returning NULL (%s:%d)\n",
fn,(unsigned long)lb,s,i);
return NULL;
}
LOCK();
if (!GC_debugging_started)
GC_start_debugging_inner();
ADD_CALL_CHAIN(p,ra);
result=GC_store_debug_info_inner(p,(word)lb,s,i);
UNLOCK();
return result;
}
#ifndef SHORT_DBG_HDRS
STATIC ptr_t GC_check_annotated_obj(oh*ohdr)
{
ptr_t body=(ptr_t)(ohdr+1);
word gc_sz=GC_size((ptr_t)ohdr);
if (ohdr->oh_sz+DEBUG_BYTES > gc_sz){
return((ptr_t)(&(ohdr->oh_sz)));
}
if (ohdr->oh_sf!=(START_FLAG^(word)body)){
return((ptr_t)(&(ohdr->oh_sf)));
}
if (((word*)ohdr)[BYTES_TO_WORDS(gc_sz)-1]!=(END_FLAG^(word)body)){
return (ptr_t)(&((word*)ohdr)[BYTES_TO_WORDS(gc_sz)-1]);
}
if (((word*)body)[SIMPLE_ROUNDED_UP_WORDS(ohdr->oh_sz)]
!=(END_FLAG^(word)body)){
return (ptr_t)(&((word*)body)[SIMPLE_ROUNDED_UP_WORDS(ohdr->oh_sz)]);
}
return(0);
}
#endif
STATIC GC_describe_type_fn GC_describe_type_fns[MAXOBJKINDS]={0};
GC_API void GC_CALL GC_register_describe_type_fn(int kind,
GC_describe_type_fn fn)
{
GC_describe_type_fns[kind]=fn;
}
#define GET_OH_LINENUM(ohdr)((int)(ohdr)->oh_int)
#ifndef SHORT_DBG_HDRS
#define IF_NOT_SHORTDBG_HDRS(x)x
#define COMMA_IFNOT_SHORTDBG_HDRS(x),x
#else
#define IF_NOT_SHORTDBG_HDRS(x)
#define COMMA_IFNOT_SHORTDBG_HDRS(x)
#endif
STATIC void GC_print_obj(ptr_t p)
{
oh*ohdr=(oh*)GC_base(p);
ptr_t q;
hdr*hhdr;
int kind;
const char*kind_str;
char buffer[GC_TYPE_DESCR_LEN+1];
GC_ASSERT(I_DONT_HOLD_LOCK());
#ifdef LINT2
if (!ohdr)ABORT("Invalid GC_print_obj argument");
#endif
q=(ptr_t)(ohdr+1);
hhdr=GC_find_header(q);
kind=hhdr->hb_obj_kind;
if (0!=GC_describe_type_fns[kind]&&GC_is_marked(ohdr)){
buffer[GC_TYPE_DESCR_LEN]=0;
(GC_describe_type_fns[kind])(q,buffer);
GC_ASSERT(buffer[GC_TYPE_DESCR_LEN]==0);
kind_str=buffer;
} else {
switch(kind){
case PTRFREE:
kind_str="PTRFREE";
break;
case NORMAL:
kind_str="NORMAL";
break;
case UNCOLLECTABLE:
kind_str="UNCOLLECTABLE";
break;
#ifdef GC_ATOMIC_UNCOLLECTABLE
case AUNCOLLECTABLE:
kind_str="ATOMIC_UNCOLLECTABLE";
break;
#endif
default:
kind_str=NULL;
}
}
if (NULL!=kind_str){
GC_err_printf("%p (%s:%d," IF_NOT_SHORTDBG_HDRS(" sz=%lu,")" %s)\n",
(void*)((ptr_t)ohdr+sizeof(oh)),
ohdr->oh_string,GET_OH_LINENUM(ohdr)
COMMA_IFNOT_SHORTDBG_HDRS((unsigned long)ohdr->oh_sz),
kind_str);
} else {
GC_err_printf("%p (%s:%d," IF_NOT_SHORTDBG_HDRS(" sz=%lu,")
" kind=%d descr=0x%lx)\n",
(void*)((ptr_t)ohdr+sizeof(oh)),
ohdr->oh_string,GET_OH_LINENUM(ohdr)
COMMA_IFNOT_SHORTDBG_HDRS((unsigned long)ohdr->oh_sz),
kind,(unsigned long)hhdr->hb_descr);
}
PRINT_CALL_CHAIN(ohdr);
}
STATIC void GC_debug_print_heap_obj_proc(ptr_t p)
{
GC_ASSERT(I_DONT_HOLD_LOCK());
if (GC_HAS_DEBUG_INFO(p)){
GC_print_obj(p);
} else {
GC_default_print_heap_obj_proc(p);
}
}
#ifndef SHORT_DBG_HDRS
STATIC void GC_print_smashed_obj(const char*msg,void*p,
ptr_t clobbered_addr)
{
oh*ohdr=(oh*)GC_base(p);
GC_ASSERT(I_DONT_HOLD_LOCK());
#ifdef LINT2
if (!ohdr)ABORT("Invalid GC_print_smashed_obj argument");
#endif
if ((word)clobbered_addr<=(word)(&ohdr->oh_sz)
||ohdr->oh_string==0){
GC_err_printf(
"%s %p in or near object at %p(<smashed>,appr. sz=%lu)\n",
msg,(void*)clobbered_addr,p,
(unsigned long)(GC_size((ptr_t)ohdr)- DEBUG_BYTES));
} else {
GC_err_printf("%s %p in or near object at %p (%s:%d,sz=%lu)\n",
msg,(void*)clobbered_addr,p,
(word)(ohdr->oh_string)< HBLKSIZE?"(smashed string)":
ohdr->oh_string[0]=='\0'?"EMPTY(smashed?)":
ohdr->oh_string,
GET_OH_LINENUM(ohdr),(unsigned long)(ohdr->oh_sz));
PRINT_CALL_CHAIN(ohdr);
}
}
STATIC void GC_check_heap_proc (void);
STATIC void GC_print_all_smashed_proc (void);
#else
STATIC void GC_do_nothing(void){}
#endif
GC_INNER void GC_start_debugging_inner(void)
{
GC_ASSERT(I_HOLD_LOCK());
#ifndef SHORT_DBG_HDRS
GC_check_heap=GC_check_heap_proc;
GC_print_all_smashed=GC_print_all_smashed_proc;
#else
GC_check_heap=GC_do_nothing;
GC_print_all_smashed=GC_do_nothing;
#endif
GC_print_heap_obj=GC_debug_print_heap_obj_proc;
GC_debugging_started=TRUE;
GC_register_displacement_inner((word)sizeof(oh));
#if defined(CPPCHECK)
GC_noop1(GC_debug_header_size);
#endif
}
const size_t GC_debug_header_size=sizeof(oh);
GC_API size_t GC_CALL GC_get_debug_header_size(void){
return sizeof(oh);
}
GC_API void GC_CALL GC_debug_register_displacement(size_t offset)
{
DCL_LOCK_STATE;
LOCK();
GC_register_displacement_inner(offset);
GC_register_displacement_inner((word)sizeof(oh)+offset);
UNLOCK();
}
#ifdef GC_ADD_CALLER
#if defined(HAVE_DLADDR)&&defined(GC_HAVE_RETURN_ADDR_PARENT)
#include <dlfcn.h>
STATIC void GC_caller_func_offset(word ad,const char**symp,int*offp)
{
Dl_info caller;
if (ad&&dladdr((void*)ad,&caller)&&caller.dli_sname!=NULL){
*symp=caller.dli_sname;
*offp=(int)((char*)ad - (char*)caller.dli_saddr);
}
if (NULL==*symp){
*symp="unknown";
}
}
#else
#define GC_caller_func_offset(ad,symp,offp)(void)(*(symp)="unknown")
#endif
#endif
GC_API GC_ATTR_MALLOC void*GC_CALL GC_debug_malloc(size_t lb,
GC_EXTRA_PARAMS)
{
void*result;
result=GC_malloc(SIZET_SAT_ADD(lb,DEBUG_BYTES));
#ifdef GC_ADD_CALLER
if (s==NULL){
GC_caller_func_offset(ra,&s,&i);
}
#endif
return store_debug_info(result,lb,"GC_debug_malloc",OPT_RA s,i);
}
GC_API GC_ATTR_MALLOC void*GC_CALL
GC_debug_malloc_ignore_off_page(size_t lb,GC_EXTRA_PARAMS)
{
void*result=GC_malloc_ignore_off_page(SIZET_SAT_ADD(lb,DEBUG_BYTES));
return store_debug_info(result,lb,"GC_debug_malloc_ignore_off_page",
OPT_RA s,i);
}
GC_API GC_ATTR_MALLOC void*GC_CALL
GC_debug_malloc_atomic_ignore_off_page(size_t lb,GC_EXTRA_PARAMS)
{
void*result=GC_malloc_atomic_ignore_off_page(
SIZET_SAT_ADD(lb,DEBUG_BYTES));
return store_debug_info(result,lb,
"GC_debug_malloc_atomic_ignore_off_page",
OPT_RA s,i);
}
STATIC void*GC_debug_generic_malloc(size_t lb,int knd,GC_EXTRA_PARAMS)
{
void*result=GC_generic_malloc(SIZET_SAT_ADD(lb,DEBUG_BYTES),knd);
return store_debug_info(result,lb,"GC_debug_generic_malloc",
OPT_RA s,i);
}
#ifdef DBG_HDRS_ALL
GC_INNER void*GC_debug_generic_malloc_inner(size_t lb,int k)
{
void*result;
GC_ASSERT(I_HOLD_LOCK());
result=GC_generic_malloc_inner(SIZET_SAT_ADD(lb,DEBUG_BYTES),k);
if (NULL==result){
GC_err_printf("GC internal allocation (%lu bytes)returning NULL\n",
(unsigned long)lb);
return(0);
}
if (!GC_debugging_started){
GC_start_debugging_inner();
}
ADD_CALL_CHAIN(result,GC_RETURN_ADDR);
return (GC_store_debug_info_inner(result,(word)lb,"INTERNAL",0));
}
GC_INNER void*GC_debug_generic_malloc_inner_ignore_off_page(size_t lb,
int k)
{
void*result;
GC_ASSERT(I_HOLD_LOCK());
result=GC_generic_malloc_inner_ignore_off_page(
SIZET_SAT_ADD(lb,DEBUG_BYTES),k);
if (NULL==result){
GC_err_printf("GC internal allocation (%lu bytes)returning NULL\n",
(unsigned long)lb);
return(0);
}
if (!GC_debugging_started){
GC_start_debugging_inner();
}
ADD_CALL_CHAIN(result,GC_RETURN_ADDR);
return (GC_store_debug_info_inner(result,(word)lb,"INTERNAL",0));
}
#endif
#ifndef CPPCHECK
GC_API void*GC_CALL GC_debug_malloc_stubborn(size_t lb,GC_EXTRA_PARAMS)
{
return GC_debug_malloc(lb,OPT_RA s,i);
}
GC_API void GC_CALL GC_debug_change_stubborn(
const void*p GC_ATTR_UNUSED){}
#endif
GC_API void GC_CALL GC_debug_end_stubborn_change(const void*p)
{
const void*q=GC_base_C(p);
if (NULL==q){
ABORT_ARG1("GC_debug_end_stubborn_change:bad arg",":%p",p);
}
GC_end_stubborn_change(q);
}
GC_API void GC_CALL GC_debug_ptr_store_and_dirty(void*p,const void*q)
{
*(void**)GC_is_visible(p)=GC_is_valid_displacement((void*)q);
GC_debug_end_stubborn_change(p);
REACHABLE_AFTER_DIRTY(q);
}
GC_API GC_ATTR_MALLOC void*GC_CALL GC_debug_malloc_atomic(size_t lb,
GC_EXTRA_PARAMS)
{
void*result=GC_malloc_atomic(SIZET_SAT_ADD(lb,DEBUG_BYTES));
return store_debug_info(result,lb,"GC_debug_malloc_atomic",
OPT_RA s,i);
}
GC_API GC_ATTR_MALLOC char*GC_CALL GC_debug_strdup(const char*str,
GC_EXTRA_PARAMS)
{
char*copy;
size_t lb;
if (str==NULL){
if (GC_find_leak)
GC_err_printf("strdup(NULL)behavior is undefined\n");
return NULL;
}
lb=strlen(str)+1;
copy=(char*)GC_debug_malloc_atomic(lb,OPT_RA s,i);
if (copy==NULL){
#ifndef MSWINCE
errno=ENOMEM;
#endif
return NULL;
}
BCOPY(str,copy,lb);
return copy;
}
GC_API GC_ATTR_MALLOC char*GC_CALL GC_debug_strndup(const char*str,
size_t size,GC_EXTRA_PARAMS)
{
char*copy;
size_t len=strlen(str);
if (len > size)
len=size;
copy=(char*)GC_debug_malloc_atomic(len+1,OPT_RA s,i);
if (copy==NULL){
#ifndef MSWINCE
errno=ENOMEM;
#endif
return NULL;
}
if (len > 0)
BCOPY(str,copy,len);
copy[len]='\0';
return copy;
}
#ifdef GC_REQUIRE_WCSDUP
#include <wchar.h>
GC_API GC_ATTR_MALLOC wchar_t*GC_CALL GC_debug_wcsdup(const wchar_t*str,
GC_EXTRA_PARAMS)
{
size_t lb=(wcslen(str)+1)*sizeof(wchar_t);
wchar_t*copy=(wchar_t*)GC_debug_malloc_atomic(lb,OPT_RA s,i);
if (copy==NULL){
#ifndef MSWINCE
errno=ENOMEM;
#endif
return NULL;
}
BCOPY(str,copy,lb);
return copy;
}
#endif
GC_API GC_ATTR_MALLOC void*GC_CALL GC_debug_malloc_uncollectable(size_t lb,
GC_EXTRA_PARAMS)
{
void*result=GC_malloc_uncollectable(
SIZET_SAT_ADD(lb,UNCOLLECTABLE_DEBUG_BYTES));
return store_debug_info(result,lb,"GC_debug_malloc_uncollectable",
OPT_RA s,i);
}
#ifdef GC_ATOMIC_UNCOLLECTABLE
GC_API GC_ATTR_MALLOC void*GC_CALL
GC_debug_malloc_atomic_uncollectable(size_t lb,GC_EXTRA_PARAMS)
{
void*result=GC_malloc_atomic_uncollectable(
SIZET_SAT_ADD(lb,UNCOLLECTABLE_DEBUG_BYTES));
return store_debug_info(result,lb,
"GC_debug_malloc_atomic_uncollectable",
OPT_RA s,i);
}
#endif
#ifndef GC_FREED_MEM_MARKER
#if CPP_WORDSZ==32
#define GC_FREED_MEM_MARKER 0xdeadbeef
#else
#define GC_FREED_MEM_MARKER GC_WORD_C(0xEFBEADDEdeadbeef)
#endif
#endif
GC_API void GC_CALL GC_debug_free(void*p)
{
ptr_t base;
if (0==p)return;
base=(ptr_t)GC_base(p);
if (NULL==base){
#if defined(REDIRECT_MALLOC)&&((defined(NEED_CALLINFO)&&defined(GC_HAVE_BUILTIN_BACKTRACE))||defined(GC_LINUX_THREADS)||defined(GC_SOLARIS_THREADS)||defined(MSWIN32))
if (!GC_is_heap_ptr(p))return;
#endif
ABORT_ARG1("Invalid pointer passed to free()",":%p",p);
}
if ((ptr_t)p - (ptr_t)base!=sizeof(oh)){
#if defined(REDIRECT_FREE)&&defined(USE_PROC_FOR_LIBRARIES)
#endif
GC_err_printf(
"GC_debug_free called on pointer %p w/o debugging info\n",p);
} else {
#ifndef SHORT_DBG_HDRS
ptr_t clobbered=GC_check_annotated_obj((oh*)base);
word sz=GC_size(base);
if (clobbered!=0){
GC_have_errors=TRUE;
if (((oh*)base)->oh_sz==sz){
GC_print_smashed_obj(
"GC_debug_free:found previously deallocated (?)object at",
p,clobbered);
return;
} else {
GC_print_smashed_obj("GC_debug_free:found smashed location at",
p,clobbered);
}
}
((oh*)base)->oh_sz=sz;
#endif
}
if (GC_find_leak
#ifndef SHORT_DBG_HDRS
&&((ptr_t)p - (ptr_t)base!=sizeof(oh)||!GC_findleak_delay_free)
#endif
){
GC_free(base);
} else {
hdr*hhdr=HDR(p);
if (hhdr->hb_obj_kind==UNCOLLECTABLE
#ifdef GC_ATOMIC_UNCOLLECTABLE
||hhdr->hb_obj_kind==AUNCOLLECTABLE
#endif
){
GC_free(base);
} else {
word i;
word sz=hhdr->hb_sz;
word obj_sz=BYTES_TO_WORDS(sz - sizeof(oh));
for (i=0;i < obj_sz;++i)
((word*)p)[i]=GC_FREED_MEM_MARKER;
GC_ASSERT((word*)p+i==(word*)(base+sz));
LOCK();
GC_bytes_freed+=sz;
UNLOCK();
}
}
}
#if defined(THREADS)&&defined(DBG_HDRS_ALL)
GC_INNER void GC_debug_free_inner(void*p)
{
ptr_t base=(ptr_t)GC_base(p);
GC_ASSERT((ptr_t)p - (ptr_t)base==sizeof(oh));
#ifdef LINT2
if (!base)ABORT("Invalid GC_debug_free_inner argument");
#endif
#ifndef SHORT_DBG_HDRS
((oh*)base)->oh_sz=GC_size(base);
#endif
GC_free_inner(base);
}
#endif
GC_API void*GC_CALL GC_debug_realloc(void*p,size_t lb,GC_EXTRA_PARAMS)
{
void*base;
void*result;
hdr*hhdr;
if (p==0){
return GC_debug_malloc(lb,OPT_RA s,i);
}
if (0==lb){
GC_debug_free(p);
return NULL;
}
#ifdef GC_ADD_CALLER
if (s==NULL){
GC_caller_func_offset(ra,&s,&i);
}
#endif
base=GC_base(p);
if (base==0){
ABORT_ARG1("Invalid pointer passed to realloc()",":%p",p);
}
if ((ptr_t)p - (ptr_t)base!=sizeof(oh)){
GC_err_printf(
"GC_debug_realloc called on pointer %p w/o debugging info\n",p);
return(GC_realloc(p,lb));
}
hhdr=HDR(base);
switch (hhdr->hb_obj_kind){
case NORMAL:
result=GC_debug_malloc(lb,OPT_RA s,i);
break;
case PTRFREE:
result=GC_debug_malloc_atomic(lb,OPT_RA s,i);
break;
case UNCOLLECTABLE:
result=GC_debug_malloc_uncollectable(lb,OPT_RA s,i);
break;
#ifdef GC_ATOMIC_UNCOLLECTABLE
case AUNCOLLECTABLE:
result=GC_debug_malloc_atomic_uncollectable(lb,OPT_RA s,i);
break;
#endif
default:
result=NULL;
ABORT_RET("GC_debug_realloc:encountered bad kind");
}
if (result!=NULL){
size_t old_sz;
#ifdef SHORT_DBG_HDRS
old_sz=GC_size(base)- sizeof(oh);
#else
old_sz=((oh*)base)->oh_sz;
#endif
if (old_sz > 0)
BCOPY(p,result,old_sz < lb?old_sz:lb);
GC_debug_free(p);
}
return(result);
}
GC_API GC_ATTR_MALLOC void*GC_CALL
GC_debug_generic_or_special_malloc(size_t lb,int knd,GC_EXTRA_PARAMS)
{
switch (knd){
case PTRFREE:
return GC_debug_malloc_atomic(lb,OPT_RA s,i);
case NORMAL:
return GC_debug_malloc(lb,OPT_RA s,i);
case UNCOLLECTABLE:
return GC_debug_malloc_uncollectable(lb,OPT_RA s,i);
#ifdef GC_ATOMIC_UNCOLLECTABLE
case AUNCOLLECTABLE:
return GC_debug_malloc_atomic_uncollectable(lb,OPT_RA s,i);
#endif
default:
return GC_debug_generic_malloc(lb,knd,OPT_RA s,i);
}
}
#ifndef SHORT_DBG_HDRS
#ifndef MAX_SMASHED
#define MAX_SMASHED 20
#endif
STATIC ptr_t GC_smashed[MAX_SMASHED]={0};
STATIC unsigned GC_n_smashed=0;
STATIC void GC_add_smashed(ptr_t smashed)
{
GC_ASSERT(GC_is_marked(GC_base(smashed)));
GC_smashed[GC_n_smashed]=smashed;
if (GC_n_smashed < MAX_SMASHED - 1)++GC_n_smashed;
GC_have_errors=TRUE;
}
STATIC void GC_print_all_smashed_proc(void)
{
unsigned i;
GC_ASSERT(I_DONT_HOLD_LOCK());
if (GC_n_smashed==0)return;
GC_err_printf("GC_check_heap_block:found %u smashed heap objects:\n",
GC_n_smashed);
for (i=0;i < GC_n_smashed;++i){
ptr_t base=(ptr_t)GC_base(GC_smashed[i]);
#ifdef LINT2
if (!base)ABORT("Invalid GC_smashed element");
#endif
GC_print_smashed_obj("",base+sizeof(oh),GC_smashed[i]);
GC_smashed[i]=0;
}
GC_n_smashed=0;
}
STATIC void GC_check_heap_block(struct hblk*hbp,word dummy GC_ATTR_UNUSED)
{
struct hblkhdr*hhdr=HDR(hbp);
word sz=hhdr->hb_sz;
word bit_no;
char*p,*plim;
p=hbp->hb_body;
if (sz > MAXOBJBYTES){
plim=p;
} else {
plim=hbp->hb_body+HBLKSIZE - sz;
}
for (bit_no=0;(word)p<=(word)plim;
bit_no+=MARK_BIT_OFFSET(sz),p+=sz){
if (mark_bit_from_hdr(hhdr,bit_no)&&GC_HAS_DEBUG_INFO((ptr_t)p)){
ptr_t clobbered=GC_check_annotated_obj((oh*)p);
if (clobbered!=0)
GC_add_smashed(clobbered);
}
}
}
STATIC void GC_check_heap_proc(void)
{
GC_STATIC_ASSERT((sizeof(oh)&(GRANULE_BYTES - 1))==0);
GC_apply_to_all_blocks(GC_check_heap_block,0);
}
GC_INNER GC_bool GC_check_leaked(ptr_t base)
{
word i;
word obj_sz;
word*p;
if (
#if defined(KEEP_BACK_PTRS)||defined(MAKE_BACK_GRAPH)
(*(word*)base&1)!=0&&
#endif
GC_has_other_debug_info(base)>=0)
return TRUE;
p=(word*)(base+sizeof(oh));
obj_sz=BYTES_TO_WORDS(HDR(base)->hb_sz - sizeof(oh));
for (i=0;i < obj_sz;++i)
if (p[i]!=GC_FREED_MEM_MARKER){
GC_set_mark_bit(base);
GC_add_smashed((ptr_t)(&p[i]));
break;
}
return FALSE;
}
#endif
#ifndef GC_NO_FINALIZATION
struct closure {
GC_finalization_proc cl_fn;
void*cl_data;
};
STATIC void*GC_make_closure(GC_finalization_proc fn,void*data)
{
struct closure*result=
#ifdef DBG_HDRS_ALL
(struct closure*)GC_debug_malloc(sizeof (struct closure),
GC_EXTRAS);
#else
(struct closure*)GC_malloc(sizeof (struct closure));
#endif
if (result!=0){
result->cl_fn=fn;
result->cl_data=data;
}
return((void*)result);
}
STATIC void GC_CALLBACK GC_debug_invoke_finalizer(void*obj,void*data)
{
struct closure*cl=(struct closure*)data;
(*(cl->cl_fn))((void*)((char*)obj+sizeof(oh)),cl->cl_data);
}
#define OFN_UNSET ((GC_finalization_proc)~(signed_word)0)
static void store_old(void*obj,GC_finalization_proc my_old_fn,
struct closure*my_old_cd,GC_finalization_proc*ofn,
void**ocd)
{
if (0!=my_old_fn){
if (my_old_fn==OFN_UNSET){
return;
}
if (my_old_fn!=GC_debug_invoke_finalizer){
GC_err_printf("Debuggable object at %p had a non-debug finalizer\n",
obj);
} else {
if (ofn)*ofn=my_old_cd->cl_fn;
if (ocd)*ocd=my_old_cd->cl_data;
}
} else {
if (ofn)*ofn=0;
if (ocd)*ocd=0;
}
}
GC_API void GC_CALL GC_debug_register_finalizer(void*obj,
GC_finalization_proc fn,
void*cd,GC_finalization_proc*ofn,
void**ocd)
{
GC_finalization_proc my_old_fn=OFN_UNSET;
void*my_old_cd;
ptr_t base=(ptr_t)GC_base(obj);
if (NULL==base){
if (ocd)*ocd=0;
if (ofn)*ofn=0;
return;
}
if ((ptr_t)obj - base!=sizeof(oh)){
GC_err_printf("GC_debug_register_finalizer called with"
" non-base-pointer %p\n",obj);
}
if (0==fn){
GC_register_finalizer(base,0,0,&my_old_fn,&my_old_cd);
} else {
cd=GC_make_closure(fn,cd);
if (cd==0)return;
GC_register_finalizer(base,GC_debug_invoke_finalizer,
cd,&my_old_fn,&my_old_cd);
}
store_old(obj,my_old_fn,(struct closure*)my_old_cd,ofn,ocd);
}
GC_API void GC_CALL GC_debug_register_finalizer_no_order
(void*obj,GC_finalization_proc fn,
void*cd,GC_finalization_proc*ofn,
void**ocd)
{
GC_finalization_proc my_old_fn=OFN_UNSET;
void*my_old_cd;
ptr_t base=(ptr_t)GC_base(obj);
if (NULL==base){
if (ocd)*ocd=0;
if (ofn)*ofn=0;
return;
}
if ((ptr_t)obj - base!=sizeof(oh)){
GC_err_printf("GC_debug_register_finalizer_no_order called with"
" non-base-pointer %p\n",obj);
}
if (0==fn){
GC_register_finalizer_no_order(base,0,0,&my_old_fn,&my_old_cd);
} else {
cd=GC_make_closure(fn,cd);
if (cd==0)return;
GC_register_finalizer_no_order(base,GC_debug_invoke_finalizer,
cd,&my_old_fn,&my_old_cd);
}
store_old(obj,my_old_fn,(struct closure*)my_old_cd,ofn,ocd);
}
GC_API void GC_CALL GC_debug_register_finalizer_unreachable
(void*obj,GC_finalization_proc fn,
void*cd,GC_finalization_proc*ofn,
void**ocd)
{
GC_finalization_proc my_old_fn=OFN_UNSET;
void*my_old_cd;
ptr_t base=(ptr_t)GC_base(obj);
if (NULL==base){
if (ocd)*ocd=0;
if (ofn)*ofn=0;
return;
}
if ((ptr_t)obj - base!=sizeof(oh)){
GC_err_printf("GC_debug_register_finalizer_unreachable called with"
" non-base-pointer %p\n",obj);
}
if (0==fn){
GC_register_finalizer_unreachable(base,0,0,&my_old_fn,&my_old_cd);
} else {
cd=GC_make_closure(fn,cd);
if (cd==0)return;
GC_register_finalizer_unreachable(base,GC_debug_invoke_finalizer,
cd,&my_old_fn,&my_old_cd);
}
store_old(obj,my_old_fn,(struct closure*)my_old_cd,ofn,ocd);
}
GC_API void GC_CALL GC_debug_register_finalizer_ignore_self
(void*obj,GC_finalization_proc fn,
void*cd,GC_finalization_proc*ofn,
void**ocd)
{
GC_finalization_proc my_old_fn=OFN_UNSET;
void*my_old_cd;
ptr_t base=(ptr_t)GC_base(obj);
if (NULL==base){
if (ocd)*ocd=0;
if (ofn)*ofn=0;
return;
}
if ((ptr_t)obj - base!=sizeof(oh)){
GC_err_printf("GC_debug_register_finalizer_ignore_self called with"
" non-base-pointer %p\n",obj);
}
if (0==fn){
GC_register_finalizer_ignore_self(base,0,0,&my_old_fn,&my_old_cd);
} else {
cd=GC_make_closure(fn,cd);
if (cd==0)return;
GC_register_finalizer_ignore_self(base,GC_debug_invoke_finalizer,
cd,&my_old_fn,&my_old_cd);
}
store_old(obj,my_old_fn,(struct closure*)my_old_cd,ofn,ocd);
}
#endif
GC_API GC_ATTR_MALLOC void*GC_CALL GC_debug_malloc_replacement(size_t lb)
{
return GC_debug_malloc(lb,GC_DBG_EXTRAS);
}
GC_API void*GC_CALL GC_debug_realloc_replacement(void*p,size_t lb)
{
return GC_debug_realloc(p,lb,GC_DBG_EXTRAS);
}
#ifndef GC_NO_FINALIZATION
#ifndef GC_JAVAXFC_H
#define GC_JAVAXFC_H
#ifndef GC_H
#endif
#ifdef __cplusplus
extern "C" {
#endif
GC_API void GC_CALL GC_finalize_all(void);
#ifdef GC_THREADS
#ifndef GC_SUSPEND_THREAD_ID
#define GC_SUSPEND_THREAD_ID void*
#endif
GC_API void GC_CALL GC_suspend_thread(GC_SUSPEND_THREAD_ID);
GC_API void GC_CALL GC_resume_thread(GC_SUSPEND_THREAD_ID);
GC_API int GC_CALL GC_is_thread_suspended(GC_SUSPEND_THREAD_ID);
#endif
#ifdef __cplusplus
}
#endif
#endif
typedef void (*finalization_mark_proc)(ptr_t);
#define HASH3(addr,size,log_size)((((word)(addr)>>3)^((word)(addr)>>(3+(log_size))))&((size)- 1))
#define HASH2(addr,log_size)HASH3(addr,(word)1<<(log_size),log_size)
struct hash_chain_entry {
word hidden_key;
struct hash_chain_entry*next;
};
struct disappearing_link {
struct hash_chain_entry prolog;
#define dl_hidden_link prolog.hidden_key
#define dl_next(x)(struct disappearing_link*)((x)->prolog.next)
#define dl_set_next(x,y)(void)((x)->prolog.next=(struct hash_chain_entry*)(y))
word dl_hidden_obj;
};
struct finalizable_object {
struct hash_chain_entry prolog;
#define fo_hidden_base prolog.hidden_key
#define fo_next(x)(struct finalizable_object*)((x)->prolog.next)
#define fo_set_next(x,y)((x)->prolog.next=(struct hash_chain_entry*)(y))
GC_finalization_proc fo_fn;
ptr_t fo_client_data;
word fo_object_size;
finalization_mark_proc fo_mark_proc;
};
#ifdef AO_HAVE_store
#define SET_FINALIZE_NOW(fo)AO_store((volatile AO_t*)&GC_fnlz_roots.finalize_now,(AO_t)(fo))
#else
#define SET_FINALIZE_NOW(fo)(void)(GC_fnlz_roots.finalize_now=(fo))
#endif
GC_API void GC_CALL GC_push_finalizer_structures(void)
{
GC_ASSERT((word)(&GC_dl_hashtbl.head)% sizeof(word)==0);
GC_ASSERT((word)(&GC_fnlz_roots)% sizeof(word)==0);
#ifndef GC_LONG_REFS_NOT_NEEDED
GC_ASSERT((word)(&GC_ll_hashtbl.head)% sizeof(word)==0);
GC_PUSH_ALL_SYM(GC_ll_hashtbl.head);
#endif
GC_PUSH_ALL_SYM(GC_dl_hashtbl.head);
GC_PUSH_ALL_SYM(GC_fnlz_roots);
}
#ifndef GC_ON_GROW_LOG_SIZE_MIN
#define GC_ON_GROW_LOG_SIZE_MIN CPP_LOG_HBLKSIZE
#endif
STATIC void GC_grow_table(struct hash_chain_entry***table,
unsigned*log_size_ptr,word*entries_ptr)
{
word i;
struct hash_chain_entry*p;
unsigned log_old_size=*log_size_ptr;
unsigned log_new_size=log_old_size+1;
word old_size=*table==NULL?0:(word)1<<log_old_size;
word new_size=(word)1<<log_new_size;
struct hash_chain_entry**new_table;
GC_ASSERT(I_HOLD_LOCK());
if (log_old_size>=GC_ON_GROW_LOG_SIZE_MIN&&!GC_incremental){
IF_CANCEL(int cancel_state;)
DISABLE_CANCEL(cancel_state);
(void)GC_try_to_collect_inner(GC_never_stop_func);
RESTORE_CANCEL(cancel_state);
if (*entries_ptr < ((word)1<<log_old_size)- (*entries_ptr>>2))
return;
}
new_table=(struct hash_chain_entry**)
GC_INTERNAL_MALLOC_IGNORE_OFF_PAGE(
(size_t)new_size*sizeof(struct hash_chain_entry*),
NORMAL);
if (new_table==0){
if (*table==0){
ABORT("Insufficient space for initial table allocation");
} else {
return;
}
}
for (i=0;i < old_size;i++){
p=(*table)[i];
while (p!=0){
ptr_t real_key=(ptr_t)GC_REVEAL_POINTER(p->hidden_key);
struct hash_chain_entry*next=p->next;
size_t new_hash=HASH3(real_key,new_size,log_new_size);
p->next=new_table[new_hash];
GC_dirty(p);
new_table[new_hash]=p;
p=next;
}
}
*log_size_ptr=log_new_size;
*table=new_table;
GC_dirty(new_table);
}
GC_API int GC_CALL GC_register_disappearing_link(void**link)
{
ptr_t base;
base=(ptr_t)GC_base(link);
if (base==0)
ABORT("Bad arg to GC_register_disappearing_link");
return(GC_general_register_disappearing_link(link,base));
}
STATIC int GC_register_disappearing_link_inner(
struct dl_hashtbl_s*dl_hashtbl,void**link,
const void*obj,const char*tbl_log_name)
{
struct disappearing_link*curr_dl;
size_t index;
struct disappearing_link*new_dl;
DCL_LOCK_STATE;
if (EXPECT(GC_find_leak,FALSE))return GC_UNIMPLEMENTED;
LOCK();
GC_ASSERT(obj!=NULL&&GC_base_C(obj)==obj);
if (EXPECT(NULL==dl_hashtbl->head,FALSE)
||EXPECT(dl_hashtbl->entries
> ((word)1<<dl_hashtbl->log_size),FALSE)){
GC_grow_table((struct hash_chain_entry***)&dl_hashtbl->head,
&dl_hashtbl->log_size,&dl_hashtbl->entries);
GC_COND_LOG_PRINTF("Grew %s table to %u entries\n",tbl_log_name,
1U<<dl_hashtbl->log_size);
}
index=HASH2(link,dl_hashtbl->log_size);
for (curr_dl=dl_hashtbl->head[index];curr_dl!=0;
curr_dl=dl_next(curr_dl)){
if (curr_dl->dl_hidden_link==GC_HIDE_POINTER(link)){
curr_dl->dl_hidden_obj=GC_HIDE_POINTER(obj);
UNLOCK();
return GC_DUPLICATE;
}
}
new_dl=(struct disappearing_link*)
GC_INTERNAL_MALLOC(sizeof(struct disappearing_link),NORMAL);
if (0==new_dl){
GC_oom_func oom_fn=GC_oom_fn;
UNLOCK();
new_dl=(struct disappearing_link*)
(*oom_fn)(sizeof(struct disappearing_link));
if (0==new_dl){
return GC_NO_MEMORY;
}
LOCK();
index=HASH2(link,dl_hashtbl->log_size);
for (curr_dl=dl_hashtbl->head[index];curr_dl!=0;
curr_dl=dl_next(curr_dl)){
if (curr_dl->dl_hidden_link==GC_HIDE_POINTER(link)){
curr_dl->dl_hidden_obj=GC_HIDE_POINTER(obj);
UNLOCK();
#ifndef DBG_HDRS_ALL
GC_free((void*)new_dl);
#endif
return GC_DUPLICATE;
}
}
}
new_dl->dl_hidden_obj=GC_HIDE_POINTER(obj);
new_dl->dl_hidden_link=GC_HIDE_POINTER(link);
dl_set_next(new_dl,dl_hashtbl->head[index]);
GC_dirty(new_dl);
dl_hashtbl->head[index]=new_dl;
dl_hashtbl->entries++;
GC_dirty(dl_hashtbl->head+index);
UNLOCK();
return GC_SUCCESS;
}
GC_API int GC_CALL GC_general_register_disappearing_link(void**link,
const void*obj)
{
if (((word)link&(ALIGNMENT-1))!=0||!NONNULL_ARG_NOT_NULL(link))
ABORT("Bad arg to GC_general_register_disappearing_link");
return GC_register_disappearing_link_inner(&GC_dl_hashtbl,link,obj,
"dl");
}
#ifdef DBG_HDRS_ALL
#define FREE_DL_ENTRY(curr_dl)dl_set_next(curr_dl,NULL)
#else
#define FREE_DL_ENTRY(curr_dl)GC_free(curr_dl)
#endif
GC_INLINE struct disappearing_link*GC_unregister_disappearing_link_inner(
struct dl_hashtbl_s*dl_hashtbl,void**link)
{
struct disappearing_link*curr_dl;
struct disappearing_link*prev_dl=NULL;
size_t index;
GC_ASSERT(I_HOLD_LOCK());
if (EXPECT(NULL==dl_hashtbl->head,FALSE))return NULL;
index=HASH2(link,dl_hashtbl->log_size);
for (curr_dl=dl_hashtbl->head[index];curr_dl;
curr_dl=dl_next(curr_dl)){
if (curr_dl->dl_hidden_link==GC_HIDE_POINTER(link)){
if (NULL==prev_dl){
dl_hashtbl->head[index]=dl_next(curr_dl);
GC_dirty(dl_hashtbl->head+index);
} else {
dl_set_next(prev_dl,dl_next(curr_dl));
GC_dirty(prev_dl);
}
dl_hashtbl->entries--;
break;
}
prev_dl=curr_dl;
}
return curr_dl;
}
GC_API int GC_CALL GC_unregister_disappearing_link(void**link)
{
struct disappearing_link*curr_dl;
DCL_LOCK_STATE;
if (((word)link&(ALIGNMENT-1))!=0)return(0);
LOCK();
curr_dl=GC_unregister_disappearing_link_inner(&GC_dl_hashtbl,link);
UNLOCK();
if (NULL==curr_dl)return 0;
FREE_DL_ENTRY(curr_dl);
return 1;
}
#ifndef GC_TOGGLE_REFS_NOT_NEEDED
typedef union toggle_ref_u GCToggleRef;
STATIC GC_toggleref_func GC_toggleref_callback=0;
GC_INNER void GC_process_togglerefs(void)
{
size_t i;
size_t new_size=0;
GC_bool needs_barrier=FALSE;
GC_ASSERT(I_HOLD_LOCK());
for (i=0;i < GC_toggleref_array_size;++i){
GCToggleRef r=GC_toggleref_arr[i];
void*obj=r.strong_ref;
if (((word)obj&1)!=0){
obj=GC_REVEAL_POINTER(r.weak_ref);
}
if (NULL==obj){
continue;
}
switch (GC_toggleref_callback(obj)){
case GC_TOGGLE_REF_DROP:
break;
case GC_TOGGLE_REF_STRONG:
GC_toggleref_arr[new_size++].strong_ref=obj;
needs_barrier=TRUE;
break;
case GC_TOGGLE_REF_WEAK:
GC_toggleref_arr[new_size++].weak_ref=GC_HIDE_POINTER(obj);
break;
default:
ABORT("Bad toggle-ref status returned by callback");
}
}
if (new_size < GC_toggleref_array_size){
BZERO(&GC_toggleref_arr[new_size],
(GC_toggleref_array_size - new_size)*sizeof(GCToggleRef));
GC_toggleref_array_size=new_size;
}
if (needs_barrier)
GC_dirty(GC_toggleref_arr);
}
STATIC void GC_normal_finalize_mark_proc(ptr_t);
static void push_and_mark_object(void*p)
{
GC_normal_finalize_mark_proc((ptr_t)p);
while (!GC_mark_stack_empty()){
MARK_FROM_MARK_STACK();
}
GC_set_mark_bit(p);
if (GC_mark_state!=MS_NONE){
while (!GC_mark_some(0)){
}
}
}
STATIC void GC_mark_togglerefs(void)
{
size_t i;
if (NULL==GC_toggleref_arr)
return;
GC_set_mark_bit(GC_toggleref_arr);
for (i=0;i < GC_toggleref_array_size;++i){
void*obj=GC_toggleref_arr[i].strong_ref;
if (obj!=NULL&&((word)obj&1)==0){
push_and_mark_object(obj);
}
}
}
STATIC void GC_clear_togglerefs(void)
{
size_t i;
for (i=0;i < GC_toggleref_array_size;++i){
if ((GC_toggleref_arr[i].weak_ref&1)!=0){
if (!GC_is_marked(GC_REVEAL_POINTER(GC_toggleref_arr[i].weak_ref))){
GC_toggleref_arr[i].weak_ref=0;
} else {
}
}
}
}
GC_API void GC_CALL GC_set_toggleref_func(GC_toggleref_func fn)
{
DCL_LOCK_STATE;
LOCK();
GC_toggleref_callback=fn;
UNLOCK();
}
GC_API GC_toggleref_func GC_CALL GC_get_toggleref_func(void)
{
GC_toggleref_func fn;
DCL_LOCK_STATE;
LOCK();
fn=GC_toggleref_callback;
UNLOCK();
return fn;
}
static GC_bool ensure_toggleref_capacity(size_t capacity_inc)
{
GC_ASSERT(I_HOLD_LOCK());
if (NULL==GC_toggleref_arr){
GC_toggleref_array_capacity=32;
GC_toggleref_arr=(GCToggleRef*)GC_INTERNAL_MALLOC_IGNORE_OFF_PAGE(
GC_toggleref_array_capacity*sizeof(GCToggleRef),
NORMAL);
if (NULL==GC_toggleref_arr)
return FALSE;
}
if (GC_toggleref_array_size+capacity_inc
>=GC_toggleref_array_capacity){
GCToggleRef*new_array;
while (GC_toggleref_array_capacity
< GC_toggleref_array_size+capacity_inc){
GC_toggleref_array_capacity*=2;
if ((GC_toggleref_array_capacity
&((size_t)1<<(sizeof(size_t)*8 - 1)))!=0)
return FALSE;
}
new_array=(GCToggleRef*)GC_INTERNAL_MALLOC_IGNORE_OFF_PAGE(
GC_toggleref_array_capacity*sizeof(GCToggleRef),
NORMAL);
if (NULL==new_array)
return FALSE;
if (EXPECT(GC_toggleref_array_size > 0,TRUE))
BCOPY(GC_toggleref_arr,new_array,
GC_toggleref_array_size*sizeof(GCToggleRef));
GC_INTERNAL_FREE(GC_toggleref_arr);
GC_toggleref_arr=new_array;
}
return TRUE;
}
GC_API int GC_CALL GC_toggleref_add(void*obj,int is_strong_ref)
{
int res=GC_SUCCESS;
DCL_LOCK_STATE;
GC_ASSERT(NONNULL_ARG_NOT_NULL(obj));
LOCK();
if (GC_toggleref_callback!=0){
if (!ensure_toggleref_capacity(1)){
res=GC_NO_MEMORY;
} else {
GC_toggleref_arr[GC_toggleref_array_size].strong_ref=
is_strong_ref?obj:(void*)GC_HIDE_POINTER(obj);
if (is_strong_ref)
GC_dirty(GC_toggleref_arr+GC_toggleref_array_size);
GC_toggleref_array_size++;
}
}
UNLOCK();
return res;
}
#endif
STATIC GC_await_finalize_proc GC_object_finalized_proc=0;
GC_API void GC_CALL GC_set_await_finalize_proc(GC_await_finalize_proc fn)
{
DCL_LOCK_STATE;
LOCK();
GC_object_finalized_proc=fn;
UNLOCK();
}
GC_API GC_await_finalize_proc GC_CALL GC_get_await_finalize_proc(void)
{
GC_await_finalize_proc fn;
DCL_LOCK_STATE;
LOCK();
fn=GC_object_finalized_proc;
UNLOCK();
return fn;
}
#ifndef GC_LONG_REFS_NOT_NEEDED
GC_API int GC_CALL GC_register_long_link(void**link,const void*obj)
{
if (((word)link&(ALIGNMENT-1))!=0||!NONNULL_ARG_NOT_NULL(link))
ABORT("Bad arg to GC_register_long_link");
return GC_register_disappearing_link_inner(&GC_ll_hashtbl,link,obj,
"long dl");
}
GC_API int GC_CALL GC_unregister_long_link(void**link)
{
struct disappearing_link*curr_dl;
DCL_LOCK_STATE;
if (((word)link&(ALIGNMENT-1))!=0)return(0);
LOCK();
curr_dl=GC_unregister_disappearing_link_inner(&GC_ll_hashtbl,link);
UNLOCK();
if (NULL==curr_dl)return 0;
FREE_DL_ENTRY(curr_dl);
return 1;
}
#endif
#ifndef GC_MOVE_DISAPPEARING_LINK_NOT_NEEDED
STATIC int GC_move_disappearing_link_inner(
struct dl_hashtbl_s*dl_hashtbl,
void**link,void**new_link)
{
struct disappearing_link*curr_dl,*new_dl;
struct disappearing_link*prev_dl=NULL;
size_t curr_index,new_index;
word curr_hidden_link,new_hidden_link;
GC_ASSERT(I_HOLD_LOCK());
if (EXPECT(NULL==dl_hashtbl->head,FALSE))return GC_NOT_FOUND;
curr_index=HASH2(link,dl_hashtbl->log_size);
curr_hidden_link=GC_HIDE_POINTER(link);
for (curr_dl=dl_hashtbl->head[curr_index];curr_dl;
curr_dl=dl_next(curr_dl)){
if (curr_dl->dl_hidden_link==curr_hidden_link)
break;
prev_dl=curr_dl;
}
if (EXPECT(NULL==curr_dl,FALSE)){
return GC_NOT_FOUND;
} else if (link==new_link){
return GC_SUCCESS;
}
new_index=HASH2(new_link,dl_hashtbl->log_size);
new_hidden_link=GC_HIDE_POINTER(new_link);
for (new_dl=dl_hashtbl->head[new_index];new_dl;
new_dl=dl_next(new_dl)){
if (new_dl->dl_hidden_link==new_hidden_link){
return GC_DUPLICATE;
}
}
if (NULL==prev_dl){
dl_hashtbl->head[curr_index]=dl_next(curr_dl);
} else {
dl_set_next(prev_dl,dl_next(curr_dl));
GC_dirty(prev_dl);
}
curr_dl->dl_hidden_link=new_hidden_link;
dl_set_next(curr_dl,dl_hashtbl->head[new_index]);
dl_hashtbl->head[new_index]=curr_dl;
GC_dirty(curr_dl);
GC_dirty(dl_hashtbl->head);
return GC_SUCCESS;
}
GC_API int GC_CALL GC_move_disappearing_link(void**link,void**new_link)
{
int result;
DCL_LOCK_STATE;
if (((word)new_link&(ALIGNMENT-1))!=0
||!NONNULL_ARG_NOT_NULL(new_link))
ABORT("Bad new_link arg to GC_move_disappearing_link");
if (((word)link&(ALIGNMENT-1))!=0)
return GC_NOT_FOUND;
LOCK();
result=GC_move_disappearing_link_inner(&GC_dl_hashtbl,link,new_link);
UNLOCK();
return result;
}
#ifndef GC_LONG_REFS_NOT_NEEDED
GC_API int GC_CALL GC_move_long_link(void**link,void**new_link)
{
int result;
DCL_LOCK_STATE;
if (((word)new_link&(ALIGNMENT-1))!=0
||!NONNULL_ARG_NOT_NULL(new_link))
ABORT("Bad new_link arg to GC_move_long_link");
if (((word)link&(ALIGNMENT-1))!=0)
return GC_NOT_FOUND;
LOCK();
result=GC_move_disappearing_link_inner(&GC_ll_hashtbl,link,new_link);
UNLOCK();
return result;
}
#endif
#endif
STATIC void GC_normal_finalize_mark_proc(ptr_t p)
{
#if defined(_MSC_VER)&&defined(I386)
hdr*hhdr=HDR(p);
#define mark_stack_top GC_mark_stack_top
mse*mark_stack_limit=GC_mark_stack+GC_mark_stack_size;
word descr=hhdr->hb_descr;
if (descr!=0){
mark_stack_top++;
if ((word)mark_stack_top>=(word)mark_stack_limit){
mark_stack_top=GC_signal_mark_stack_overflow(mark_stack_top);
}
mark_stack_top->mse_start=p;
mark_stack_top->mse_descr.w=descr;
}
#undef mark_stack_top
#else
GC_mark_stack_top=GC_push_obj(p,HDR(p),GC_mark_stack_top,
GC_mark_stack+GC_mark_stack_size);
#endif
}
STATIC void GC_ignore_self_finalize_mark_proc(ptr_t p)
{
hdr*hhdr=HDR(p);
word descr=hhdr->hb_descr;
ptr_t q;
ptr_t scan_limit;
ptr_t target_limit=p+hhdr->hb_sz - 1;
if ((descr&GC_DS_TAGS)==GC_DS_LENGTH){
scan_limit=p+descr - sizeof(word);
} else {
scan_limit=target_limit+1 - sizeof(word);
}
for (q=p;(word)q<=(word)scan_limit;q+=ALIGNMENT){
word r=*(word*)q;
if (r < (word)p||r > (word)target_limit){
GC_PUSH_ONE_HEAP(r,q,GC_mark_stack_top);
}
}
}
STATIC void GC_null_finalize_mark_proc(ptr_t p GC_ATTR_UNUSED){}
STATIC void GC_unreachable_finalize_mark_proc(ptr_t p)
{
GC_normal_finalize_mark_proc(p);
}
STATIC void GC_register_finalizer_inner(void*obj,
GC_finalization_proc fn,void*cd,
GC_finalization_proc*ofn,void**ocd,
finalization_mark_proc mp)
{
struct finalizable_object*curr_fo;
size_t index;
struct finalizable_object*new_fo=0;
hdr*hhdr=NULL;
DCL_LOCK_STATE;
if (EXPECT(GC_find_leak,FALSE))return;
LOCK();
if (EXPECT(NULL==GC_fnlz_roots.fo_head,FALSE)
||EXPECT(GC_fo_entries > ((word)1<<GC_log_fo_table_size),FALSE)){
GC_grow_table((struct hash_chain_entry***)&GC_fnlz_roots.fo_head,
&GC_log_fo_table_size,&GC_fo_entries);
GC_COND_LOG_PRINTF("Grew fo table to %u entries\n",
1U<<GC_log_fo_table_size);
}
for (;;){
struct finalizable_object*prev_fo=NULL;
GC_oom_func oom_fn;
index=HASH2(obj,GC_log_fo_table_size);
curr_fo=GC_fnlz_roots.fo_head[index];
while (curr_fo!=0){
GC_ASSERT(GC_size(curr_fo)>=sizeof(struct finalizable_object));
if (curr_fo->fo_hidden_base==GC_HIDE_POINTER(obj)){
if (ocd)*ocd=(void*)(curr_fo->fo_client_data);
if (ofn)*ofn=curr_fo->fo_fn;
if (prev_fo==0){
GC_fnlz_roots.fo_head[index]=fo_next(curr_fo);
} else {
fo_set_next(prev_fo,fo_next(curr_fo));
GC_dirty(prev_fo);
}
if (fn==0){
GC_fo_entries--;
#if!defined(THREADS)&&!defined(DBG_HDRS_ALL)
GC_free((void*)curr_fo);
#endif
} else {
curr_fo->fo_fn=fn;
curr_fo->fo_client_data=(ptr_t)cd;
curr_fo->fo_mark_proc=mp;
GC_dirty(curr_fo);
if (prev_fo==0){
GC_fnlz_roots.fo_head[index]=curr_fo;
} else {
fo_set_next(prev_fo,curr_fo);
GC_dirty(prev_fo);
}
}
if (NULL==prev_fo)
GC_dirty(GC_fnlz_roots.fo_head+index);
UNLOCK();
#ifndef DBG_HDRS_ALL
GC_free((void*)new_fo);
#endif
return;
}
prev_fo=curr_fo;
curr_fo=fo_next(curr_fo);
}
if (EXPECT(new_fo!=0,FALSE)){
GC_ASSERT(fn!=0);
#ifdef LINT2
if (NULL==hhdr)ABORT("Bad hhdr in GC_register_finalizer_inner");
#endif
break;
}
if (fn==0){
if (ocd)*ocd=0;
if (ofn)*ofn=0;
UNLOCK();
return;
}
GET_HDR(obj,hhdr);
if (EXPECT(0==hhdr,FALSE)){
if (ocd)*ocd=0;
if (ofn)*ofn=0;
UNLOCK();
return;
}
new_fo=(struct finalizable_object*)
GC_INTERNAL_MALLOC(sizeof(struct finalizable_object),NORMAL);
if (EXPECT(new_fo!=0,TRUE))
break;
oom_fn=GC_oom_fn;
UNLOCK();
new_fo=(struct finalizable_object*)
(*oom_fn)(sizeof(struct finalizable_object));
if (0==new_fo){
return;
}
LOCK();
}
GC_ASSERT(GC_size(new_fo)>=sizeof(struct finalizable_object));
if (ocd)*ocd=0;
if (ofn)*ofn=0;
new_fo->fo_hidden_base=GC_HIDE_POINTER(obj);
new_fo->fo_fn=fn;
new_fo->fo_client_data=(ptr_t)cd;
new_fo->fo_object_size=hhdr->hb_sz;
new_fo->fo_mark_proc=mp;
fo_set_next(new_fo,GC_fnlz_roots.fo_head[index]);
GC_dirty(new_fo);
GC_fo_entries++;
GC_fnlz_roots.fo_head[index]=new_fo;
GC_dirty(GC_fnlz_roots.fo_head+index);
UNLOCK();
}
GC_API void GC_CALL GC_register_finalizer(void*obj,
GC_finalization_proc fn,void*cd,
GC_finalization_proc*ofn,void**ocd)
{
GC_register_finalizer_inner(obj,fn,cd,ofn,
ocd,GC_normal_finalize_mark_proc);
}
GC_API void GC_CALL GC_register_finalizer_ignore_self(void*obj,
GC_finalization_proc fn,void*cd,
GC_finalization_proc*ofn,void**ocd)
{
GC_register_finalizer_inner(obj,fn,cd,ofn,
ocd,GC_ignore_self_finalize_mark_proc);
}
GC_API void GC_CALL GC_register_finalizer_no_order(void*obj,
GC_finalization_proc fn,void*cd,
GC_finalization_proc*ofn,void**ocd)
{
GC_register_finalizer_inner(obj,fn,cd,ofn,
ocd,GC_null_finalize_mark_proc);
}
static GC_bool need_unreachable_finalization=FALSE;
GC_API void GC_CALL GC_register_finalizer_unreachable(void*obj,
GC_finalization_proc fn,void*cd,
GC_finalization_proc*ofn,void**ocd)
{
need_unreachable_finalization=TRUE;
GC_ASSERT(GC_java_finalization);
GC_register_finalizer_inner(obj,fn,cd,ofn,
ocd,GC_unreachable_finalize_mark_proc);
}
#ifndef NO_DEBUGGING
STATIC void GC_dump_finalization_links(
const struct dl_hashtbl_s*dl_hashtbl)
{
size_t dl_size=(size_t)1<<dl_hashtbl->log_size;
size_t i;
if (NULL==dl_hashtbl->head)return;
for (i=0;i < dl_size;i++){
struct disappearing_link*curr_dl;
for (curr_dl=dl_hashtbl->head[i];curr_dl!=0;
curr_dl=dl_next(curr_dl)){
ptr_t real_ptr=(ptr_t)GC_REVEAL_POINTER(curr_dl->dl_hidden_obj);
ptr_t real_link=(ptr_t)GC_REVEAL_POINTER(curr_dl->dl_hidden_link);
GC_printf("Object:%p,link:%p\n",
(void*)real_ptr,(void*)real_link);
}
}
}
GC_API void GC_CALL GC_dump_finalization(void)
{
struct finalizable_object*curr_fo;
size_t i;
size_t fo_size=GC_fnlz_roots.fo_head==NULL?0:
(size_t)1<<GC_log_fo_table_size;
GC_printf("Disappearing (short)links:\n");
GC_dump_finalization_links(&GC_dl_hashtbl);
#ifndef GC_LONG_REFS_NOT_NEEDED
GC_printf("Disappearing long links:\n");
GC_dump_finalization_links(&GC_ll_hashtbl);
#endif
GC_printf("Finalizers:\n");
for (i=0;i < fo_size;i++){
for (curr_fo=GC_fnlz_roots.fo_head[i];
curr_fo!=NULL;curr_fo=fo_next(curr_fo)){
ptr_t real_ptr=(ptr_t)GC_REVEAL_POINTER(curr_fo->fo_hidden_base);
GC_printf("Finalizable object:%p\n",(void*)real_ptr);
}
}
}
#endif
#ifndef SMALL_CONFIG
STATIC word GC_old_dl_entries=0;
#ifndef GC_LONG_REFS_NOT_NEEDED
STATIC word GC_old_ll_entries=0;
#endif
#endif
#ifndef THREADS
STATIC int GC_finalizer_nested=0;
STATIC unsigned GC_finalizer_skipped=0;
STATIC unsigned char*GC_check_finalizer_nested(void)
{
unsigned nesting_level=*(unsigned char*)&GC_finalizer_nested;
if (nesting_level){
if (++GC_finalizer_skipped < (1U<<nesting_level))return NULL;
GC_finalizer_skipped=0;
}
*(char*)&GC_finalizer_nested=(char)(nesting_level+1);
return (unsigned char*)&GC_finalizer_nested;
}
#endif
GC_INLINE void GC_make_disappearing_links_disappear(
struct dl_hashtbl_s*dl_hashtbl,
GC_bool is_remove_dangling)
{
size_t i;
size_t dl_size=(size_t)1<<dl_hashtbl->log_size;
GC_bool needs_barrier=FALSE;
GC_ASSERT(I_HOLD_LOCK());
if (NULL==dl_hashtbl->head)return;
for (i=0;i < dl_size;i++){
struct disappearing_link*curr_dl,*next_dl;
struct disappearing_link*prev_dl=NULL;
for (curr_dl=dl_hashtbl->head[i];curr_dl!=NULL;curr_dl=next_dl){
next_dl=dl_next(curr_dl);
if (is_remove_dangling){
ptr_t real_link=(ptr_t)GC_base(GC_REVEAL_POINTER(
curr_dl->dl_hidden_link));
if (NULL==real_link||EXPECT(GC_is_marked(real_link),TRUE)){
prev_dl=curr_dl;
continue;
}
} else {
if (EXPECT(GC_is_marked((ptr_t)GC_REVEAL_POINTER(
curr_dl->dl_hidden_obj)),TRUE)){
prev_dl=curr_dl;
continue;
}
*(ptr_t*)GC_REVEAL_POINTER(curr_dl->dl_hidden_link)=NULL;
}
if (NULL==prev_dl){
dl_hashtbl->head[i]=next_dl;
needs_barrier=TRUE;
} else {
dl_set_next(prev_dl,next_dl);
GC_dirty(prev_dl);
}
GC_clear_mark_bit(curr_dl);
dl_hashtbl->entries--;
}
}
if (needs_barrier)
GC_dirty(dl_hashtbl->head);
}
GC_INNER void GC_finalize(void)
{
struct finalizable_object*curr_fo,*prev_fo,*next_fo;
ptr_t real_ptr;
size_t i;
size_t fo_size=GC_fnlz_roots.fo_head==NULL?0:
(size_t)1<<GC_log_fo_table_size;
GC_bool needs_barrier=FALSE;
GC_ASSERT(I_HOLD_LOCK());
#ifndef SMALL_CONFIG
GC_old_dl_entries=GC_dl_hashtbl.entries;
#ifndef GC_LONG_REFS_NOT_NEEDED
GC_old_ll_entries=GC_ll_hashtbl.entries;
#endif
#endif
#ifndef GC_TOGGLE_REFS_NOT_NEEDED
GC_mark_togglerefs();
#endif
GC_make_disappearing_links_disappear(&GC_dl_hashtbl,FALSE);
GC_ASSERT(GC_mark_state==MS_NONE);
for (i=0;i < fo_size;i++){
for (curr_fo=GC_fnlz_roots.fo_head[i];
curr_fo!=NULL;curr_fo=fo_next(curr_fo)){
GC_ASSERT(GC_size(curr_fo)>=sizeof(struct finalizable_object));
real_ptr=(ptr_t)GC_REVEAL_POINTER(curr_fo->fo_hidden_base);
if (!GC_is_marked(real_ptr)){
GC_MARKED_FOR_FINALIZATION(real_ptr);
GC_MARK_FO(real_ptr,curr_fo->fo_mark_proc);
if (GC_is_marked(real_ptr)){
WARN("Finalization cycle involving %p\n",real_ptr);
}
}
}
}
GC_bytes_finalized=0;
for (i=0;i < fo_size;i++){
curr_fo=GC_fnlz_roots.fo_head[i];
prev_fo=0;
while (curr_fo!=0){
real_ptr=(ptr_t)GC_REVEAL_POINTER(curr_fo->fo_hidden_base);
if (!GC_is_marked(real_ptr)){
if (!GC_java_finalization){
GC_set_mark_bit(real_ptr);
}
next_fo=fo_next(curr_fo);
if (NULL==prev_fo){
GC_fnlz_roots.fo_head[i]=next_fo;
if (GC_object_finalized_proc){
GC_dirty(GC_fnlz_roots.fo_head+i);
} else {
needs_barrier=TRUE;
}
} else {
fo_set_next(prev_fo,next_fo);
GC_dirty(prev_fo);
}
GC_fo_entries--;
if (GC_object_finalized_proc)
GC_object_finalized_proc(real_ptr);
fo_set_next(curr_fo,GC_fnlz_roots.finalize_now);
GC_dirty(curr_fo);
SET_FINALIZE_NOW(curr_fo);
curr_fo->fo_hidden_base=
(word)GC_REVEAL_POINTER(curr_fo->fo_hidden_base);
GC_bytes_finalized+=
curr_fo->fo_object_size
+sizeof(struct finalizable_object);
GC_ASSERT(GC_is_marked(GC_base(curr_fo)));
curr_fo=next_fo;
} else {
prev_fo=curr_fo;
curr_fo=fo_next(curr_fo);
}
}
}
if (GC_java_finalization){
for (curr_fo=GC_fnlz_roots.finalize_now;
curr_fo!=NULL;curr_fo=fo_next(curr_fo)){
real_ptr=(ptr_t)curr_fo->fo_hidden_base;
if (!GC_is_marked(real_ptr)){
if (curr_fo->fo_mark_proc==GC_null_finalize_mark_proc){
GC_MARK_FO(real_ptr,GC_normal_finalize_mark_proc);
}
if (curr_fo->fo_mark_proc!=GC_unreachable_finalize_mark_proc){
GC_set_mark_bit(real_ptr);
}
}
}
if (need_unreachable_finalization){
curr_fo=GC_fnlz_roots.finalize_now;
GC_ASSERT(NULL==curr_fo||GC_fnlz_roots.fo_head!=NULL);
prev_fo=NULL;
while (curr_fo!=NULL){
next_fo=fo_next(curr_fo);
if (curr_fo->fo_mark_proc==GC_unreachable_finalize_mark_proc){
real_ptr=(ptr_t)curr_fo->fo_hidden_base;
if (!GC_is_marked(real_ptr)){
GC_set_mark_bit(real_ptr);
} else {
if (NULL==prev_fo){
SET_FINALIZE_NOW(next_fo);
} else {
fo_set_next(prev_fo,next_fo);
GC_dirty(prev_fo);
}
curr_fo->fo_hidden_base=
GC_HIDE_POINTER(curr_fo->fo_hidden_base);
GC_bytes_finalized-=
curr_fo->fo_object_size+sizeof(struct finalizable_object);
i=HASH2(real_ptr,GC_log_fo_table_size);
fo_set_next(curr_fo,GC_fnlz_roots.fo_head[i]);
GC_dirty(curr_fo);
GC_fo_entries++;
GC_fnlz_roots.fo_head[i]=curr_fo;
curr_fo=prev_fo;
needs_barrier=TRUE;
}
}
prev_fo=curr_fo;
curr_fo=next_fo;
}
}
}
if (needs_barrier)
GC_dirty(GC_fnlz_roots.fo_head);
GC_make_disappearing_links_disappear(&GC_dl_hashtbl,TRUE);
#ifndef GC_TOGGLE_REFS_NOT_NEEDED
GC_clear_togglerefs();
#endif
#ifndef GC_LONG_REFS_NOT_NEEDED
GC_make_disappearing_links_disappear(&GC_ll_hashtbl,FALSE);
GC_make_disappearing_links_disappear(&GC_ll_hashtbl,TRUE);
#endif
if (GC_fail_count){
#ifdef THREADS
GC_reset_finalizer_nested();
#else
GC_finalizer_nested=0;
#endif
}
}
#ifndef JAVA_FINALIZATION_NOT_NEEDED
STATIC void GC_enqueue_all_finalizers(void)
{
struct finalizable_object*next_fo;
size_t i;
size_t fo_size=GC_fnlz_roots.fo_head==NULL?0:
(size_t)1<<GC_log_fo_table_size;
GC_ASSERT(I_HOLD_LOCK());
GC_bytes_finalized=0;
for (i=0;i < fo_size;i++){
struct finalizable_object*curr_fo=GC_fnlz_roots.fo_head[i];
GC_fnlz_roots.fo_head[i]=NULL;
while (curr_fo!=NULL){
ptr_t real_ptr=(ptr_t)GC_REVEAL_POINTER(curr_fo->fo_hidden_base);
GC_MARK_FO(real_ptr,GC_normal_finalize_mark_proc);
GC_set_mark_bit(real_ptr);
next_fo=fo_next(curr_fo);
fo_set_next(curr_fo,GC_fnlz_roots.finalize_now);
GC_dirty(curr_fo);
SET_FINALIZE_NOW(curr_fo);
curr_fo->fo_hidden_base=
(word)GC_REVEAL_POINTER(curr_fo->fo_hidden_base);
GC_bytes_finalized+=
curr_fo->fo_object_size+sizeof(struct finalizable_object);
curr_fo=next_fo;
}
}
GC_fo_entries=0;
}
GC_API void GC_CALL GC_finalize_all(void)
{
DCL_LOCK_STATE;
LOCK();
while (GC_fo_entries > 0){
GC_enqueue_all_finalizers();
UNLOCK();
GC_invoke_finalizers();
LOCK();
}
UNLOCK();
}
#endif
GC_API int GC_CALL GC_should_invoke_finalizers(void)
{
#ifdef AO_HAVE_load
return AO_load((volatile AO_t*)&GC_fnlz_roots.finalize_now)!=0;
#else
return GC_fnlz_roots.finalize_now!=NULL;
#endif
}
GC_API int GC_CALL GC_invoke_finalizers(void)
{
int count=0;
word bytes_freed_before=0;
DCL_LOCK_STATE;
while (GC_should_invoke_finalizers()){
struct finalizable_object*curr_fo;
#ifdef THREADS
LOCK();
#endif
if (count==0){
bytes_freed_before=GC_bytes_freed;
}
curr_fo=GC_fnlz_roots.finalize_now;
#ifdef THREADS
if (curr_fo!=NULL)
SET_FINALIZE_NOW(fo_next(curr_fo));
UNLOCK();
if (curr_fo==0)break;
#else
GC_fnlz_roots.finalize_now=fo_next(curr_fo);
#endif
fo_set_next(curr_fo,0);
(*(curr_fo->fo_fn))((ptr_t)(curr_fo->fo_hidden_base),
curr_fo->fo_client_data);
curr_fo->fo_client_data=0;
++count;
}
if (count!=0
#if defined(THREADS)&&!defined(THREAD_SANITIZER)
&&bytes_freed_before!=GC_bytes_freed
#endif
){
LOCK();
GC_finalizer_bytes_freed+=(GC_bytes_freed - bytes_freed_before);
UNLOCK();
}
return count;
}
static word last_finalizer_notification=0;
GC_INNER void GC_notify_or_invoke_finalizers(void)
{
GC_finalizer_notifier_proc notifier_fn=0;
#if defined(KEEP_BACK_PTRS)||defined(MAKE_BACK_GRAPH)
static word last_back_trace_gc_no=1;
#endif
DCL_LOCK_STATE;
#if defined(THREADS)&&!defined(KEEP_BACK_PTRS)&&!defined(MAKE_BACK_GRAPH)
if (!GC_should_invoke_finalizers())
return;
#endif
LOCK();
#if defined(KEEP_BACK_PTRS)||defined(MAKE_BACK_GRAPH)
if (GC_gc_no > last_back_trace_gc_no){
#ifdef KEEP_BACK_PTRS
long i;
last_back_trace_gc_no=GC_WORD_MAX;
for (i=0;i < GC_backtraces;++i){
UNLOCK();
GC_generate_random_backtrace_no_gc();
LOCK();
}
last_back_trace_gc_no=GC_gc_no;
#endif
#ifdef MAKE_BACK_GRAPH
if (GC_print_back_height){
GC_print_back_graph_stats();
}
#endif
}
#endif
if (NULL==GC_fnlz_roots.finalize_now){
UNLOCK();
return;
}
if (!GC_finalize_on_demand){
unsigned char*pnested=GC_check_finalizer_nested();
UNLOCK();
if (pnested!=NULL){
(void)GC_invoke_finalizers();
*pnested=0;
#ifndef THREADS
GC_ASSERT(NULL==GC_fnlz_roots.finalize_now);
#endif
}
return;
}
if (last_finalizer_notification!=GC_gc_no){
notifier_fn=GC_finalizer_notifier;
last_finalizer_notification=GC_gc_no;
}
UNLOCK();
if (notifier_fn!=0)
(*notifier_fn)();
}
#ifndef SMALL_CONFIG
#ifndef GC_LONG_REFS_NOT_NEEDED
#define IF_LONG_REFS_PRESENT_ELSE(x,y)(x)
#else
#define IF_LONG_REFS_PRESENT_ELSE(x,y)(y)
#endif
GC_INNER void GC_print_finalization_stats(void)
{
struct finalizable_object*fo;
unsigned long ready=0;
GC_log_printf("%lu finalization entries;"
" %lu/%lu short/long disappearing links alive\n",
(unsigned long)GC_fo_entries,
(unsigned long)GC_dl_hashtbl.entries,
(unsigned long)IF_LONG_REFS_PRESENT_ELSE(
GC_ll_hashtbl.entries,0));
for (fo=GC_fnlz_roots.finalize_now;fo!=NULL;fo=fo_next(fo))
++ready;
GC_log_printf("%lu finalization-ready objects;"
" %ld/%ld short/long links cleared\n",
ready,
(long)GC_old_dl_entries - (long)GC_dl_hashtbl.entries,
(long)IF_LONG_REFS_PRESENT_ELSE(
GC_old_ll_entries - GC_ll_hashtbl.entries,0));
}
#endif
#endif
#ifdef ENABLE_DISCLAIM
#ifndef GC_DISCLAIM_H
#define GC_DISCLAIM_H
#ifdef __cplusplus
extern "C" {
#endif
GC_API void GC_CALL GC_init_finalized_malloc(void);
typedef int (GC_CALLBACK*GC_disclaim_proc)(void*);
GC_API void GC_CALL GC_register_disclaim_proc(int,
GC_disclaim_proc,
int)GC_ATTR_NONNULL(2);
struct GC_finalizer_closure {
GC_finalization_proc proc;
void*cd;
};
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_finalized_malloc(size_t,
const struct GC_finalizer_closure*)GC_ATTR_NONNULL(2);
#ifdef __cplusplus
}
#endif
#endif
#if defined(KEEP_BACK_PTRS)||defined(MAKE_BACK_GRAPH)
#define FINALIZER_CLOSURE_FLAG 0x2
#else
#define FINALIZER_CLOSURE_FLAG 0x1
#endif
STATIC int GC_CALLBACK GC_finalized_disclaim(void*obj)
{
word fc_word=*(word*)obj;
if ((fc_word&FINALIZER_CLOSURE_FLAG)!=0){
const struct GC_finalizer_closure*fc
=(struct GC_finalizer_closure*)(fc_word
&~(word)FINALIZER_CLOSURE_FLAG);
GC_ASSERT(!GC_find_leak);
(*fc->proc)((word*)obj+1,fc->cd);
}
return 0;
}
GC_API void GC_CALL GC_init_finalized_malloc(void)
{
DCL_LOCK_STATE;
GC_init();
LOCK();
if (GC_finalized_kind!=0){
UNLOCK();
return;
}
GC_register_displacement_inner(sizeof(word));
GC_register_displacement_inner(FINALIZER_CLOSURE_FLAG);
GC_register_displacement_inner(sizeof(oh)+FINALIZER_CLOSURE_FLAG);
GC_finalized_kind=GC_new_kind_inner(GC_new_free_list_inner(),
GC_DS_LENGTH,TRUE,TRUE);
GC_ASSERT(GC_finalized_kind!=0);
GC_register_disclaim_proc(GC_finalized_kind,GC_finalized_disclaim,TRUE);
UNLOCK();
}
GC_API void GC_CALL GC_register_disclaim_proc(int kind,GC_disclaim_proc proc,
int mark_unconditionally)
{
GC_ASSERT((unsigned)kind < MAXOBJKINDS);
GC_ASSERT(NONNULL_ARG_NOT_NULL(proc));
if (!EXPECT(GC_find_leak,FALSE)){
GC_obj_kinds[kind].ok_disclaim_proc=proc;
GC_obj_kinds[kind].ok_mark_unconditionally=
(GC_bool)mark_unconditionally;
}
}
GC_API GC_ATTR_MALLOC void*GC_CALL GC_finalized_malloc(size_t lb,
const struct GC_finalizer_closure*fclos)
{
word*op;
GC_ASSERT(GC_finalized_kind!=0);
GC_ASSERT(NONNULL_ARG_NOT_NULL(fclos));
GC_ASSERT(((word)fclos&FINALIZER_CLOSURE_FLAG)==0);
op=(word*)GC_malloc_kind(SIZET_SAT_ADD(lb,sizeof(word)),
GC_finalized_kind);
if (EXPECT(NULL==op,FALSE))
return NULL;
*op=(word)fclos|FINALIZER_CLOSURE_FLAG;
GC_dirty(op);
REACHABLE_AFTER_DIRTY(fclos);
return op+1;
}
#endif
#include <stdio.h>
#include <string.h>
STATIC GC_bool GC_alloc_reclaim_list(struct obj_kind*kind)
{
struct hblk**result=(struct hblk**)
GC_scratch_alloc((MAXOBJGRANULES+1)*sizeof(struct hblk*));
if (result==0)return(FALSE);
BZERO(result,(MAXOBJGRANULES+1)*sizeof(struct hblk*));
kind->ok_reclaim_list=result;
return(TRUE);
}
GC_INNER ptr_t GC_alloc_large(size_t lb,int k,unsigned flags)
{
struct hblk*h;
word n_blocks;
ptr_t result;
GC_bool retry=FALSE;
GC_ASSERT(I_HOLD_LOCK());
lb=ROUNDUP_GRANULE_SIZE(lb);
n_blocks=OBJ_SZ_TO_BLOCKS_CHECKED(lb);
if (!EXPECT(GC_is_initialized,TRUE)){
DCL_LOCK_STATE;
UNLOCK();
GC_init();
LOCK();
}
if (GC_incremental&&!GC_dont_gc){
ENTER_GC();
GC_collect_a_little_inner((int)n_blocks);
EXIT_GC();
}
h=GC_allochblk(lb,k,flags);
#ifdef USE_MUNMAP
if (0==h){
GC_merge_unmapped();
h=GC_allochblk(lb,k,flags);
}
#endif
while (0==h&&GC_collect_or_expand(n_blocks,flags!=0,retry)){
h=GC_allochblk(lb,k,flags);
retry=TRUE;
}
if (h==0){
result=0;
} else {
size_t total_bytes=n_blocks*HBLKSIZE;
if (n_blocks > 1){
GC_large_allocd_bytes+=total_bytes;
if (GC_large_allocd_bytes > GC_max_large_allocd_bytes)
GC_max_large_allocd_bytes=GC_large_allocd_bytes;
}
result=h->hb_body;
}
return result;
}
STATIC ptr_t GC_alloc_large_and_clear(size_t lb,int k,unsigned flags)
{
ptr_t result;
GC_ASSERT(I_HOLD_LOCK());
result=GC_alloc_large(lb,k,flags);
if (result!=NULL
&&(GC_debugging_started||GC_obj_kinds[k].ok_init)){
word n_blocks=OBJ_SZ_TO_BLOCKS(lb);
BZERO(result,n_blocks*HBLKSIZE);
}
return result;
}
STATIC void GC_extend_size_map(size_t i)
{
size_t orig_granule_sz=ROUNDED_UP_GRANULES(i);
size_t granule_sz;
size_t byte_sz=GRANULES_TO_BYTES(orig_granule_sz);
size_t smaller_than_i=byte_sz - (byte_sz>>3);
size_t low_limit;
size_t number_of_objs;
GC_ASSERT(I_HOLD_LOCK());
GC_ASSERT(0==GC_size_map[i]);
if (0==GC_size_map[smaller_than_i]){
low_limit=byte_sz - (byte_sz>>2);
granule_sz=orig_granule_sz;
while (GC_size_map[low_limit]!=0)
low_limit++;
} else {
low_limit=smaller_than_i+1;
while (GC_size_map[low_limit]!=0)
low_limit++;
granule_sz=ROUNDED_UP_GRANULES(low_limit);
granule_sz+=granule_sz>>3;
if (granule_sz < orig_granule_sz)
granule_sz=orig_granule_sz;
}
granule_sz=(granule_sz+1)&~1;
if (granule_sz > MAXOBJGRANULES)
granule_sz=MAXOBJGRANULES;
number_of_objs=HBLK_GRANULES/granule_sz;
GC_ASSERT(number_of_objs!=0);
granule_sz=(HBLK_GRANULES/number_of_objs)&~1;
byte_sz=GRANULES_TO_BYTES(granule_sz)- EXTRA_BYTES;
for (;low_limit<=byte_sz;low_limit++)
GC_size_map[low_limit]=granule_sz;
}
GC_INNER void*GC_generic_malloc_inner(size_t lb,int k)
{
void*op;
GC_ASSERT(I_HOLD_LOCK());
GC_ASSERT(k < MAXOBJKINDS);
if (SMALL_OBJ(lb)){
struct obj_kind*kind=GC_obj_kinds+k;
size_t lg=GC_size_map[lb];
void**opp=&(kind->ok_freelist[lg]);
op=*opp;
if (EXPECT(0==op,FALSE)){
if (lg==0){
if (!EXPECT(GC_is_initialized,TRUE)){
DCL_LOCK_STATE;
UNLOCK();
GC_init();
LOCK();
lg=GC_size_map[lb];
}
if (0==lg){
GC_extend_size_map(lb);
lg=GC_size_map[lb];
GC_ASSERT(lg!=0);
}
opp=&(kind->ok_freelist[lg]);
op=*opp;
}
if (0==op){
if (0==kind->ok_reclaim_list&&
!GC_alloc_reclaim_list(kind))
return NULL;
op=GC_allocobj(lg,k);
if (0==op)
return NULL;
}
}
*opp=obj_link(op);
obj_link(op)=0;
GC_bytes_allocd+=GRANULES_TO_BYTES((word)lg);
} else {
op=(ptr_t)GC_alloc_large_and_clear(ADD_SLOP(lb),k,0);
if (op!=NULL)
GC_bytes_allocd+=lb;
}
return op;
}
#if defined(DBG_HDRS_ALL)||defined(GC_GCJ_SUPPORT)||!defined(GC_NO_FINALIZATION)
GC_INNER void*GC_generic_malloc_inner_ignore_off_page(size_t lb,int k)
{
word lb_adjusted;
void*op;
GC_ASSERT(I_HOLD_LOCK());
if (lb<=HBLKSIZE)
return GC_generic_malloc_inner(lb,k);
GC_ASSERT(k < MAXOBJKINDS);
lb_adjusted=ADD_SLOP(lb);
op=GC_alloc_large_and_clear(lb_adjusted,k,IGNORE_OFF_PAGE);
if (op!=NULL)
GC_bytes_allocd+=lb_adjusted;
return op;
}
#endif
#ifdef GC_COLLECT_AT_MALLOC
#if defined(CPPCHECK)
size_t GC_dbg_collect_at_malloc_min_lb=16*1024;
#else
size_t GC_dbg_collect_at_malloc_min_lb=(GC_COLLECT_AT_MALLOC);
#endif
#endif
GC_API GC_ATTR_MALLOC void*GC_CALL GC_generic_malloc(size_t lb,int k)
{
void*result;
DCL_LOCK_STATE;
GC_ASSERT(k < MAXOBJKINDS);
if (EXPECT(GC_have_errors,FALSE))
GC_print_all_errors();
GC_INVOKE_FINALIZERS();
GC_DBG_COLLECT_AT_MALLOC(lb);
if (SMALL_OBJ(lb)){
LOCK();
result=GC_generic_malloc_inner(lb,k);
UNLOCK();
} else {
size_t lg;
size_t lb_rounded;
word n_blocks;
GC_bool init;
lg=ROUNDED_UP_GRANULES(lb);
lb_rounded=GRANULES_TO_BYTES(lg);
n_blocks=OBJ_SZ_TO_BLOCKS(lb_rounded);
init=GC_obj_kinds[k].ok_init;
LOCK();
result=(ptr_t)GC_alloc_large(lb_rounded,k,0);
if (0!=result){
if (GC_debugging_started){
BZERO(result,n_blocks*HBLKSIZE);
} else {
#ifdef THREADS
((word*)result)[0]=0;
((word*)result)[1]=0;
((word*)result)[GRANULES_TO_WORDS(lg)-1]=0;
((word*)result)[GRANULES_TO_WORDS(lg)-2]=0;
#endif
}
GC_bytes_allocd+=lb_rounded;
}
UNLOCK();
if (init&&!GC_debugging_started&&0!=result){
BZERO(result,n_blocks*HBLKSIZE);
}
}
if (0==result){
return((*GC_get_oom_fn())(lb));
} else {
return(result);
}
}
GC_API GC_ATTR_MALLOC void*GC_CALL GC_malloc_kind_global(size_t lb,int k)
{
GC_ASSERT(k < MAXOBJKINDS);
if (SMALL_OBJ(lb)){
void*op;
void**opp;
size_t lg;
DCL_LOCK_STATE;
GC_DBG_COLLECT_AT_MALLOC(lb);
LOCK();
lg=GC_size_map[lb];
opp=&GC_obj_kinds[k].ok_freelist[lg];
op=*opp;
if (EXPECT(op!=NULL,TRUE)){
if (k==PTRFREE){
*opp=obj_link(op);
} else {
GC_ASSERT(0==obj_link(op)
||((word)obj_link(op)
<=(word)GC_greatest_plausible_heap_addr
&&(word)obj_link(op)
>=(word)GC_least_plausible_heap_addr));
*opp=obj_link(op);
obj_link(op)=0;
}
GC_bytes_allocd+=GRANULES_TO_BYTES((word)lg);
UNLOCK();
return op;
}
UNLOCK();
}
return GC_clear_stack(GC_generic_malloc(lb,k));
}
#if defined(THREADS)&&!defined(THREAD_LOCAL_ALLOC)
GC_API GC_ATTR_MALLOC void*GC_CALL GC_malloc_kind(size_t lb,int k)
{
return GC_malloc_kind_global(lb,k);
}
#endif
GC_API GC_ATTR_MALLOC void*GC_CALL GC_malloc_atomic(size_t lb)
{
return GC_malloc_kind(lb,PTRFREE);
}
GC_API GC_ATTR_MALLOC void*GC_CALL GC_malloc(size_t lb)
{
return GC_malloc_kind(lb,NORMAL);
}
GC_API GC_ATTR_MALLOC void*GC_CALL GC_generic_malloc_uncollectable(
size_t lb,int k)
{
void*op;
DCL_LOCK_STATE;
GC_ASSERT(k < MAXOBJKINDS);
if (SMALL_OBJ(lb)){
void**opp;
size_t lg;
GC_DBG_COLLECT_AT_MALLOC(lb);
if (EXTRA_BYTES!=0&&lb!=0)lb--;
LOCK();
lg=GC_size_map[lb];
opp=&GC_obj_kinds[k].ok_freelist[lg];
op=*opp;
if (EXPECT(op!=NULL,TRUE)){
*opp=obj_link(op);
obj_link(op)=0;
GC_bytes_allocd+=GRANULES_TO_BYTES((word)lg);
GC_non_gc_bytes+=GRANULES_TO_BYTES((word)lg);
UNLOCK();
} else {
UNLOCK();
op=GC_generic_malloc(lb,k);
}
GC_ASSERT(0==op||GC_is_marked(op));
} else {
op=GC_generic_malloc(lb,k);
if (op){
hdr*hhdr=HDR(op);
GC_ASSERT(((word)op&(HBLKSIZE - 1))==0);
LOCK();
set_mark_bit_from_hdr(hhdr,0);
#ifndef THREADS
GC_ASSERT(hhdr->hb_n_marks==0);
#endif
hhdr->hb_n_marks=1;
UNLOCK();
}
}
return op;
}
GC_API GC_ATTR_MALLOC void*GC_CALL GC_malloc_uncollectable(size_t lb)
{
return GC_generic_malloc_uncollectable(lb,UNCOLLECTABLE);
}
#ifdef GC_ATOMIC_UNCOLLECTABLE
GC_API GC_ATTR_MALLOC void*GC_CALL
GC_malloc_atomic_uncollectable(size_t lb)
{
return GC_generic_malloc_uncollectable(lb,AUNCOLLECTABLE);
}
#endif
#if defined(REDIRECT_MALLOC)&&!defined(REDIRECT_MALLOC_IN_HEADER)
#ifndef MSWINCE
#include <errno.h>
#endif
#define GC_debug_malloc_replacement(lb)GC_debug_malloc(lb,GC_DBG_EXTRAS)
#if defined(CPPCHECK)
#define REDIRECT_MALLOC_F GC_malloc
#else
#define REDIRECT_MALLOC_F REDIRECT_MALLOC
#endif
void*malloc(size_t lb)
{
#if defined(I386)&&defined(GC_SOLARIS_THREADS)
if (!EXPECT(GC_is_initialized,TRUE))return sbrk(lb);
#endif
return (void*)REDIRECT_MALLOC_F(lb);
}
#if defined(GC_LINUX_THREADS)
STATIC ptr_t GC_libpthread_start=0;
STATIC ptr_t GC_libpthread_end=0;
STATIC ptr_t GC_libld_start=0;
STATIC ptr_t GC_libld_end=0;
STATIC void GC_init_lib_bounds(void)
{
IF_CANCEL(int cancel_state;)
if (GC_libpthread_start!=0)return;
DISABLE_CANCEL(cancel_state);
GC_init();
if (!GC_text_mapping("libpthread-",
&GC_libpthread_start,&GC_libpthread_end)){
WARN("Failed to find libpthread.so text mapping:Expect crash\n",0);
GC_libpthread_start=(ptr_t)1;
}
if (!GC_text_mapping("ld-",&GC_libld_start,&GC_libld_end)){
WARN("Failed to find ld.so text mapping:Expect crash\n",0);
}
RESTORE_CANCEL(cancel_state);
}
#endif
void*calloc(size_t n,size_t lb)
{
if ((lb|n)> GC_SQRT_SIZE_MAX
&&lb&&n > GC_SIZE_MAX/lb)
return (*GC_get_oom_fn())(GC_SIZE_MAX);
#if defined(GC_LINUX_THREADS)
{
static GC_bool lib_bounds_set=FALSE;
ptr_t caller=(ptr_t)__builtin_return_address(0);
if (!EXPECT(lib_bounds_set,TRUE)){
GC_init_lib_bounds();
lib_bounds_set=TRUE;
}
if (((word)caller>=(word)GC_libpthread_start
&&(word)caller < (word)GC_libpthread_end)
||((word)caller>=(word)GC_libld_start
&&(word)caller < (word)GC_libld_end))
return GC_generic_malloc_uncollectable(n*lb,UNCOLLECTABLE);
}
#endif
return (void*)REDIRECT_MALLOC_F(n*lb);
}
#ifndef strdup
char*strdup(const char*s)
{
size_t lb=strlen(s)+1;
char*result=(char*)REDIRECT_MALLOC_F(lb);
if (result==0){
errno=ENOMEM;
return 0;
}
BCOPY(s,result,lb);
return result;
}
#endif
#ifndef strndup
char*strndup(const char*str,size_t size)
{
char*copy;
size_t len=strlen(str);
if (len > size)
len=size;
copy=(char*)REDIRECT_MALLOC_F(len+1);
if (copy==NULL){
errno=ENOMEM;
return NULL;
}
if (EXPECT(len > 0,TRUE))
BCOPY(str,copy,len);
copy[len]='\0';
return copy;
}
#endif
#undef GC_debug_malloc_replacement
#endif
GC_API void GC_CALL GC_free(void*p)
{
struct hblk*h;
hdr*hhdr;
size_t sz;
size_t ngranules;
int knd;
struct obj_kind*ok;
DCL_LOCK_STATE;
if (p){
} else {
return;
}
#ifdef LOG_ALLOCS
GC_log_printf("GC_free(%p)after GC #%lu\n",
p,(unsigned long)GC_gc_no);
#endif
h=HBLKPTR(p);
hhdr=HDR(h);
#if defined(REDIRECT_MALLOC)&&((defined(NEED_CALLINFO)&&defined(GC_HAVE_BUILTIN_BACKTRACE))||defined(GC_SOLARIS_THREADS)||defined(GC_LINUX_THREADS)||defined(MSWIN32))
if (0==hhdr)return;
#endif
GC_ASSERT(GC_base(p)==p);
sz=(size_t)hhdr->hb_sz;
ngranules=BYTES_TO_GRANULES(sz);
knd=hhdr->hb_obj_kind;
ok=&GC_obj_kinds[knd];
if (EXPECT(ngranules<=MAXOBJGRANULES,TRUE)){
void**flh;
LOCK();
GC_bytes_freed+=sz;
if (IS_UNCOLLECTABLE(knd))GC_non_gc_bytes-=sz;
if (ok->ok_init&&EXPECT(sz > sizeof(word),TRUE)){
BZERO((word*)p+1,sz-sizeof(word));
}
flh=&(ok->ok_freelist[ngranules]);
obj_link(p)=*flh;
*flh=(ptr_t)p;
UNLOCK();
} else {
size_t nblocks=OBJ_SZ_TO_BLOCKS(sz);
LOCK();
GC_bytes_freed+=sz;
if (IS_UNCOLLECTABLE(knd))GC_non_gc_bytes-=sz;
if (nblocks > 1){
GC_large_allocd_bytes-=nblocks*HBLKSIZE;
}
GC_freehblk(h);
UNLOCK();
}
}
#ifdef THREADS
GC_INNER void GC_free_inner(void*p)
{
struct hblk*h;
hdr*hhdr;
size_t sz;
size_t ngranules;
int knd;
struct obj_kind*ok;
h=HBLKPTR(p);
hhdr=HDR(h);
knd=hhdr->hb_obj_kind;
sz=(size_t)hhdr->hb_sz;
ngranules=BYTES_TO_GRANULES(sz);
ok=&GC_obj_kinds[knd];
if (ngranules<=MAXOBJGRANULES){
void**flh;
GC_bytes_freed+=sz;
if (IS_UNCOLLECTABLE(knd))GC_non_gc_bytes-=sz;
if (ok->ok_init&&EXPECT(sz > sizeof(word),TRUE)){
BZERO((word*)p+1,sz-sizeof(word));
}
flh=&(ok->ok_freelist[ngranules]);
obj_link(p)=*flh;
*flh=(ptr_t)p;
} else {
size_t nblocks=OBJ_SZ_TO_BLOCKS(sz);
GC_bytes_freed+=sz;
if (IS_UNCOLLECTABLE(knd))GC_non_gc_bytes-=sz;
if (nblocks > 1){
GC_large_allocd_bytes-=nblocks*HBLKSIZE;
}
GC_freehblk(h);
}
}
#endif
#if defined(REDIRECT_MALLOC)&&!defined(REDIRECT_FREE)
#define REDIRECT_FREE GC_free
#endif
#if defined(REDIRECT_FREE)&&!defined(REDIRECT_MALLOC_IN_HEADER)
#if defined(CPPCHECK)
#define REDIRECT_FREE_F GC_free
#else
#define REDIRECT_FREE_F REDIRECT_FREE
#endif
void free(void*p)
{
#ifndef IGNORE_FREE
#if defined(GC_LINUX_THREADS)&&!defined(USE_PROC_FOR_LIBRARIES)
ptr_t caller=(ptr_t)__builtin_return_address(0);
if (((word)caller>=(word)GC_libpthread_start
&&(word)caller < (word)GC_libpthread_end)
||((word)caller>=(word)GC_libld_start
&&(word)caller < (word)GC_libld_end)){
GC_free(p);
return;
}
#endif
REDIRECT_FREE_F(p);
#endif
}
#endif
#include <stdio.h>
#include <string.h>
#ifndef MSWINCE
#include <errno.h>
#endif
#ifndef GC_ALLOC_PTRS_H
#define GC_ALLOC_PTRS_H
#ifdef __cplusplus
extern "C" {
#endif
GC_API void**const GC_objfreelist_ptr;
GC_API void**const GC_aobjfreelist_ptr;
GC_API void**const GC_uobjfreelist_ptr;
#ifdef GC_ATOMIC_UNCOLLECTABLE
GC_API void**const GC_auobjfreelist_ptr;
#endif
GC_API void GC_CALL GC_incr_bytes_allocd(size_t bytes);
GC_API void GC_CALL GC_incr_bytes_freed(size_t bytes);
#ifdef __cplusplus
}
#endif
#endif
void**const GC_objfreelist_ptr=GC_objfreelist;
void**const GC_aobjfreelist_ptr=GC_aobjfreelist;
void**const GC_uobjfreelist_ptr=GC_uobjfreelist;
#ifdef GC_ATOMIC_UNCOLLECTABLE
void**const GC_auobjfreelist_ptr=GC_auobjfreelist;
#endif
GC_API int GC_CALL GC_get_kind_and_size(const void*p,size_t*psize)
{
hdr*hhdr=HDR(p);
if (psize!=NULL){
*psize=(size_t)hhdr->hb_sz;
}
return hhdr->hb_obj_kind;
}
GC_API GC_ATTR_MALLOC void*GC_CALL GC_generic_or_special_malloc(size_t lb,
int knd)
{
switch(knd){
case PTRFREE:
case NORMAL:
return GC_malloc_kind(lb,knd);
case UNCOLLECTABLE:
#ifdef GC_ATOMIC_UNCOLLECTABLE
case AUNCOLLECTABLE:
#endif
return GC_generic_malloc_uncollectable(lb,knd);
default:
return GC_generic_malloc(lb,knd);
}
}
GC_API void*GC_CALL GC_realloc(void*p,size_t lb)
{
struct hblk*h;
hdr*hhdr;
void*result;
size_t sz;
size_t orig_sz;
int obj_kind;
if (p==0)return(GC_malloc(lb));
if (0==lb){
#ifndef IGNORE_FREE
GC_free(p);
#endif
return NULL;
}
h=HBLKPTR(p);
hhdr=HDR(h);
sz=(size_t)hhdr->hb_sz;
obj_kind=hhdr->hb_obj_kind;
orig_sz=sz;
if (sz > MAXOBJBYTES){
word descr=GC_obj_kinds[obj_kind].ok_descriptor;
sz=(sz+HBLKSIZE-1)&~HBLKMASK;
if (GC_obj_kinds[obj_kind].ok_relocate_descr)
descr+=sz;
#ifdef AO_HAVE_store
GC_STATIC_ASSERT(sizeof(hhdr->hb_sz)==sizeof(AO_t));
AO_store((volatile AO_t*)&hhdr->hb_sz,(AO_t)sz);
AO_store((volatile AO_t*)&hhdr->hb_descr,(AO_t)descr);
#else
{
DCL_LOCK_STATE;
LOCK();
hhdr->hb_sz=sz;
hhdr->hb_descr=descr;
UNLOCK();
}
#endif
#ifdef MARK_BIT_PER_OBJ
GC_ASSERT(hhdr->hb_inv_sz==LARGE_INV_SZ);
#endif
#ifdef MARK_BIT_PER_GRANULE
GC_ASSERT((hhdr->hb_flags&LARGE_BLOCK)!=0
&&hhdr->hb_map[ANY_INDEX]==1);
#endif
if (IS_UNCOLLECTABLE(obj_kind))GC_non_gc_bytes+=(sz - orig_sz);
}
if (ADD_SLOP(lb)<=sz){
if (lb>=(sz>>1)){
if (orig_sz > lb){
BZERO(((ptr_t)p)+lb,orig_sz - lb);
}
return(p);
}
sz=lb;
}
result=GC_generic_or_special_malloc((word)lb,obj_kind);
if (result!=NULL){
BCOPY(p,result,sz);
#ifndef IGNORE_FREE
GC_free(p);
#endif
}
return result;
}
#if defined(REDIRECT_MALLOC)&&!defined(REDIRECT_REALLOC)
#define REDIRECT_REALLOC GC_realloc
#endif
#ifdef REDIRECT_REALLOC
#define GC_debug_realloc_replacement(p,lb)GC_debug_realloc(p,lb,GC_DBG_EXTRAS)
#if!defined(REDIRECT_MALLOC_IN_HEADER)
void*realloc(void*p,size_t lb)
{
return(REDIRECT_REALLOC(p,lb));
}
#endif
#undef GC_debug_realloc_replacement
#endif
GC_API GC_ATTR_MALLOC void*GC_CALL
GC_generic_malloc_ignore_off_page(size_t lb,int k)
{
void*result;
size_t lg;
size_t lb_rounded;
word n_blocks;
GC_bool init;
DCL_LOCK_STATE;
if (SMALL_OBJ(lb))
return GC_generic_malloc(lb,k);
GC_ASSERT(k < MAXOBJKINDS);
lg=ROUNDED_UP_GRANULES(lb);
lb_rounded=GRANULES_TO_BYTES(lg);
n_blocks=OBJ_SZ_TO_BLOCKS(lb_rounded);
init=GC_obj_kinds[k].ok_init;
if (EXPECT(GC_have_errors,FALSE))
GC_print_all_errors();
GC_INVOKE_FINALIZERS();
GC_DBG_COLLECT_AT_MALLOC(lb);
LOCK();
result=(ptr_t)GC_alloc_large(ADD_SLOP(lb),k,IGNORE_OFF_PAGE);
if (NULL==result){
GC_oom_func oom_fn=GC_oom_fn;
UNLOCK();
return (*oom_fn)(lb);
}
if (GC_debugging_started){
BZERO(result,n_blocks*HBLKSIZE);
} else {
#ifdef THREADS
((word*)result)[0]=0;
((word*)result)[1]=0;
((word*)result)[GRANULES_TO_WORDS(lg)-1]=0;
((word*)result)[GRANULES_TO_WORDS(lg)-2]=0;
#endif
}
GC_bytes_allocd+=lb_rounded;
UNLOCK();
if (init&&!GC_debugging_started){
BZERO(result,n_blocks*HBLKSIZE);
}
return(result);
}
GC_API GC_ATTR_MALLOC void*GC_CALL GC_malloc_ignore_off_page(size_t lb)
{
return GC_generic_malloc_ignore_off_page(lb,NORMAL);
}
GC_API GC_ATTR_MALLOC void*GC_CALL
GC_malloc_atomic_ignore_off_page(size_t lb)
{
return GC_generic_malloc_ignore_off_page(lb,PTRFREE);
}
GC_API void GC_CALL GC_incr_bytes_allocd(size_t n)
{
GC_bytes_allocd+=n;
}
GC_API void GC_CALL GC_incr_bytes_freed(size_t n)
{
GC_bytes_freed+=n;
}
GC_API size_t GC_CALL GC_get_expl_freed_bytes_since_gc(void)
{
return (size_t)GC_bytes_freed;
}
#ifdef PARALLEL_MARK
STATIC volatile AO_t GC_bytes_allocd_tmp=0;
#endif
GC_API void GC_CALL GC_generic_malloc_many(size_t lb,int k,void**result)
{
void*op;
void*p;
void**opp;
size_t lw;
size_t lg;
signed_word my_bytes_allocd=0;
struct obj_kind*ok=&(GC_obj_kinds[k]);
struct hblk**rlh;
DCL_LOCK_STATE;
GC_ASSERT(lb!=0&&(lb&(GRANULE_BYTES-1))==0);
if (!SMALL_OBJ(lb)||GC_manual_vdb){
op=GC_generic_malloc(lb,k);
if (EXPECT(0!=op,TRUE))
obj_link(op)=0;
*result=op;
#ifndef GC_DISABLE_INCREMENTAL
if (GC_manual_vdb&&GC_is_heap_ptr(result)){
GC_dirty_inner(result);
REACHABLE_AFTER_DIRTY(op);
}
#endif
return;
}
GC_ASSERT(k < MAXOBJKINDS);
lw=BYTES_TO_WORDS(lb);
lg=BYTES_TO_GRANULES(lb);
if (EXPECT(GC_have_errors,FALSE))
GC_print_all_errors();
GC_INVOKE_FINALIZERS();
GC_DBG_COLLECT_AT_MALLOC(lb);
if (!EXPECT(GC_is_initialized,TRUE))GC_init();
LOCK();
if (GC_incremental&&!GC_dont_gc){
ENTER_GC();
GC_collect_a_little_inner(1);
EXIT_GC();
}
rlh=ok->ok_reclaim_list;
if (rlh!=NULL){
struct hblk*hbp;
hdr*hhdr;
for (rlh+=lg;(hbp=*rlh)!=NULL;){
hhdr=HDR(hbp);
*rlh=hhdr->hb_next;
GC_ASSERT(hhdr->hb_sz==lb);
hhdr->hb_last_reclaimed=(unsigned short)GC_gc_no;
#ifdef PARALLEL_MARK
if (GC_parallel){
signed_word my_bytes_allocd_tmp=
(signed_word)AO_load(&GC_bytes_allocd_tmp);
GC_ASSERT(my_bytes_allocd_tmp>=0);
if (my_bytes_allocd_tmp!=0){
(void)AO_fetch_and_add(&GC_bytes_allocd_tmp,
(AO_t)(-my_bytes_allocd_tmp));
GC_bytes_allocd+=my_bytes_allocd_tmp;
}
GC_acquire_mark_lock();
++GC_fl_builder_count;
UNLOCK();
GC_release_mark_lock();
}
#endif
op=GC_reclaim_generic(hbp,hhdr,lb,
ok->ok_init,0,&my_bytes_allocd);
if (op!=0){
#ifdef PARALLEL_MARK
if (GC_parallel){
*result=op;
(void)AO_fetch_and_add(&GC_bytes_allocd_tmp,
(AO_t)my_bytes_allocd);
GC_acquire_mark_lock();
--GC_fl_builder_count;
if (GC_fl_builder_count==0)GC_notify_all_builder();
#ifdef THREAD_SANITIZER
GC_release_mark_lock();
LOCK();
GC_bytes_found+=my_bytes_allocd;
UNLOCK();
#else
GC_bytes_found+=my_bytes_allocd;
GC_release_mark_lock();
#endif
(void)GC_clear_stack(0);
return;
}
#endif
GC_bytes_found+=my_bytes_allocd;
GC_bytes_allocd+=my_bytes_allocd;
goto out;
}
#ifdef PARALLEL_MARK
if (GC_parallel){
GC_acquire_mark_lock();
--GC_fl_builder_count;
if (GC_fl_builder_count==0)GC_notify_all_builder();
GC_release_mark_lock();
LOCK();
}
#endif
}
}
opp=&(GC_obj_kinds[k].ok_freelist[lg]);
if ( (op=*opp)!=0){
*opp=0;
my_bytes_allocd=0;
for (p=op;p!=0;p=obj_link(p)){
my_bytes_allocd+=lb;
if ((word)my_bytes_allocd>=HBLKSIZE){
*opp=obj_link(p);
obj_link(p)=0;
break;
}
}
GC_bytes_allocd+=my_bytes_allocd;
goto out;
}
{
struct hblk*h=GC_allochblk(lb,k,0);
if (h){
if (IS_UNCOLLECTABLE(k))GC_set_hdr_marks(HDR(h));
GC_bytes_allocd+=HBLKSIZE - HBLKSIZE % lb;
#ifdef PARALLEL_MARK
if (GC_parallel){
GC_acquire_mark_lock();
++GC_fl_builder_count;
UNLOCK();
GC_release_mark_lock();
op=GC_build_fl(h,lw,
(ok->ok_init||GC_debugging_started),0);
*result=op;
GC_acquire_mark_lock();
--GC_fl_builder_count;
if (GC_fl_builder_count==0)GC_notify_all_builder();
GC_release_mark_lock();
(void)GC_clear_stack(0);
return;
}
#endif
op=GC_build_fl(h,lw,(ok->ok_init||GC_debugging_started),0);
goto out;
}
}
op=GC_generic_malloc_inner(lb,k);
if (0!=op)obj_link(op)=0;
out:
*result=op;
UNLOCK();
(void)GC_clear_stack(0);
}
GC_API GC_ATTR_MALLOC void*GC_CALL GC_malloc_many(size_t lb)
{
void*result;
lb=SIZET_SAT_ADD(lb,EXTRA_BYTES+GRANULE_BYTES - 1)
&~(GRANULE_BYTES - 1);
GC_generic_malloc_many(lb,NORMAL,&result);
return result;
}
#include <limits.h>
GC_API GC_ATTR_MALLOC void*GC_CALL GC_memalign(size_t align,size_t lb)
{
size_t new_lb;
size_t offset;
ptr_t result;
if (align<=GRANULE_BYTES)return GC_malloc(lb);
if (align>=HBLKSIZE/2||lb>=HBLKSIZE/2){
if (align > HBLKSIZE){
return (*GC_get_oom_fn())(LONG_MAX-1024);
}
return GC_malloc(lb<=HBLKSIZE?HBLKSIZE:lb);
}
new_lb=SIZET_SAT_ADD(lb,align - 1);
result=(ptr_t)GC_malloc(new_lb);
offset=(word)result % align;
if (offset!=0){
offset=align - offset;
if (!GC_all_interior_pointers){
GC_STATIC_ASSERT(VALID_OFFSET_SZ<=HBLKSIZE);
GC_ASSERT(offset < VALID_OFFSET_SZ);
GC_register_displacement(offset);
}
}
result+=offset;
GC_ASSERT((word)result % align==0);
return result;
}
GC_API int GC_CALL GC_posix_memalign(void**memptr,size_t align,size_t lb)
{
size_t align_minus_one=align - 1;
if (align < sizeof(void*)||(align_minus_one&align)!=0){
#ifdef MSWINCE
return ERROR_INVALID_PARAMETER;
#else
return EINVAL;
#endif
}
if ((*memptr=GC_memalign(align,lb))==NULL){
#ifdef MSWINCE
return ERROR_NOT_ENOUGH_MEMORY;
#else
return ENOMEM;
#endif
}
return 0;
}
GC_API GC_ATTR_MALLOC char*GC_CALL GC_strdup(const char*s)
{
char*copy;
size_t lb;
if (s==NULL)return NULL;
lb=strlen(s)+1;
copy=(char*)GC_malloc_atomic(lb);
if (NULL==copy){
#ifndef MSWINCE
errno=ENOMEM;
#endif
return NULL;
}
BCOPY(s,copy,lb);
return copy;
}
GC_API GC_ATTR_MALLOC char*GC_CALL GC_strndup(const char*str,size_t size)
{
char*copy;
size_t len=strlen(str);
if (len > size)
len=size;
copy=(char*)GC_malloc_atomic(len+1);
if (copy==NULL){
#ifndef MSWINCE
errno=ENOMEM;
#endif
return NULL;
}
if (EXPECT(len > 0,TRUE))
BCOPY(str,copy,len);
copy[len]='\0';
return copy;
}
#ifdef GC_REQUIRE_WCSDUP
#include <wchar.h>
GC_API GC_ATTR_MALLOC wchar_t*GC_CALL GC_wcsdup(const wchar_t*str)
{
size_t lb=(wcslen(str)+1)*sizeof(wchar_t);
wchar_t*copy=(wchar_t*)GC_malloc_atomic(lb);
if (copy==NULL){
#ifndef MSWINCE
errno=ENOMEM;
#endif
return NULL;
}
BCOPY(str,copy,lb);
return copy;
}
#endif
#ifndef CPPCHECK
GC_API void*GC_CALL GC_malloc_stubborn(size_t lb)
{
return GC_malloc(lb);
}
GC_API void GC_CALL GC_change_stubborn(const void*p GC_ATTR_UNUSED)
{
}
#endif
GC_API void GC_CALL GC_end_stubborn_change(const void*p)
{
GC_dirty(p);
}
GC_API void GC_CALL GC_ptr_store_and_dirty(void*p,const void*q)
{
*(const void**)p=q;
GC_dirty(p);
REACHABLE_AFTER_DIRTY(q);
}
#if defined(__MINGW32__)&&!defined(__MINGW_EXCPT_DEFINE_PSDK)&&defined(__i386__)
#define __MINGW_EXCPT_DEFINE_PSDK 1
#endif
#include <stdio.h>
#if defined(MSWIN32)&&defined(__GNUC__)
#include <excpt.h>
#endif
GC_ATTR_NOINLINE
void GC_noop6(word arg1 GC_ATTR_UNUSED,word arg2 GC_ATTR_UNUSED,
word arg3 GC_ATTR_UNUSED,word arg4 GC_ATTR_UNUSED,
word arg5 GC_ATTR_UNUSED,word arg6 GC_ATTR_UNUSED)
{
#if defined(AO_HAVE_compiler_barrier)&&!defined(BASE_ATOMIC_OPS_EMULATED)
AO_compiler_barrier();
#else
GC_noop1(0);
#endif
}
volatile word GC_noop_sink;
GC_ATTR_NO_SANITIZE_THREAD
GC_API void GC_CALL GC_noop1(word x)
{
GC_noop_sink=x;
}
GC_INNER struct obj_kind GC_obj_kinds[MAXOBJKINDS]={
{&GC_aobjfreelist[0],0,
GC_DS_LENGTH,FALSE,FALSE
OK_DISCLAIM_INITZ },
{&GC_objfreelist[0],0,
GC_DS_LENGTH,
TRUE,TRUE
OK_DISCLAIM_INITZ },
{&GC_uobjfreelist[0],0,
GC_DS_LENGTH,TRUE,TRUE
OK_DISCLAIM_INITZ },
#ifdef GC_ATOMIC_UNCOLLECTABLE
{&GC_auobjfreelist[0],0,
GC_DS_LENGTH,FALSE,FALSE
OK_DISCLAIM_INITZ },
#endif
};
#ifndef INITIAL_MARK_STACK_SIZE
#define INITIAL_MARK_STACK_SIZE (1*HBLKSIZE)
#endif
#if!defined(GC_DISABLE_INCREMENTAL)
STATIC word GC_n_rescuing_pages=0;
#endif
#ifdef PARALLEL_MARK
GC_INNER GC_bool GC_parallel_mark_disabled=FALSE;
#endif
GC_INNER GC_bool GC_collection_in_progress(void)
{
return(GC_mark_state!=MS_NONE);
}
GC_INNER void GC_clear_hdr_marks(hdr*hhdr)
{
size_t last_bit;
#ifdef AO_HAVE_load
last_bit=FINAL_MARK_BIT((size_t)AO_load((volatile AO_t*)&hhdr->hb_sz));
#else
last_bit=FINAL_MARK_BIT((size_t)hhdr->hb_sz);
#endif
BZERO(hhdr->hb_marks,sizeof(hhdr->hb_marks));
set_mark_bit_from_hdr(hhdr,last_bit);
hhdr->hb_n_marks=0;
}
GC_INNER void GC_set_hdr_marks(hdr*hhdr)
{
unsigned i;
size_t sz=(size_t)hhdr->hb_sz;
unsigned n_marks=(unsigned)FINAL_MARK_BIT(sz);
#ifdef USE_MARK_BYTES
for (i=0;i<=n_marks;i+=(unsigned)MARK_BIT_OFFSET(sz)){
hhdr->hb_marks[i]=1;
}
#else
for (i=0;i < divWORDSZ(n_marks+WORDSZ);++i){
hhdr->hb_marks[i]=GC_WORD_MAX;
}
#endif
#ifdef MARK_BIT_PER_OBJ
hhdr->hb_n_marks=n_marks;
#else
hhdr->hb_n_marks=HBLK_OBJS(sz);
#endif
}
static void clear_marks_for_block(struct hblk*h,word dummy GC_ATTR_UNUSED)
{
hdr*hhdr=HDR(h);
if (IS_UNCOLLECTABLE(hhdr->hb_obj_kind))return;
GC_clear_hdr_marks(hhdr);
}
GC_API void GC_CALL GC_set_mark_bit(const void*p)
{
struct hblk*h=HBLKPTR(p);
hdr*hhdr=HDR(h);
word bit_no=MARK_BIT_NO((ptr_t)p - (ptr_t)h,hhdr->hb_sz);
if (!mark_bit_from_hdr(hhdr,bit_no)){
set_mark_bit_from_hdr(hhdr,bit_no);
++hhdr->hb_n_marks;
}
}
GC_API void GC_CALL GC_clear_mark_bit(const void*p)
{
struct hblk*h=HBLKPTR(p);
hdr*hhdr=HDR(h);
word bit_no=MARK_BIT_NO((ptr_t)p - (ptr_t)h,hhdr->hb_sz);
if (mark_bit_from_hdr(hhdr,bit_no)){
size_t n_marks=hhdr->hb_n_marks;
GC_ASSERT(n_marks!=0);
clear_mark_bit_from_hdr(hhdr,bit_no);
n_marks--;
#ifdef PARALLEL_MARK
if (n_marks!=0||!GC_parallel)
hhdr->hb_n_marks=n_marks;
#else
hhdr->hb_n_marks=n_marks;
#endif
}
}
GC_API int GC_CALL GC_is_marked(const void*p)
{
struct hblk*h=HBLKPTR(p);
hdr*hhdr=HDR(h);
word bit_no=MARK_BIT_NO((ptr_t)p - (ptr_t)h,hhdr->hb_sz);
return (int)mark_bit_from_hdr(hhdr,bit_no);
}
GC_INNER void GC_clear_marks(void)
{
GC_apply_to_all_blocks(clear_marks_for_block,(word)0);
GC_objects_are_marked=FALSE;
GC_mark_state=MS_INVALID;
GC_scan_ptr=NULL;
}
GC_INNER void GC_initiate_gc(void)
{
GC_ASSERT(I_HOLD_LOCK());
#ifndef GC_DISABLE_INCREMENTAL
if (GC_incremental){
#ifdef CHECKSUMS
GC_read_dirty(FALSE);
GC_check_dirty();
#else
GC_read_dirty(GC_mark_state==MS_INVALID);
#endif
}
GC_n_rescuing_pages=0;
#endif
if (GC_mark_state==MS_NONE){
GC_mark_state=MS_PUSH_RESCUERS;
} else if (GC_mark_state!=MS_INVALID){
ABORT("Unexpected state");
}
GC_scan_ptr=NULL;
}
#ifdef PARALLEL_MARK
STATIC void GC_do_parallel_mark(void);
#endif
#ifdef GC_DISABLE_INCREMENTAL
#define GC_push_next_marked_dirty(h)GC_push_next_marked(h)
#else
STATIC struct hblk*GC_push_next_marked_dirty(struct hblk*h);
#endif
STATIC struct hblk*GC_push_next_marked(struct hblk*h);
STATIC struct hblk*GC_push_next_marked_uncollectable(struct hblk*h);
static void alloc_mark_stack(size_t);
#ifdef WRAP_MARK_SOME
STATIC GC_bool GC_mark_some_inner(ptr_t cold_gc_frame)
#else
GC_INNER GC_bool GC_mark_some(ptr_t cold_gc_frame)
#endif
{
switch(GC_mark_state){
case MS_NONE:
break;
case MS_PUSH_RESCUERS:
if ((word)GC_mark_stack_top
>=(word)(GC_mark_stack_limit - INITIAL_MARK_STACK_SIZE/2)){
GC_mark_stack_too_small=TRUE;
MARK_FROM_MARK_STACK();
break;
} else {
GC_scan_ptr=GC_push_next_marked_dirty(GC_scan_ptr);
if (NULL==GC_scan_ptr){
#if!defined(GC_DISABLE_INCREMENTAL)
GC_COND_LOG_PRINTF("Marked from %lu dirty pages\n",
(unsigned long)GC_n_rescuing_pages);
#endif
GC_push_roots(FALSE,cold_gc_frame);
GC_objects_are_marked=TRUE;
if (GC_mark_state!=MS_INVALID){
GC_mark_state=MS_ROOTS_PUSHED;
}
}
}
break;
case MS_PUSH_UNCOLLECTABLE:
if ((word)GC_mark_stack_top
>=(word)(GC_mark_stack+GC_mark_stack_size/4)){
#ifdef PARALLEL_MARK
if (GC_parallel)GC_mark_stack_too_small=TRUE;
#endif
MARK_FROM_MARK_STACK();
break;
} else {
GC_scan_ptr=GC_push_next_marked_uncollectable(GC_scan_ptr);
if (NULL==GC_scan_ptr){
GC_push_roots(TRUE,cold_gc_frame);
GC_objects_are_marked=TRUE;
if (GC_mark_state!=MS_INVALID){
GC_mark_state=MS_ROOTS_PUSHED;
}
}
}
break;
case MS_ROOTS_PUSHED:
#ifdef PARALLEL_MARK
if (GC_parallel&&!GC_parallel_mark_disabled){
GC_do_parallel_mark();
GC_ASSERT((word)GC_mark_stack_top < (word)GC_first_nonempty);
GC_mark_stack_top=GC_mark_stack - 1;
if (GC_mark_stack_too_small){
alloc_mark_stack(2*GC_mark_stack_size);
}
if (GC_mark_state==MS_ROOTS_PUSHED){
GC_mark_state=MS_NONE;
return(TRUE);
}
break;
}
#endif
if ((word)GC_mark_stack_top>=(word)GC_mark_stack){
MARK_FROM_MARK_STACK();
break;
} else {
GC_mark_state=MS_NONE;
if (GC_mark_stack_too_small){
alloc_mark_stack(2*GC_mark_stack_size);
}
return(TRUE);
}
case MS_INVALID:
case MS_PARTIALLY_INVALID:
if (!GC_objects_are_marked){
GC_mark_state=MS_PUSH_UNCOLLECTABLE;
break;
}
if ((word)GC_mark_stack_top>=(word)GC_mark_stack){
MARK_FROM_MARK_STACK();
break;
}
if (NULL==GC_scan_ptr&&GC_mark_state==MS_INVALID){
if (GC_mark_stack_too_small){
alloc_mark_stack(2*GC_mark_stack_size);
}
GC_mark_state=MS_PARTIALLY_INVALID;
}
GC_scan_ptr=GC_push_next_marked(GC_scan_ptr);
if (NULL==GC_scan_ptr&&GC_mark_state==MS_PARTIALLY_INVALID){
GC_push_roots(TRUE,cold_gc_frame);
GC_objects_are_marked=TRUE;
if (GC_mark_state!=MS_INVALID){
GC_mark_state=MS_ROOTS_PUSHED;
}
}
break;
default:
ABORT("GC_mark_some:bad state");
}
return(FALSE);
}
#ifdef WRAP_MARK_SOME
#if (defined(MSWIN32)||defined(MSWINCE))&&defined(__GNUC__)
typedef struct {
EXCEPTION_REGISTRATION ex_reg;
void*alt_path;
} ext_ex_regn;
static EXCEPTION_DISPOSITION mark_ex_handler(
struct _EXCEPTION_RECORD*ex_rec,
void*est_frame,
struct _CONTEXT*context,
void*disp_ctxt GC_ATTR_UNUSED)
{
if (ex_rec->ExceptionCode==STATUS_ACCESS_VIOLATION){
ext_ex_regn*xer=(ext_ex_regn*)est_frame;
context->Esp=context->Ebp;
context->Ebp=*((DWORD*)context->Esp);
context->Esp=context->Esp - 8;
context->Eip=(DWORD)(xer->alt_path);
return ExceptionContinueExecution;
} else {
return ExceptionContinueSearch;
}
}
#endif
GC_INNER GC_bool GC_mark_some(ptr_t cold_gc_frame)
{
GC_bool ret_val;
#if defined(MSWIN32)||defined(MSWINCE)
#ifndef __GNUC__
__try {
ret_val=GC_mark_some_inner(cold_gc_frame);
} __except (GetExceptionCode()==EXCEPTION_ACCESS_VIOLATION?
EXCEPTION_EXECUTE_HANDLER:EXCEPTION_CONTINUE_SEARCH){
goto handle_ex;
}
#if defined(GC_WIN32_THREADS)&&!defined(GC_PTHREADS)
if (GC_started_thread_while_stopped())
goto handle_thr_start;
#endif
rm_handler:
return ret_val;
#else
ext_ex_regn er;
#if GC_GNUC_PREREQ(4,7)||GC_CLANG_PREREQ(3,3)
#pragma GCC diagnostic push
#if defined(__clang__)||GC_GNUC_PREREQ(6,4)
#pragma GCC diagnostic ignored "-Wpedantic"
#else
#pragma GCC diagnostic ignored "-pedantic"
#endif
er.alt_path=&&handle_ex;
#pragma GCC diagnostic pop
#elif!defined(CPPCHECK)
er.alt_path=&&handle_ex;
#endif
er.ex_reg.handler=mark_ex_handler;
__asm__ __volatile__ ("movl %%fs:0,%0":"=r" (er.ex_reg.prev));
__asm__ __volatile__ ("movl %0,%%fs:0"::"r" (&er));
ret_val=GC_mark_some_inner(cold_gc_frame);
if (er.alt_path==0)
goto handle_ex;
#if defined(GC_WIN32_THREADS)&&!defined(GC_PTHREADS)
if (GC_started_thread_while_stopped())
goto handle_thr_start;
#endif
rm_handler:
__asm__ __volatile__ ("mov %0,%%fs:0"::"r" (er.ex_reg.prev));
return ret_val;
#endif
#else
#ifndef DEFAULT_VDB
if (GC_auto_incremental){
WARN("Incremental GC incompatible with/proc roots\n",0);
}
#endif
GC_setup_temporary_fault_handler();
if(SETJMP(GC_jmp_buf)!=0)goto handle_ex;
ret_val=GC_mark_some_inner(cold_gc_frame);
rm_handler:
GC_reset_fault_handler();
return ret_val;
#endif
handle_ex:
{
static word warned_gc_no;
if (warned_gc_no!=GC_gc_no){
WARN("Caught ACCESS_VIOLATION in marker;"
" memory mapping disappeared\n",0);
warned_gc_no=GC_gc_no;
}
}
#if (defined(MSWIN32)||defined(MSWINCE))&&defined(GC_WIN32_THREADS)&&!defined(GC_PTHREADS)
handle_thr_start:
#endif
#ifdef REGISTER_LIBRARIES_EARLY
START_WORLD();
GC_cond_register_dynamic_libraries();
STOP_WORLD();
#endif
GC_invalidate_mark_state();
GC_scan_ptr=NULL;
ret_val=FALSE;
goto rm_handler;
}
#endif
GC_INNER void GC_invalidate_mark_state(void)
{
GC_mark_state=MS_INVALID;
GC_mark_stack_top=GC_mark_stack-1;
}
GC_INNER mse*GC_signal_mark_stack_overflow(mse*msp)
{
GC_mark_state=MS_INVALID;
#ifdef PARALLEL_MARK
if (!GC_parallel)
GC_mark_stack_too_small=TRUE;
#else
GC_mark_stack_too_small=TRUE;
#endif
GC_COND_LOG_PRINTF("Mark stack overflow;current size=%lu entries\n",
(unsigned long)GC_mark_stack_size);
return(msp - GC_MARK_STACK_DISCARDS);
}
GC_ATTR_NO_SANITIZE_ADDR GC_ATTR_NO_SANITIZE_MEMORY GC_ATTR_NO_SANITIZE_THREAD
GC_INNER mse*GC_mark_from(mse*mark_stack_top,mse*mark_stack,
mse*mark_stack_limit)
{
signed_word credit=HBLKSIZE;
ptr_t current_p;
word current;
ptr_t limit=0;
word descr;
ptr_t greatest_ha=(ptr_t)GC_greatest_plausible_heap_addr;
ptr_t least_ha=(ptr_t)GC_least_plausible_heap_addr;
DECLARE_HDR_CACHE;
#define SPLIT_RANGE_WORDS 128
GC_objects_are_marked=TRUE;
INIT_HDR_CACHE;
#ifdef OS2
while ((word)mark_stack_top>=(word)mark_stack&&credit>=0)
#else
while (((((word)mark_stack_top - (word)mark_stack)|(word)credit)
&SIGNB)==0)
#endif
{
current_p=mark_stack_top->mse_start;
descr=mark_stack_top->mse_descr.w;
retry:
if (descr&((~(WORDS_TO_BYTES(SPLIT_RANGE_WORDS)- 1))|GC_DS_TAGS)){
word tag=descr&GC_DS_TAGS;
GC_STATIC_ASSERT(GC_DS_TAGS==0x3);
switch(tag){
case GC_DS_LENGTH:
GC_ASSERT(descr < (word)GC_greatest_plausible_heap_addr
- (word)GC_least_plausible_heap_addr
||(word)(current_p+descr)
<=(word)GC_least_plausible_heap_addr
||(word)current_p>=(word)GC_greatest_plausible_heap_addr);
#ifdef PARALLEL_MARK
#define SHARE_BYTES 2048
if (descr > SHARE_BYTES&&GC_parallel
&&(word)mark_stack_top < (word)(mark_stack_limit - 1)){
word new_size=(descr/2)&~(word)(sizeof(word)-1);
mark_stack_top->mse_start=current_p;
mark_stack_top->mse_descr.w=new_size+sizeof(word);
mark_stack_top++;
#ifdef ENABLE_TRACE
if ((word)GC_trace_addr>=(word)current_p
&&(word)GC_trace_addr < (word)(current_p+descr)){
GC_log_printf("GC #%u:large section;start %p,len %lu,"
" splitting (parallel)at %p\n",
(unsigned)GC_gc_no,(void*)current_p,
(unsigned long)descr,
(void*)(current_p+new_size));
}
#endif
current_p+=new_size;
descr-=new_size;
goto retry;
}
#endif
mark_stack_top->mse_start=
limit=current_p+WORDS_TO_BYTES(SPLIT_RANGE_WORDS-1);
mark_stack_top->mse_descr.w=
descr - WORDS_TO_BYTES(SPLIT_RANGE_WORDS-1);
#ifdef ENABLE_TRACE
if ((word)GC_trace_addr>=(word)current_p
&&(word)GC_trace_addr < (word)(current_p+descr)){
GC_log_printf("GC #%u:large section;start %p,len %lu,"
" splitting at %p\n",
(unsigned)GC_gc_no,(void*)current_p,
(unsigned long)descr,(void*)limit);
}
#endif
limit+=sizeof(word)- ALIGNMENT;
break;
case GC_DS_BITMAP:
mark_stack_top--;
#ifdef ENABLE_TRACE
if ((word)GC_trace_addr>=(word)current_p
&&(word)GC_trace_addr < (word)(current_p
+WORDS_TO_BYTES(WORDSZ-2))){
GC_log_printf("GC #%u:tracing from %p bitmap descr %lu\n",
(unsigned)GC_gc_no,(void*)current_p,
(unsigned long)descr);
}
#endif
descr&=~GC_DS_TAGS;
credit-=WORDS_TO_BYTES(WORDSZ/2);
while (descr!=0){
if ((descr&SIGNB)!=0){
current=*(word*)current_p;
FIXUP_POINTER(current);
if (current>=(word)least_ha&&current < (word)greatest_ha){
PREFETCH((ptr_t)current);
#ifdef ENABLE_TRACE
if (GC_trace_addr==current_p){
GC_log_printf("GC #%u:considering(3)%p->%p\n",
(unsigned)GC_gc_no,(void*)current_p,
(void*)current);
}
#endif
PUSH_CONTENTS((ptr_t)current,mark_stack_top,
mark_stack_limit,current_p);
}
}
descr<<=1;
current_p+=sizeof(word);
}
continue;
case GC_DS_PROC:
mark_stack_top--;
#ifdef ENABLE_TRACE
if ((word)GC_trace_addr>=(word)current_p
&&GC_base(current_p)!=0
&&GC_base(current_p)==GC_base(GC_trace_addr)){
GC_log_printf("GC #%u:tracing from %p,proc descr %lu\n",
(unsigned)GC_gc_no,(void*)current_p,
(unsigned long)descr);
}
#endif
credit-=GC_PROC_BYTES;
mark_stack_top=(*PROC(descr))((word*)current_p,mark_stack_top,
mark_stack_limit,ENV(descr));
continue;
case GC_DS_PER_OBJECT:
if ((signed_word)descr>=0){
descr=*(word*)(current_p+descr - GC_DS_PER_OBJECT);
} else {
ptr_t type_descr=*(ptr_t*)current_p;
if (EXPECT(0==type_descr,FALSE)){
mark_stack_top--;
continue;
}
descr=*(word*)(type_descr
- ((signed_word)descr+(GC_INDIR_PER_OBJ_BIAS
- GC_DS_PER_OBJECT)));
}
if (0==descr){
mark_stack_top--;
continue;
}
goto retry;
}
} else {
mark_stack_top--;
#ifndef SMALL_CONFIG
if (descr < sizeof(word))
continue;
#endif
#ifdef ENABLE_TRACE
if ((word)GC_trace_addr>=(word)current_p
&&(word)GC_trace_addr < (word)(current_p+descr)){
GC_log_printf("GC #%u:small object;start %p,len %lu\n",
(unsigned)GC_gc_no,(void*)current_p,
(unsigned long)descr);
}
#endif
limit=current_p+(word)descr;
}
GC_ASSERT(!((word)current_p&(ALIGNMENT-1)));
credit-=limit - current_p;
limit-=sizeof(word);
{
#define PREF_DIST 4
#ifndef SMALL_CONFIG
word deferred;
for(;;){
PREFETCH(limit - PREF_DIST*CACHE_LINE_SIZE);
GC_ASSERT((word)limit>=(word)current_p);
deferred=*(word*)limit;
FIXUP_POINTER(deferred);
limit-=ALIGNMENT;
if (deferred>=(word)least_ha&&deferred < (word)greatest_ha){
PREFETCH((ptr_t)deferred);
break;
}
if ((word)current_p > (word)limit)goto next_object;
deferred=*(word*)limit;
FIXUP_POINTER(deferred);
limit-=ALIGNMENT;
if (deferred>=(word)least_ha&&deferred < (word)greatest_ha){
PREFETCH((ptr_t)deferred);
break;
}
if ((word)current_p > (word)limit)goto next_object;
}
#endif
while ((word)current_p<=(word)limit){
current=*(word*)current_p;
FIXUP_POINTER(current);
PREFETCH(current_p+PREF_DIST*CACHE_LINE_SIZE);
if (current>=(word)least_ha&&current < (word)greatest_ha){
PREFETCH((ptr_t)current);
#ifdef ENABLE_TRACE
if (GC_trace_addr==current_p){
GC_log_printf("GC #%u:considering(1)%p->%p\n",
(unsigned)GC_gc_no,(void*)current_p,
(void*)current);
}
#endif
PUSH_CONTENTS((ptr_t)current,mark_stack_top,
mark_stack_limit,current_p);
}
current_p+=ALIGNMENT;
}
#ifndef SMALL_CONFIG
#ifdef ENABLE_TRACE
if (GC_trace_addr==current_p){
GC_log_printf("GC #%u:considering(2)%p->%p\n",
(unsigned)GC_gc_no,(void*)current_p,
(void*)deferred);
}
#endif
PUSH_CONTENTS((ptr_t)deferred,mark_stack_top,
mark_stack_limit,current_p);
next_object:;
#endif
}
}
return mark_stack_top;
}
#ifdef PARALLEL_MARK
STATIC GC_bool GC_help_wanted=FALSE;
STATIC unsigned GC_helper_count=0;
STATIC unsigned GC_active_count=0;
GC_INNER word GC_mark_no=0;
#ifdef LINT2
#define LOCAL_MARK_STACK_SIZE (HBLKSIZE/8)
#else
#define LOCAL_MARK_STACK_SIZE HBLKSIZE
#endif
GC_INNER void GC_wait_for_markers_init(void)
{
signed_word count;
if (GC_markers_m1==0)
return;
#ifndef CAN_HANDLE_FORK
GC_ASSERT(NULL==GC_main_local_mark_stack);
#else
if (NULL==GC_main_local_mark_stack)
#endif
{
size_t bytes_to_get=
ROUNDUP_PAGESIZE_IF_MMAP(LOCAL_MARK_STACK_SIZE*sizeof(mse));
GC_ASSERT(GC_page_size!=0);
GC_main_local_mark_stack=(mse*)GET_MEM(bytes_to_get);
if (NULL==GC_main_local_mark_stack)
ABORT("Insufficient memory for main local_mark_stack");
GC_add_to_our_memory((ptr_t)GC_main_local_mark_stack,bytes_to_get);
}
GC_acquire_mark_lock();
GC_fl_builder_count+=GC_markers_m1;
count=GC_fl_builder_count;
GC_release_mark_lock();
if (count!=0){
GC_ASSERT(count > 0);
GC_wait_for_reclaim();
}
}
STATIC mse*GC_steal_mark_stack(mse*low,mse*high,mse*local,
unsigned max,mse**next)
{
mse*p;
mse*top=local - 1;
unsigned i=0;
GC_ASSERT((word)high>=(word)(low - 1)
&&(word)(high - low+1)<=GC_mark_stack_size);
for (p=low;(word)p<=(word)high&&i<=max;++p){
word descr=(word)AO_load(&p->mse_descr.ao);
if (descr!=0){
AO_store_release_write(&p->mse_descr.ao,0);
++top;
top->mse_descr.w=descr;
top->mse_start=p->mse_start;
GC_ASSERT((descr&GC_DS_TAGS)!=GC_DS_LENGTH
||descr < (word)GC_greatest_plausible_heap_addr
- (word)GC_least_plausible_heap_addr
||(word)(p->mse_start+descr)
<=(word)GC_least_plausible_heap_addr
||(word)p->mse_start
>=(word)GC_greatest_plausible_heap_addr);
++i;
if ((descr&GC_DS_TAGS)==GC_DS_LENGTH)i+=(int)(descr>>8);
}
}
*next=p;
return top;
}
STATIC void GC_return_mark_stack(mse*low,mse*high)
{
mse*my_top;
mse*my_start;
size_t stack_size;
if ((word)high < (word)low)return;
stack_size=high - low+1;
GC_acquire_mark_lock();
my_top=GC_mark_stack_top;
my_start=my_top+1;
if ((word)(my_start - GC_mark_stack+stack_size)
> (word)GC_mark_stack_size){
GC_COND_LOG_PRINTF("No room to copy back mark stack\n");
GC_mark_state=MS_INVALID;
GC_mark_stack_too_small=TRUE;
} else {
BCOPY(low,my_start,stack_size*sizeof(mse));
GC_ASSERT((mse*)AO_load((volatile AO_t*)(&GC_mark_stack_top))
==my_top);
AO_store_release_write((volatile AO_t*)(&GC_mark_stack_top),
(AO_t)(my_top+stack_size));
}
GC_release_mark_lock();
GC_notify_all_marker();
}
#ifndef N_LOCAL_ITERS
#define N_LOCAL_ITERS 1
#endif
static GC_bool has_inactive_helpers(void)
{
GC_bool res;
GC_acquire_mark_lock();
res=GC_active_count < GC_helper_count;
GC_release_mark_lock();
return res;
}
STATIC void GC_do_local_mark(mse*local_mark_stack,mse*local_top)
{
unsigned n;
for (;;){
for (n=0;n < N_LOCAL_ITERS;++n){
local_top=GC_mark_from(local_top,local_mark_stack,
local_mark_stack+LOCAL_MARK_STACK_SIZE);
if ((word)local_top < (word)local_mark_stack)return;
if ((word)(local_top - local_mark_stack)
>=LOCAL_MARK_STACK_SIZE/2){
GC_return_mark_stack(local_mark_stack,local_top);
return;
}
}
if ((word)AO_load((volatile AO_t*)&GC_mark_stack_top)
< (word)AO_load(&GC_first_nonempty)
&&(word)local_top > (word)(local_mark_stack+1)
&&has_inactive_helpers()){
mse*new_bottom=local_mark_stack
+(local_top - local_mark_stack)/2;
GC_ASSERT((word)new_bottom > (word)local_mark_stack
&&(word)new_bottom < (word)local_top);
GC_return_mark_stack(local_mark_stack,new_bottom - 1);
memmove(local_mark_stack,new_bottom,
(local_top - new_bottom+1)*sizeof(mse));
local_top-=(new_bottom - local_mark_stack);
}
}
}
#ifndef ENTRIES_TO_GET
#define ENTRIES_TO_GET 5
#endif
STATIC void GC_mark_local(mse*local_mark_stack,int id)
{
mse*my_first_nonempty;
GC_active_count++;
my_first_nonempty=(mse*)AO_load(&GC_first_nonempty);
GC_ASSERT((word)GC_mark_stack<=(word)my_first_nonempty);
GC_ASSERT((word)my_first_nonempty
<=(word)AO_load((volatile AO_t*)&GC_mark_stack_top)+sizeof(mse));
GC_VERBOSE_LOG_PRINTF("Starting mark helper %d\n",id);
GC_release_mark_lock();
for (;;){
size_t n_on_stack;
unsigned n_to_get;
mse*my_top;
mse*local_top;
mse*global_first_nonempty=(mse*)AO_load(&GC_first_nonempty);
GC_ASSERT((word)my_first_nonempty>=(word)GC_mark_stack&&
(word)my_first_nonempty<=
(word)AO_load((volatile AO_t*)&GC_mark_stack_top)
+sizeof(mse));
GC_ASSERT((word)global_first_nonempty>=(word)GC_mark_stack);
if ((word)my_first_nonempty < (word)global_first_nonempty){
my_first_nonempty=global_first_nonempty;
} else if ((word)global_first_nonempty < (word)my_first_nonempty){
(void)AO_compare_and_swap(&GC_first_nonempty,
(AO_t)global_first_nonempty,
(AO_t)my_first_nonempty);
}
my_top=(mse*)AO_load_acquire((volatile AO_t*)(&GC_mark_stack_top));
if ((word)my_top < (word)my_first_nonempty){
GC_acquire_mark_lock();
my_top=GC_mark_stack_top;
n_on_stack=my_top - my_first_nonempty+1;
if (0==n_on_stack){
GC_active_count--;
GC_ASSERT(GC_active_count<=GC_helper_count);
if (0==GC_active_count)GC_notify_all_marker();
while (GC_active_count > 0
&&(word)AO_load(&GC_first_nonempty)
> (word)GC_mark_stack_top){
GC_wait_marker();
}
if (GC_active_count==0
&&(word)AO_load(&GC_first_nonempty)
> (word)GC_mark_stack_top){
GC_bool need_to_notify=FALSE;
GC_helper_count--;
if (0==GC_helper_count)need_to_notify=TRUE;
GC_VERBOSE_LOG_PRINTF("Finished mark helper %d\n",id);
if (need_to_notify)GC_notify_all_marker();
return;
}
GC_active_count++;
GC_ASSERT(GC_active_count > 0);
GC_release_mark_lock();
continue;
} else {
GC_release_mark_lock();
}
} else {
n_on_stack=my_top - my_first_nonempty+1;
}
n_to_get=ENTRIES_TO_GET;
if (n_on_stack < 2*ENTRIES_TO_GET)n_to_get=1;
local_top=GC_steal_mark_stack(my_first_nonempty,my_top,
local_mark_stack,n_to_get,
&my_first_nonempty);
GC_ASSERT((word)my_first_nonempty>=(word)GC_mark_stack&&
(word)my_first_nonempty<=
(word)AO_load((volatile AO_t*)&GC_mark_stack_top)
+sizeof(mse));
GC_do_local_mark(local_mark_stack,local_top);
}
}
STATIC void GC_do_parallel_mark(void)
{
GC_acquire_mark_lock();
GC_ASSERT(I_HOLD_LOCK());
if (GC_help_wanted||GC_active_count!=0||GC_helper_count!=0)
ABORT("Tried to start parallel mark in bad state");
GC_VERBOSE_LOG_PRINTF("Starting marking for mark phase number %lu\n",
(unsigned long)GC_mark_no);
GC_first_nonempty=(AO_t)GC_mark_stack;
GC_active_count=0;
GC_helper_count=1;
GC_help_wanted=TRUE;
GC_notify_all_marker();
GC_mark_local(GC_main_local_mark_stack,0);
GC_help_wanted=FALSE;
while (GC_helper_count > 0){
GC_wait_marker();
}
GC_VERBOSE_LOG_PRINTF("Finished marking for mark phase number %lu\n",
(unsigned long)GC_mark_no);
GC_mark_no++;
GC_release_mark_lock();
GC_notify_all_marker();
}
GC_INNER void GC_help_marker(word my_mark_no)
{
#define my_id my_id_mse.mse_descr.w
mse my_id_mse;
mse local_mark_stack[LOCAL_MARK_STACK_SIZE];
GC_ASSERT(GC_parallel);
while (GC_mark_no < my_mark_no
||(!GC_help_wanted&&GC_mark_no==my_mark_no)){
GC_wait_marker();
}
my_id=GC_helper_count;
if (GC_mark_no!=my_mark_no||my_id > (unsigned)GC_markers_m1){
return;
}
GC_helper_count=(unsigned)my_id+1;
GC_mark_local(local_mark_stack,(int)my_id);
#undef my_id
}
#endif
GC_INNER void GC_scratch_recycle_inner(void*ptr,size_t bytes)
{
if (ptr!=NULL){
size_t page_offset=(word)ptr&(GC_page_size - 1);
size_t displ=0;
size_t recycled_bytes;
GC_ASSERT(bytes!=0);
GC_ASSERT(GC_page_size!=0);
if (page_offset!=0)
displ=GC_page_size - page_offset;
recycled_bytes=(bytes - displ)&~(GC_page_size - 1);
GC_COND_LOG_PRINTF("Recycle %lu scratch-allocated bytes at %p\n",
(unsigned long)recycled_bytes,ptr);
if (recycled_bytes > 0)
GC_add_to_heap((struct hblk*)((word)ptr+displ),recycled_bytes);
}
}
static void alloc_mark_stack(size_t n)
{
mse*new_stack=(mse*)GC_scratch_alloc(n*sizeof(struct GC_ms_entry));
#ifdef GWW_VDB
static GC_bool GC_incremental_at_stack_alloc=FALSE;
GC_bool recycle_old=!GC_auto_incremental
||GC_incremental_at_stack_alloc;
GC_incremental_at_stack_alloc=GC_auto_incremental;
#else
#define recycle_old TRUE
#endif
GC_mark_stack_too_small=FALSE;
if (GC_mark_stack!=NULL){
if (new_stack!=0){
if (recycle_old){
GC_scratch_recycle_inner(GC_mark_stack,
GC_mark_stack_size*sizeof(struct GC_ms_entry));
}
GC_mark_stack=new_stack;
GC_mark_stack_size=n;
GC_mark_stack_limit=new_stack+n;
GC_COND_LOG_PRINTF("Grew mark stack to %lu frames\n",
(unsigned long)GC_mark_stack_size);
} else {
WARN("Failed to grow mark stack to %" WARN_PRIdPTR " frames\n",n);
}
} else if (NULL==new_stack){
GC_err_printf("No space for mark stack\n");
EXIT();
} else {
GC_mark_stack=new_stack;
GC_mark_stack_size=n;
GC_mark_stack_limit=new_stack+n;
}
GC_mark_stack_top=GC_mark_stack-1;
}
GC_INNER void GC_mark_init(void)
{
alloc_mark_stack(INITIAL_MARK_STACK_SIZE);
}
GC_API void GC_CALL GC_push_all(void*bottom,void*top)
{
word length;
bottom=(void*)(((word)bottom+ALIGNMENT-1)&~(ALIGNMENT-1));
top=(void*)((word)top&~(ALIGNMENT-1));
if ((word)bottom>=(word)top)return;
GC_mark_stack_top++;
if ((word)GC_mark_stack_top>=(word)GC_mark_stack_limit){
ABORT("Unexpected mark stack overflow");
}
length=(word)top - (word)bottom;
#if GC_DS_TAGS > ALIGNMENT - 1
length+=GC_DS_TAGS;
length&=~GC_DS_TAGS;
#endif
GC_mark_stack_top->mse_start=(ptr_t)bottom;
GC_mark_stack_top->mse_descr.w=length;
}
#ifndef GC_DISABLE_INCREMENTAL
STATIC void GC_push_selected(ptr_t bottom,ptr_t top,
GC_bool (*dirty_fn)(struct hblk*))
{
struct hblk*h;
bottom=(ptr_t)(((word)bottom+ALIGNMENT-1)&~(ALIGNMENT-1));
top=(ptr_t)(((word)top)&~(ALIGNMENT-1));
if ((word)bottom>=(word)top)return;
h=HBLKPTR(bottom+HBLKSIZE);
if ((word)top<=(word)h){
if ((*dirty_fn)(h-1)){
GC_push_all(bottom,top);
}
return;
}
if ((*dirty_fn)(h-1)){
if ((word)(GC_mark_stack_top - GC_mark_stack)
> 3*GC_mark_stack_size/4){
GC_push_all(bottom,top);
return;
}
GC_push_all(bottom,h);
}
while ((word)(h+1)<=(word)top){
if ((*dirty_fn)(h)){
if ((word)(GC_mark_stack_top - GC_mark_stack)
> 3*GC_mark_stack_size/4){
GC_push_all(h,top);
return;
} else {
GC_push_all(h,h+1);
}
}
h++;
}
if ((ptr_t)h!=top&&(*dirty_fn)(h)){
GC_push_all(h,top);
}
}
GC_API void GC_CALL GC_push_conditional(void*bottom,void*top,int all)
{
if (!all){
GC_push_selected((ptr_t)bottom,(ptr_t)top,GC_page_was_dirty);
} else {
#ifdef PROC_VDB
if (GC_auto_incremental){
GC_push_selected((ptr_t)bottom,(ptr_t)top,GC_page_was_ever_dirty);
} else
#endif
{
GC_push_all(bottom,top);
}
}
}
#else
GC_API void GC_CALL GC_push_conditional(void*bottom,void*top,
int all GC_ATTR_UNUSED)
{
GC_push_all(bottom,top);
}
#endif
#if defined(AMIGA)||defined(MACOS)||defined(GC_DARWIN_THREADS)
void GC_push_one(word p)
{
GC_PUSH_ONE_STACK(p,MARKED_FROM_REGISTER);
}
#endif
#ifdef GC_WIN32_THREADS
GC_INNER void GC_push_many_regs(const word*regs,unsigned count)
{
unsigned i;
for (i=0;i < count;i++)
GC_PUSH_ONE_STACK(regs[i],MARKED_FROM_REGISTER);
}
#endif
GC_API struct GC_ms_entry*GC_CALL GC_mark_and_push(void*obj,
mse*mark_stack_ptr,
mse*mark_stack_limit,
void**src GC_ATTR_UNUSED)
{
hdr*hhdr;
PREFETCH(obj);
GET_HDR(obj,hhdr);
if ((EXPECT(IS_FORWARDING_ADDR_OR_NIL(hhdr),FALSE)
&&(!GC_all_interior_pointers
||NULL==(hhdr=GC_find_header((ptr_t)GC_base(obj)))))
||EXPECT(HBLK_IS_FREE(hhdr),FALSE)){
GC_ADD_TO_BLACK_LIST_NORMAL(obj,(ptr_t)src);
return mark_stack_ptr;
}
return GC_push_contents_hdr((ptr_t)obj,mark_stack_ptr,mark_stack_limit,
(ptr_t)src,hhdr,TRUE);
}
#if defined(PRINT_BLACK_LIST)||defined(KEEP_BACK_PTRS)
GC_INNER void GC_mark_and_push_stack(ptr_t p,ptr_t source)
#else
GC_INNER void GC_mark_and_push_stack(ptr_t p)
#define source ((ptr_t)0)
#endif
{
hdr*hhdr;
ptr_t r=p;
PREFETCH(p);
GET_HDR(p,hhdr);
if (EXPECT(IS_FORWARDING_ADDR_OR_NIL(hhdr),FALSE)){
if (NULL==hhdr
||(r=(ptr_t)GC_base(p))==NULL
||(hhdr=HDR(r))==NULL){
GC_ADD_TO_BLACK_LIST_STACK(p,source);
return;
}
}
if (EXPECT(HBLK_IS_FREE(hhdr),FALSE)){
GC_ADD_TO_BLACK_LIST_NORMAL(p,source);
return;
}
#ifdef THREADS
GC_dirty(p);
#endif
GC_mark_stack_top=GC_push_contents_hdr(r,GC_mark_stack_top,
GC_mark_stack_limit,
source,hhdr,FALSE);
}
#undef source
#ifdef TRACE_BUF
#ifndef TRACE_ENTRIES
#define TRACE_ENTRIES 1000
#endif
struct trace_entry {
char*kind;
word gc_no;
word bytes_allocd;
word arg1;
word arg2;
} GC_trace_buf[TRACE_ENTRIES]={ { NULL,0,0,0,0 } };
void GC_add_trace_entry(char*kind,word arg1,word arg2)
{
GC_trace_buf[GC_trace_buf_ptr].kind=kind;
GC_trace_buf[GC_trace_buf_ptr].gc_no=GC_gc_no;
GC_trace_buf[GC_trace_buf_ptr].bytes_allocd=GC_bytes_allocd;
GC_trace_buf[GC_trace_buf_ptr].arg1=arg1^0x80000000;
GC_trace_buf[GC_trace_buf_ptr].arg2=arg2^0x80000000;
GC_trace_buf_ptr++;
if (GC_trace_buf_ptr>=TRACE_ENTRIES)GC_trace_buf_ptr=0;
}
GC_API void GC_CALL GC_print_trace_inner(word gc_no)
{
int i;
for (i=GC_trace_buf_ptr-1;i!=GC_trace_buf_ptr;i--){
struct trace_entry*p;
if (i < 0)i=TRACE_ENTRIES-1;
p=GC_trace_buf+i;
if (p->gc_no < gc_no||p->kind==0){
return;
}
GC_printf("Trace:%s (gc:%u,bytes:%lu)0x%lX,0x%lX\n",
p->kind,(unsigned)p->gc_no,
(unsigned long)p->bytes_allocd,
(long)p->arg1^0x80000000L,(long)p->arg2^0x80000000L);
}
GC_printf("Trace incomplete\n");
}
GC_API void GC_CALL GC_print_trace(word gc_no)
{
DCL_LOCK_STATE;
LOCK();
GC_print_trace_inner(gc_no);
UNLOCK();
}
#endif
GC_ATTR_NO_SANITIZE_ADDR GC_ATTR_NO_SANITIZE_MEMORY GC_ATTR_NO_SANITIZE_THREAD
GC_API void GC_CALL GC_push_all_eager(void*bottom,void*top)
{
word*b=(word*)(((word)bottom+ALIGNMENT-1)&~(ALIGNMENT-1));
word*t=(word*)(((word)top)&~(ALIGNMENT-1));
REGISTER word*p;
REGISTER word*lim;
REGISTER ptr_t greatest_ha=(ptr_t)GC_greatest_plausible_heap_addr;
REGISTER ptr_t least_ha=(ptr_t)GC_least_plausible_heap_addr;
#define GC_greatest_plausible_heap_addr greatest_ha
#define GC_least_plausible_heap_addr least_ha
if (top==0)return;
lim=t - 1;
for (p=b;(word)p<=(word)lim;
p=(word*)(((ptr_t)p)+ALIGNMENT)){
REGISTER word q=*p;
GC_PUSH_ONE_STACK(q,p);
}
#undef GC_greatest_plausible_heap_addr
#undef GC_least_plausible_heap_addr
}
GC_INNER void GC_push_all_stack(ptr_t bottom,ptr_t top)
{
#ifndef NEED_FIXUP_POINTER
if (GC_all_interior_pointers
#if defined(THREADS)&&defined(MPROTECT_VDB)
&&!GC_auto_incremental
#endif
&&(word)GC_mark_stack_top
< (word)(GC_mark_stack_limit - INITIAL_MARK_STACK_SIZE/8)){
GC_push_all(bottom,top);
} else
#endif
{
GC_push_all_eager(bottom,top);
}
}
#if defined(WRAP_MARK_SOME)&&defined(PARALLEL_MARK)
GC_ATTR_NO_SANITIZE_ADDR GC_ATTR_NO_SANITIZE_MEMORY
GC_ATTR_NO_SANITIZE_THREAD
GC_INNER void GC_push_conditional_eager(void*bottom,void*top,
GC_bool all)
{
word*b=(word*)(((word)bottom+ALIGNMENT-1)&~(ALIGNMENT-1));
word*t=(word*)(((word)top)&~(ALIGNMENT-1));
REGISTER word*p;
REGISTER word*lim;
REGISTER ptr_t greatest_ha=(ptr_t)GC_greatest_plausible_heap_addr;
REGISTER ptr_t least_ha=(ptr_t)GC_least_plausible_heap_addr;
#define GC_greatest_plausible_heap_addr greatest_ha
#define GC_least_plausible_heap_addr least_ha
if (top==NULL)
return;
(void)all;
lim=t - 1;
for (p=b;(word)p<=(word)lim;p=(word*)((ptr_t)p+ALIGNMENT)){
REGISTER word q=*p;
GC_PUSH_ONE_HEAP(q,p,GC_mark_stack_top);
}
#undef GC_greatest_plausible_heap_addr
#undef GC_least_plausible_heap_addr
}
#endif
#if!defined(SMALL_CONFIG)&&!defined(USE_MARK_BYTES)&&defined(MARK_BIT_PER_GRANULE)
#if GC_GRANULE_WORDS==1
#define USE_PUSH_MARKED_ACCELERATORS
#define PUSH_GRANULE(q)do { word qcontents=(q)[0];GC_PUSH_ONE_HEAP(qcontents,q,GC_mark_stack_top);} while (0)
#elif GC_GRANULE_WORDS==2
#define USE_PUSH_MARKED_ACCELERATORS
#define PUSH_GRANULE(q)do { word qcontents=(q)[0];GC_PUSH_ONE_HEAP(qcontents,q,GC_mark_stack_top);qcontents=(q)[1];GC_PUSH_ONE_HEAP(qcontents,(q)+1,GC_mark_stack_top);} while (0)
#elif GC_GRANULE_WORDS==4
#define USE_PUSH_MARKED_ACCELERATORS
#define PUSH_GRANULE(q)do { word qcontents=(q)[0];GC_PUSH_ONE_HEAP(qcontents,q,GC_mark_stack_top);qcontents=(q)[1];GC_PUSH_ONE_HEAP(qcontents,(q)+1,GC_mark_stack_top);qcontents=(q)[2];GC_PUSH_ONE_HEAP(qcontents,(q)+2,GC_mark_stack_top);qcontents=(q)[3];GC_PUSH_ONE_HEAP(qcontents,(q)+3,GC_mark_stack_top);} while (0)
#endif
#endif
#ifdef USE_PUSH_MARKED_ACCELERATORS
STATIC void GC_push_marked1(struct hblk*h,hdr*hhdr)
{
word*mark_word_addr=&(hhdr->hb_marks[0]);
word*p;
word*plim;
ptr_t greatest_ha=(ptr_t)GC_greatest_plausible_heap_addr;
ptr_t least_ha=(ptr_t)GC_least_plausible_heap_addr;
mse*mark_stack_top=GC_mark_stack_top;
mse*mark_stack_limit=GC_mark_stack_limit;
#undef GC_mark_stack_top
#undef GC_mark_stack_limit
#define GC_mark_stack_top mark_stack_top
#define GC_mark_stack_limit mark_stack_limit
#define GC_greatest_plausible_heap_addr greatest_ha
#define GC_least_plausible_heap_addr least_ha
p=(word*)(h->hb_body);
plim=(word*)(((word)h)+HBLKSIZE);
while ((word)p < (word)plim){
word mark_word=*mark_word_addr++;
word*q=p;
while(mark_word!=0){
if (mark_word&1){
PUSH_GRANULE(q);
}
q+=GC_GRANULE_WORDS;
mark_word>>=1;
}
p+=WORDSZ*GC_GRANULE_WORDS;
}
#undef GC_greatest_plausible_heap_addr
#undef GC_least_plausible_heap_addr
#undef GC_mark_stack_top
#undef GC_mark_stack_limit
#define GC_mark_stack_limit GC_arrays._mark_stack_limit
#define GC_mark_stack_top GC_arrays._mark_stack_top
GC_mark_stack_top=mark_stack_top;
}
#ifndef UNALIGNED_PTRS
STATIC void GC_push_marked2(struct hblk*h,hdr*hhdr)
{
word*mark_word_addr=&(hhdr->hb_marks[0]);
word*p;
word*plim;
ptr_t greatest_ha=(ptr_t)GC_greatest_plausible_heap_addr;
ptr_t least_ha=(ptr_t)GC_least_plausible_heap_addr;
mse*mark_stack_top=GC_mark_stack_top;
mse*mark_stack_limit=GC_mark_stack_limit;
#undef GC_mark_stack_top
#undef GC_mark_stack_limit
#define GC_mark_stack_top mark_stack_top
#define GC_mark_stack_limit mark_stack_limit
#define GC_greatest_plausible_heap_addr greatest_ha
#define GC_least_plausible_heap_addr least_ha
p=(word*)(h->hb_body);
plim=(word*)(((word)h)+HBLKSIZE);
while ((word)p < (word)plim){
word mark_word=*mark_word_addr++;
word*q=p;
while(mark_word!=0){
if (mark_word&1){
PUSH_GRANULE(q);
PUSH_GRANULE(q+GC_GRANULE_WORDS);
}
q+=2*GC_GRANULE_WORDS;
mark_word>>=2;
}
p+=WORDSZ*GC_GRANULE_WORDS;
}
#undef GC_greatest_plausible_heap_addr
#undef GC_least_plausible_heap_addr
#undef GC_mark_stack_top
#undef GC_mark_stack_limit
#define GC_mark_stack_limit GC_arrays._mark_stack_limit
#define GC_mark_stack_top GC_arrays._mark_stack_top
GC_mark_stack_top=mark_stack_top;
}
#if GC_GRANULE_WORDS < 4
STATIC void GC_push_marked4(struct hblk*h,hdr*hhdr)
{
word*mark_word_addr=&(hhdr->hb_marks[0]);
word*p;
word*plim;
ptr_t greatest_ha=(ptr_t)GC_greatest_plausible_heap_addr;
ptr_t least_ha=(ptr_t)GC_least_plausible_heap_addr;
mse*mark_stack_top=GC_mark_stack_top;
mse*mark_stack_limit=GC_mark_stack_limit;
#undef GC_mark_stack_top
#undef GC_mark_stack_limit
#define GC_mark_stack_top mark_stack_top
#define GC_mark_stack_limit mark_stack_limit
#define GC_greatest_plausible_heap_addr greatest_ha
#define GC_least_plausible_heap_addr least_ha
p=(word*)(h->hb_body);
plim=(word*)(((word)h)+HBLKSIZE);
while ((word)p < (word)plim){
word mark_word=*mark_word_addr++;
word*q=p;
while(mark_word!=0){
if (mark_word&1){
PUSH_GRANULE(q);
PUSH_GRANULE(q+GC_GRANULE_WORDS);
PUSH_GRANULE(q+2*GC_GRANULE_WORDS);
PUSH_GRANULE(q+3*GC_GRANULE_WORDS);
}
q+=4*GC_GRANULE_WORDS;
mark_word>>=4;
}
p+=WORDSZ*GC_GRANULE_WORDS;
}
#undef GC_greatest_plausible_heap_addr
#undef GC_least_plausible_heap_addr
#undef GC_mark_stack_top
#undef GC_mark_stack_limit
#define GC_mark_stack_limit GC_arrays._mark_stack_limit
#define GC_mark_stack_top GC_arrays._mark_stack_top
GC_mark_stack_top=mark_stack_top;
}
#endif
#endif
#endif
STATIC void GC_push_marked(struct hblk*h,hdr*hhdr)
{
word sz=hhdr->hb_sz;
word descr=hhdr->hb_descr;
ptr_t p;
word bit_no;
ptr_t lim;
mse*GC_mark_stack_top_reg;
mse*mark_stack_limit=GC_mark_stack_limit;
if (( GC_DS_LENGTH)==descr)return;
if (GC_block_empty(hhdr))return;
#if!defined(GC_DISABLE_INCREMENTAL)
GC_n_rescuing_pages++;
#endif
GC_objects_are_marked=TRUE;
if (sz > MAXOBJBYTES){
lim=h->hb_body;
} else {
lim=(ptr_t)((word)(h+1)->hb_body - sz);
}
switch(BYTES_TO_GRANULES(sz)){
#if defined(USE_PUSH_MARKED_ACCELERATORS)
case 1:
GC_push_marked1(h,hhdr);
break;
#if!defined(UNALIGNED_PTRS)
case 2:
GC_push_marked2(h,hhdr);
break;
#if GC_GRANULE_WORDS < 4
case 4:
GC_push_marked4(h,hhdr);
break;
#endif
#endif
#else
case 1:
#endif
default:
GC_mark_stack_top_reg=GC_mark_stack_top;
for (p=h->hb_body,bit_no=0;(word)p<=(word)lim;
p+=sz,bit_no+=MARK_BIT_OFFSET(sz)){
if (mark_bit_from_hdr(hhdr,bit_no)){
GC_mark_stack_top_reg=GC_push_obj(p,hhdr,GC_mark_stack_top_reg,
mark_stack_limit);
}
}
GC_mark_stack_top=GC_mark_stack_top_reg;
}
}
#ifdef ENABLE_DISCLAIM
STATIC void GC_push_unconditionally(struct hblk*h,hdr*hhdr)
{
word sz=hhdr->hb_sz;
word descr=hhdr->hb_descr;
ptr_t p;
ptr_t lim;
mse*GC_mark_stack_top_reg;
mse*mark_stack_limit=GC_mark_stack_limit;
if (( GC_DS_LENGTH)==descr)
return;
#if!defined(GC_DISABLE_INCREMENTAL)
GC_n_rescuing_pages++;
#endif
GC_objects_are_marked=TRUE;
if (sz > MAXOBJBYTES)
lim=h->hb_body;
else
lim=(ptr_t)((word)(h+1)->hb_body - sz);
GC_mark_stack_top_reg=GC_mark_stack_top;
for (p=h->hb_body;(word)p<=(word)lim;p+=sz)
if ((*(word*)p&0x3)!=0)
GC_mark_stack_top_reg=GC_push_obj(p,hhdr,GC_mark_stack_top_reg,
mark_stack_limit);
GC_mark_stack_top=GC_mark_stack_top_reg;
}
#endif
#ifndef GC_DISABLE_INCREMENTAL
STATIC GC_bool GC_block_was_dirty(struct hblk*h,hdr*hhdr)
{
word sz=hhdr->hb_sz;
if (sz<=MAXOBJBYTES){
return(GC_page_was_dirty(h));
} else {
ptr_t p=(ptr_t)h;
while ((word)p < (word)h+sz){
if (GC_page_was_dirty((struct hblk*)p))return(TRUE);
p+=HBLKSIZE;
}
return(FALSE);
}
}
#endif
STATIC struct hblk*GC_push_next_marked(struct hblk*h)
{
hdr*hhdr=HDR(h);
if (EXPECT(IS_FORWARDING_ADDR_OR_NIL(hhdr)||HBLK_IS_FREE(hhdr),FALSE)){
h=GC_next_block(h,FALSE);
if (NULL==h)return NULL;
hhdr=GC_find_header((ptr_t)h);
} else {
#ifdef LINT2
if (NULL==h)ABORT("Bad HDR()definition");
#endif
}
GC_push_marked(h,hhdr);
return(h+OBJ_SZ_TO_BLOCKS(hhdr->hb_sz));
}
#ifndef GC_DISABLE_INCREMENTAL
STATIC struct hblk*GC_push_next_marked_dirty(struct hblk*h)
{
hdr*hhdr=HDR(h);
if (!GC_incremental)ABORT("Dirty bits not set up");
for (;;){
if (EXPECT(IS_FORWARDING_ADDR_OR_NIL(hhdr)
||HBLK_IS_FREE(hhdr),FALSE)){
h=GC_next_block(h,FALSE);
if (NULL==h)return NULL;
hhdr=GC_find_header((ptr_t)h);
} else {
#ifdef LINT2
if (NULL==h)ABORT("Bad HDR()definition");
#endif
}
if (GC_block_was_dirty(h,hhdr))
break;
h+=OBJ_SZ_TO_BLOCKS(hhdr->hb_sz);
hhdr=HDR(h);
}
#ifdef ENABLE_DISCLAIM
if ((hhdr->hb_flags&MARK_UNCONDITIONALLY)!=0){
GC_push_unconditionally(h,hhdr);
} else
#endif
{
GC_push_marked(h,hhdr);
}
return(h+OBJ_SZ_TO_BLOCKS(hhdr->hb_sz));
}
#endif
STATIC struct hblk*GC_push_next_marked_uncollectable(struct hblk*h)
{
hdr*hhdr=HDR(h);
for (;;){
if (EXPECT(IS_FORWARDING_ADDR_OR_NIL(hhdr)
||HBLK_IS_FREE(hhdr),FALSE)){
h=GC_next_block(h,FALSE);
if (NULL==h)return NULL;
hhdr=GC_find_header((ptr_t)h);
} else {
#ifdef LINT2
if (NULL==h)ABORT("Bad HDR()definition");
#endif
}
if (hhdr->hb_obj_kind==UNCOLLECTABLE){
GC_push_marked(h,hhdr);
break;
}
#ifdef ENABLE_DISCLAIM
if ((hhdr->hb_flags&MARK_UNCONDITIONALLY)!=0){
GC_push_unconditionally(h,hhdr);
break;
}
#endif
h+=OBJ_SZ_TO_BLOCKS(hhdr->hb_sz);
hhdr=HDR(h);
}
return(h+OBJ_SZ_TO_BLOCKS(hhdr->hb_sz));
}
#include <stdio.h>
int GC_no_dls=0;
#if!defined(NO_DEBUGGING)||defined(GC_ASSERTIONS)
GC_INNER word GC_compute_root_size(void)
{
int i;
word size=0;
for (i=0;i < n_root_sets;i++){
size+=GC_static_roots[i].r_end - GC_static_roots[i].r_start;
}
return size;
}
#endif
#if!defined(NO_DEBUGGING)
void GC_print_static_roots(void)
{
int i;
word size;
for (i=0;i < n_root_sets;i++){
GC_printf("From %p to %p%s\n",
(void*)GC_static_roots[i].r_start,
(void*)GC_static_roots[i].r_end,
GC_static_roots[i].r_tmp?" (temporary)":"");
}
GC_printf("GC_root_size:%lu\n",(unsigned long)GC_root_size);
if ((size=GC_compute_root_size())!=GC_root_size)
GC_err_printf("GC_root_size incorrect!!Should be:%lu\n",
(unsigned long)size);
}
#endif
#ifndef THREADS
GC_INNER GC_bool GC_is_static_root(void*p)
{
static int last_root_set=MAX_ROOT_SETS;
int i;
if (last_root_set < n_root_sets
&&(word)p>=(word)GC_static_roots[last_root_set].r_start
&&(word)p < (word)GC_static_roots[last_root_set].r_end)
return(TRUE);
for (i=0;i < n_root_sets;i++){
if ((word)p>=(word)GC_static_roots[i].r_start
&&(word)p < (word)GC_static_roots[i].r_end){
last_root_set=i;
return(TRUE);
}
}
return(FALSE);
}
#endif
#if!defined(MSWIN32)&&!defined(MSWINCE)&&!defined(CYGWIN32)
GC_INLINE int rt_hash(ptr_t addr)
{
word result=(word)addr;
#if CPP_WORDSZ > 8*LOG_RT_SIZE
result^=result>>8*LOG_RT_SIZE;
#endif
#if CPP_WORDSZ > 4*LOG_RT_SIZE
result^=result>>4*LOG_RT_SIZE;
#endif
result^=result>>2*LOG_RT_SIZE;
result^=result>>LOG_RT_SIZE;
result&=(RT_SIZE-1);
return(result);
}
GC_INNER void*GC_roots_present(ptr_t b)
{
int h=rt_hash(b);
struct roots*p=GC_root_index[h];
while (p!=0){
if (p->r_start==(ptr_t)b)return(p);
p=p->r_next;
}
return NULL;
}
GC_INLINE void add_roots_to_index(struct roots*p)
{
int h=rt_hash(p->r_start);
p->r_next=GC_root_index[h];
GC_root_index[h]=p;
}
#endif
GC_INNER word GC_root_size=0;
GC_API void GC_CALL GC_add_roots(void*b,void*e)
{
DCL_LOCK_STATE;
if (!EXPECT(GC_is_initialized,TRUE))GC_init();
LOCK();
GC_add_roots_inner((ptr_t)b,(ptr_t)e,FALSE);
UNLOCK();
}
void GC_add_roots_inner(ptr_t b,ptr_t e,GC_bool tmp)
{
GC_ASSERT((word)b<=(word)e);
b=(ptr_t)(((word)b+(sizeof(word)- 1))&~(word)(sizeof(word)- 1));
e=(ptr_t)((word)e&~(word)(sizeof(word)- 1));
if ((word)b>=(word)e)return;
#if defined(MSWIN32)||defined(MSWINCE)||defined(CYGWIN32)
{
int i;
struct roots*old=NULL;
for (i=0;i < n_root_sets;i++){
old=GC_static_roots+i;
if ((word)b<=(word)old->r_end
&&(word)e>=(word)old->r_start){
if ((word)b < (word)old->r_start){
GC_root_size+=old->r_start - b;
old->r_start=b;
}
if ((word)e > (word)old->r_end){
GC_root_size+=e - old->r_end;
old->r_end=e;
}
old->r_tmp&=tmp;
break;
}
}
if (i < n_root_sets){
struct roots*other;
for (i++;i < n_root_sets;i++){
other=GC_static_roots+i;
b=other->r_start;
e=other->r_end;
if ((word)b<=(word)old->r_end
&&(word)e>=(word)old->r_start){
if ((word)b < (word)old->r_start){
GC_root_size+=old->r_start - b;
old->r_start=b;
}
if ((word)e > (word)old->r_end){
GC_root_size+=e - old->r_end;
old->r_end=e;
}
old->r_tmp&=other->r_tmp;
GC_root_size-=(other->r_end - other->r_start);
other->r_start=GC_static_roots[n_root_sets-1].r_start;
other->r_end=GC_static_roots[n_root_sets-1].r_end;
n_root_sets--;
}
}
return;
}
}
#else
{
struct roots*old=(struct roots*)GC_roots_present(b);
if (old!=0){
if ((word)e<=(word)old->r_end){
old->r_tmp&=tmp;
return;
}
if (old->r_tmp==tmp||!tmp){
GC_root_size+=e - old->r_end;
old->r_end=e;
old->r_tmp=tmp;
return;
}
b=old->r_end;
}
}
#endif
if (n_root_sets==MAX_ROOT_SETS){
ABORT("Too many root sets");
}
#ifdef DEBUG_ADD_DEL_ROOTS
GC_log_printf("Adding data root section %d:%p .. %p%s\n",
n_root_sets,(void*)b,(void*)e,
tmp?" (temporary)":"");
#endif
GC_static_roots[n_root_sets].r_start=(ptr_t)b;
GC_static_roots[n_root_sets].r_end=(ptr_t)e;
GC_static_roots[n_root_sets].r_tmp=tmp;
#if!defined(MSWIN32)&&!defined(MSWINCE)&&!defined(CYGWIN32)
GC_static_roots[n_root_sets].r_next=0;
add_roots_to_index(GC_static_roots+n_root_sets);
#endif
GC_root_size+=e - b;
n_root_sets++;
}
GC_API void GC_CALL GC_clear_roots(void)
{
DCL_LOCK_STATE;
if (!EXPECT(GC_is_initialized,TRUE))GC_init();
LOCK();
#ifdef THREADS
GC_roots_were_cleared=TRUE;
#endif
n_root_sets=0;
GC_root_size=0;
#if!defined(MSWIN32)&&!defined(MSWINCE)&&!defined(CYGWIN32)
BZERO(GC_root_index,RT_SIZE*sizeof(void*));
#endif
#ifdef DEBUG_ADD_DEL_ROOTS
GC_log_printf("Clear all data root sections\n");
#endif
UNLOCK();
}
STATIC void GC_remove_root_at_pos(int i)
{
#ifdef DEBUG_ADD_DEL_ROOTS
GC_log_printf("Remove data root section at %d:%p .. %p%s\n",
i,(void*)GC_static_roots[i].r_start,
(void*)GC_static_roots[i].r_end,
GC_static_roots[i].r_tmp?" (temporary)":"");
#endif
GC_root_size-=(GC_static_roots[i].r_end - GC_static_roots[i].r_start);
GC_static_roots[i].r_start=GC_static_roots[n_root_sets-1].r_start;
GC_static_roots[i].r_end=GC_static_roots[n_root_sets-1].r_end;
GC_static_roots[i].r_tmp=GC_static_roots[n_root_sets-1].r_tmp;
n_root_sets--;
}
#if!defined(MSWIN32)&&!defined(MSWINCE)&&!defined(CYGWIN32)
STATIC void GC_rebuild_root_index(void)
{
int i;
BZERO(GC_root_index,RT_SIZE*sizeof(void*));
for (i=0;i < n_root_sets;i++)
add_roots_to_index(GC_static_roots+i);
}
#endif
#if defined(DYNAMIC_LOADING)||defined(MSWIN32)||defined(MSWINCE)||defined(PCR)||defined(CYGWIN32)
STATIC void GC_remove_tmp_roots(void)
{
int i;
#if!defined(MSWIN32)&&!defined(MSWINCE)&&!defined(CYGWIN32)
int old_n_roots=n_root_sets;
#endif
for (i=0;i < n_root_sets;){
if (GC_static_roots[i].r_tmp){
GC_remove_root_at_pos(i);
} else {
i++;
}
}
#if!defined(MSWIN32)&&!defined(MSWINCE)&&!defined(CYGWIN32)
if (n_root_sets < old_n_roots)
GC_rebuild_root_index();
#endif
}
#endif
#if!defined(MSWIN32)&&!defined(MSWINCE)&&!defined(CYGWIN32)
STATIC void GC_remove_roots_inner(ptr_t b,ptr_t e);
GC_API void GC_CALL GC_remove_roots(void*b,void*e)
{
DCL_LOCK_STATE;
if ((((word)b+(sizeof(word)- 1))&~(word)(sizeof(word)- 1))>=
((word)e&~(word)(sizeof(word)- 1)))
return;
LOCK();
GC_remove_roots_inner((ptr_t)b,(ptr_t)e);
UNLOCK();
}
STATIC void GC_remove_roots_inner(ptr_t b,ptr_t e)
{
int i;
GC_bool rebuild=FALSE;
for (i=0;i < n_root_sets;){
if ((word)GC_static_roots[i].r_start>=(word)b
&&(word)GC_static_roots[i].r_end<=(word)e){
GC_remove_root_at_pos(i);
rebuild=TRUE;
} else {
i++;
}
}
if (rebuild)
GC_rebuild_root_index();
}
#endif
#ifdef USE_PROC_FOR_LIBRARIES
GC_INLINE void swap_static_roots(int i,int j)
{
ptr_t r_start=GC_static_roots[i].r_start;
ptr_t r_end=GC_static_roots[i].r_end;
GC_bool r_tmp=GC_static_roots[i].r_tmp;
GC_static_roots[i].r_start=GC_static_roots[j].r_start;
GC_static_roots[i].r_end=GC_static_roots[j].r_end;
GC_static_roots[i].r_tmp=GC_static_roots[j].r_tmp;
GC_static_roots[j].r_start=r_start;
GC_static_roots[j].r_end=r_end;
GC_static_roots[j].r_tmp=r_tmp;
}
GC_INNER void GC_remove_roots_subregion(ptr_t b,ptr_t e)
{
int i;
GC_bool rebuild=FALSE;
GC_ASSERT(I_HOLD_LOCK());
GC_ASSERT((word)b % sizeof(word)==0&&(word)e % sizeof(word)==0);
for (i=0;i < n_root_sets;i++){
ptr_t r_start,r_end;
if (GC_static_roots[i].r_tmp){
#ifdef GC_ASSERTIONS
int j;
for (j=i+1;j < n_root_sets;j++){
GC_ASSERT(GC_static_roots[j].r_tmp);
}
#endif
break;
}
r_start=GC_static_roots[i].r_start;
r_end=GC_static_roots[i].r_end;
if (!EXPECT((word)e<=(word)r_start||(word)r_end<=(word)b,TRUE)){
#ifdef DEBUG_ADD_DEL_ROOTS
GC_log_printf("Removing %p .. %p from root section %d (%p .. %p)\n",
(void*)b,(void*)e,
i,(void*)r_start,(void*)r_end);
#endif
if ((word)r_start < (word)b){
GC_root_size-=r_end - b;
GC_static_roots[i].r_end=b;
if ((word)e < (word)r_end){
int j;
if (rebuild){
GC_rebuild_root_index();
rebuild=FALSE;
}
GC_add_roots_inner(e,r_end,FALSE);
for (j=i+1;j < n_root_sets;j++)
if (GC_static_roots[j].r_tmp)
break;
if (j < n_root_sets-1&&!GC_static_roots[n_root_sets-1].r_tmp){
swap_static_roots(j,n_root_sets - 1);
rebuild=TRUE;
}
}
} else {
if ((word)e < (word)r_end){
GC_root_size-=e - r_start;
GC_static_roots[i].r_start=e;
} else {
GC_remove_root_at_pos(i);
if (i < n_root_sets - 1&&GC_static_roots[i].r_tmp
&&!GC_static_roots[i+1].r_tmp){
int j;
for (j=i+2;j < n_root_sets;j++)
if (GC_static_roots[j].r_tmp)
break;
swap_static_roots(i,j - 1);
}
i--;
}
rebuild=TRUE;
}
}
}
if (rebuild)
GC_rebuild_root_index();
}
#endif
#if!defined(NO_DEBUGGING)
GC_API int GC_CALL GC_is_tmp_root(void*p)
{
static int last_root_set=MAX_ROOT_SETS;
int i;
if (last_root_set < n_root_sets
&&(word)p>=(word)GC_static_roots[last_root_set].r_start
&&(word)p < (word)GC_static_roots[last_root_set].r_end)
return GC_static_roots[last_root_set].r_tmp;
for (i=0;i < n_root_sets;i++){
if ((word)p>=(word)GC_static_roots[i].r_start
&&(word)p < (word)GC_static_roots[i].r_end){
last_root_set=i;
return GC_static_roots[i].r_tmp;
}
}
return(FALSE);
}
#endif
GC_INNER ptr_t GC_approx_sp(void)
{
volatile word sp;
#if defined(S390)&&!defined(CPPCHECK)&&(__clang_major__ < 8)
sp=(word)&sp;
#elif defined(CPPCHECK)||(__GNUC__>=4&&!defined(STACK_NOT_SCANNED))
sp=(word)__builtin_frame_address(0);
#else
sp=(word)&sp;
#endif
return((ptr_t)sp);
}
GC_API void GC_CALL GC_clear_exclusion_table(void)
{
GC_excl_table_entries=0;
}
STATIC struct exclusion*GC_next_exclusion(ptr_t start_addr)
{
size_t low=0;
size_t high;
GC_ASSERT(GC_excl_table_entries > 0);
high=GC_excl_table_entries - 1;
while (high > low){
size_t mid=(low+high)>>1;
if ((word)GC_excl_table[mid].e_end<=(word)start_addr){
low=mid+1;
} else {
high=mid;
}
}
if ((word)GC_excl_table[low].e_end<=(word)start_addr)return 0;
return GC_excl_table+low;
}
GC_INNER void GC_exclude_static_roots_inner(void*start,void*finish)
{
struct exclusion*next;
size_t next_index;
GC_ASSERT((word)start % sizeof(word)==0);
GC_ASSERT((word)start < (word)finish);
if (0==GC_excl_table_entries){
next=0;
} else {
next=GC_next_exclusion((ptr_t)start);
}
if (0!=next){
size_t i;
if ((word)(next->e_start)< (word)finish){
ABORT("Exclusion ranges overlap");
}
if ((word)(next->e_start)==(word)finish){
next->e_start=(ptr_t)start;
return;
}
next_index=next - GC_excl_table;
for (i=GC_excl_table_entries;i > next_index;--i){
GC_excl_table[i]=GC_excl_table[i-1];
}
} else {
next_index=GC_excl_table_entries;
}
if (GC_excl_table_entries==MAX_EXCLUSIONS)ABORT("Too many exclusions");
GC_excl_table[next_index].e_start=(ptr_t)start;
GC_excl_table[next_index].e_end=(ptr_t)finish;
++GC_excl_table_entries;
}
GC_API void GC_CALL GC_exclude_static_roots(void*b,void*e)
{
DCL_LOCK_STATE;
if (b==e)return;
b=(void*)((word)b&~(word)(sizeof(word)- 1));
e=(void*)(((word)e+(sizeof(word)- 1))&~(word)(sizeof(word)- 1));
if (NULL==e)
e=(void*)(~(word)(sizeof(word)- 1));
LOCK();
GC_exclude_static_roots_inner(b,e);
UNLOCK();
}
#if defined(WRAP_MARK_SOME)&&defined(PARALLEL_MARK)
#define GC_PUSH_CONDITIONAL(b,t,all)(GC_parallel?GC_push_conditional_eager(b,t,all):GC_push_conditional(b,t,all))
#elif defined(GC_DISABLE_INCREMENTAL)
#define GC_PUSH_CONDITIONAL(b,t,all)GC_push_all(b,t)
#else
#define GC_PUSH_CONDITIONAL(b,t,all)GC_push_conditional(b,t,all)
#endif
STATIC void GC_push_conditional_with_exclusions(ptr_t bottom,ptr_t top,
GC_bool all GC_ATTR_UNUSED)
{
while ((word)bottom < (word)top){
struct exclusion*next=GC_next_exclusion(bottom);
ptr_t excl_start;
if (0==next
||(word)(excl_start=next->e_start)>=(word)top){
GC_PUSH_CONDITIONAL(bottom,top,all);
break;
}
if ((word)excl_start > (word)bottom)
GC_PUSH_CONDITIONAL(bottom,excl_start,all);
bottom=next->e_end;
}
}
#ifdef IA64
GC_INNER void GC_push_all_register_sections(ptr_t bs_lo,ptr_t bs_hi,
int eager,struct GC_traced_stack_sect_s*traced_stack_sect)
{
while (traced_stack_sect!=NULL){
ptr_t frame_bs_lo=traced_stack_sect->backing_store_end;
GC_ASSERT((word)frame_bs_lo<=(word)bs_hi);
if (eager){
GC_push_all_eager(frame_bs_lo,bs_hi);
} else {
GC_push_all_stack(frame_bs_lo,bs_hi);
}
bs_hi=traced_stack_sect->saved_backing_store_ptr;
traced_stack_sect=traced_stack_sect->prev;
}
GC_ASSERT((word)bs_lo<=(word)bs_hi);
if (eager){
GC_push_all_eager(bs_lo,bs_hi);
} else {
GC_push_all_stack(bs_lo,bs_hi);
}
}
#endif
#ifdef THREADS
GC_INNER void GC_push_all_stack_sections(ptr_t lo,ptr_t hi,
struct GC_traced_stack_sect_s*traced_stack_sect)
{
while (traced_stack_sect!=NULL){
GC_ASSERT((word)lo HOTTER_THAN (word)traced_stack_sect);
#ifdef STACK_GROWS_UP
GC_push_all_stack((ptr_t)traced_stack_sect,lo);
#else
GC_push_all_stack(lo,(ptr_t)traced_stack_sect);
#endif
lo=traced_stack_sect->saved_stack_ptr;
GC_ASSERT(lo!=NULL);
traced_stack_sect=traced_stack_sect->prev;
}
GC_ASSERT(!((word)hi HOTTER_THAN (word)lo));
#ifdef STACK_GROWS_UP
GC_push_all_stack(hi,lo);
#else
GC_push_all_stack(lo,hi);
#endif
}
#else
STATIC void GC_push_all_stack_partially_eager(ptr_t bottom,ptr_t top,
ptr_t cold_gc_frame)
{
#ifndef NEED_FIXUP_POINTER
if (GC_all_interior_pointers){
if (0==cold_gc_frame){
GC_push_all_stack(bottom,top);
return;
}
GC_ASSERT((word)bottom<=(word)cold_gc_frame
&&(word)cold_gc_frame<=(word)top);
#ifdef STACK_GROWS_DOWN
GC_push_all(cold_gc_frame - sizeof(ptr_t),top);
GC_push_all_eager(bottom,cold_gc_frame);
#else
GC_push_all(bottom,cold_gc_frame+sizeof(ptr_t));
GC_push_all_eager(cold_gc_frame,top);
#endif
} else
#endif
{
GC_push_all_eager(bottom,top);
}
#ifdef TRACE_BUF
GC_add_trace_entry("GC_push_all_stack",(word)bottom,(word)top);
#endif
}
STATIC void GC_push_all_stack_part_eager_sections(ptr_t lo,ptr_t hi,
ptr_t cold_gc_frame,struct GC_traced_stack_sect_s*traced_stack_sect)
{
GC_ASSERT(traced_stack_sect==NULL||cold_gc_frame==NULL||
(word)cold_gc_frame HOTTER_THAN (word)traced_stack_sect);
while (traced_stack_sect!=NULL){
GC_ASSERT((word)lo HOTTER_THAN (word)traced_stack_sect);
#ifdef STACK_GROWS_UP
GC_push_all_stack_partially_eager((ptr_t)traced_stack_sect,lo,
cold_gc_frame);
#else
GC_push_all_stack_partially_eager(lo,(ptr_t)traced_stack_sect,
cold_gc_frame);
#endif
lo=traced_stack_sect->saved_stack_ptr;
GC_ASSERT(lo!=NULL);
traced_stack_sect=traced_stack_sect->prev;
cold_gc_frame=NULL;
}
GC_ASSERT(!((word)hi HOTTER_THAN (word)lo));
#ifdef STACK_GROWS_UP
GC_push_all_stack_partially_eager(hi,lo,cold_gc_frame);
#else
GC_push_all_stack_partially_eager(lo,hi,cold_gc_frame);
#endif
}
#endif
STATIC void GC_push_current_stack(ptr_t cold_gc_frame,
void*context GC_ATTR_UNUSED)
{
#if defined(THREADS)
#ifdef STACK_GROWS_DOWN
GC_push_all_eager(GC_approx_sp(),cold_gc_frame);
#else
GC_push_all_eager(cold_gc_frame,GC_approx_sp());
#endif
#else
GC_push_all_stack_part_eager_sections(GC_approx_sp(),GC_stackbottom,
cold_gc_frame,GC_traced_stack_sect);
#ifdef IA64
{
ptr_t bsp=GC_save_regs_ret_val;
ptr_t cold_gc_bs_pointer=bsp - 2048;
if (GC_all_interior_pointers
&&(word)cold_gc_bs_pointer > (word)BACKING_STORE_BASE){
if (GC_traced_stack_sect!=NULL
&&(word)cold_gc_bs_pointer
< (word)GC_traced_stack_sect->backing_store_end)
cold_gc_bs_pointer=
GC_traced_stack_sect->backing_store_end;
GC_push_all_register_sections(BACKING_STORE_BASE,
cold_gc_bs_pointer,FALSE,GC_traced_stack_sect);
GC_push_all_eager(cold_gc_bs_pointer,bsp);
} else {
GC_push_all_register_sections(BACKING_STORE_BASE,bsp,
TRUE,GC_traced_stack_sect);
}
}
#endif
#endif
}
GC_INNER void (*GC_push_typed_structures)(void)=0;
GC_INNER void GC_cond_register_dynamic_libraries(void)
{
#if (defined(DYNAMIC_LOADING)&&!defined(MSWIN_XBOX1))||defined(CYGWIN32)||defined(MSWIN32)||defined(MSWINCE)||defined(PCR)
GC_remove_tmp_roots();
if (!GC_no_dls)GC_register_dynamic_libraries();
#else
GC_no_dls=TRUE;
#endif
}
STATIC void GC_push_regs_and_stack(ptr_t cold_gc_frame)
{
#ifdef THREADS
if (NULL==cold_gc_frame)
return;
#endif
GC_with_callee_saves_pushed(GC_push_current_stack,cold_gc_frame);
}
GC_INNER void GC_push_roots(GC_bool all,ptr_t cold_gc_frame GC_ATTR_UNUSED)
{
int i;
unsigned kind;
#if!defined(REGISTER_LIBRARIES_EARLY)
GC_cond_register_dynamic_libraries();
#endif
for (i=0;i < n_root_sets;i++){
GC_push_conditional_with_exclusions(
GC_static_roots[i].r_start,
GC_static_roots[i].r_end,all);
}
for (kind=0;kind < GC_n_kinds;kind++){
void*base=GC_base(GC_obj_kinds[kind].ok_freelist);
if (base!=NULL){
GC_set_mark_bit(base);
}
}
#ifndef GC_NO_FINALIZATION
GC_push_finalizer_structures();
#endif
#ifdef THREADS
if (GC_no_dls||GC_roots_were_cleared)
GC_push_thread_structures();
#endif
if (GC_push_typed_structures)
GC_push_typed_structures();
#if defined(THREAD_LOCAL_ALLOC)
if (GC_world_stopped)
GC_mark_thread_local_free_lists();
#endif
#ifndef STACK_NOT_SCANNED
GC_push_regs_and_stack(cold_gc_frame);
#endif
if (GC_push_other_roots!=0){
(*GC_push_other_roots)();
}
}
#ifdef ENABLE_DISCLAIM
#endif
#include <stdio.h>
GC_INNER signed_word GC_bytes_found=0;
#if defined(PARALLEL_MARK)
GC_INNER signed_word GC_fl_builder_count=0;
#endif
#ifndef MAX_LEAKED
#define MAX_LEAKED 40
#endif
STATIC ptr_t GC_leaked[MAX_LEAKED]={ NULL };
STATIC unsigned GC_n_leaked=0;
GC_INNER GC_bool GC_have_errors=FALSE;
#if!defined(EAGER_SWEEP)&&defined(ENABLE_DISCLAIM)
STATIC void GC_reclaim_unconditionally_marked(void);
#endif
GC_INLINE void GC_add_leaked(ptr_t leaked)
{
#ifndef SHORT_DBG_HDRS
if (GC_findleak_delay_free&&!GC_check_leaked(leaked))
return;
#endif
GC_have_errors=TRUE;
if (GC_n_leaked < MAX_LEAKED){
GC_leaked[GC_n_leaked++]=leaked;
GC_set_mark_bit(leaked);
}
}
GC_INNER void GC_print_all_errors(void)
{
static GC_bool printing_errors=FALSE;
GC_bool have_errors;
unsigned i,n_leaked;
ptr_t leaked[MAX_LEAKED];
DCL_LOCK_STATE;
LOCK();
if (printing_errors){
UNLOCK();
return;
}
have_errors=GC_have_errors;
printing_errors=TRUE;
n_leaked=GC_n_leaked;
if (n_leaked > 0){
GC_ASSERT(n_leaked<=MAX_LEAKED);
BCOPY(GC_leaked,leaked,n_leaked*sizeof(ptr_t));
GC_n_leaked=0;
BZERO(GC_leaked,n_leaked*sizeof(ptr_t));
}
UNLOCK();
if (GC_debugging_started){
GC_print_all_smashed();
} else {
have_errors=FALSE;
}
if (n_leaked > 0){
GC_err_printf("Found %u leaked objects:\n",n_leaked);
have_errors=TRUE;
}
for (i=0;i < n_leaked;i++){
ptr_t p=leaked[i];
#ifndef SKIP_LEAKED_OBJECTS_PRINTING
GC_print_heap_obj(p);
#endif
GC_free(p);
}
if (have_errors
#ifndef GC_ABORT_ON_LEAK
&&GETENV("GC_ABORT_ON_LEAK")!=NULL
#endif
){
ABORT("Leaked or smashed objects encountered");
}
LOCK();
printing_errors=FALSE;
UNLOCK();
}
GC_INNER GC_bool GC_block_empty(hdr*hhdr)
{
return (hhdr->hb_n_marks==0);
}
STATIC GC_bool GC_block_nearly_full(hdr*hhdr,word sz)
{
return hhdr->hb_n_marks > HBLK_OBJS(sz)*7/8;
}
STATIC ptr_t GC_reclaim_clear(struct hblk*hbp,hdr*hhdr,word sz,
ptr_t list,signed_word*count)
{
word bit_no=0;
word*p,*q,*plim;
signed_word n_bytes_found=0;
GC_ASSERT(hhdr==GC_find_header((ptr_t)hbp));
#ifndef THREADS
GC_ASSERT(sz==hhdr->hb_sz);
#else
#endif
GC_ASSERT((sz&(BYTES_PER_WORD-1))==0);
p=(word*)(hbp->hb_body);
plim=(word*)(hbp->hb_body+HBLKSIZE - sz);
while ((word)p<=(word)plim){
if (mark_bit_from_hdr(hhdr,bit_no)){
p=(word*)((ptr_t)p+sz);
} else {
n_bytes_found+=sz;
obj_link(p)=list;
list=((ptr_t)p);
q=(word*)((ptr_t)p+sz);
#ifdef USE_MARK_BYTES
GC_ASSERT(!(sz&1)
&&!((word)p&(2*sizeof(word)- 1)));
p[1]=0;
p+=2;
while ((word)p < (word)q){
CLEAR_DOUBLE(p);
p+=2;
}
#else
p++;
while ((word)p < (word)q){
*p++=0;
}
#endif
}
bit_no+=MARK_BIT_OFFSET(sz);
}
*count+=n_bytes_found;
return(list);
}
STATIC ptr_t GC_reclaim_uninit(struct hblk*hbp,hdr*hhdr,word sz,
ptr_t list,signed_word*count)
{
word bit_no=0;
word*p,*plim;
signed_word n_bytes_found=0;
#ifndef THREADS
GC_ASSERT(sz==hhdr->hb_sz);
#endif
p=(word*)(hbp->hb_body);
plim=(word*)((ptr_t)hbp+HBLKSIZE - sz);
while ((word)p<=(word)plim){
if (!mark_bit_from_hdr(hhdr,bit_no)){
n_bytes_found+=sz;
obj_link(p)=list;
list=((ptr_t)p);
}
p=(word*)((ptr_t)p+sz);
bit_no+=MARK_BIT_OFFSET(sz);
}
*count+=n_bytes_found;
return(list);
}
#ifdef ENABLE_DISCLAIM
STATIC ptr_t GC_disclaim_and_reclaim(struct hblk*hbp,hdr*hhdr,word sz,
ptr_t list,signed_word*count)
{
word bit_no=0;
word*p,*q,*plim;
signed_word n_bytes_found=0;
struct obj_kind*ok=&GC_obj_kinds[hhdr->hb_obj_kind];
int (GC_CALLBACK*disclaim)(void*)=ok->ok_disclaim_proc;
#ifndef THREADS
GC_ASSERT(sz==hhdr->hb_sz);
#endif
p=(word*)(hbp->hb_body);
plim=(word*)((ptr_t)p+HBLKSIZE - sz);
while ((word)p<=(word)plim){
int marked=mark_bit_from_hdr(hhdr,bit_no);
if (!marked&&(*disclaim)(p)){
set_mark_bit_from_hdr(hhdr,bit_no);
hhdr->hb_n_marks++;
marked=1;
}
if (marked)
p=(word*)((ptr_t)p+sz);
else {
n_bytes_found+=sz;
obj_link(p)=list;
list=((ptr_t)p);
q=(word*)((ptr_t)p+sz);
#ifdef USE_MARK_BYTES
GC_ASSERT((sz&1)==0);
GC_ASSERT(((word)p&(2*sizeof(word)- 1))==0);
p[1]=0;
p+=2;
while ((word)p < (word)q){
CLEAR_DOUBLE(p);
p+=2;
}
#else
p++;
while ((word)p < (word)q){
*p++=0;
}
#endif
}
bit_no+=MARK_BIT_OFFSET(sz);
}
*count+=n_bytes_found;
return list;
}
#endif
STATIC void GC_reclaim_check(struct hblk*hbp,hdr*hhdr,word sz)
{
word bit_no;
ptr_t p,plim;
#ifndef THREADS
GC_ASSERT(sz==hhdr->hb_sz);
#endif
p=hbp->hb_body;
plim=p+HBLKSIZE - sz;
for (bit_no=0;(word)p<=(word)plim;
p+=sz,bit_no+=MARK_BIT_OFFSET(sz)){
if (!mark_bit_from_hdr(hhdr,bit_no)){
GC_add_leaked(p);
}
}
}
#ifdef AO_HAVE_load
#define IS_PTRFREE_SAFE(hhdr)(AO_load((volatile AO_t*)&(hhdr)->hb_descr)==0)
#else
#define IS_PTRFREE_SAFE(hhdr)((hhdr)->hb_descr==0)
#endif
GC_INNER ptr_t GC_reclaim_generic(struct hblk*hbp,hdr*hhdr,size_t sz,
GC_bool init,ptr_t list,
signed_word*count)
{
ptr_t result;
GC_ASSERT(GC_find_header((ptr_t)hbp)==hhdr);
#ifndef GC_DISABLE_INCREMENTAL
GC_remove_protection(hbp,1,IS_PTRFREE_SAFE(hhdr));
#endif
#ifdef ENABLE_DISCLAIM
if ((hhdr->hb_flags&HAS_DISCLAIM)!=0){
result=GC_disclaim_and_reclaim(hbp,hhdr,sz,list,count);
} else
#endif
if (init||GC_debugging_started){
result=GC_reclaim_clear(hbp,hhdr,sz,list,count);
} else {
GC_ASSERT(IS_PTRFREE_SAFE(hhdr));
result=GC_reclaim_uninit(hbp,hhdr,sz,list,count);
}
if (IS_UNCOLLECTABLE(hhdr->hb_obj_kind))GC_set_hdr_marks(hhdr);
return result;
}
STATIC void GC_reclaim_small_nonempty_block(struct hblk*hbp,word sz,
GC_bool report_if_found)
{
hdr*hhdr=HDR(hbp);
struct obj_kind*ok=&GC_obj_kinds[hhdr->hb_obj_kind];
void**flh=&(ok->ok_freelist[BYTES_TO_GRANULES(sz)]);
hhdr->hb_last_reclaimed=(unsigned short)GC_gc_no;
if (report_if_found){
GC_reclaim_check(hbp,hhdr,sz);
} else {
*flh=GC_reclaim_generic(hbp,hhdr,sz,ok->ok_init,
(ptr_t)(*flh),&GC_bytes_found);
}
}
#ifdef ENABLE_DISCLAIM
STATIC void GC_disclaim_and_reclaim_or_free_small_block(struct hblk*hbp)
{
hdr*hhdr=HDR(hbp);
word sz=hhdr->hb_sz;
struct obj_kind*ok=&GC_obj_kinds[hhdr->hb_obj_kind];
void**flh=&(ok->ok_freelist[BYTES_TO_GRANULES(sz)]);
void*flh_next;
hhdr->hb_last_reclaimed=(unsigned short)GC_gc_no;
flh_next=GC_reclaim_generic(hbp,hhdr,sz,ok->ok_init,
(ptr_t)(*flh),&GC_bytes_found);
if (hhdr->hb_n_marks)
*flh=flh_next;
else {
GC_bytes_found+=HBLKSIZE;
GC_freehblk(hbp);
}
}
#endif
STATIC void GC_reclaim_block(struct hblk*hbp,word report_if_found)
{
hdr*hhdr=HDR(hbp);
word sz;
struct obj_kind*ok=&GC_obj_kinds[hhdr->hb_obj_kind];
#ifdef AO_HAVE_load
sz=(word)AO_load((volatile AO_t*)&hhdr->hb_sz);
#else
sz=hhdr->hb_sz;
#endif
if( sz > MAXOBJBYTES){
if(!mark_bit_from_hdr(hhdr,0)){
if (report_if_found){
GC_add_leaked((ptr_t)hbp);
} else {
word blocks;
#ifdef ENABLE_DISCLAIM
if (EXPECT(hhdr->hb_flags&HAS_DISCLAIM,0)){
if ((*ok->ok_disclaim_proc)(hbp)){
set_mark_bit_from_hdr(hhdr,0);
goto in_use;
}
}
#endif
blocks=OBJ_SZ_TO_BLOCKS(sz);
#if defined(CPPCHECK)
GC_noop1((word)&blocks);
#endif
if (blocks > 1){
GC_large_allocd_bytes-=blocks*HBLKSIZE;
}
GC_bytes_found+=sz;
GC_freehblk(hbp);
}
} else {
#ifdef ENABLE_DISCLAIM
in_use:
#endif
if (IS_PTRFREE_SAFE(hhdr)){
GC_atomic_in_use+=sz;
} else {
GC_composite_in_use+=sz;
}
}
} else {
GC_bool empty=GC_block_empty(hhdr);
#ifdef PARALLEL_MARK
GC_ASSERT(hhdr->hb_n_marks<=2*(HBLKSIZE/sz+1)+16);
#else
GC_ASSERT(sz*hhdr->hb_n_marks<=HBLKSIZE);
#endif
if (report_if_found){
GC_reclaim_small_nonempty_block(hbp,sz,
TRUE);
} else if (empty){
#ifdef ENABLE_DISCLAIM
if ((hhdr->hb_flags&HAS_DISCLAIM)!=0){
GC_disclaim_and_reclaim_or_free_small_block(hbp);
} else
#endif
{
GC_bytes_found+=HBLKSIZE;
GC_freehblk(hbp);
}
} else if (GC_find_leak||!GC_block_nearly_full(hhdr,sz)){
struct hblk**rlh=ok->ok_reclaim_list;
if (rlh!=NULL){
rlh+=BYTES_TO_GRANULES(sz);
hhdr->hb_next=*rlh;
*rlh=hbp;
}
}
if (IS_PTRFREE_SAFE(hhdr)){
GC_atomic_in_use+=sz*hhdr->hb_n_marks;
} else {
GC_composite_in_use+=sz*hhdr->hb_n_marks;
}
}
}
#if!defined(NO_DEBUGGING)
struct Print_stats
{
size_t number_of_blocks;
size_t total_bytes;
};
#ifdef USE_MARK_BYTES
unsigned GC_n_set_marks(hdr*hhdr)
{
unsigned result=0;
word i;
word sz=hhdr->hb_sz;
word offset=MARK_BIT_OFFSET(sz);
word limit=FINAL_MARK_BIT(sz);
for (i=0;i < limit;i+=offset){
result+=hhdr->hb_marks[i];
}
GC_ASSERT(hhdr->hb_marks[limit]);
return(result);
}
#else
static unsigned set_bits(word n)
{
word m=n;
unsigned result=0;
while (m > 0){
if (m&1)result++;
m>>=1;
}
return(result);
}
unsigned GC_n_set_marks(hdr*hhdr)
{
unsigned result=0;
word i;
word n_mark_words;
#ifdef MARK_BIT_PER_OBJ
word n_objs=HBLK_OBJS(hhdr->hb_sz);
if (0==n_objs)n_objs=1;
n_mark_words=divWORDSZ(n_objs+WORDSZ - 1);
#else
n_mark_words=MARK_BITS_SZ;
#endif
for (i=0;i < n_mark_words - 1;i++){
result+=set_bits(hhdr->hb_marks[i]);
}
#ifdef MARK_BIT_PER_OBJ
result+=set_bits((hhdr->hb_marks[n_mark_words - 1])
<<(n_mark_words*WORDSZ - n_objs));
#else
result+=set_bits(hhdr->hb_marks[n_mark_words - 1]);
#endif
return result;
}
#endif
STATIC void GC_print_block_descr(struct hblk*h,
word raw_ps)
{
hdr*hhdr=HDR(h);
size_t bytes=hhdr->hb_sz;
struct Print_stats*ps;
unsigned n_marks=GC_n_set_marks(hhdr);
unsigned n_objs=(unsigned)HBLK_OBJS(bytes);
if (0==n_objs)n_objs=1;
if (hhdr->hb_n_marks!=n_marks){
GC_printf("%u,%u,%u!=%u,%u\n",hhdr->hb_obj_kind,(unsigned)bytes,
(unsigned)hhdr->hb_n_marks,n_marks,n_objs);
} else {
GC_printf("%u,%u,%u,%u\n",hhdr->hb_obj_kind,(unsigned)bytes,
n_marks,n_objs);
}
ps=(struct Print_stats*)raw_ps;
ps->total_bytes+=(bytes+(HBLKSIZE-1))&~(HBLKSIZE-1);
ps->number_of_blocks++;
}
void GC_print_block_list(void)
{
struct Print_stats pstats;
GC_printf("kind(0=ptrfree,1=normal,2=unc.),"
"size_in_bytes,#_marks_set,#objs\n");
pstats.number_of_blocks=0;
pstats.total_bytes=0;
GC_apply_to_all_blocks(GC_print_block_descr,(word)&pstats);
GC_printf("blocks=%lu,bytes=%lu\n",
(unsigned long)pstats.number_of_blocks,
(unsigned long)pstats.total_bytes);
}
GC_API void GC_CALL GC_print_free_list(int kind,size_t sz_in_granules)
{
void*flh_next;
int n;
GC_ASSERT(kind < MAXOBJKINDS);
GC_ASSERT(sz_in_granules<=MAXOBJGRANULES);
flh_next=GC_obj_kinds[kind].ok_freelist[sz_in_granules];
for (n=0;flh_next;n++){
GC_printf("Free object in heap block %p [%d]:%p\n",
(void*)HBLKPTR(flh_next),n,flh_next);
flh_next=obj_link(flh_next);
}
}
#endif
STATIC void GC_clear_fl_links(void**flp)
{
void*next=*flp;
while (0!=next){
*flp=0;
flp=&(obj_link(next));
next=*flp;
}
}
GC_INNER void GC_start_reclaim(GC_bool report_if_found)
{
unsigned kind;
#if defined(PARALLEL_MARK)
GC_ASSERT(0==GC_fl_builder_count);
#endif
GC_composite_in_use=0;
GC_atomic_in_use=0;
for (kind=0;kind < GC_n_kinds;kind++){
struct hblk**rlist=GC_obj_kinds[kind].ok_reclaim_list;
GC_bool should_clobber=(GC_obj_kinds[kind].ok_descriptor!=0);
if (rlist==0)continue;
if (!report_if_found){
void**fop;
void**lim=&(GC_obj_kinds[kind].ok_freelist[MAXOBJGRANULES+1]);
for (fop=GC_obj_kinds[kind].ok_freelist;
(word)fop < (word)lim;(*(word**)&fop)++){
if (*fop!=0){
if (should_clobber){
GC_clear_fl_links(fop);
} else {
*fop=0;
}
}
}
}
BZERO(rlist,(MAXOBJGRANULES+1)*sizeof(void*));
}
GC_apply_to_all_blocks(GC_reclaim_block,(word)report_if_found);
#ifdef EAGER_SWEEP
GC_reclaim_all((GC_stop_func)0,FALSE);
#elif defined(ENABLE_DISCLAIM)
GC_reclaim_unconditionally_marked();
#endif
#if defined(PARALLEL_MARK)
GC_ASSERT(0==GC_fl_builder_count);
#endif
}
GC_INNER void GC_continue_reclaim(word sz,int kind)
{
hdr*hhdr;
struct hblk*hbp;
struct obj_kind*ok=&(GC_obj_kinds[kind]);
struct hblk**rlh=ok->ok_reclaim_list;
void**flh=&(ok->ok_freelist[sz]);
if (NULL==rlh)
return;
for (rlh+=sz;(hbp=*rlh)!=NULL;){
hhdr=HDR(hbp);
*rlh=hhdr->hb_next;
GC_reclaim_small_nonempty_block(hbp,hhdr->hb_sz,FALSE);
if (*flh!=0)
break;
}
}
GC_INNER GC_bool GC_reclaim_all(GC_stop_func stop_func,GC_bool ignore_old)
{
word sz;
unsigned kind;
hdr*hhdr;
struct hblk*hbp;
struct obj_kind*ok;
struct hblk**rlp;
struct hblk**rlh;
#ifndef NO_CLOCK
CLOCK_TYPE start_time=CLOCK_TYPE_INITIALIZER;
if (GC_print_stats==VERBOSE)
GET_TIME(start_time);
#endif
for (kind=0;kind < GC_n_kinds;kind++){
ok=&(GC_obj_kinds[kind]);
rlp=ok->ok_reclaim_list;
if (rlp==0)continue;
for (sz=1;sz<=MAXOBJGRANULES;sz++){
for (rlh=rlp+sz;(hbp=*rlh)!=NULL;){
if (stop_func!=(GC_stop_func)0&&(*stop_func)()){
return(FALSE);
}
hhdr=HDR(hbp);
*rlh=hhdr->hb_next;
if (!ignore_old
||(word)hhdr->hb_last_reclaimed==GC_gc_no - 1){
GC_reclaim_small_nonempty_block(hbp,hhdr->hb_sz,FALSE);
}
}
}
}
#ifndef NO_CLOCK
if (GC_print_stats==VERBOSE){
CLOCK_TYPE done_time;
GET_TIME(done_time);
GC_verbose_log_printf(
"Disposing of reclaim lists took %lu ms %lu ns\n",
MS_TIME_DIFF(done_time,start_time),
NS_FRAC_TIME_DIFF(done_time,start_time));
}
#endif
return(TRUE);
}
#if!defined(EAGER_SWEEP)&&defined(ENABLE_DISCLAIM)
STATIC void GC_reclaim_unconditionally_marked(void)
{
word sz;
unsigned kind;
hdr*hhdr;
struct hblk*hbp;
struct obj_kind*ok;
struct hblk**rlp;
struct hblk**rlh;
for (kind=0;kind < GC_n_kinds;kind++){
ok=&(GC_obj_kinds[kind]);
if (!ok->ok_mark_unconditionally)
continue;
rlp=ok->ok_reclaim_list;
if (rlp==0)
continue;
for (sz=1;sz<=MAXOBJGRANULES;sz++){
rlh=rlp+sz;
while ((hbp=*rlh)!=0){
hhdr=HDR(hbp);
*rlh=hhdr->hb_next;
GC_reclaim_small_nonempty_block(hbp,hhdr->hb_sz,FALSE);
}
}
}
}
#endif
struct enumerate_reachable_s {
GC_reachable_object_proc proc;
void*client_data;
};
STATIC void GC_do_enumerate_reachable_objects(struct hblk*hbp,word ped)
{
struct hblkhdr*hhdr=HDR(hbp);
size_t sz=(size_t)hhdr->hb_sz;
size_t bit_no;
char*p,*plim;
if (GC_block_empty(hhdr)){
return;
}
p=hbp->hb_body;
if (sz > MAXOBJBYTES){
plim=p;
} else {
plim=hbp->hb_body+HBLKSIZE - sz;
}
for (bit_no=0;p<=plim;bit_no+=MARK_BIT_OFFSET(sz),p+=sz){
if (mark_bit_from_hdr(hhdr,bit_no)){
((struct enumerate_reachable_s*)ped)->proc(p,sz,
((struct enumerate_reachable_s*)ped)->client_data);
}
}
}
GC_API void GC_CALL GC_enumerate_reachable_objects_inner(
GC_reachable_object_proc proc,
void*client_data)
{
struct enumerate_reachable_s ed;
GC_ASSERT(I_HOLD_LOCK());
ed.proc=proc;
ed.client_data=client_data;
GC_apply_to_all_blocks(GC_do_enumerate_reachable_objects,(word)&ed);
}
#ifndef GC_TYPED_H
#define GC_TYPED_H
#ifndef GC_H
#endif
#ifdef __cplusplus
extern "C" {
#endif
typedef GC_word*GC_bitmap;
#define GC_WORDSZ (8*sizeof(GC_word))
#define GC_get_bit(bm,index)(((bm)[(index)/GC_WORDSZ]>>((index)% GC_WORDSZ))&1)
#define GC_set_bit(bm,index)((bm)[(index)/GC_WORDSZ]|=(GC_word)1<<((index)% GC_WORDSZ))
#define GC_WORD_OFFSET(t,f)(offsetof(t,f)/sizeof(GC_word))
#define GC_WORD_LEN(t)(sizeof(t)/sizeof(GC_word))
#define GC_BITMAP_SIZE(t)((GC_WORD_LEN(t)+GC_WORDSZ - 1)/GC_WORDSZ)
typedef GC_word GC_descr;
GC_API GC_descr GC_CALL GC_make_descriptor(const GC_word*,
size_t);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_malloc_explicitly_typed(size_t,
GC_descr);
GC_API GC_ATTR_MALLOC GC_ATTR_ALLOC_SIZE(1)void*GC_CALL
GC_malloc_explicitly_typed_ignore_off_page(size_t,
GC_descr);
GC_API GC_ATTR_MALLOC GC_ATTR_CALLOC_SIZE(1,2)void*GC_CALL
GC_calloc_explicitly_typed(size_t,
size_t,
GC_descr);
#ifdef GC_DEBUG
#define GC_MALLOC_EXPLICITLY_TYPED(bytes,d)((void)(d),GC_MALLOC(bytes))
#define GC_CALLOC_EXPLICITLY_TYPED(n,bytes,d)((void)(d),GC_MALLOC((n)*(bytes)))
#else
#define GC_MALLOC_EXPLICITLY_TYPED(bytes,d)GC_malloc_explicitly_typed(bytes,d)
#define GC_CALLOC_EXPLICITLY_TYPED(n,bytes,d)GC_calloc_explicitly_typed(n,bytes,d)
#endif
#ifdef __cplusplus
}
#endif
#endif
#define TYPD_EXTRA_BYTES (sizeof(word)- EXTRA_BYTES)
STATIC int GC_explicit_kind=0;
STATIC int GC_array_kind=0;
struct LeafDescriptor {
word ld_tag;
#define LEAF_TAG 1
size_t ld_size;
size_t ld_nelements;
GC_descr ld_descriptor;
};
struct ComplexArrayDescriptor {
word ad_tag;
#define ARRAY_TAG 2
size_t ad_nelements;
union ComplexDescriptor*ad_element_descr;
};
struct SequenceDescriptor {
word sd_tag;
#define SEQUENCE_TAG 3
union ComplexDescriptor*sd_first;
union ComplexDescriptor*sd_second;
};
typedef union ComplexDescriptor {
struct LeafDescriptor ld;
struct ComplexArrayDescriptor ad;
struct SequenceDescriptor sd;
} complex_descriptor;
#define TAG ad.ad_tag
#define ED_INITIAL_SIZE 100
STATIC int GC_typed_mark_proc_index=0;
STATIC int GC_array_mark_proc_index=0;
STATIC void GC_push_typed_structures_proc(void)
{
GC_PUSH_ALL_SYM(GC_ext_descriptors);
}
STATIC signed_word GC_add_ext_descriptor(const word*bm,word nbits)
{
size_t nwords=divWORDSZ(nbits+WORDSZ-1);
signed_word result;
size_t i;
word last_part;
size_t extra_bits;
DCL_LOCK_STATE;
LOCK();
while (GC_avail_descr+nwords>=GC_ed_size){
typed_ext_descr_t*newExtD;
size_t new_size;
word ed_size=GC_ed_size;
if (ed_size==0){
GC_ASSERT((word)(&GC_ext_descriptors)% sizeof(word)==0);
GC_push_typed_structures=GC_push_typed_structures_proc;
UNLOCK();
new_size=ED_INITIAL_SIZE;
} else {
UNLOCK();
new_size=2*ed_size;
if (new_size > MAX_ENV)return(-1);
}
newExtD=(typed_ext_descr_t*)GC_malloc_atomic(new_size
*sizeof(typed_ext_descr_t));
if (NULL==newExtD)
return -1;
LOCK();
if (ed_size==GC_ed_size){
if (GC_avail_descr!=0){
BCOPY(GC_ext_descriptors,newExtD,
GC_avail_descr*sizeof(typed_ext_descr_t));
}
GC_ed_size=new_size;
GC_ext_descriptors=newExtD;
}
}
result=GC_avail_descr;
for (i=0;i < nwords-1;i++){
GC_ext_descriptors[result+i].ed_bitmap=bm[i];
GC_ext_descriptors[result+i].ed_continued=TRUE;
}
last_part=bm[i];
extra_bits=nwords*WORDSZ - nbits;
last_part<<=extra_bits;
last_part>>=extra_bits;
GC_ext_descriptors[result+i].ed_bitmap=last_part;
GC_ext_descriptors[result+i].ed_continued=FALSE;
GC_avail_descr+=nwords;
UNLOCK();
return(result);
}
STATIC GC_descr GC_bm_table[WORDSZ/2];
STATIC GC_descr GC_double_descr(GC_descr descriptor,word nwords)
{
if ((descriptor&GC_DS_TAGS)==GC_DS_LENGTH){
descriptor=GC_bm_table[BYTES_TO_WORDS((word)descriptor)];
};
descriptor|=(descriptor&~GC_DS_TAGS)>>nwords;
return(descriptor);
}
STATIC complex_descriptor*
GC_make_sequence_descriptor(complex_descriptor*first,
complex_descriptor*second);
#define COMPLEX 2
#define LEAF 1
#define SIMPLE 0
#define NO_MEM (-1)
STATIC int GC_make_array_descriptor(size_t nelements,size_t size,
GC_descr descriptor,GC_descr*simple_d,
complex_descriptor**complex_d,
struct LeafDescriptor*leaf)
{
#define OPT_THRESHOLD 50
if ((descriptor&GC_DS_TAGS)==GC_DS_LENGTH){
if (descriptor==(GC_descr)size){
*simple_d=nelements*descriptor;
return(SIMPLE);
} else if ((word)descriptor==0){
*simple_d=(GC_descr)0;
return(SIMPLE);
}
}
if (nelements<=OPT_THRESHOLD){
if (nelements<=1){
if (nelements==1){
*simple_d=descriptor;
return(SIMPLE);
} else {
*simple_d=(GC_descr)0;
return(SIMPLE);
}
}
} else if (size<=BITMAP_BITS/2
&&(descriptor&GC_DS_TAGS)!=GC_DS_PROC
&&(size&(sizeof(word)-1))==0){
int result=
GC_make_array_descriptor(nelements/2,2*size,
GC_double_descr(descriptor,
BYTES_TO_WORDS(size)),
simple_d,complex_d,leaf);
if ((nelements&1)==0){
return(result);
} else {
struct LeafDescriptor*one_element=
(struct LeafDescriptor*)
GC_malloc_atomic(sizeof(struct LeafDescriptor));
if (result==NO_MEM||one_element==0)return(NO_MEM);
one_element->ld_tag=LEAF_TAG;
one_element->ld_size=size;
one_element->ld_nelements=1;
one_element->ld_descriptor=descriptor;
switch(result){
case SIMPLE:
{
struct LeafDescriptor*beginning=
(struct LeafDescriptor*)
GC_malloc_atomic(sizeof(struct LeafDescriptor));
if (beginning==0)return(NO_MEM);
beginning->ld_tag=LEAF_TAG;
beginning->ld_size=size;
beginning->ld_nelements=1;
beginning->ld_descriptor=*simple_d;
*complex_d=GC_make_sequence_descriptor(
(complex_descriptor*)beginning,
(complex_descriptor*)one_element);
break;
}
case LEAF:
{
struct LeafDescriptor*beginning=
(struct LeafDescriptor*)
GC_malloc_atomic(sizeof(struct LeafDescriptor));
if (beginning==0)return(NO_MEM);
beginning->ld_tag=LEAF_TAG;
beginning->ld_size=leaf->ld_size;
beginning->ld_nelements=leaf->ld_nelements;
beginning->ld_descriptor=leaf->ld_descriptor;
*complex_d=GC_make_sequence_descriptor(
(complex_descriptor*)beginning,
(complex_descriptor*)one_element);
break;
}
case COMPLEX:
*complex_d=GC_make_sequence_descriptor(
*complex_d,
(complex_descriptor*)one_element);
break;
}
return(COMPLEX);
}
}
leaf->ld_size=size;
leaf->ld_nelements=nelements;
leaf->ld_descriptor=descriptor;
return(LEAF);
}
STATIC complex_descriptor*
GC_make_sequence_descriptor(complex_descriptor*first,
complex_descriptor*second)
{
struct SequenceDescriptor*result=
(struct SequenceDescriptor*)
GC_malloc(sizeof(struct SequenceDescriptor));
if (result!=0){
result->sd_tag=SEQUENCE_TAG;
result->sd_first=first;
result->sd_second=second;
GC_dirty(result);
REACHABLE_AFTER_DIRTY(first);
REACHABLE_AFTER_DIRTY(second);
}
return((complex_descriptor*)result);
}
STATIC mse*GC_typed_mark_proc(word*addr,mse*mark_stack_ptr,
mse*mark_stack_limit,word env);
STATIC mse*GC_array_mark_proc(word*addr,mse*mark_stack_ptr,
mse*mark_stack_limit,word env);
STATIC void GC_init_explicit_typing(void)
{
unsigned i;
GC_STATIC_ASSERT(sizeof(struct LeafDescriptor)% sizeof(word)==0);
GC_explicit_kind=GC_new_kind_inner(GC_new_free_list_inner(),
(WORDS_TO_BYTES((word)-1)|GC_DS_PER_OBJECT),
TRUE,TRUE);
GC_typed_mark_proc_index=GC_new_proc_inner(GC_typed_mark_proc);
GC_array_mark_proc_index=GC_new_proc_inner(GC_array_mark_proc);
GC_array_kind=GC_new_kind_inner(GC_new_free_list_inner(),
GC_MAKE_PROC(GC_array_mark_proc_index,0),
FALSE,TRUE);
GC_bm_table[0]=GC_DS_BITMAP;
for (i=1;i < WORDSZ/2;i++){
GC_bm_table[i]=(((word)-1)<<(WORDSZ - i))|GC_DS_BITMAP;
}
}
STATIC mse*GC_typed_mark_proc(word*addr,mse*mark_stack_ptr,
mse*mark_stack_limit,word env)
{
word bm=GC_ext_descriptors[env].ed_bitmap;
word*current_p=addr;
word current;
ptr_t greatest_ha=(ptr_t)GC_greatest_plausible_heap_addr;
ptr_t least_ha=(ptr_t)GC_least_plausible_heap_addr;
DECLARE_HDR_CACHE;
INIT_HDR_CACHE;
for (;bm!=0;bm>>=1,current_p++){
if (bm&1){
current=*current_p;
FIXUP_POINTER(current);
if (current>=(word)least_ha&&current<=(word)greatest_ha){
PUSH_CONTENTS((ptr_t)current,mark_stack_ptr,
mark_stack_limit,(ptr_t)current_p);
}
}
}
if (GC_ext_descriptors[env].ed_continued){
mark_stack_ptr++;
if ((word)mark_stack_ptr>=(word)mark_stack_limit){
mark_stack_ptr=GC_signal_mark_stack_overflow(mark_stack_ptr);
}
mark_stack_ptr->mse_start=(ptr_t)(addr+WORDSZ);
mark_stack_ptr->mse_descr.w=
GC_MAKE_PROC(GC_typed_mark_proc_index,env+1);
}
return(mark_stack_ptr);
}
STATIC word GC_descr_obj_size(complex_descriptor*d)
{
switch(d->TAG){
case LEAF_TAG:
return(d->ld.ld_nelements*d->ld.ld_size);
case ARRAY_TAG:
return(d->ad.ad_nelements
*GC_descr_obj_size(d->ad.ad_element_descr));
case SEQUENCE_TAG:
return(GC_descr_obj_size(d->sd.sd_first)
+GC_descr_obj_size(d->sd.sd_second));
default:
ABORT_RET("Bad complex descriptor");
return 0;
}
}
STATIC mse*GC_push_complex_descriptor(word*addr,complex_descriptor*d,
mse*msp,mse*msl)
{
ptr_t current=(ptr_t)addr;
word nelements;
word sz;
word i;
switch(d->TAG){
case LEAF_TAG:
{
GC_descr descr=d->ld.ld_descriptor;
nelements=d->ld.ld_nelements;
if (msl - msp<=(ptrdiff_t)nelements)return(0);
sz=d->ld.ld_size;
for (i=0;i < nelements;i++){
msp++;
msp->mse_start=current;
msp->mse_descr.w=descr;
current+=sz;
}
return(msp);
}
case ARRAY_TAG:
{
complex_descriptor*descr=d->ad.ad_element_descr;
nelements=d->ad.ad_nelements;
sz=GC_descr_obj_size(descr);
for (i=0;i < nelements;i++){
msp=GC_push_complex_descriptor((word*)current,descr,
msp,msl);
if (msp==0)return(0);
current+=sz;
}
return(msp);
}
case SEQUENCE_TAG:
{
sz=GC_descr_obj_size(d->sd.sd_first);
msp=GC_push_complex_descriptor((word*)current,d->sd.sd_first,
msp,msl);
if (msp==0)return(0);
current+=sz;
msp=GC_push_complex_descriptor((word*)current,d->sd.sd_second,
msp,msl);
return(msp);
}
default:
ABORT_RET("Bad complex descriptor");
return 0;
}
}
STATIC mse*GC_array_mark_proc(word*addr,mse*mark_stack_ptr,
mse*mark_stack_limit,
word env GC_ATTR_UNUSED)
{
hdr*hhdr=HDR(addr);
word sz=hhdr->hb_sz;
word nwords=BYTES_TO_WORDS(sz);
complex_descriptor*descr=(complex_descriptor*)(addr[nwords-1]);
mse*orig_mark_stack_ptr=mark_stack_ptr;
mse*new_mark_stack_ptr;
if (descr==0){
return(orig_mark_stack_ptr);
}
new_mark_stack_ptr=GC_push_complex_descriptor(addr,descr,
mark_stack_ptr,
mark_stack_limit-1);
if (new_mark_stack_ptr==0){
if (NULL==mark_stack_ptr)ABORT("Bad mark_stack_ptr");
#ifdef PARALLEL_MARK
if (GC_mark_stack+GC_mark_stack_size==mark_stack_limit)
#endif
{
GC_mark_stack_too_small=TRUE;
}
new_mark_stack_ptr=orig_mark_stack_ptr+1;
new_mark_stack_ptr->mse_start=(ptr_t)addr;
new_mark_stack_ptr->mse_descr.w=sz|GC_DS_LENGTH;
} else {
new_mark_stack_ptr++;
new_mark_stack_ptr->mse_start=(ptr_t)(addr+nwords - 1);
new_mark_stack_ptr->mse_descr.w=sizeof(word)|GC_DS_LENGTH;
}
return new_mark_stack_ptr;
}
GC_API GC_descr GC_CALL GC_make_descriptor(const GC_word*bm,size_t len)
{
signed_word last_set_bit=len - 1;
GC_descr result;
DCL_LOCK_STATE;
#if defined(AO_HAVE_load_acquire)&&defined(AO_HAVE_store_release)
if (!EXPECT(AO_load_acquire(&GC_explicit_typing_initialized),TRUE)){
LOCK();
if (!GC_explicit_typing_initialized){
GC_init_explicit_typing();
AO_store_release(&GC_explicit_typing_initialized,TRUE);
}
UNLOCK();
}
#else
LOCK();
if (!EXPECT(GC_explicit_typing_initialized,TRUE)){
GC_init_explicit_typing();
GC_explicit_typing_initialized=TRUE;
}
UNLOCK();
#endif
while (last_set_bit>=0&&!GC_get_bit(bm,last_set_bit))
last_set_bit--;
if (last_set_bit < 0)return(0);
#if ALIGNMENT==CPP_WORDSZ/8
{
signed_word i;
for (i=0;i < last_set_bit;i++){
if (!GC_get_bit(bm,i)){
break;
}
}
if (i==last_set_bit){
return (WORDS_TO_BYTES(last_set_bit+1)|GC_DS_LENGTH);
}
}
#endif
if ((word)last_set_bit < BITMAP_BITS){
signed_word i;
result=SIGNB;
for (i=last_set_bit - 1;i>=0;i--){
result>>=1;
if (GC_get_bit(bm,i))result|=SIGNB;
}
result|=GC_DS_BITMAP;
} else {
signed_word index=GC_add_ext_descriptor(bm,(word)last_set_bit+1);
if (index==-1)return(WORDS_TO_BYTES(last_set_bit+1)|GC_DS_LENGTH);
result=GC_MAKE_PROC(GC_typed_mark_proc_index,(word)index);
}
return result;
}
GC_API GC_ATTR_MALLOC void*GC_CALL GC_malloc_explicitly_typed(size_t lb,
GC_descr d)
{
word*op;
size_t lg;
GC_ASSERT(GC_explicit_typing_initialized);
lb=SIZET_SAT_ADD(lb,TYPD_EXTRA_BYTES);
op=(word*)GC_malloc_kind(lb,GC_explicit_kind);
if (EXPECT(NULL==op,FALSE))
return NULL;
lg=BYTES_TO_GRANULES(GC_size(op));
op[GRANULES_TO_WORDS(lg)- 1]=d;
GC_dirty(op+GRANULES_TO_WORDS(lg)- 1);
REACHABLE_AFTER_DIRTY(d);
return op;
}
#define GENERAL_MALLOC_IOP(lb,k)GC_clear_stack(GC_generic_malloc_ignore_off_page(lb,k))
GC_API GC_ATTR_MALLOC void*GC_CALL
GC_malloc_explicitly_typed_ignore_off_page(size_t lb,GC_descr d)
{
ptr_t op;
size_t lg;
DCL_LOCK_STATE;
GC_ASSERT(GC_explicit_typing_initialized);
lb=SIZET_SAT_ADD(lb,TYPD_EXTRA_BYTES);
if (SMALL_OBJ(lb)){
void**opp;
GC_DBG_COLLECT_AT_MALLOC(lb);
LOCK();
lg=GC_size_map[lb];
opp=&GC_obj_kinds[GC_explicit_kind].ok_freelist[lg];
op=(ptr_t)(*opp);
if (EXPECT(0==op,FALSE)){
UNLOCK();
op=(ptr_t)GENERAL_MALLOC_IOP(lb,GC_explicit_kind);
if (0==op)return 0;
lg=BYTES_TO_GRANULES(GC_size(op));
} else {
*opp=obj_link(op);
obj_link(op)=0;
GC_bytes_allocd+=GRANULES_TO_BYTES((word)lg);
UNLOCK();
}
} else {
op=(ptr_t)GENERAL_MALLOC_IOP(lb,GC_explicit_kind);
if (NULL==op)return NULL;
lg=BYTES_TO_GRANULES(GC_size(op));
}
((word*)op)[GRANULES_TO_WORDS(lg)- 1]=d;
GC_dirty(op+GRANULES_TO_WORDS(lg)- 1);
REACHABLE_AFTER_DIRTY(d);
return op;
}
GC_API GC_ATTR_MALLOC void*GC_CALL GC_calloc_explicitly_typed(size_t n,
size_t lb,GC_descr d)
{
word*op;
size_t lg;
GC_descr simple_descr;
complex_descriptor*complex_descr;
int descr_type;
struct LeafDescriptor leaf;
GC_ASSERT(GC_explicit_typing_initialized);
descr_type=GC_make_array_descriptor((word)n,(word)lb,d,&simple_descr,
&complex_descr,&leaf);
if ((lb|n)> GC_SQRT_SIZE_MAX
&&lb > 0&&n > GC_SIZE_MAX/lb)
return (*GC_get_oom_fn())(GC_SIZE_MAX);
lb*=n;
switch(descr_type){
case NO_MEM:return(0);
case SIMPLE:
return GC_malloc_explicitly_typed(lb,simple_descr);
case LEAF:
lb=SIZET_SAT_ADD(lb,
sizeof(struct LeafDescriptor)+TYPD_EXTRA_BYTES);
break;
case COMPLEX:
lb=SIZET_SAT_ADD(lb,TYPD_EXTRA_BYTES);
break;
}
op=(word*)GC_malloc_kind(lb,GC_array_kind);
if (EXPECT(NULL==op,FALSE))
return NULL;
lg=BYTES_TO_GRANULES(GC_size(op));
if (descr_type==LEAF){
volatile struct LeafDescriptor*lp=
(struct LeafDescriptor*)
(op+GRANULES_TO_WORDS(lg)
- (BYTES_TO_WORDS(sizeof(struct LeafDescriptor))+1));
lp->ld_tag=LEAF_TAG;
lp->ld_size=leaf.ld_size;
lp->ld_nelements=leaf.ld_nelements;
lp->ld_descriptor=leaf.ld_descriptor;
((volatile word*)op)[GRANULES_TO_WORDS(lg)- 1]=(word)lp;
} else {
#ifndef GC_NO_FINALIZATION
size_t lw=GRANULES_TO_WORDS(lg);
op[lw - 1]=(word)complex_descr;
GC_dirty(op+lw - 1);
REACHABLE_AFTER_DIRTY(complex_descr);
if (EXPECT(GC_general_register_disappearing_link(
(void**)(op+lw - 1),op)
==GC_NO_MEMORY,FALSE))
#endif
{
return (*GC_get_oom_fn())(lb);
}
}
return op;
}
#include <stdio.h>
#include <limits.h>
#include <stdarg.h>
#ifndef MSWINCE
#include <signal.h>
#endif
#ifdef GC_SOLARIS_THREADS
#include <sys/syscall.h>
#endif
#if defined(UNIX_LIKE)||defined(CYGWIN32)||defined(SYMBIAN)||(defined(CONSOLE_LOG)&&defined(MSWIN32))
#include <fcntl.h>
#include <sys/types.h>
#include <sys/stat.h>
#endif
#if defined(CONSOLE_LOG)&&defined(MSWIN32)&&defined(_MSC_VER)
#include <io.h>
#endif
#ifdef NONSTOP
#include <floss.h>
#endif
#ifdef THREADS
#ifdef PCR
#include "il/PCR_IL.h"
GC_INNER PCR_Th_ML GC_allocate_ml;
#elif defined(SN_TARGET_PSP2)
GC_INNER WapiMutex GC_allocate_ml_PSP2={ 0,NULL };
#elif defined(SN_TARGET_ORBIS)||defined(SN_TARGET_PS3)
#include <pthread.h>
GC_INNER pthread_mutex_t GC_allocate_ml;
#endif
#endif
#ifdef DYNAMIC_LOADING
#define GC_REGISTER_MAIN_STATIC_DATA()GC_register_main_static_data()
#elif defined(GC_DONT_REGISTER_MAIN_STATIC_DATA)
#define GC_REGISTER_MAIN_STATIC_DATA()FALSE
#else
#define GC_REGISTER_MAIN_STATIC_DATA()TRUE
#endif
#ifdef NEED_CANCEL_DISABLE_COUNT
__thread unsigned char GC_cancel_disable_count=0;
#endif
GC_FAR struct _GC_arrays GC_arrays;
GC_INNER unsigned GC_n_mark_procs=GC_RESERVED_MARK_PROCS;
GC_INNER unsigned GC_n_kinds=GC_N_KINDS_INITIAL_VALUE;
GC_INNER GC_bool GC_debugging_started=FALSE;
ptr_t GC_stackbottom=0;
#ifdef IA64
ptr_t GC_register_stackbottom=0;
#endif
int GC_dont_gc=FALSE;
int GC_dont_precollect=FALSE;
GC_bool GC_quiet=0;
#if!defined(NO_CLOCK)||!defined(SMALL_CONFIG)
int GC_print_stats=0;
#endif
#ifdef GC_PRINT_BACK_HEIGHT
GC_INNER GC_bool GC_print_back_height=TRUE;
#else
GC_INNER GC_bool GC_print_back_height=FALSE;
#endif
#ifndef NO_DEBUGGING
#ifdef GC_DUMP_REGULARLY
GC_INNER GC_bool GC_dump_regularly=TRUE;
#else
GC_INNER GC_bool GC_dump_regularly=FALSE;
#endif
#ifndef NO_CLOCK
STATIC CLOCK_TYPE GC_init_time;
#endif
#endif
#ifdef KEEP_BACK_PTRS
GC_INNER long GC_backtraces=0;
#endif
#ifdef FIND_LEAK
int GC_find_leak=1;
#else
int GC_find_leak=0;
#endif
#ifndef SHORT_DBG_HDRS
#ifdef GC_FINDLEAK_DELAY_FREE
GC_INNER GC_bool GC_findleak_delay_free=TRUE;
#else
GC_INNER GC_bool GC_findleak_delay_free=FALSE;
#endif
#endif
#ifdef ALL_INTERIOR_POINTERS
int GC_all_interior_pointers=1;
#else
int GC_all_interior_pointers=0;
#endif
#ifdef FINALIZE_ON_DEMAND
int GC_finalize_on_demand=1;
#else
int GC_finalize_on_demand=0;
#endif
#ifdef JAVA_FINALIZATION
int GC_java_finalization=1;
#else
int GC_java_finalization=0;
#endif
GC_finalizer_notifier_proc GC_finalizer_notifier=
(GC_finalizer_notifier_proc)0;
#ifdef GC_FORCE_UNMAP_ON_GCOLLECT
GC_INNER GC_bool GC_force_unmap_on_gcollect=TRUE;
#else
GC_INNER GC_bool GC_force_unmap_on_gcollect=FALSE;
#endif
#ifndef GC_LARGE_ALLOC_WARN_INTERVAL
#define GC_LARGE_ALLOC_WARN_INTERVAL 5
#endif
GC_INNER long GC_large_alloc_warn_interval=GC_LARGE_ALLOC_WARN_INTERVAL;
STATIC void*GC_CALLBACK GC_default_oom_fn(
size_t bytes_requested GC_ATTR_UNUSED)
{
return(0);
}
GC_oom_func GC_oom_fn=GC_default_oom_fn;
#ifdef CAN_HANDLE_FORK
#ifdef HANDLE_FORK
GC_INNER int GC_handle_fork=1;
#else
GC_INNER int GC_handle_fork=FALSE;
#endif
#elif!defined(HAVE_NO_FORK)
GC_API void GC_CALL GC_atfork_prepare(void)
{
#ifdef THREADS
ABORT("fork()handling unsupported");
#endif
}
GC_API void GC_CALL GC_atfork_parent(void)
{
}
GC_API void GC_CALL GC_atfork_child(void)
{
}
#endif
GC_API void GC_CALL GC_set_handle_fork(int value GC_ATTR_UNUSED)
{
#ifdef CAN_HANDLE_FORK
if (!GC_is_initialized)
GC_handle_fork=value>=-1?value:1;
#elif defined(THREADS)||(defined(DARWIN)&&defined(MPROTECT_VDB))
if (!GC_is_initialized&&value){
#ifndef SMALL_CONFIG
GC_init();
#ifndef THREADS
if (GC_manual_vdb)
return;
#endif
#endif
ABORT("fork()handling unsupported");
}
#else
#endif
}
STATIC void GC_init_size_map(void)
{
size_t i;
GC_size_map[0]=1;
for (i=1;i<=GRANULES_TO_BYTES(TINY_FREELISTS-1)- EXTRA_BYTES;i++){
GC_size_map[i]=ROUNDED_UP_GRANULES(i);
#ifndef _MSC_VER
GC_ASSERT(GC_size_map[i] < TINY_FREELISTS);
#endif
}
}
#ifndef SMALL_CLEAR_SIZE
#define SMALL_CLEAR_SIZE 256
#endif
#if defined(ALWAYS_SMALL_CLEAR_STACK)||defined(STACK_NOT_SCANNED)
GC_API void*GC_CALL GC_clear_stack(void*arg)
{
#ifndef STACK_NOT_SCANNED
word volatile dummy[SMALL_CLEAR_SIZE];
BZERO(( void*)dummy,sizeof(dummy));
#endif
return arg;
}
#else
#ifdef THREADS
#define BIG_CLEAR_SIZE 2048
#else
STATIC word GC_stack_last_cleared=0;
STATIC ptr_t GC_min_sp=NULL;
STATIC ptr_t GC_high_water=NULL;
STATIC word GC_bytes_allocd_at_reset=0;
#define DEGRADE_RATE 50
#endif
#if defined(ASM_CLEAR_CODE)
void*GC_clear_stack_inner(void*,ptr_t);
#else
void*GC_clear_stack_inner(void*arg,
#if defined(__APPLE_CC__)&&!GC_CLANG_PREREQ(6,0)
volatile
#endif
ptr_t limit)
{
#define CLEAR_SIZE 213
volatile word dummy[CLEAR_SIZE];
BZERO(( void*)dummy,sizeof(dummy));
if ((word)GC_approx_sp()COOLER_THAN (word)limit){
(void)GC_clear_stack_inner(arg,limit);
}
#if defined(CPPCHECK)
GC_noop1(dummy[0]);
#else
GC_noop1(COVERT_DATAFLOW(dummy));
#endif
return(arg);
}
#endif
#ifdef THREADS
GC_ATTR_NO_SANITIZE_THREAD
static unsigned next_random_no(void)
{
static unsigned random_no=0;
return++random_no % 13;
}
#endif
GC_API void*GC_CALL GC_clear_stack(void*arg)
{
ptr_t sp=GC_approx_sp();
#ifdef THREADS
word volatile dummy[SMALL_CLEAR_SIZE];
#endif
#define SLOP 400
#define GC_SLOP 4000
#define CLEAR_THRESHOLD 100000
#ifdef THREADS
if (next_random_no()==0){
ptr_t limit=sp;
MAKE_HOTTER(limit,BIG_CLEAR_SIZE*sizeof(word));
limit=(ptr_t)((word)limit&~0xf);
return GC_clear_stack_inner(arg,limit);
}
BZERO((void*)dummy,SMALL_CLEAR_SIZE*sizeof(word));
#else
if (GC_gc_no > GC_stack_last_cleared){
if (GC_stack_last_cleared==0)
GC_high_water=(ptr_t)GC_stackbottom;
GC_min_sp=GC_high_water;
GC_stack_last_cleared=GC_gc_no;
GC_bytes_allocd_at_reset=GC_bytes_allocd;
}
MAKE_COOLER(GC_high_water,WORDS_TO_BYTES(DEGRADE_RATE)+GC_SLOP);
if ((word)sp HOTTER_THAN (word)GC_high_water){
GC_high_water=sp;
}
MAKE_HOTTER(GC_high_water,GC_SLOP);
{
ptr_t limit=GC_min_sp;
MAKE_HOTTER(limit,SLOP);
if ((word)sp COOLER_THAN (word)limit){
limit=(ptr_t)((word)limit&~0xf);
GC_min_sp=sp;
return GC_clear_stack_inner(arg,limit);
}
}
if (GC_bytes_allocd - GC_bytes_allocd_at_reset > CLEAR_THRESHOLD){
GC_min_sp=sp;
MAKE_HOTTER(GC_min_sp,CLEAR_THRESHOLD/4);
if ((word)GC_min_sp HOTTER_THAN (word)GC_high_water)
GC_min_sp=GC_high_water;
GC_bytes_allocd_at_reset=GC_bytes_allocd;
}
#endif
return arg;
}
#endif
GC_API void*GC_CALL GC_base(void*p)
{
ptr_t r;
struct hblk*h;
bottom_index*bi;
hdr*candidate_hdr;
r=(ptr_t)p;
if (!EXPECT(GC_is_initialized,TRUE))return 0;
h=HBLKPTR(r);
GET_BI(r,bi);
candidate_hdr=HDR_FROM_BI(bi,r);
if (candidate_hdr==0)return(0);
while (IS_FORWARDING_ADDR_OR_NIL(candidate_hdr)){
h=FORWARDED_ADDR(h,candidate_hdr);
r=(ptr_t)h;
candidate_hdr=HDR(h);
}
if (HBLK_IS_FREE(candidate_hdr))return(0);
r=(ptr_t)((word)r&~(WORDS_TO_BYTES(1)- 1));
{
size_t offset=HBLKDISPL(r);
word sz=candidate_hdr->hb_sz;
size_t obj_displ=offset % sz;
ptr_t limit;
r-=obj_displ;
limit=r+sz;
if ((word)limit > (word)(h+1)&&sz<=HBLKSIZE){
return(0);
}
if ((word)p>=(word)limit)return(0);
}
return((void*)r);
}
GC_API int GC_CALL GC_is_heap_ptr(const void*p)
{
bottom_index*bi;
GC_ASSERT(GC_is_initialized);
GET_BI(p,bi);
return HDR_FROM_BI(bi,p)!=0;
}
GC_API size_t GC_CALL GC_size(const void*p)
{
hdr*hhdr=HDR(p);
return (size_t)hhdr->hb_sz;
}
GC_API size_t GC_CALL GC_get_heap_size(void)
{
return (size_t)(GC_heapsize - GC_unmapped_bytes);
}
GC_API size_t GC_CALL GC_get_free_bytes(void)
{
return (size_t)(GC_large_free_bytes - GC_unmapped_bytes);
}
GC_API size_t GC_CALL GC_get_unmapped_bytes(void)
{
return (size_t)GC_unmapped_bytes;
}
GC_API size_t GC_CALL GC_get_bytes_since_gc(void)
{
return (size_t)GC_bytes_allocd;
}
GC_API size_t GC_CALL GC_get_total_bytes(void)
{
return (size_t)(GC_bytes_allocd+GC_bytes_allocd_before_gc);
}
#ifndef GC_GET_HEAP_USAGE_NOT_NEEDED
GC_API size_t GC_CALL GC_get_size_map_at(int i)
{
if ((unsigned)i > MAXOBJBYTES)
return GC_SIZE_MAX;
return GRANULES_TO_BYTES(GC_size_map[i]);
}
GC_API void GC_CALL GC_get_heap_usage_safe(GC_word*pheap_size,
GC_word*pfree_bytes,GC_word*punmapped_bytes,
GC_word*pbytes_since_gc,GC_word*ptotal_bytes)
{
DCL_LOCK_STATE;
LOCK();
if (pheap_size!=NULL)
*pheap_size=GC_heapsize - GC_unmapped_bytes;
if (pfree_bytes!=NULL)
*pfree_bytes=GC_large_free_bytes - GC_unmapped_bytes;
if (punmapped_bytes!=NULL)
*punmapped_bytes=GC_unmapped_bytes;
if (pbytes_since_gc!=NULL)
*pbytes_since_gc=GC_bytes_allocd;
if (ptotal_bytes!=NULL)
*ptotal_bytes=GC_bytes_allocd+GC_bytes_allocd_before_gc;
UNLOCK();
}
GC_INNER word GC_reclaimed_bytes_before_gc=0;
static void fill_prof_stats(struct GC_prof_stats_s*pstats)
{
pstats->heapsize_full=GC_heapsize;
pstats->free_bytes_full=GC_large_free_bytes;
pstats->unmapped_bytes=GC_unmapped_bytes;
pstats->bytes_allocd_since_gc=GC_bytes_allocd;
pstats->allocd_bytes_before_gc=GC_bytes_allocd_before_gc;
pstats->non_gc_bytes=GC_non_gc_bytes;
pstats->gc_no=GC_gc_no;
#ifdef PARALLEL_MARK
pstats->markers_m1=(word)((signed_word)GC_markers_m1);
#else
pstats->markers_m1=0;
#endif
pstats->bytes_reclaimed_since_gc=GC_bytes_found > 0?
(word)GC_bytes_found:0;
pstats->reclaimed_bytes_before_gc=GC_reclaimed_bytes_before_gc;
pstats->expl_freed_bytes_since_gc=GC_bytes_freed;
}
#include <string.h>
GC_API size_t GC_CALL GC_get_prof_stats(struct GC_prof_stats_s*pstats,
size_t stats_sz)
{
struct GC_prof_stats_s stats;
DCL_LOCK_STATE;
LOCK();
fill_prof_stats(stats_sz>=sizeof(stats)?pstats:&stats);
UNLOCK();
if (stats_sz==sizeof(stats)){
return sizeof(stats);
} else if (stats_sz > sizeof(stats)){
memset((char*)pstats+sizeof(stats),0xff,stats_sz - sizeof(stats));
return sizeof(stats);
} else {
if (EXPECT(stats_sz > 0,TRUE))
BCOPY(&stats,pstats,stats_sz);
return stats_sz;
}
}
#ifdef THREADS
GC_API size_t GC_CALL GC_get_prof_stats_unsafe(
struct GC_prof_stats_s*pstats,
size_t stats_sz)
{
struct GC_prof_stats_s stats;
if (stats_sz>=sizeof(stats)){
fill_prof_stats(pstats);
if (stats_sz > sizeof(stats))
memset((char*)pstats+sizeof(stats),0xff,
stats_sz - sizeof(stats));
return sizeof(stats);
} else {
if (EXPECT(stats_sz > 0,TRUE)){
fill_prof_stats(&stats);
BCOPY(&stats,pstats,stats_sz);
}
return stats_sz;
}
}
#endif
#endif
#if defined(GC_DARWIN_THREADS)||defined(GC_OPENBSD_UTHREADS)||defined(GC_WIN32_THREADS)||(defined(NACL)&&defined(THREADS))
GC_API void GC_CALL GC_set_suspend_signal(int sig GC_ATTR_UNUSED)
{
}
GC_API void GC_CALL GC_set_thr_restart_signal(int sig GC_ATTR_UNUSED)
{
}
GC_API int GC_CALL GC_get_suspend_signal(void)
{
return -1;
}
GC_API int GC_CALL GC_get_thr_restart_signal(void)
{
return -1;
}
#endif
#if!defined(_MAX_PATH)&&(defined(MSWIN32)||defined(MSWINCE)||defined(CYGWIN32))
#define _MAX_PATH MAX_PATH
#endif
#ifdef GC_READ_ENV_FILE
STATIC char*GC_envfile_content=NULL;
STATIC unsigned GC_envfile_length=0;
#ifndef GC_ENVFILE_MAXLEN
#define GC_ENVFILE_MAXLEN 0x4000
#endif
#define GC_ENV_FILE_EXT ".gc.env"
STATIC void GC_envfile_init(void)
{
#if defined(MSWIN32)||defined(MSWINCE)||defined(CYGWIN32)
HANDLE hFile;
char*content;
unsigned ofs;
unsigned len;
DWORD nBytesRead;
TCHAR path[_MAX_PATH+0x10];
len=(unsigned)GetModuleFileName(NULL,path,
_MAX_PATH+1);
if (len > 4&&path[len - 4]==(TCHAR)'.'){
len-=4;
}
BCOPY(TEXT(GC_ENV_FILE_EXT),&path[len],sizeof(TEXT(GC_ENV_FILE_EXT)));
hFile=CreateFile(path,GENERIC_READ,
FILE_SHARE_READ|FILE_SHARE_WRITE,
NULL,OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL,NULL);
if (hFile==INVALID_HANDLE_VALUE)
return;
len=(unsigned)GetFileSize(hFile,NULL);
if (len<=1||len>=GC_ENVFILE_MAXLEN){
CloseHandle(hFile);
return;
}
GC_ASSERT(GC_page_size!=0);
content=(char*)GET_MEM(ROUNDUP_PAGESIZE_IF_MMAP((size_t)len+1));
if (content==NULL){
CloseHandle(hFile);
return;
}
ofs=0;
nBytesRead=(DWORD)-1L;
while (ReadFile(hFile,content+ofs,len - ofs+1,&nBytesRead,
NULL)&&nBytesRead!=0){
if ((ofs+=nBytesRead)> len)
break;
}
CloseHandle(hFile);
if (ofs!=len||nBytesRead!=0)
return;
content[ofs]='\0';
while (ofs--> 0){
if (content[ofs]=='\r'||content[ofs]=='\n')
content[ofs]='\0';
}
GC_ASSERT(NULL==GC_envfile_content);
GC_envfile_length=len+1;
GC_envfile_content=content;
#endif
}
GC_INNER char*GC_envfile_getenv(const char*name)
{
char*p;
char*end_of_content;
unsigned namelen;
#ifndef NO_GETENV
p=getenv(name);
if (p!=NULL)
return*p!='\0'?p:NULL;
#endif
p=GC_envfile_content;
if (p==NULL)
return NULL;
namelen=strlen(name);
if (namelen==0)
return NULL;
for (end_of_content=p+GC_envfile_length;
p!=end_of_content;p+=strlen(p)+1){
if (strncmp(p,name,namelen)==0&&*(p+=namelen)=='='){
p++;
return*p!='\0'?p:NULL;
}
}
return NULL;
}
#endif
GC_INNER GC_bool GC_is_initialized=FALSE;
GC_API int GC_CALL GC_is_init_called(void)
{
return GC_is_initialized;
}
#if defined(GC_WIN32_THREADS)&&((defined(MSWIN32)&&!defined(CONSOLE_LOG))||defined(MSWINCE))
GC_INNER CRITICAL_SECTION GC_write_cs;
#endif
#ifndef DONT_USE_ATEXIT
#if!defined(PCR)&&!defined(SMALL_CONFIG)
static GC_bool skip_gc_atexit=FALSE;
#else
#define skip_gc_atexit FALSE
#endif
STATIC void GC_exit_check(void)
{
if (GC_find_leak&&!skip_gc_atexit){
#ifdef THREADS
GC_in_thread_creation=TRUE;
GC_gcollect();
GC_in_thread_creation=FALSE;
#else
GC_gcollect();
#endif
}
}
#endif
#if defined(UNIX_LIKE)&&!defined(NO_DEBUGGING)
static void looping_handler(int sig)
{
GC_err_printf("Caught signal %d:looping in handler\n",sig);
for (;;){
}
}
static GC_bool installed_looping_handler=FALSE;
static void maybe_install_looping_handler(void)
{
if (!installed_looping_handler&&0!=GETENV("GC_LOOP_ON_ABORT")){
GC_set_and_save_fault_handler(looping_handler);
installed_looping_handler=TRUE;
}
}
#else
#define maybe_install_looping_handler()
#endif
#define GC_DEFAULT_STDOUT_FD 1
#define GC_DEFAULT_STDERR_FD 2
#if!defined(OS2)&&!defined(MACOS)&&!defined(GC_ANDROID_LOG)&&!defined(NN_PLATFORM_CTR)&&!defined(NINTENDO_SWITCH)&&(!defined(MSWIN32)||defined(CONSOLE_LOG))&&!defined(MSWINCE)
STATIC int GC_stdout=GC_DEFAULT_STDOUT_FD;
STATIC int GC_stderr=GC_DEFAULT_STDERR_FD;
STATIC int GC_log=GC_DEFAULT_STDERR_FD;
#ifndef MSWIN32
GC_API void GC_CALL GC_set_log_fd(int fd)
{
GC_log=fd;
}
#endif
#endif
#ifdef MSGBOX_ON_ERROR
STATIC void GC_win32_MessageBoxA(const char*msg,const char*caption,
unsigned flags)
{
#ifndef DONT_USE_USER32_DLL
(void)MessageBoxA(NULL,msg,caption,flags);
#else
HINSTANCE hU32=LoadLibrary(TEXT("user32.dll"));
if (hU32){
FARPROC pfn=GetProcAddress(hU32,"MessageBoxA");
if (pfn)
(void)(*(int (WINAPI*)(HWND,LPCSTR,LPCSTR,UINT))(word)pfn)(
NULL,msg,caption,flags);
(void)FreeLibrary(hU32);
}
#endif
}
#endif
#if defined(THREADS)&&defined(UNIX_LIKE)&&!defined(NO_GETCONTEXT)
static void callee_saves_pushed_dummy_fn(ptr_t data GC_ATTR_UNUSED,
void*context GC_ATTR_UNUSED){}
#endif
#ifndef SMALL_CONFIG
#ifdef MANUAL_VDB
static GC_bool manual_vdb_allowed=TRUE;
#else
static GC_bool manual_vdb_allowed=FALSE;
#endif
GC_API void GC_CALL GC_set_manual_vdb_allowed(int value)
{
manual_vdb_allowed=(GC_bool)value;
}
GC_API int GC_CALL GC_get_manual_vdb_allowed(void)
{
return (int)manual_vdb_allowed;
}
#endif
STATIC word GC_parse_mem_size_arg(const char*str)
{
word result=0;
if (*str!='\0'){
char*endptr;
char ch;
result=(word)STRTOULL(str,&endptr,10);
ch=*endptr;
if (ch!='\0'){
if (*(endptr+1)!='\0')
return 0;
switch (ch){
case 'K':
case 'k':
result<<=10;
break;
case 'M':
case 'm':
result<<=20;
break;
case 'G':
case 'g':
result<<=30;
break;
default:
result=0;
}
}
}
return result;
}
#define GC_LOG_STD_NAME "gc.log"
GC_API void GC_CALL GC_init(void)
{
word initial_heap_sz;
IF_CANCEL(int cancel_state;)
#if defined(GC_ASSERTIONS)&&defined(GC_ALWAYS_MULTITHREADED)
DCL_LOCK_STATE;
#endif
if (EXPECT(GC_is_initialized,TRUE))return;
#ifdef REDIRECT_MALLOC
{
static GC_bool init_started=FALSE;
if (init_started)
ABORT("Redirected malloc()called during GC init");
init_started=TRUE;
}
#endif
#if defined(GC_INITIAL_HEAP_SIZE)&&!defined(CPPCHECK)
initial_heap_sz=GC_INITIAL_HEAP_SIZE;
#else
initial_heap_sz=MINHINCR*HBLKSIZE;
#endif
DISABLE_CANCEL(cancel_state);
#ifdef THREADS
#ifndef GC_ALWAYS_MULTITHREADED
GC_ASSERT(!GC_need_to_lock);
#endif
#ifdef SN_TARGET_PS3
{
pthread_mutexattr_t mattr;
if (0!=pthread_mutexattr_init(&mattr)){
ABORT("pthread_mutexattr_init failed");
}
if (0!=pthread_mutex_init(&GC_allocate_ml,&mattr)){
ABORT("pthread_mutex_init failed");
}
(void)pthread_mutexattr_destroy(&mattr);
}
#endif
#endif
#if defined(GC_WIN32_THREADS)&&!defined(GC_PTHREADS)
#ifndef SPIN_COUNT
#define SPIN_COUNT 4000
#endif
#ifdef MSWINRT_FLAVOR
InitializeCriticalSectionAndSpinCount(&GC_allocate_ml,SPIN_COUNT);
#else
{
#ifndef MSWINCE
FARPROC pfn=0;
HMODULE hK32=GetModuleHandle(TEXT("kernel32.dll"));
if (hK32)
pfn=GetProcAddress(hK32,
"InitializeCriticalSectionAndSpinCount");
if (pfn){
(*(BOOL (WINAPI*)(LPCRITICAL_SECTION,DWORD))(word)pfn)(
&GC_allocate_ml,SPIN_COUNT);
} else
#endif
InitializeCriticalSection(&GC_allocate_ml);
}
#endif
#endif
#if defined(GC_WIN32_THREADS)&&((defined(MSWIN32)&&!defined(CONSOLE_LOG))||defined(MSWINCE))
InitializeCriticalSection(&GC_write_cs);
#endif
GC_setpagesize();
#ifdef MSWIN32
GC_init_win32();
#endif
#ifdef GC_READ_ENV_FILE
GC_envfile_init();
#endif
#if!defined(NO_CLOCK)||!defined(SMALL_CONFIG)
#ifdef GC_PRINT_VERBOSE_STATS
GC_print_stats=VERBOSE;
#else
if (0!=GETENV("GC_PRINT_VERBOSE_STATS")){
GC_print_stats=VERBOSE;
} else if (0!=GETENV("GC_PRINT_STATS")){
GC_print_stats=1;
}
#endif
#endif
#if ((defined(UNIX_LIKE)&&!defined(GC_ANDROID_LOG))||(defined(CONSOLE_LOG)&&defined(MSWIN32))||defined(CYGWIN32)||defined(SYMBIAN))&&!defined(SMALL_CONFIG)
{
char*file_name=TRUSTED_STRING(GETENV("GC_LOG_FILE"));
#ifdef GC_LOG_TO_FILE_ALWAYS
if (NULL==file_name)
file_name=GC_LOG_STD_NAME;
#else
if (0!=file_name)
#endif
{
#if defined(_MSC_VER)
int log_d=_open(file_name,O_CREAT|O_WRONLY|O_APPEND);
#else
int log_d=open(file_name,O_CREAT|O_WRONLY|O_APPEND,0644);
#endif
if (log_d < 0){
GC_err_printf("Failed to open %s as log file\n",file_name);
} else {
char*str;
GC_log=log_d;
str=GETENV("GC_ONLY_LOG_TO_FILE");
#ifdef GC_ONLY_LOG_TO_FILE
if (str!=NULL&&*str=='0'&&*(str+1)=='\0')
#else
if (str==NULL||(*str=='0'&&*(str+1)=='\0'))
#endif
{
GC_stdout=log_d;
GC_stderr=log_d;
}
}
}
}
#endif
#if!defined(NO_DEBUGGING)&&!defined(GC_DUMP_REGULARLY)
if (0!=GETENV("GC_DUMP_REGULARLY")){
GC_dump_regularly=TRUE;
}
#endif
#ifdef KEEP_BACK_PTRS
{
char*backtraces_string=GETENV("GC_BACKTRACES");
if (0!=backtraces_string){
GC_backtraces=atol(backtraces_string);
if (backtraces_string[0]=='\0')GC_backtraces=1;
}
}
#endif
if (0!=GETENV("GC_FIND_LEAK")){
GC_find_leak=1;
}
#ifndef SHORT_DBG_HDRS
if (0!=GETENV("GC_FINDLEAK_DELAY_FREE")){
GC_findleak_delay_free=TRUE;
}
#endif
if (0!=GETENV("GC_ALL_INTERIOR_POINTERS")){
GC_all_interior_pointers=1;
}
if (0!=GETENV("GC_DONT_GC")){
GC_dont_gc=1;
}
if (0!=GETENV("GC_PRINT_BACK_HEIGHT")){
GC_print_back_height=TRUE;
}
if (0!=GETENV("GC_NO_BLACKLIST_WARNING")){
GC_large_alloc_warn_interval=LONG_MAX;
}
{
char*addr_string=GETENV("GC_TRACE");
if (0!=addr_string){
#ifndef ENABLE_TRACE
WARN("Tracing not enabled:Ignoring GC_TRACE value\n",0);
#else
word addr=(word)STRTOULL(addr_string,NULL,16);
if (addr < 0x1000)
WARN("Unlikely trace address:%p\n",(void*)addr);
GC_trace_addr=(ptr_t)addr;
#endif
}
}
#ifdef GC_COLLECT_AT_MALLOC
{
char*string=GETENV("GC_COLLECT_AT_MALLOC");
if (0!=string){
size_t min_lb=(size_t)STRTOULL(string,NULL,10);
if (min_lb > 0)
GC_dbg_collect_at_malloc_min_lb=min_lb;
}
}
#endif
#if!defined(GC_DISABLE_INCREMENTAL)&&!defined(NO_CLOCK)
{
char*time_limit_string=GETENV("GC_PAUSE_TIME_TARGET");
if (0!=time_limit_string){
long time_limit=atol(time_limit_string);
if (time_limit > 0){
GC_time_limit=time_limit;
}
}
}
#endif
#ifndef SMALL_CONFIG
{
char*full_freq_string=GETENV("GC_FULL_FREQUENCY");
if (full_freq_string!=NULL){
int full_freq=atoi(full_freq_string);
if (full_freq > 0)
GC_full_freq=full_freq;
}
}
#endif
{
char*interval_string=GETENV("GC_LARGE_ALLOC_WARN_INTERVAL");
if (0!=interval_string){
long interval=atol(interval_string);
if (interval<=0){
WARN("GC_LARGE_ALLOC_WARN_INTERVAL environment variable has "
"bad value:Ignoring\n",0);
} else {
GC_large_alloc_warn_interval=interval;
}
}
}
{
char*space_divisor_string=GETENV("GC_FREE_SPACE_DIVISOR");
if (space_divisor_string!=NULL){
int space_divisor=atoi(space_divisor_string);
if (space_divisor > 0)
GC_free_space_divisor=(unsigned)space_divisor;
}
}
#ifdef USE_MUNMAP
{
char*string=GETENV("GC_UNMAP_THRESHOLD");
if (string!=NULL){
if (*string=='0'&&*(string+1)=='\0'){
GC_unmap_threshold=0;
} else {
int unmap_threshold=atoi(string);
if (unmap_threshold > 0)
GC_unmap_threshold=unmap_threshold;
}
}
}
{
char*string=GETENV("GC_FORCE_UNMAP_ON_GCOLLECT");
if (string!=NULL){
if (*string=='0'&&*(string+1)=='\0'){
GC_force_unmap_on_gcollect=FALSE;
} else {
GC_force_unmap_on_gcollect=TRUE;
}
}
}
{
char*string=GETENV("GC_USE_ENTIRE_HEAP");
if (string!=NULL){
if (*string=='0'&&*(string+1)=='\0'){
GC_use_entire_heap=FALSE;
} else {
GC_use_entire_heap=TRUE;
}
}
}
#endif
#if!defined(NO_DEBUGGING)&&!defined(NO_CLOCK)
GET_TIME(GC_init_time);
#endif
maybe_install_looping_handler();
#if ALIGNMENT > GC_DS_TAGS
if (EXTRA_BYTES!=0)
GC_obj_kinds[NORMAL].ok_descriptor=(word)(-ALIGNMENT)|GC_DS_LENGTH;
#endif
GC_exclude_static_roots_inner(beginGC_arrays,endGC_arrays);
GC_exclude_static_roots_inner(beginGC_obj_kinds,endGC_obj_kinds);
#ifdef SEPARATE_GLOBALS
GC_exclude_static_roots_inner(beginGC_objfreelist,endGC_objfreelist);
GC_exclude_static_roots_inner(beginGC_aobjfreelist,endGC_aobjfreelist);
#endif
#if defined(USE_PROC_FOR_LIBRARIES)&&defined(GC_LINUX_THREADS)
WARN("USE_PROC_FOR_LIBRARIES+GC_LINUX_THREADS performs poorly.\n",0);
#endif
#if defined(SEARCH_FOR_DATA_START)
GC_init_linux_data_start();
#endif
#if defined(NETBSD)&&defined(__ELF__)
GC_init_netbsd_elf();
#endif
#if!defined(THREADS)||defined(GC_PTHREADS)||defined(NN_PLATFORM_CTR)||defined(NINTENDO_SWITCH)||defined(GC_WIN32_THREADS)||defined(GC_SOLARIS_THREADS)
if (GC_stackbottom==0){
GC_stackbottom=GC_get_main_stack_base();
#if (defined(LINUX)||defined(HPUX))&&defined(IA64)
GC_register_stackbottom=GC_get_register_stack_base();
#endif
} else {
#if (defined(LINUX)||defined(HPUX))&&defined(IA64)
if (GC_register_stackbottom==0){
WARN("GC_register_stackbottom should be set with GC_stackbottom\n",0);
GC_register_stackbottom=GC_get_register_stack_base();
}
#endif
}
#endif
#if!defined(CPPCHECK)
GC_STATIC_ASSERT(sizeof(ptr_t)==sizeof(word));
GC_STATIC_ASSERT(sizeof(signed_word)==sizeof(word));
#if!defined(_AUX_SOURCE)||defined(__GNUC__)
GC_STATIC_ASSERT((word)(-1)> (word)0);
#endif
GC_STATIC_ASSERT((signed_word)(-1)< (signed_word)0);
#endif
GC_STATIC_ASSERT(sizeof (struct hblk)==HBLKSIZE);
#ifndef THREADS
GC_ASSERT(!((word)GC_stackbottom HOTTER_THAN (word)GC_approx_sp()));
#endif
#ifndef GC_DISABLE_INCREMENTAL
if (GC_incremental||0!=GETENV("GC_ENABLE_INCREMENTAL")){
#if defined(BASE_ATOMIC_OPS_EMULATED)||defined(CHECKSUMS)||defined(REDIRECT_MALLOC)||defined(REDIRECT_MALLOC_IN_HEADER)||defined(SMALL_CONFIG)
#else
if (manual_vdb_allowed){
GC_manual_vdb=TRUE;
GC_incremental=TRUE;
} else
#endif
{
GC_incremental=GC_dirty_init();
GC_ASSERT(GC_bytes_allocd==0);
}
}
#endif
if (GC_REGISTER_MAIN_STATIC_DATA())GC_register_data_segments();
GC_init_headers();
GC_bl_init();
GC_mark_init();
{
char*sz_str=GETENV("GC_INITIAL_HEAP_SIZE");
if (sz_str!=NULL){
initial_heap_sz=GC_parse_mem_size_arg(sz_str);
if (initial_heap_sz<=MINHINCR*HBLKSIZE){
WARN("Bad initial heap size %s - ignoring it.\n",sz_str);
}
}
}
{
char*sz_str=GETENV("GC_MAXIMUM_HEAP_SIZE");
if (sz_str!=NULL){
word max_heap_sz=GC_parse_mem_size_arg(sz_str);
if (max_heap_sz < initial_heap_sz){
WARN("Bad maximum heap size %s - ignoring it.\n",sz_str);
}
if (0==GC_max_retries)GC_max_retries=2;
GC_set_max_heap_size(max_heap_sz);
}
}
#if defined(GC_ASSERTIONS)&&defined(GC_ALWAYS_MULTITHREADED)
LOCK();
#endif
if (!GC_expand_hp_inner(divHBLKSZ(initial_heap_sz))){
GC_err_printf("Can't start up:not enough memory\n");
EXIT();
} else {
GC_requested_heapsize+=initial_heap_sz;
}
if (GC_all_interior_pointers)
GC_initialize_offsets();
GC_register_displacement_inner(0L);
#if defined(GC_LINUX_THREADS)&&defined(REDIRECT_MALLOC)
if (!GC_all_interior_pointers){
GC_register_displacement_inner(sizeof(void*));
}
#endif
GC_init_size_map();
#ifdef PCR
if (PCR_IL_Lock(PCR_Bool_false,PCR_allSigsBlocked,PCR_waitForever)
!=PCR_ERes_okay){
ABORT("Can't lock load state");
} else if (PCR_IL_Unlock()!=PCR_ERes_okay){
ABORT("Can't unlock load state");
}
PCR_IL_Unlock();
GC_pcr_install();
#endif
GC_is_initialized=TRUE;
#if defined(GC_PTHREADS)||defined(GC_WIN32_THREADS)
GC_thr_init();
#ifdef PARALLEL_MARK
#if defined(GC_ASSERTIONS)&&defined(GC_ALWAYS_MULTITHREADED)
UNLOCK();
#endif
GC_start_mark_threads_inner();
#if defined(GC_ASSERTIONS)&&defined(GC_ALWAYS_MULTITHREADED)
LOCK();
#endif
#endif
#endif
COND_DUMP;
if (!GC_dont_precollect||GC_incremental){
GC_gcollect_inner();
}
#if defined(GC_ASSERTIONS)&&defined(GC_ALWAYS_MULTITHREADED)
UNLOCK();
#endif
#if defined(THREADS)&&defined(UNIX_LIKE)&&!defined(NO_GETCONTEXT)
if (GC_dont_gc||GC_dont_precollect)
GC_with_callee_saves_pushed(callee_saves_pushed_dummy_fn,NULL);
#endif
#ifndef DONT_USE_ATEXIT
if (GC_find_leak){
atexit(GC_exit_check);
}
#endif
#if defined(PARALLEL_MARK)||defined(THREAD_LOCAL_ALLOC)||(defined(GC_ALWAYS_MULTITHREADED)&&defined(GC_WIN32_THREADS)&&!defined(GC_NO_THREADS_DISCOVERY))
GC_init_parallel();
#endif
#if defined(DYNAMIC_LOADING)&&defined(DARWIN)
GC_init_dyld();
#endif
RESTORE_CANCEL(cancel_state);
}
GC_API void GC_CALL GC_enable_incremental(void)
{
#if!defined(GC_DISABLE_INCREMENTAL)&&!defined(KEEP_BACK_PTRS)
DCL_LOCK_STATE;
if (!GC_find_leak&&0==GETENV("GC_DISABLE_INCREMENTAL")){
LOCK();
if (!GC_incremental){
GC_setpagesize();
maybe_install_looping_handler();
if (!GC_is_initialized){
UNLOCK();
GC_incremental=TRUE;
GC_init();
LOCK();
} else {
#if!defined(BASE_ATOMIC_OPS_EMULATED)&&!defined(CHECKSUMS)&&!defined(REDIRECT_MALLOC)&&!defined(REDIRECT_MALLOC_IN_HEADER)&&!defined(SMALL_CONFIG)
if (manual_vdb_allowed){
GC_manual_vdb=TRUE;
GC_incremental=TRUE;
} else
#endif
{
GC_incremental=GC_dirty_init();
}
}
if (GC_incremental&&!GC_dont_gc){
IF_CANCEL(int cancel_state;)
DISABLE_CANCEL(cancel_state);
if (GC_bytes_allocd > 0){
GC_gcollect_inner();
}
GC_read_dirty(FALSE);
RESTORE_CANCEL(cancel_state);
}
}
UNLOCK();
return;
}
#endif
GC_init();
}
#if defined(THREADS)
GC_API void GC_CALL GC_start_mark_threads(void)
{
#if defined(PARALLEL_MARK)&&defined(CAN_HANDLE_FORK)&&!defined(THREAD_SANITIZER)
IF_CANCEL(int cancel_state;)
DISABLE_CANCEL(cancel_state);
GC_start_mark_threads_inner();
RESTORE_CANCEL(cancel_state);
#else
GC_ASSERT(I_DONT_HOLD_LOCK());
#endif
}
#endif
GC_API void GC_CALL GC_deinit(void)
{
if (GC_is_initialized){
GC_is_initialized=FALSE;
#if defined(GC_WIN32_THREADS)&&(defined(MSWIN32)||defined(MSWINCE))
#if!defined(CONSOLE_LOG)||defined(MSWINCE)
DeleteCriticalSection(&GC_write_cs);
#endif
DeleteCriticalSection(&GC_allocate_ml);
#endif
}
}
#if (defined(MSWIN32)&&!defined(CONSOLE_LOG))||defined(MSWINCE)
#if defined(_MSC_VER)&&defined(_DEBUG)&&!defined(MSWINCE)
#include <crtdbg.h>
#endif
STATIC HANDLE GC_log=0;
#ifdef THREADS
#if defined(PARALLEL_MARK)&&!defined(GC_ALWAYS_MULTITHREADED)
#define IF_NEED_TO_LOCK(x)if (GC_parallel||GC_need_to_lock)x
#else
#define IF_NEED_TO_LOCK(x)if (GC_need_to_lock)x
#endif
#else
#define IF_NEED_TO_LOCK(x)
#endif
#ifdef MSWINRT_FLAVOR
#include <windows.storage.h>
DECLSPEC_IMPORT HRESULT WINAPI RoGetActivationFactory(
HSTRING activatableClassId,
REFIID iid,void**factory);
static GC_bool getWinRTLogPath(wchar_t*buf,size_t bufLen)
{
static const GUID kIID_IApplicationDataStatics={
0x5612147B,0xE843,0x45E3,
0x94,0xD8,0x06,0x16,0x9E,0x3C,0x8E,0x17
};
static const GUID kIID_IStorageItem={
0x4207A996,0xCA2F,0x42F7,
0xBD,0xE8,0x8B,0x10,0x45,0x7A,0x7F,0x30
};
GC_bool result=FALSE;
HSTRING_HEADER appDataClassNameHeader;
HSTRING appDataClassName;
__x_ABI_CWindows_CStorage_CIApplicationDataStatics*appDataStatics=0;
GC_ASSERT(bufLen > 0);
if (SUCCEEDED(WindowsCreateStringReference(
RuntimeClass_Windows_Storage_ApplicationData,
(sizeof(RuntimeClass_Windows_Storage_ApplicationData)-1)
/sizeof(wchar_t),
&appDataClassNameHeader,&appDataClassName))
&&SUCCEEDED(RoGetActivationFactory(appDataClassName,
&kIID_IApplicationDataStatics,
&appDataStatics))){
__x_ABI_CWindows_CStorage_CIApplicationData*appData=NULL;
__x_ABI_CWindows_CStorage_CIStorageFolder*tempFolder=NULL;
__x_ABI_CWindows_CStorage_CIStorageItem*tempFolderItem=NULL;
HSTRING tempPath=NULL;
if (SUCCEEDED(appDataStatics->lpVtbl->get_Current(appDataStatics,
&appData))
&&SUCCEEDED(appData->lpVtbl->get_TemporaryFolder(appData,
&tempFolder))
&&SUCCEEDED(tempFolder->lpVtbl->QueryInterface(tempFolder,
&kIID_IStorageItem,
&tempFolderItem))
&&SUCCEEDED(tempFolderItem->lpVtbl->get_Path(tempFolderItem,
&tempPath))){
UINT32 tempPathLen;
const wchar_t*tempPathBuf=
WindowsGetStringRawBuffer(tempPath,&tempPathLen);
buf[0]='\0';
if (wcsncat_s(buf,bufLen,tempPathBuf,tempPathLen)==0
&&wcscat_s(buf,bufLen,L"\\")==0
&&wcscat_s(buf,bufLen,TEXT(GC_LOG_STD_NAME))==0)
result=TRUE;
WindowsDeleteString(tempPath);
}
if (tempFolderItem!=NULL)
tempFolderItem->lpVtbl->Release(tempFolderItem);
if (tempFolder!=NULL)
tempFolder->lpVtbl->Release(tempFolder);
if (appData!=NULL)
appData->lpVtbl->Release(appData);
appDataStatics->lpVtbl->Release(appDataStatics);
}
return result;
}
#endif
STATIC HANDLE GC_CreateLogFile(void)
{
HANDLE hFile;
#ifdef MSWINRT_FLAVOR
TCHAR pathBuf[_MAX_PATH+0x10];
hFile=INVALID_HANDLE_VALUE;
if (getWinRTLogPath(pathBuf,_MAX_PATH+1)){
CREATEFILE2_EXTENDED_PARAMETERS extParams;
BZERO(&extParams,sizeof(extParams));
extParams.dwSize=sizeof(extParams);
extParams.dwFileAttributes=FILE_ATTRIBUTE_NORMAL;
extParams.dwFileFlags=GC_print_stats==VERBOSE?0
:FILE_FLAG_WRITE_THROUGH;
hFile=CreateFile2(pathBuf,GENERIC_WRITE,FILE_SHARE_READ,
CREATE_ALWAYS,&extParams);
}
#else
TCHAR*logPath;
#if defined(NO_GETENV_WIN32)&&defined(CPPCHECK)
#define appendToFile FALSE
#else
BOOL appendToFile=FALSE;
#endif
#if!defined(NO_GETENV_WIN32)||!defined(OLD_WIN32_LOG_FILE)
TCHAR pathBuf[_MAX_PATH+0x10];
logPath=pathBuf;
#endif
#ifndef NO_GETENV_WIN32
if (GetEnvironmentVariable(TEXT("GC_LOG_FILE"),pathBuf,
_MAX_PATH+1)- 1U < (DWORD)_MAX_PATH){
appendToFile=TRUE;
} else
#endif
{
#ifdef OLD_WIN32_LOG_FILE
logPath=TEXT(GC_LOG_STD_NAME);
#else
int len=(int)GetModuleFileName(NULL,pathBuf,
_MAX_PATH+1);
if (len > 4&&pathBuf[len - 4]==(TCHAR)'.'){
len-=4;
}
BCOPY(TEXT(".")TEXT(GC_LOG_STD_NAME),&pathBuf[len],
sizeof(TEXT(".")TEXT(GC_LOG_STD_NAME)));
#endif
}
hFile=CreateFile(logPath,GENERIC_WRITE,FILE_SHARE_READ,
NULL,
appendToFile?OPEN_ALWAYS:CREATE_ALWAYS,
GC_print_stats==VERBOSE?FILE_ATTRIBUTE_NORMAL:
FILE_ATTRIBUTE_NORMAL|FILE_FLAG_WRITE_THROUGH,
NULL);
#ifndef NO_GETENV_WIN32
if (appendToFile&&hFile!=INVALID_HANDLE_VALUE){
LONG posHigh=0;
(void)SetFilePointer(hFile,0,&posHigh,FILE_END);
}
#endif
#undef appendToFile
#endif
return hFile;
}
STATIC int GC_write(const char*buf,size_t len)
{
BOOL res;
DWORD written;
#if defined(THREADS)&&defined(GC_ASSERTIONS)
static GC_bool inside_write=FALSE;
if (inside_write)
return -1;
#endif
if (len==0)
return 0;
IF_NEED_TO_LOCK(EnterCriticalSection(&GC_write_cs));
#if defined(THREADS)&&defined(GC_ASSERTIONS)
if (GC_write_disabled){
inside_write=TRUE;
ABORT("Assertion failure:GC_write called with write_disabled");
}
#endif
if (GC_log==0){
GC_log=GC_CreateLogFile();
}
if (GC_log==INVALID_HANDLE_VALUE){
IF_NEED_TO_LOCK(LeaveCriticalSection(&GC_write_cs));
#ifdef NO_DEBUGGING
return 0;
#else
return -1;
#endif
}
res=WriteFile(GC_log,buf,(DWORD)len,&written,NULL);
#if defined(_MSC_VER)&&defined(_DEBUG)&&!defined(NO_CRT)
#ifdef MSWINCE
{
WCHAR wbuf[1024];
wbuf[MultiByteToWideChar(CP_ACP,0,
buf,len,wbuf,
sizeof(wbuf)/sizeof(wbuf[0])- 1)]=0;
OutputDebugStringW(wbuf);
}
#else
_CrtDbgReport(_CRT_WARN,NULL,0,NULL,"%.*s",len,buf);
#endif
#endif
IF_NEED_TO_LOCK(LeaveCriticalSection(&GC_write_cs));
return res?(int)written:-1;
}
#define WRITE(f,buf,len)GC_write(buf,len)
#elif defined(OS2)||defined(MACOS)
STATIC FILE*GC_stdout=NULL;
STATIC FILE*GC_stderr=NULL;
STATIC FILE*GC_log=NULL;
STATIC void GC_set_files(void)
{
if (GC_stdout==NULL){
GC_stdout=stdout;
}
if (GC_stderr==NULL){
GC_stderr=stderr;
}
if (GC_log==NULL){
GC_log=stderr;
}
}
GC_INLINE int GC_write(FILE*f,const char*buf,size_t len)
{
int res=fwrite(buf,1,len,f);
fflush(f);
return res;
}
#define WRITE(f,buf,len)(GC_set_files(),GC_write(f,buf,len))
#elif defined(GC_ANDROID_LOG)
#include <android/log.h>
#ifndef GC_ANDROID_LOG_TAG
#define GC_ANDROID_LOG_TAG "BDWGC"
#endif
#define GC_stdout ANDROID_LOG_DEBUG
#define GC_stderr ANDROID_LOG_ERROR
#define GC_log GC_stdout
#define WRITE(level,buf,unused_len)__android_log_write(level,GC_ANDROID_LOG_TAG,buf)
#elif defined(NN_PLATFORM_CTR)
int n3ds_log_write(const char*text,int length);
#define WRITE(level,buf,len)n3ds_log_write(buf,len)
#elif defined(NINTENDO_SWITCH)
int switch_log_write(const char*text,int length);
#define WRITE(level,buf,len)switch_log_write(buf,len)
#else
#if!defined(SN_TARGET_ORBIS)&&!defined(SN_TARGET_PSP2)
#if!defined(AMIGA)&&!defined(MSWIN32)&&!defined(MSWIN_XBOX1)&&!defined(__CC_ARM)
#include <unistd.h>
#endif
#if!defined(ECOS)&&!defined(NOSYS)
#include <errno.h>
#endif
#endif
STATIC int GC_write(int fd,const char*buf,size_t len)
{
#if defined(ECOS)||defined(SN_TARGET_ORBIS)||defined(SN_TARGET_PSP2)||defined(NOSYS)
#ifdef ECOS
#else
#endif
return len;
#else
int bytes_written=0;
IF_CANCEL(int cancel_state;)
DISABLE_CANCEL(cancel_state);
while ((unsigned)bytes_written < len){
#ifdef GC_SOLARIS_THREADS
int result=syscall(SYS_write,fd,buf+bytes_written,
len - bytes_written);
#elif defined(_MSC_VER)
int result=_write(fd,buf+bytes_written,
(unsigned)(len - bytes_written));
#else
int result=write(fd,buf+bytes_written,len - bytes_written);
#endif
if (-1==result){
if (EAGAIN==errno)
continue;
RESTORE_CANCEL(cancel_state);
return(result);
}
bytes_written+=result;
}
RESTORE_CANCEL(cancel_state);
return(bytes_written);
#endif
}
#define WRITE(f,buf,len)GC_write(f,buf,len)
#endif
#define BUFSZ 1024
#if defined(DJGPP)||defined(__STRICT_ANSI__)
#define GC_VSNPRINTF(buf,bufsz,format,args)vsprintf(buf,format,args)
#elif defined(_MSC_VER)
#ifdef MSWINCE
#define GC_VSNPRINTF StringCchVPrintfA
#else
#define GC_VSNPRINTF _vsnprintf
#endif
#else
#define GC_VSNPRINTF vsnprintf
#endif
#define GC_PRINTF_FILLBUF(buf,format)do { va_list args;va_start(args,format);(buf)[sizeof(buf)- 1]=0x15;(void)GC_VSNPRINTF(buf,sizeof(buf)- 1,format,args);va_end(args);if ((buf)[sizeof(buf)- 1]!=0x15)ABORT("GC_printf clobbered stack");} while (0)
void GC_printf(const char*format,...)
{
if (!GC_quiet){
char buf[BUFSZ+1];
GC_PRINTF_FILLBUF(buf,format);
#ifdef NACL
(void)WRITE(GC_stdout,buf,strlen(buf));
#else
if (WRITE(GC_stdout,buf,strlen(buf))< 0
#if defined(CYGWIN32)||(defined(CONSOLE_LOG)&&defined(MSWIN32))
&&GC_stdout!=GC_DEFAULT_STDOUT_FD
#endif
){
ABORT("write to stdout failed");
}
#endif
}
}
void GC_err_printf(const char*format,...)
{
char buf[BUFSZ+1];
GC_PRINTF_FILLBUF(buf,format);
GC_err_puts(buf);
}
void GC_log_printf(const char*format,...)
{
char buf[BUFSZ+1];
GC_PRINTF_FILLBUF(buf,format);
#ifdef NACL
(void)WRITE(GC_log,buf,strlen(buf));
#else
if (WRITE(GC_log,buf,strlen(buf))< 0
#if defined(CYGWIN32)||(defined(CONSOLE_LOG)&&defined(MSWIN32))
&&GC_log!=GC_DEFAULT_STDERR_FD
#endif
){
ABORT("write to GC log failed");
}
#endif
}
#ifndef GC_ANDROID_LOG
#define GC_warn_printf GC_err_printf
#else
GC_INNER void GC_info_log_printf(const char*format,...)
{
char buf[BUFSZ+1];
GC_PRINTF_FILLBUF(buf,format);
(void)WRITE(ANDROID_LOG_INFO,buf,0);
}
GC_INNER void GC_verbose_log_printf(const char*format,...)
{
char buf[BUFSZ+1];
GC_PRINTF_FILLBUF(buf,format);
(void)WRITE(ANDROID_LOG_VERBOSE,buf,0);
}
STATIC void GC_warn_printf(const char*format,...)
{
char buf[BUFSZ+1];
GC_PRINTF_FILLBUF(buf,format);
(void)WRITE(ANDROID_LOG_WARN,buf,0);
}
#endif
void GC_err_puts(const char*s)
{
(void)WRITE(GC_stderr,s,strlen(s));
}
STATIC void GC_CALLBACK GC_default_warn_proc(char*msg,GC_word arg)
{
GC_warn_printf(msg,arg);
}
GC_INNER GC_warn_proc GC_current_warn_proc=GC_default_warn_proc;
GC_API void GC_CALLBACK GC_ignore_warn_proc(char*msg,GC_word arg)
{
if (GC_print_stats){
GC_default_warn_proc(msg,arg);
}
}
GC_API void GC_CALL GC_set_warn_proc(GC_warn_proc p)
{
DCL_LOCK_STATE;
GC_ASSERT(NONNULL_ARG_NOT_NULL(p));
#ifdef GC_WIN32_THREADS
#ifdef CYGWIN32
GC_ASSERT(GC_is_initialized);
#else
if (!GC_is_initialized)GC_init();
#endif
#endif
LOCK();
GC_current_warn_proc=p;
UNLOCK();
}
GC_API GC_warn_proc GC_CALL GC_get_warn_proc(void)
{
GC_warn_proc result;
DCL_LOCK_STATE;
LOCK();
result=GC_current_warn_proc;
UNLOCK();
return(result);
}
#if!defined(PCR)&&!defined(SMALL_CONFIG)
STATIC void GC_CALLBACK GC_default_on_abort(const char*msg)
{
#ifndef DONT_USE_ATEXIT
skip_gc_atexit=TRUE;
#endif
if (msg!=NULL){
#ifdef MSGBOX_ON_ERROR
GC_win32_MessageBoxA(msg,"Fatal error in GC",MB_ICONERROR|MB_OK);
#endif
#ifndef GC_ANDROID_LOG
#if defined(GC_WIN32_THREADS)&&defined(GC_ASSERTIONS)&&((defined(MSWIN32)&&!defined(CONSOLE_LOG))||defined(MSWINCE))
if (!GC_write_disabled)
#endif
{
if (WRITE(GC_stderr,msg,strlen(msg))>=0)
(void)WRITE(GC_stderr,"\n",1);
}
#else
__android_log_assert("*",GC_ANDROID_LOG_TAG,"%s\n",msg);
#endif
}
#if!defined(NO_DEBUGGING)&&!defined(GC_ANDROID_LOG)
if (GETENV("GC_LOOP_ON_ABORT")!=NULL){
for(;;){
}
}
#endif
}
GC_abort_func GC_on_abort=GC_default_on_abort;
GC_API void GC_CALL GC_set_abort_func(GC_abort_func fn)
{
DCL_LOCK_STATE;
GC_ASSERT(NONNULL_ARG_NOT_NULL(fn));
LOCK();
GC_on_abort=fn;
UNLOCK();
}
GC_API GC_abort_func GC_CALL GC_get_abort_func(void)
{
GC_abort_func fn;
DCL_LOCK_STATE;
LOCK();
fn=GC_on_abort;
UNLOCK();
return fn;
}
#endif
GC_API void GC_CALL GC_enable(void)
{
DCL_LOCK_STATE;
LOCK();
GC_ASSERT(GC_dont_gc!=0);
GC_dont_gc--;
UNLOCK();
}
GC_API void GC_CALL GC_disable(void)
{
DCL_LOCK_STATE;
LOCK();
GC_dont_gc++;
UNLOCK();
}
GC_API int GC_CALL GC_is_disabled(void)
{
return GC_dont_gc!=0;
}
GC_API void**GC_CALL GC_new_free_list_inner(void)
{
void*result;
GC_ASSERT(I_HOLD_LOCK());
result=GC_INTERNAL_MALLOC((MAXOBJGRANULES+1)*sizeof(ptr_t),PTRFREE);
if (NULL==result)ABORT("Failed to allocate freelist for new kind");
BZERO(result,(MAXOBJGRANULES+1)*sizeof(ptr_t));
return (void**)result;
}
GC_API void**GC_CALL GC_new_free_list(void)
{
void**result;
DCL_LOCK_STATE;
LOCK();
result=GC_new_free_list_inner();
UNLOCK();
return result;
}
GC_API unsigned GC_CALL GC_new_kind_inner(void**fl,GC_word descr,
int adjust,int clear)
{
unsigned result=GC_n_kinds;
GC_ASSERT(NONNULL_ARG_NOT_NULL(fl));
GC_ASSERT(adjust==FALSE||adjust==TRUE);
GC_ASSERT(clear==TRUE
||(descr==0&&adjust==FALSE&&clear==FALSE));
if (result < MAXOBJKINDS){
GC_ASSERT(result > 0);
GC_n_kinds++;
GC_obj_kinds[result].ok_freelist=fl;
GC_obj_kinds[result].ok_reclaim_list=0;
GC_obj_kinds[result].ok_descriptor=descr;
GC_obj_kinds[result].ok_relocate_descr=adjust;
GC_obj_kinds[result].ok_init=(GC_bool)clear;
#ifdef ENABLE_DISCLAIM
GC_obj_kinds[result].ok_mark_unconditionally=FALSE;
GC_obj_kinds[result].ok_disclaim_proc=0;
#endif
} else {
ABORT("Too many kinds");
}
return result;
}
GC_API unsigned GC_CALL GC_new_kind(void**fl,GC_word descr,int adjust,
int clear)
{
unsigned result;
DCL_LOCK_STATE;
LOCK();
result=GC_new_kind_inner(fl,descr,adjust,clear);
UNLOCK();
return result;
}
GC_API unsigned GC_CALL GC_new_proc_inner(GC_mark_proc proc)
{
unsigned result=GC_n_mark_procs;
if (result < MAX_MARK_PROCS){
GC_n_mark_procs++;
GC_mark_procs[result]=proc;
} else {
ABORT("Too many mark procedures");
}
return result;
}
GC_API unsigned GC_CALL GC_new_proc(GC_mark_proc proc)
{
unsigned result;
DCL_LOCK_STATE;
LOCK();
result=GC_new_proc_inner(proc);
UNLOCK();
return result;
}
GC_API void*GC_CALL GC_call_with_alloc_lock(GC_fn_type fn,void*client_data)
{
void*result;
DCL_LOCK_STATE;
#ifdef THREADS
LOCK();
#endif
result=(*fn)(client_data);
#ifdef THREADS
UNLOCK();
#endif
return(result);
}
GC_API void*GC_CALL GC_call_with_stack_base(GC_stack_base_func fn,void*arg)
{
struct GC_stack_base base;
void*result;
base.mem_base=(void*)&base;
#ifdef IA64
base.reg_base=(void*)GC_save_regs_in_stack();
#endif
result=fn(&base,arg);
GC_noop1(COVERT_DATAFLOW(&base));
return result;
}
#ifndef THREADS
GC_INNER ptr_t GC_blocked_sp=NULL;
#ifdef IA64
STATIC ptr_t GC_blocked_register_sp=NULL;
#endif
GC_INNER struct GC_traced_stack_sect_s*GC_traced_stack_sect=NULL;
GC_API void*GC_CALL GC_call_with_gc_active(GC_fn_type fn,
void*client_data)
{
struct GC_traced_stack_sect_s stacksect;
GC_ASSERT(GC_is_initialized);
if ((word)GC_stackbottom HOTTER_THAN (word)(&stacksect))
GC_stackbottom=(ptr_t)COVERT_DATAFLOW(&stacksect);
if (GC_blocked_sp==NULL){
client_data=fn(client_data);
GC_noop1(COVERT_DATAFLOW(&stacksect));
return client_data;
}
stacksect.saved_stack_ptr=GC_blocked_sp;
#ifdef IA64
stacksect.backing_store_end=GC_save_regs_in_stack();
stacksect.saved_backing_store_ptr=GC_blocked_register_sp;
#endif
stacksect.prev=GC_traced_stack_sect;
GC_blocked_sp=NULL;
GC_traced_stack_sect=&stacksect;
client_data=fn(client_data);
GC_ASSERT(GC_blocked_sp==NULL);
GC_ASSERT(GC_traced_stack_sect==&stacksect);
#if defined(CPPCHECK)
GC_noop1((word)GC_traced_stack_sect - (word)GC_blocked_sp);
#endif
GC_traced_stack_sect=stacksect.prev;
#ifdef IA64
GC_blocked_register_sp=stacksect.saved_backing_store_ptr;
#endif
GC_blocked_sp=stacksect.saved_stack_ptr;
return client_data;
}
STATIC void GC_do_blocking_inner(ptr_t data,void*context GC_ATTR_UNUSED)
{
struct blocking_data*d=(struct blocking_data*)data;
GC_ASSERT(GC_is_initialized);
GC_ASSERT(GC_blocked_sp==NULL);
#ifdef SPARC
GC_blocked_sp=GC_save_regs_in_stack();
#else
GC_blocked_sp=(ptr_t)&d;
#endif
#ifdef IA64
GC_blocked_register_sp=GC_save_regs_in_stack();
#endif
d->client_data=(d->fn)(d->client_data);
#ifdef SPARC
GC_ASSERT(GC_blocked_sp!=NULL);
#else
GC_ASSERT(GC_blocked_sp==(ptr_t)(&d));
#endif
#if defined(CPPCHECK)
GC_noop1((word)GC_blocked_sp);
#endif
GC_blocked_sp=NULL;
}
GC_API void GC_CALL GC_set_stackbottom(void*gc_thread_handle,
const struct GC_stack_base*sb)
{
GC_ASSERT(sb->mem_base!=NULL);
GC_ASSERT(NULL==gc_thread_handle||&GC_stackbottom==gc_thread_handle);
GC_ASSERT(NULL==GC_blocked_sp
&&NULL==GC_traced_stack_sect);
(void)gc_thread_handle;
GC_stackbottom=(char*)sb->mem_base;
#ifdef IA64
GC_register_stackbottom=(ptr_t)sb->reg_base;
#endif
}
GC_API void*GC_CALL GC_get_my_stackbottom(struct GC_stack_base*sb)
{
GC_ASSERT(GC_is_initialized);
sb->mem_base=GC_stackbottom;
#ifdef IA64
sb->reg_base=GC_register_stackbottom;
#endif
return&GC_stackbottom;
}
#endif
GC_API void*GC_CALL GC_do_blocking(GC_fn_type fn,void*client_data)
{
struct blocking_data my_data;
my_data.fn=fn;
my_data.client_data=client_data;
GC_with_callee_saves_pushed(GC_do_blocking_inner,(ptr_t)(&my_data));
return my_data.client_data;
}
#if!defined(NO_DEBUGGING)
GC_API void GC_CALL GC_dump(void)
{
DCL_LOCK_STATE;
LOCK();
GC_dump_named(NULL);
UNLOCK();
}
GC_API void GC_CALL GC_dump_named(const char*name)
{
#ifndef NO_CLOCK
CLOCK_TYPE current_time;
GET_TIME(current_time);
#endif
if (name!=NULL){
GC_printf("***GC Dump %s\n",name);
} else {
GC_printf("***GC Dump collection #%lu\n",(unsigned long)GC_gc_no);
}
#ifndef NO_CLOCK
GC_printf("Time since GC init:%lu ms\n",
MS_TIME_DIFF(current_time,GC_init_time));
#endif
GC_printf("\n***Static roots:\n");
GC_print_static_roots();
GC_printf("\n***Heap sections:\n");
GC_print_heap_sects();
GC_printf("\n***Free blocks:\n");
GC_print_hblkfreelist();
GC_printf("\n***Blocks in use:\n");
GC_print_block_list();
}
#endif
static void block_add_size(struct hblk*h,word pbytes)
{
hdr*hhdr=HDR(h);
*(word*)pbytes+=(WORDS_TO_BYTES(hhdr->hb_sz)+(HBLKSIZE - 1))
&~(word)(HBLKSIZE - 1);
}
GC_API size_t GC_CALL GC_get_memory_use(void)
{
word bytes=0;
DCL_LOCK_STATE;
LOCK();
GC_apply_to_all_blocks(block_add_size,(word)(&bytes));
UNLOCK();
return (size_t)bytes;
}
GC_API GC_word GC_CALL GC_get_gc_no(void)
{
return GC_gc_no;
}
#ifdef THREADS
GC_API int GC_CALL GC_get_parallel(void)
{
return GC_parallel;
}
GC_API void GC_CALL GC_alloc_lock(void)
{
DCL_LOCK_STATE;
LOCK();
}
GC_API void GC_CALL GC_alloc_unlock(void)
{
UNLOCK();
}
GC_INNER GC_on_thread_event_proc GC_on_thread_event=0;
GC_API void GC_CALL GC_set_on_thread_event(GC_on_thread_event_proc fn)
{
DCL_LOCK_STATE;
LOCK();
GC_on_thread_event=fn;
UNLOCK();
}
GC_API GC_on_thread_event_proc GC_CALL GC_get_on_thread_event(void)
{
GC_on_thread_event_proc fn;
DCL_LOCK_STATE;
LOCK();
fn=GC_on_thread_event;
UNLOCK();
return fn;
}
#endif
GC_API void GC_CALL GC_set_oom_fn(GC_oom_func fn)
{
GC_ASSERT(NONNULL_ARG_NOT_NULL(fn));
DCL_LOCK_STATE;
LOCK();
GC_oom_fn=fn;
UNLOCK();
}
GC_API GC_oom_func GC_CALL GC_get_oom_fn(void)
{
GC_oom_func fn;
DCL_LOCK_STATE;
LOCK();
fn=GC_oom_fn;
UNLOCK();
return fn;
}
GC_API void GC_CALL GC_set_on_heap_resize(GC_on_heap_resize_proc fn)
{
DCL_LOCK_STATE;
LOCK();
GC_on_heap_resize=fn;
UNLOCK();
}
GC_API GC_on_heap_resize_proc GC_CALL GC_get_on_heap_resize(void)
{
GC_on_heap_resize_proc fn;
DCL_LOCK_STATE;
LOCK();
fn=GC_on_heap_resize;
UNLOCK();
return fn;
}
GC_API void GC_CALL GC_set_finalizer_notifier(GC_finalizer_notifier_proc fn)
{
DCL_LOCK_STATE;
LOCK();
GC_finalizer_notifier=fn;
UNLOCK();
}
GC_API GC_finalizer_notifier_proc GC_CALL GC_get_finalizer_notifier(void)
{
GC_finalizer_notifier_proc fn;
DCL_LOCK_STATE;
LOCK();
fn=GC_finalizer_notifier;
UNLOCK();
return fn;
}
GC_API void GC_CALL GC_set_find_leak(int value)
{
GC_find_leak=value;
}
GC_API int GC_CALL GC_get_find_leak(void)
{
return GC_find_leak;
}
GC_API void GC_CALL GC_set_all_interior_pointers(int value)
{
DCL_LOCK_STATE;
GC_all_interior_pointers=value?1:0;
if (GC_is_initialized){
LOCK();
GC_initialize_offsets();
if (!GC_all_interior_pointers)
GC_bl_init_no_interiors();
UNLOCK();
}
}
GC_API int GC_CALL GC_get_all_interior_pointers(void)
{
return GC_all_interior_pointers;
}
GC_API void GC_CALL GC_set_finalize_on_demand(int value)
{
GC_ASSERT(value!=-1);
GC_finalize_on_demand=value;
}
GC_API int GC_CALL GC_get_finalize_on_demand(void)
{
return GC_finalize_on_demand;
}
GC_API void GC_CALL GC_set_java_finalization(int value)
{
GC_ASSERT(value!=-1);
GC_java_finalization=value;
}
GC_API int GC_CALL GC_get_java_finalization(void)
{
return GC_java_finalization;
}
GC_API void GC_CALL GC_set_dont_expand(int value)
{
GC_ASSERT(value!=-1);
GC_dont_expand=value;
}
GC_API int GC_CALL GC_get_dont_expand(void)
{
return GC_dont_expand;
}
GC_API void GC_CALL GC_set_no_dls(int value)
{
GC_ASSERT(value!=-1);
GC_no_dls=value;
}
GC_API int GC_CALL GC_get_no_dls(void)
{
return GC_no_dls;
}
GC_API void GC_CALL GC_set_non_gc_bytes(GC_word value)
{
GC_non_gc_bytes=value;
}
GC_API GC_word GC_CALL GC_get_non_gc_bytes(void)
{
return GC_non_gc_bytes;
}
GC_API void GC_CALL GC_set_free_space_divisor(GC_word value)
{
GC_ASSERT(value > 0);
GC_free_space_divisor=value;
}
GC_API GC_word GC_CALL GC_get_free_space_divisor(void)
{
return GC_free_space_divisor;
}
GC_API void GC_CALL GC_set_max_retries(GC_word value)
{
GC_ASSERT((GC_signed_word)value!=-1);
GC_max_retries=value;
}
GC_API GC_word GC_CALL GC_get_max_retries(void)
{
return GC_max_retries;
}
GC_API void GC_CALL GC_set_dont_precollect(int value)
{
GC_ASSERT(value!=-1);
GC_dont_precollect=value;
}
GC_API int GC_CALL GC_get_dont_precollect(void)
{
return GC_dont_precollect;
}
GC_API void GC_CALL GC_set_full_freq(int value)
{
GC_ASSERT(value>=0);
GC_full_freq=value;
}
GC_API int GC_CALL GC_get_full_freq(void)
{
return GC_full_freq;
}
GC_API void GC_CALL GC_set_time_limit(unsigned long value)
{
GC_ASSERT((long)value!=-1L);
GC_time_limit=value;
}
GC_API unsigned long GC_CALL GC_get_time_limit(void)
{
return GC_time_limit;
}
GC_API void GC_CALL GC_set_force_unmap_on_gcollect(int value)
{
GC_force_unmap_on_gcollect=(GC_bool)value;
}
GC_API int GC_CALL GC_get_force_unmap_on_gcollect(void)
{
return (int)GC_force_unmap_on_gcollect;
}
GC_API void GC_CALL GC_abort_on_oom(void)
{
GC_err_printf("Insufficient memory for the allocation\n");
EXIT();
}
#ifdef THREADS
GC_API void GC_CALL GC_stop_world_external(void)
{
GC_ASSERT(GC_is_initialized);
LOCK();
#ifdef THREAD_LOCAL_ALLOC
GC_ASSERT(!GC_world_stopped);
#endif
STOP_WORLD();
#ifdef THREAD_LOCAL_ALLOC
GC_world_stopped=TRUE;
#endif
}
GC_API void GC_CALL GC_start_world_external(void)
{
#ifdef THREAD_LOCAL_ALLOC
GC_ASSERT(GC_world_stopped);
GC_world_stopped=FALSE;
#else
GC_ASSERT(GC_is_initialized);
#endif
START_WORLD();
UNLOCK();
}
#endif
#if!defined(OS2)&&!defined(PCR)&&!defined(AMIGA)&&!defined(MACOS)&&!defined(MSWINCE)&&!defined(SN_TARGET_ORBIS)&&!defined(SN_TARGET_PSP2)&&!defined(__CC_ARM)
#include <sys/types.h>
#if!defined(MSWIN32)&&!defined(MSWIN_XBOX1)
#include <unistd.h>
#endif
#endif
#include <stdio.h>
#if defined(MSWINCE)||defined(SN_TARGET_PS3)
#define SIGSEGV 0
#else
#include <signal.h>
#endif
#if defined(UNIX_LIKE)||defined(CYGWIN32)||defined(NACL)||defined(SYMBIAN)
#include <fcntl.h>
#endif
#if defined(LINUX)||defined(LINUX_STACKBOTTOM)
#include <ctype.h>
#endif
#ifdef AMIGA
#define GC_AMIGA_DEF
#if!defined(GC_AMIGA_DEF)&&!defined(GC_AMIGA_SB)&&!defined(GC_AMIGA_DS)&&!defined(GC_AMIGA_AM)
#include <stdio.h>
#include <signal.h>
#define GC_AMIGA_DEF
#define GC_AMIGA_SB
#define GC_AMIGA_DS
#define GC_AMIGA_AM
#endif
#ifdef GC_AMIGA_DEF
#ifndef __GNUC__
#include <exec/exec.h>
#endif
#include <proto/exec.h>
#include <proto/dos.h>
#include <dos/dosextens.h>
#include <workbench/startup.h>
#endif
#ifdef GC_AMIGA_SB
ptr_t GC_get_main_stack_base(void)
{
struct Process*proc=(struct Process*)SysBase->ThisTask;
if (proc->pr_Task.tc_Node.ln_Type==NT_PROCESS
&&proc->pr_CLI!=NULL){
return (char*)proc->pr_ReturnAddr+sizeof(ULONG);
} else {
return (char*)proc->pr_Task.tc_SPUpper;
}
}
#endif
#ifdef GC_AMIGA_DS
void GC_register_data_segments(void)
{
struct Process*proc;
struct CommandLineInterface*cli;
BPTR myseglist;
ULONG*data;
#ifdef __GNUC__
ULONG dataSegSize;
GC_bool found_segment=FALSE;
extern char __data_size[];
dataSegSize=__data_size+8;
#endif
proc=(struct Process*)SysBase->ThisTask;
myseglist=proc->pr_SegList;
if (proc->pr_Task.tc_Node.ln_Type==NT_PROCESS){
if (proc->pr_CLI!=NULL){
cli=BADDR(proc->pr_CLI);
myseglist=cli->cli_Module;
}
} else {
ABORT("Not a Process.");
}
if (myseglist==NULL){
ABORT("Arrrgh.. can't find segments,aborting");
}
for (data=(ULONG*)BADDR(myseglist);data!=NULL;
data=(ULONG*)BADDR(data[0])){
if ((ULONG)GC_register_data_segments < (ULONG)(&data[1])
||(ULONG)GC_register_data_segments > (ULONG)(&data[1])
+data[-1]){
#ifdef __GNUC__
if (dataSegSize==data[-1]){
found_segment=TRUE;
}
#endif
GC_add_roots_inner((char*)&data[1],
((char*)&data[1])+data[-1],FALSE);
}
}
#ifdef __GNUC__
if (!found_segment){
ABORT("Can`t find correct Segments.\nSolution:Use an newer version of ixemul.library");
}
#endif
}
#endif
#ifdef GC_AMIGA_AM
#ifndef GC_AMIGA_FASTALLOC
void*GC_amiga_allocwrapper(size_t size,void*(*AllocFunction)(size_t size2)){
return (*AllocFunction)(size);
}
void*(*GC_amiga_allocwrapper_do)(size_t size,void*(*AllocFunction)(size_t size2))
=GC_amiga_allocwrapper;
#else
void*GC_amiga_allocwrapper_firsttime(size_t size,void*(*AllocFunction)(size_t size2));
void*(*GC_amiga_allocwrapper_do)(size_t size,void*(*AllocFunction)(size_t size2))
=GC_amiga_allocwrapper_firsttime;
struct GC_Amiga_AllocedMemoryHeader{
ULONG size;
struct GC_Amiga_AllocedMemoryHeader*next;
};
struct GC_Amiga_AllocedMemoryHeader*GC_AMIGAMEM=(struct GC_Amiga_AllocedMemoryHeader*)(int)~(NULL);
ULONG GC_AMIGA_MEMF=MEMF_FAST|MEMF_CLEAR;
#ifndef GC_AMIGA_ONLYFAST
BOOL GC_amiga_dontalloc=FALSE;
#endif
#ifdef GC_AMIGA_PRINTSTATS
int succ=0,succ2=0;
int nsucc=0,nsucc2=0;
int nullretries=0;
int numcollects=0;
int chipa=0;
int allochip=0;
int allocfast=0;
int cur0=0;
int cur1=0;
int cur10=0;
int cur50=0;
int cur150=0;
int cur151=0;
int ncur0=0;
int ncur1=0;
int ncur10=0;
int ncur50=0;
int ncur150=0;
int ncur151=0;
#endif
void GC_amiga_free_all_mem(void){
struct GC_Amiga_AllocedMemoryHeader*gc_am=(struct GC_Amiga_AllocedMemoryHeader*)(~(int)(GC_AMIGAMEM));
#ifdef GC_AMIGA_PRINTSTATS
printf("\n\n"
"%d bytes of chip-mem,and %d bytes of fast-mem where allocated from the OS.\n",
allochip,allocfast
);
printf(
"%d bytes of chip-mem were returned from the GC_AMIGA_FASTALLOC supported allocating functions.\n",
chipa
);
printf("\n");
printf("GC_gcollect was called %d times to avoid returning NULL or start allocating with the MEMF_ANY flag.\n",numcollects);
printf("%d of them was a success. (the others had to use allocation from the OS.)\n",nullretries);
printf("\n");
printf("Succeeded forcing %d gc-allocations (%d bytes)of chip-mem to be fast-mem.\n",succ,succ2);
printf("Failed forcing %d gc-allocations (%d bytes)of chip-mem to be fast-mem.\n",nsucc,nsucc2);
printf("\n");
printf(
"Number of retries before succeeding a chip->fast force:\n"
"0:%d,1:%d,2-9:%d,10-49:%d,50-149:%d,>150:%d\n",
cur0,cur1,cur10,cur50,cur150,cur151
);
printf(
"Number of retries before giving up a chip->fast force:\n"
"0:%d,1:%d,2-9:%d,10-49:%d,50-149:%d,>150:%d\n",
ncur0,ncur1,ncur10,ncur50,ncur150,ncur151
);
#endif
while(gc_am!=NULL){
struct GC_Amiga_AllocedMemoryHeader*temp=gc_am->next;
FreeMem(gc_am,gc_am->size);
gc_am=(struct GC_Amiga_AllocedMemoryHeader*)(~(int)(temp));
}
}
#ifndef GC_AMIGA_ONLYFAST
char*chipmax;
size_t latestsize;
#endif
#ifdef GC_AMIGA_FASTALLOC
void*GC_amiga_get_mem(size_t size){
struct GC_Amiga_AllocedMemoryHeader*gc_am;
#ifndef GC_AMIGA_ONLYFAST
if(GC_amiga_dontalloc==TRUE){
return NULL;
}
if(GC_AMIGA_MEMF==(MEMF_ANY|MEMF_CLEAR)&&size>100000&&latestsize<50000)return NULL;
#endif
gc_am=AllocMem((ULONG)(size+sizeof(struct GC_Amiga_AllocedMemoryHeader)),GC_AMIGA_MEMF);
if(gc_am==NULL)return NULL;
gc_am->next=GC_AMIGAMEM;
gc_am->size=size+sizeof(struct GC_Amiga_AllocedMemoryHeader);
GC_AMIGAMEM=(struct GC_Amiga_AllocedMemoryHeader*)(~(int)(gc_am));
#ifdef GC_AMIGA_PRINTSTATS
if((char*)gc_am<chipmax){
allochip+=size;
}else{
allocfast+=size;
}
#endif
return gc_am+1;
}
#endif
#ifndef GC_AMIGA_ONLYFAST
#ifdef GC_AMIGA_RETRY
void*GC_amiga_rec_alloc(size_t size,void*(*AllocFunction)(size_t size2),const int rec){
void*ret;
ret=(*AllocFunction)(size);
#ifdef GC_AMIGA_PRINTSTATS
if((char*)ret>chipmax||ret==NULL){
if(ret==NULL){
nsucc++;
nsucc2+=size;
if(rec==0)ncur0++;
if(rec==1)ncur1++;
if(rec>1&&rec<10)ncur10++;
if(rec>=10&&rec<50)ncur50++;
if(rec>=50&&rec<150)ncur150++;
if(rec>=150)ncur151++;
}else{
succ++;
succ2+=size;
if(rec==0)cur0++;
if(rec==1)cur1++;
if(rec>1&&rec<10)cur10++;
if(rec>=10&&rec<50)cur50++;
if(rec>=50&&rec<150)cur150++;
if(rec>=150)cur151++;
}
}
#endif
if (((char*)ret)<=chipmax&&ret!=NULL&&(rec<(size>500000?9:size/5000))){
ret=GC_amiga_rec_alloc(size,AllocFunction,rec+1);
}
return ret;
}
#endif
void*GC_amiga_allocwrapper_any(size_t size,void*(*AllocFunction)(size_t size2)){
void*ret;
GC_amiga_dontalloc=TRUE;
latestsize=size;
ret=(*AllocFunction)(size);
if(((char*)ret)<=chipmax){
if(ret==NULL){
#ifdef GC_AMIGA_GC
if(!GC_dont_gc){
GC_gcollect();
#ifdef GC_AMIGA_PRINTSTATS
numcollects++;
#endif
ret=(*AllocFunction)(size);
}
if(ret==NULL)
#endif
{
GC_amiga_dontalloc=FALSE;
ret=(*AllocFunction)(size);
if(ret==NULL){
WARN("Out of Memory!Returning NIL!\n",0);
}
}
#ifdef GC_AMIGA_PRINTSTATS
else{
nullretries++;
}
if(ret!=NULL&&(char*)ret<=chipmax)chipa+=size;
#endif
}
#ifdef GC_AMIGA_RETRY
else{
void*ret2;
if(
AllocFunction!=GC_malloc_uncollectable
#ifdef GC_ATOMIC_UNCOLLECTABLE
&&AllocFunction!=GC_malloc_atomic_uncollectable
#endif
){
ret2=GC_amiga_rec_alloc(size,AllocFunction,0);
}else{
ret2=(*AllocFunction)(size);
#ifdef GC_AMIGA_PRINTSTATS
if((char*)ret2<chipmax||ret2==NULL){
nsucc++;
nsucc2+=size;
ncur0++;
}else{
succ++;
succ2+=size;
cur0++;
}
#endif
}
if(((char*)ret2)>chipmax){
GC_free(ret);
ret=ret2;
}else{
GC_free(ret2);
}
}
#endif
}
#if defined(CPPCHECK)
if (GC_amiga_dontalloc)
#endif
GC_amiga_dontalloc=FALSE;
return ret;
}
void (*GC_amiga_toany)(void)=NULL;
void GC_amiga_set_toany(void (*func)(void)){
GC_amiga_toany=func;
}
#endif
void*GC_amiga_allocwrapper_fast(size_t size,void*(*AllocFunction)(size_t size2)){
void*ret;
ret=(*AllocFunction)(size);
if(ret==NULL){
#ifdef GC_AMIGA_GC
if(!GC_dont_gc){
GC_gcollect();
#ifdef GC_AMIGA_PRINTSTATS
numcollects++;
#endif
ret=(*AllocFunction)(size);
}
if(ret==NULL)
#endif
{
#ifndef GC_AMIGA_ONLYFAST
GC_AMIGA_MEMF=MEMF_ANY|MEMF_CLEAR;
if(GC_amiga_toany!=NULL)(*GC_amiga_toany)();
GC_amiga_allocwrapper_do=GC_amiga_allocwrapper_any;
return GC_amiga_allocwrapper_any(size,AllocFunction);
#endif
}
#ifdef GC_AMIGA_PRINTSTATS
else{
nullretries++;
}
#endif
}
return ret;
}
void*GC_amiga_allocwrapper_firsttime(size_t size,void*(*AllocFunction)(size_t size2)){
atexit(&GC_amiga_free_all_mem);
chipmax=(char*)SysBase->MaxLocMem;
GC_amiga_allocwrapper_do=GC_amiga_allocwrapper_fast;
return GC_amiga_allocwrapper_fast(size,AllocFunction);
}
#endif
void*GC_amiga_realloc(void*old_object,size_t new_size_in_bytes){
#ifndef GC_AMIGA_FASTALLOC
return GC_realloc(old_object,new_size_in_bytes);
#else
void*ret;
latestsize=new_size_in_bytes;
ret=GC_realloc(old_object,new_size_in_bytes);
if(ret==NULL&&new_size_in_bytes!=0
&&GC_AMIGA_MEMF==(MEMF_FAST|MEMF_CLEAR)){
#ifdef GC_AMIGA_GC
if(!GC_dont_gc){
GC_gcollect();
#ifdef GC_AMIGA_PRINTSTATS
numcollects++;
#endif
ret=GC_realloc(old_object,new_size_in_bytes);
}
if(ret==NULL)
#endif
{
#ifndef GC_AMIGA_ONLYFAST
GC_AMIGA_MEMF=MEMF_ANY|MEMF_CLEAR;
if(GC_amiga_toany!=NULL)(*GC_amiga_toany)();
GC_amiga_allocwrapper_do=GC_amiga_allocwrapper_any;
ret=GC_realloc(old_object,new_size_in_bytes);
#endif
}
#ifdef GC_AMIGA_PRINTSTATS
else{
nullretries++;
}
#endif
}
if(ret==NULL&&new_size_in_bytes!=0){
WARN("Out of Memory!Returning NIL!\n",0);
}
#ifdef GC_AMIGA_PRINTSTATS
if(((char*)ret)<chipmax&&ret!=NULL){
chipa+=new_size_in_bytes;
}
#endif
return ret;
#endif
}
#endif
#undef GC_AMIGA_DEF
#endif
#ifdef MACOS
#include <Processes.h>
#endif
#ifdef IRIX5
#include <sys/uio.h>
#include <malloc.h>
#endif
#if defined(MMAP_SUPPORTED)||defined(ADD_HEAP_GUARD_PAGES)
#if defined(USE_MUNMAP)&&!defined(USE_MMAP)&&!defined(CPPCHECK)
#error Invalid config:USE_MUNMAP requires USE_MMAP
#endif
#include <sys/types.h>
#include <sys/mman.h>
#include <sys/stat.h>
#include <errno.h>
#endif
#ifdef DARWIN
#include <mach-o/getsect.h>
#endif
#ifdef DJGPP
typedef long unsigned int caddr_t;
#endif
#ifdef PCR
#include "mm/PCR_MM.h"
#endif
#if defined(GC_DARWIN_THREADS)&&defined(MPROTECT_VDB)
#ifndef GC_DARWIN_STOP_WORLD_H
#define GC_DARWIN_STOP_WORLD_H
#if!defined(GC_DARWIN_THREADS)
#error darwin_stop_world.h included without GC_DARWIN_THREADS defined
#endif
#include <mach/mach.h>
#include <mach/thread_act.h>
EXTERN_C_BEGIN
struct thread_stop_info {
mach_port_t mach_thread;
ptr_t stack_ptr;
};
#ifndef DARWIN_DONT_PARSE_STACK
GC_INNER ptr_t GC_FindTopOfStack(unsigned long);
#endif
#ifdef MPROTECT_VDB
GC_INNER void GC_mprotect_stop(void);
GC_INNER void GC_mprotect_resume(void);
#ifndef GC_NO_THREADS_DISCOVERY
GC_INNER void GC_darwin_register_mach_handler_thread(mach_port_t thread);
#endif
#endif
#if defined(PARALLEL_MARK)&&!defined(GC_NO_THREADS_DISCOVERY)
GC_INNER GC_bool GC_is_mach_marker(thread_act_t);
#endif
EXTERN_C_END
#endif
#endif
#if!defined(NO_EXECUTE_PERMISSION)
STATIC GC_bool GC_pages_executable=TRUE;
#else
STATIC GC_bool GC_pages_executable=FALSE;
#endif
#define IGNORE_PAGES_EXECUTABLE 1
#ifdef NEED_PROC_MAPS
STATIC ssize_t GC_repeat_read(int fd,char*buf,size_t count)
{
#define READ read
size_t num_read=0;
ASSERT_CANCEL_DISABLED();
while (num_read < count){
ssize_t result=READ(fd,buf+num_read,count - num_read);
if (result < 0)return result;
if (result==0)break;
num_read+=result;
}
#undef READ
return num_read;
}
#ifdef THREADS
STATIC size_t GC_get_file_len(int f)
{
size_t total=0;
ssize_t result;
#define GET_FILE_LEN_BUF_SZ 500
char buf[GET_FILE_LEN_BUF_SZ];
do {
result=read(f,buf,GET_FILE_LEN_BUF_SZ);
if (result==-1)return 0;
total+=result;
} while (result > 0);
return total;
}
STATIC size_t GC_get_maps_len(void)
{
int f=open("/proc/self/maps",O_RDONLY);
size_t result;
if (f < 0)return 0;
result=GC_get_file_len(f);
close(f);
return result;
}
#endif
GC_INNER char*GC_get_maps(void)
{
ssize_t result;
static char*maps_buf=NULL;
static size_t maps_buf_sz=1;
size_t maps_size;
#ifdef THREADS
size_t old_maps_size=0;
#endif
GC_ASSERT(I_HOLD_LOCK());
#ifdef THREADS
maps_size=GC_get_maps_len();
if (0==maps_size)return 0;
#else
maps_size=4000;
#endif
do {
int f;
while (maps_size>=maps_buf_sz){
#ifdef LINT2
GC_noop1((word)maps_buf);
#else
GC_scratch_recycle_no_gww(maps_buf,maps_buf_sz);
#endif
while (maps_size>=maps_buf_sz)maps_buf_sz*=2;
maps_buf=GC_scratch_alloc(maps_buf_sz);
#ifdef THREADS
maps_size=GC_get_maps_len();
if (0==maps_size)return 0;
#endif
if (maps_buf==0)return 0;
}
GC_ASSERT(maps_buf_sz>=maps_size+1);
f=open("/proc/self/maps",O_RDONLY);
if (-1==f)return 0;
#ifdef THREADS
old_maps_size=maps_size;
#endif
maps_size=0;
do {
result=GC_repeat_read(f,maps_buf,maps_buf_sz-1);
if (result<=0){
close(f);
return 0;
}
maps_size+=result;
} while ((size_t)result==maps_buf_sz-1);
close(f);
#ifdef THREADS
if (maps_size > old_maps_size){
WARN("Unexpected asynchronous/proc/self/maps growth"
" (to %" WARN_PRIdPTR " bytes)\n",maps_size);
}
#endif
} while (maps_size>=maps_buf_sz
#ifdef THREADS
||maps_size < old_maps_size
#endif
);
maps_buf[maps_size]='\0';
return maps_buf;
}
#if (defined(DYNAMIC_LOADING)&&defined(USE_PROC_FOR_LIBRARIES))||defined(IA64)||defined(INCLUDE_LINUX_THREAD_DESCR)||defined(REDIRECT_MALLOC)
GC_INNER char*GC_parse_map_entry(char*buf_ptr,ptr_t*start,ptr_t*end,
char**prot,unsigned int*maj_dev,
char**mapping_name)
{
unsigned char*start_start,*end_start,*maj_dev_start;
unsigned char*p;
if (buf_ptr==NULL||*buf_ptr=='\0'){
return NULL;
}
p=(unsigned char*)buf_ptr;
while (isspace(*p))++p;
start_start=p;
GC_ASSERT(isxdigit(*start_start));
*start=(ptr_t)strtoul((char*)start_start,(char**)&p,16);
GC_ASSERT(*p=='-');
++p;
end_start=p;
GC_ASSERT(isxdigit(*end_start));
*end=(ptr_t)strtoul((char*)end_start,(char**)&p,16);
GC_ASSERT(isspace(*p));
while (isspace(*p))++p;
GC_ASSERT(*p=='r'||*p=='-');
*prot=(char*)p;
while (!isspace(*p))++p;
while (isspace(*p))p++;
GC_ASSERT(isxdigit(*p));
while (!isspace(*p))++p;
while (isspace(*p))p++;
maj_dev_start=p;
GC_ASSERT(isxdigit(*maj_dev_start));
*maj_dev=strtoul((char*)maj_dev_start,NULL,16);
if (mapping_name==0){
while (*p&&*p++!='\n');
} else {
while (*p&&*p!='\n'&&*p!='/'&&*p!='[')p++;
*mapping_name=(char*)p;
while (*p&&*p++!='\n');
}
return (char*)p;
}
#endif
#if defined(IA64)||defined(INCLUDE_LINUX_THREAD_DESCR)
GC_INNER GC_bool GC_enclosing_mapping(ptr_t addr,ptr_t*startp,
ptr_t*endp)
{
char*prot;
ptr_t my_start,my_end;
unsigned int maj_dev;
char*maps=GC_get_maps();
char*buf_ptr=maps;
if (0==maps)return(FALSE);
for (;;){
buf_ptr=GC_parse_map_entry(buf_ptr,&my_start,&my_end,
&prot,&maj_dev,0);
if (buf_ptr==NULL)return FALSE;
if (prot[1]=='w'&&maj_dev==0){
if ((word)my_end > (word)addr&&(word)my_start<=(word)addr){
*startp=my_start;
*endp=my_end;
return TRUE;
}
}
}
return FALSE;
}
#endif
#if defined(REDIRECT_MALLOC)
GC_INNER GC_bool GC_text_mapping(char*nm,ptr_t*startp,ptr_t*endp)
{
size_t nm_len=strlen(nm);
char*prot;
char*map_path;
ptr_t my_start,my_end;
unsigned int maj_dev;
char*maps=GC_get_maps();
char*buf_ptr=maps;
if (0==maps)return(FALSE);
for (;;){
buf_ptr=GC_parse_map_entry(buf_ptr,&my_start,&my_end,
&prot,&maj_dev,&map_path);
if (buf_ptr==NULL)return FALSE;
if (prot[0]=='r'&&prot[1]=='-'&&prot[2]=='x'){
char*p=map_path;
while (*p!='\0'&&*p!='\n'&&*p!=' '&&*p!='\t')++p;
while (*p!='/'&&(word)p>=(word)map_path)--p;
++p;
if (strncmp(nm,p,nm_len)==0){
*startp=my_start;
*endp=my_end;
return TRUE;
}
}
}
return FALSE;
}
#endif
#ifdef IA64
static ptr_t backing_store_base_from_proc(void)
{
ptr_t my_start,my_end;
if (!GC_enclosing_mapping(GC_save_regs_in_stack(),&my_start,&my_end)){
GC_COND_LOG_PRINTF("Failed to find backing store base from/proc\n");
return 0;
}
return my_start;
}
#endif
#endif
#if defined(SEARCH_FOR_DATA_START)
#if defined(LINUX)||defined(HURD)
EXTERN_C_BEGIN
#pragma weak __data_start
#pragma weak data_start
extern int __data_start[],data_start[];
EXTERN_C_END
#endif
ptr_t GC_data_start=NULL;
GC_INNER void GC_init_linux_data_start(void)
{
ptr_t data_end=DATAEND;
#if (defined(LINUX)||defined(HURD))&&defined(USE_PROG_DATA_START)
if (COVERT_DATAFLOW(__data_start)!=0){
GC_data_start=(ptr_t)(__data_start);
} else {
GC_data_start=(ptr_t)(data_start);
}
if (COVERT_DATAFLOW(GC_data_start)!=0){
if ((word)GC_data_start > (word)data_end)
ABORT_ARG2("Wrong __data_start/_end pair",
":%p .. %p",(void*)GC_data_start,(void*)data_end);
return;
}
#ifdef DEBUG_ADD_DEL_ROOTS
GC_log_printf("__data_start not provided\n");
#endif
#endif
if (GC_no_dls){
GC_data_start=data_end;
return;
}
GC_data_start=(ptr_t)GC_find_limit(data_end,FALSE);
}
#endif
#ifdef ECOS
#ifndef ECOS_GC_MEMORY_SIZE
#define ECOS_GC_MEMORY_SIZE (448*1024)
#endif
static char ecos_gc_memory[ECOS_GC_MEMORY_SIZE];
static char*ecos_gc_brk=ecos_gc_memory;
static void*tiny_sbrk(ptrdiff_t increment)
{
void*p=ecos_gc_brk;
ecos_gc_brk+=increment;
if ((word)ecos_gc_brk > (word)(ecos_gc_memory+sizeof(ecos_gc_memory))){
ecos_gc_brk-=increment;
return NULL;
}
return p;
}
#define sbrk tiny_sbrk
#endif
#if defined(NETBSD)&&defined(__ELF__)
ptr_t GC_data_start=NULL;
EXTERN_C_BEGIN
extern char**environ;
EXTERN_C_END
GC_INNER void GC_init_netbsd_elf(void)
{
GC_data_start=(ptr_t)GC_find_limit(&environ,FALSE);
}
#endif
#if defined(ADDRESS_SANITIZER)&&(defined(UNIX_LIKE)||defined(NEED_FIND_LIMIT)||defined(MPROTECT_VDB))&&!defined(CUSTOM_ASAN_DEF_OPTIONS)
GC_API const char*__asan_default_options(void)
{
return "allow_user_segv_handler=1";
}
#endif
#ifdef OPENBSD
static struct sigaction old_segv_act;
STATIC JMP_BUF GC_jmp_buf_openbsd;
STATIC void GC_fault_handler_openbsd(int sig GC_ATTR_UNUSED)
{
LONGJMP(GC_jmp_buf_openbsd,1);
}
#ifdef GC_OPENBSD_UTHREADS
#include <sys/syscall.h>
EXTERN_C_BEGIN
extern sigset_t __syscall(quad_t,...);
EXTERN_C_END
STATIC ptr_t GC_find_limit_openbsd(ptr_t p,ptr_t bound)
{
static volatile ptr_t result;
struct sigaction act;
word pgsz=(word)sysconf(_SC_PAGESIZE);
GC_ASSERT((word)bound>=pgsz);
GC_ASSERT(I_HOLD_LOCK());
act.sa_handler=GC_fault_handler_openbsd;
sigemptyset(&act.sa_mask);
act.sa_flags=SA_NODEFER|SA_RESTART;
sigaction(SIGSEGV,&act,&old_segv_act);
if (SETJMP(GC_jmp_buf_openbsd)==0){
result=(ptr_t)((word)p&~(pgsz-1));
for (;;){
if ((word)result>=(word)bound - pgsz){
result=bound;
break;
}
result+=pgsz;
GC_noop1((word)(*result));
}
}
#ifdef THREADS
__syscall(SYS_sigprocmask,SIG_UNBLOCK,sigmask(SIGPROF));
#endif
sigaction(SIGSEGV,&old_segv_act,0);
return(result);
}
#endif
static volatile int firstpass;
STATIC ptr_t GC_skip_hole_openbsd(ptr_t p,ptr_t bound)
{
static volatile ptr_t result;
struct sigaction act;
word pgsz=(word)sysconf(_SC_PAGESIZE);
GC_ASSERT((word)bound>=pgsz);
GC_ASSERT(I_HOLD_LOCK());
act.sa_handler=GC_fault_handler_openbsd;
sigemptyset(&act.sa_mask);
act.sa_flags=SA_NODEFER|SA_RESTART;
sigaction(SIGSEGV,&act,&old_segv_act);
firstpass=1;
result=(ptr_t)((word)p&~(pgsz-1));
if (SETJMP(GC_jmp_buf_openbsd)!=0||firstpass){
firstpass=0;
if ((word)result>=(word)bound - pgsz){
result=bound;
} else {
result+=pgsz;
GC_noop1((word)(*result));
}
}
sigaction(SIGSEGV,&old_segv_act,0);
return(result);
}
#endif
#ifdef OS2
#include <stddef.h>
#if!defined(__IBMC__)&&!defined(__WATCOMC__)
struct exe_hdr {
unsigned short magic_number;
unsigned short padding[29];
long new_exe_offset;
};
#define E_MAGIC(x)(x).magic_number
#define EMAGIC 0x5A4D
#define E_LFANEW(x)(x).new_exe_offset
struct e32_exe {
unsigned char magic_number[2];
unsigned char byte_order;
unsigned char word_order;
unsigned long exe_format_level;
unsigned short cpu;
unsigned short os;
unsigned long padding1[13];
unsigned long object_table_offset;
unsigned long object_count;
unsigned long padding2[31];
};
#define E32_MAGIC1(x)(x).magic_number[0]
#define E32MAGIC1 'L'
#define E32_MAGIC2(x)(x).magic_number[1]
#define E32MAGIC2 'X'
#define E32_BORDER(x)(x).byte_order
#define E32LEBO 0
#define E32_WORDER(x)(x).word_order
#define E32LEWO 0
#define E32_CPU(x)(x).cpu
#define E32CPU286 1
#define E32_OBJTAB(x)(x).object_table_offset
#define E32_OBJCNT(x)(x).object_count
struct o32_obj {
unsigned long size;
unsigned long base;
unsigned long flags;
unsigned long pagemap;
unsigned long mapsize;
unsigned long reserved;
};
#define O32_FLAGS(x)(x).flags
#define OBJREAD 0x0001L
#define OBJWRITE 0x0002L
#define OBJINVALID 0x0080L
#define O32_SIZE(x)(x).size
#define O32_BASE(x)(x).base
#else
#ifndef WORD
#define WORD unsigned short
#endif
#ifndef DWORD
#define DWORD unsigned long
#endif
#define EXE386 1
#include <newexe.h>
#include <exe386.h>
#endif
#define INCL_DOSEXCEPTIONS
#define INCL_DOSPROCESS
#define INCL_DOSERRORS
#define INCL_DOSMODULEMGR
#define INCL_DOSMEMMGR
#include <os2.h>
#endif
GC_INNER size_t GC_page_size=0;
#if defined(MSWIN32)||defined(MSWINCE)||defined(CYGWIN32)
#ifndef VER_PLATFORM_WIN32_CE
#define VER_PLATFORM_WIN32_CE 3
#endif
#if defined(MSWINCE)&&defined(THREADS)
GC_INNER GC_bool GC_dont_query_stack_min=FALSE;
#endif
GC_INNER SYSTEM_INFO GC_sysinfo;
GC_INNER void GC_setpagesize(void)
{
GetSystemInfo(&GC_sysinfo);
#if defined(CYGWIN32)&&(defined(MPROTECT_VDB)||defined(USE_MUNMAP))
GC_page_size=(size_t)GC_sysinfo.dwAllocationGranularity;
GC_ASSERT(GC_page_size>=(size_t)GC_sysinfo.dwPageSize);
#else
GC_page_size=(size_t)GC_sysinfo.dwPageSize;
#endif
#if defined(MSWINCE)&&!defined(_WIN32_WCE_EMULATION)
{
OSVERSIONINFO verInfo;
verInfo.dwOSVersionInfoSize=sizeof(OSVERSIONINFO);
if (!GetVersionEx(&verInfo))
ABORT("GetVersionEx failed");
if (verInfo.dwPlatformId==VER_PLATFORM_WIN32_CE&&
verInfo.dwMajorVersion < 6){
GC_sysinfo.lpMaximumApplicationAddress=(LPVOID)((word)32<<20);
#ifdef THREADS
if (verInfo.dwMajorVersion < 5)
GC_dont_query_stack_min=TRUE;
#endif
}
}
#endif
}
#ifndef CYGWIN32
#define is_writable(prot)((prot)==PAGE_READWRITE||(prot)==PAGE_WRITECOPY||(prot)==PAGE_EXECUTE_READWRITE||(prot)==PAGE_EXECUTE_WRITECOPY)
STATIC word GC_get_writable_length(ptr_t p,ptr_t*base)
{
MEMORY_BASIC_INFORMATION buf;
word result;
word protect;
result=VirtualQuery(p,&buf,sizeof(buf));
if (result!=sizeof(buf))ABORT("Weird VirtualQuery result");
if (base!=0)*base=(ptr_t)(buf.AllocationBase);
protect=(buf.Protect&~(PAGE_GUARD|PAGE_NOCACHE));
if (!is_writable(protect)){
return(0);
}
if (buf.State!=MEM_COMMIT)return(0);
return(buf.RegionSize);
}
GC_API int GC_CALL GC_get_stack_base(struct GC_stack_base*sb)
{
ptr_t trunc_sp;
word size;
if (!GC_page_size)GC_setpagesize();
trunc_sp=(ptr_t)((word)GC_approx_sp()&~(GC_page_size - 1));
size=GC_get_writable_length(trunc_sp,0);
GC_ASSERT(size!=0);
sb->mem_base=trunc_sp+size;
return GC_SUCCESS;
}
#else
GC_API int GC_CALL GC_get_stack_base(struct GC_stack_base*sb)
{
#ifdef X86_64
sb->mem_base=((NT_TIB*)NtCurrentTeb())->StackBase;
#else
void*_tlsbase;
__asm__ ("movl %%fs:4,%0"
:"=r" (_tlsbase));
sb->mem_base=_tlsbase;
#endif
return GC_SUCCESS;
}
#endif
#define HAVE_GET_STACK_BASE
#else
GC_INNER void GC_setpagesize(void)
{
#if defined(MPROTECT_VDB)||defined(PROC_VDB)||defined(USE_MMAP)
GC_page_size=(size_t)GETPAGESIZE();
#if!defined(CPPCHECK)
if (0==GC_page_size)
ABORT("getpagesize failed");
#endif
#else
GC_page_size=HBLKSIZE;
#endif
}
#endif
#ifdef HAIKU
#include <kernel/OS.h>
GC_API int GC_CALL GC_get_stack_base(struct GC_stack_base*sb)
{
thread_info th;
get_thread_info(find_thread(NULL),&th);
sb->mem_base=th.stack_end;
return GC_SUCCESS;
}
#define HAVE_GET_STACK_BASE
#endif
#ifdef OS2
GC_API int GC_CALL GC_get_stack_base(struct GC_stack_base*sb)
{
PTIB ptib;
PPIB ppib;
if (DosGetInfoBlocks(&ptib,&ppib)!=NO_ERROR){
WARN("DosGetInfoBlocks failed\n",0);
return GC_UNIMPLEMENTED;
}
sb->mem_base=ptib->tib_pstacklimit;
return GC_SUCCESS;
}
#define HAVE_GET_STACK_BASE
#endif
#ifdef AMIGA
#define GC_AMIGA_SB
#undef GC_AMIGA_SB
#define GET_MAIN_STACKBASE_SPECIAL
#endif
#if defined(NEED_FIND_LIMIT)||defined(UNIX_LIKE)
typedef void (*GC_fault_handler_t)(int);
#if defined(SUNOS5SIGS)||defined(IRIX5)||defined(OSF1)||defined(HAIKU)||defined(HURD)||defined(FREEBSD)||defined(NETBSD)
static struct sigaction old_segv_act;
#if defined(_sigargs)||defined(HURD)||defined(NETBSD)||defined(FREEBSD)
static struct sigaction old_bus_act;
#endif
#else
static GC_fault_handler_t old_segv_handler;
#ifdef HAVE_SIGBUS
static GC_fault_handler_t old_bus_handler;
#endif
#endif
GC_INNER void GC_set_and_save_fault_handler(GC_fault_handler_t h)
{
#if defined(SUNOS5SIGS)||defined(IRIX5)||defined(OSF1)||defined(HAIKU)||defined(HURD)||defined(FREEBSD)||defined(NETBSD)||defined(OPENBSD)
struct sigaction act;
act.sa_handler=h;
#ifdef SIGACTION_FLAGS_NODEFER_HACK
act.sa_flags=SA_RESTART|SA_NODEFER;
#else
act.sa_flags=SA_RESTART;
#endif
(void)sigemptyset(&act.sa_mask);
#ifdef GC_IRIX_THREADS
(void)sigaction(SIGSEGV,0,&old_segv_act);
(void)sigaction(SIGSEGV,&act,0);
#else
(void)sigaction(SIGSEGV,&act,&old_segv_act);
#if defined(IRIX5)&&defined(_sigargs)||defined(HURD)||defined(NETBSD)||defined(FREEBSD)
(void)sigaction(SIGBUS,&act,&old_bus_act);
#endif
#endif
#else
old_segv_handler=signal(SIGSEGV,h);
#ifdef HAVE_SIGBUS
old_bus_handler=signal(SIGBUS,h);
#endif
#endif
#if defined(CPPCHECK)&&defined(ADDRESS_SANITIZER)
GC_noop1((word)&__asan_default_options);
#endif
}
#endif
#if defined(NEED_FIND_LIMIT)||(defined(USE_PROC_FOR_LIBRARIES)&&defined(THREADS))
#define MIN_PAGE_SIZE 256
GC_INNER JMP_BUF GC_jmp_buf;
STATIC void GC_fault_handler(int sig GC_ATTR_UNUSED)
{
LONGJMP(GC_jmp_buf,1);
}
GC_INNER void GC_setup_temporary_fault_handler(void)
{
GC_ASSERT(I_HOLD_LOCK());
GC_set_and_save_fault_handler(GC_fault_handler);
}
GC_INNER void GC_reset_fault_handler(void)
{
#if defined(SUNOS5SIGS)||defined(IRIX5)||defined(OSF1)||defined(HAIKU)||defined(HURD)||defined(FREEBSD)||defined(NETBSD)||defined(OPENBSD)
(void)sigaction(SIGSEGV,&old_segv_act,0);
#if defined(IRIX5)&&defined(_sigargs)||defined(HURD)||defined(NETBSD)
(void)sigaction(SIGBUS,&old_bus_act,0);
#endif
#else
(void)signal(SIGSEGV,old_segv_handler);
#ifdef HAVE_SIGBUS
(void)signal(SIGBUS,old_bus_handler);
#endif
#endif
}
GC_ATTR_NO_SANITIZE_ADDR
STATIC ptr_t GC_find_limit_with_bound(ptr_t p,GC_bool up,ptr_t bound)
{
static volatile ptr_t result;
GC_ASSERT(up?(word)bound>=MIN_PAGE_SIZE
:(word)bound<=~(word)MIN_PAGE_SIZE);
GC_ASSERT(I_HOLD_LOCK());
GC_setup_temporary_fault_handler();
if (SETJMP(GC_jmp_buf)==0){
result=(ptr_t)(((word)(p))
&~(MIN_PAGE_SIZE-1));
for (;;){
if (up){
if ((word)result>=(word)bound - MIN_PAGE_SIZE){
result=bound;
break;
}
result+=MIN_PAGE_SIZE;
} else {
if ((word)result<=(word)bound+MIN_PAGE_SIZE){
result=bound - MIN_PAGE_SIZE;
break;
}
result-=MIN_PAGE_SIZE;
}
GC_noop1((word)(*result));
}
}
GC_reset_fault_handler();
if (!up){
result+=MIN_PAGE_SIZE;
}
return(result);
}
void*GC_find_limit(void*p,int up)
{
return GC_find_limit_with_bound((ptr_t)p,(GC_bool)up,
up?(ptr_t)GC_WORD_MAX:0);
}
#endif
#ifdef HPUX_MAIN_STACKBOTTOM
#include <sys/param.h>
#include <sys/pstat.h>
STATIC ptr_t GC_hpux_main_stack_base(void)
{
struct pst_vm_status vm_status;
int i=0;
while (pstat_getprocvm(&vm_status,sizeof(vm_status),0,i++)==1){
if (vm_status.pst_type==PS_STACK)
return (ptr_t)vm_status.pst_vaddr;
}
#ifdef STACK_GROWS_UP
return (ptr_t)GC_find_limit(GC_approx_sp(),FALSE);
#else
return (ptr_t)GC_find_limit(GC_approx_sp(),TRUE);
#endif
}
#endif
#ifdef HPUX_STACKBOTTOM
#include <sys/param.h>
#include <sys/pstat.h>
GC_INNER ptr_t GC_get_register_stack_base(void)
{
struct pst_vm_status vm_status;
int i=0;
while (pstat_getprocvm(&vm_status,sizeof(vm_status),0,i++)==1){
if (vm_status.pst_type==PS_RSESTACK){
return (ptr_t)vm_status.pst_vaddr;
}
}
return (ptr_t)(((word)GC_stackbottom - BACKING_STORE_DISPLACEMENT - 1)
&~(BACKING_STORE_ALIGNMENT - 1));
}
#endif
#ifdef LINUX_STACKBOTTOM
#include <sys/types.h>
#include <sys/stat.h>
#define STAT_SKIP 27
#ifdef USE_LIBC_PRIVATES
EXTERN_C_BEGIN
#pragma weak __libc_stack_end
extern ptr_t __libc_stack_end;
#ifdef IA64
#pragma weak __libc_ia64_register_backing_store_base
extern ptr_t __libc_ia64_register_backing_store_base;
#endif
EXTERN_C_END
#endif
#ifdef IA64
GC_INNER ptr_t GC_get_register_stack_base(void)
{
ptr_t result;
#ifdef USE_LIBC_PRIVATES
if (0!=&__libc_ia64_register_backing_store_base
&&0!=__libc_ia64_register_backing_store_base){
return __libc_ia64_register_backing_store_base;
}
#endif
result=backing_store_base_from_proc();
if (0==result){
result=(ptr_t)GC_find_limit(GC_save_regs_in_stack(),FALSE);
}
return result;
}
#endif
STATIC ptr_t GC_linux_main_stack_base(void)
{
#ifndef STAT_READ
#define STAT_BUF_SIZE 4096
#define STAT_READ read
#endif
char stat_buf[STAT_BUF_SIZE];
int f;
word result;
int i,buf_offset=0,len;
#ifdef USE_LIBC_PRIVATES
if (0!=&__libc_stack_end&&0!=__libc_stack_end){
#if defined(IA64)
if (((word)__libc_stack_end&0xfff)+0x10 < 0x1000){
return __libc_stack_end+0x10;
}
#elif defined(SPARC)
if (__libc_stack_end!=(ptr_t)(unsigned long)0x1)
return __libc_stack_end;
#else
return __libc_stack_end;
#endif
}
#endif
f=open("/proc/self/stat",O_RDONLY);
if (f < 0)
ABORT("Couldn't read/proc/self/stat");
len=STAT_READ(f,stat_buf,STAT_BUF_SIZE);
close(f);
for (i=0;i < STAT_SKIP;++i){
while (buf_offset < len&&isspace(stat_buf[buf_offset++])){
}
while (buf_offset < len&&!isspace(stat_buf[buf_offset++])){
}
}
while (buf_offset < len&&isspace(stat_buf[buf_offset])){
buf_offset++;
}
for (i=0;buf_offset+i < len;i++){
if (!isdigit(stat_buf[buf_offset+i]))break;
}
if (buf_offset+i>=len)ABORT("Could not parse/proc/self/stat");
stat_buf[buf_offset+i]='\0';
result=(word)STRTOULL(&stat_buf[buf_offset],NULL,10);
if (result < 0x100000||(result&(sizeof(word)- 1))!=0)
ABORT("Absurd stack bottom value");
return (ptr_t)result;
}
#endif
#ifdef FREEBSD_STACKBOTTOM
#include <unistd.h>
#include <sys/types.h>
#include <sys/sysctl.h>
STATIC ptr_t GC_freebsd_main_stack_base(void)
{
int nm[2]={CTL_KERN,KERN_USRSTACK};
ptr_t base;
size_t len=sizeof(ptr_t);
int r=sysctl(nm,2,&base,&len,NULL,0);
if (r)ABORT("Error getting main stack base");
return base;
}
#endif
#if defined(ECOS)||defined(NOSYS)
ptr_t GC_get_main_stack_base(void)
{
return STACKBOTTOM;
}
#define GET_MAIN_STACKBASE_SPECIAL
#elif defined(SYMBIAN)
EXTERN_C_BEGIN
extern int GC_get_main_symbian_stack_base(void);
EXTERN_C_END
ptr_t GC_get_main_stack_base(void)
{
return (ptr_t)GC_get_main_symbian_stack_base();
}
#define GET_MAIN_STACKBASE_SPECIAL
#elif defined(__EMSCRIPTEN__)
#include <emscripten.h>
static void*emscripten_stack_base;
static void scan_stack_cb(void*begin,void*end)
{
(void)begin;
emscripten_stack_base=end;
}
ptr_t GC_get_main_stack_base(void)
{
emscripten_scan_stack(scan_stack_cb);
return (ptr_t)emscripten_stack_base;
}
#define GET_MAIN_STACKBASE_SPECIAL
#elif!defined(AMIGA)&&!defined(HAIKU)&&!defined(OS2)&&!defined(MSWIN32)&&!defined(MSWINCE)&&!defined(CYGWIN32)&&!defined(GC_OPENBSD_THREADS)&&(!defined(GC_SOLARIS_THREADS)||defined(_STRICT_STDC))
#if (defined(HAVE_PTHREAD_ATTR_GET_NP)||defined(HAVE_PTHREAD_GETATTR_NP))&&(defined(THREADS)||defined(USE_GET_STACKBASE_FOR_MAIN))
#include <pthread.h>
#ifdef HAVE_PTHREAD_NP_H
#include <pthread_np.h>
#endif
#elif defined(DARWIN)&&!defined(NO_PTHREAD_GET_STACKADDR_NP)
#include <pthread.h>
#undef STACKBOTTOM
#define STACKBOTTOM (ptr_t)pthread_get_stackaddr_np(pthread_self())
#endif
ptr_t GC_get_main_stack_base(void)
{
ptr_t result;
#if (defined(HAVE_PTHREAD_ATTR_GET_NP)||defined(HAVE_PTHREAD_GETATTR_NP))&&(defined(USE_GET_STACKBASE_FOR_MAIN)||(defined(THREADS)&&!defined(REDIRECT_MALLOC)))
pthread_attr_t attr;
void*stackaddr;
size_t size;
#ifdef HAVE_PTHREAD_ATTR_GET_NP
if (pthread_attr_init(&attr)==0
&&(pthread_attr_get_np(pthread_self(),&attr)==0
?TRUE:(pthread_attr_destroy(&attr),FALSE)))
#else
if (pthread_getattr_np(pthread_self(),&attr)==0)
#endif
{
if (pthread_attr_getstack(&attr,&stackaddr,&size)==0
&&stackaddr!=NULL){
(void)pthread_attr_destroy(&attr);
#ifdef STACK_GROWS_DOWN
stackaddr=(char*)stackaddr+size;
#endif
return (ptr_t)stackaddr;
}
(void)pthread_attr_destroy(&attr);
}
WARN("pthread_getattr_np or pthread_attr_getstack failed"
" for main thread\n",0);
#endif
#ifdef STACKBOTTOM
result=STACKBOTTOM;
#else
#ifdef HEURISTIC1
#define STACKBOTTOM_ALIGNMENT_M1 ((word)STACK_GRAN - 1)
#ifdef STACK_GROWS_DOWN
result=(ptr_t)(((word)GC_approx_sp()+STACKBOTTOM_ALIGNMENT_M1)
&~STACKBOTTOM_ALIGNMENT_M1);
#else
result=(ptr_t)((word)GC_approx_sp()&~STACKBOTTOM_ALIGNMENT_M1);
#endif
#elif defined(HPUX_MAIN_STACKBOTTOM)
result=GC_hpux_main_stack_base();
#elif defined(LINUX_STACKBOTTOM)
result=GC_linux_main_stack_base();
#elif defined(FREEBSD_STACKBOTTOM)
result=GC_freebsd_main_stack_base();
#elif defined(HEURISTIC2)
{
ptr_t sp=GC_approx_sp();
#ifdef STACK_GROWS_DOWN
result=(ptr_t)GC_find_limit(sp,TRUE);
#if defined(HEURISTIC2_LIMIT)&&!defined(CPPCHECK)
if ((word)result > (word)HEURISTIC2_LIMIT
&&(word)sp < (word)HEURISTIC2_LIMIT){
result=HEURISTIC2_LIMIT;
}
#endif
#else
result=(ptr_t)GC_find_limit(sp,FALSE);
#if defined(HEURISTIC2_LIMIT)&&!defined(CPPCHECK)
if ((word)result < (word)HEURISTIC2_LIMIT
&&(word)sp > (word)HEURISTIC2_LIMIT){
result=HEURISTIC2_LIMIT;
}
#endif
#endif
}
#elif defined(STACK_NOT_SCANNED)||defined(CPPCHECK)
result=NULL;
#else
#error None of HEURISTIC*and*STACKBOTTOM defined!
#endif
#if defined(STACK_GROWS_DOWN)&&!defined(CPPCHECK)
if (result==0)
result=(ptr_t)(signed_word)(-sizeof(ptr_t));
#endif
#endif
#if!defined(CPPCHECK)
GC_ASSERT((word)GC_approx_sp()HOTTER_THAN (word)result);
#endif
return(result);
}
#define GET_MAIN_STACKBASE_SPECIAL
#endif
#if (defined(HAVE_PTHREAD_ATTR_GET_NP)||defined(HAVE_PTHREAD_GETATTR_NP))&&defined(THREADS)&&!defined(HAVE_GET_STACK_BASE)
#include <pthread.h>
#ifdef HAVE_PTHREAD_NP_H
#include <pthread_np.h>
#endif
GC_API int GC_CALL GC_get_stack_base(struct GC_stack_base*b)
{
pthread_attr_t attr;
size_t size;
#ifdef IA64
DCL_LOCK_STATE;
#endif
#ifdef HAVE_PTHREAD_ATTR_GET_NP
if (pthread_attr_init(&attr)!=0)
ABORT("pthread_attr_init failed");
if (pthread_attr_get_np(pthread_self(),&attr)!=0){
WARN("pthread_attr_get_np failed\n",0);
(void)pthread_attr_destroy(&attr);
return GC_UNIMPLEMENTED;
}
#else
if (pthread_getattr_np(pthread_self(),&attr)!=0){
WARN("pthread_getattr_np failed\n",0);
return GC_UNIMPLEMENTED;
}
#endif
if (pthread_attr_getstack(&attr,&(b->mem_base),&size)!=0){
ABORT("pthread_attr_getstack failed");
}
(void)pthread_attr_destroy(&attr);
#ifdef STACK_GROWS_DOWN
b->mem_base=(char*)(b->mem_base)+size;
#endif
#ifdef IA64
LOCK();
{
IF_CANCEL(int cancel_state;)
ptr_t bsp;
ptr_t next_stack;
DISABLE_CANCEL(cancel_state);
bsp=GC_save_regs_in_stack();
next_stack=GC_greatest_stack_base_below(bsp);
if (0==next_stack){
b->reg_base=GC_find_limit(bsp,FALSE);
} else {
b->reg_base=GC_find_limit_with_bound(bsp,FALSE,next_stack);
}
RESTORE_CANCEL(cancel_state);
}
UNLOCK();
#endif
return GC_SUCCESS;
}
#define HAVE_GET_STACK_BASE
#endif
#if defined(GC_DARWIN_THREADS)&&!defined(NO_PTHREAD_GET_STACKADDR_NP)
#include <pthread.h>
GC_API int GC_CALL GC_get_stack_base(struct GC_stack_base*b)
{
b->mem_base=pthread_get_stackaddr_np(pthread_self());
GC_ASSERT((word)GC_approx_sp()HOTTER_THAN (word)b->mem_base);
return GC_SUCCESS;
}
#define HAVE_GET_STACK_BASE
#endif
#ifdef GC_OPENBSD_THREADS
#include <sys/signal.h>
#include <pthread.h>
#include <pthread_np.h>
GC_API int GC_CALL GC_get_stack_base(struct GC_stack_base*sb)
{
stack_t stack;
if (pthread_stackseg_np(pthread_self(),&stack))
ABORT("pthread_stackseg_np(self)failed");
sb->mem_base=stack.ss_sp;
return GC_SUCCESS;
}
#define HAVE_GET_STACK_BASE
#endif
#if defined(GC_SOLARIS_THREADS)&&!defined(_STRICT_STDC)
#include <thread.h>
#include <signal.h>
#include <pthread.h>
static pthread_t stackbase_main_self=0;
static void*stackbase_main_ss_sp=NULL;
GC_API int GC_CALL GC_get_stack_base(struct GC_stack_base*b)
{
stack_t s;
pthread_t self=pthread_self();
if (self==stackbase_main_self)
{
b->mem_base=stackbase_main_ss_sp;
GC_ASSERT(b->mem_base!=NULL);
return GC_SUCCESS;
}
if (thr_stksegment(&s)){
ABORT("thr_stksegment failed");
}
GC_ASSERT((word)GC_approx_sp()HOTTER_THAN (word)s.ss_sp);
if (!stackbase_main_self&&thr_main()!=0)
{
stackbase_main_ss_sp=s.ss_sp;
stackbase_main_self=self;
}
b->mem_base=s.ss_sp;
return GC_SUCCESS;
}
#define HAVE_GET_STACK_BASE
#endif
#ifdef GC_RTEMS_PTHREADS
GC_API int GC_CALL GC_get_stack_base(struct GC_stack_base*sb)
{
sb->mem_base=rtems_get_stack_bottom();
return GC_SUCCESS;
}
#define HAVE_GET_STACK_BASE
#endif
#ifndef HAVE_GET_STACK_BASE
#ifdef NEED_FIND_LIMIT
GC_API int GC_CALL GC_get_stack_base(struct GC_stack_base*b)
{
IF_CANCEL(int cancel_state;)
DCL_LOCK_STATE;
LOCK();
DISABLE_CANCEL(cancel_state);
#ifdef STACK_GROWS_DOWN
b->mem_base=GC_find_limit(GC_approx_sp(),TRUE);
#ifdef IA64
b->reg_base=GC_find_limit(GC_save_regs_in_stack(),FALSE);
#endif
#else
b->mem_base=GC_find_limit(GC_approx_sp(),FALSE);
#endif
RESTORE_CANCEL(cancel_state);
UNLOCK();
return GC_SUCCESS;
}
#else
GC_API int GC_CALL GC_get_stack_base(
struct GC_stack_base*b GC_ATTR_UNUSED)
{
#if defined(GET_MAIN_STACKBASE_SPECIAL)&&!defined(THREADS)&&!defined(IA64)
b->mem_base=GC_get_main_stack_base();
return GC_SUCCESS;
#else
return GC_UNIMPLEMENTED;
#endif
}
#endif
#endif
#ifndef GET_MAIN_STACKBASE_SPECIAL
ptr_t GC_get_main_stack_base(void)
{
struct GC_stack_base sb;
if (GC_get_stack_base(&sb)!=GC_SUCCESS)
ABORT("GC_get_stack_base failed");
GC_ASSERT((word)GC_approx_sp()HOTTER_THAN (word)sb.mem_base);
return (ptr_t)sb.mem_base;
}
#endif
#ifdef OS2
void GC_register_data_segments(void)
{
PTIB ptib;
PPIB ppib;
HMODULE module_handle;
#define PBUFSIZ 512
UCHAR path[PBUFSIZ];
FILE*myexefile;
struct exe_hdr hdrdos;
struct e32_exe hdr386;
struct o32_obj seg;
int nsegs;
#if defined(CPPCHECK)
hdrdos.padding[0]=0;
hdr386.exe_format_level=0;
hdr386.os=0;
hdr386.padding1[0]=0;
hdr386.padding2[0]=0;
seg.pagemap=0;
seg.mapsize=0;
seg.reserved=0;
#endif
if (DosGetInfoBlocks(&ptib,&ppib)!=NO_ERROR){
ABORT("DosGetInfoBlocks failed");
}
module_handle=ppib->pib_hmte;
if (DosQueryModuleName(module_handle,PBUFSIZ,path)!=NO_ERROR){
ABORT("DosQueryModuleName failed");
}
myexefile=fopen(path,"rb");
if (myexefile==0){
ABORT_ARG1("Failed to open executable",":%s",path);
}
if (fread((char*)(&hdrdos),1,sizeof(hdrdos),myexefile)
< sizeof(hdrdos)){
ABORT_ARG1("Could not read MSDOS header"," from:%s",path);
}
if (E_MAGIC(hdrdos)!=EMAGIC){
ABORT_ARG1("Bad DOS magic number"," in file:%s",path);
}
if (fseek(myexefile,E_LFANEW(hdrdos),SEEK_SET)!=0){
ABORT_ARG1("Bad DOS magic number"," in file:%s",path);
}
if (fread((char*)(&hdr386),1,sizeof(hdr386),myexefile)
< sizeof(hdr386)){
ABORT_ARG1("Could not read OS/2 header"," from:%s",path);
}
if (E32_MAGIC1(hdr386)!=E32MAGIC1||E32_MAGIC2(hdr386)!=E32MAGIC2){
ABORT_ARG1("Bad OS/2 magic number"," in file:%s",path);
}
if (E32_BORDER(hdr386)!=E32LEBO||E32_WORDER(hdr386)!=E32LEWO){
ABORT_ARG1("Bad byte order in executable"," file:%s",path);
}
if (E32_CPU(hdr386)==E32CPU286){
ABORT_ARG1("GC cannot handle 80286 executables",":%s",path);
}
if (fseek(myexefile,E_LFANEW(hdrdos)+E32_OBJTAB(hdr386),
SEEK_SET)!=0){
ABORT_ARG1("Seek to object table failed"," in file:%s",path);
}
for (nsegs=E32_OBJCNT(hdr386);nsegs > 0;nsegs--){
int flags;
if (fread((char*)(&seg),1,sizeof(seg),myexefile)< sizeof(seg)){
ABORT_ARG1("Could not read obj table entry"," from file:%s",path);
}
flags=O32_FLAGS(seg);
if (!(flags&OBJWRITE))continue;
if (!(flags&OBJREAD))continue;
if (flags&OBJINVALID){
GC_err_printf("Object with invalid pages?\n");
continue;
}
GC_add_roots_inner((ptr_t)O32_BASE(seg),
(ptr_t)(O32_BASE(seg)+O32_SIZE(seg)),FALSE);
}
(void)fclose(myexefile);
}
#else
#if defined(GWW_VDB)
#ifndef MEM_WRITE_WATCH
#define MEM_WRITE_WATCH 0x200000
#endif
#ifndef WRITE_WATCH_FLAG_RESET
#define WRITE_WATCH_FLAG_RESET 1
#endif
#define GC_ULONG_PTR word
typedef UINT (WINAPI*GetWriteWatch_type)(
DWORD,PVOID,GC_ULONG_PTR,
PVOID*,GC_ULONG_PTR*,PULONG);
static FARPROC GetWriteWatch_func;
static DWORD GetWriteWatch_alloc_flag;
#define GC_GWW_AVAILABLE()(GetWriteWatch_func!=0)
static void detect_GetWriteWatch(void)
{
static GC_bool done;
HMODULE hK32;
if (done)
return;
#if defined(MPROTECT_VDB)
{
char*str=GETENV("GC_USE_GETWRITEWATCH");
#if defined(GC_PREFER_MPROTECT_VDB)
if (str==NULL||(*str=='0'&&*(str+1)=='\0')){
done=TRUE;
return;
}
#else
if (str!=NULL&&*str=='0'&&*(str+1)=='\0'){
done=TRUE;
return;
}
#endif
}
#endif
#ifdef MSWINRT_FLAVOR
{
MEMORY_BASIC_INFORMATION memInfo;
SIZE_T result=VirtualQuery(GetProcAddress,
&memInfo,sizeof(memInfo));
if (result!=sizeof(memInfo))
ABORT("Weird VirtualQuery result");
hK32=(HMODULE)memInfo.AllocationBase;
}
#else
hK32=GetModuleHandle(TEXT("kernel32.dll"));
#endif
if (hK32!=(HMODULE)0&&
(GetWriteWatch_func=GetProcAddress(hK32,"GetWriteWatch"))!=0){
void*page;
GC_ASSERT(GC_page_size!=0);
page=VirtualAlloc(NULL,GC_page_size,MEM_WRITE_WATCH|MEM_RESERVE,
PAGE_READWRITE);
if (page!=NULL){
PVOID pages[16];
GC_ULONG_PTR count=16;
DWORD page_size;
if ((*(GetWriteWatch_type)(word)GetWriteWatch_func)(
WRITE_WATCH_FLAG_RESET,page,
GC_page_size,pages,&count,
&page_size)!=0){
GetWriteWatch_func=0;
} else {
GetWriteWatch_alloc_flag=MEM_WRITE_WATCH;
}
VirtualFree(page,0,MEM_RELEASE);
} else {
GetWriteWatch_func=0;
}
}
#ifndef SMALL_CONFIG
if (!GetWriteWatch_func){
GC_COND_LOG_PRINTF("Did not find a usable GetWriteWatch()\n");
} else {
GC_COND_LOG_PRINTF("Using GetWriteWatch()\n");
}
#endif
done=TRUE;
}
#else
#define GetWriteWatch_alloc_flag 0
#endif
#if defined(MSWIN32)||defined(MSWINCE)||defined(CYGWIN32)
#ifdef MSWIN32
GC_INNER GC_bool GC_no_win32_dlls=FALSE;
GC_INNER GC_bool GC_wnt=FALSE;
GC_INNER void GC_init_win32(void)
{
#if defined(_WIN64)||(defined(_MSC_VER)&&_MSC_VER>=1800)
GC_wnt=TRUE;
#else
DWORD v=GetVersion();
GC_wnt=!(v&0x80000000);
GC_no_win32_dlls|=((!GC_wnt)&&(v&0xff)<=3);
#endif
#ifdef USE_MUNMAP
if (GC_no_win32_dlls){
GC_unmap_threshold=0;
}
#endif
}
STATIC ptr_t GC_least_described_address(ptr_t start)
{
MEMORY_BASIC_INFORMATION buf;
LPVOID limit=GC_sysinfo.lpMinimumApplicationAddress;
ptr_t p=(ptr_t)((word)start&~(GC_page_size - 1));
GC_ASSERT(GC_page_size!=0);
for (;;){
size_t result;
LPVOID q=(LPVOID)(p - GC_page_size);
if ((word)q > (word)p||(word)q < (word)limit)break;
result=VirtualQuery(q,&buf,sizeof(buf));
if (result!=sizeof(buf)||buf.AllocationBase==0)break;
p=(ptr_t)(buf.AllocationBase);
}
return p;
}
#endif
#if defined(USE_WINALLOC)&&!defined(REDIRECT_MALLOC)
STATIC size_t GC_max_root_size=100000;
STATIC struct GC_malloc_heap_list {
void*allocation_base;
struct GC_malloc_heap_list*next;
}*GC_malloc_heap_l=0;
STATIC GC_bool GC_is_malloc_heap_base(void*p)
{
struct GC_malloc_heap_list*q=GC_malloc_heap_l;
while (0!=q){
if (q->allocation_base==p)return TRUE;
q=q->next;
}
return FALSE;
}
STATIC void*GC_get_allocation_base(void*p)
{
MEMORY_BASIC_INFORMATION buf;
size_t result=VirtualQuery(p,&buf,sizeof(buf));
if (result!=sizeof(buf)){
ABORT("Weird VirtualQuery result");
}
return buf.AllocationBase;
}
GC_INNER void GC_add_current_malloc_heap(void)
{
struct GC_malloc_heap_list*new_l=(struct GC_malloc_heap_list*)
malloc(sizeof(struct GC_malloc_heap_list));
void*candidate;
if (NULL==new_l)return;
candidate=GC_get_allocation_base(new_l);
if (GC_is_malloc_heap_base(candidate)){
size_t req_size=10000;
do {
void*p=malloc(req_size);
if (0==p){
free(new_l);
return;
}
candidate=GC_get_allocation_base(p);
free(p);
req_size*=2;
} while (GC_is_malloc_heap_base(candidate)
&&req_size < GC_max_root_size/10&&req_size < 500000);
if (GC_is_malloc_heap_base(candidate)){
free(new_l);
return;
}
}
GC_COND_LOG_PRINTF("Found new system malloc AllocationBase at %p\n",
candidate);
new_l->allocation_base=candidate;
new_l->next=GC_malloc_heap_l;
GC_malloc_heap_l=new_l;
}
STATIC void GC_free_malloc_heap_list(void)
{
struct GC_malloc_heap_list*q=GC_malloc_heap_l;
GC_malloc_heap_l=NULL;
while (q!=NULL){
struct GC_malloc_heap_list*next=q->next;
free(q);
q=next;
}
}
#endif
GC_INNER GC_bool GC_is_heap_base(void*p)
{
int i;
#if defined(USE_WINALLOC)&&!defined(REDIRECT_MALLOC)
if (GC_root_size > GC_max_root_size)
GC_max_root_size=GC_root_size;
if (GC_is_malloc_heap_base(p))
return TRUE;
#endif
for (i=0;i < (int)GC_n_heap_bases;i++){
if (GC_heap_bases[i]==p)return TRUE;
}
return FALSE;
}
#ifdef MSWIN32
STATIC void GC_register_root_section(ptr_t static_root)
{
MEMORY_BASIC_INFORMATION buf;
LPVOID p;
char*base;
char*limit;
if (!GC_no_win32_dlls)return;
p=base=limit=GC_least_described_address(static_root);
while ((word)p < (word)GC_sysinfo.lpMaximumApplicationAddress){
size_t result=VirtualQuery(p,&buf,sizeof(buf));
char*new_limit;
DWORD protect;
if (result!=sizeof(buf)||buf.AllocationBase==0
||GC_is_heap_base(buf.AllocationBase))break;
new_limit=(char*)p+buf.RegionSize;
protect=buf.Protect;
if (buf.State==MEM_COMMIT
&&is_writable(protect)){
if ((char*)p==limit){
limit=new_limit;
} else {
if (base!=limit)GC_add_roots_inner(base,limit,FALSE);
base=(char*)p;
limit=new_limit;
}
}
if ((word)p > (word)new_limit)break;
p=(LPVOID)new_limit;
}
if (base!=limit)GC_add_roots_inner(base,limit,FALSE);
}
#endif
void GC_register_data_segments(void)
{
#ifdef MSWIN32
GC_register_root_section((ptr_t)&GC_pages_executable);
#endif
}
#else
#if (defined(SVR4)||defined(AIX)||defined(DGUX)||(defined(LINUX)&&defined(SPARC)))&&!defined(PCR)
ptr_t GC_SysVGetDataStart(size_t max_page_size,ptr_t etext_addr)
{
word text_end=((word)(etext_addr)+sizeof(word)- 1)
&~(word)(sizeof(word)- 1);
word next_page=((text_end+(word)max_page_size - 1)
&~((word)max_page_size - 1));
word page_offset=(text_end&((word)max_page_size - 1));
volatile ptr_t result=(char*)(next_page+page_offset);
GC_setup_temporary_fault_handler();
if (SETJMP(GC_jmp_buf)==0){
#ifdef AO_HAVE_fetch_and_add
volatile AO_t zero=0;
(void)AO_fetch_and_add((volatile AO_t*)result,zero);
#else
char v=*result;
#if defined(CPPCHECK)
GC_noop1((word)&v);
#endif
*result=v;
#endif
GC_reset_fault_handler();
} else {
GC_reset_fault_handler();
result=(char*)GC_find_limit(DATAEND,FALSE);
}
return ( ptr_t)result;
}
#endif
#ifdef DATASTART_USES_BSDGETDATASTART
GC_INNER ptr_t GC_FreeBSDGetDataStart(size_t max_page_size,
ptr_t etext_addr)
{
word text_end=((word)(etext_addr)+sizeof(word)- 1)
&~(word)(sizeof(word)- 1);
volatile word next_page=(text_end+(word)max_page_size - 1)
&~((word)max_page_size - 1);
volatile ptr_t result=(ptr_t)text_end;
GC_setup_temporary_fault_handler();
if (SETJMP(GC_jmp_buf)==0){
for (;next_page < (word)DATAEND;next_page+=(word)max_page_size)
*(volatile char*)next_page;
GC_reset_fault_handler();
} else {
GC_reset_fault_handler();
result=(ptr_t)GC_find_limit(DATAEND,FALSE);
}
return(result);
}
#endif
#ifdef AMIGA
#define GC_AMIGA_DS
#undef GC_AMIGA_DS
#elif defined(OPENBSD)
void GC_register_data_segments(void)
{
ptr_t region_start=DATASTART;
if ((word)region_start - 1U>=(word)DATAEND)
ABORT_ARG2("Wrong DATASTART/END pair",
":%p .. %p",(void*)region_start,(void*)DATAEND);
for (;;){
#ifdef GC_OPENBSD_UTHREADS
ptr_t region_end=GC_find_limit_openbsd(region_start,DATAEND);
#else
ptr_t region_end=GC_find_limit_with_bound(region_start,TRUE,DATAEND);
#endif
GC_add_roots_inner(region_start,region_end,FALSE);
if ((word)region_end>=(word)DATAEND)
break;
region_start=GC_skip_hole_openbsd(region_end,DATAEND);
}
}
#else
#if!defined(PCR)&&!defined(MACOS)&&defined(REDIRECT_MALLOC)&&defined(GC_SOLARIS_THREADS)
EXTERN_C_BEGIN
extern caddr_t sbrk(int);
EXTERN_C_END
#endif
void GC_register_data_segments(void)
{
#if!defined(PCR)&&!defined(MACOS)
#if defined(REDIRECT_MALLOC)&&defined(GC_SOLARIS_THREADS)
GC_ASSERT(DATASTART);
{
ptr_t p=(ptr_t)sbrk(0);
if ((word)DATASTART < (word)p)
GC_add_roots_inner(DATASTART,p,FALSE);
}
#else
if ((word)DATASTART - 1U>=(word)DATAEND){
ABORT_ARG2("Wrong DATASTART/END pair",
":%p .. %p",(void*)DATASTART,(void*)DATAEND);
}
GC_add_roots_inner(DATASTART,DATAEND,FALSE);
#ifdef GC_HAVE_DATAREGION2
if ((word)DATASTART2 - 1U>=(word)DATAEND2)
ABORT_ARG2("Wrong DATASTART/END2 pair",
":%p .. %p",(void*)DATASTART2,(void*)DATAEND2);
GC_add_roots_inner(DATASTART2,DATAEND2,FALSE);
#endif
#endif
#endif
#if defined(MACOS)
{
#if defined(THINK_C)
extern void*GC_MacGetDataStart(void);
GC_add_roots_inner((ptr_t)GC_MacGetDataStart(),
(ptr_t)LMGetCurrentA5(),FALSE);
#else
#if defined(__MWERKS__)
#if!__POWERPC__
extern void*GC_MacGetDataStart(void);
#if __option(far_data)
extern void*GC_MacGetDataEnd(void);
#endif
GC_add_roots_inner((ptr_t)GC_MacGetDataStart(),
(ptr_t)LMGetCurrentA5(),FALSE);
#if __option(far_data)
GC_add_roots_inner((ptr_t)LMGetCurrentA5(),
(ptr_t)GC_MacGetDataEnd(),FALSE);
#endif
#else
extern char __data_start__[],__data_end__[];
GC_add_roots_inner((ptr_t)&__data_start__,
(ptr_t)&__data_end__,FALSE);
#endif
#endif
#endif
}
#endif
}
#endif
#endif
#endif
#if!defined(OS2)&&!defined(PCR)&&!defined(AMIGA)&&!defined(USE_WINALLOC)&&!defined(MACOS)&&!defined(DOS4GW)&&!defined(NINTENDO_SWITCH)&&!defined(NONSTOP)&&!defined(SN_TARGET_ORBIS)&&!defined(SN_TARGET_PS3)&&!defined(SN_TARGET_PSP2)&&!defined(RTEMS)&&!defined(__CC_ARM)
#define SBRK_ARG_T ptrdiff_t
#if defined(MMAP_SUPPORTED)
#ifdef USE_MMAP_FIXED
#define GC_MMAP_FLAGS MAP_FIXED|MAP_PRIVATE
#else
#define GC_MMAP_FLAGS MAP_PRIVATE
#endif
#ifdef USE_MMAP_ANON
#define zero_fd -1
#if defined(MAP_ANONYMOUS)&&!defined(CPPCHECK)
#define OPT_MAP_ANON MAP_ANONYMOUS
#else
#define OPT_MAP_ANON MAP_ANON
#endif
#else
static int zero_fd=-1;
#define OPT_MAP_ANON 0
#endif
#ifndef MSWIN_XBOX1
#if defined(SYMBIAN)&&!defined(USE_MMAP_ANON)
EXTERN_C_BEGIN
extern char*GC_get_private_path_and_zero_file(void);
EXTERN_C_END
#endif
STATIC ptr_t GC_unix_mmap_get_mem(size_t bytes)
{
void*result;
static ptr_t last_addr=HEAP_START;
#ifndef USE_MMAP_ANON
static GC_bool initialized=FALSE;
if (!EXPECT(initialized,TRUE)){
#ifdef SYMBIAN
char*path=GC_get_private_path_and_zero_file();
if (path!=NULL){
zero_fd=open(path,O_RDWR|O_CREAT,0644);
free(path);
}
#else
zero_fd=open("/dev/zero",O_RDONLY);
#endif
if (zero_fd==-1)
ABORT("Could not open/dev/zero");
if (fcntl(zero_fd,F_SETFD,FD_CLOEXEC)==-1)
WARN("Could not set FD_CLOEXEC for/dev/zero\n",0);
initialized=TRUE;
}
#endif
GC_ASSERT(GC_page_size!=0);
if (bytes&(GC_page_size - 1))ABORT("Bad GET_MEM arg");
result=mmap(last_addr,bytes,(PROT_READ|PROT_WRITE)
|(GC_pages_executable?PROT_EXEC:0),
GC_MMAP_FLAGS|OPT_MAP_ANON,zero_fd,0);
#undef IGNORE_PAGES_EXECUTABLE
if (EXPECT(MAP_FAILED==result,FALSE)){
if (HEAP_START==last_addr&&GC_pages_executable&&EACCES==errno)
ABORT("Cannot allocate executable pages");
return NULL;
}
last_addr=(ptr_t)(((word)result+bytes+GC_page_size - 1)
&~(GC_page_size - 1));
#if!defined(LINUX)
if (last_addr==0){
munmap(result,~GC_page_size - (size_t)result+1);
return GC_unix_mmap_get_mem(bytes);
}
#else
GC_ASSERT(last_addr!=0);
#endif
if (((word)result % HBLKSIZE)!=0)
ABORT(
"GC_unix_get_mem:Memory returned by mmap is not aligned to HBLKSIZE.");
return((ptr_t)result);
}
#endif
#endif
#if defined(USE_MMAP)
ptr_t GC_unix_get_mem(size_t bytes)
{
return GC_unix_mmap_get_mem(bytes);
}
#else
STATIC ptr_t GC_unix_sbrk_get_mem(size_t bytes)
{
ptr_t result;
#ifdef IRIX5
__LOCK_MALLOC();
#endif
{
ptr_t cur_brk=(ptr_t)sbrk(0);
SBRK_ARG_T lsbs=(word)cur_brk&(GC_page_size-1);
GC_ASSERT(GC_page_size!=0);
if ((SBRK_ARG_T)bytes < 0){
result=0;
goto out;
}
if (lsbs!=0){
if((ptr_t)sbrk((SBRK_ARG_T)GC_page_size - lsbs)==(ptr_t)(-1)){
result=0;
goto out;
}
}
#ifdef ADD_HEAP_GUARD_PAGES
{
ptr_t guard=(ptr_t)sbrk((SBRK_ARG_T)GC_page_size);
if (mprotect(guard,GC_page_size,PROT_NONE)!=0)
ABORT("ADD_HEAP_GUARD_PAGES:mprotect failed");
}
#endif
result=(ptr_t)sbrk((SBRK_ARG_T)bytes);
if (result==(ptr_t)(-1))result=0;
}
out:
#ifdef IRIX5
__UNLOCK_MALLOC();
#endif
return(result);
}
ptr_t GC_unix_get_mem(size_t bytes)
{
#if defined(MMAP_SUPPORTED)
static GC_bool sbrk_failed=FALSE;
ptr_t result=0;
if (GC_pages_executable){
return GC_unix_mmap_get_mem(bytes);
}
if (!sbrk_failed)result=GC_unix_sbrk_get_mem(bytes);
if (0==result){
sbrk_failed=TRUE;
result=GC_unix_mmap_get_mem(bytes);
}
if (0==result){
result=GC_unix_sbrk_get_mem(bytes);
}
return result;
#else
return GC_unix_sbrk_get_mem(bytes);
#endif
}
#endif
#endif
#ifdef OS2
void*os2_alloc(size_t bytes)
{
void*result;
if (DosAllocMem(&result,bytes,(PAG_READ|PAG_WRITE|PAG_COMMIT)
|(GC_pages_executable?PAG_EXECUTE:0))
!=NO_ERROR){
return(0);
}
if (result==0)return(os2_alloc(bytes));
return(result);
}
#endif
#ifdef MSWIN_XBOX1
ptr_t GC_durango_get_mem(size_t bytes)
{
if (0==bytes)return NULL;
return (ptr_t)VirtualAlloc(NULL,bytes,MEM_COMMIT|MEM_TOP_DOWN,
PAGE_READWRITE);
}
#elif defined(MSWINCE)
ptr_t GC_wince_get_mem(size_t bytes)
{
ptr_t result=0;
word i;
GC_ASSERT(GC_page_size!=0);
bytes=ROUNDUP_PAGESIZE(bytes);
for (i=0;i < GC_n_heap_bases;i++){
if (((word)(-(signed_word)GC_heap_lengths[i])
&(GC_sysinfo.dwAllocationGranularity-1))
>=bytes){
result=GC_heap_bases[i]+GC_heap_lengths[i];
break;
}
}
if (i==GC_n_heap_bases){
size_t res_bytes=
SIZET_SAT_ADD(bytes,(size_t)GC_sysinfo.dwAllocationGranularity-1)
&~((size_t)GC_sysinfo.dwAllocationGranularity-1);
result=(ptr_t)VirtualAlloc(NULL,res_bytes,
MEM_RESERVE|MEM_TOP_DOWN,
GC_pages_executable?PAGE_EXECUTE_READWRITE:
PAGE_READWRITE);
if (HBLKDISPL(result)!=0)ABORT("Bad VirtualAlloc result");
if (GC_n_heap_bases>=MAX_HEAP_SECTS)ABORT("Too many heap sections");
if (result==NULL)return NULL;
GC_heap_bases[GC_n_heap_bases]=result;
GC_heap_lengths[GC_n_heap_bases]=0;
GC_n_heap_bases++;
}
result=(ptr_t)VirtualAlloc(result,bytes,MEM_COMMIT,
GC_pages_executable?PAGE_EXECUTE_READWRITE:
PAGE_READWRITE);
#undef IGNORE_PAGES_EXECUTABLE
if (result!=NULL){
if (HBLKDISPL(result)!=0)ABORT("Bad VirtualAlloc result");
GC_heap_lengths[i]+=bytes;
}
return(result);
}
#elif (defined(USE_WINALLOC)&&!defined(MSWIN_XBOX1))||defined(CYGWIN32)
#ifdef USE_GLOBAL_ALLOC
#define GLOBAL_ALLOC_TEST 1
#else
#define GLOBAL_ALLOC_TEST GC_no_win32_dlls
#endif
#if (defined(GC_USE_MEM_TOP_DOWN)&&defined(USE_WINALLOC))||defined(CPPCHECK)
DWORD GC_mem_top_down=MEM_TOP_DOWN;
#else
#define GC_mem_top_down 0
#endif
ptr_t GC_win32_get_mem(size_t bytes)
{
ptr_t result;
#ifndef USE_WINALLOC
result=GC_unix_get_mem(bytes);
#else
#if defined(MSWIN32)&&!defined(MSWINRT_FLAVOR)
if (GLOBAL_ALLOC_TEST){
result=(ptr_t)GlobalAlloc(0,SIZET_SAT_ADD(bytes,HBLKSIZE));
result=(ptr_t)(((word)result+HBLKSIZE - 1)
&~(word)(HBLKSIZE - 1));
} else
#endif
{
#ifdef MPROTECT_VDB
#ifdef GWW_VDB
#define VIRTUAL_ALLOC_PAD (GC_GWW_AVAILABLE()?0:1)
#else
#define VIRTUAL_ALLOC_PAD 1
#endif
#else
#define VIRTUAL_ALLOC_PAD 0
#endif
result=(ptr_t)VirtualAlloc(NULL,
SIZET_SAT_ADD(bytes,VIRTUAL_ALLOC_PAD),
GetWriteWatch_alloc_flag
|(MEM_COMMIT|MEM_RESERVE)
|GC_mem_top_down,
GC_pages_executable?PAGE_EXECUTE_READWRITE:
PAGE_READWRITE);
#undef IGNORE_PAGES_EXECUTABLE
}
#endif
if (HBLKDISPL(result)!=0)ABORT("Bad VirtualAlloc result");
if (GC_n_heap_bases>=MAX_HEAP_SECTS)ABORT("Too many heap sections");
if (0!=result)GC_heap_bases[GC_n_heap_bases++]=result;
return(result);
}
#endif
#if defined(MSWIN32)||defined(MSWINCE)||defined(CYGWIN32)||defined(MSWIN_XBOX1)
GC_API void GC_CALL GC_win32_free_heap(void)
{
#if defined(USE_WINALLOC)&&!defined(REDIRECT_MALLOC)&&!defined(MSWIN_XBOX1)
GC_free_malloc_heap_list();
#endif
#if (defined(USE_WINALLOC)&&!defined(MSWIN_XBOX1)&&!defined(MSWINCE))||defined(CYGWIN32)
#ifndef MSWINRT_FLAVOR
#ifndef CYGWIN32
if (GLOBAL_ALLOC_TEST)
#endif
{
while (GC_n_heap_bases--> 0){
#ifdef CYGWIN32
#else
GlobalFree(GC_heap_bases[GC_n_heap_bases]);
#endif
GC_heap_bases[GC_n_heap_bases]=0;
}
return;
}
#endif
#ifndef CYGWIN32
while (GC_n_heap_bases > 0){
VirtualFree(GC_heap_bases[--GC_n_heap_bases],0,MEM_RELEASE);
GC_heap_bases[GC_n_heap_bases]=0;
}
#endif
#endif
}
#endif
#ifdef AMIGA
#define GC_AMIGA_AM
#undef GC_AMIGA_AM
#endif
#if defined(HAIKU)
#include <stdlib.h>
ptr_t GC_haiku_get_mem(size_t bytes)
{
void*mem;
GC_ASSERT(GC_page_size!=0);
if (posix_memalign(&mem,GC_page_size,bytes)==0)
return mem;
return NULL;
}
#endif
#ifdef USE_MUNMAP
#if!defined(NN_PLATFORM_CTR)&&!defined(MSWIN32)&&!defined(MSWINCE)&&!defined(MSWIN_XBOX1)
#include <unistd.h>
#ifdef SN_TARGET_PS3
#include <sys/memory.h>
#else
#include <sys/mman.h>
#endif
#include <sys/stat.h>
#include <sys/types.h>
#endif
STATIC ptr_t GC_unmap_start(ptr_t start,size_t bytes)
{
ptr_t result=(ptr_t)(((word)start+GC_page_size - 1)
&~(GC_page_size - 1));
GC_ASSERT(GC_page_size!=0);
if ((word)(result+GC_page_size)> (word)(start+bytes))return 0;
return result;
}
GC_INNER void GC_unmap(ptr_t start,size_t bytes)
{
ptr_t start_addr=GC_unmap_start(start,bytes);
ptr_t end_addr=GC_unmap_end(start,bytes);
word len=end_addr - start_addr;
if (0==start_addr)return;
#ifdef USE_WINALLOC
while (len!=0){
MEMORY_BASIC_INFORMATION mem_info;
word free_len;
if (VirtualQuery(start_addr,&mem_info,sizeof(mem_info))
!=sizeof(mem_info))
ABORT("Weird VirtualQuery result");
free_len=(len < mem_info.RegionSize)?len:mem_info.RegionSize;
if (!VirtualFree(start_addr,free_len,MEM_DECOMMIT))
ABORT("VirtualFree failed");
GC_unmapped_bytes+=free_len;
start_addr+=free_len;
len-=free_len;
}
#elif defined(SN_TARGET_PS3)
ps3_free_mem(start_addr,len);
#else
{
#if defined(AIX)||defined(CYGWIN32)||defined(HAIKU)||defined(HPUX)
if (mprotect(start_addr,len,PROT_NONE))
ABORT("mprotect(PROT_NONE)failed");
#elif defined(__EMSCRIPTEN__)
#else
void*result=mmap(start_addr,len,PROT_NONE,
MAP_PRIVATE|MAP_FIXED|OPT_MAP_ANON,
zero_fd,0);
if (result!=(void*)start_addr)
ABORT("mmap(PROT_NONE)failed");
#if defined(CPPCHECK)||defined(LINT2)
GC_noop1((word)result);
#endif
#endif
}
GC_unmapped_bytes+=len;
#endif
}
GC_INNER void GC_remap(ptr_t start,size_t bytes)
{
ptr_t start_addr=GC_unmap_start(start,bytes);
ptr_t end_addr=GC_unmap_end(start,bytes);
word len=end_addr - start_addr;
if (0==start_addr)return;
#ifdef USE_WINALLOC
while (len!=0){
MEMORY_BASIC_INFORMATION mem_info;
word alloc_len;
ptr_t result;
if (VirtualQuery(start_addr,&mem_info,sizeof(mem_info))
!=sizeof(mem_info))
ABORT("Weird VirtualQuery result");
alloc_len=(len < mem_info.RegionSize)?len:mem_info.RegionSize;
result=(ptr_t)VirtualAlloc(start_addr,alloc_len,MEM_COMMIT,
GC_pages_executable
?PAGE_EXECUTE_READWRITE
:PAGE_READWRITE);
if (result!=start_addr){
if (GetLastError()==ERROR_NOT_ENOUGH_MEMORY||
GetLastError()==ERROR_OUTOFMEMORY){
ABORT("Not enough memory to process remapping");
} else {
ABORT("VirtualAlloc remapping failed");
}
}
#ifdef LINT2
GC_noop1((word)result);
#endif
GC_unmapped_bytes-=alloc_len;
start_addr+=alloc_len;
len-=alloc_len;
}
#else
{
#if defined(NACL)||defined(NETBSD)
void*result=mmap(start_addr,len,(PROT_READ|PROT_WRITE)
|(GC_pages_executable?PROT_EXEC:0),
MAP_PRIVATE|MAP_FIXED|OPT_MAP_ANON,
zero_fd,0);
if (result!=(void*)start_addr)
ABORT("mmap as mprotect failed");
#if defined(CPPCHECK)||defined(LINT2)
GC_noop1((word)result);
#endif
#else
if (mprotect(start_addr,len,(PROT_READ|PROT_WRITE)
|(GC_pages_executable?PROT_EXEC:0))!=0){
ABORT_ARG3("mprotect remapping failed",
" at %p (length %lu),errcode=%d",
(void*)start_addr,(unsigned long)len,errno);
}
#endif
}
#undef IGNORE_PAGES_EXECUTABLE
GC_unmapped_bytes-=len;
#endif
}
GC_INNER void GC_unmap_gap(ptr_t start1,size_t bytes1,ptr_t start2,
size_t bytes2)
{
ptr_t start1_addr=GC_unmap_start(start1,bytes1);
ptr_t end1_addr=GC_unmap_end(start1,bytes1);
ptr_t start2_addr=GC_unmap_start(start2,bytes2);
ptr_t start_addr=end1_addr;
ptr_t end_addr=start2_addr;
size_t len;
GC_ASSERT(start1+bytes1==start2);
if (0==start1_addr)start_addr=GC_unmap_start(start1,bytes1+bytes2);
if (0==start2_addr)end_addr=GC_unmap_end(start1,bytes1+bytes2);
if (0==start_addr)return;
len=end_addr - start_addr;
#ifdef USE_WINALLOC
while (len!=0){
MEMORY_BASIC_INFORMATION mem_info;
word free_len;
if (VirtualQuery(start_addr,&mem_info,sizeof(mem_info))
!=sizeof(mem_info))
ABORT("Weird VirtualQuery result");
free_len=(len < mem_info.RegionSize)?len:mem_info.RegionSize;
if (!VirtualFree(start_addr,free_len,MEM_DECOMMIT))
ABORT("VirtualFree failed");
GC_unmapped_bytes+=free_len;
start_addr+=free_len;
len-=free_len;
}
#else
if (len!=0){
#if defined(AIX)||defined(CYGWIN32)||defined(HAIKU)||defined(HPUX)
if (mprotect(start_addr,len,PROT_NONE))
ABORT("mprotect(PROT_NONE)failed");
#else
void*result=mmap(start_addr,len,PROT_NONE,
MAP_PRIVATE|MAP_FIXED|OPT_MAP_ANON,
zero_fd,0);
if (result!=(void*)start_addr)
ABORT("mmap(PROT_NONE)failed");
#if defined(CPPCHECK)||defined(LINT2)
GC_noop1((word)result);
#endif
#endif
GC_unmapped_bytes+=len;
}
#endif
}
#endif
#ifndef THREADS
#if defined(__EMSCRIPTEN__)
static void scan_regs_cb(void*begin,void*end)
{
GC_push_all_stack((ptr_t)begin,(ptr_t)end);
}
STATIC void GC_CALLBACK GC_default_push_other_roots(void)
{
emscripten_scan_registers(scan_regs_cb);
}
#else
#define GC_default_push_other_roots 0
#endif
#else
#ifdef PCR
PCR_ERes GC_push_thread_stack(PCR_Th_T*t,PCR_Any dummy)
{
struct PCR_ThCtl_TInfoRep info;
PCR_ERes result;
info.ti_stkLow=info.ti_stkHi=0;
result=PCR_ThCtl_GetInfo(t,&info);
GC_push_all_stack((ptr_t)(info.ti_stkLow),(ptr_t)(info.ti_stkHi));
return(result);
}
PCR_ERes GC_push_old_obj(void*p,size_t size,PCR_Any data)
{
GC_push_all_stack((ptr_t)p,(ptr_t)p+size);
return(PCR_ERes_okay);
}
extern struct PCR_MM_ProcsRep*GC_old_allocator;
STATIC void GC_CALLBACK GC_default_push_other_roots(void)
{
if ((*(GC_old_allocator->mmp_enumerate))(PCR_Bool_false,
GC_push_old_obj,0)
!=PCR_ERes_okay){
ABORT("Old object enumeration failed");
}
if (PCR_ERes_IsErr(
PCR_ThCtl_ApplyToAllOtherThreads(GC_push_thread_stack,0))
||PCR_ERes_IsErr(GC_push_thread_stack(PCR_Th_CurrThread(),0))){
ABORT("Thread stack marking failed");
}
}
#elif defined(SN_TARGET_PS3)
STATIC void GC_CALLBACK GC_default_push_other_roots(void)
{
ABORT("GC_default_push_other_roots is not implemented");
}
void GC_push_thread_structures(void)
{
ABORT("GC_push_thread_structures is not implemented");
}
#else
STATIC void GC_CALLBACK GC_default_push_other_roots(void)
{
GC_push_all_stacks();
}
#endif
#endif
GC_push_other_roots_proc GC_push_other_roots=GC_default_push_other_roots;
GC_API void GC_CALL GC_set_push_other_roots(GC_push_other_roots_proc fn)
{
GC_push_other_roots=fn;
}
GC_API GC_push_other_roots_proc GC_CALL GC_get_push_other_roots(void)
{
return GC_push_other_roots;
}
#if (defined(CHECKSUMS)&&defined(GWW_VDB))||defined(PROC_VDB)
STATIC void GC_or_pages(page_hash_table pht1,page_hash_table pht2)
{
unsigned i;
for (i=0;i < PHT_SIZE;i++)pht1[i]|=pht2[i];
}
#endif
#ifdef GWW_VDB
#define GC_GWW_BUF_LEN (MAXHINCR*HBLKSIZE/4096)
static PVOID gww_buf[GC_GWW_BUF_LEN];
#ifndef MPROTECT_VDB
#define GC_gww_dirty_init GC_dirty_init
#endif
GC_INNER GC_bool GC_gww_dirty_init(void)
{
detect_GetWriteWatch();
return GC_GWW_AVAILABLE();
}
GC_INLINE void GC_gww_read_dirty(GC_bool output_unneeded)
{
word i;
if (!output_unneeded)
BZERO(GC_grungy_pages,sizeof(GC_grungy_pages));
for (i=0;i!=GC_n_heap_sects;++i){
GC_ULONG_PTR count;
do {
PVOID*pages=gww_buf;
DWORD page_size;
count=GC_GWW_BUF_LEN;
if ((*(GetWriteWatch_type)(word)GetWriteWatch_func)(
WRITE_WATCH_FLAG_RESET,
GC_heap_sects[i].hs_start,
GC_heap_sects[i].hs_bytes,
pages,&count,&page_size)!=0){
static int warn_count=0;
struct hblk*start=(struct hblk*)GC_heap_sects[i].hs_start;
static struct hblk*last_warned=0;
size_t nblocks=divHBLKSZ(GC_heap_sects[i].hs_bytes);
if (i!=0&&last_warned!=start&&warn_count++< 5){
last_warned=start;
WARN("GC_gww_read_dirty unexpectedly failed at %p:"
"Falling back to marking all pages dirty\n",start);
}
if (!output_unneeded){
unsigned j;
for (j=0;j < nblocks;++j){
word hash=PHT_HASH(start+j);
set_pht_entry_from_index(GC_grungy_pages,hash);
}
}
count=1;
} else if (!output_unneeded){
PVOID*pages_end=pages+count;
while (pages!=pages_end){
struct hblk*h=(struct hblk*)*pages++;
struct hblk*h_end=(struct hblk*)((char*)h+page_size);
do {
set_pht_entry_from_index(GC_grungy_pages,PHT_HASH(h));
} while ((word)(++h)< (word)h_end);
}
}
} while (count==GC_GWW_BUF_LEN);
}
#ifdef CHECKSUMS
GC_ASSERT(!output_unneeded);
GC_or_pages(GC_written_pages,GC_grungy_pages);
#endif
}
#else
#define GC_GWW_AVAILABLE()FALSE
#endif
#ifdef DEFAULT_VDB
GC_INNER GC_bool GC_dirty_init(void)
{
GC_VERBOSE_LOG_PRINTF("Initializing DEFAULT_VDB...\n");
return TRUE;
}
#endif
#ifndef GC_DISABLE_INCREMENTAL
#if!defined(THREADS)||defined(HAVE_LOCKFREE_AO_OR)
#define async_set_pht_entry_from_index(db,index)set_pht_entry_from_index_concurrent(db,index)
#elif defined(AO_HAVE_test_and_set_acquire)
GC_INNER volatile AO_TS_t GC_fault_handler_lock=AO_TS_INITIALIZER;
static void async_set_pht_entry_from_index(volatile page_hash_table db,
size_t index)
{
GC_acquire_dirty_lock();
set_pht_entry_from_index(db,index);
GC_release_dirty_lock();
}
#else
#error No test_and_set operation:Introduces a race.
#endif
#endif
#ifdef MPROTECT_VDB
#ifdef DARWIN
#include <mach/vm_map.h>
STATIC mach_port_t GC_task_self=0;
#define PROTECT(addr,len)if (vm_protect(GC_task_self,(vm_address_t)(addr),(vm_size_t)(len),FALSE,VM_PROT_READ|(GC_pages_executable?VM_PROT_EXECUTE:0))==KERN_SUCCESS){} else ABORT("vm_protect(PROTECT)failed")
#define UNPROTECT(addr,len)if (vm_protect(GC_task_self,(vm_address_t)(addr),(vm_size_t)(len),FALSE,(VM_PROT_READ|VM_PROT_WRITE)|(GC_pages_executable?VM_PROT_EXECUTE:0))==KERN_SUCCESS){} else ABORT("vm_protect(UNPROTECT)failed")
#elif!defined(USE_WINALLOC)
#include <sys/mman.h>
#include <signal.h>
#if!defined(CYGWIN32)&&!defined(HAIKU)
#include <sys/syscall.h>
#endif
#define PROTECT(addr,len)if (mprotect((caddr_t)(addr),(size_t)(len),PROT_READ|(GC_pages_executable?PROT_EXEC:0))>=0){ } else ABORT("mprotect failed")
#define UNPROTECT(addr,len)if (mprotect((caddr_t)(addr),(size_t)(len),(PROT_READ|PROT_WRITE)|(GC_pages_executable?PROT_EXEC:0))>=0){ } else ABORT(GC_pages_executable?"un-mprotect executable page failed" " (probably disabled by OS)":"un-mprotect failed")
#undef IGNORE_PAGES_EXECUTABLE
#else
#ifndef MSWINCE
#include <signal.h>
#endif
static DWORD protect_junk;
#define PROTECT(addr,len)if (VirtualProtect((addr),(len),GC_pages_executable?PAGE_EXECUTE_READ:PAGE_READONLY,&protect_junk)){ } else ABORT_ARG1("VirtualProtect failed",":errcode=0x%X",(unsigned)GetLastError())
#define UNPROTECT(addr,len)if (VirtualProtect((addr),(len),GC_pages_executable?PAGE_EXECUTE_READWRITE:PAGE_READWRITE,&protect_junk)){ } else ABORT("un-VirtualProtect failed")
#endif
#if defined(MSWIN32)
typedef LPTOP_LEVEL_EXCEPTION_FILTER SIG_HNDLR_PTR;
#undef SIG_DFL
#define SIG_DFL (LPTOP_LEVEL_EXCEPTION_FILTER)((signed_word)-1)
#elif defined(MSWINCE)
typedef LONG (WINAPI*SIG_HNDLR_PTR)(struct _EXCEPTION_POINTERS*);
#undef SIG_DFL
#define SIG_DFL (SIG_HNDLR_PTR)(-1)
#elif defined(DARWIN)
typedef void (*SIG_HNDLR_PTR)();
#else
typedef void (*SIG_HNDLR_PTR)(int,siginfo_t*,void*);
typedef void (*PLAIN_HNDLR_PTR)(int);
#endif
#if defined(__GLIBC__)
#if __GLIBC__ < 2||__GLIBC__==2&&__GLIBC_MINOR__ < 2
#error glibc too old?
#endif
#endif
#ifndef DARWIN
STATIC SIG_HNDLR_PTR GC_old_segv_handler=0;
#if defined(FREEBSD)||defined(HPUX)||defined(HURD)||defined(LINUX)
STATIC SIG_HNDLR_PTR GC_old_bus_handler=0;
#ifndef LINUX
STATIC GC_bool GC_old_bus_handler_used_si=FALSE;
#endif
#endif
#if!defined(MSWIN32)&&!defined(MSWINCE)
STATIC GC_bool GC_old_segv_handler_used_si=FALSE;
#endif
#endif
#ifdef THREADS
GC_ATTR_NO_SANITIZE_THREAD
static GC_bool is_header_found_async(void*addr)
{
#ifdef HASH_TL
hdr*result;
GET_HDR((ptr_t)addr,result);
return result!=NULL;
#else
return HDR_INNER(addr)!=NULL;
#endif
}
#else
#define is_header_found_async(addr)(HDR(addr)!=NULL)
#endif
#ifndef DARWIN
#if!defined(MSWIN32)&&!defined(MSWINCE)
#include <errno.h>
#if defined(FREEBSD)||defined(HURD)||defined(HPUX)
#define SIG_OK (sig==SIGBUS||sig==SIGSEGV)
#else
#define SIG_OK (sig==SIGSEGV)
#endif
#if defined(FREEBSD)
#ifndef SEGV_ACCERR
#define SEGV_ACCERR 2
#endif
#if defined(AARCH64)||defined(ARM32)||defined(MIPS)
#define CODE_OK (si->si_code==SEGV_ACCERR)
#elif defined(POWERPC)
#define AIM
#include <machine/trap.h>
#define CODE_OK (si->si_code==EXC_DSI||si->si_code==SEGV_ACCERR)
#else
#define CODE_OK (si->si_code==BUS_PAGE_FAULT||si->si_code==SEGV_ACCERR)
#endif
#elif defined(OSF1)
#define CODE_OK (si->si_code==2)
#elif defined(IRIX5)
#define CODE_OK (si->si_code==EACCES)
#elif defined(CYGWIN32)||defined(HAIKU)||defined(HURD)
#define CODE_OK TRUE
#elif defined(LINUX)
#define CODE_OK TRUE
#elif defined(HPUX)
#define CODE_OK (si->si_code==SEGV_ACCERR||si->si_code==BUS_ADRERR||si->si_code==BUS_UNKNOWN||si->si_code==SEGV_UNKNOWN||si->si_code==BUS_OBJERR)
#elif defined(SUNOS5SIGS)
#define CODE_OK (si->si_code==SEGV_ACCERR)
#endif
#ifndef NO_GETCONTEXT
#include <ucontext.h>
#endif
STATIC void GC_write_fault_handler(int sig,siginfo_t*si,void*raw_sc)
#else
#define SIG_OK (exc_info->ExceptionRecord->ExceptionCode==STATUS_ACCESS_VIOLATION)
#define CODE_OK (exc_info->ExceptionRecord->ExceptionInformation[0]==1)
STATIC LONG WINAPI GC_write_fault_handler(
struct _EXCEPTION_POINTERS*exc_info)
#endif
{
#if!defined(MSWIN32)&&!defined(MSWINCE)
char*addr=(char*)si->si_addr;
#else
char*addr=(char*)(exc_info->ExceptionRecord
->ExceptionInformation[1]);
#endif
if (SIG_OK&&CODE_OK){
struct hblk*h=(struct hblk*)((word)addr&~(GC_page_size-1));
GC_bool in_allocd_block;
size_t i;
GC_ASSERT(GC_page_size!=0);
#ifdef CHECKSUMS
GC_record_fault(h);
#endif
#ifdef SUNOS5SIGS
in_allocd_block=FALSE;
for (i=0;i < divHBLKSZ(GC_page_size);i++){
if (is_header_found_async(&h[i])){
in_allocd_block=TRUE;
break;
}
}
#else
in_allocd_block=is_header_found_async(addr);
#endif
if (!in_allocd_block){
SIG_HNDLR_PTR old_handler;
#if defined(MSWIN32)||defined(MSWINCE)
old_handler=GC_old_segv_handler;
#else
GC_bool used_si;
#if defined(FREEBSD)||defined(HURD)||defined(HPUX)
if (sig==SIGBUS){
old_handler=GC_old_bus_handler;
used_si=GC_old_bus_handler_used_si;
} else
#endif
{
old_handler=GC_old_segv_handler;
used_si=GC_old_segv_handler_used_si;
}
#endif
if (old_handler==(SIG_HNDLR_PTR)(signed_word)SIG_DFL){
#if!defined(MSWIN32)&&!defined(MSWINCE)
ABORT_ARG1("Unexpected bus error or segmentation fault",
" at %p",(void*)addr);
#else
return(EXCEPTION_CONTINUE_SEARCH);
#endif
} else {
#if defined(MSWIN32)||defined(MSWINCE)
return((*old_handler)(exc_info));
#else
if (used_si)
((SIG_HNDLR_PTR)old_handler)(sig,si,raw_sc);
else
((PLAIN_HNDLR_PTR)(signed_word)old_handler)(sig);
return;
#endif
}
}
UNPROTECT(h,GC_page_size);
for (i=0;i < divHBLKSZ(GC_page_size);i++){
word index=PHT_HASH(h+i);
async_set_pht_entry_from_index(GC_dirty_pages,index);
}
#if defined(MSWIN32)||defined(MSWINCE)
return(EXCEPTION_CONTINUE_EXECUTION);
#else
return;
#endif
}
#if defined(MSWIN32)||defined(MSWINCE)
return EXCEPTION_CONTINUE_SEARCH;
#else
ABORT_ARG1("Unexpected bus error or segmentation fault",
" at %p",(void*)addr);
#endif
}
#if defined(GC_WIN32_THREADS)&&!defined(CYGWIN32)
GC_INNER void GC_set_write_fault_handler(void)
{
SetUnhandledExceptionFilter(GC_write_fault_handler);
}
#endif
#endif
#if!defined(DARWIN)
GC_INNER GC_bool GC_dirty_init(void)
{
#if!defined(MSWIN32)&&!defined(MSWINCE)
struct sigaction act,oldact;
act.sa_flags=SA_RESTART|SA_SIGINFO;
act.sa_sigaction=GC_write_fault_handler;
(void)sigemptyset(&act.sa_mask);
#if defined(THREADS)&&!defined(GC_OPENBSD_UTHREADS)&&!defined(GC_WIN32_THREADS)&&!defined(NACL)
(void)sigaddset(&act.sa_mask,GC_get_suspend_signal());
#endif
#endif
GC_VERBOSE_LOG_PRINTF(
"Initializing mprotect virtual dirty bit implementation\n");
if (GC_page_size % HBLKSIZE!=0){
ABORT("Page size not multiple of HBLKSIZE");
}
#if!defined(MSWIN32)&&!defined(MSWINCE)
#if defined(GC_IRIX_THREADS)
sigaction(SIGSEGV,0,&oldact);
sigaction(SIGSEGV,&act,0);
#else
{
int res=sigaction(SIGSEGV,&act,&oldact);
if (res!=0)ABORT("Sigaction failed");
}
#endif
if (oldact.sa_flags&SA_SIGINFO){
GC_old_segv_handler=oldact.sa_sigaction;
GC_old_segv_handler_used_si=TRUE;
} else {
GC_old_segv_handler=(SIG_HNDLR_PTR)(signed_word)oldact.sa_handler;
GC_old_segv_handler_used_si=FALSE;
}
if (GC_old_segv_handler==(SIG_HNDLR_PTR)(signed_word)SIG_IGN){
WARN("Previously ignored segmentation violation!?\n",0);
GC_old_segv_handler=(SIG_HNDLR_PTR)(signed_word)SIG_DFL;
}
if (GC_old_segv_handler!=(SIG_HNDLR_PTR)(signed_word)SIG_DFL){
GC_VERBOSE_LOG_PRINTF("Replaced other SIGSEGV handler\n");
}
#if defined(HPUX)||defined(LINUX)||defined(HURD)||(defined(FREEBSD)&&(defined(__GLIBC__)||defined(SUNOS5SIGS)))
sigaction(SIGBUS,&act,&oldact);
if ((oldact.sa_flags&SA_SIGINFO)!=0){
GC_old_bus_handler=oldact.sa_sigaction;
#if!defined(LINUX)
GC_old_bus_handler_used_si=TRUE;
#endif
} else {
GC_old_bus_handler=(SIG_HNDLR_PTR)(signed_word)oldact.sa_handler;
}
if (GC_old_bus_handler==(SIG_HNDLR_PTR)(signed_word)SIG_IGN){
WARN("Previously ignored bus error!?\n",0);
#if!defined(LINUX)
GC_old_bus_handler=(SIG_HNDLR_PTR)(signed_word)SIG_DFL;
#else
#endif
} else if (GC_old_bus_handler!=(SIG_HNDLR_PTR)(signed_word)SIG_DFL){
GC_VERBOSE_LOG_PRINTF("Replaced other SIGBUS handler\n");
}
#endif
#endif
#if defined(GWW_VDB)
if (GC_gww_dirty_init())
return TRUE;
#endif
#if defined(MSWIN32)
GC_old_segv_handler=SetUnhandledExceptionFilter(GC_write_fault_handler);
if (GC_old_segv_handler!=NULL){
GC_COND_LOG_PRINTF("Replaced other UnhandledExceptionFilter\n");
} else {
GC_old_segv_handler=SIG_DFL;
}
#elif defined(MSWINCE)
#endif
#if defined(CPPCHECK)&&defined(ADDRESS_SANITIZER)
GC_noop1((word)&__asan_default_options);
#endif
return TRUE;
}
#endif
GC_API int GC_CALL GC_incremental_protection_needs(void)
{
GC_ASSERT(GC_is_initialized);
if (GC_page_size==HBLKSIZE){
return GC_PROTECTS_POINTER_HEAP;
} else {
return GC_PROTECTS_POINTER_HEAP|GC_PROTECTS_PTRFREE_HEAP;
}
}
#define HAVE_INCREMENTAL_PROTECTION_NEEDS
#define IS_PTRFREE(hhdr)((hhdr)->hb_descr==0)
#define PAGE_ALIGNED(x)!((word)(x)&(GC_page_size - 1))
STATIC void GC_protect_heap(void)
{
unsigned i;
GC_bool protect_all=
(0!=(GC_incremental_protection_needs()&GC_PROTECTS_PTRFREE_HEAP));
GC_ASSERT(GC_page_size!=0);
for (i=0;i < GC_n_heap_sects;i++){
ptr_t start=GC_heap_sects[i].hs_start;
size_t len=GC_heap_sects[i].hs_bytes;
if (protect_all){
PROTECT(start,len);
} else {
struct hblk*current;
struct hblk*current_start;
struct hblk*limit;
GC_ASSERT(PAGE_ALIGNED(len));
GC_ASSERT(PAGE_ALIGNED(start));
current_start=current=(struct hblk*)start;
limit=(struct hblk*)(start+len);
while ((word)current < (word)limit){
hdr*hhdr;
word nhblks;
GC_bool is_ptrfree;
GC_ASSERT(PAGE_ALIGNED(current));
GET_HDR(current,hhdr);
if (IS_FORWARDING_ADDR_OR_NIL(hhdr)){
GC_ASSERT(current_start==current);
current_start=++current;
continue;
}
if (HBLK_IS_FREE(hhdr)){
GC_ASSERT(PAGE_ALIGNED(hhdr->hb_sz));
nhblks=divHBLKSZ(hhdr->hb_sz);
is_ptrfree=TRUE;
} else {
nhblks=OBJ_SZ_TO_BLOCKS(hhdr->hb_sz);
is_ptrfree=IS_PTRFREE(hhdr);
}
if (is_ptrfree){
if ((word)current_start < (word)current){
PROTECT(current_start,(ptr_t)current - (ptr_t)current_start);
}
current_start=(current+=nhblks);
} else {
current+=nhblks;
}
}
if ((word)current_start < (word)current){
PROTECT(current_start,(ptr_t)current - (ptr_t)current_start);
}
}
}
}
#endif
#ifdef PROC_VDB
#include <errno.h>
#include <sys/types.h>
#include <sys/signal.h>
#include <sys/syscall.h>
#include <sys/stat.h>
#ifdef GC_NO_SYS_FAULT_H
#define PG_MODIFIED 1
struct prpageheader {
int dummy[2];
unsigned long pr_nmap;
unsigned long pr_npage;
};
struct prasmap {
char*pr_vaddr;
size_t pr_npage;
char dummy1[64+8];
unsigned pr_mflags;
unsigned pr_pagesize;
int dummy2[2];
};
#else
#include <sys/fault.h>
#include <sys/procfs.h>
#endif
#define INITIAL_BUF_SZ 16384
STATIC size_t GC_proc_buf_size=INITIAL_BUF_SZ;
STATIC char*GC_proc_buf=NULL;
STATIC int GC_proc_fd=0;
GC_INNER GC_bool GC_dirty_init(void)
{
char buf[40];
if (GC_bytes_allocd!=0||GC_bytes_allocd_before_gc!=0){
memset(GC_written_pages,0xff,sizeof(page_hash_table));
GC_VERBOSE_LOG_PRINTF(
"Allocated %lu bytes:all pages may have been written\n",
(unsigned long)(GC_bytes_allocd+GC_bytes_allocd_before_gc));
}
(void)snprintf(buf,sizeof(buf),"/proc/%ld/pagedata",(long)getpid());
buf[sizeof(buf)- 1]='\0';
GC_proc_fd=open(buf,O_RDONLY);
if (GC_proc_fd < 0){
WARN("/proc open failed;cannot enable GC incremental mode\n",0);
return FALSE;
}
if (syscall(SYS_fcntl,GC_proc_fd,F_SETFD,FD_CLOEXEC)==-1)
WARN("Could not set FD_CLOEXEC for/proc\n",0);
GC_proc_buf=GC_scratch_alloc(GC_proc_buf_size);
if (GC_proc_buf==NULL)
ABORT("Insufficient space for/proc read");
return TRUE;
}
GC_INLINE void GC_proc_read_dirty(GC_bool output_unneeded)
{
#define READ read
int nmaps;
char*bufp=GC_proc_buf;
int i;
BZERO(GC_grungy_pages,sizeof(GC_grungy_pages));
if (READ(GC_proc_fd,bufp,GC_proc_buf_size)<=0){
size_t new_size=2*GC_proc_buf_size;
char*new_buf;
WARN("/proc read failed:GC_proc_buf_size=%" WARN_PRIdPTR "\n",
(signed_word)GC_proc_buf_size);
new_buf=GC_scratch_alloc(new_size);
if (new_buf!=0){
GC_scratch_recycle_no_gww(bufp,GC_proc_buf_size);
GC_proc_buf=bufp=new_buf;
GC_proc_buf_size=new_size;
}
if (READ(GC_proc_fd,bufp,GC_proc_buf_size)<=0){
WARN("Insufficient space for/proc read\n",0);
if (!output_unneeded)
memset(GC_grungy_pages,0xff,sizeof (page_hash_table));
memset(GC_written_pages,0xff,sizeof(page_hash_table));
return;
}
}
nmaps=((struct prpageheader*)bufp)->pr_nmap;
#ifdef DEBUG_DIRTY_BITS
GC_log_printf("Proc VDB read:pr_nmap=%u,pr_npage=%lu\n",
nmaps,((struct prpageheader*)bufp)->pr_npage);
#endif
#if defined(GC_NO_SYS_FAULT_H)&&defined(CPPCHECK)
GC_noop1(((struct prpageheader*)bufp)->dummy[0]);
#endif
bufp+=sizeof(struct prpageheader);
for (i=0;i < nmaps;i++){
struct prasmap*map=(struct prasmap*)bufp;
ptr_t vaddr=(ptr_t)(map->pr_vaddr);
unsigned long npages=map->pr_npage;
unsigned pagesize=map->pr_pagesize;
ptr_t limit;
#if defined(GC_NO_SYS_FAULT_H)&&defined(CPPCHECK)
GC_noop1(map->dummy1[0]+map->dummy2[0]);
#endif
#ifdef DEBUG_DIRTY_BITS
GC_log_printf(
"pr_vaddr=%p,npage=%lu,mflags=0x%x,pagesize=0x%x\n",
(void*)vaddr,npages,map->pr_mflags,pagesize);
#endif
bufp+=sizeof(struct prasmap);
limit=vaddr+pagesize*npages;
for (;(word)vaddr < (word)limit;vaddr+=pagesize){
if ((*bufp++)&PG_MODIFIED){
struct hblk*h;
ptr_t next_vaddr=vaddr+pagesize;
#ifdef DEBUG_DIRTY_BITS
GC_log_printf("dirty page at:%p\n",(void*)vaddr);
#endif
for (h=(struct hblk*)vaddr;
(word)h < (word)next_vaddr;h++){
word index=PHT_HASH(h);
set_pht_entry_from_index(GC_grungy_pages,index);
}
}
}
bufp=(char*)(((word)bufp+(sizeof(long)-1))
&~(word)(sizeof(long)-1));
}
#ifdef DEBUG_DIRTY_BITS
GC_log_printf("Proc VDB read done\n");
#endif
GC_or_pages(GC_written_pages,GC_grungy_pages);
#undef READ
}
#endif
#ifdef PCR_VDB
#include "vd/PCR_VD.h"
#define NPAGES (32*1024)
PCR_VD_DB GC_grungy_bits[NPAGES];
STATIC ptr_t GC_vd_base=NULL;
GC_INNER GC_bool GC_dirty_init(void)
{
GC_vd_base=GC_heap_sects[0].hs_start;
if (GC_vd_base==0){
ABORT("Bad initial heap segment");
}
if (PCR_VD_Start(HBLKSIZE,GC_vd_base,NPAGES*HBLKSIZE)
!=PCR_ERes_okay){
ABORT("Dirty bit initialization failed");
}
return TRUE;
}
#endif
#ifndef GC_DISABLE_INCREMENTAL
GC_INNER GC_bool GC_manual_vdb=FALSE;
GC_INNER void GC_dirty_inner(const void*p)
{
word index=PHT_HASH(p);
#if defined(MPROTECT_VDB)
GC_ASSERT(GC_manual_vdb);
#endif
async_set_pht_entry_from_index(GC_dirty_pages,index);
}
GC_INNER void GC_read_dirty(GC_bool output_unneeded)
{
if (GC_manual_vdb
#if defined(MPROTECT_VDB)
||!GC_GWW_AVAILABLE()
#endif
){
if (!output_unneeded)
BCOPY(( void*)GC_dirty_pages,GC_grungy_pages,
sizeof(GC_dirty_pages));
BZERO(( void*)GC_dirty_pages,
sizeof(GC_dirty_pages));
#ifdef MPROTECT_VDB
if (!GC_manual_vdb)
GC_protect_heap();
#endif
return;
}
#ifdef GWW_VDB
GC_gww_read_dirty(output_unneeded);
#elif defined(PROC_VDB)
GC_proc_read_dirty(output_unneeded);
#elif defined(PCR_VDB)
{
static int onhs=0;
int nhs=GC_n_heap_sects;
for (;onhs < nhs;onhs++){
PCR_VD_WriteProtectEnable(
GC_heap_sects[onhs].hs_start,
GC_heap_sects[onhs].hs_bytes);
}
}
if (PCR_VD_Clear(GC_vd_base,NPAGES*HBLKSIZE,GC_grungy_bits)
!=PCR_ERes_okay){
ABORT("Dirty bit read failed");
}
#endif
}
GC_INNER GC_bool GC_page_was_dirty(struct hblk*h)
{
#ifdef PCR_VDB
if (!GC_manual_vdb){
if ((word)h < (word)GC_vd_base
||(word)h>=(word)(GC_vd_base+NPAGES*HBLKSIZE)){
return TRUE;
}
return GC_grungy_bits[h-(struct hblk*)GC_vd_base]&PCR_VD_DB_dirtyBit;
}
#elif defined(DEFAULT_VDB)
if (!GC_manual_vdb)
return TRUE;
#endif
return NULL==HDR(h)
||get_pht_entry_from_index(GC_grungy_pages,PHT_HASH(h));
}
#if defined(CHECKSUMS)||defined(PROC_VDB)
GC_INNER GC_bool GC_page_was_ever_dirty(struct hblk*h)
{
#if defined(GWW_VDB)||defined(PROC_VDB)
#ifdef MPROTECT_VDB
if (!GC_GWW_AVAILABLE())
return TRUE;
#endif
return NULL==HDR(h)
||get_pht_entry_from_index(GC_written_pages,PHT_HASH(h));
#else
(void)h;
return TRUE;
#endif
}
#endif
GC_INNER void GC_remove_protection(struct hblk*h,word nblocks,
GC_bool is_ptrfree)
{
#ifdef PCR_VDB
(void)is_ptrfree;
if (!GC_auto_incremental)
return;
PCR_VD_WriteProtectDisable(h,nblocks*HBLKSIZE);
PCR_VD_WriteProtectEnable(h,nblocks*HBLKSIZE);
#elif defined(MPROTECT_VDB)
struct hblk*h_trunc;
struct hblk*h_end;
struct hblk*current;
if (!GC_auto_incremental||GC_GWW_AVAILABLE())
return;
GC_ASSERT(GC_page_size!=0);
h_trunc=(struct hblk*)((word)h&~(GC_page_size-1));
h_end=(struct hblk*)(((word)(h+nblocks)+GC_page_size - 1)
&~(GC_page_size - 1));
if (h_end==h_trunc+1&&
get_pht_entry_from_index(GC_dirty_pages,PHT_HASH(h_trunc))){
return;
}
for (current=h_trunc;(word)current < (word)h_end;++current){
word index=PHT_HASH(current);
if (!is_ptrfree||(word)current < (word)h
||(word)current>=(word)(h+nblocks)){
async_set_pht_entry_from_index(GC_dirty_pages,index);
}
}
UNPROTECT(h_trunc,(ptr_t)h_end - (ptr_t)h_trunc);
#else
(void)h;(void)nblocks;(void)is_ptrfree;
#endif
}
#endif
#if defined(MPROTECT_VDB)&&defined(DARWIN)
#include <mach/mach.h>
#include <mach/mach_error.h>
#include <mach/exception.h>
#include <mach/task.h>
#include <pthread.h>
EXTERN_C_BEGIN
extern boolean_t
exc_server(mach_msg_header_t*,mach_msg_header_t*);
extern kern_return_t
exception_raise(mach_port_t,mach_port_t,mach_port_t,exception_type_t,
exception_data_t,mach_msg_type_number_t);
extern kern_return_t
exception_raise_state(mach_port_t,mach_port_t,mach_port_t,exception_type_t,
exception_data_t,mach_msg_type_number_t,
thread_state_flavor_t*,thread_state_t,
mach_msg_type_number_t,thread_state_t,
mach_msg_type_number_t*);
extern kern_return_t
exception_raise_state_identity(mach_port_t,mach_port_t,mach_port_t,
exception_type_t,exception_data_t,
mach_msg_type_number_t,thread_state_flavor_t*,
thread_state_t,mach_msg_type_number_t,
thread_state_t,mach_msg_type_number_t*);
GC_API_OSCALL kern_return_t
catch_exception_raise(mach_port_t exception_port,mach_port_t thread,
mach_port_t task,exception_type_t exception,
exception_data_t code,
mach_msg_type_number_t code_count);
GC_API_OSCALL kern_return_t
catch_exception_raise_state(mach_port_name_t exception_port,
int exception,exception_data_t code,
mach_msg_type_number_t codeCnt,int flavor,
thread_state_t old_state,int old_stateCnt,
thread_state_t new_state,int new_stateCnt);
GC_API_OSCALL kern_return_t
catch_exception_raise_state_identity(mach_port_name_t exception_port,
mach_port_t thread,mach_port_t task,int exception,
exception_data_t code,mach_msg_type_number_t codeCnt,
int flavor,thread_state_t old_state,int old_stateCnt,
thread_state_t new_state,int new_stateCnt);
EXTERN_C_END
GC_API_OSCALL kern_return_t
catch_exception_raise_state(mach_port_name_t exception_port GC_ATTR_UNUSED,
int exception GC_ATTR_UNUSED,exception_data_t code GC_ATTR_UNUSED,
mach_msg_type_number_t codeCnt GC_ATTR_UNUSED,int flavor GC_ATTR_UNUSED,
thread_state_t old_state GC_ATTR_UNUSED,int old_stateCnt GC_ATTR_UNUSED,
thread_state_t new_state GC_ATTR_UNUSED,int new_stateCnt GC_ATTR_UNUSED)
{
ABORT_RET("Unexpected catch_exception_raise_state invocation");
return(KERN_INVALID_ARGUMENT);
}
GC_API_OSCALL kern_return_t
catch_exception_raise_state_identity(
mach_port_name_t exception_port GC_ATTR_UNUSED,
mach_port_t thread GC_ATTR_UNUSED,mach_port_t task GC_ATTR_UNUSED,
int exception GC_ATTR_UNUSED,exception_data_t code GC_ATTR_UNUSED,
mach_msg_type_number_t codeCnt GC_ATTR_UNUSED,int flavor GC_ATTR_UNUSED,
thread_state_t old_state GC_ATTR_UNUSED,int old_stateCnt GC_ATTR_UNUSED,
thread_state_t new_state GC_ATTR_UNUSED,int new_stateCnt GC_ATTR_UNUSED)
{
ABORT_RET("Unexpected catch_exception_raise_state_identity invocation");
return(KERN_INVALID_ARGUMENT);
}
#define MAX_EXCEPTION_PORTS 16
static struct {
mach_msg_type_number_t count;
exception_mask_t masks[MAX_EXCEPTION_PORTS];
exception_handler_t ports[MAX_EXCEPTION_PORTS];
exception_behavior_t behaviors[MAX_EXCEPTION_PORTS];
thread_state_flavor_t flavors[MAX_EXCEPTION_PORTS];
} GC_old_exc_ports;
STATIC struct ports_s {
void (*volatile os_callback[3])(void);
mach_port_t exception;
#if defined(THREADS)
mach_port_t reply;
#endif
} GC_ports={
{
(void (*)(void))catch_exception_raise,
(void (*)(void))catch_exception_raise_state,
(void (*)(void))catch_exception_raise_state_identity
},
#ifdef THREADS
0,
#endif
0
};
typedef struct {
mach_msg_header_t head;
} GC_msg_t;
typedef enum {
GC_MP_NORMAL,
GC_MP_DISCARDING,
GC_MP_STOPPED
} GC_mprotect_state_t;
#ifdef THREADS
#define ID_STOP 1
#define ID_RESUME 2
#define ID_ACK 3
STATIC GC_mprotect_state_t GC_mprotect_state=GC_MP_NORMAL;
STATIC void GC_mprotect_thread_notify(mach_msg_id_t id)
{
struct buf_s {
GC_msg_t msg;
mach_msg_trailer_t trailer;
} buf;
mach_msg_return_t r;
buf.msg.head.msgh_bits=MACH_MSGH_BITS(MACH_MSG_TYPE_MAKE_SEND,0);
buf.msg.head.msgh_size=sizeof(buf.msg);
buf.msg.head.msgh_remote_port=GC_ports.exception;
buf.msg.head.msgh_local_port=MACH_PORT_NULL;
buf.msg.head.msgh_id=id;
r=mach_msg(&buf.msg.head,MACH_SEND_MSG|MACH_RCV_MSG|MACH_RCV_LARGE,
sizeof(buf.msg),sizeof(buf),GC_ports.reply,
MACH_MSG_TIMEOUT_NONE,MACH_PORT_NULL);
if (r!=MACH_MSG_SUCCESS)
ABORT("mach_msg failed in GC_mprotect_thread_notify");
if (buf.msg.head.msgh_id!=ID_ACK)
ABORT("Invalid ack in GC_mprotect_thread_notify");
}
STATIC void GC_mprotect_thread_reply(void)
{
GC_msg_t msg;
mach_msg_return_t r;
msg.head.msgh_bits=MACH_MSGH_BITS(MACH_MSG_TYPE_MAKE_SEND,0);
msg.head.msgh_size=sizeof(msg);
msg.head.msgh_remote_port=GC_ports.reply;
msg.head.msgh_local_port=MACH_PORT_NULL;
msg.head.msgh_id=ID_ACK;
r=mach_msg(&msg.head,MACH_SEND_MSG,sizeof(msg),0,MACH_PORT_NULL,
MACH_MSG_TIMEOUT_NONE,MACH_PORT_NULL);
if (r!=MACH_MSG_SUCCESS)
ABORT("mach_msg failed in GC_mprotect_thread_reply");
}
GC_INNER void GC_mprotect_stop(void)
{
GC_mprotect_thread_notify(ID_STOP);
}
GC_INNER void GC_mprotect_resume(void)
{
GC_mprotect_thread_notify(ID_RESUME);
}
#else
#define GC_mprotect_state GC_MP_NORMAL
#endif
struct mp_reply_s {
mach_msg_header_t head;
char data[256];
};
struct mp_msg_s {
mach_msg_header_t head;
mach_msg_body_t msgh_body;
char data[1024];
};
STATIC void*GC_mprotect_thread(void*arg)
{
mach_msg_return_t r;
struct mp_reply_s reply;
struct mp_msg_s msg;
mach_msg_id_t id;
if ((word)arg==GC_WORD_MAX)return 0;
#if defined(CPPCHECK)
reply.data[0]=0;
msg.data[0]=0;
#endif
#if defined(HAVE_PTHREAD_SETNAME_NP_WITHOUT_TID)
(void)pthread_setname_np("GC-mprotect");
#endif
#if defined(THREADS)&&!defined(GC_NO_THREADS_DISCOVERY)
GC_darwin_register_mach_handler_thread(mach_thread_self());
#endif
for(;;){
r=mach_msg(&msg.head,MACH_RCV_MSG|MACH_RCV_LARGE|
(GC_mprotect_state==GC_MP_DISCARDING?MACH_RCV_TIMEOUT:0),
0,sizeof(msg),GC_ports.exception,
GC_mprotect_state==GC_MP_DISCARDING?0
:MACH_MSG_TIMEOUT_NONE,MACH_PORT_NULL);
id=r==MACH_MSG_SUCCESS?msg.head.msgh_id:-1;
#if defined(THREADS)
if(GC_mprotect_state==GC_MP_DISCARDING){
if(r==MACH_RCV_TIMED_OUT){
GC_mprotect_state=GC_MP_STOPPED;
GC_mprotect_thread_reply();
continue;
}
if(r==MACH_MSG_SUCCESS&&(id==ID_STOP||id==ID_RESUME))
ABORT("Out of order mprotect thread request");
}
#endif
if (r!=MACH_MSG_SUCCESS){
ABORT_ARG2("mach_msg failed",
":errcode=%d (%s)",(int)r,mach_error_string(r));
}
switch(id){
#if defined(THREADS)
case ID_STOP:
if(GC_mprotect_state!=GC_MP_NORMAL)
ABORT("Called mprotect_stop when state wasn't normal");
GC_mprotect_state=GC_MP_DISCARDING;
break;
case ID_RESUME:
if(GC_mprotect_state!=GC_MP_STOPPED)
ABORT("Called mprotect_resume when state wasn't stopped");
GC_mprotect_state=GC_MP_NORMAL;
GC_mprotect_thread_reply();
break;
#endif
default:
if(!exc_server(&msg.head,&reply.head))
ABORT("exc_server failed");
r=mach_msg(&reply.head,MACH_SEND_MSG,reply.head.msgh_size,0,
MACH_PORT_NULL,MACH_MSG_TIMEOUT_NONE,
MACH_PORT_NULL);
if(r!=MACH_MSG_SUCCESS){
#ifdef BROKEN_EXCEPTION_HANDLING
GC_err_printf("mach_msg failed with %d %s while sending "
"exc reply\n",(int)r,mach_error_string(r));
#else
ABORT("mach_msg failed while sending exception reply");
#endif
}
}
}
}
#ifdef BROKEN_EXCEPTION_HANDLING
STATIC int GC_sigbus_count=0;
STATIC void GC_darwin_sigbus(int num,siginfo_t*sip,void*context)
{
if (num!=SIGBUS)
ABORT("Got a non-sigbus signal in the sigbus handler");
if (GC_sigbus_count>=8){
ABORT("Got more than 8 SIGBUSs in a row!");
} else {
GC_sigbus_count++;
WARN("Ignoring SIGBUS\n",0);
}
}
#endif
GC_INNER GC_bool GC_dirty_init(void)
{
kern_return_t r;
mach_port_t me;
pthread_t thread;
pthread_attr_t attr;
exception_mask_t mask;
#ifdef CAN_HANDLE_FORK
if (GC_handle_fork){
WARN("Can't turn on GC incremental mode as fork()"
" handling requested\n",0);
return FALSE;
}
#endif
GC_VERBOSE_LOG_PRINTF("Initializing mach/darwin mprotect"
" virtual dirty bit implementation\n");
#ifdef BROKEN_EXCEPTION_HANDLING
WARN("Enabling workarounds for various darwin "
"exception handling bugs\n",0);
#endif
if (GC_page_size % HBLKSIZE!=0){
ABORT("Page size not multiple of HBLKSIZE");
}
GC_task_self=me=mach_task_self();
r=mach_port_allocate(me,MACH_PORT_RIGHT_RECEIVE,&GC_ports.exception);
if (r!=KERN_SUCCESS)
ABORT("mach_port_allocate failed (exception port)");
r=mach_port_insert_right(me,GC_ports.exception,GC_ports.exception,
MACH_MSG_TYPE_MAKE_SEND);
if (r!=KERN_SUCCESS)
ABORT("mach_port_insert_right failed (exception port)");
#if defined(THREADS)
r=mach_port_allocate(me,MACH_PORT_RIGHT_RECEIVE,&GC_ports.reply);
if(r!=KERN_SUCCESS)
ABORT("mach_port_allocate failed (reply port)");
#endif
mask=EXC_MASK_BAD_ACCESS;
r=task_get_exception_ports(me,mask,GC_old_exc_ports.masks,
&GC_old_exc_ports.count,GC_old_exc_ports.ports,
GC_old_exc_ports.behaviors,
GC_old_exc_ports.flavors);
if (r!=KERN_SUCCESS)
ABORT("task_get_exception_ports failed");
r=task_set_exception_ports(me,mask,GC_ports.exception,EXCEPTION_DEFAULT,
GC_MACH_THREAD_STATE);
if (r!=KERN_SUCCESS)
ABORT("task_set_exception_ports failed");
if (pthread_attr_init(&attr)!=0)
ABORT("pthread_attr_init failed");
if (pthread_attr_setdetachstate(&attr,PTHREAD_CREATE_DETACHED)!=0)
ABORT("pthread_attr_setdetachedstate failed");
#undef pthread_create
if (pthread_create(&thread,&attr,GC_mprotect_thread,NULL)!=0)
ABORT("pthread_create failed");
(void)pthread_attr_destroy(&attr);
#ifdef BROKEN_EXCEPTION_HANDLING
{
struct sigaction sa,oldsa;
sa.sa_handler=(SIG_HNDLR_PTR)GC_darwin_sigbus;
sigemptyset(&sa.sa_mask);
sa.sa_flags=SA_RESTART|SA_SIGINFO;
if (sigaction(SIGBUS,&sa,&oldsa)< 0)
ABORT("sigaction failed");
if (oldsa.sa_handler!=(SIG_HNDLR_PTR)(signed_word)SIG_DFL){
GC_VERBOSE_LOG_PRINTF("Replaced other SIGBUS handler\n");
}
}
#endif
#if defined(CPPCHECK)
GC_noop1((word)GC_ports.os_callback[0]);
#endif
return TRUE;
}
STATIC kern_return_t GC_forward_exception(mach_port_t thread,mach_port_t task,
exception_type_t exception,
exception_data_t data,
mach_msg_type_number_t data_count)
{
unsigned int i;
kern_return_t r;
mach_port_t port;
exception_behavior_t behavior;
thread_state_flavor_t flavor;
thread_state_data_t thread_state;
mach_msg_type_number_t thread_state_count=THREAD_STATE_MAX;
for (i=0;i < GC_old_exc_ports.count;i++)
if (GC_old_exc_ports.masks[i]&(1<<exception))
break;
if (i==GC_old_exc_ports.count)
ABORT("No handler for exception!");
port=GC_old_exc_ports.ports[i];
behavior=GC_old_exc_ports.behaviors[i];
flavor=GC_old_exc_ports.flavors[i];
if (behavior==EXCEPTION_STATE||behavior==EXCEPTION_STATE_IDENTITY){
r=thread_get_state(thread,flavor,thread_state,&thread_state_count);
if(r!=KERN_SUCCESS)
ABORT("thread_get_state failed in forward_exception");
}
switch(behavior){
case EXCEPTION_STATE:
r=exception_raise_state(port,thread,task,exception,data,data_count,
&flavor,thread_state,thread_state_count,
thread_state,&thread_state_count);
break;
case EXCEPTION_STATE_IDENTITY:
r=exception_raise_state_identity(port,thread,task,exception,data,
data_count,&flavor,thread_state,
thread_state_count,thread_state,
&thread_state_count);
break;
default:
r=exception_raise(port,thread,task,exception,data,data_count);
}
if (behavior==EXCEPTION_STATE||behavior==EXCEPTION_STATE_IDENTITY){
r=thread_set_state(thread,flavor,thread_state,thread_state_count);
if (r!=KERN_SUCCESS)
ABORT("thread_set_state failed in forward_exception");
}
return r;
}
#define FWD()GC_forward_exception(thread,task,exception,code,code_count)
#ifdef ARM32
#define DARWIN_EXC_STATE ARM_EXCEPTION_STATE
#define DARWIN_EXC_STATE_COUNT ARM_EXCEPTION_STATE_COUNT
#define DARWIN_EXC_STATE_T arm_exception_state_t
#define DARWIN_EXC_STATE_DAR THREAD_FLD_NAME(far)
#elif defined(AARCH64)
#define DARWIN_EXC_STATE ARM_EXCEPTION_STATE64
#define DARWIN_EXC_STATE_COUNT ARM_EXCEPTION_STATE64_COUNT
#define DARWIN_EXC_STATE_T arm_exception_state64_t
#define DARWIN_EXC_STATE_DAR THREAD_FLD_NAME(far)
#elif defined(POWERPC)
#if CPP_WORDSZ==32
#define DARWIN_EXC_STATE PPC_EXCEPTION_STATE
#define DARWIN_EXC_STATE_COUNT PPC_EXCEPTION_STATE_COUNT
#define DARWIN_EXC_STATE_T ppc_exception_state_t
#else
#define DARWIN_EXC_STATE PPC_EXCEPTION_STATE64
#define DARWIN_EXC_STATE_COUNT PPC_EXCEPTION_STATE64_COUNT
#define DARWIN_EXC_STATE_T ppc_exception_state64_t
#endif
#define DARWIN_EXC_STATE_DAR THREAD_FLD_NAME(dar)
#elif defined(I386)||defined(X86_64)
#if CPP_WORDSZ==32
#if defined(i386_EXCEPTION_STATE_COUNT)&&!defined(x86_EXCEPTION_STATE32_COUNT)
#define DARWIN_EXC_STATE i386_EXCEPTION_STATE
#define DARWIN_EXC_STATE_COUNT i386_EXCEPTION_STATE_COUNT
#define DARWIN_EXC_STATE_T i386_exception_state_t
#else
#define DARWIN_EXC_STATE x86_EXCEPTION_STATE32
#define DARWIN_EXC_STATE_COUNT x86_EXCEPTION_STATE32_COUNT
#define DARWIN_EXC_STATE_T x86_exception_state32_t
#endif
#else
#define DARWIN_EXC_STATE x86_EXCEPTION_STATE64
#define DARWIN_EXC_STATE_COUNT x86_EXCEPTION_STATE64_COUNT
#define DARWIN_EXC_STATE_T x86_exception_state64_t
#endif
#define DARWIN_EXC_STATE_DAR THREAD_FLD_NAME(faultvaddr)
#elif!defined(CPPCHECK)
#error FIXME for non-arm/ppc/x86 darwin
#endif
GC_API_OSCALL kern_return_t
catch_exception_raise(mach_port_t exception_port GC_ATTR_UNUSED,
mach_port_t thread,mach_port_t task GC_ATTR_UNUSED,
exception_type_t exception,exception_data_t code,
mach_msg_type_number_t code_count GC_ATTR_UNUSED)
{
kern_return_t r;
char*addr;
thread_state_flavor_t flavor=DARWIN_EXC_STATE;
mach_msg_type_number_t exc_state_count=DARWIN_EXC_STATE_COUNT;
DARWIN_EXC_STATE_T exc_state;
if (exception!=EXC_BAD_ACCESS||code[0]!=KERN_PROTECTION_FAILURE){
#ifdef DEBUG_EXCEPTION_HANDLING
GC_log_printf("Exception:0x%x Code:0x%x 0x%x in catch...\n",
exception,code_count > 0?code[0]:-1,
code_count > 1?code[1]:-1);
#endif
return FWD();
}
r=thread_get_state(thread,flavor,(natural_t*)&exc_state,
&exc_state_count);
if(r!=KERN_SUCCESS){
#ifdef BROKEN_EXCEPTION_HANDLING
GC_err_printf("thread_get_state failed in catch_exception_raise\n");
return KERN_SUCCESS;
#else
ABORT("thread_get_state failed in catch_exception_raise");
#endif
}
addr=(char*)exc_state.DARWIN_EXC_STATE_DAR;
if (!is_header_found_async(addr)){
#ifdef BROKEN_EXCEPTION_HANDLING
static char*last_fault;
static int last_fault_count;
if(addr!=last_fault){
last_fault=addr;
last_fault_count=0;
}
if(++last_fault_count < 32){
if(last_fault_count==1)
WARN("Ignoring KERN_PROTECTION_FAILURE at %p\n",addr);
return KERN_SUCCESS;
}
GC_err_printf("Unexpected KERN_PROTECTION_FAILURE at %p;aborting...\n",
(void*)addr);
EXIT();
#else
return FWD();
#endif
}
#ifdef BROKEN_EXCEPTION_HANDLING
GC_sigbus_count=0;
#endif
GC_ASSERT(GC_page_size!=0);
if (GC_mprotect_state==GC_MP_NORMAL){
struct hblk*h=(struct hblk*)((word)addr&~(GC_page_size-1));
size_t i;
UNPROTECT(h,GC_page_size);
for (i=0;i < divHBLKSZ(GC_page_size);i++){
word index=PHT_HASH(h+i);
async_set_pht_entry_from_index(GC_dirty_pages,index);
}
} else if (GC_mprotect_state==GC_MP_DISCARDING){
} else {
GC_err_printf("KERN_PROTECTION_FAILURE while world is stopped\n");
return FWD();
}
return KERN_SUCCESS;
}
#undef FWD
#ifndef NO_DESC_CATCH_EXCEPTION_RAISE
__asm__(".desc _catch_exception_raise,0x10");
__asm__(".desc _catch_exception_raise_state,0x10");
__asm__(".desc _catch_exception_raise_state_identity,0x10");
#endif
#endif
#ifndef HAVE_INCREMENTAL_PROTECTION_NEEDS
GC_API int GC_CALL GC_incremental_protection_needs(void)
{
GC_ASSERT(GC_is_initialized);
return GC_PROTECTS_NONE;
}
#endif
#ifdef ECOS
#undef sbrk
#endif
GC_API void GC_CALL GC_set_pages_executable(int value)
{
GC_ASSERT(!GC_is_initialized);
GC_pages_executable=(GC_bool)(value!=0);
}
GC_API int GC_CALL GC_get_pages_executable(void)
{
#ifdef IGNORE_PAGES_EXECUTABLE
return 1;
#else
return (int)GC_pages_executable;
#endif
}
#if defined(I386)&&defined(LINUX)&&defined(SAVE_CALL_CHAIN)
#include <features.h>
struct frame {
struct frame*fr_savfp;
long fr_savpc;
#if NARGS > 0
long fr_arg[NARGS];
#endif
};
#endif
#if defined(SPARC)
#if defined(LINUX)
#include <features.h>
#if defined(SAVE_CALL_CHAIN)
struct frame {
long fr_local[8];
long fr_arg[6];
struct frame*fr_savfp;
long fr_savpc;
#ifndef __arch64__
char*fr_stret;
#endif
long fr_argd[6];
long fr_argx[0];
};
#endif
#elif defined (DRSNX)
#include <sys/sparc/frame.h>
#elif defined(OPENBSD)
#include <frame.h>
#elif defined(FREEBSD)||defined(NETBSD)
#include <machine/frame.h>
#else
#include <sys/frame.h>
#endif
#if NARGS > 6
#error We only know how to get the first 6 arguments
#endif
#endif
#ifdef NEED_CALLINFO
#ifdef LINUX
#include <unistd.h>
#endif
#endif
#if defined(GC_HAVE_BUILTIN_BACKTRACE)
#ifdef _MSC_VER
#ifndef GC_MSVC_DBG_H
#define GC_MSVC_DBG_H
#include <stdlib.h>
#ifdef __cplusplus
extern "C" {
#endif
#if!MSVC_DBG_DLL
#define MSVC_DBG_EXPORT
#elif MSVC_DBG_BUILD
#define MSVC_DBG_EXPORT __declspec(dllexport)
#else
#define MSVC_DBG_EXPORT __declspec(dllimport)
#endif
#ifndef MAX_SYM_NAME
#define MAX_SYM_NAME 2000
#endif
typedef void*HANDLE;
typedef struct _CONTEXT CONTEXT;
MSVC_DBG_EXPORT size_t GetStackFrames(size_t skip,void*frames[],size_t maxFrames);
MSVC_DBG_EXPORT size_t GetStackFramesFromContext(HANDLE hProcess,HANDLE hThread,CONTEXT*context,size_t skip,void*frames[],size_t maxFrames);
MSVC_DBG_EXPORT size_t GetModuleNameFromAddress(void*address,char*moduleName,size_t size);
MSVC_DBG_EXPORT size_t GetModuleNameFromStack(size_t skip,char*moduleName,size_t size);
MSVC_DBG_EXPORT size_t GetSymbolNameFromAddress(void*address,char*symbolName,size_t size,size_t*offsetBytes);
MSVC_DBG_EXPORT size_t GetSymbolNameFromStack(size_t skip,char*symbolName,size_t size,size_t*offsetBytes);
MSVC_DBG_EXPORT size_t GetFileLineFromAddress(void*address,char*fileName,size_t size,size_t*lineNumber,size_t*offsetBytes);
MSVC_DBG_EXPORT size_t GetFileLineFromStack(size_t skip,char*fileName,size_t size,size_t*lineNumber,size_t*offsetBytes);
MSVC_DBG_EXPORT size_t GetDescriptionFromAddress(void*address,const char*format,char*description,size_t size);
MSVC_DBG_EXPORT size_t GetDescriptionFromStack(void*const frames[],size_t count,const char*format,char*description[],size_t size);
MSVC_DBG_EXPORT int backtrace(void*addresses[],int count);
MSVC_DBG_EXPORT char**backtrace_symbols(void*const addresses[],int count);
#ifdef __cplusplus
}
#endif
#endif
#else
#include <execinfo.h>
#endif
#endif
#ifdef SAVE_CALL_CHAIN
#if NARGS==0&&NFRAMES % 2==0&&defined(GC_HAVE_BUILTIN_BACKTRACE)
#ifdef REDIRECT_MALLOC
#ifdef THREADS
__thread
#endif
GC_bool GC_in_save_callers=FALSE;
#endif
GC_INNER void GC_save_callers(struct callinfo info[NFRAMES])
{
void*tmp_info[NFRAMES+1];
int npcs,i;
#define IGNORE_FRAMES 1
#ifdef REDIRECT_MALLOC
if (GC_in_save_callers){
info[0].ci_pc=(word)(&GC_save_callers);
for (i=1;i < NFRAMES;++i)info[i].ci_pc=0;
return;
}
GC_in_save_callers=TRUE;
#endif
GC_ASSERT(I_HOLD_LOCK());
GC_STATIC_ASSERT(sizeof(struct callinfo)==sizeof(void*));
npcs=backtrace((void**)tmp_info,NFRAMES+IGNORE_FRAMES);
if (npcs > IGNORE_FRAMES)
BCOPY(&tmp_info[IGNORE_FRAMES],info,
(npcs - IGNORE_FRAMES)*sizeof(void*));
for (i=npcs - IGNORE_FRAMES;i < NFRAMES;++i)info[i].ci_pc=0;
#ifdef REDIRECT_MALLOC
GC_in_save_callers=FALSE;
#endif
}
#else
#if (defined(OPENBSD)||defined(NETBSD)||defined(FREEBSD))&&defined(SPARC)
#define FR_SAVFP fr_fp
#define FR_SAVPC fr_pc
#else
#define FR_SAVFP fr_savfp
#define FR_SAVPC fr_savpc
#endif
#if defined(SPARC)&&(defined(__arch64__)||defined(__sparcv9))
#define BIAS 2047
#else
#define BIAS 0
#endif
GC_INNER void GC_save_callers(struct callinfo info[NFRAMES])
{
struct frame*frame;
struct frame*fp;
int nframes=0;
#ifdef I386
asm("movl %%ebp,%0":"=r"(frame));
fp=frame;
#else
frame=(struct frame*)GC_save_regs_in_stack();
fp=(struct frame*)((long)frame->FR_SAVFP+BIAS);
#endif
for (;!((word)fp HOTTER_THAN (word)frame)
#ifndef THREADS
&&!((word)GC_stackbottom HOTTER_THAN (word)fp)
#elif defined(STACK_GROWS_UP)
&&fp!=NULL
#endif
&&nframes < NFRAMES;
fp=(struct frame*)((long)fp->FR_SAVFP+BIAS),nframes++){
#if NARGS > 0
int i;
#endif
info[nframes].ci_pc=fp->FR_SAVPC;
#if NARGS > 0
for (i=0;i < NARGS;i++){
info[nframes].ci_arg[i]=~(fp->fr_arg[i]);
}
#endif
}
if (nframes < NFRAMES)info[nframes].ci_pc=0;
}
#endif
#endif
#ifdef NEED_CALLINFO
GC_INNER void GC_print_callers(struct callinfo info[NFRAMES])
{
int i;
static int reentry_count=0;
DCL_LOCK_STATE;
LOCK();
++reentry_count;
UNLOCK();
#if NFRAMES==1
GC_err_printf("\tCaller at allocation:\n");
#else
GC_err_printf("\tCall chain at allocation:\n");
#endif
for (i=0;i < NFRAMES;i++){
#if defined(LINUX)&&!defined(SMALL_CONFIG)
GC_bool stop=FALSE;
#endif
if (0==info[i].ci_pc)
break;
#if NARGS > 0
{
int j;
GC_err_printf("\t\targs:");
for (j=0;j < NARGS;j++){
if (j!=0)GC_err_printf(",");
GC_err_printf("%d (0x%X)",~(info[i].ci_arg[j]),
~(info[i].ci_arg[j]));
}
GC_err_printf("\n");
}
#endif
if (reentry_count > 1){
GC_err_printf("\t\t##PC##=0x%lx\n",
(unsigned long)info[i].ci_pc);
continue;
}
{
char buf[40];
char*name;
#if defined(GC_HAVE_BUILTIN_BACKTRACE)&&!defined(GC_BACKTRACE_SYMBOLS_BROKEN)
char**sym_name=
backtrace_symbols((void**)(&(info[i].ci_pc)),1);
if (sym_name!=NULL){
name=sym_name[0];
} else
#endif
{
(void)snprintf(buf,sizeof(buf),"##PC##=0x%lx",
(unsigned long)info[i].ci_pc);
buf[sizeof(buf)- 1]='\0';
name=buf;
}
#if defined(LINUX)&&!defined(SMALL_CONFIG)
do {
FILE*pipe;
#define EXE_SZ 100
static char exe_name[EXE_SZ];
#define CMD_SZ 200
char cmd_buf[CMD_SZ];
#define RESULT_SZ 200
static char result_buf[RESULT_SZ];
size_t result_len;
char*old_preload;
#define PRELOAD_SZ 200
char preload_buf[PRELOAD_SZ];
static GC_bool found_exe_name=FALSE;
static GC_bool will_fail=FALSE;
if (will_fail)
break;
if (!found_exe_name){
int ret_code=readlink("/proc/self/exe",exe_name,EXE_SZ);
if (ret_code < 0||ret_code>=EXE_SZ
||exe_name[0]!='/'){
will_fail=TRUE;
break;
}
exe_name[ret_code]='\0';
found_exe_name=TRUE;
}
(void)snprintf(cmd_buf,sizeof(cmd_buf),
"/usr/bin/addr2line -f -e %s 0x%lx",
exe_name,(unsigned long)info[i].ci_pc);
cmd_buf[sizeof(cmd_buf)- 1]='\0';
old_preload=GETENV("LD_PRELOAD");
if (0!=old_preload){
size_t old_len=strlen(old_preload);
if (old_len>=PRELOAD_SZ){
will_fail=TRUE;
break;
}
BCOPY(old_preload,preload_buf,old_len+1);
unsetenv ("LD_PRELOAD");
}
pipe=popen(cmd_buf,"r");
if (0!=old_preload
&&0!=setenv ("LD_PRELOAD",preload_buf,0)){
WARN("Failed to reset LD_PRELOAD\n",0);
}
if (NULL==pipe){
will_fail=TRUE;
break;
}
result_len=fread(result_buf,1,RESULT_SZ - 1,pipe);
(void)pclose(pipe);
if (0==result_len){
will_fail=TRUE;
break;
}
if (result_buf[result_len - 1]=='\n')--result_len;
result_buf[result_len]=0;
if (result_buf[0]=='?'
||(result_buf[result_len-2]==':'
&&result_buf[result_len-1]=='0'))
break;
{
char*nl=strchr(result_buf,'\n');
if (nl!=NULL
&&(word)nl < (word)(result_buf+result_len)){
*nl=':';
}
if (strncmp(result_buf,"main",
nl!=NULL
?(size_t)((word)nl
- COVERT_DATAFLOW(result_buf))
:result_len)==0){
stop=TRUE;
}
}
if (result_len < RESULT_SZ - 25){
(void)snprintf(&result_buf[result_len],
sizeof(result_buf)- result_len,
" [0x%lx]",(unsigned long)info[i].ci_pc);
result_buf[sizeof(result_buf)- 1]='\0';
}
#if defined(CPPCHECK)
GC_noop1((unsigned char)name[0]);
#endif
name=result_buf;
} while (0);
#endif
GC_err_printf("\t\t%s\n",name);
#if defined(GC_HAVE_BUILTIN_BACKTRACE)&&!defined(GC_BACKTRACE_SYMBOLS_BROKEN)
if (sym_name!=NULL)
free(sym_name);
#endif
}
#if defined(LINUX)&&!defined(SMALL_CONFIG)
if (stop)
break;
#endif
}
LOCK();
--reentry_count;
UNLOCK();
}
#endif
#if defined(LINUX)&&defined(__ELF__)&&!defined(SMALL_CONFIG)
void GC_print_address_map(void)
{
char*maps;
GC_err_printf("----------Begin address map----------\n");
maps=GC_get_maps();
GC_err_puts(maps!=NULL?maps:"Failed to get map!\n");
GC_err_printf("----------End address map----------\n");
}
#endif
#if defined(THREAD_LOCAL_ALLOC)
#ifndef THREADS
#error "invalid config - THREAD_LOCAL_ALLOC requires GC_THREADS"
#endif
#ifndef GC_THREAD_LOCAL_ALLOC_H
#define GC_THREAD_LOCAL_ALLOC_H
#ifdef THREAD_LOCAL_ALLOC
#if defined(USE_HPUX_TLS)
#error USE_HPUX_TLS macro was replaced by USE_COMPILER_TLS
#endif
#include <stdlib.h>
EXTERN_C_BEGIN
#if!defined(USE_PTHREAD_SPECIFIC)&&!defined(USE_WIN32_SPECIFIC)&&!defined(USE_WIN32_COMPILER_TLS)&&!defined(USE_COMPILER_TLS)&&!defined(USE_CUSTOM_SPECIFIC)
#if defined(MSWIN32)||defined(MSWINCE)||defined(CYGWIN32)
#if defined(CYGWIN32)&&GC_GNUC_PREREQ(4,0)
#if defined(__clang__)
#define USE_PTHREAD_SPECIFIC
#else
#define USE_COMPILER_TLS
#endif
#elif defined(__GNUC__)||defined(MSWINCE)
#define USE_WIN32_SPECIFIC
#else
#define USE_WIN32_COMPILER_TLS
#endif
#elif (defined(LINUX)&&!defined(ARM32)&&!defined(AVR32)&&GC_GNUC_PREREQ(3,3)&&!(defined(__clang__)&&defined(HOST_ANDROID)))||(defined(FREEBSD)||(defined(NETBSD)&&__NetBSD_Version__>=600000000)&&(GC_GNUC_PREREQ(4,4)||GC_CLANG_PREREQ(3,9)))||(defined(HOST_ANDROID)&&defined(ARM32)&&(GC_GNUC_PREREQ(4,6)||GC_CLANG_PREREQ_FULL(3,8,256229)))
#define USE_COMPILER_TLS
#elif defined(GC_DGUX386_THREADS)||defined(GC_OSF1_THREADS)||defined(GC_AIX_THREADS)||defined(GC_DARWIN_THREADS)||defined(GC_FREEBSD_THREADS)||defined(GC_NETBSD_THREADS)||defined(GC_LINUX_THREADS)||defined(GC_HAIKU_THREADS)||defined(GC_RTEMS_PTHREADS)
#define USE_PTHREAD_SPECIFIC
#elif defined(GC_HPUX_THREADS)
#ifdef __GNUC__
#define USE_PTHREAD_SPECIFIC
#else
#define USE_COMPILER_TLS
#endif
#else
#define USE_CUSTOM_SPECIFIC
#endif
#endif
#ifndef THREAD_FREELISTS_KINDS
#ifdef ENABLE_DISCLAIM
#define THREAD_FREELISTS_KINDS (NORMAL+2)
#else
#define THREAD_FREELISTS_KINDS (NORMAL+1)
#endif
#endif
typedef struct thread_local_freelists {
void*_freelists[THREAD_FREELISTS_KINDS][TINY_FREELISTS];
#define ptrfree_freelists _freelists[PTRFREE]
#define normal_freelists _freelists[NORMAL]
#ifdef GC_GCJ_SUPPORT
void*gcj_freelists[TINY_FREELISTS];
#define ERROR_FL ((void*)GC_WORD_MAX)
#endif
#define DIRECT_GRANULES (HBLKSIZE/GRANULE_BYTES)
}*GC_tlfs;
#if defined(USE_PTHREAD_SPECIFIC)
#define GC_getspecific pthread_getspecific
#define GC_setspecific pthread_setspecific
#define GC_key_create pthread_key_create
#define GC_remove_specific(key)pthread_setspecific(key,NULL)
#define GC_remove_specific_after_fork(key,t)(void)0
typedef pthread_key_t GC_key_t;
#elif defined(USE_COMPILER_TLS)||defined(USE_WIN32_COMPILER_TLS)
#define GC_getspecific(x)(x)
#define GC_setspecific(key,v)((key)=(v),0)
#define GC_key_create(key,d)0
#define GC_remove_specific(key)
#define GC_remove_specific_after_fork(key,t)(void)0
typedef void*GC_key_t;
#elif defined(USE_WIN32_SPECIFIC)
#define GC_getspecific TlsGetValue
#define GC_setspecific(key,v)!TlsSetValue(key,v)
#ifndef TLS_OUT_OF_INDEXES
#define TLS_OUT_OF_INDEXES (DWORD)0xFFFFFFFF
#endif
#define GC_key_create(key,d)((d)!=0||(*(key)=TlsAlloc())==TLS_OUT_OF_INDEXES?-1:0)
#define GC_remove_specific(key)
#define GC_remove_specific_after_fork(key,t)(void)0
typedef DWORD GC_key_t;
#elif defined(USE_CUSTOM_SPECIFIC)
EXTERN_C_END
#ifndef GC_SPECIFIC_H
#define GC_SPECIFIC_H
#include <errno.h>
EXTERN_C_BEGIN
#define MALLOC_CLEAR(n)GC_INTERNAL_MALLOC(n,NORMAL)
#define TS_CACHE_SIZE 1024
#define CACHE_HASH(n)((((n)>>8)^(n))&(TS_CACHE_SIZE - 1))
#define TS_HASH_SIZE 1024
#define HASH(p)((unsigned)((((word)(p))>>8)^(word)(p))&(TS_HASH_SIZE - 1))
#ifdef GC_ASSERTIONS
typedef GC_hidden_pointer ts_entry_value_t;
#define TS_HIDE_VALUE(p)GC_HIDE_POINTER(p)
#define TS_REVEAL_PTR(p)GC_REVEAL_POINTER(p)
#else
typedef void*ts_entry_value_t;
#define TS_HIDE_VALUE(p)(p)
#define TS_REVEAL_PTR(p)(p)
#endif
typedef struct thread_specific_entry {
volatile AO_t qtid;
ts_entry_value_t value;
struct thread_specific_entry*next;
pthread_t thread;
} tse;
#define quick_thread_id()(((word)GC_approx_sp())>>12)
#define INVALID_QTID ((word)0)
#define INVALID_THREADID ((pthread_t)0)
union ptse_ao_u {
tse*p;
volatile AO_t ao;
};
typedef struct thread_specific_data {
tse*volatile cache[TS_CACHE_SIZE];
union ptse_ao_u hash[TS_HASH_SIZE];
pthread_mutex_t lock;
} tsd;
typedef tsd*GC_key_t;
#define GC_key_create(key,d)GC_key_create_inner(key)
GC_INNER int GC_key_create_inner(tsd**key_ptr);
GC_INNER int GC_setspecific(tsd*key,void*value);
#define GC_remove_specific(key)GC_remove_specific_after_fork(key,pthread_self())
GC_INNER void GC_remove_specific_after_fork(tsd*key,pthread_t t);
GC_INNER void*GC_slow_getspecific(tsd*key,word qtid,
tse*volatile*cache_entry);
GC_INLINE void*GC_getspecific(tsd*key)
{
word qtid=quick_thread_id();
tse*volatile*entry_ptr=&key->cache[CACHE_HASH(qtid)];
tse*entry=*entry_ptr;
GC_ASSERT(qtid!=INVALID_QTID);
if (EXPECT(entry->qtid==qtid,TRUE)){
GC_ASSERT(entry->thread==pthread_self());
return TS_REVEAL_PTR(entry->value);
}
return GC_slow_getspecific(key,qtid,entry_ptr);
}
EXTERN_C_END
#endif
EXTERN_C_BEGIN
#else
#error implement me
#endif
GC_INNER void GC_init_thread_local(GC_tlfs p);
GC_INNER void GC_destroy_thread_local(GC_tlfs p);
GC_INNER void GC_mark_thread_local_fls_for(GC_tlfs p);
#ifdef GC_ASSERTIONS
GC_bool GC_is_thread_tsd_valid(void*tsd);
void GC_check_tls_for(GC_tlfs p);
#if defined(USE_CUSTOM_SPECIFIC)
void GC_check_tsd_marks(tsd*key);
#endif
#endif
#ifndef GC_ATTR_TLS_FAST
#define GC_ATTR_TLS_FAST
#endif
extern
#if defined(USE_COMPILER_TLS)
__thread GC_ATTR_TLS_FAST
#elif defined(USE_WIN32_COMPILER_TLS)
__declspec(thread)GC_ATTR_TLS_FAST
#endif
GC_key_t GC_thread_key;
EXTERN_C_END
#endif
#endif
#include <stdlib.h>
#if defined(USE_COMPILER_TLS)
__thread GC_ATTR_TLS_FAST
#elif defined(USE_WIN32_COMPILER_TLS)
__declspec(thread)GC_ATTR_TLS_FAST
#endif
GC_key_t GC_thread_key;
static GC_bool keys_initialized;
static void return_single_freelist(void*fl,void**gfl)
{
if (*gfl==0){
*gfl=fl;
} else {
void*q,**qptr;
GC_ASSERT(GC_size(fl)==GC_size(*gfl));
qptr=&(obj_link(fl));
while ((word)(q=*qptr)>=HBLKSIZE)
qptr=&(obj_link(q));
GC_ASSERT(0==q);
*qptr=*gfl;
*gfl=fl;
}
}
static void return_freelists(void**fl,void**gfl)
{
int i;
for (i=1;i < TINY_FREELISTS;++i){
if ((word)(fl[i])>=HBLKSIZE){
return_single_freelist(fl[i],&gfl[i]);
}
fl[i]=(ptr_t)HBLKSIZE;
}
#ifdef GC_GCJ_SUPPORT
if (fl[0]==ERROR_FL)return;
#endif
if ((word)(fl[0])>=HBLKSIZE){
return_single_freelist(fl[0],&gfl[1]);
}
}
#ifdef USE_PTHREAD_SPECIFIC
static void reset_thread_key(void*v){
pthread_setspecific(GC_thread_key,v);
}
#else
#define reset_thread_key 0
#endif
GC_INNER void GC_init_thread_local(GC_tlfs p)
{
int i,j,res;
GC_ASSERT(I_HOLD_LOCK());
if (!EXPECT(keys_initialized,TRUE)){
GC_ASSERT((word)&GC_thread_key % sizeof(word)==0);
res=GC_key_create(&GC_thread_key,reset_thread_key);
if (COVERT_DATAFLOW(res)!=0){
ABORT("Failed to create key for local allocator");
}
keys_initialized=TRUE;
}
res=GC_setspecific(GC_thread_key,p);
if (COVERT_DATAFLOW(res)!=0){
ABORT("Failed to set thread specific allocation pointers");
}
for (j=0;j < TINY_FREELISTS;++j){
for (i=0;i < THREAD_FREELISTS_KINDS;++i){
p->_freelists[i][j]=(void*)(word)1;
}
#ifdef GC_GCJ_SUPPORT
p->gcj_freelists[j]=(void*)(word)1;
#endif
}
#ifdef GC_GCJ_SUPPORT
p->gcj_freelists[0]=ERROR_FL;
#endif
}
GC_INNER void GC_destroy_thread_local(GC_tlfs p)
{
int k;
GC_STATIC_ASSERT(THREAD_FREELISTS_KINDS<=MAXOBJKINDS);
for (k=0;k < THREAD_FREELISTS_KINDS;++k){
if (k==(int)GC_n_kinds)
break;
return_freelists(p->_freelists[k],GC_obj_kinds[k].ok_freelist);
}
#ifdef GC_GCJ_SUPPORT
return_freelists(p->gcj_freelists,(void**)GC_gcjobjfreelist);
#endif
}
GC_API GC_ATTR_MALLOC void*GC_CALL GC_malloc_kind(size_t bytes,int kind)
{
size_t granules;
void*tsd;
void*result;
#if MAXOBJKINDS > THREAD_FREELISTS_KINDS
if (EXPECT(kind>=THREAD_FREELISTS_KINDS,FALSE)){
return GC_malloc_kind_global(bytes,kind);
}
#endif
#if!defined(USE_PTHREAD_SPECIFIC)&&!defined(USE_WIN32_SPECIFIC)
{
GC_key_t k=GC_thread_key;
if (EXPECT(0==k,FALSE)){
return GC_malloc_kind_global(bytes,kind);
}
tsd=GC_getspecific(k);
}
#else
if (!EXPECT(keys_initialized,TRUE))
return GC_malloc_kind_global(bytes,kind);
tsd=GC_getspecific(GC_thread_key);
#endif
#if!defined(USE_COMPILER_TLS)&&!defined(USE_WIN32_COMPILER_TLS)
if (EXPECT(0==tsd,FALSE)){
return GC_malloc_kind_global(bytes,kind);
}
#endif
GC_ASSERT(GC_is_initialized);
GC_ASSERT(GC_is_thread_tsd_valid(tsd));
granules=ROUNDED_UP_GRANULES(bytes);
#if defined(CPPCHECK)
#define MALLOC_KIND_PTRFREE_INIT (void*)1
#else
#define MALLOC_KIND_PTRFREE_INIT NULL
#endif
GC_FAST_MALLOC_GRANS(result,granules,
((GC_tlfs)tsd)->_freelists[kind],DIRECT_GRANULES,
kind,GC_malloc_kind_global(bytes,kind),
(void)(kind==PTRFREE?MALLOC_KIND_PTRFREE_INIT
:(obj_link(result)=0)));
#ifdef LOG_ALLOCS
GC_log_printf("GC_malloc_kind(%lu,%d)returned %p,recent GC #%lu\n",
(unsigned long)bytes,kind,result,
(unsigned long)GC_gc_no);
#endif
return result;
}
#ifdef GC_GCJ_SUPPORT
GC_API GC_ATTR_MALLOC void*GC_CALL GC_gcj_malloc(size_t bytes,
void*ptr_to_struct_containing_descr)
{
if (EXPECT(GC_incremental,FALSE)){
return GC_core_gcj_malloc(bytes,ptr_to_struct_containing_descr);
} else {
size_t granules=ROUNDED_UP_GRANULES(bytes);
void*result;
void**tiny_fl;
GC_ASSERT(GC_gcjobjfreelist!=NULL);
tiny_fl=((GC_tlfs)GC_getspecific(GC_thread_key))->gcj_freelists;
GC_FAST_MALLOC_GRANS(result,granules,tiny_fl,DIRECT_GRANULES,
GC_gcj_kind,
GC_core_gcj_malloc(bytes,
ptr_to_struct_containing_descr),
{AO_compiler_barrier();
*(void**)result=ptr_to_struct_containing_descr;});
return result;
}
}
#endif
GC_INNER void GC_mark_thread_local_fls_for(GC_tlfs p)
{
ptr_t q;
int i,j;
for (j=0;j < TINY_FREELISTS;++j){
for (i=0;i < THREAD_FREELISTS_KINDS;++i){
q=(ptr_t)AO_load((volatile AO_t*)&p->_freelists[i][j]);
if ((word)q > HBLKSIZE)
GC_set_fl_marks(q);
}
#ifdef GC_GCJ_SUPPORT
if (EXPECT(j > 0,TRUE)){
q=(ptr_t)AO_load((volatile AO_t*)&p->gcj_freelists[j]);
if ((word)q > HBLKSIZE)
GC_set_fl_marks(q);
}
#endif
}
}
#if defined(GC_ASSERTIONS)
void GC_check_tls_for(GC_tlfs p)
{
int i,j;
for (j=1;j < TINY_FREELISTS;++j){
for (i=0;i < THREAD_FREELISTS_KINDS;++i){
GC_check_fl_marks(&p->_freelists[i][j]);
}
#ifdef GC_GCJ_SUPPORT
GC_check_fl_marks(&p->gcj_freelists[j]);
#endif
}
}
#endif
#endif
#ifndef GC_PTHREAD_SUPPORT_H
#define GC_PTHREAD_SUPPORT_H
#if defined(GC_PTHREADS)&&!defined(GC_WIN32_THREADS)
#if defined(GC_DARWIN_THREADS)
#else
#ifndef GC_PTHREAD_STOP_WORLD_H
#define GC_PTHREAD_STOP_WORLD_H
EXTERN_C_BEGIN
struct thread_stop_info {
#if!defined(GC_OPENBSD_UTHREADS)&&!defined(NACL)&&!defined(SN_TARGET_ORBIS)&&!defined(SN_TARGET_PSP2)
volatile AO_t last_stop_count;
#endif
ptr_t stack_ptr;
#ifdef NACL
#ifdef ARM32
#define NACL_GC_REG_STORAGE_SIZE 9
#else
#define NACL_GC_REG_STORAGE_SIZE 20
#endif
ptr_t reg_storage[NACL_GC_REG_STORAGE_SIZE];
#elif defined(SN_TARGET_ORBIS)
#define ORBIS_GC_REG_STORAGE_SIZE 27
word registers[ORBIS_GC_REG_STORAGE_SIZE];
#endif
};
GC_INNER void GC_stop_init(void);
EXTERN_C_END
#endif
#endif
#ifdef THREAD_LOCAL_ALLOC
#endif
#ifdef THREAD_SANITIZER
#endif
EXTERN_C_BEGIN
typedef struct GC_Thread_Rep {
#ifdef THREAD_SANITIZER
char dummy[sizeof(oh)];
#endif
struct GC_Thread_Rep*next;
pthread_t id;
#ifdef USE_TKILL_ON_ANDROID
pid_t kernel_id;
#endif
struct thread_stop_info stop_info;
#if defined(GC_ENABLE_SUSPEND_THREAD)&&!defined(GC_DARWIN_THREADS)&&!defined(GC_OPENBSD_UTHREADS)&&!defined(NACL)
volatile AO_t suspended_ext;
#endif
unsigned char flags;
#define FINISHED 1
#define DETACHED 2
#define MAIN_THREAD 4
#define DISABLED_GC 0x10
unsigned char thread_blocked;
unsigned short finalizer_skipped;
unsigned char finalizer_nested;
ptr_t stack_end;
ptr_t altstack;
word altstack_size;
ptr_t stack;
word stack_size;
#if defined(GC_DARWIN_THREADS)&&!defined(DARWIN_DONT_PARSE_STACK)
ptr_t topOfStack;
#endif
#ifdef IA64
ptr_t backing_store_end;
ptr_t backing_store_ptr;
#endif
struct GC_traced_stack_sect_s*traced_stack_sect;
void*status;
#ifdef THREAD_LOCAL_ALLOC
struct thread_local_freelists tlfs GC_ATTR_WORD_ALIGNED;
#endif
}*GC_thread;
#ifndef THREAD_TABLE_SZ
#define THREAD_TABLE_SZ 256
#endif
#if CPP_WORDSZ==64
#define THREAD_TABLE_INDEX(id)(int)(((((NUMERIC_THREAD_ID(id)>>8)^NUMERIC_THREAD_ID(id))>>16)^((NUMERIC_THREAD_ID(id)>>8)^NUMERIC_THREAD_ID(id)))% THREAD_TABLE_SZ)
#else
#define THREAD_TABLE_INDEX(id)(int)(((NUMERIC_THREAD_ID(id)>>16)^(NUMERIC_THREAD_ID(id)>>8)^NUMERIC_THREAD_ID(id))% THREAD_TABLE_SZ)
#endif
GC_EXTERN volatile GC_thread GC_threads[THREAD_TABLE_SZ];
GC_EXTERN GC_bool GC_thr_initialized;
GC_INNER GC_thread GC_lookup_thread(pthread_t id);
#ifdef NACL
GC_EXTERN __thread GC_thread GC_nacl_gc_thread_self;
GC_INNER void GC_nacl_initialize_gc_thread(void);
GC_INNER void GC_nacl_shutdown_gc_thread(void);
#endif
#ifdef GC_EXPLICIT_SIGNALS_UNBLOCK
GC_INNER void GC_unblock_gc_signals(void);
#endif
#ifdef GC_PTHREAD_START_STANDALONE
#define GC_INNER_PTHRSTART
#else
#define GC_INNER_PTHRSTART GC_INNER
#endif
GC_INNER_PTHRSTART void*GC_CALLBACK GC_inner_start_routine(
struct GC_stack_base*sb,void*arg);
GC_INNER_PTHRSTART GC_thread GC_start_rtn_prepare_thread(
void*(**pstart)(void*),
void**pstart_arg,
struct GC_stack_base*sb,void*arg);
GC_INNER_PTHRSTART void GC_thread_exit_proc(void*);
EXTERN_C_END
#endif
#endif
#if defined(GC_DARWIN_THREADS)
#include <sys/sysctl.h>
#include <mach/machine.h>
#include <CoreFoundation/CoreFoundation.h>
#ifdef POWERPC
#if CPP_WORDSZ==32
#define PPC_RED_ZONE_SIZE 224
#elif CPP_WORDSZ==64
#define PPC_RED_ZONE_SIZE 320
#endif
#endif
#ifndef DARWIN_DONT_PARSE_STACK
typedef struct StackFrame {
unsigned long savedSP;
unsigned long savedCR;
unsigned long savedLR;
unsigned long reserved[2];
unsigned long savedRTOC;
} StackFrame;
GC_INNER ptr_t GC_FindTopOfStack(unsigned long stack_start)
{
StackFrame*frame=(StackFrame*)stack_start;
if (stack_start==0){
#ifdef POWERPC
#if CPP_WORDSZ==32
__asm__ __volatile__ ("lwz %0,0(r1)":"=r" (frame));
#else
__asm__ __volatile__ ("ld %0,0(r1)":"=r" (frame));
#endif
#elif defined(ARM32)
volatile ptr_t sp_reg;
__asm__ __volatile__ ("mov %0,r7\n":"=r" (sp_reg));
frame=(StackFrame*)sp_reg;
#elif defined(AARCH64)
volatile ptr_t sp_reg;
__asm__ __volatile__ ("mov %0,x29\n":"=r" (sp_reg));
frame=(StackFrame*)sp_reg;
#else
#if defined(CPPCHECK)
GC_noop1((word)&frame);
#endif
ABORT("GC_FindTopOfStack(0)is not implemented");
#endif
}
#ifdef DEBUG_THREADS_EXTRA
GC_log_printf("FindTopOfStack start at sp=%p\n",(void*)frame);
#endif
while (frame->savedSP!=0){
frame=(StackFrame*)frame->savedSP;
if ((frame->savedLR&~0x3)==0||(frame->savedLR&~0x3)==~0x3UL)
break;
}
#ifdef DEBUG_THREADS_EXTRA
GC_log_printf("FindTopOfStack finish at sp=%p\n",(void*)frame);
#endif
return (ptr_t)frame;
}
#endif
#ifdef GC_NO_THREADS_DISCOVERY
#define GC_query_task_threads FALSE
#elif defined(GC_DISCOVER_TASK_THREADS)
#define GC_query_task_threads TRUE
#else
STATIC GC_bool GC_query_task_threads=FALSE;
#endif
GC_API void GC_CALL GC_use_threads_discovery(void)
{
#if defined(GC_NO_THREADS_DISCOVERY)||defined(DARWIN_DONT_PARSE_STACK)
ABORT("Darwin task-threads-based stop and push unsupported");
#else
#ifndef GC_ALWAYS_MULTITHREADED
GC_ASSERT(!GC_need_to_lock);
#endif
#ifndef GC_DISCOVER_TASK_THREADS
GC_query_task_threads=TRUE;
#endif
GC_init_parallel();
#endif
}
#ifndef kCFCoreFoundationVersionNumber_iOS_8_0
#define kCFCoreFoundationVersionNumber_iOS_8_0 1140.1
#endif
STATIC ptr_t GC_stack_range_for(ptr_t*phi,thread_act_t thread,GC_thread p,
GC_bool thread_blocked,mach_port_t my_thread,
ptr_t*paltstack_lo,
ptr_t*paltstack_hi GC_ATTR_UNUSED)
{
ptr_t lo;
if (thread==my_thread){
GC_ASSERT(!thread_blocked);
lo=GC_approx_sp();
#ifndef DARWIN_DONT_PARSE_STACK
*phi=GC_FindTopOfStack(0);
#endif
} else if (thread_blocked){
#if defined(CPPCHECK)
if (NULL==p)ABORT("Invalid GC_thread passed to GC_stack_range_for");
#endif
lo=p->stop_info.stack_ptr;
#ifndef DARWIN_DONT_PARSE_STACK
*phi=p->topOfStack;
#endif
} else {
kern_return_t kern_result;
GC_THREAD_STATE_T state;
#if defined(ARM32)&&defined(ARM_THREAD_STATE32)
size_t size;
static cpu_type_t cputype=0;
if (cputype==0){
sysctlbyname("hw.cputype",&cputype,&size,NULL,0);
}
if (cputype==CPU_TYPE_ARM64
||kCFCoreFoundationVersionNumber
>=kCFCoreFoundationVersionNumber_iOS_8_0){
arm_unified_thread_state_t unified_state;
mach_msg_type_number_t unified_thread_state_count
=ARM_UNIFIED_THREAD_STATE_COUNT;
#if defined(CPPCHECK)
#define GC_ARM_UNIFIED_THREAD_STATE 1
#else
#define GC_ARM_UNIFIED_THREAD_STATE ARM_UNIFIED_THREAD_STATE
#endif
kern_result=thread_get_state(thread,GC_ARM_UNIFIED_THREAD_STATE,
(natural_t*)&unified_state,
&unified_thread_state_count);
#if!defined(CPPCHECK)
if (unified_state.ash.flavor!=ARM_THREAD_STATE32){
ABORT("unified_state flavor should be ARM_THREAD_STATE32");
}
#endif
state=unified_state;
} else
#endif
{
mach_msg_type_number_t thread_state_count=GC_MACH_THREAD_STATE_COUNT;
kern_result=thread_get_state(thread,GC_MACH_THREAD_STATE,
(natural_t*)&state,
&thread_state_count);
}
#ifdef DEBUG_THREADS
GC_log_printf("thread_get_state returns value=%d\n",kern_result);
#endif
if (kern_result!=KERN_SUCCESS)
ABORT("thread_get_state failed");
#if defined(I386)
lo=(ptr_t)state.THREAD_FLD(esp);
#ifndef DARWIN_DONT_PARSE_STACK
*phi=GC_FindTopOfStack(state.THREAD_FLD(esp));
#endif
GC_push_one(state.THREAD_FLD(eax));
GC_push_one(state.THREAD_FLD(ebx));
GC_push_one(state.THREAD_FLD(ecx));
GC_push_one(state.THREAD_FLD(edx));
GC_push_one(state.THREAD_FLD(edi));
GC_push_one(state.THREAD_FLD(esi));
GC_push_one(state.THREAD_FLD(ebp));
#elif defined(X86_64)
lo=(ptr_t)state.THREAD_FLD(rsp);
#ifndef DARWIN_DONT_PARSE_STACK
*phi=GC_FindTopOfStack(state.THREAD_FLD(rsp));
#endif
GC_push_one(state.THREAD_FLD(rax));
GC_push_one(state.THREAD_FLD(rbx));
GC_push_one(state.THREAD_FLD(rcx));
GC_push_one(state.THREAD_FLD(rdx));
GC_push_one(state.THREAD_FLD(rdi));
GC_push_one(state.THREAD_FLD(rsi));
GC_push_one(state.THREAD_FLD(rbp));
GC_push_one(state.THREAD_FLD(r8));
GC_push_one(state.THREAD_FLD(r9));
GC_push_one(state.THREAD_FLD(r10));
GC_push_one(state.THREAD_FLD(r11));
GC_push_one(state.THREAD_FLD(r12));
GC_push_one(state.THREAD_FLD(r13));
GC_push_one(state.THREAD_FLD(r14));
GC_push_one(state.THREAD_FLD(r15));
#elif defined(POWERPC)
lo=(ptr_t)(state.THREAD_FLD(r1)- PPC_RED_ZONE_SIZE);
#ifndef DARWIN_DONT_PARSE_STACK
*phi=GC_FindTopOfStack(state.THREAD_FLD(r1));
#endif
GC_push_one(state.THREAD_FLD(r0));
GC_push_one(state.THREAD_FLD(r2));
GC_push_one(state.THREAD_FLD(r3));
GC_push_one(state.THREAD_FLD(r4));
GC_push_one(state.THREAD_FLD(r5));
GC_push_one(state.THREAD_FLD(r6));
GC_push_one(state.THREAD_FLD(r7));
GC_push_one(state.THREAD_FLD(r8));
GC_push_one(state.THREAD_FLD(r9));
GC_push_one(state.THREAD_FLD(r10));
GC_push_one(state.THREAD_FLD(r11));
GC_push_one(state.THREAD_FLD(r12));
GC_push_one(state.THREAD_FLD(r13));
GC_push_one(state.THREAD_FLD(r14));
GC_push_one(state.THREAD_FLD(r15));
GC_push_one(state.THREAD_FLD(r16));
GC_push_one(state.THREAD_FLD(r17));
GC_push_one(state.THREAD_FLD(r18));
GC_push_one(state.THREAD_FLD(r19));
GC_push_one(state.THREAD_FLD(r20));
GC_push_one(state.THREAD_FLD(r21));
GC_push_one(state.THREAD_FLD(r22));
GC_push_one(state.THREAD_FLD(r23));
GC_push_one(state.THREAD_FLD(r24));
GC_push_one(state.THREAD_FLD(r25));
GC_push_one(state.THREAD_FLD(r26));
GC_push_one(state.THREAD_FLD(r27));
GC_push_one(state.THREAD_FLD(r28));
GC_push_one(state.THREAD_FLD(r29));
GC_push_one(state.THREAD_FLD(r30));
GC_push_one(state.THREAD_FLD(r31));
#elif defined(ARM32)
lo=(ptr_t)state.THREAD_FLD(sp);
#ifndef DARWIN_DONT_PARSE_STACK
*phi=GC_FindTopOfStack(state.THREAD_FLD(r[7]));
#endif
{
int j;
for (j=0;j < 7;j++)
GC_push_one(state.THREAD_FLD(r[j]));
j++;
for (;j<=12;j++)
GC_push_one(state.THREAD_FLD(r[j]));
}
GC_push_one(state.THREAD_FLD(lr));
#elif defined(AARCH64)
lo=(ptr_t)state.THREAD_FLD(sp);
#ifndef DARWIN_DONT_PARSE_STACK
*phi=GC_FindTopOfStack(state.THREAD_FLD(fp));
#endif
{
int j;
for (j=0;j<=28;j++){
GC_push_one(state.THREAD_FLD(x[j]));
}
}
GC_push_one(state.THREAD_FLD(lr));
#elif defined(CPPCHECK)
lo=NULL;
#else
#error FIXME for non-arm/ppc/x86 architectures
#endif
}
#ifdef DARWIN_DONT_PARSE_STACK
*phi=(p->flags&MAIN_THREAD)!=0?GC_stackbottom:p->stack_end;
#endif
#ifdef DARWIN_DONT_PARSE_STACK
if (p->altstack!=NULL&&(word)p->altstack<=(word)lo
&&(word)lo<=(word)p->altstack+p->altstack_size){
*paltstack_lo=lo;
*paltstack_hi=p->altstack+p->altstack_size;
lo=p->stack;
*phi=p->stack+p->stack_size;
} else
#endif
{
*paltstack_lo=NULL;
}
#ifdef DEBUG_THREADS
GC_log_printf("Darwin:Stack for thread %p=[%p,%p)\n",
(void*)(word)thread,(void*)lo,(void*)(*phi));
#endif
return lo;
}
GC_INNER void GC_push_all_stacks(void)
{
ptr_t hi,altstack_lo,altstack_hi;
task_t my_task=current_task();
mach_port_t my_thread=mach_thread_self();
GC_bool found_me=FALSE;
int nthreads=0;
word total_size=0;
mach_msg_type_number_t listcount=(mach_msg_type_number_t)THREAD_TABLE_SZ;
if (!EXPECT(GC_thr_initialized,TRUE))
GC_thr_init();
#ifndef DARWIN_DONT_PARSE_STACK
if (GC_query_task_threads){
int i;
kern_return_t kern_result;
thread_act_array_t act_list=0;
kern_result=task_threads(my_task,&act_list,&listcount);
if (kern_result!=KERN_SUCCESS)
ABORT("task_threads failed");
for (i=0;i < (int)listcount;i++){
thread_act_t thread=act_list[i];
ptr_t lo=GC_stack_range_for(&hi,thread,NULL,FALSE,my_thread,
&altstack_lo,&altstack_hi);
if (lo){
GC_ASSERT((word)lo<=(word)hi);
total_size+=hi - lo;
GC_push_all_stack(lo,hi);
}
nthreads++;
if (thread==my_thread)
found_me=TRUE;
mach_port_deallocate(my_task,thread);
}
vm_deallocate(my_task,(vm_address_t)act_list,
sizeof(thread_t)*listcount);
} else
#endif
{
int i;
for (i=0;i < (int)listcount;i++){
GC_thread p;
for (p=GC_threads[i];p!=NULL;p=p->next)
if ((p->flags&FINISHED)==0){
thread_act_t thread=(thread_act_t)p->stop_info.mach_thread;
ptr_t lo=GC_stack_range_for(&hi,thread,p,
(GC_bool)p->thread_blocked,
my_thread,&altstack_lo,
&altstack_hi);
if (lo){
GC_ASSERT((word)lo<=(word)hi);
total_size+=hi - lo;
GC_push_all_stack_sections(lo,hi,p->traced_stack_sect);
}
if (altstack_lo){
total_size+=altstack_hi - altstack_lo;
GC_push_all_stack(altstack_lo,altstack_hi);
}
nthreads++;
if (thread==my_thread)
found_me=TRUE;
}
}
}
mach_port_deallocate(my_task,my_thread);
GC_VERBOSE_LOG_PRINTF("Pushed %d thread stacks\n",nthreads);
if (!found_me&&!GC_in_thread_creation)
ABORT("Collecting from unknown thread");
GC_total_stacksize=total_size;
}
#ifndef GC_NO_THREADS_DISCOVERY
#ifdef MPROTECT_VDB
STATIC mach_port_t GC_mach_handler_thread=0;
STATIC GC_bool GC_use_mach_handler_thread=FALSE;
GC_INNER void GC_darwin_register_mach_handler_thread(mach_port_t thread)
{
GC_mach_handler_thread=thread;
GC_use_mach_handler_thread=TRUE;
}
#endif
#ifndef GC_MAX_MACH_THREADS
#define GC_MAX_MACH_THREADS THREAD_TABLE_SZ
#endif
struct GC_mach_thread {
thread_act_t thread;
GC_bool suspended;
};
struct GC_mach_thread GC_mach_threads[GC_MAX_MACH_THREADS];
STATIC int GC_mach_threads_count=0;
STATIC GC_bool GC_suspend_thread_list(thread_act_array_t act_list,int count,
thread_act_array_t old_list,
int old_count,task_t my_task,
mach_port_t my_thread)
{
int i;
int j=-1;
GC_bool changed=FALSE;
for (i=0;i < count;i++){
thread_act_t thread=act_list[i];
GC_bool found;
kern_return_t kern_result;
if (thread==my_thread
#ifdef MPROTECT_VDB
||(GC_mach_handler_thread==thread&&GC_use_mach_handler_thread)
#endif
#ifdef PARALLEL_MARK
||GC_is_mach_marker(thread)
#endif
){
mach_port_deallocate(my_task,thread);
continue;
}
found=FALSE;
{
int last_found=j;
while (++j < old_count)
if (old_list[j]==thread){
found=TRUE;
break;
}
if (!found){
for (j=0;j < last_found;j++)
if (old_list[j]==thread){
found=TRUE;
break;
}
}
}
if (found){
mach_port_deallocate(my_task,thread);
continue;
}
if (GC_mach_threads_count==GC_MAX_MACH_THREADS)
ABORT("Too many threads");
GC_mach_threads[GC_mach_threads_count].thread=thread;
GC_mach_threads[GC_mach_threads_count].suspended=FALSE;
changed=TRUE;
#ifdef DEBUG_THREADS
GC_log_printf("Suspending %p\n",(void*)(word)thread);
#endif
GC_acquire_dirty_lock();
do {
kern_result=thread_suspend(thread);
} while (kern_result==KERN_ABORTED);
GC_release_dirty_lock();
if (kern_result!=KERN_SUCCESS){
GC_mach_threads[GC_mach_threads_count].suspended=FALSE;
} else {
GC_mach_threads[GC_mach_threads_count].suspended=TRUE;
if (GC_on_thread_event)
GC_on_thread_event(GC_EVENT_THREAD_SUSPENDED,(void*)(word)thread);
}
GC_mach_threads_count++;
}
return changed;
}
#endif
GC_INNER void GC_stop_world(void)
{
task_t my_task=current_task();
mach_port_t my_thread=mach_thread_self();
kern_return_t kern_result;
#ifdef DEBUG_THREADS
GC_log_printf("Stopping the world from thread %p\n",
(void*)(word)my_thread);
#endif
#ifdef PARALLEL_MARK
if (GC_parallel){
GC_acquire_mark_lock();
GC_ASSERT(GC_fl_builder_count==0);
}
#endif
if (GC_query_task_threads){
#ifndef GC_NO_THREADS_DISCOVERY
GC_bool changed;
thread_act_array_t act_list,prev_list;
mach_msg_type_number_t listcount,prevcount;
GC_mach_threads_count=0;
changed=TRUE;
prev_list=NULL;
prevcount=0;
do {
kern_result=task_threads(my_task,&act_list,&listcount);
if (kern_result==KERN_SUCCESS){
changed=GC_suspend_thread_list(act_list,listcount,prev_list,
prevcount,my_task,my_thread);
if (prev_list!=NULL){
vm_deallocate(my_task,(vm_address_t)prev_list,
sizeof(thread_t)*prevcount);
}
prev_list=act_list;
prevcount=listcount;
}
} while (changed);
GC_ASSERT(prev_list!=0);
vm_deallocate(my_task,(vm_address_t)act_list,
sizeof(thread_t)*listcount);
#endif
} else {
unsigned i;
for (i=0;i < THREAD_TABLE_SZ;i++){
GC_thread p;
for (p=GC_threads[i];p!=NULL;p=p->next){
if ((p->flags&FINISHED)==0&&!p->thread_blocked&&
p->stop_info.mach_thread!=my_thread){
GC_acquire_dirty_lock();
do {
kern_result=thread_suspend(p->stop_info.mach_thread);
} while (kern_result==KERN_ABORTED);
GC_release_dirty_lock();
if (kern_result!=KERN_SUCCESS)
ABORT("thread_suspend failed");
if (GC_on_thread_event)
GC_on_thread_event(GC_EVENT_THREAD_SUSPENDED,
(void*)(word)p->stop_info.mach_thread);
}
}
}
}
#ifdef MPROTECT_VDB
if (GC_auto_incremental){
GC_mprotect_stop();
}
#endif
#ifdef PARALLEL_MARK
if (GC_parallel)
GC_release_mark_lock();
#endif
#ifdef DEBUG_THREADS
GC_log_printf("World stopped from %p\n",(void*)(word)my_thread);
#endif
mach_port_deallocate(my_task,my_thread);
}
GC_INLINE void GC_thread_resume(thread_act_t thread)
{
kern_return_t kern_result;
#if defined(DEBUG_THREADS)||defined(GC_ASSERTIONS)
struct thread_basic_info info;
mach_msg_type_number_t outCount=THREAD_BASIC_INFO_COUNT;
#if defined(CPPCHECK)&&defined(DEBUG_THREADS)
info.run_state=0;
#endif
kern_result=thread_info(thread,THREAD_BASIC_INFO,
(thread_info_t)&info,&outCount);
if (kern_result!=KERN_SUCCESS)
ABORT("thread_info failed");
#endif
#ifdef DEBUG_THREADS
GC_log_printf("Resuming thread %p with state %d\n",(void*)(word)thread,
info.run_state);
#endif
kern_result=thread_resume(thread);
if (kern_result!=KERN_SUCCESS){
WARN("thread_resume(%p)failed:mach port invalid\n",thread);
} else if (GC_on_thread_event){
GC_on_thread_event(GC_EVENT_THREAD_UNSUSPENDED,(void*)(word)thread);
}
}
GC_INNER void GC_start_world(void)
{
task_t my_task=current_task();
#ifdef DEBUG_THREADS
GC_log_printf("World starting\n");
#endif
#ifdef MPROTECT_VDB
if (GC_auto_incremental){
GC_mprotect_resume();
}
#endif
if (GC_query_task_threads){
#ifndef GC_NO_THREADS_DISCOVERY
int i,j;
kern_return_t kern_result;
thread_act_array_t act_list;
mach_msg_type_number_t listcount;
kern_result=task_threads(my_task,&act_list,&listcount);
if (kern_result!=KERN_SUCCESS)
ABORT("task_threads failed");
j=(int)listcount;
for (i=0;i < GC_mach_threads_count;i++){
thread_act_t thread=GC_mach_threads[i].thread;
if (GC_mach_threads[i].suspended){
int last_found=j;
while (++j < (int)listcount){
if (act_list[j]==thread)
break;
}
if (j>=(int)listcount){
for (j=0;j < last_found;j++){
if (act_list[j]==thread)
break;
}
}
if (j!=last_found){
GC_thread_resume(thread);
}
} else {
#ifdef DEBUG_THREADS
GC_log_printf("Not resuming thread %p as it is not suspended\n",
(void*)(word)thread);
#endif
}
mach_port_deallocate(my_task,thread);
}
for (i=0;i < (int)listcount;i++)
mach_port_deallocate(my_task,act_list[i]);
vm_deallocate(my_task,(vm_address_t)act_list,
sizeof(thread_t)*listcount);
#endif
} else {
int i;
mach_port_t my_thread=mach_thread_self();
for (i=0;i < THREAD_TABLE_SZ;i++){
GC_thread p;
for (p=GC_threads[i];p!=NULL;p=p->next){
if ((p->flags&FINISHED)==0&&!p->thread_blocked&&
p->stop_info.mach_thread!=my_thread)
GC_thread_resume(p->stop_info.mach_thread);
}
}
mach_port_deallocate(my_task,my_thread);
}
#ifdef DEBUG_THREADS
GC_log_printf("World started\n");
#endif
}
#endif
#if!defined(MACOS)&&!defined(SN_TARGET_ORBIS)&&!defined(SN_TARGET_PSP2)&&!defined(_WIN32_WCE)&&!defined(__CC_ARM)
#include <sys/types.h>
#endif
#undef GC_MUST_RESTORE_REDEFINED_DLOPEN
#if defined(GC_PTHREADS)&&!defined(GC_NO_DLOPEN)&&!defined(GC_NO_THREAD_REDIRECTS)&&!defined(GC_USE_LD_WRAP)
#undef dlopen
#define GC_MUST_RESTORE_REDEFINED_DLOPEN
#endif
STATIC GC_has_static_roots_func GC_has_static_roots=0;
#if (defined(DYNAMIC_LOADING)||defined(MSWIN32)||defined(MSWINCE)||defined(CYGWIN32))&&!defined(PCR)
#if!defined(DARWIN)&&!defined(SCO_ELF)&&!defined(SOLARISDL)&&!defined(AIX)&&!defined(DGUX)&&!defined(IRIX5)&&!defined(HPUX)&&!defined(CYGWIN32)&&!defined(MSWIN32)&&!defined(MSWINCE)&&!(defined(ALPHA)&&defined(OSF1))&&!(defined(FREEBSD)&&defined(__ELF__))&&!(defined(LINUX)&&defined(__ELF__))&&!(defined(NETBSD)&&defined(__ELF__))&&!(defined(OPENBSD)&&(defined(__ELF__)||defined(M68K)))&&!defined(HAIKU)&&!defined(HURD)&&!defined(NACL)&&!defined(CPPCHECK)
#error We only know how to find data segments of dynamic libraries for above.
#error Additional SVR4 variants might not be too hard to add.
#endif
#include <stdio.h>
#ifdef SOLARISDL
#include <sys/elf.h>
#include <dlfcn.h>
#include <link.h>
#endif
#if defined(NETBSD)
#include <sys/param.h>
#include <dlfcn.h>
#include <machine/elf_machdep.h>
#define ELFSIZE ARCH_ELFSIZE
#endif
#if defined(OPENBSD)
#include <sys/param.h>
#if (OpenBSD>=200519)&&!defined(HAVE_DL_ITERATE_PHDR)
#define HAVE_DL_ITERATE_PHDR
#endif
#endif
#if defined(SCO_ELF)||defined(DGUX)||defined(HURD)||defined(NACL)||(defined(__ELF__)&&(defined(LINUX)||defined(FREEBSD)||defined(NETBSD)||defined(OPENBSD)))
#include <stddef.h>
#if!defined(OPENBSD)&&!defined(HOST_ANDROID)
#include <elf.h>
#endif
#ifdef HOST_ANDROID
#ifdef BIONIC_ELFDATA_REDEF_BUG
#include <asm/elf.h>
#include <linux/elf-em.h>
#undef ELF_DATA
#undef EM_ALPHA
#endif
#include <link.h>
#if!defined(GC_DONT_DEFINE_LINK_MAP)&&!(__ANDROID_API__>=21)
struct link_map {
uintptr_t l_addr;
char*l_name;
uintptr_t l_ld;
struct link_map*l_next;
struct link_map*l_prev;
};
struct r_debug {
int32_t r_version;
struct link_map*r_map;
void (*r_brk)(void);
int32_t r_state;
uintptr_t r_ldbase;
};
#endif
#else
EXTERN_C_BEGIN
#include <link.h>
EXTERN_C_END
#endif
#endif
#ifndef ElfW
#if defined(FREEBSD)
#if __ELF_WORD_SIZE==32
#define ElfW(type)Elf32_##type
#else
#define ElfW(type)Elf64_##type
#endif
#elif defined(NETBSD)||defined(OPENBSD)
#if ELFSIZE==32
#define ElfW(type)Elf32_##type
#elif ELFSIZE==64
#define ElfW(type)Elf64_##type
#else
#error Missing ELFSIZE define
#endif
#else
#if!defined(ELF_CLASS)||ELF_CLASS==ELFCLASS32
#define ElfW(type)Elf32_##type
#else
#define ElfW(type)Elf64_##type
#endif
#endif
#endif
#if defined(SOLARISDL)&&!defined(USE_PROC_FOR_LIBRARIES)
EXTERN_C_BEGIN
extern ElfW(Dyn)_DYNAMIC;
EXTERN_C_END
STATIC struct link_map*
GC_FirstDLOpenedLinkMap(void)
{
ElfW(Dyn)*dp;
static struct link_map*cachedResult=0;
static ElfW(Dyn)*dynStructureAddr=0;
#ifdef SUNOS53_SHARED_LIB
if( dynStructureAddr==0){
void*startupSyms=dlopen(0,RTLD_LAZY);
dynStructureAddr=(ElfW(Dyn)*)(word)dlsym(startupSyms,"_DYNAMIC");
}
#else
dynStructureAddr=&_DYNAMIC;
#endif
if (0==COVERT_DATAFLOW(dynStructureAddr)){
return(0);
}
if (cachedResult==0){
int tag;
for( dp=((ElfW(Dyn)*)(&_DYNAMIC));(tag=dp->d_tag)!=0;dp++){
if (tag==DT_DEBUG){
struct r_debug*rd=(struct r_debug*)dp->d_un.d_ptr;
if (rd!=NULL){
struct link_map*lm=rd->r_map;
if (lm!=NULL)
cachedResult=lm->l_next;
}
break;
}
}
}
return cachedResult;
}
#endif
#ifdef GC_MUST_RESTORE_REDEFINED_DLOPEN
#define dlopen GC_dlopen
#endif
#if defined(SOLARISDL)
#if!defined(PCR)&&!defined(GC_SOLARIS_THREADS)&&defined(THREADS)&&!defined(CPPCHECK)
#error Fix mutual exclusion with dlopen
#endif
#ifndef USE_PROC_FOR_LIBRARIES
GC_INNER void GC_register_dynamic_libraries(void)
{
struct link_map*lm;
for (lm=GC_FirstDLOpenedLinkMap();lm!=0;lm=lm->l_next){
ElfW(Ehdr)*e;
ElfW(Phdr)*p;
unsigned long offset;
char*start;
int i;
e=(ElfW(Ehdr)*)lm->l_addr;
p=((ElfW(Phdr)*)(((char*)(e))+e->e_phoff));
offset=((unsigned long)(lm->l_addr));
for( i=0;i < (int)e->e_phnum;i++,p++){
switch( p->p_type){
case PT_LOAD:
{
if(!(p->p_flags&PF_W))break;
start=((char*)(p->p_vaddr))+offset;
GC_add_roots_inner(start,start+p->p_memsz,TRUE);
}
break;
default:
break;
}
}
}
}
#endif
#endif
#if defined(SCO_ELF)||defined(DGUX)||defined(HURD)||defined(NACL)||(defined(__ELF__)&&(defined(LINUX)||defined(FREEBSD)||defined(NETBSD)||defined(OPENBSD)))
#ifdef USE_PROC_FOR_LIBRARIES
#include <string.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#define MAPS_BUF_SIZE (32*1024)
static void sort_heap_sects(struct HeapSect*base,size_t number_of_elements)
{
signed_word n=(signed_word)number_of_elements;
signed_word nsorted=1;
while (nsorted < n){
signed_word i;
while (nsorted < n&&
(word)base[nsorted-1].hs_start < (word)base[nsorted].hs_start)
++nsorted;
if (nsorted==n)break;
GC_ASSERT((word)base[nsorted-1].hs_start > (word)base[nsorted].hs_start);
i=nsorted - 1;
while (i>=0&&(word)base[i].hs_start > (word)base[i+1].hs_start){
struct HeapSect tmp=base[i];
base[i]=base[i+1];
base[i+1]=tmp;
--i;
}
GC_ASSERT((word)base[nsorted-1].hs_start < (word)base[nsorted].hs_start);
++nsorted;
}
}
STATIC void GC_register_map_entries(char*maps)
{
char*prot;
char*buf_ptr=maps;
ptr_t start,end;
unsigned int maj_dev;
ptr_t least_ha,greatest_ha;
unsigned i;
GC_ASSERT(I_HOLD_LOCK());
sort_heap_sects(GC_our_memory,GC_n_memory);
least_ha=GC_our_memory[0].hs_start;
greatest_ha=GC_our_memory[GC_n_memory-1].hs_start
+GC_our_memory[GC_n_memory-1].hs_bytes;
for (;;){
buf_ptr=GC_parse_map_entry(buf_ptr,&start,&end,&prot,
&maj_dev,0);
if (NULL==buf_ptr)
break;
if (prot[1]=='w'){
if ((word)start<=(word)GC_stackbottom
&&(word)end>=(word)GC_stackbottom){
continue;
}
#ifdef THREADS
if (GC_segment_is_thread_stack(start,end))continue;
#endif
if ((word)end<=(word)least_ha
||(word)start>=(word)greatest_ha){
GC_add_roots_inner(start,end,TRUE);
continue;
}
i=0;
while ((word)(GC_our_memory[i].hs_start
+GC_our_memory[i].hs_bytes)< (word)start)
++i;
GC_ASSERT(i < GC_n_memory);
if ((word)GC_our_memory[i].hs_start<=(word)start){
start=GC_our_memory[i].hs_start
+GC_our_memory[i].hs_bytes;
++i;
}
while (i < GC_n_memory
&&(word)GC_our_memory[i].hs_start < (word)end
&&(word)start < (word)end){
if ((word)start < (word)GC_our_memory[i].hs_start)
GC_add_roots_inner(start,
GC_our_memory[i].hs_start,TRUE);
start=GC_our_memory[i].hs_start
+GC_our_memory[i].hs_bytes;
++i;
}
if ((word)start < (word)end)
GC_add_roots_inner(start,end,TRUE);
} else if (prot[0]=='-'&&prot[1]=='-'&&prot[2]=='-'){
GC_remove_roots_subregion(start,end);
}
}
}
GC_INNER void GC_register_dynamic_libraries(void)
{
char*maps=GC_get_maps();
if (NULL==maps)
ABORT("Failed to read/proc for library registration");
GC_register_map_entries(maps);
}
GC_INNER GC_bool GC_register_main_static_data(void)
{
return FALSE;
}
#define HAVE_REGISTER_MAIN_STATIC_DATA
#else
#if __GLIBC__ > 2||(__GLIBC__==2&&__GLIBC_MINOR__ > 2)||(__GLIBC__==2&&__GLIBC_MINOR__==2&&defined(DT_CONFIG))||defined(HOST_ANDROID)
#ifndef HAVE_DL_ITERATE_PHDR
#define HAVE_DL_ITERATE_PHDR
#endif
#ifdef HOST_ANDROID
EXTERN_C_BEGIN
extern int dl_iterate_phdr(int (*cb)(struct dl_phdr_info*,
size_t,void*),
void*data);
EXTERN_C_END
#endif
#endif
#if defined(__DragonFly__)||defined(__FreeBSD_kernel__)||(defined(FREEBSD)&&__FreeBSD__>=7)
#ifndef HAVE_DL_ITERATE_PHDR
#define HAVE_DL_ITERATE_PHDR
#endif
#define DL_ITERATE_PHDR_STRONG
#elif defined(HAVE_DL_ITERATE_PHDR)
EXTERN_C_BEGIN
#pragma weak dl_iterate_phdr
EXTERN_C_END
#endif
#if defined(HAVE_DL_ITERATE_PHDR)
#ifdef PT_GNU_RELRO
#define MAX_LOAD_SEGS MAX_ROOT_SETS
static struct load_segment {
ptr_t start;
ptr_t end;
ptr_t start2;
ptr_t end2;
} load_segs[MAX_LOAD_SEGS];
static int n_load_segs;
static GC_bool load_segs_overflow;
#endif
STATIC int GC_register_dynlib_callback(struct dl_phdr_info*info,
size_t size,void*ptr)
{
const ElfW(Phdr)*p;
ptr_t start,end;
int i;
if (size < offsetof (struct dl_phdr_info,dlpi_phnum)
+sizeof (info->dlpi_phnum))
return -1;
p=info->dlpi_phdr;
for (i=0;i < (int)info->dlpi_phnum;i++,p++){
if (p->p_type==PT_LOAD){
GC_has_static_roots_func callback=GC_has_static_roots;
if ((p->p_flags&PF_W)==0)continue;
start=(ptr_t)p->p_vaddr+info->dlpi_addr;
end=start+p->p_memsz;
if (callback!=0&&!callback(info->dlpi_name,start,p->p_memsz))
continue;
#ifdef PT_GNU_RELRO
#if CPP_WORDSZ==64
start=(ptr_t)((word)start&~(word)(sizeof(word)- 1));
#endif
if (n_load_segs>=MAX_LOAD_SEGS){
if (!load_segs_overflow){
WARN("Too many PT_LOAD segments;"
" registering as roots directly...\n",0);
load_segs_overflow=TRUE;
}
GC_add_roots_inner(start,end,TRUE);
} else {
load_segs[n_load_segs].start=start;
load_segs[n_load_segs].end=end;
load_segs[n_load_segs].start2=0;
load_segs[n_load_segs].end2=0;
++n_load_segs;
}
#else
GC_add_roots_inner(start,end,TRUE);
#endif
}
}
#ifdef PT_GNU_RELRO
p=info->dlpi_phdr;
for (i=0;i < (int)info->dlpi_phnum;i++,p++){
if (p->p_type==PT_GNU_RELRO){
int j;
start=(ptr_t)p->p_vaddr+info->dlpi_addr;
end=start+p->p_memsz;
for (j=n_load_segs;--j>=0;){
if ((word)start>=(word)load_segs[j].start
&&(word)start < (word)load_segs[j].end){
if (load_segs[j].start2!=0){
WARN("More than one GNU_RELRO segment per load one\n",0);
} else {
GC_ASSERT((word)end<=(word)load_segs[j].end);
load_segs[j].end2=load_segs[j].end;
load_segs[j].end=start;
load_segs[j].start2=end;
}
break;
}
if (0==j&&0==GC_has_static_roots)
WARN("Failed to find PT_GNU_RELRO segment"
" inside PT_LOAD region\n",0);
}
}
}
#endif
*(int*)ptr=1;
return 0;
}
GC_INNER GC_bool GC_register_main_static_data(void)
{
#ifdef DL_ITERATE_PHDR_STRONG
return FALSE;
#else
return 0==COVERT_DATAFLOW(dl_iterate_phdr);
#endif
}
STATIC GC_bool GC_register_dynamic_libraries_dl_iterate_phdr(void)
{
int did_something;
if (GC_register_main_static_data())
return FALSE;
#ifdef PT_GNU_RELRO
{
static GC_bool excluded_segs=FALSE;
n_load_segs=0;
load_segs_overflow=FALSE;
if (!EXPECT(excluded_segs,TRUE)){
GC_exclude_static_roots_inner((ptr_t)load_segs,
(ptr_t)load_segs+sizeof(load_segs));
excluded_segs=TRUE;
}
}
#endif
did_something=0;
dl_iterate_phdr(GC_register_dynlib_callback,&did_something);
if (did_something){
#ifdef PT_GNU_RELRO
int i;
for (i=0;i < n_load_segs;++i){
if ((word)load_segs[i].end > (word)load_segs[i].start){
GC_add_roots_inner(load_segs[i].start,load_segs[i].end,TRUE);
}
if ((word)load_segs[i].end2 > (word)load_segs[i].start2){
GC_add_roots_inner(load_segs[i].start2,load_segs[i].end2,TRUE);
}
}
#endif
} else {
ptr_t datastart,dataend;
#ifdef DATASTART_IS_FUNC
static ptr_t datastart_cached=(ptr_t)GC_WORD_MAX;
if (datastart_cached==(ptr_t)GC_WORD_MAX){
datastart_cached=DATASTART;
}
datastart=datastart_cached;
#else
datastart=DATASTART;
#endif
#ifdef DATAEND_IS_FUNC
{
static ptr_t dataend_cached=0;
if (dataend_cached==0){
dataend_cached=DATAEND;
}
dataend=dataend_cached;
}
#else
dataend=DATAEND;
#endif
if (NULL==*(char*volatile*)&datastart
||(word)datastart > (word)dataend)
ABORT_ARG2("Wrong DATASTART/END pair",
":%p .. %p",(void*)datastart,(void*)dataend);
GC_add_roots_inner(datastart,dataend,TRUE);
#ifdef GC_HAVE_DATAREGION2
if ((word)DATASTART2 - 1U>=(word)DATAEND2){
ABORT_ARG2("Wrong DATASTART/END2 pair",
":%p .. %p",(void*)DATASTART2,(void*)DATAEND2);
}
GC_add_roots_inner(DATASTART2,DATAEND2,TRUE);
#endif
}
return TRUE;
}
#define HAVE_REGISTER_MAIN_STATIC_DATA
#else
#if defined(NETBSD)||defined(OPENBSD)
#include <sys/exec_elf.h>
#ifndef DT_DEBUG
#define DT_DEBUG 21
#endif
#ifndef PT_LOAD
#define PT_LOAD 1
#endif
#ifndef PF_W
#define PF_W 2
#endif
#elif!defined(HOST_ANDROID)
#include <elf.h>
#endif
#ifndef HOST_ANDROID
#include <link.h>
#endif
#endif
EXTERN_C_BEGIN
#ifdef __GNUC__
#pragma weak _DYNAMIC
#endif
extern ElfW(Dyn)_DYNAMIC[];
EXTERN_C_END
STATIC struct link_map*
GC_FirstDLOpenedLinkMap(void)
{
static struct link_map*cachedResult=0;
if (0==COVERT_DATAFLOW(_DYNAMIC)){
return(0);
}
if( cachedResult==0){
#if defined(NETBSD)&&defined(RTLD_DI_LINKMAP)
#if defined(CPPCHECK)
#define GC_RTLD_DI_LINKMAP 2
#else
#define GC_RTLD_DI_LINKMAP RTLD_DI_LINKMAP
#endif
struct link_map*lm=NULL;
if (!dlinfo(RTLD_SELF,GC_RTLD_DI_LINKMAP,&lm)&&lm!=NULL){
while (lm->l_prev!=NULL){
lm=lm->l_prev;
}
cachedResult=lm->l_next;
}
#else
ElfW(Dyn)*dp;
int tag;
for( dp=_DYNAMIC;(tag=dp->d_tag)!=0;dp++){
if (tag==DT_DEBUG){
struct r_debug*rd=(struct r_debug*)dp->d_un.d_ptr;
if (rd!=NULL){
struct link_map*lm=rd->r_map;
if (lm!=NULL)
cachedResult=lm->l_next;
}
break;
}
}
#endif
}
return cachedResult;
}
GC_INNER void GC_register_dynamic_libraries(void)
{
struct link_map*lm;
#ifdef HAVE_DL_ITERATE_PHDR
if (GC_register_dynamic_libraries_dl_iterate_phdr()){
return;
}
#endif
for (lm=GC_FirstDLOpenedLinkMap();lm!=0;lm=lm->l_next)
{
ElfW(Ehdr)*e;
ElfW(Phdr)*p;
unsigned long offset;
char*start;
int i;
e=(ElfW(Ehdr)*)lm->l_addr;
#ifdef HOST_ANDROID
if (e==NULL)
continue;
#endif
p=((ElfW(Phdr)*)(((char*)(e))+e->e_phoff));
offset=((unsigned long)(lm->l_addr));
for( i=0;i < (int)e->e_phnum;i++,p++){
switch( p->p_type){
case PT_LOAD:
{
if(!(p->p_flags&PF_W))break;
start=((char*)(p->p_vaddr))+offset;
GC_add_roots_inner(start,start+p->p_memsz,TRUE);
}
break;
default:
break;
}
}
}
}
#endif
#endif
#if defined(IRIX5)||(defined(USE_PROC_FOR_LIBRARIES)&&!defined(LINUX))
#include <sys/procfs.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <elf.h>
#include <errno.h>
#include <signal.h>
#ifndef _sigargs
#define IRIX6
#endif
GC_INNER void GC_register_dynamic_libraries(void)
{
static int fd=-1;
char buf[30];
static prmap_t*addr_map=0;
static int current_sz=0;
int needed_sz=0;
int i;
long flags;
ptr_t start;
ptr_t limit;
ptr_t heap_start=HEAP_START;
ptr_t heap_end=heap_start;
#ifdef SOLARISDL
#define MA_PHYS 0
#endif
if (fd < 0){
(void)snprintf(buf,sizeof(buf),"/proc/%ld",(long)getpid());
buf[sizeof(buf)- 1]='\0';
fd=open(buf,O_RDONLY);
if (fd < 0){
ABORT("/proc open failed");
}
}
if (ioctl(fd,PIOCNMAP,&needed_sz)< 0){
ABORT_ARG2("/proc PIOCNMAP ioctl failed",
":fd=%d,errno=%d",fd,errno);
}
if (needed_sz>=current_sz){
GC_scratch_recycle_no_gww(addr_map,
(size_t)current_sz*sizeof(prmap_t));
current_sz=needed_sz*2+1;
addr_map=(prmap_t*)GC_scratch_alloc(
(size_t)current_sz*sizeof(prmap_t));
if (addr_map==NULL)
ABORT("Insufficient memory for address map");
}
if (ioctl(fd,PIOCMAP,addr_map)< 0){
ABORT_ARG3("/proc PIOCMAP ioctl failed",
":errcode=%d,needed_sz=%d,addr_map=%p",
errno,needed_sz,(void*)addr_map);
};
if (GC_n_heap_sects > 0){
heap_end=GC_heap_sects[GC_n_heap_sects-1].hs_start
+GC_heap_sects[GC_n_heap_sects-1].hs_bytes;
if ((word)heap_end < (word)GC_scratch_last_end_ptr)
heap_end=GC_scratch_last_end_ptr;
}
for (i=0;i < needed_sz;i++){
flags=addr_map[i].pr_mflags;
if ((flags&(MA_BREAK|MA_STACK|MA_PHYS
|MA_FETCHOP|MA_NOTCACHED))!=0)goto irrelevant;
if ((flags&(MA_READ|MA_WRITE))!=(MA_READ|MA_WRITE))
goto irrelevant;
start=(ptr_t)(addr_map[i].pr_vaddr);
if (GC_roots_present(start))goto irrelevant;
if ((word)start < (word)heap_end&&(word)start>=(word)heap_start)
goto irrelevant;
limit=start+addr_map[i].pr_size;
#ifndef IRIX6
if (addr_map[i].pr_off==0&&strncmp(start,ELFMAG,4)==0){
caddr_t arg;
int obj;
#define MAP_IRR_SZ 10
static ptr_t map_irr[MAP_IRR_SZ];
static int n_irr=0;
struct stat buf;
int j;
for (j=0;j < n_irr;j++){
if (map_irr[j]==start)goto irrelevant;
}
arg=(caddr_t)start;
obj=ioctl(fd,PIOCOPENM,&arg);
if (obj>=0){
fstat(obj,&buf);
close(obj);
if ((buf.st_mode&0111)!=0){
if (n_irr < MAP_IRR_SZ){
map_irr[n_irr++]=start;
}
goto irrelevant;
}
}
}
#endif
GC_add_roots_inner(start,limit,TRUE);
irrelevant:;
}
if (close(fd)< 0)ABORT("Couldn't close/proc file");
fd=-1;
}
#endif
#if defined(MSWIN32)||defined(MSWINCE)||defined(CYGWIN32)
#include <stdlib.h>
STATIC void GC_cond_add_roots(char*base,char*limit)
{
#ifdef GC_WIN32_THREADS
char*curr_base=base;
char*next_stack_lo;
char*next_stack_hi;
if (base==limit)return;
for(;;){
GC_get_next_stack(curr_base,limit,&next_stack_lo,&next_stack_hi);
if ((word)next_stack_lo>=(word)limit)break;
if ((word)next_stack_lo > (word)curr_base)
GC_add_roots_inner(curr_base,next_stack_lo,TRUE);
curr_base=next_stack_hi;
}
if ((word)curr_base < (word)limit)
GC_add_roots_inner(curr_base,limit,TRUE);
#else
char*stack_top
=(char*)((word)GC_approx_sp()&
~(word)(GC_sysinfo.dwAllocationGranularity - 1));
if (base==limit)return;
if ((word)limit > (word)stack_top
&&(word)base < (word)GC_stackbottom){
return;
}
GC_add_roots_inner(base,limit,TRUE);
#endif
}
#ifdef DYNAMIC_LOADING
GC_INNER GC_bool GC_register_main_static_data(void)
{
#if defined(MSWINCE)||defined(CYGWIN32)
return FALSE;
#else
return GC_no_win32_dlls;
#endif
}
#define HAVE_REGISTER_MAIN_STATIC_DATA
#endif
#ifdef DEBUG_VIRTUALQUERY
void GC_dump_meminfo(MEMORY_BASIC_INFORMATION*buf)
{
GC_printf("BaseAddress=0x%lx,AllocationBase=0x%lx,"
" RegionSize=0x%lx(%lu)\n",buf->BaseAddress,
buf->AllocationBase,buf->RegionSize,buf->RegionSize);
GC_printf("\tAllocationProtect=0x%lx,State=0x%lx,Protect=0x%lx,"
"Type=0x%lx\n",buf->AllocationProtect,buf->State,
buf->Protect,buf->Type);
}
#endif
#if defined(MSWINCE)||defined(CYGWIN32)
#define GC_wnt TRUE
#endif
GC_INNER void GC_register_dynamic_libraries(void)
{
MEMORY_BASIC_INFORMATION buf;
DWORD protect;
LPVOID p;
char*base;
char*limit,*new_limit;
#ifdef MSWIN32
if (GC_no_win32_dlls)return;
#endif
p=GC_sysinfo.lpMinimumApplicationAddress;
base=limit=(char*)p;
while ((word)p < (word)GC_sysinfo.lpMaximumApplicationAddress){
size_t result=VirtualQuery(p,&buf,sizeof(buf));
#ifdef MSWINCE
if (result==0){
new_limit=(char*)
(((DWORD)p+GC_sysinfo.dwAllocationGranularity)
&~(GC_sysinfo.dwAllocationGranularity-1));
} else
#endif
{
if (result!=sizeof(buf)){
ABORT("Weird VirtualQuery result");
}
new_limit=(char*)p+buf.RegionSize;
protect=buf.Protect;
if (buf.State==MEM_COMMIT
&&(protect==PAGE_EXECUTE_READWRITE
||protect==PAGE_EXECUTE_WRITECOPY
||protect==PAGE_READWRITE
||protect==PAGE_WRITECOPY)
&&(buf.Type==MEM_IMAGE
#ifdef GC_REGISTER_MEM_PRIVATE
||(protect==PAGE_READWRITE&&buf.Type==MEM_PRIVATE)
#else
||(!GC_wnt&&buf.Type==MEM_PRIVATE)
#endif
)
&&!GC_is_heap_base(buf.AllocationBase)){
#ifdef DEBUG_VIRTUALQUERY
GC_dump_meminfo(&buf);
#endif
if ((char*)p!=limit){
GC_cond_add_roots(base,limit);
base=(char*)p;
}
limit=new_limit;
}
}
if ((word)p > (word)new_limit)break;
p=(LPVOID)new_limit;
}
GC_cond_add_roots(base,limit);
}
#endif
#if defined(ALPHA)&&defined(OSF1)
#include <loader.h>
EXTERN_C_BEGIN
extern char*sys_errlist[];
extern int sys_nerr;
extern int errno;
EXTERN_C_END
GC_INNER void GC_register_dynamic_libraries(void)
{
ldr_module_t moduleid=LDR_NULL_MODULE;
ldr_process_t mypid=ldr_my_process();
while (TRUE){
ldr_module_info_t moduleinfo;
size_t modulereturnsize;
ldr_region_t region;
ldr_region_info_t regioninfo;
size_t regionreturnsize;
int status=ldr_next_module(mypid,&moduleid);
if (moduleid==LDR_NULL_MODULE)
break;
if (status!=0){
ABORT_ARG3("ldr_next_module failed",
":status=%d,errcode=%d (%s)",status,errno,
errno < sys_nerr?sys_errlist[errno]:"");
}
status=ldr_inq_module(mypid,moduleid,&moduleinfo,
sizeof(moduleinfo),&modulereturnsize);
if (status!=0)
ABORT("ldr_inq_module failed");
if (moduleinfo.lmi_flags&LDR_MAIN)
continue;
#ifdef DL_VERBOSE
GC_log_printf("---Module---\n");
GC_log_printf("Module ID\t=%16ld\n",moduleinfo.lmi_modid);
GC_log_printf("Count of regions=%16d\n",moduleinfo.lmi_nregion);
GC_log_printf("flags for module=%16lx\n",moduleinfo.lmi_flags);
GC_log_printf("module pathname\t=\"%s\"\n",moduleinfo.lmi_name);
#endif
for (region=0;region < moduleinfo.lmi_nregion;region++){
status=ldr_inq_region(mypid,moduleid,region,&regioninfo,
sizeof(regioninfo),&regionreturnsize);
if (status!=0)
ABORT("ldr_inq_region failed");
if (!(regioninfo.lri_prot&LDR_W))
continue;
#ifdef DL_VERBOSE
GC_log_printf("--- Region---\n");
GC_log_printf("Region number\t=%16ld\n",
regioninfo.lri_region_no);
GC_log_printf("Protection flags=%016x\n",regioninfo.lri_prot);
GC_log_printf("Virtual address\t=%16p\n",regioninfo.lri_vaddr);
GC_log_printf("Mapped address\t=%16p\n",
regioninfo.lri_mapaddr);
GC_log_printf("Region size\t=%16ld\n",regioninfo.lri_size);
GC_log_printf("Region name\t=\"%s\"\n",regioninfo.lri_name);
#endif
GC_add_roots_inner((char*)regioninfo.lri_mapaddr,
(char*)regioninfo.lri_mapaddr+regioninfo.lri_size,
TRUE);
}
}
}
#endif
#if defined(HPUX)
#include <errno.h>
#include <dl.h>
EXTERN_C_BEGIN
extern char*sys_errlist[];
extern int sys_nerr;
EXTERN_C_END
GC_INNER void GC_register_dynamic_libraries(void)
{
int index=1;
while (TRUE){
struct shl_descriptor*shl_desc;
int status=shl_get(index,&shl_desc);
if (status!=0){
#ifdef GC_HPUX_THREADS
break;
#else
if (errno==EINVAL){
break;
} else {
ABORT_ARG3("shl_get failed",
":status=%d,errcode=%d (%s)",status,errno,
errno < sys_nerr?sys_errlist[errno]:"");
}
#endif
}
#ifdef DL_VERBOSE
GC_log_printf("---Shared library---\n");
GC_log_printf("\tfilename\t=\"%s\"\n",shl_desc->filename);
GC_log_printf("\tindex\t\t=%d\n",index);
GC_log_printf("\thandle\t\t=%08x\n",
(unsigned long)shl_desc->handle);
GC_log_printf("\ttext seg.start\t=%08x\n",shl_desc->tstart);
GC_log_printf("\ttext seg.end\t=%08x\n",shl_desc->tend);
GC_log_printf("\tdata seg.start\t=%08x\n",shl_desc->dstart);
GC_log_printf("\tdata seg.end\t=%08x\n",shl_desc->dend);
GC_log_printf("\tref.count\t=%lu\n",shl_desc->ref_count);
#endif
GC_add_roots_inner((char*)shl_desc->dstart,
(char*)shl_desc->dend,TRUE);
index++;
}
}
#endif
#ifdef AIX
#include <alloca.h>
#include <sys/ldr.h>
#include <sys/errno.h>
GC_INNER void GC_register_dynamic_libraries(void)
{
int ldibuflen=8192;
for (;;){
int len;
struct ld_info*ldi;
#if defined(CPPCHECK)
char ldibuf[ldibuflen];
#else
char*ldibuf=alloca(ldibuflen);
#endif
len=loadquery(L_GETINFO,ldibuf,ldibuflen);
if (len < 0){
if (errno!=ENOMEM){
ABORT("loadquery failed");
}
ldibuflen*=2;
continue;
}
ldi=(struct ld_info*)ldibuf;
while (ldi){
len=ldi->ldinfo_next;
GC_add_roots_inner(
ldi->ldinfo_dataorg,
(ptr_t)(unsigned long)ldi->ldinfo_dataorg
+ldi->ldinfo_datasize,
TRUE);
ldi=len?(struct ld_info*)((char*)ldi+len):0;
}
break;
}
}
#endif
#ifdef DARWIN
#ifndef __private_extern__
#define __private_extern__ extern
#include <mach-o/dyld.h>
#undef __private_extern__
#else
#include <mach-o/dyld.h>
#endif
#include <mach-o/getsect.h>
STATIC const struct dyld_sections_s {
const char*seg;
const char*sect;
} GC_dyld_sections[]={
{ SEG_DATA,SECT_DATA },
{ SEG_DATA,"__static_data" },
{ SEG_DATA,SECT_BSS },
{ SEG_DATA,SECT_COMMON },
{ SEG_DATA,"__zobj_data" },
{ SEG_DATA,"__zobj_bss" }
};
STATIC const char*const GC_dyld_add_sect_fmts[]={
"__bss%u",
"__pu_bss%u",
"__zo_bss%u",
"__zo_pu_bss%u"
};
#ifndef L2_MAX_OFILE_ALIGNMENT
#define L2_MAX_OFILE_ALIGNMENT 15
#endif
STATIC const char*GC_dyld_name_for_hdr(const struct GC_MACH_HEADER*hdr)
{
unsigned long i,c;
c=_dyld_image_count();
for (i=0;i < c;i++)
if ((const struct GC_MACH_HEADER*)_dyld_get_image_header(i)==hdr)
return _dyld_get_image_name(i);
return NULL;
}
STATIC void GC_dyld_image_add(const struct GC_MACH_HEADER*hdr,
intptr_t slide)
{
unsigned long start,end;
unsigned i,j;
const struct GC_MACH_SECTION*sec;
const char*name;
GC_has_static_roots_func callback=GC_has_static_roots;
DCL_LOCK_STATE;
if (GC_no_dls)return;
#ifdef DARWIN_DEBUG
name=GC_dyld_name_for_hdr(hdr);
#else
name=callback!=0?GC_dyld_name_for_hdr(hdr):NULL;
#endif
for (i=0;i < sizeof(GC_dyld_sections)/sizeof(GC_dyld_sections[0]);i++){
sec=GC_GETSECTBYNAME(hdr,GC_dyld_sections[i].seg,
GC_dyld_sections[i].sect);
if (sec==NULL||sec->size < sizeof(word))
continue;
start=slide+sec->addr;
end=start+sec->size;
LOCK();
if (callback==0||callback(name,(void*)start,(size_t)sec->size)){
#ifdef DARWIN_DEBUG
GC_log_printf(
"Adding section __DATA,%s at %p-%p (%lu bytes)from image %s\n",
GC_dyld_sections[i].sect,(void*)start,(void*)end,
(unsigned long)sec->size,name);
#endif
GC_add_roots_inner((ptr_t)start,(ptr_t)end,FALSE);
}
UNLOCK();
}
for (j=0;j < sizeof(GC_dyld_add_sect_fmts)/sizeof(char*);j++){
const char*fmt=GC_dyld_add_sect_fmts[j];
for (i=0;i<=L2_MAX_OFILE_ALIGNMENT;i++){
char secnam[16];
(void)snprintf(secnam,sizeof(secnam),fmt,(unsigned)i);
secnam[sizeof(secnam)- 1]='\0';
sec=GC_GETSECTBYNAME(hdr,SEG_DATA,secnam);
if (sec==NULL||sec->size==0)
continue;
start=slide+sec->addr;
end=start+sec->size;
#ifdef DARWIN_DEBUG
GC_log_printf("Adding on-demand section __DATA,%s at"
" %p-%p (%lu bytes)from image %s\n",
secnam,(void*)start,(void*)end,
(unsigned long)sec->size,name);
#endif
GC_add_roots((char*)start,(char*)end);
}
}
#if defined(DARWIN_DEBUG)&&!defined(NO_DEBUGGING)
LOCK();
GC_print_static_roots();
UNLOCK();
#endif
}
STATIC void GC_dyld_image_remove(const struct GC_MACH_HEADER*hdr,
intptr_t slide)
{
unsigned long start,end;
unsigned i,j;
const struct GC_MACH_SECTION*sec;
#if defined(DARWIN_DEBUG)&&!defined(NO_DEBUGGING)
DCL_LOCK_STATE;
#endif
for (i=0;i < sizeof(GC_dyld_sections)/sizeof(GC_dyld_sections[0]);i++){
sec=GC_GETSECTBYNAME(hdr,GC_dyld_sections[i].seg,
GC_dyld_sections[i].sect);
if (sec==NULL||sec->size==0)
continue;
start=slide+sec->addr;
end=start+sec->size;
#ifdef DARWIN_DEBUG
GC_log_printf(
"Removing section __DATA,%s at %p-%p (%lu bytes)from image %s\n",
GC_dyld_sections[i].sect,(void*)start,(void*)end,
(unsigned long)sec->size,GC_dyld_name_for_hdr(hdr));
#endif
GC_remove_roots((char*)start,(char*)end);
}
for (j=0;j < sizeof(GC_dyld_add_sect_fmts)/sizeof(char*);j++){
const char*fmt=GC_dyld_add_sect_fmts[j];
for (i=0;i<=L2_MAX_OFILE_ALIGNMENT;i++){
char secnam[16];
(void)snprintf(secnam,sizeof(secnam),fmt,(unsigned)i);
secnam[sizeof(secnam)- 1]='\0';
sec=GC_GETSECTBYNAME(hdr,SEG_DATA,secnam);
if (sec==NULL||sec->size==0)
continue;
start=slide+sec->addr;
end=start+sec->size;
#ifdef DARWIN_DEBUG
GC_log_printf("Removing on-demand section __DATA,%s at"
" %p-%p (%lu bytes)from image %s\n",secnam,
(void*)start,(void*)end,(unsigned long)sec->size,
GC_dyld_name_for_hdr(hdr));
#endif
GC_remove_roots((char*)start,(char*)end);
}
}
#if defined(DARWIN_DEBUG)&&!defined(NO_DEBUGGING)
LOCK();
GC_print_static_roots();
UNLOCK();
#endif
}
GC_INNER void GC_register_dynamic_libraries(void)
{
}
GC_INNER void GC_init_dyld(void)
{
static GC_bool initialized=FALSE;
if (initialized)return;
#ifdef DARWIN_DEBUG
GC_log_printf("Registering dyld callbacks...\n");
#endif
_dyld_register_func_for_add_image(
(void (*)(const struct mach_header*,intptr_t))GC_dyld_image_add);
_dyld_register_func_for_remove_image(
(void (*)(const struct mach_header*,intptr_t))GC_dyld_image_remove);
initialized=TRUE;
#ifdef NO_DYLD_BIND_FULLY_IMAGE
#else
if (GC_no_dls)return;
if (GETENV("DYLD_BIND_AT_LAUNCH")==0){
#ifdef DARWIN_DEBUG
GC_log_printf("Forcing full bind of GC code...\n");
#endif
if (!_dyld_bind_fully_image_containing_address(
(unsigned long*)GC_malloc))
ABORT("_dyld_bind_fully_image_containing_address failed");
}
#endif
}
#define HAVE_REGISTER_MAIN_STATIC_DATA
GC_INNER GC_bool GC_register_main_static_data(void)
{
return FALSE;
}
#endif
#if defined(HAIKU)
#include <kernel/image.h>
GC_INNER void GC_register_dynamic_libraries(void)
{
image_info info;
int32 cookie=0;
while (get_next_image_info(0,&cookie,&info)==B_OK){
ptr_t data=(ptr_t)info.data;
GC_add_roots_inner(data,data+info.data_size,TRUE);
}
}
#endif
#elif defined(PCR)
GC_INNER void GC_register_dynamic_libraries(void)
{
PCR_IL_LoadedFile*p=PCR_IL_GetLastLoadedFile();
PCR_IL_LoadedSegment*q;
while (p!=NIL&&!(p->lf_commitPoint)){
p=p->lf_prev;
}
for (;p!=NIL;p=p->lf_prev){
for (q=p->lf_ls;q!=NIL;q=q->ls_next){
if ((q->ls_flags&PCR_IL_SegFlags_Traced_MASK)
==PCR_IL_SegFlags_Traced_on){
GC_add_roots_inner((ptr_t)q->ls_addr,
(ptr_t)q->ls_addr+q->ls_bytes,TRUE);
}
}
}
}
#endif
#if!defined(HAVE_REGISTER_MAIN_STATIC_DATA)&&defined(DYNAMIC_LOADING)
GC_INNER GC_bool GC_register_main_static_data(void)
{
return TRUE;
}
#endif
GC_API void GC_CALL GC_register_has_static_roots_callback(
GC_has_static_roots_func callback)
{
GC_has_static_roots=callback;
}
#if defined(GC_PTHREADS)&&!defined(GC_NO_DLOPEN)
#undef GC_MUST_RESTORE_REDEFINED_DLOPEN
#if defined(dlopen)&&!defined(GC_USE_LD_WRAP)
#undef dlopen
#define GC_MUST_RESTORE_REDEFINED_DLOPEN
#endif
#ifndef USE_PROC_FOR_LIBRARIES
static void disable_gc_for_dlopen(void)
{
DCL_LOCK_STATE;
LOCK();
while (GC_incremental&&GC_collection_in_progress()){
ENTER_GC();
GC_collect_a_little_inner(1000);
EXIT_GC();
}
++GC_dont_gc;
UNLOCK();
}
#endif
#ifdef GC_USE_LD_WRAP
#define WRAP_DLFUNC(f)__wrap_##f
#define REAL_DLFUNC(f)__real_##f
void*REAL_DLFUNC(dlopen)(const char*,int);
#else
#define WRAP_DLFUNC(f)GC_##f
#define REAL_DLFUNC(f)f
#endif
GC_API void*WRAP_DLFUNC(dlopen)(const char*path,int mode)
{
void*result;
#ifndef USE_PROC_FOR_LIBRARIES
disable_gc_for_dlopen();
#endif
result=REAL_DLFUNC(dlopen)(path,mode);
#ifndef USE_PROC_FOR_LIBRARIES
GC_enable();
#endif
return(result);
}
#ifdef GC_USE_LD_WRAP
GC_API void*GC_dlopen(const char*path,int mode)
{
return dlopen(path,mode);
}
#endif
#ifdef GC_MUST_RESTORE_REDEFINED_DLOPEN
#define dlopen GC_dlopen
#endif
#endif
#if!defined(SN_TARGET_ORBIS)&&!defined(SN_TARGET_PSP2)
#include <stdio.h>
#ifdef AMIGA
#ifndef __GNUC__
#include <dos.h>
#else
#include <machine/reg.h>
#endif
#endif
#if defined(MACOS)&&defined(__MWERKS__)
#if defined(POWERPC)
#define NONVOLATILE_GPR_COUNT 19
struct ppc_registers {
unsigned long gprs[NONVOLATILE_GPR_COUNT];
};
typedef struct ppc_registers ppc_registers;
#if defined(CPPCHECK)
void getRegisters(ppc_registers*regs);
#else
asm static void getRegisters(register ppc_registers*regs)
{
stmw r13,regs->gprs
blr
}
#endif
static void PushMacRegisters(void)
{
ppc_registers regs;
int i;
getRegisters(&regs);
for (i=0;i < NONVOLATILE_GPR_COUNT;i++)
GC_push_one(regs.gprs[i]);
}
#else
asm static void PushMacRegisters(void)
{
sub.w #4,sp
move.l a2,(sp)
jsr GC_push_one
move.l a3,(sp)
jsr GC_push_one
move.l a4,(sp)
jsr GC_push_one
#if!__option(a6frames)
move.l a6,(sp)
jsr GC_push_one
#endif
move.l d2,(sp)
jsr GC_push_one
move.l d3,(sp)
jsr GC_push_one
move.l d4,(sp)
jsr GC_push_one
move.l d5,(sp)
jsr GC_push_one
move.l d6,(sp)
jsr GC_push_one
move.l d7,(sp)
jsr GC_push_one
add.w #4,sp
rts
}
#endif
#endif
#if defined(SPARC)||defined(IA64)
GC_INNER ptr_t GC_save_regs_ret_val=NULL;
#endif
#undef HAVE_PUSH_REGS
#if defined(USE_ASM_PUSH_REGS)
#define HAVE_PUSH_REGS
#else
#ifdef STACK_NOT_SCANNED
void GC_push_regs(void)
{
}
#define HAVE_PUSH_REGS
#elif defined(M68K)&&defined(AMIGA)
void GC_push_regs(void)
{
#ifdef __GNUC__
asm("subq.w&0x4,%sp");
asm("mov.l %a2,(%sp)");asm("jsr _GC_push_one");
asm("mov.l %a3,(%sp)");asm("jsr _GC_push_one");
asm("mov.l %a4,(%sp)");asm("jsr _GC_push_one");
asm("mov.l %a5,(%sp)");asm("jsr _GC_push_one");
asm("mov.l %a6,(%sp)");asm("jsr _GC_push_one");
asm("mov.l %d2,(%sp)");asm("jsr _GC_push_one");
asm("mov.l %d3,(%sp)");asm("jsr _GC_push_one");
asm("mov.l %d4,(%sp)");asm("jsr _GC_push_one");
asm("mov.l %d5,(%sp)");asm("jsr _GC_push_one");
asm("mov.l %d6,(%sp)");asm("jsr _GC_push_one");
asm("mov.l %d7,(%sp)");asm("jsr _GC_push_one");
asm("addq.w&0x4,%sp");
#else
GC_push_one(getreg(REG_A2));
GC_push_one(getreg(REG_A3));
#ifndef __SASC
GC_push_one(getreg(REG_A4));
#endif
GC_push_one(getreg(REG_A5));
GC_push_one(getreg(REG_A6));
GC_push_one(getreg(REG_D2));
GC_push_one(getreg(REG_D3));
GC_push_one(getreg(REG_D4));
GC_push_one(getreg(REG_D5));
GC_push_one(getreg(REG_D6));
GC_push_one(getreg(REG_D7));
#endif
}
#define HAVE_PUSH_REGS
#elif defined(MACOS)
#if defined(M68K)&&defined(THINK_C)&&!defined(CPPCHECK)
#define PushMacReg(reg)move.l reg,(sp)jsr GC_push_one
void GC_push_regs(void)
{
asm {
sub.w #4,sp;reserve space for one parameter.
PushMacReg(a2);
PushMacReg(a3);
PushMacReg(a4);
;skip a5 (globals),a6 (frame pointer),and a7 (stack pointer)
PushMacReg(d2);
PushMacReg(d3);
PushMacReg(d4);
PushMacReg(d5);
PushMacReg(d6);
PushMacReg(d7);
add.w #4,sp;fix stack.
}
}
#define HAVE_PUSH_REGS
#undef PushMacReg
#elif defined(__MWERKS__)
void GC_push_regs(void)
{
PushMacRegisters();
}
#define HAVE_PUSH_REGS
#endif
#endif
#endif
#if defined(HAVE_PUSH_REGS)&&defined(THREADS)
#error GC_push_regs cannot be used with threads
#undef HAVE_PUSH_REGS
#endif
#if!defined(HAVE_PUSH_REGS)&&defined(UNIX_LIKE)
#include <signal.h>
#ifndef NO_GETCONTEXT
#if defined(DARWIN)&&(MAC_OS_X_VERSION_MAX_ALLOWED>=1060)
#include <sys/ucontext.h>
#else
#include <ucontext.h>
#endif
#ifdef GETCONTEXT_FPU_EXCMASK_BUG
#include <fenv.h>
#endif
#endif
#endif
GC_ATTR_NO_SANITIZE_ADDR
GC_INNER void GC_with_callee_saves_pushed(void (*fn)(ptr_t,void*),
volatile ptr_t arg)
{
volatile int dummy;
volatile ptr_t context=0;
#if defined(HAVE_PUSH_REGS)
GC_push_regs();
#elif defined(__EMSCRIPTEN__)
#else
#if defined(UNIX_LIKE)&&!defined(NO_GETCONTEXT)
static signed char getcontext_works=0;
ucontext_t ctxt;
#ifdef GETCONTEXT_FPU_EXCMASK_BUG
#ifdef X86_64
unsigned short old_fcw;
#if defined(CPPCHECK)
GC_noop1((word)&old_fcw);
#endif
__asm__ __volatile__ ("fstcw %0":"=m" (*&old_fcw));
#else
int except_mask=fegetexcept();
#endif
#endif
if (getcontext_works>=0){
if (getcontext(&ctxt)< 0){
WARN("getcontext failed:"
" using another register retrieval method...\n",0);
} else {
context=(ptr_t)&ctxt;
}
if (EXPECT(0==getcontext_works,FALSE))
getcontext_works=context!=NULL?1:-1;
}
#ifdef GETCONTEXT_FPU_EXCMASK_BUG
#ifdef X86_64
__asm__ __volatile__ ("fldcw %0"::"m" (*&old_fcw));
{
unsigned mxcsr;
__asm__ __volatile__ ("stmxcsr %0":"=m" (*&mxcsr));
mxcsr=(mxcsr&~(FE_ALL_EXCEPT<<7))|
((old_fcw&FE_ALL_EXCEPT)<<7);
__asm__ __volatile__ ("ldmxcsr %0"::"m" (*&mxcsr));
}
#else
if (feenableexcept(except_mask)< 0)
ABORT("feenableexcept failed");
#endif
#endif
#if defined(SPARC)||defined(IA64)
GC_save_regs_ret_val=GC_save_regs_in_stack();
#endif
if (NULL==context)
#endif
{
#if defined(HAVE_BUILTIN_UNWIND_INIT)
__builtin_unwind_init();
#elif defined(NO_CRT)&&defined(MSWIN32)
CONTEXT ctx;
RtlCaptureContext(&ctx);
#else
jmp_buf regs;
word*i=(word*)&regs;
ptr_t lim=(ptr_t)(&regs)+sizeof(regs);
for (;(word)i < (word)lim;i++){
*i=0;
}
#if defined(MSWIN32)||defined(MSWINCE)||defined(UTS4)||defined(OS2)||defined(CX_UX)||defined(__CC_ARM)||defined(LINUX)||defined(EWS4800)||defined(RTEMS)
(void)setjmp(regs);
#else
(void)_setjmp(regs);
#endif
#endif
}
#endif
fn(arg,( void*)context);
GC_noop1(COVERT_DATAFLOW(&dummy));
}
#endif
#if defined(GC_PTHREADS)&&!defined(GC_WIN32_THREADS)&&!defined(GC_DARWIN_THREADS)&&!defined(SN_TARGET_ORBIS)&&!defined(SN_TARGET_PSP2)
#ifdef NACL
#include <unistd.h>
#include <sys/time.h>
STATIC int GC_nacl_num_gc_threads=0;
STATIC __thread int GC_nacl_thread_idx=-1;
STATIC volatile int GC_nacl_park_threads_now=0;
STATIC volatile pthread_t GC_nacl_thread_parker=-1;
GC_INNER __thread GC_thread GC_nacl_gc_thread_self=NULL;
volatile int GC_nacl_thread_parked[MAX_NACL_GC_THREADS];
int GC_nacl_thread_used[MAX_NACL_GC_THREADS];
#elif defined(GC_OPENBSD_UTHREADS)
#include <pthread_np.h>
#else
#include <signal.h>
#include <semaphore.h>
#include <errno.h>
#include <time.h>
#include <unistd.h>
#if (!defined(AO_HAVE_load_acquire)||!defined(AO_HAVE_store_release))&&!defined(CPPCHECK)
#error AO_load_acquire and/or AO_store_release are missing;
#error please define AO_REQUIRE_CAS manually
#endif
#undef pthread_sigmask
#ifdef GC_ENABLE_SUSPEND_THREAD
static void*GC_CALLBACK suspend_self_inner(void*client_data);
#endif
#ifdef DEBUG_THREADS
#ifndef NSIG
#if defined(MAXSIG)
#define NSIG (MAXSIG+1)
#elif defined(_NSIG)
#define NSIG _NSIG
#elif defined(__SIGRTMAX)
#define NSIG (__SIGRTMAX+1)
#else
#error define NSIG
#endif
#endif
void GC_print_sig_mask(void)
{
sigset_t blocked;
int i;
if (pthread_sigmask(SIG_BLOCK,NULL,&blocked)!=0)
ABORT("pthread_sigmask failed");
for (i=1;i < NSIG;i++){
if (sigismember(&blocked,i))
GC_printf("Signal blocked:%d\n",i);
}
}
#endif
STATIC void GC_remove_allowed_signals(sigset_t*set)
{
if (sigdelset(set,SIGINT)!=0
||sigdelset(set,SIGQUIT)!=0
||sigdelset(set,SIGABRT)!=0
||sigdelset(set,SIGTERM)!=0){
ABORT("sigdelset failed");
}
#ifdef MPROTECT_VDB
if (sigdelset(set,SIGSEGV)!=0
#ifdef HAVE_SIGBUS
||sigdelset(set,SIGBUS)!=0
#endif
){
ABORT("sigdelset failed");
}
#endif
}
static sigset_t suspend_handler_mask;
#define THREAD_RESTARTED 0x1
STATIC volatile AO_t GC_stop_count=0;
STATIC volatile AO_t GC_world_is_stopped=FALSE;
#if defined(GC_OSF1_THREADS)||defined(THREAD_SANITIZER)||defined(ADDRESS_SANITIZER)||defined(MEMORY_SANITIZER)
STATIC GC_bool GC_retry_signals=TRUE;
#else
STATIC GC_bool GC_retry_signals=FALSE;
#endif
#ifndef SIG_THR_RESTART
#if defined(GC_HPUX_THREADS)||defined(GC_OSF1_THREADS)||defined(GC_NETBSD_THREADS)||defined(GC_USESIGRT_SIGNALS)
#if defined(_SIGRTMIN)&&!defined(CPPCHECK)
#define SIG_THR_RESTART _SIGRTMIN+5
#else
#define SIG_THR_RESTART SIGRTMIN+5
#endif
#else
#define SIG_THR_RESTART SIGXCPU
#endif
#endif
#define SIGNAL_UNSET (-1)
STATIC int GC_sig_suspend=SIGNAL_UNSET;
STATIC int GC_sig_thr_restart=SIGNAL_UNSET;
GC_API void GC_CALL GC_set_suspend_signal(int sig)
{
if (GC_is_initialized)return;
GC_sig_suspend=sig;
}
GC_API void GC_CALL GC_set_thr_restart_signal(int sig)
{
if (GC_is_initialized)return;
GC_sig_thr_restart=sig;
}
GC_API int GC_CALL GC_get_suspend_signal(void)
{
return GC_sig_suspend!=SIGNAL_UNSET?GC_sig_suspend:SIG_SUSPEND;
}
GC_API int GC_CALL GC_get_thr_restart_signal(void)
{
return GC_sig_thr_restart!=SIGNAL_UNSET
?GC_sig_thr_restart:SIG_THR_RESTART;
}
#if defined(GC_EXPLICIT_SIGNALS_UNBLOCK)||!defined(NO_SIGNALS_UNBLOCK_IN_MAIN)
GC_INNER void GC_unblock_gc_signals(void)
{
sigset_t set;
sigemptyset(&set);
GC_ASSERT(GC_sig_suspend!=SIGNAL_UNSET);
GC_ASSERT(GC_sig_thr_restart!=SIGNAL_UNSET);
sigaddset(&set,GC_sig_suspend);
sigaddset(&set,GC_sig_thr_restart);
if (pthread_sigmask(SIG_UNBLOCK,&set,NULL)!=0)
ABORT("pthread_sigmask failed");
}
#endif
STATIC sem_t GC_suspend_ack_sem;
STATIC void GC_suspend_handler_inner(ptr_t dummy,void*context);
#ifndef NO_SA_SIGACTION
STATIC void GC_suspend_handler(int sig,siginfo_t*info GC_ATTR_UNUSED,
void*context GC_ATTR_UNUSED)
#else
STATIC void GC_suspend_handler(int sig)
#endif
{
int old_errno=errno;
if (sig!=GC_sig_suspend){
#if defined(GC_FREEBSD_THREADS)
if (0==sig)return;
#endif
ABORT("Bad signal in suspend_handler");
}
#if defined(IA64)||defined(HP_PA)||defined(M68K)
GC_with_callee_saves_pushed(GC_suspend_handler_inner,NULL);
#else
{
#ifdef NO_SA_SIGACTION
void*context=0;
#endif
GC_suspend_handler_inner(NULL,context);
}
#endif
errno=old_errno;
}
#ifdef BASE_ATOMIC_OPS_EMULATED
#define ao_load_acquire_async(p)(*(p))
#define ao_load_async(p)ao_load_acquire_async(p)
#define ao_store_release_async(p,v)(void)(*(p)=(v))
#define ao_store_async(p,v)ao_store_release_async(p,v)
#else
#define ao_load_acquire_async(p)AO_load_acquire(p)
#define ao_load_async(p)AO_load(p)
#define ao_store_release_async(p,v)AO_store_release(p,v)
#define ao_store_async(p,v)AO_store(p,v)
#endif
#ifdef THREAD_SANITIZER
GC_ATTR_NO_SANITIZE_THREAD
static GC_thread GC_lookup_thread_async(pthread_t id)
{
GC_thread p=GC_threads[THREAD_TABLE_INDEX(id)];
while (p!=NULL&&!THREAD_EQUAL(p->id,id))
p=p->next;
return p;
}
#else
#define GC_lookup_thread_async GC_lookup_thread
#endif
GC_INLINE void GC_store_stack_ptr(GC_thread me)
{
#ifdef SPARC
ao_store_async((volatile AO_t*)&me->stop_info.stack_ptr,
(AO_t)GC_save_regs_in_stack());
#else
#ifdef IA64
me->backing_store_ptr=GC_save_regs_in_stack();
#endif
ao_store_async((volatile AO_t*)&me->stop_info.stack_ptr,
(AO_t)GC_approx_sp());
#endif
}
STATIC void GC_suspend_handler_inner(ptr_t dummy GC_ATTR_UNUSED,
void*context GC_ATTR_UNUSED)
{
pthread_t self=pthread_self();
GC_thread me;
IF_CANCEL(int cancel_state;)
AO_t my_stop_count=ao_load_acquire_async(&GC_stop_count);
DISABLE_CANCEL(cancel_state);
#ifdef DEBUG_THREADS
GC_log_printf("Suspending %p\n",(void*)self);
#endif
GC_ASSERT(((word)my_stop_count&THREAD_RESTARTED)==0);
me=GC_lookup_thread_async(self);
#ifdef GC_ENABLE_SUSPEND_THREAD
if (ao_load_async(&me->suspended_ext)){
GC_store_stack_ptr(me);
sem_post(&GC_suspend_ack_sem);
suspend_self_inner(me);
#ifdef DEBUG_THREADS
GC_log_printf("Continuing %p on GC_resume_thread\n",(void*)self);
#endif
RESTORE_CANCEL(cancel_state);
return;
}
#endif
if (((word)me->stop_info.last_stop_count&~(word)THREAD_RESTARTED)
==(word)my_stop_count){
if (!GC_retry_signals){
WARN("Duplicate suspend signal in thread %p\n",self);
}
RESTORE_CANCEL(cancel_state);
return;
}
GC_store_stack_ptr(me);
#ifdef THREAD_SANITIZER
{
sigset_t set;
sigemptyset(&set);
GC_ASSERT(GC_sig_suspend!=SIGNAL_UNSET);
GC_ASSERT(GC_sig_thr_restart!=SIGNAL_UNSET);
sigaddset(&set,GC_sig_suspend);
sigaddset(&set,GC_sig_thr_restart);
if (pthread_sigmask(SIG_UNBLOCK,&set,NULL)!=0)
ABORT("pthread_sigmask failed in suspend handler");
}
#endif
sem_post(&GC_suspend_ack_sem);
ao_store_release_async(&me->stop_info.last_stop_count,my_stop_count);
do {
sigsuspend (&suspend_handler_mask);
} while (ao_load_acquire_async(&GC_world_is_stopped)
&&ao_load_async(&GC_stop_count)==my_stop_count);
#ifdef DEBUG_THREADS
GC_log_printf("Continuing %p\n",(void*)self);
#endif
#ifndef GC_NETBSD_THREADS_WORKAROUND
if (GC_retry_signals)
#endif
{
sem_post(&GC_suspend_ack_sem);
#ifdef GC_NETBSD_THREADS_WORKAROUND
if (GC_retry_signals)
#endif
{
ao_store_release_async(&me->stop_info.last_stop_count,
(AO_t)((word)my_stop_count|THREAD_RESTARTED));
}
}
RESTORE_CANCEL(cancel_state);
}
static void suspend_restart_barrier(int n_live_threads)
{
int i;
for (i=0;i < n_live_threads;i++){
while (0!=sem_wait(&GC_suspend_ack_sem)){
if (errno!=EINTR)
ABORT("sem_wait failed");
}
}
#ifdef GC_ASSERTIONS
sem_getvalue(&GC_suspend_ack_sem,&i);
GC_ASSERT(0==i);
#endif
}
static int resend_lost_signals(int n_live_threads,
int (*suspend_restart_all)(void))
{
#define WAIT_UNIT 3000
#define RETRY_INTERVAL 100000
if (n_live_threads > 0){
unsigned long wait_usecs=0;
for (;;){
int ack_count;
sem_getvalue(&GC_suspend_ack_sem,&ack_count);
if (ack_count==n_live_threads)
break;
if (wait_usecs > RETRY_INTERVAL){
int newly_sent=suspend_restart_all();
GC_COND_LOG_PRINTF("Resent %d signals after timeout\n",newly_sent);
sem_getvalue(&GC_suspend_ack_sem,&ack_count);
if (newly_sent < n_live_threads - ack_count){
WARN("Lost some threads while stopping or starting world?!\n",0);
n_live_threads=ack_count+newly_sent;
}
wait_usecs=0;
}
#ifdef LINT2
#undef WAIT_UNIT
#define WAIT_UNIT 1
sched_yield();
#elif defined(CPPCHECK)
{
struct timespec ts;
ts.tv_sec=0;
ts.tv_nsec=WAIT_UNIT*1000;
(void)nanosleep(&ts,NULL);
}
#else
usleep(WAIT_UNIT);
#endif
wait_usecs+=WAIT_UNIT;
}
}
return n_live_threads;
}
STATIC void GC_restart_handler(int sig)
{
#if defined(DEBUG_THREADS)
int old_errno=errno;
#endif
if (sig!=GC_sig_thr_restart)
ABORT("Bad signal in restart handler");
#ifdef DEBUG_THREADS
GC_log_printf("In GC_restart_handler for %p\n",(void*)pthread_self());
errno=old_errno;
#endif
}
#ifdef USE_TKILL_ON_ANDROID
EXTERN_C_BEGIN
extern int tkill(pid_t tid,int sig);
EXTERN_C_END
static int android_thread_kill(pid_t tid,int sig)
{
int ret;
int old_errno=errno;
ret=tkill(tid,sig);
if (ret < 0){
ret=errno;
errno=old_errno;
}
return ret;
}
#define THREAD_SYSTEM_ID(t)(t)->kernel_id
#define RAISE_SIGNAL(t,sig)android_thread_kill(THREAD_SYSTEM_ID(t),sig)
#else
#define THREAD_SYSTEM_ID(t)(t)->id
#define RAISE_SIGNAL(t,sig)pthread_kill(THREAD_SYSTEM_ID(t),sig)
#endif
#ifdef GC_ENABLE_SUSPEND_THREAD
#include <sys/time.h>
STATIC void GC_brief_async_signal_safe_sleep(void)
{
struct timeval tv;
tv.tv_sec=0;
#if defined(GC_TIME_LIMIT)&&!defined(CPPCHECK)
tv.tv_usec=1000*GC_TIME_LIMIT/2;
#else
tv.tv_usec=1000*50/2;
#endif
(void)select(0,0,0,0,&tv);
}
static void*GC_CALLBACK suspend_self_inner(void*client_data){
GC_thread me=(GC_thread)client_data;
while (ao_load_acquire_async(&me->suspended_ext)){
GC_brief_async_signal_safe_sleep();
}
return NULL;
}
GC_API void GC_CALL GC_suspend_thread(GC_SUSPEND_THREAD_ID thread){
GC_thread t;
IF_CANCEL(int cancel_state;)
DCL_LOCK_STATE;
LOCK();
t=GC_lookup_thread((pthread_t)thread);
if (t==NULL||t->suspended_ext){
UNLOCK();
return;
}
AO_store_release(&t->suspended_ext,TRUE);
if (THREAD_EQUAL((pthread_t)thread,pthread_self())){
UNLOCK();
(void)GC_do_blocking(suspend_self_inner,t);
return;
}
if ((t->flags&FINISHED)!=0){
UNLOCK();
return;
}
DISABLE_CANCEL(cancel_state);
#ifdef PARALLEL_MARK
if (GC_parallel)
GC_wait_for_reclaim();
#endif
if (GC_manual_vdb){
GC_acquire_dirty_lock();
}
switch (RAISE_SIGNAL(t,GC_sig_suspend)){
case 0:
break;
default:
ABORT("pthread_kill failed");
}
GC_ASSERT(GC_thr_initialized);
while (sem_wait(&GC_suspend_ack_sem)!=0){
if (errno!=EINTR)
ABORT("sem_wait for handler failed (suspend_self)");
}
if (GC_manual_vdb)
GC_release_dirty_lock();
RESTORE_CANCEL(cancel_state);
UNLOCK();
}
GC_API void GC_CALL GC_resume_thread(GC_SUSPEND_THREAD_ID thread){
GC_thread t;
DCL_LOCK_STATE;
LOCK();
t=GC_lookup_thread((pthread_t)thread);
if (t!=NULL)
AO_store(&t->suspended_ext,FALSE);
UNLOCK();
}
GC_API int GC_CALL GC_is_thread_suspended(GC_SUSPEND_THREAD_ID thread){
GC_thread t;
int is_suspended=0;
DCL_LOCK_STATE;
LOCK();
t=GC_lookup_thread((pthread_t)thread);
if (t!=NULL&&t->suspended_ext)
is_suspended=(int)TRUE;
UNLOCK();
return is_suspended;
}
#endif
#undef ao_load_acquire_async
#undef ao_load_async
#undef ao_store_async
#undef ao_store_release_async
#endif
#ifdef IA64
#define IF_IA64(x)x
#else
#define IF_IA64(x)
#endif
GC_INNER void GC_push_all_stacks(void)
{
GC_bool found_me=FALSE;
size_t nthreads=0;
int i;
GC_thread p;
ptr_t lo,hi;
IF_IA64(ptr_t bs_lo;ptr_t bs_hi;)
struct GC_traced_stack_sect_s*traced_stack_sect;
pthread_t self=pthread_self();
word total_size=0;
if (!EXPECT(GC_thr_initialized,TRUE))
GC_thr_init();
#ifdef DEBUG_THREADS
GC_log_printf("Pushing stacks from thread %p\n",(void*)self);
#endif
for (i=0;i < THREAD_TABLE_SZ;i++){
for (p=GC_threads[i];p!=0;p=p->next){
if (p->flags&FINISHED)continue;
++nthreads;
traced_stack_sect=p->traced_stack_sect;
if (THREAD_EQUAL(p->id,self)){
GC_ASSERT(!p->thread_blocked);
#ifdef SPARC
lo=(ptr_t)GC_save_regs_in_stack();
#else
lo=GC_approx_sp();
#endif
found_me=TRUE;
IF_IA64(bs_hi=(ptr_t)GC_save_regs_in_stack();)
} else {
lo=(ptr_t)AO_load((volatile AO_t*)&p->stop_info.stack_ptr);
IF_IA64(bs_hi=p->backing_store_ptr;)
if (traced_stack_sect!=NULL
&&traced_stack_sect->saved_stack_ptr==lo){
traced_stack_sect=traced_stack_sect->prev;
}
}
if ((p->flags&MAIN_THREAD)==0){
hi=p->stack_end;
IF_IA64(bs_lo=p->backing_store_end);
} else {
hi=GC_stackbottom;
IF_IA64(bs_lo=BACKING_STORE_BASE;)
}
#ifdef DEBUG_THREADS
GC_log_printf("Stack for thread %p=[%p,%p)\n",
(void*)p->id,(void*)lo,(void*)hi);
#endif
if (0==lo)ABORT("GC_push_all_stacks:sp not set!");
if (p->altstack!=NULL&&(word)p->altstack<=(word)lo
&&(word)lo<=(word)p->altstack+p->altstack_size){
hi=p->altstack+p->altstack_size;
}
GC_push_all_stack_sections(lo,hi,traced_stack_sect);
#ifdef STACK_GROWS_UP
total_size+=lo - hi;
#else
total_size+=hi - lo;
#endif
#ifdef NACL
GC_push_all_stack((ptr_t)p->stop_info.reg_storage,
(ptr_t)(p->stop_info.reg_storage+NACL_GC_REG_STORAGE_SIZE));
total_size+=NACL_GC_REG_STORAGE_SIZE*sizeof(ptr_t);
#endif
#ifdef IA64
#ifdef DEBUG_THREADS
GC_log_printf("Reg stack for thread %p=[%p,%p)\n",
(void*)p->id,(void*)bs_lo,(void*)bs_hi);
#endif
GC_push_all_register_sections(bs_lo,bs_hi,
THREAD_EQUAL(p->id,self),
traced_stack_sect);
total_size+=bs_hi - bs_lo;
#endif
}
}
GC_VERBOSE_LOG_PRINTF("Pushed %d thread stacks\n",(int)nthreads);
if (!found_me&&!GC_in_thread_creation)
ABORT("Collecting from unknown thread");
GC_total_stacksize=total_size;
}
#ifdef DEBUG_THREADS
pthread_t GC_stopping_thread;
int GC_stopping_pid=0;
#endif
STATIC int GC_suspend_all(void)
{
int n_live_threads=0;
int i;
#ifndef NACL
GC_thread p;
#ifndef GC_OPENBSD_UTHREADS
int result;
#endif
pthread_t self=pthread_self();
for (i=0;i < THREAD_TABLE_SZ;i++){
for (p=GC_threads[i];p!=0;p=p->next){
if (!THREAD_EQUAL(p->id,self)){
if ((p->flags&FINISHED)!=0)continue;
if (p->thread_blocked)continue;
#ifndef GC_OPENBSD_UTHREADS
#ifdef GC_ENABLE_SUSPEND_THREAD
if (p->suspended_ext)continue;
#endif
if (AO_load(&p->stop_info.last_stop_count)==GC_stop_count)
continue;
n_live_threads++;
#endif
#ifdef DEBUG_THREADS
GC_log_printf("Sending suspend signal to %p\n",(void*)p->id);
#endif
#ifdef GC_OPENBSD_UTHREADS
{
stack_t stack;
GC_acquire_dirty_lock();
if (pthread_suspend_np(p->id)!=0)
ABORT("pthread_suspend_np failed");
GC_release_dirty_lock();
if (pthread_stackseg_np(p->id,&stack))
ABORT("pthread_stackseg_np failed");
p->stop_info.stack_ptr=(ptr_t)stack.ss_sp - stack.ss_size;
if (GC_on_thread_event)
GC_on_thread_event(GC_EVENT_THREAD_SUSPENDED,
(void*)p->id);
}
#else
result=RAISE_SIGNAL(p,GC_sig_suspend);
switch(result){
case ESRCH:
n_live_threads--;
break;
case 0:
if (GC_on_thread_event)
GC_on_thread_event(GC_EVENT_THREAD_SUSPENDED,
(void*)(word)THREAD_SYSTEM_ID(p));
break;
default:
ABORT_ARG1("pthread_kill failed at suspend",
":errcode=%d",result);
}
#endif
}
}
}
#else
#ifndef NACL_PARK_WAIT_NANOSECONDS
#define NACL_PARK_WAIT_NANOSECONDS (100*1000)
#endif
#define NANOS_PER_SECOND (1000UL*1000*1000)
unsigned long num_sleeps=0;
#ifdef DEBUG_THREADS
GC_log_printf("pthread_stop_world:num_threads=%d\n",
GC_nacl_num_gc_threads - 1);
#endif
GC_nacl_thread_parker=pthread_self();
GC_nacl_park_threads_now=1;
if (GC_manual_vdb)
GC_acquire_dirty_lock();
while (1){
int num_threads_parked=0;
struct timespec ts;
int num_used=0;
for (i=0;i < MAX_NACL_GC_THREADS
&&num_used < GC_nacl_num_gc_threads;i++){
if (GC_nacl_thread_used[i]==1){
num_used++;
if (GC_nacl_thread_parked[i]==1){
num_threads_parked++;
if (GC_on_thread_event)
GC_on_thread_event(GC_EVENT_THREAD_SUSPENDED,(void*)(word)i);
}
}
}
if (num_threads_parked>=GC_nacl_num_gc_threads - 1)
break;
ts.tv_sec=0;
ts.tv_nsec=NACL_PARK_WAIT_NANOSECONDS;
#ifdef DEBUG_THREADS
GC_log_printf("Sleep waiting for %d threads to park...\n",
GC_nacl_num_gc_threads - num_threads_parked - 1);
#endif
nanosleep(&ts,0);
if (++num_sleeps > NANOS_PER_SECOND/NACL_PARK_WAIT_NANOSECONDS){
WARN("GC appears stalled waiting for %" WARN_PRIdPTR
" threads to park...\n",
GC_nacl_num_gc_threads - num_threads_parked - 1);
num_sleeps=0;
}
}
if (GC_manual_vdb)
GC_release_dirty_lock();
#endif
return n_live_threads;
}
GC_INNER void GC_stop_world(void)
{
#if!defined(GC_OPENBSD_UTHREADS)&&!defined(NACL)
int n_live_threads;
#endif
GC_ASSERT(I_HOLD_LOCK());
#ifdef DEBUG_THREADS
GC_stopping_thread=pthread_self();
GC_stopping_pid=getpid();
GC_log_printf("Stopping the world from %p\n",(void*)GC_stopping_thread);
#endif
#ifdef PARALLEL_MARK
if (GC_parallel){
GC_acquire_mark_lock();
GC_ASSERT(GC_fl_builder_count==0);
}
#endif
#if defined(GC_OPENBSD_UTHREADS)||defined(NACL)
(void)GC_suspend_all();
#else
AO_store(&GC_stop_count,
(AO_t)((word)GC_stop_count+(THREAD_RESTARTED+1)));
if (GC_manual_vdb){
GC_acquire_dirty_lock();
}
AO_store_release(&GC_world_is_stopped,TRUE);
n_live_threads=GC_suspend_all();
if (GC_retry_signals)
n_live_threads=resend_lost_signals(n_live_threads,GC_suspend_all);
suspend_restart_barrier(n_live_threads);
if (GC_manual_vdb)
GC_release_dirty_lock();
#endif
#ifdef PARALLEL_MARK
if (GC_parallel)
GC_release_mark_lock();
#endif
#ifdef DEBUG_THREADS
GC_log_printf("World stopped from %p\n",(void*)pthread_self());
GC_stopping_thread=0;
#endif
}
#ifdef NACL
#if defined(__x86_64__)
#define NACL_STORE_REGS()do { __asm__ __volatile__ ("push %rbx");__asm__ __volatile__ ("push %rbp");__asm__ __volatile__ ("push %r12");__asm__ __volatile__ ("push %r13");__asm__ __volatile__ ("push %r14");__asm__ __volatile__ ("push %r15");__asm__ __volatile__ ("mov %%esp,%0":"=m" (GC_nacl_gc_thread_self->stop_info.stack_ptr));BCOPY(GC_nacl_gc_thread_self->stop_info.stack_ptr,GC_nacl_gc_thread_self->stop_info.reg_storage,NACL_GC_REG_STORAGE_SIZE*sizeof(ptr_t));__asm__ __volatile__ ("naclasp $48,%r15");} while (0)
#elif defined(__i386__)
#define NACL_STORE_REGS()do { __asm__ __volatile__ ("push %ebx");__asm__ __volatile__ ("push %ebp");__asm__ __volatile__ ("push %esi");__asm__ __volatile__ ("push %edi");__asm__ __volatile__ ("mov %%esp,%0":"=m" (GC_nacl_gc_thread_self->stop_info.stack_ptr));BCOPY(GC_nacl_gc_thread_self->stop_info.stack_ptr,GC_nacl_gc_thread_self->stop_info.reg_storage,NACL_GC_REG_STORAGE_SIZE*sizeof(ptr_t));__asm__ __volatile__ ("add $16,%esp");} while (0)
#elif defined(__arm__)
#define NACL_STORE_REGS()do { __asm__ __volatile__ ("push {r4-r8,r10-r12,lr}");__asm__ __volatile__ ("mov r0,%0"::"r" (&GC_nacl_gc_thread_self->stop_info.stack_ptr));__asm__ __volatile__ ("bic r0,r0,#0xc0000000");__asm__ __volatile__ ("str sp,[r0]");BCOPY(GC_nacl_gc_thread_self->stop_info.stack_ptr,GC_nacl_gc_thread_self->stop_info.reg_storage,NACL_GC_REG_STORAGE_SIZE*sizeof(ptr_t));__asm__ __volatile__ ("add sp,sp,#40");__asm__ __volatile__ ("bic sp,sp,#0xc0000000");} while (0)
#else
#error TODO Please port NACL_STORE_REGS
#endif
GC_API_OSCALL void nacl_pre_syscall_hook(void)
{
if (GC_nacl_thread_idx!=-1){
NACL_STORE_REGS();
GC_nacl_gc_thread_self->stop_info.stack_ptr=GC_approx_sp();
GC_nacl_thread_parked[GC_nacl_thread_idx]=1;
}
}
GC_API_OSCALL void __nacl_suspend_thread_if_needed(void)
{
if (GC_nacl_park_threads_now){
pthread_t self=pthread_self();
if (GC_nacl_thread_parker==self)
return;
if (GC_nacl_thread_idx < 0)
return;
if (!GC_nacl_thread_parked[GC_nacl_thread_idx]){
NACL_STORE_REGS();
GC_nacl_gc_thread_self->stop_info.stack_ptr=GC_approx_sp();
}
GC_nacl_thread_parked[GC_nacl_thread_idx]=1;
while (GC_nacl_park_threads_now){
}
GC_nacl_thread_parked[GC_nacl_thread_idx]=0;
BZERO(GC_nacl_gc_thread_self->stop_info.reg_storage,
NACL_GC_REG_STORAGE_SIZE*sizeof(ptr_t));
}
}
GC_API_OSCALL void nacl_post_syscall_hook(void)
{
__nacl_suspend_thread_if_needed();
if (GC_nacl_thread_idx!=-1){
GC_nacl_thread_parked[GC_nacl_thread_idx]=0;
}
}
STATIC GC_bool GC_nacl_thread_parking_inited=FALSE;
STATIC pthread_mutex_t GC_nacl_thread_alloc_lock=PTHREAD_MUTEX_INITIALIZER;
struct nacl_irt_blockhook {
int (*register_block_hooks)(void (*pre)(void),void (*post)(void));
};
EXTERN_C_BEGIN
extern size_t nacl_interface_query(const char*interface_ident,
void*table,size_t tablesize);
EXTERN_C_END
GC_INNER void GC_nacl_initialize_gc_thread(void)
{
int i;
static struct nacl_irt_blockhook gc_hook;
pthread_mutex_lock(&GC_nacl_thread_alloc_lock);
if (!EXPECT(GC_nacl_thread_parking_inited,TRUE)){
BZERO(GC_nacl_thread_parked,sizeof(GC_nacl_thread_parked));
BZERO(GC_nacl_thread_used,sizeof(GC_nacl_thread_used));
nacl_interface_query("nacl-irt-blockhook-0.1",
&gc_hook,sizeof(gc_hook));
gc_hook.register_block_hooks(nacl_pre_syscall_hook,
nacl_post_syscall_hook);
GC_nacl_thread_parking_inited=TRUE;
}
GC_ASSERT(GC_nacl_num_gc_threads<=MAX_NACL_GC_THREADS);
for (i=0;i < MAX_NACL_GC_THREADS;i++){
if (GC_nacl_thread_used[i]==0){
GC_nacl_thread_used[i]=1;
GC_nacl_thread_idx=i;
GC_nacl_num_gc_threads++;
break;
}
}
pthread_mutex_unlock(&GC_nacl_thread_alloc_lock);
}
GC_INNER void GC_nacl_shutdown_gc_thread(void)
{
pthread_mutex_lock(&GC_nacl_thread_alloc_lock);
GC_ASSERT(GC_nacl_thread_idx>=0);
GC_ASSERT(GC_nacl_thread_idx < MAX_NACL_GC_THREADS);
GC_ASSERT(GC_nacl_thread_used[GC_nacl_thread_idx]!=0);
GC_nacl_thread_used[GC_nacl_thread_idx]=0;
GC_nacl_thread_idx=-1;
GC_nacl_num_gc_threads--;
pthread_mutex_unlock(&GC_nacl_thread_alloc_lock);
}
#else
STATIC int GC_restart_all(void)
{
int n_live_threads=0;
int i;
pthread_t self=pthread_self();
GC_thread p;
#ifndef GC_OPENBSD_UTHREADS
int result;
#endif
for (i=0;i < THREAD_TABLE_SZ;i++){
for (p=GC_threads[i];p!=NULL;p=p->next){
if (!THREAD_EQUAL(p->id,self)){
if ((p->flags&FINISHED)!=0)continue;
if (p->thread_blocked)continue;
#ifndef GC_OPENBSD_UTHREADS
#ifdef GC_ENABLE_SUSPEND_THREAD
if (p->suspended_ext)continue;
#endif
if (GC_retry_signals
&&AO_load(&p->stop_info.last_stop_count)
==(AO_t)((word)GC_stop_count|THREAD_RESTARTED))
continue;
n_live_threads++;
#endif
#ifdef DEBUG_THREADS
GC_log_printf("Sending restart signal to %p\n",(void*)p->id);
#endif
#ifdef GC_OPENBSD_UTHREADS
if (pthread_resume_np(p->id)!=0)
ABORT("pthread_resume_np failed");
if (GC_on_thread_event)
GC_on_thread_event(GC_EVENT_THREAD_UNSUSPENDED,(void*)p->id);
#else
result=RAISE_SIGNAL(p,GC_sig_thr_restart);
switch(result){
case ESRCH:
n_live_threads--;
break;
case 0:
if (GC_on_thread_event)
GC_on_thread_event(GC_EVENT_THREAD_UNSUSPENDED,
(void*)(word)THREAD_SYSTEM_ID(p));
break;
default:
ABORT_ARG1("pthread_kill failed at resume",
":errcode=%d",result);
}
#endif
}
}
}
return n_live_threads;
}
#endif
GC_INNER void GC_start_world(void)
{
#ifndef NACL
int n_live_threads;
GC_ASSERT(I_HOLD_LOCK());
#ifdef DEBUG_THREADS
GC_log_printf("World starting\n");
#endif
#ifndef GC_OPENBSD_UTHREADS
AO_store_release(&GC_world_is_stopped,FALSE);
#endif
n_live_threads=GC_restart_all();
#ifdef GC_OPENBSD_UTHREADS
(void)n_live_threads;
#elif defined(GC_NETBSD_THREADS_WORKAROUND)
if (GC_retry_signals)
n_live_threads=resend_lost_signals(n_live_threads,GC_restart_all);
suspend_restart_barrier(n_live_threads);
#else
if (GC_retry_signals){
n_live_threads=resend_lost_signals(n_live_threads,GC_restart_all);
suspend_restart_barrier(n_live_threads);
}
#endif
#ifdef DEBUG_THREADS
GC_log_printf("World started\n");
#endif
#else
#ifdef DEBUG_THREADS
GC_log_printf("World starting...\n");
#endif
GC_nacl_park_threads_now=0;
if (GC_on_thread_event)
GC_on_thread_event(GC_EVENT_THREAD_UNSUSPENDED,NULL);
#endif
}
GC_INNER void GC_stop_init(void)
{
#if!defined(GC_OPENBSD_UTHREADS)&&!defined(NACL)
struct sigaction act;
char*str;
if (SIGNAL_UNSET==GC_sig_suspend)
GC_sig_suspend=SIG_SUSPEND;
if (SIGNAL_UNSET==GC_sig_thr_restart)
GC_sig_thr_restart=SIG_THR_RESTART;
if (GC_sig_suspend==GC_sig_thr_restart)
ABORT("Cannot use same signal for thread suspend and resume");
if (sem_init(&GC_suspend_ack_sem,GC_SEM_INIT_PSHARED,0)!=0)
ABORT("sem_init failed");
#ifdef SA_RESTART
act.sa_flags=SA_RESTART
#else
act.sa_flags=0
#endif
#ifndef NO_SA_SIGACTION
|SA_SIGINFO
#endif
;
if (sigfillset(&act.sa_mask)!=0){
ABORT("sigfillset failed");
}
#ifdef GC_RTEMS_PTHREADS
if(sigprocmask(SIG_UNBLOCK,&act.sa_mask,NULL)!=0){
ABORT("sigprocmask failed");
}
#endif
GC_remove_allowed_signals(&act.sa_mask);
#ifndef NO_SA_SIGACTION
act.sa_sigaction=GC_suspend_handler;
#else
act.sa_handler=GC_suspend_handler;
#endif
if (sigaction(GC_sig_suspend,&act,NULL)!=0){
ABORT("Cannot set SIG_SUSPEND handler");
}
#ifndef NO_SA_SIGACTION
act.sa_flags&=~SA_SIGINFO;
#endif
act.sa_handler=GC_restart_handler;
if (sigaction(GC_sig_thr_restart,&act,NULL)!=0){
ABORT("Cannot set SIG_THR_RESTART handler");
}
if (sigfillset(&suspend_handler_mask)!=0)ABORT("sigfillset failed");
GC_remove_allowed_signals(&suspend_handler_mask);
if (sigdelset(&suspend_handler_mask,GC_sig_thr_restart)!=0)
ABORT("sigdelset failed");
str=GETENV("GC_RETRY_SIGNALS");
if (str!=NULL){
if (*str=='0'&&*(str+1)=='\0'){
GC_retry_signals=FALSE;
} else {
GC_retry_signals=TRUE;
}
}
if (GC_retry_signals){
GC_COND_LOG_PRINTF(
"Will retry suspend and restart signals if necessary\n");
}
#ifndef NO_SIGNALS_UNBLOCK_IN_MAIN
GC_unblock_gc_signals();
#endif
#endif
}
#endif
#if defined(GC_PTHREADS)&&!defined(GC_WIN32_THREADS)
#include <stdlib.h>
#include <pthread.h>
#include <sched.h>
#include <time.h>
#include <errno.h>
#include <unistd.h>
#if!defined(SN_TARGET_ORBIS)&&!defined(SN_TARGET_PSP2)
#if!defined(GC_RTEMS_PTHREADS)
#include <sys/mman.h>
#endif
#include <sys/time.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#endif
#include <signal.h>
#if defined(GC_DARWIN_THREADS)
#ifndef GC_DARWIN_SEMAPHORE_H
#define GC_DARWIN_SEMAPHORE_H
#if!defined(GC_DARWIN_THREADS)
#error darwin_semaphore.h included with GC_DARWIN_THREADS not defined
#endif
#ifdef __cplusplus
extern "C" {
#endif
typedef struct {
pthread_mutex_t mutex;
pthread_cond_t cond;
int value;
} sem_t;
GC_INLINE int sem_init(sem_t*sem,int pshared,int value){
if (pshared!=0){
errno=EPERM;
return -1;
}
sem->value=value;
if (pthread_mutex_init(&sem->mutex,NULL)!=0)
return -1;
if (pthread_cond_init(&sem->cond,NULL)!=0){
(void)pthread_mutex_destroy(&sem->mutex);
return -1;
}
return 0;
}
GC_INLINE int sem_post(sem_t*sem){
if (pthread_mutex_lock(&sem->mutex)!=0)
return -1;
sem->value++;
if (pthread_cond_signal(&sem->cond)!=0){
(void)pthread_mutex_unlock(&sem->mutex);
return -1;
}
return pthread_mutex_unlock(&sem->mutex)!=0?-1:0;
}
GC_INLINE int sem_wait(sem_t*sem){
if (pthread_mutex_lock(&sem->mutex)!=0)
return -1;
while (sem->value==0){
if (pthread_cond_wait(&sem->cond,&sem->mutex)!=0){
(void)pthread_mutex_unlock(&sem->mutex);
return -1;
}
}
sem->value--;
return pthread_mutex_unlock(&sem->mutex)!=0?-1:0;
}
GC_INLINE int sem_destroy(sem_t*sem){
return pthread_cond_destroy(&sem->cond)!=0
||pthread_mutex_destroy(&sem->mutex)!=0?-1:0;
}
#ifdef __cplusplus
}
#endif
#endif
#else
#include <semaphore.h>
#endif
#if defined(GC_DARWIN_THREADS)||defined(GC_FREEBSD_THREADS)
#include <sys/sysctl.h>
#endif
#if defined(GC_NETBSD_THREADS)||defined(GC_OPENBSD_THREADS)
#include <sys/param.h>
#include <sys/sysctl.h>
#endif
#if!defined(USE_SPIN_LOCK)
GC_INNER pthread_mutex_t GC_allocate_ml=PTHREAD_MUTEX_INITIALIZER;
#endif
#ifdef GC_ASSERTIONS
GC_INNER unsigned long GC_lock_holder=NO_THREAD;
#endif
#if defined(GC_DGUX386_THREADS)
#include <sys/dg_sys_info.h>
#include <sys/_int_psem.h>
typedef unsigned int sem_t;
#endif
#undef pthread_create
#ifndef GC_NO_PTHREAD_SIGMASK
#undef pthread_sigmask
#endif
#ifndef GC_NO_PTHREAD_CANCEL
#undef pthread_cancel
#endif
#ifdef GC_HAVE_PTHREAD_EXIT
#undef pthread_exit
#endif
#undef pthread_join
#undef pthread_detach
#if defined(GC_OSF1_THREADS)&&defined(_PTHREAD_USE_MANGLED_NAMES_)&&!defined(_PTHREAD_USE_PTDNAM_)
#define pthread_create __pthread_create
#define pthread_join __pthread_join
#define pthread_detach __pthread_detach
#ifndef GC_NO_PTHREAD_CANCEL
#define pthread_cancel __pthread_cancel
#endif
#ifdef GC_HAVE_PTHREAD_EXIT
#define pthread_exit __pthread_exit
#endif
#endif
#ifdef GC_USE_LD_WRAP
#define WRAP_FUNC(f)__wrap_##f
#define REAL_FUNC(f)__real_##f
int REAL_FUNC(pthread_create)(pthread_t*,
GC_PTHREAD_CREATE_CONST pthread_attr_t*,
void*(*start_routine)(void*),void*);
int REAL_FUNC(pthread_join)(pthread_t,void**);
int REAL_FUNC(pthread_detach)(pthread_t);
#ifndef GC_NO_PTHREAD_SIGMASK
int REAL_FUNC(pthread_sigmask)(int,const sigset_t*,sigset_t*);
#endif
#ifndef GC_NO_PTHREAD_CANCEL
int REAL_FUNC(pthread_cancel)(pthread_t);
#endif
#ifdef GC_HAVE_PTHREAD_EXIT
void REAL_FUNC(pthread_exit)(void*)GC_PTHREAD_EXIT_ATTRIBUTE;
#endif
#else
#ifdef GC_USE_DLOPEN_WRAP
#include <dlfcn.h>
#define WRAP_FUNC(f)f
#define REAL_FUNC(f)GC_real_##f
typedef int (*GC_pthread_create_t)(pthread_t*,
GC_PTHREAD_CREATE_CONST pthread_attr_t*,
void*(*)(void*),void*);
static GC_pthread_create_t REAL_FUNC(pthread_create);
#ifndef GC_NO_PTHREAD_SIGMASK
typedef int (*GC_pthread_sigmask_t)(int,const sigset_t*,
sigset_t*);
static GC_pthread_sigmask_t REAL_FUNC(pthread_sigmask);
#endif
typedef int (*GC_pthread_join_t)(pthread_t,void**);
static GC_pthread_join_t REAL_FUNC(pthread_join);
typedef int (*GC_pthread_detach_t)(pthread_t);
static GC_pthread_detach_t REAL_FUNC(pthread_detach);
#ifndef GC_NO_PTHREAD_CANCEL
typedef int (*GC_pthread_cancel_t)(pthread_t);
static GC_pthread_cancel_t REAL_FUNC(pthread_cancel);
#endif
#ifdef GC_HAVE_PTHREAD_EXIT
typedef void (*GC_pthread_exit_t)(void*)GC_PTHREAD_EXIT_ATTRIBUTE;
static GC_pthread_exit_t REAL_FUNC(pthread_exit);
#endif
#else
#define WRAP_FUNC(f)GC_##f
#if!defined(GC_DGUX386_THREADS)
#define REAL_FUNC(f)f
#else
#define REAL_FUNC(f)__d10_##f
#endif
#endif
#endif
#if defined(GC_USE_LD_WRAP)||defined(GC_USE_DLOPEN_WRAP)
GC_API int GC_pthread_create(pthread_t*t,
GC_PTHREAD_CREATE_CONST pthread_attr_t*a,
void*(*fn)(void*),void*arg)
{
return pthread_create(t,a,fn,arg);
}
#ifndef GC_NO_PTHREAD_SIGMASK
GC_API int GC_pthread_sigmask(int how,const sigset_t*mask,
sigset_t*old)
{
return pthread_sigmask(how,mask,old);
}
#endif
GC_API int GC_pthread_join(pthread_t t,void**res)
{
return pthread_join(t,res);
}
GC_API int GC_pthread_detach(pthread_t t)
{
return pthread_detach(t);
}
#ifndef GC_NO_PTHREAD_CANCEL
GC_API int GC_pthread_cancel(pthread_t t)
{
return pthread_cancel(t);
}
#endif
#ifdef GC_HAVE_PTHREAD_EXIT
GC_API GC_PTHREAD_EXIT_ATTRIBUTE void GC_pthread_exit(void*retval)
{
pthread_exit(retval);
}
#endif
#endif
#ifdef GC_USE_DLOPEN_WRAP
STATIC GC_bool GC_syms_initialized=FALSE;
STATIC void GC_init_real_syms(void)
{
void*dl_handle;
if (GC_syms_initialized)return;
#ifdef RTLD_NEXT
dl_handle=RTLD_NEXT;
#else
dl_handle=dlopen("libpthread.so.0",RTLD_LAZY);
if (NULL==dl_handle){
dl_handle=dlopen("libpthread.so",RTLD_LAZY);
}
if (NULL==dl_handle)ABORT("Couldn't open libpthread");
#endif
REAL_FUNC(pthread_create)=(GC_pthread_create_t)(word)
dlsym(dl_handle,"pthread_create");
#ifdef RTLD_NEXT
if (REAL_FUNC(pthread_create)==0)
ABORT("pthread_create not found"
" (probably -lgc is specified after -lpthread)");
#endif
#ifndef GC_NO_PTHREAD_SIGMASK
REAL_FUNC(pthread_sigmask)=(GC_pthread_sigmask_t)(word)
dlsym(dl_handle,"pthread_sigmask");
#endif
REAL_FUNC(pthread_join)=(GC_pthread_join_t)(word)
dlsym(dl_handle,"pthread_join");
REAL_FUNC(pthread_detach)=(GC_pthread_detach_t)(word)
dlsym(dl_handle,"pthread_detach");
#ifndef GC_NO_PTHREAD_CANCEL
REAL_FUNC(pthread_cancel)=(GC_pthread_cancel_t)(word)
dlsym(dl_handle,"pthread_cancel");
#endif
#ifdef GC_HAVE_PTHREAD_EXIT
REAL_FUNC(pthread_exit)=(GC_pthread_exit_t)(word)
dlsym(dl_handle,"pthread_exit");
#endif
GC_syms_initialized=TRUE;
}
#define INIT_REAL_SYMS()if (EXPECT(GC_syms_initialized,TRUE)){} else GC_init_real_syms()
#define ASSERT_SYMS_INITIALIZED()GC_ASSERT(GC_syms_initialized)
#else
#define INIT_REAL_SYMS()(void)0
#define ASSERT_SYMS_INITIALIZED()GC_ASSERT(parallel_initialized)
#endif
static GC_bool parallel_initialized=FALSE;
#ifndef GC_ALWAYS_MULTITHREADED
GC_INNER GC_bool GC_need_to_lock=FALSE;
#endif
STATIC int GC_nprocs=1;
#ifdef THREAD_LOCAL_ALLOC
GC_INNER void GC_mark_thread_local_free_lists(void)
{
int i;
GC_thread p;
for (i=0;i < THREAD_TABLE_SZ;++i){
for (p=GC_threads[i];0!=p;p=p->next){
if (!(p->flags&FINISHED))
GC_mark_thread_local_fls_for(&(p->tlfs));
}
}
}
#if defined(GC_ASSERTIONS)
void GC_check_tls(void)
{
int i;
GC_thread p;
for (i=0;i < THREAD_TABLE_SZ;++i){
for (p=GC_threads[i];0!=p;p=p->next){
if (!(p->flags&FINISHED))
GC_check_tls_for(&(p->tlfs));
}
}
#if defined(USE_CUSTOM_SPECIFIC)
if (GC_thread_key!=0)
GC_check_tsd_marks(GC_thread_key);
#endif
}
#endif
#endif
#ifdef PARALLEL_MARK
#ifndef MAX_MARKERS
#define MAX_MARKERS 16
#endif
static ptr_t marker_sp[MAX_MARKERS - 1]={0};
#ifdef IA64
static ptr_t marker_bsp[MAX_MARKERS - 1]={0};
#endif
#if defined(GC_DARWIN_THREADS)&&!defined(GC_NO_THREADS_DISCOVERY)
static mach_port_t marker_mach_threads[MAX_MARKERS - 1]={0};
GC_INNER GC_bool GC_is_mach_marker(thread_act_t thread)
{
int i;
for (i=0;i < GC_markers_m1;i++){
if (marker_mach_threads[i]==thread)
return TRUE;
}
return FALSE;
}
#endif
#ifdef HAVE_PTHREAD_SETNAME_NP_WITH_TID_AND_ARG
static void set_marker_thread_name(unsigned id)
{
int err=pthread_setname_np(pthread_self(),"GC-marker-%zu",
(void*)(size_t)id);
if (err!=0)
WARN("pthread_setname_np failed,errno=%" WARN_PRIdPTR "\n",err);
}
#elif defined(HAVE_PTHREAD_SETNAME_NP_WITH_TID)||defined(HAVE_PTHREAD_SETNAME_NP_WITHOUT_TID)
static void set_marker_thread_name(unsigned id)
{
char name_buf[16];
int len=sizeof("GC-marker-")- 1;
BCOPY("GC-marker-",name_buf,len);
if (id>=10)
name_buf[len++]=(char)('0'+(id/10)% 10);
name_buf[len]=(char)('0'+id % 10);
name_buf[len+1]='\0';
#ifdef HAVE_PTHREAD_SETNAME_NP_WITHOUT_TID
(void)pthread_setname_np(name_buf);
#else
if (pthread_setname_np(pthread_self(),name_buf)!=0)
WARN("pthread_setname_np failed\n",0);
#endif
}
#else
#define set_marker_thread_name(id)(void)(id)
#endif
STATIC void*GC_mark_thread(void*id)
{
word my_mark_no=0;
IF_CANCEL(int cancel_state;)
if ((word)id==GC_WORD_MAX)return 0;
DISABLE_CANCEL(cancel_state);
set_marker_thread_name((unsigned)(word)id);
marker_sp[(word)id]=GC_approx_sp();
#ifdef IA64
marker_bsp[(word)id]=GC_save_regs_in_stack();
#endif
#if defined(GC_DARWIN_THREADS)&&!defined(GC_NO_THREADS_DISCOVERY)
marker_mach_threads[(word)id]=mach_thread_self();
#endif
GC_acquire_mark_lock();
if (0==--GC_fl_builder_count)
GC_notify_all_builder();
for (;;++my_mark_no){
if (my_mark_no < GC_mark_no||my_mark_no > GC_mark_no+2){
my_mark_no=GC_mark_no;
}
#ifdef DEBUG_THREADS
GC_log_printf("Starting mark helper for mark number %lu\n",
(unsigned long)my_mark_no);
#endif
GC_help_marker(my_mark_no);
}
}
STATIC pthread_t GC_mark_threads[MAX_MARKERS];
#ifdef CAN_HANDLE_FORK
static int available_markers_m1=0;
static pthread_cond_t mark_cv;
#else
#define available_markers_m1 GC_markers_m1
static pthread_cond_t mark_cv=PTHREAD_COND_INITIALIZER;
#endif
GC_INNER void GC_start_mark_threads_inner(void)
{
int i;
pthread_attr_t attr;
#ifndef NO_MARKER_SPECIAL_SIGMASK
sigset_t set,oldset;
#endif
GC_ASSERT(I_DONT_HOLD_LOCK());
if (available_markers_m1<=0)return;
#ifdef CAN_HANDLE_FORK
if (GC_parallel)return;
{
pthread_cond_t mark_cv_local=PTHREAD_COND_INITIALIZER;
BCOPY(&mark_cv_local,&mark_cv,sizeof(mark_cv));
}
#endif
GC_ASSERT(GC_fl_builder_count==0);
INIT_REAL_SYMS();
if (0!=pthread_attr_init(&attr))ABORT("pthread_attr_init failed");
if (0!=pthread_attr_setdetachstate(&attr,PTHREAD_CREATE_DETACHED))
ABORT("pthread_attr_setdetachstate failed");
#ifdef DEFAULT_STACK_MAYBE_SMALL
{
size_t old_size;
if (pthread_attr_getstacksize(&attr,&old_size)!=0)
ABORT("pthread_attr_getstacksize failed");
if (old_size < MIN_STACK_SIZE
&&old_size!=0){
if (pthread_attr_setstacksize(&attr,MIN_STACK_SIZE)!=0)
ABORT("pthread_attr_setstacksize failed");
}
}
#endif
#ifndef NO_MARKER_SPECIAL_SIGMASK
if (sigfillset(&set)!=0)
ABORT("sigfillset failed");
#if!defined(GC_DARWIN_THREADS)&&!defined(GC_OPENBSD_UTHREADS)&&!defined(NACL)
if (sigdelset(&set,GC_get_suspend_signal())!=0
||sigdelset(&set,GC_get_thr_restart_signal())!=0)
ABORT("sigdelset failed");
#endif
if (REAL_FUNC(pthread_sigmask)(SIG_BLOCK,&set,&oldset)< 0){
WARN("pthread_sigmask set failed,no markers started,"
" errno=%" WARN_PRIdPTR "\n",errno);
GC_markers_m1=0;
(void)pthread_attr_destroy(&attr);
return;
}
#endif
#ifdef CAN_HANDLE_FORK
GC_markers_m1=available_markers_m1;
#endif
for (i=0;i < available_markers_m1;++i){
if (0!=REAL_FUNC(pthread_create)(GC_mark_threads+i,&attr,
GC_mark_thread,(void*)(word)i)){
WARN("Marker thread creation failed,errno=%" WARN_PRIdPTR "\n",
errno);
GC_markers_m1=i;
break;
}
}
#ifndef NO_MARKER_SPECIAL_SIGMASK
if (REAL_FUNC(pthread_sigmask)(SIG_SETMASK,&oldset,NULL)< 0){
WARN("pthread_sigmask restore failed,errno=%" WARN_PRIdPTR "\n",
errno);
}
#endif
(void)pthread_attr_destroy(&attr);
GC_wait_for_markers_init();
GC_COND_LOG_PRINTF("Started %d mark helper threads\n",GC_markers_m1);
}
#endif
GC_INNER GC_bool GC_thr_initialized=FALSE;
GC_INNER volatile GC_thread GC_threads[THREAD_TABLE_SZ]={0};
void GC_push_thread_structures(void)
{
GC_ASSERT(I_HOLD_LOCK());
GC_PUSH_ALL_SYM(GC_threads);
#if defined(THREAD_LOCAL_ALLOC)
GC_PUSH_ALL_SYM(GC_thread_key);
#endif
}
#ifdef DEBUG_THREADS
STATIC int GC_count_threads(void)
{
int i;
int count=0;
GC_ASSERT(I_HOLD_LOCK());
for (i=0;i < THREAD_TABLE_SZ;++i){
GC_thread th=GC_threads[i];
while (th){
if (!(th->flags&FINISHED))
++count;
th=th->next;
}
}
return count;
}
#endif
static struct GC_Thread_Rep first_thread;
STATIC GC_thread GC_new_thread(pthread_t id)
{
int hv=THREAD_TABLE_INDEX(id);
GC_thread result;
static GC_bool first_thread_used=FALSE;
#ifdef DEBUG_THREADS
GC_log_printf("Creating thread %p\n",(void*)id);
for (result=GC_threads[hv];result!=NULL;result=result->next)
if (!THREAD_EQUAL(result->id,id)){
GC_log_printf("Hash collision at GC_threads[%d]\n",hv);
break;
}
#endif
GC_ASSERT(I_HOLD_LOCK());
if (!EXPECT(first_thread_used,TRUE)){
result=&first_thread;
first_thread_used=TRUE;
GC_ASSERT(NULL==GC_threads[hv]);
#if defined(THREAD_SANITIZER)&&defined(CPPCHECK)
GC_noop1(result->dummy[0]);
#endif
} else {
result=(struct GC_Thread_Rep*)
GC_INTERNAL_MALLOC(sizeof(struct GC_Thread_Rep),NORMAL);
if (result==0)return(0);
}
result->id=id;
#ifdef USE_TKILL_ON_ANDROID
result->kernel_id=gettid();
#endif
result->next=GC_threads[hv];
GC_threads[hv]=result;
#ifdef NACL
GC_nacl_gc_thread_self=result;
GC_nacl_initialize_gc_thread();
#endif
GC_ASSERT(result->flags==0&&result->thread_blocked==0);
if (EXPECT(result!=&first_thread,TRUE))
GC_dirty(result);
return(result);
}
STATIC void GC_delete_thread(pthread_t id)
{
int hv=THREAD_TABLE_INDEX(id);
GC_thread p=GC_threads[hv];
GC_thread prev=NULL;
#ifdef DEBUG_THREADS
GC_log_printf("Deleting thread %p,n_threads=%d\n",
(void*)id,GC_count_threads());
#endif
#ifdef NACL
GC_nacl_shutdown_gc_thread();
GC_nacl_gc_thread_self=NULL;
#endif
GC_ASSERT(I_HOLD_LOCK());
while (!THREAD_EQUAL(p->id,id)){
prev=p;
p=p->next;
}
if (prev==0){
GC_threads[hv]=p->next;
} else {
GC_ASSERT(prev!=&first_thread);
prev->next=p->next;
GC_dirty(prev);
}
if (p!=&first_thread){
#ifdef GC_DARWIN_THREADS
mach_port_deallocate(mach_task_self(),p->stop_info.mach_thread);
#endif
GC_INTERNAL_FREE(p);
}
}
STATIC void GC_delete_gc_thread(GC_thread t)
{
pthread_t id=t->id;
int hv=THREAD_TABLE_INDEX(id);
GC_thread p=GC_threads[hv];
GC_thread prev=NULL;
GC_ASSERT(I_HOLD_LOCK());
while (p!=t){
prev=p;
p=p->next;
}
if (prev==0){
GC_threads[hv]=p->next;
} else {
GC_ASSERT(prev!=&first_thread);
prev->next=p->next;
GC_dirty(prev);
}
#ifdef GC_DARWIN_THREADS
mach_port_deallocate(mach_task_self(),p->stop_info.mach_thread);
#endif
GC_INTERNAL_FREE(p);
#ifdef DEBUG_THREADS
GC_log_printf("Deleted thread %p,n_threads=%d\n",
(void*)id,GC_count_threads());
#endif
}
GC_INNER GC_thread GC_lookup_thread(pthread_t id)
{
GC_thread p=GC_threads[THREAD_TABLE_INDEX(id)];
while (p!=0&&!THREAD_EQUAL(p->id,id))p=p->next;
return(p);
}
GC_INNER void GC_reset_finalizer_nested(void)
{
GC_thread me=GC_lookup_thread(pthread_self());
me->finalizer_nested=0;
}
GC_INNER unsigned char*GC_check_finalizer_nested(void)
{
GC_thread me=GC_lookup_thread(pthread_self());
unsigned nesting_level=me->finalizer_nested;
if (nesting_level){
if (++me->finalizer_skipped < (1U<<nesting_level))return NULL;
me->finalizer_skipped=0;
}
me->finalizer_nested=(unsigned char)(nesting_level+1);
return&me->finalizer_nested;
}
#if defined(GC_ASSERTIONS)&&defined(THREAD_LOCAL_ALLOC)
GC_bool GC_is_thread_tsd_valid(void*tsd)
{
GC_thread me;
DCL_LOCK_STATE;
LOCK();
me=GC_lookup_thread(pthread_self());
UNLOCK();
return (word)tsd>=(word)(&me->tlfs)
&&(word)tsd < (word)(&me->tlfs)+sizeof(me->tlfs);
}
#endif
GC_API int GC_CALL GC_thread_is_registered(void)
{
pthread_t self=pthread_self();
GC_thread me;
DCL_LOCK_STATE;
LOCK();
me=GC_lookup_thread(self);
UNLOCK();
return me!=NULL;
}
static pthread_t main_pthread_id;
static void*main_stack,*main_altstack;
static word main_stack_size,main_altstack_size;
GC_API void GC_CALL GC_register_altstack(void*stack,GC_word stack_size,
void*altstack,
GC_word altstack_size)
{
GC_thread me;
pthread_t self=pthread_self();
DCL_LOCK_STATE;
LOCK();
me=GC_lookup_thread(self);
if (me!=NULL){
me->stack=(ptr_t)stack;
me->stack_size=stack_size;
me->altstack=(ptr_t)altstack;
me->altstack_size=altstack_size;
} else {
main_pthread_id=self;
main_stack=stack;
main_stack_size=stack_size;
main_altstack=altstack;
main_altstack_size=altstack_size;
}
UNLOCK();
}
#ifdef CAN_HANDLE_FORK
#ifdef CAN_CALL_ATFORK
GC_ATTR_NO_SANITIZE_THREAD
#endif
static void store_to_threads_table(int hv,GC_thread me)
{
GC_threads[hv]=me;
}
STATIC void GC_remove_all_threads_but_me(void)
{
pthread_t self=pthread_self();
int hv;
for (hv=0;hv < THREAD_TABLE_SZ;++hv){
GC_thread p,next;
GC_thread me=NULL;
for (p=GC_threads[hv];0!=p;p=next){
next=p->next;
if (THREAD_EQUAL(p->id,self)
&&me==NULL){
me=p;
p->next=0;
#ifdef GC_DARWIN_THREADS
me->stop_info.mach_thread=mach_thread_self();
#endif
#ifdef USE_TKILL_ON_ANDROID
me->kernel_id=gettid();
#endif
#if defined(THREAD_LOCAL_ALLOC)&&!defined(USE_CUSTOM_SPECIFIC)
{
int res;
res=GC_setspecific(GC_thread_key,&me->tlfs);
if (COVERT_DATAFLOW(res)!=0)
ABORT("GC_setspecific failed (in child)");
}
#endif
} else {
#ifdef THREAD_LOCAL_ALLOC
if (!(p->flags&FINISHED)){
GC_remove_specific_after_fork(GC_thread_key,p->id);
}
#endif
#if!defined(THREAD_SANITIZER)||!defined(CAN_CALL_ATFORK)
if (p!=&first_thread)GC_INTERNAL_FREE(p);
#endif
}
}
store_to_threads_table(hv,me);
}
}
#endif
#ifdef USE_PROC_FOR_LIBRARIES
GC_INNER GC_bool GC_segment_is_thread_stack(ptr_t lo,ptr_t hi)
{
int i;
GC_thread p;
GC_ASSERT(I_HOLD_LOCK());
#ifdef PARALLEL_MARK
for (i=0;i < GC_markers_m1;++i){
if ((word)marker_sp[i] > (word)lo&&(word)marker_sp[i] < (word)hi)
return TRUE;
#ifdef IA64
if ((word)marker_bsp[i] > (word)lo
&&(word)marker_bsp[i] < (word)hi)
return TRUE;
#endif
}
#endif
for (i=0;i < THREAD_TABLE_SZ;i++){
for (p=GC_threads[i];p!=0;p=p->next){
if (0!=p->stack_end){
#ifdef STACK_GROWS_UP
if ((word)p->stack_end>=(word)lo
&&(word)p->stack_end < (word)hi)
return TRUE;
#else
if ((word)p->stack_end > (word)lo
&&(word)p->stack_end<=(word)hi)
return TRUE;
#endif
}
}
}
return FALSE;
}
#endif
#ifdef IA64
GC_INNER ptr_t GC_greatest_stack_base_below(ptr_t bound)
{
int i;
GC_thread p;
ptr_t result=0;
GC_ASSERT(I_HOLD_LOCK());
#ifdef PARALLEL_MARK
for (i=0;i < GC_markers_m1;++i){
if ((word)marker_sp[i] > (word)result
&&(word)marker_sp[i] < (word)bound)
result=marker_sp[i];
}
#endif
for (i=0;i < THREAD_TABLE_SZ;i++){
for (p=GC_threads[i];p!=0;p=p->next){
if ((word)p->stack_end > (word)result
&&(word)p->stack_end < (word)bound){
result=p->stack_end;
}
}
}
return result;
}
#endif
#ifndef STAT_READ
#define STAT_BUF_SIZE 4096
#define STAT_READ read
#endif
#ifdef GC_HPUX_THREADS
#define GC_get_nprocs()pthread_num_processors_np()
#elif defined(GC_OSF1_THREADS)||defined(GC_AIX_THREADS)||defined(GC_HAIKU_THREADS)||defined(GC_SOLARIS_THREADS)||defined(HURD)||defined(HOST_ANDROID)||defined(NACL)
GC_INLINE int GC_get_nprocs(void)
{
int nprocs=(int)sysconf(_SC_NPROCESSORS_ONLN);
return nprocs > 0?nprocs:1;
}
#elif defined(GC_IRIX_THREADS)
GC_INLINE int GC_get_nprocs(void)
{
int nprocs=(int)sysconf(_SC_NPROC_ONLN);
return nprocs > 0?nprocs:1;
}
#elif defined(GC_LINUX_THREADS)
STATIC int GC_get_nprocs(void)
{
char stat_buf[STAT_BUF_SIZE];
int f;
int result,i,len;
f=open("/proc/stat",O_RDONLY);
if (f < 0){
WARN("Couldn't read/proc/stat\n",0);
return 1;
}
len=STAT_READ(f,stat_buf,STAT_BUF_SIZE);
close(f);
result=1;
for (i=0;i < len - 100;++i){
if (stat_buf[i]=='\n'&&stat_buf[i+1]=='c'
&&stat_buf[i+2]=='p'&&stat_buf[i+3]=='u'){
int cpu_no=atoi(&stat_buf[i+4]);
if (cpu_no>=result)
result=cpu_no+1;
}
}
return result;
}
#elif defined(GC_DGUX386_THREADS)
STATIC int GC_get_nprocs(void)
{
int numCpus;
struct dg_sys_info_pm_info pm_sysinfo;
int status=0;
status=dg_sys_info((long int*)&pm_sysinfo,
DG_SYS_INFO_PM_INFO_TYPE,DG_SYS_INFO_PM_CURRENT_VERSION);
if (status < 0)
numCpus=-1;
else
numCpus=pm_sysinfo.idle_vp_count;
return(numCpus);
}
#elif defined(GC_DARWIN_THREADS)||defined(GC_FREEBSD_THREADS)||defined(GC_NETBSD_THREADS)||defined(GC_OPENBSD_THREADS)
STATIC int GC_get_nprocs(void)
{
int mib[]={CTL_HW,HW_NCPU};
int res;
size_t len=sizeof(res);
sysctl(mib,sizeof(mib)/sizeof(int),&res,&len,NULL,0);
return res;
}
#else
#define GC_get_nprocs()1
#endif
#if defined(ARM32)&&defined(GC_LINUX_THREADS)&&!defined(NACL)
STATIC int GC_get_nprocs_present(void)
{
char stat_buf[16];
int f;
int len;
f=open("/sys/devices/system/cpu/present",O_RDONLY);
if (f < 0)
return -1;
len=STAT_READ(f,stat_buf,sizeof(stat_buf));
close(f);
if (len < 2||stat_buf[0]!='0'||stat_buf[len - 1]!='\n'){
return 0;
} else if (len==2){
return 1;
} else if (stat_buf[1]!='-'){
return 0;
}
stat_buf[len - 1]='\0';
return atoi(&stat_buf[2])+1;
}
#endif
STATIC void GC_wait_for_gc_completion(GC_bool wait_for_all)
{
DCL_LOCK_STATE;
#if!defined(THREAD_SANITIZER)||!defined(CAN_CALL_ATFORK)
GC_ASSERT(I_HOLD_LOCK());
#endif
ASSERT_CANCEL_DISABLED();
if (GC_incremental&&GC_collection_in_progress()){
word old_gc_no=GC_gc_no;
while (GC_incremental&&GC_collection_in_progress()
&&(wait_for_all||old_gc_no==GC_gc_no)){
ENTER_GC();
GC_in_thread_creation=TRUE;
GC_collect_a_little_inner(1);
GC_in_thread_creation=FALSE;
EXIT_GC();
UNLOCK();
sched_yield();
LOCK();
}
}
}
#ifdef CAN_HANDLE_FORK
IF_CANCEL(static int fork_cancel_state;)
#if defined(GC_ASSERTIONS)&&defined(CAN_CALL_ATFORK)
GC_ATTR_NO_SANITIZE_THREAD
#endif
static void fork_prepare_proc(void)
{
LOCK();
DISABLE_CANCEL(fork_cancel_state);
#if defined(PARALLEL_MARK)
if (GC_parallel)
GC_wait_for_reclaim();
#endif
GC_wait_for_gc_completion(TRUE);
#if defined(PARALLEL_MARK)
if (GC_parallel)
GC_acquire_mark_lock();
#endif
GC_acquire_dirty_lock();
}
#if defined(GC_ASSERTIONS)&&defined(CAN_CALL_ATFORK)
GC_ATTR_NO_SANITIZE_THREAD
#endif
static void fork_parent_proc(void)
{
GC_release_dirty_lock();
#if defined(PARALLEL_MARK)
if (GC_parallel)
GC_release_mark_lock();
#endif
RESTORE_CANCEL(fork_cancel_state);
UNLOCK();
}
#if defined(GC_ASSERTIONS)&&defined(CAN_CALL_ATFORK)
GC_ATTR_NO_SANITIZE_THREAD
#endif
static void fork_child_proc(void)
{
GC_release_dirty_lock();
#if defined(PARALLEL_MARK)
if (GC_parallel)
GC_release_mark_lock();
#endif
GC_remove_all_threads_but_me();
#ifdef PARALLEL_MARK
GC_parallel=FALSE;
#endif
RESTORE_CANCEL(fork_cancel_state);
UNLOCK();
#ifdef USE_PTHREAD_LOCKS
GC_ASSERT(I_DONT_HOLD_LOCK());
(void)pthread_mutex_destroy(&GC_allocate_ml);
if (0!=pthread_mutex_init(&GC_allocate_ml,NULL))
ABORT("pthread_mutex_init failed (in child)");
#endif
}
GC_API void GC_CALL GC_atfork_prepare(void)
{
if (!EXPECT(GC_is_initialized,TRUE))GC_init();
#if defined(GC_DARWIN_THREADS)&&defined(MPROTECT_VDB)
if (GC_auto_incremental){
GC_ASSERT(0==GC_handle_fork);
ABORT("Unable to fork while mprotect_thread is running");
}
#endif
if (GC_handle_fork<=0)
fork_prepare_proc();
}
GC_API void GC_CALL GC_atfork_parent(void)
{
if (GC_handle_fork<=0)
fork_parent_proc();
}
GC_API void GC_CALL GC_atfork_child(void)
{
if (GC_handle_fork<=0)
fork_child_proc();
}
#endif
#ifdef INCLUDE_LINUX_THREAD_DESCR
__thread int GC_dummy_thread_local;
#endif
#ifdef PARALLEL_MARK
static void setup_mark_lock(void);
static unsigned required_markers_cnt=0;
#endif
GC_API void GC_CALL GC_set_markers_count(unsigned markers GC_ATTR_UNUSED)
{
#ifdef PARALLEL_MARK
required_markers_cnt=markers < MAX_MARKERS?markers:MAX_MARKERS;
#endif
}
GC_INNER void GC_thr_init(void)
{
GC_ASSERT(I_HOLD_LOCK());
if (GC_thr_initialized)return;
GC_thr_initialized=TRUE;
GC_ASSERT((word)&GC_threads % sizeof(word)==0);
#ifdef CAN_HANDLE_FORK
if (GC_handle_fork){
#ifdef CAN_CALL_ATFORK
if (pthread_atfork(fork_prepare_proc,fork_parent_proc,
fork_child_proc)==0){
GC_handle_fork=1;
} else
#endif
if (GC_handle_fork!=-1)
ABORT("pthread_atfork failed");
}
#endif
#ifdef INCLUDE_LINUX_THREAD_DESCR
{
ptr_t thread_local_addr=(ptr_t)(&GC_dummy_thread_local);
ptr_t main_thread_start,main_thread_end;
if (!GC_enclosing_mapping(thread_local_addr,&main_thread_start,
&main_thread_end)){
ABORT("Failed to find mapping for main thread thread locals");
} else {
GC_add_roots_inner(main_thread_start,main_thread_end,FALSE);
}
}
#endif
{
pthread_t self=pthread_self();
GC_thread t=GC_new_thread(self);
if (t==NULL)
ABORT("Failed to allocate memory for the initial thread");
#ifdef GC_DARWIN_THREADS
t->stop_info.mach_thread=mach_thread_self();
#else
t->stop_info.stack_ptr=GC_approx_sp();
#endif
t->flags=DETACHED|MAIN_THREAD;
if (THREAD_EQUAL(self,main_pthread_id)){
t->stack=(ptr_t)main_stack;
t->stack_size=main_stack_size;
t->altstack=(ptr_t)main_altstack;
t->altstack_size=main_altstack_size;
}
}
#ifndef GC_DARWIN_THREADS
GC_stop_init();
#endif
{
char*nprocs_string=GETENV("GC_NPROCS");
GC_nprocs=-1;
if (nprocs_string!=NULL)GC_nprocs=atoi(nprocs_string);
}
if (GC_nprocs<=0
#if defined(ARM32)&&defined(GC_LINUX_THREADS)&&!defined(NACL)
&&(GC_nprocs=GC_get_nprocs_present())<=1
#endif
)
{
GC_nprocs=GC_get_nprocs();
}
if (GC_nprocs<=0){
WARN("GC_get_nprocs()returned %" WARN_PRIdPTR "\n",GC_nprocs);
GC_nprocs=2;
#ifdef PARALLEL_MARK
available_markers_m1=0;
#endif
} else {
#ifdef PARALLEL_MARK
{
char*markers_string=GETENV("GC_MARKERS");
int markers=required_markers_cnt;
if (markers_string!=NULL){
markers=atoi(markers_string);
if (markers<=0||markers > MAX_MARKERS){
WARN("Too big or invalid number of mark threads:%" WARN_PRIdPTR
";using maximum threads\n",(signed_word)markers);
markers=MAX_MARKERS;
}
} else if (0==markers){
markers=GC_nprocs;
#if defined(GC_MIN_MARKERS)&&!defined(CPPCHECK)
if (markers < GC_MIN_MARKERS)
markers=GC_MIN_MARKERS;
#endif
if (markers > MAX_MARKERS)
markers=MAX_MARKERS;
}
available_markers_m1=markers - 1;
}
#endif
}
GC_COND_LOG_PRINTF("Number of processors=%d\n",GC_nprocs);
#ifdef PARALLEL_MARK
if (available_markers_m1<=0){
GC_parallel=FALSE;
GC_COND_LOG_PRINTF(
"Single marker thread,turning off parallel marking\n");
} else {
setup_mark_lock();
}
#endif
}
GC_INNER void GC_init_parallel(void)
{
#if defined(THREAD_LOCAL_ALLOC)
DCL_LOCK_STATE;
#endif
if (parallel_initialized)return;
parallel_initialized=TRUE;
if (!GC_is_initialized)GC_init();
#if defined(THREAD_LOCAL_ALLOC)
LOCK();
GC_init_thread_local(&(GC_lookup_thread(pthread_self())->tlfs));
UNLOCK();
#endif
}
#ifndef GC_NO_PTHREAD_SIGMASK
GC_API int WRAP_FUNC(pthread_sigmask)(int how,const sigset_t*set,
sigset_t*oset)
{
sigset_t fudged_set;
INIT_REAL_SYMS();
if (set!=NULL&&(how==SIG_BLOCK||how==SIG_SETMASK)){
int sig_suspend=GC_get_suspend_signal();
fudged_set=*set;
GC_ASSERT(sig_suspend>=0);
if (sigdelset(&fudged_set,sig_suspend)!=0)
ABORT("sigdelset failed");
set=&fudged_set;
}
return(REAL_FUNC(pthread_sigmask)(how,set,oset));
}
#endif
GC_INNER void GC_do_blocking_inner(ptr_t data,void*context GC_ATTR_UNUSED)
{
struct blocking_data*d=(struct blocking_data*)data;
pthread_t self=pthread_self();
GC_thread me;
#if defined(SPARC)||defined(IA64)
ptr_t stack_ptr=GC_save_regs_in_stack();
#endif
#if defined(GC_DARWIN_THREADS)&&!defined(DARWIN_DONT_PARSE_STACK)
GC_bool topOfStackUnset=FALSE;
#endif
DCL_LOCK_STATE;
LOCK();
me=GC_lookup_thread(self);
GC_ASSERT(!(me->thread_blocked));
#ifdef SPARC
me->stop_info.stack_ptr=stack_ptr;
#else
me->stop_info.stack_ptr=GC_approx_sp();
#endif
#if defined(GC_DARWIN_THREADS)&&!defined(DARWIN_DONT_PARSE_STACK)
if (me->topOfStack==NULL){
topOfStackUnset=TRUE;
me->topOfStack=GC_FindTopOfStack(0);
}
#endif
#ifdef IA64
me->backing_store_ptr=stack_ptr;
#endif
me->thread_blocked=(unsigned char)TRUE;
UNLOCK();
d->client_data=(d->fn)(d->client_data);
LOCK();
#if defined(CPPCHECK)
GC_noop1((word)&me->thread_blocked);
#endif
me->thread_blocked=FALSE;
#if defined(GC_DARWIN_THREADS)&&!defined(DARWIN_DONT_PARSE_STACK)
if (topOfStackUnset)
me->topOfStack=NULL;
#endif
UNLOCK();
}
GC_API void GC_CALL GC_set_stackbottom(void*gc_thread_handle,
const struct GC_stack_base*sb)
{
GC_thread t=(GC_thread)gc_thread_handle;
GC_ASSERT(sb->mem_base!=NULL);
if (!EXPECT(GC_is_initialized,TRUE)){
GC_ASSERT(NULL==t);
} else {
GC_ASSERT(I_HOLD_LOCK());
if (NULL==t)
t=GC_lookup_thread(pthread_self());
GC_ASSERT((t->flags&FINISHED)==0);
GC_ASSERT(!(t->thread_blocked)
&&NULL==t->traced_stack_sect);
if ((t->flags&MAIN_THREAD)==0){
t->stack_end=(ptr_t)sb->mem_base;
#ifdef IA64
t->backing_store_end=(ptr_t)sb->reg_base;
#endif
return;
}
}
GC_stackbottom=(char*)sb->mem_base;
#ifdef IA64
GC_register_stackbottom=(ptr_t)sb->reg_base;
#endif
}
GC_API void*GC_CALL GC_get_my_stackbottom(struct GC_stack_base*sb)
{
pthread_t self=pthread_self();
GC_thread me;
DCL_LOCK_STATE;
LOCK();
me=GC_lookup_thread(self);
if ((me->flags&MAIN_THREAD)==0){
sb->mem_base=me->stack_end;
#ifdef IA64
sb->reg_base=me->backing_store_end;
#endif
} else {
sb->mem_base=GC_stackbottom;
#ifdef IA64
sb->reg_base=GC_register_stackbottom;
#endif
}
UNLOCK();
return (void*)me;
}
GC_API void*GC_CALL GC_call_with_gc_active(GC_fn_type fn,
void*client_data)
{
struct GC_traced_stack_sect_s stacksect;
pthread_t self=pthread_self();
GC_thread me;
DCL_LOCK_STATE;
LOCK();
me=GC_lookup_thread(self);
if ((me->flags&MAIN_THREAD)==0){
GC_ASSERT(me->stack_end!=NULL);
if ((word)me->stack_end HOTTER_THAN (word)(&stacksect))
me->stack_end=(ptr_t)(&stacksect);
} else {
if ((word)GC_stackbottom HOTTER_THAN (word)(&stacksect))
GC_stackbottom=(ptr_t)COVERT_DATAFLOW(&stacksect);
}
if (!me->thread_blocked){
UNLOCK();
client_data=fn(client_data);
GC_noop1(COVERT_DATAFLOW(&stacksect));
return client_data;
}
stacksect.saved_stack_ptr=me->stop_info.stack_ptr;
#ifdef IA64
stacksect.backing_store_end=GC_save_regs_in_stack();
stacksect.saved_backing_store_ptr=me->backing_store_ptr;
#endif
stacksect.prev=me->traced_stack_sect;
me->thread_blocked=FALSE;
me->traced_stack_sect=&stacksect;
UNLOCK();
client_data=fn(client_data);
GC_ASSERT(me->thread_blocked==FALSE);
GC_ASSERT(me->traced_stack_sect==&stacksect);
#if defined(CPPCHECK)
GC_noop1((word)me->traced_stack_sect);
#endif
LOCK();
me->traced_stack_sect=stacksect.prev;
#ifdef IA64
me->backing_store_ptr=stacksect.saved_backing_store_ptr;
#endif
me->thread_blocked=(unsigned char)TRUE;
me->stop_info.stack_ptr=stacksect.saved_stack_ptr;
UNLOCK();
return client_data;
}
STATIC void GC_unregister_my_thread_inner(GC_thread me)
{
#ifdef DEBUG_THREADS
GC_log_printf(
"Unregistering thread %p,gc_thread=%p,n_threads=%d\n",
(void*)me->id,(void*)me,GC_count_threads());
#endif
GC_ASSERT(!(me->flags&FINISHED));
#if defined(THREAD_LOCAL_ALLOC)
GC_ASSERT(GC_getspecific(GC_thread_key)==&me->tlfs);
GC_destroy_thread_local(&(me->tlfs));
#endif
#if defined(GC_HAVE_PTHREAD_EXIT)||!defined(GC_NO_PTHREAD_CANCEL)
if ((me->flags&DISABLED_GC)!=0){
GC_dont_gc--;
}
#endif
if (me->flags&DETACHED){
GC_delete_thread(pthread_self());
} else {
me->flags|=FINISHED;
}
#if defined(THREAD_LOCAL_ALLOC)
GC_remove_specific(GC_thread_key);
#endif
}
GC_API int GC_CALL GC_unregister_my_thread(void)
{
pthread_t self=pthread_self();
GC_thread me;
IF_CANCEL(int cancel_state;)
DCL_LOCK_STATE;
LOCK();
DISABLE_CANCEL(cancel_state);
GC_wait_for_gc_completion(FALSE);
me=GC_lookup_thread(self);
#ifdef DEBUG_THREADS
GC_log_printf(
"Called GC_unregister_my_thread on %p,gc_thread=%p\n",
(void*)self,(void*)me);
#endif
GC_ASSERT(THREAD_EQUAL(me->id,self));
GC_unregister_my_thread_inner(me);
RESTORE_CANCEL(cancel_state);
UNLOCK();
return GC_SUCCESS;
}
GC_INNER_PTHRSTART void GC_thread_exit_proc(void*arg)
{
IF_CANCEL(int cancel_state;)
DCL_LOCK_STATE;
#ifdef DEBUG_THREADS
GC_log_printf("Called GC_thread_exit_proc on %p,gc_thread=%p\n",
(void*)((GC_thread)arg)->id,arg);
#endif
LOCK();
DISABLE_CANCEL(cancel_state);
GC_wait_for_gc_completion(FALSE);
GC_unregister_my_thread_inner((GC_thread)arg);
RESTORE_CANCEL(cancel_state);
UNLOCK();
}
#if!defined(SN_TARGET_ORBIS)&&!defined(SN_TARGET_PSP2)
GC_API int WRAP_FUNC(pthread_join)(pthread_t thread,void**retval)
{
int result;
GC_thread t;
DCL_LOCK_STATE;
ASSERT_SYMS_INITIALIZED();
LOCK();
t=GC_lookup_thread(thread);
UNLOCK();
result=REAL_FUNC(pthread_join)(thread,retval);
#if defined(GC_FREEBSD_THREADS)
if (result==EINTR)result=0;
#endif
if (result==0){
LOCK();
if ((t->flags&FINISHED)!=0)
GC_delete_gc_thread(t);
UNLOCK();
}
return result;
}
GC_API int WRAP_FUNC(pthread_detach)(pthread_t thread)
{
int result;
GC_thread t;
DCL_LOCK_STATE;
ASSERT_SYMS_INITIALIZED();
LOCK();
t=GC_lookup_thread(thread);
UNLOCK();
result=REAL_FUNC(pthread_detach)(thread);
if (result==0){
LOCK();
t->flags|=DETACHED;
if ((t->flags&FINISHED)!=0){
GC_delete_gc_thread(t);
}
UNLOCK();
}
return result;
}
#endif
#ifndef GC_NO_PTHREAD_CANCEL
GC_API int WRAP_FUNC(pthread_cancel)(pthread_t thread)
{
#ifdef CANCEL_SAFE
GC_thread t;
DCL_LOCK_STATE;
#endif
INIT_REAL_SYMS();
#ifdef CANCEL_SAFE
LOCK();
t=GC_lookup_thread(thread);
if (t!=NULL&&(t->flags&DISABLED_GC)==0){
t->flags|=DISABLED_GC;
GC_dont_gc++;
}
UNLOCK();
#endif
return REAL_FUNC(pthread_cancel)(thread);
}
#endif
#ifdef GC_HAVE_PTHREAD_EXIT
GC_API GC_PTHREAD_EXIT_ATTRIBUTE void WRAP_FUNC(pthread_exit)(void*retval)
{
pthread_t self=pthread_self();
GC_thread me;
DCL_LOCK_STATE;
INIT_REAL_SYMS();
LOCK();
me=GC_lookup_thread(self);
if (me!=0&&(me->flags&DISABLED_GC)==0){
me->flags|=DISABLED_GC;
GC_dont_gc++;
}
UNLOCK();
REAL_FUNC(pthread_exit)(retval);
}
#endif
GC_INNER GC_bool GC_in_thread_creation=FALSE;
GC_INLINE void GC_record_stack_base(GC_thread me,
const struct GC_stack_base*sb)
{
#ifndef GC_DARWIN_THREADS
me->stop_info.stack_ptr=(ptr_t)sb->mem_base;
#endif
me->stack_end=(ptr_t)sb->mem_base;
if (me->stack_end==NULL)
ABORT("Bad stack base in GC_register_my_thread");
#ifdef IA64
me->backing_store_end=(ptr_t)sb->reg_base;
#endif
}
STATIC GC_thread GC_register_my_thread_inner(const struct GC_stack_base*sb,
pthread_t my_pthread)
{
GC_thread me;
GC_in_thread_creation=TRUE;
me=GC_new_thread(my_pthread);
GC_in_thread_creation=FALSE;
if (me==0)
ABORT("Failed to allocate memory for thread registering");
#ifdef GC_DARWIN_THREADS
me->stop_info.mach_thread=mach_thread_self();
#endif
GC_record_stack_base(me,sb);
#ifdef GC_EXPLICIT_SIGNALS_UNBLOCK
GC_unblock_gc_signals();
#endif
return me;
}
GC_API void GC_CALL GC_allow_register_threads(void)
{
GC_ASSERT(GC_lookup_thread(pthread_self())!=0);
set_need_to_lock();
}
GC_API int GC_CALL GC_register_my_thread(const struct GC_stack_base*sb)
{
pthread_t self=pthread_self();
GC_thread me;
DCL_LOCK_STATE;
if (GC_need_to_lock==FALSE)
ABORT("Threads explicit registering is not previously enabled");
LOCK();
me=GC_lookup_thread(self);
if (0==me){
me=GC_register_my_thread_inner(sb,self);
#if defined(CPPCHECK)
GC_noop1(me->flags);
#endif
me->flags|=DETACHED;
#if defined(THREAD_LOCAL_ALLOC)
GC_init_thread_local(&(me->tlfs));
#endif
UNLOCK();
return GC_SUCCESS;
} else if ((me->flags&FINISHED)!=0){
#ifdef GC_DARWIN_THREADS
me->stop_info.mach_thread=mach_thread_self();
#endif
GC_record_stack_base(me,sb);
me->flags&=~FINISHED;
#ifdef GC_EXPLICIT_SIGNALS_UNBLOCK
GC_unblock_gc_signals();
#endif
#if defined(THREAD_LOCAL_ALLOC)
GC_init_thread_local(&(me->tlfs));
#endif
UNLOCK();
return GC_SUCCESS;
} else {
UNLOCK();
return GC_DUPLICATE;
}
}
struct start_info {
void*(*start_routine)(void*);
void*arg;
word flags;
sem_t registered;
};
GC_INNER_PTHRSTART GC_thread GC_start_rtn_prepare_thread(
void*(**pstart)(void*),
void**pstart_arg,
struct GC_stack_base*sb,void*arg)
{
struct start_info*si=(struct start_info*)arg;
pthread_t self=pthread_self();
GC_thread me;
DCL_LOCK_STATE;
#ifdef DEBUG_THREADS
GC_log_printf("Starting thread %p,pid=%ld,sp=%p\n",
(void*)self,(long)getpid(),(void*)&arg);
#endif
LOCK();
me=GC_register_my_thread_inner(sb,self);
me->flags=si->flags;
#if defined(THREAD_LOCAL_ALLOC)
GC_init_thread_local(&(me->tlfs));
#endif
UNLOCK();
*pstart=si->start_routine;
#ifdef DEBUG_THREADS
GC_log_printf("start_routine=%p\n",(void*)(signed_word)(*pstart));
#endif
*pstart_arg=si->arg;
sem_post(&(si->registered));
return me;
}
#if!defined(SN_TARGET_ORBIS)&&!defined(SN_TARGET_PSP2)
STATIC void*GC_start_routine(void*arg)
{
#ifdef INCLUDE_LINUX_THREAD_DESCR
struct GC_stack_base sb;
#ifdef REDIRECT_MALLOC
GC_disable();
#endif
if (GC_get_stack_base(&sb)!=GC_SUCCESS)
ABORT("Failed to get thread stack base");
#ifdef REDIRECT_MALLOC
GC_enable();
#endif
return GC_inner_start_routine(&sb,arg);
#else
return GC_call_with_stack_base(GC_inner_start_routine,arg);
#endif
}
GC_API int WRAP_FUNC(pthread_create)(pthread_t*new_thread,
GC_PTHREAD_CREATE_CONST pthread_attr_t*attr,
void*(*start_routine)(void*),void*arg)
{
int result;
int detachstate;
word my_flags=0;
struct start_info si;
DCL_LOCK_STATE;
INIT_REAL_SYMS();
if (!EXPECT(parallel_initialized,TRUE))
GC_init_parallel();
if (sem_init(&si.registered,GC_SEM_INIT_PSHARED,0)!=0)
ABORT("sem_init failed");
si.start_routine=start_routine;
si.arg=arg;
LOCK();
if (!EXPECT(GC_thr_initialized,TRUE))
GC_thr_init();
#ifdef GC_ASSERTIONS
{
size_t stack_size=0;
if (NULL!=attr){
if (pthread_attr_getstacksize(attr,&stack_size)!=0)
ABORT("pthread_attr_getstacksize failed");
}
if (0==stack_size){
pthread_attr_t my_attr;
if (pthread_attr_init(&my_attr)!=0)
ABORT("pthread_attr_init failed");
if (pthread_attr_getstacksize(&my_attr,&stack_size)!=0)
ABORT("pthread_attr_getstacksize failed");
(void)pthread_attr_destroy(&my_attr);
}
if (0==stack_size){
#ifndef SOLARIS
WARN("Failed to get stack size for assertion checking\n",0);
#endif
stack_size=1000000;
}
GC_ASSERT(stack_size>=65536);
}
#endif
if (NULL==attr){
detachstate=PTHREAD_CREATE_JOINABLE;
} else {
pthread_attr_getdetachstate(attr,&detachstate);
}
if (PTHREAD_CREATE_DETACHED==detachstate)my_flags|=DETACHED;
si.flags=my_flags;
UNLOCK();
#ifdef DEBUG_THREADS
GC_log_printf("About to start new thread from thread %p\n",
(void*)pthread_self());
#endif
set_need_to_lock();
result=REAL_FUNC(pthread_create)(new_thread,attr,GC_start_routine,
&si);
if (0==result){
IF_CANCEL(int cancel_state;)
#ifdef DEBUG_THREADS
GC_log_printf("Started thread %p\n",(void*)(*new_thread));
#endif
DISABLE_CANCEL(cancel_state);
while (0!=sem_wait(&si.registered)){
#if defined(GC_HAIKU_THREADS)
if (EACCES==errno)continue;
#endif
if (EINTR!=errno)ABORT("sem_wait failed");
}
RESTORE_CANCEL(cancel_state);
}
sem_destroy(&si.registered);
return(result);
}
#endif
#if defined(USE_SPIN_LOCK)||!defined(NO_PTHREAD_TRYLOCK)
#define GC_PAUSE_SPIN_CYCLES 10
STATIC void GC_pause(void)
{
int i;
for (i=0;i < GC_PAUSE_SPIN_CYCLES;++i){
#if defined(AO_HAVE_compiler_barrier)&&!defined(BASE_ATOMIC_OPS_EMULATED)
AO_compiler_barrier();
#else
GC_noop1(i);
#endif
}
}
#endif
#ifndef SPIN_MAX
#define SPIN_MAX 128
#endif
GC_INNER volatile GC_bool GC_collecting=FALSE;
#if (!defined(USE_SPIN_LOCK)&&!defined(NO_PTHREAD_TRYLOCK))||defined(PARALLEL_MARK)
#ifdef LOCK_STATS
volatile AO_t GC_spin_count=0;
volatile AO_t GC_block_count=0;
volatile AO_t GC_unlocked_count=0;
#endif
STATIC void GC_generic_lock(pthread_mutex_t*lock)
{
#ifndef NO_PTHREAD_TRYLOCK
unsigned pause_length=1;
unsigned i;
if (0==pthread_mutex_trylock(lock)){
#ifdef LOCK_STATS
(void)AO_fetch_and_add1(&GC_unlocked_count);
#endif
return;
}
for (;pause_length<=SPIN_MAX;pause_length<<=1){
for (i=0;i < pause_length;++i){
GC_pause();
}
switch(pthread_mutex_trylock(lock)){
case 0:
#ifdef LOCK_STATS
(void)AO_fetch_and_add1(&GC_spin_count);
#endif
return;
case EBUSY:
break;
default:
ABORT("Unexpected error from pthread_mutex_trylock");
}
}
#endif
#ifdef LOCK_STATS
(void)AO_fetch_and_add1(&GC_block_count);
#endif
pthread_mutex_lock(lock);
}
#endif
#if defined(AO_HAVE_char_load)&&!defined(BASE_ATOMIC_OPS_EMULATED)
#define is_collecting()((GC_bool)AO_char_load((unsigned char*)&GC_collecting))
#else
#define is_collecting()GC_collecting
#endif
#if defined(USE_SPIN_LOCK)
GC_INNER volatile AO_TS_t GC_allocate_lock=AO_TS_INITIALIZER;
#define low_spin_max 30
#define high_spin_max SPIN_MAX
static volatile AO_t spin_max=low_spin_max;
static volatile AO_t last_spins=0;
GC_INNER void GC_lock(void)
{
unsigned my_spin_max;
unsigned my_last_spins;
unsigned i;
if (AO_test_and_set_acquire(&GC_allocate_lock)==AO_TS_CLEAR){
return;
}
my_spin_max=(unsigned)AO_load(&spin_max);
my_last_spins=(unsigned)AO_load(&last_spins);
for (i=0;i < my_spin_max;i++){
if (is_collecting()||GC_nprocs==1)
goto yield;
if (i < my_last_spins/2){
GC_pause();
continue;
}
if (AO_test_and_set_acquire(&GC_allocate_lock)==AO_TS_CLEAR){
AO_store(&last_spins,(AO_t)i);
AO_store(&spin_max,(AO_t)high_spin_max);
return;
}
}
AO_store(&spin_max,(AO_t)low_spin_max);
yield:
for (i=0;;++i){
if (AO_test_and_set_acquire(&GC_allocate_lock)==AO_TS_CLEAR){
return;
}
#define SLEEP_THRESHOLD 12
if (i < SLEEP_THRESHOLD){
sched_yield();
} else {
struct timespec ts;
if (i > 24)i=24;
ts.tv_sec=0;
ts.tv_nsec=1<<i;
nanosleep(&ts,0);
}
}
}
#elif defined(USE_PTHREAD_LOCKS)
#ifndef NO_PTHREAD_TRYLOCK
GC_INNER void GC_lock(void)
{
if (1==GC_nprocs||is_collecting()){
pthread_mutex_lock(&GC_allocate_ml);
} else {
GC_generic_lock(&GC_allocate_ml);
}
}
#elif defined(GC_ASSERTIONS)
GC_INNER void GC_lock(void)
{
pthread_mutex_lock(&GC_allocate_ml);
}
#endif
#endif
#ifdef PARALLEL_MARK
#ifdef GC_ASSERTIONS
STATIC unsigned long GC_mark_lock_holder=NO_THREAD;
#define SET_MARK_LOCK_HOLDER (void)(GC_mark_lock_holder=NUMERIC_THREAD_ID(pthread_self()))
#define UNSET_MARK_LOCK_HOLDER do { GC_ASSERT(GC_mark_lock_holder==NUMERIC_THREAD_ID(pthread_self()));GC_mark_lock_holder=NO_THREAD;} while (0)
#else
#define SET_MARK_LOCK_HOLDER (void)0
#define UNSET_MARK_LOCK_HOLDER (void)0
#endif
#ifdef GLIBC_2_1_MUTEX_HACK
static pthread_mutex_t mark_mutex=
{0,0,0,PTHREAD_MUTEX_ERRORCHECK_NP,{0,0}};
#else
static pthread_mutex_t mark_mutex=PTHREAD_MUTEX_INITIALIZER;
#endif
static pthread_cond_t builder_cv=PTHREAD_COND_INITIALIZER;
#ifdef GLIBC_2_19_TSX_BUG
static int parse_version(int*pminor,const char*pverstr){
char*endp;
unsigned long value=strtoul(pverstr,&endp,10);
int major=(int)value;
if (major < 0||(char*)pverstr==endp||(unsigned)major!=value){
return -1;
}
if (*endp!='.'){
*pminor=-1;
} else {
value=strtoul(endp+1,&endp,10);
*pminor=(int)value;
if (*pminor < 0||(unsigned)(*pminor)!=value){
return -1;
}
}
return major;
}
#endif
static void setup_mark_lock(void)
{
#ifdef GLIBC_2_19_TSX_BUG
pthread_mutexattr_t mattr;
int glibc_minor=-1;
int glibc_major=parse_version(&glibc_minor,gnu_get_libc_version());
if (glibc_major > 2||(glibc_major==2&&glibc_minor>=19)){
if (0!=pthread_mutexattr_init(&mattr)){
ABORT("pthread_mutexattr_init failed");
}
if (0!=pthread_mutexattr_settype(&mattr,PTHREAD_MUTEX_NORMAL)){
ABORT("pthread_mutexattr_settype failed");
}
if (0!=pthread_mutex_init(&mark_mutex,&mattr)){
ABORT("pthread_mutex_init failed");
}
(void)pthread_mutexattr_destroy(&mattr);
}
#endif
}
GC_INNER void GC_acquire_mark_lock(void)
{
#if defined(NUMERIC_THREAD_ID_UNIQUE)&&!defined(THREAD_SANITIZER)
GC_ASSERT(GC_mark_lock_holder!=NUMERIC_THREAD_ID(pthread_self()));
#endif
GC_generic_lock(&mark_mutex);
SET_MARK_LOCK_HOLDER;
}
GC_INNER void GC_release_mark_lock(void)
{
UNSET_MARK_LOCK_HOLDER;
if (pthread_mutex_unlock(&mark_mutex)!=0){
ABORT("pthread_mutex_unlock failed");
}
}
STATIC void GC_wait_builder(void)
{
ASSERT_CANCEL_DISABLED();
UNSET_MARK_LOCK_HOLDER;
if (pthread_cond_wait(&builder_cv,&mark_mutex)!=0){
ABORT("pthread_cond_wait failed");
}
GC_ASSERT(GC_mark_lock_holder==NO_THREAD);
SET_MARK_LOCK_HOLDER;
}
GC_INNER void GC_wait_for_reclaim(void)
{
GC_acquire_mark_lock();
while (GC_fl_builder_count > 0){
GC_wait_builder();
}
GC_release_mark_lock();
}
GC_INNER void GC_notify_all_builder(void)
{
GC_ASSERT(GC_mark_lock_holder==NUMERIC_THREAD_ID(pthread_self()));
if (pthread_cond_broadcast(&builder_cv)!=0){
ABORT("pthread_cond_broadcast failed");
}
}
GC_INNER void GC_wait_marker(void)
{
ASSERT_CANCEL_DISABLED();
GC_ASSERT(GC_parallel);
UNSET_MARK_LOCK_HOLDER;
if (pthread_cond_wait(&mark_cv,&mark_mutex)!=0){
ABORT("pthread_cond_wait failed");
}
GC_ASSERT(GC_mark_lock_holder==NO_THREAD);
SET_MARK_LOCK_HOLDER;
}
GC_INNER void GC_notify_all_marker(void)
{
GC_ASSERT(GC_parallel);
if (pthread_cond_broadcast(&mark_cv)!=0){
ABORT("pthread_cond_broadcast failed");
}
}
#endif
#ifdef PTHREAD_REGISTER_CANCEL_WEAK_STUBS
EXTERN_C_BEGIN
extern void __pthread_register_cancel(void)__attribute__((__weak__));
extern void __pthread_unregister_cancel(void)__attribute__((__weak__));
EXTERN_C_END
void __pthread_register_cancel(void){}
void __pthread_unregister_cancel(void){}
#endif
#endif
#if defined(USE_CUSTOM_SPECIFIC)
static const tse invalid_tse={INVALID_QTID,0,0,INVALID_THREADID};
GC_INNER int GC_key_create_inner(tsd**key_ptr)
{
int i;
int ret;
tsd*result;
GC_ASSERT(I_HOLD_LOCK());
GC_ASSERT((word)(&invalid_tse.next)% sizeof(tse*)==0);
result=(tsd*)MALLOC_CLEAR(sizeof(tsd));
if (NULL==result)return ENOMEM;
ret=pthread_mutex_init(&result->lock,NULL);
if (ret!=0)return ret;
for (i=0;i < TS_CACHE_SIZE;++i){
result->cache[i]=( tse*)&invalid_tse;
}
#ifdef GC_ASSERTIONS
for (i=0;i < TS_HASH_SIZE;++i){
GC_ASSERT(result->hash[i].p==0);
}
#endif
*key_ptr=result;
return 0;
}
GC_INNER int GC_setspecific(tsd*key,void*value)
{
pthread_t self=pthread_self();
unsigned hash_val=HASH(self);
volatile tse*entry;
GC_ASSERT(I_HOLD_LOCK());
GC_ASSERT(self!=INVALID_THREADID);
GC_dont_gc++;
entry=(volatile tse*)MALLOC_CLEAR(sizeof(tse));
GC_dont_gc--;
if (0==entry)return ENOMEM;
pthread_mutex_lock(&(key->lock));
entry->next=key->hash[hash_val].p;
entry->thread=self;
entry->value=TS_HIDE_VALUE(value);
GC_ASSERT(entry->qtid==INVALID_QTID);
AO_store_release(&key->hash[hash_val].ao,(AO_t)entry);
GC_dirty(( void*)entry);
GC_dirty(key->hash+hash_val);
if (pthread_mutex_unlock(&key->lock)!=0)
ABORT("pthread_mutex_unlock failed (setspecific)");
return 0;
}
GC_INNER void GC_remove_specific_after_fork(tsd*key,pthread_t t)
{
unsigned hash_val=HASH(t);
tse*entry;
tse*prev=NULL;
#ifdef CAN_HANDLE_FORK
GC_ASSERT(I_HOLD_LOCK());
#endif
pthread_mutex_lock(&(key->lock));
entry=key->hash[hash_val].p;
while (entry!=NULL&&!THREAD_EQUAL(entry->thread,t)){
prev=entry;
entry=entry->next;
}
if (entry!=NULL){
entry->qtid=INVALID_QTID;
if (NULL==prev){
key->hash[hash_val].p=entry->next;
GC_dirty(key->hash+hash_val);
} else {
prev->next=entry->next;
GC_dirty(prev);
}
}
#ifdef LINT2
GC_noop1((word)entry);
#endif
if (pthread_mutex_unlock(&key->lock)!=0)
ABORT("pthread_mutex_unlock failed (remove_specific after fork)");
}
GC_INNER void*GC_slow_getspecific(tsd*key,word qtid,
tse*volatile*cache_ptr)
{
pthread_t self=pthread_self();
tse*entry=key->hash[HASH(self)].p;
GC_ASSERT(qtid!=INVALID_QTID);
while (entry!=NULL&&!THREAD_EQUAL(entry->thread,self)){
entry=entry->next;
}
if (entry==NULL)return NULL;
entry->qtid=(AO_t)qtid;
*cache_ptr=entry;
return TS_REVEAL_PTR(entry->value);
}
#ifdef GC_ASSERTIONS
void GC_check_tsd_marks(tsd*key)
{
int i;
tse*p;
if (!GC_is_marked(GC_base(key))){
ABORT("Unmarked thread-specific-data table");
}
for (i=0;i < TS_HASH_SIZE;++i){
for (p=key->hash[i].p;p!=0;p=p->next){
if (!GC_is_marked(GC_base(p))){
ABORT_ARG1("Unmarked thread-specific-data entry",
" at %p",(void*)p);
}
}
}
for (i=0;i < TS_CACHE_SIZE;++i){
p=key->cache[i];
if (p!=&invalid_tse&&!GC_is_marked(GC_base(p))){
ABORT_ARG1("Unmarked cached thread-specific-data entry",
" at %p",(void*)p);
}
}
}
#endif
#endif
#if defined(GC_WIN32_THREADS)
#ifdef THREAD_LOCAL_ALLOC
#endif
#if!defined(USE_PTHREAD_LOCKS)
GC_INNER CRITICAL_SECTION GC_allocate_ml;
#ifdef GC_ASSERTIONS
GC_INNER DWORD GC_lock_holder=NO_THREAD;
#endif
#else
GC_INNER pthread_mutex_t GC_allocate_ml=PTHREAD_MUTEX_INITIALIZER;
#ifdef GC_ASSERTIONS
GC_INNER unsigned long GC_lock_holder=NO_THREAD;
#endif
#endif
#undef CreateThread
#undef ExitThread
#undef _beginthreadex
#undef _endthreadex
#ifdef GC_PTHREADS
#include <errno.h>
#undef pthread_create
#undef pthread_join
#undef pthread_detach
#ifndef GC_NO_PTHREAD_SIGMASK
#undef pthread_sigmask
#endif
STATIC void*GC_pthread_start(void*arg);
STATIC void GC_thread_exit_proc(void*arg);
#include <pthread.h>
#ifdef CAN_CALL_ATFORK
#include <unistd.h>
#endif
#elif!defined(MSWINCE)
#include <process.h>
#include <errno.h>
#endif
static ptr_t copy_ptr_regs(word*regs,const CONTEXT*pcontext);
#if defined(I386)
#ifdef WOW64_THREAD_CONTEXT_WORKAROUND
#define PUSHED_REGS_COUNT 9
#else
#define PUSHED_REGS_COUNT 7
#endif
#elif defined(X86_64)||defined(SHx)
#define PUSHED_REGS_COUNT 15
#elif defined(ARM32)
#define PUSHED_REGS_COUNT 13
#elif defined(AARCH64)
#define PUSHED_REGS_COUNT 30
#elif defined(MIPS)||defined(ALPHA)
#define PUSHED_REGS_COUNT 28
#elif defined(PPC)
#define PUSHED_REGS_COUNT 29
#endif
#if (defined(GC_DLL)||defined(GC_INSIDE_DLL))&&!defined(NO_CRT)&&!defined(GC_NO_THREADS_DISCOVERY)&&!defined(MSWINCE)&&!defined(THREAD_LOCAL_ALLOC)&&!defined(GC_PTHREADS)
#ifdef GC_DISCOVER_TASK_THREADS
#define GC_win32_dll_threads TRUE
#else
STATIC GC_bool GC_win32_dll_threads=FALSE;
#endif
#else
#ifndef GC_NO_THREADS_DISCOVERY
#define GC_NO_THREADS_DISCOVERY
#endif
#define GC_win32_dll_threads FALSE
#undef MAX_THREADS
#define MAX_THREADS 1
#endif
typedef LONG*IE_t;
STATIC GC_bool GC_thr_initialized=FALSE;
#ifndef GC_ALWAYS_MULTITHREADED
GC_INNER GC_bool GC_need_to_lock=FALSE;
#endif
static GC_bool parallel_initialized=FALSE;
GC_API void GC_CALL GC_use_threads_discovery(void)
{
#ifdef GC_NO_THREADS_DISCOVERY
ABORT("GC DllMain-based thread registration unsupported");
#else
GC_ASSERT(!parallel_initialized);
#ifndef GC_DISCOVER_TASK_THREADS
GC_win32_dll_threads=TRUE;
#endif
GC_init_parallel();
#endif
}
#define ADDR_LIMIT ((ptr_t)GC_WORD_MAX)
struct GC_Thread_Rep {
union {
#ifndef GC_NO_THREADS_DISCOVERY
volatile AO_t in_use;
LONG long_in_use;
#endif
struct GC_Thread_Rep*next;
} tm;
DWORD id;
#ifdef MSWINCE
#define THREAD_HANDLE(t)(HANDLE)(word)(t)->id
#else
HANDLE handle;
#define THREAD_HANDLE(t)(t)->handle
#endif
ptr_t stack_base;
ptr_t last_stack_min;
#ifdef IA64
ptr_t backing_store_end;
ptr_t backing_store_ptr;
#elif defined(I386)
ptr_t initial_stack_base;
#endif
ptr_t thread_blocked_sp;
struct GC_traced_stack_sect_s*traced_stack_sect;
unsigned short finalizer_skipped;
unsigned char finalizer_nested;
unsigned char suspended;
#ifdef GC_PTHREADS
unsigned char flags;
#define FINISHED 1
#define DETACHED 2
#define KNOWN_FINISHED(t)(((t)->flags)&FINISHED)
pthread_t pthread_id;
void*status;
#else
#define KNOWN_FINISHED(t)0
#endif
#ifdef THREAD_LOCAL_ALLOC
struct thread_local_freelists tlfs;
#endif
#ifdef RETRY_GET_THREAD_CONTEXT
ptr_t context_sp;
word context_regs[PUSHED_REGS_COUNT];
#endif
};
typedef struct GC_Thread_Rep*GC_thread;
typedef volatile struct GC_Thread_Rep*GC_vthread;
#ifndef GC_NO_THREADS_DISCOVERY
STATIC DWORD GC_main_thread=0;
STATIC volatile AO_t GC_attached_thread=FALSE;
STATIC volatile GC_bool GC_please_stop=FALSE;
#elif defined(GC_ASSERTIONS)
STATIC GC_bool GC_please_stop=FALSE;
#endif
#if defined(WRAP_MARK_SOME)&&!defined(GC_PTHREADS)
GC_INNER GC_bool GC_started_thread_while_stopped(void)
{
#ifndef GC_NO_THREADS_DISCOVERY
if (GC_win32_dll_threads){
#ifdef AO_HAVE_compare_and_swap_release
if (AO_compare_and_swap_release(&GC_attached_thread,TRUE,
FALSE))
return TRUE;
#else
AO_nop_full();
if (AO_load(&GC_attached_thread)){
AO_store(&GC_attached_thread,FALSE);
return TRUE;
}
#endif
}
#endif
return FALSE;
}
#endif
#ifndef MAX_THREADS
#define MAX_THREADS 512
#endif
volatile struct GC_Thread_Rep dll_thread_table[MAX_THREADS];
STATIC volatile LONG GC_max_thread_index=0;
#ifndef THREAD_TABLE_SZ
#define THREAD_TABLE_SZ 256
#endif
#define THREAD_TABLE_INDEX(id)(int)((((id)>>8)^(id))% THREAD_TABLE_SZ)
STATIC GC_thread GC_threads[THREAD_TABLE_SZ];
static struct GC_Thread_Rep first_thread;
static GC_bool first_thread_used=FALSE;
STATIC GC_thread GC_new_thread(DWORD id)
{
int hv=THREAD_TABLE_INDEX(id);
GC_thread result;
#ifdef DEBUG_THREADS
GC_log_printf("Creating thread 0x%lx\n",(long)id);
if (GC_threads[hv]!=NULL)
GC_log_printf("Hash collision at GC_threads[%d]\n",hv);
#endif
GC_ASSERT(I_HOLD_LOCK());
if (!EXPECT(first_thread_used,TRUE)){
result=&first_thread;
first_thread_used=TRUE;
GC_ASSERT(NULL==GC_threads[hv]);
} else {
GC_ASSERT(!GC_win32_dll_threads);
result=(struct GC_Thread_Rep*)
GC_INTERNAL_MALLOC(sizeof(struct GC_Thread_Rep),NORMAL);
if (result==0)return(0);
}
result->tm.next=GC_threads[hv];
GC_threads[hv]=result;
#ifdef GC_PTHREADS
GC_ASSERT(result->flags==0);
#endif
GC_ASSERT(result->thread_blocked_sp==NULL);
if (EXPECT(result!=&first_thread,TRUE))
GC_dirty(result);
return(result);
}
GC_INNER GC_bool GC_in_thread_creation=FALSE;
GC_INLINE void GC_record_stack_base(GC_vthread me,
const struct GC_stack_base*sb)
{
me->stack_base=(ptr_t)sb->mem_base;
#ifdef IA64
me->backing_store_end=(ptr_t)sb->reg_base;
#elif defined(I386)
me->initial_stack_base=(ptr_t)sb->mem_base;
#endif
if (me->stack_base==NULL)
ABORT("Bad stack base in GC_register_my_thread");
}
STATIC GC_thread GC_register_my_thread_inner(const struct GC_stack_base*sb,
DWORD thread_id)
{
GC_vthread me;
#if defined(MPROTECT_VDB)&&!defined(CYGWIN32)
if (GC_auto_incremental
#ifdef GWW_VDB
&&!GC_gww_dirty_init()
#endif
)
GC_set_write_fault_handler();
#endif
#ifndef GC_NO_THREADS_DISCOVERY
if (GC_win32_dll_threads){
int i;
for (i=0;
InterlockedExchange(&dll_thread_table[i].tm.long_in_use,1)!=0;
i++){
if (i==MAX_THREADS - 1)
ABORT("Too many threads");
}
while (i > GC_max_thread_index){
InterlockedIncrement((IE_t)&GC_max_thread_index);
}
if (GC_max_thread_index>=MAX_THREADS){
GC_max_thread_index=MAX_THREADS - 1;
}
me=dll_thread_table+i;
} else
#endif
{
GC_ASSERT(I_HOLD_LOCK());
GC_in_thread_creation=TRUE;
me=GC_new_thread(thread_id);
GC_in_thread_creation=FALSE;
if (me==0)
ABORT("Failed to allocate memory for thread registering");
}
#ifdef GC_PTHREADS
me->pthread_id=pthread_self();
#endif
#ifndef MSWINCE
if (!DuplicateHandle(GetCurrentProcess(),GetCurrentThread(),
GetCurrentProcess(),
(HANDLE*)&(me->handle),
0,FALSE,
DUPLICATE_SAME_ACCESS)){
ABORT_ARG1("DuplicateHandle failed",
":errcode=0x%X",(unsigned)GetLastError());
}
#endif
me->last_stack_min=ADDR_LIMIT;
GC_record_stack_base(me,sb);
me->id=thread_id;
#if defined(THREAD_LOCAL_ALLOC)
GC_init_thread_local((GC_tlfs)(&(me->tlfs)));
#endif
#ifndef GC_NO_THREADS_DISCOVERY
if (GC_win32_dll_threads){
if (GC_please_stop){
AO_store(&GC_attached_thread,TRUE);
AO_nop_full();
}
} else
#endif
{
GC_ASSERT(!GC_please_stop);
}
return (GC_thread)(me);
}
GC_INLINE LONG GC_get_max_thread_index(void)
{
LONG my_max=GC_max_thread_index;
if (my_max>=MAX_THREADS)return MAX_THREADS - 1;
return my_max;
}
STATIC GC_thread GC_lookup_thread_inner(DWORD thread_id)
{
#ifndef GC_NO_THREADS_DISCOVERY
if (GC_win32_dll_threads){
int i;
LONG my_max=GC_get_max_thread_index();
for (i=0;i<=my_max&&
(!AO_load_acquire(&dll_thread_table[i].tm.in_use)
||dll_thread_table[i].id!=thread_id);
i++){
}
return i<=my_max?(GC_thread)(dll_thread_table+i):NULL;
} else
#endif
{
GC_thread p=GC_threads[THREAD_TABLE_INDEX(thread_id)];
GC_ASSERT(I_HOLD_LOCK());
while (p!=0&&p->id!=thread_id)p=p->tm.next;
return(p);
}
}
#ifdef LINT2
#define CHECK_LOOKUP_MY_THREAD(me)if (!(me))ABORT("GC_lookup_thread_inner(GetCurrentThreadId)failed")
#else
#define CHECK_LOOKUP_MY_THREAD(me)
#endif
GC_INNER void GC_reset_finalizer_nested(void)
{
GC_thread me=GC_lookup_thread_inner(GetCurrentThreadId());
CHECK_LOOKUP_MY_THREAD(me);
me->finalizer_nested=0;
}
GC_INNER unsigned char*GC_check_finalizer_nested(void)
{
GC_thread me=GC_lookup_thread_inner(GetCurrentThreadId());
unsigned nesting_level;
CHECK_LOOKUP_MY_THREAD(me);
nesting_level=me->finalizer_nested;
if (nesting_level){
if (++me->finalizer_skipped < (1U<<nesting_level))return NULL;
me->finalizer_skipped=0;
}
me->finalizer_nested=(unsigned char)(nesting_level+1);
return&me->finalizer_nested;
}
#if defined(GC_ASSERTIONS)&&defined(THREAD_LOCAL_ALLOC)
GC_bool GC_is_thread_tsd_valid(void*tsd)
{
GC_thread me;
DCL_LOCK_STATE;
LOCK();
me=GC_lookup_thread_inner(GetCurrentThreadId());
UNLOCK();
return (word)tsd>=(word)(&me->tlfs)
&&(word)tsd < (word)(&me->tlfs)+sizeof(me->tlfs);
}
#endif
GC_API int GC_CALL GC_thread_is_registered(void)
{
DWORD thread_id=GetCurrentThreadId();
GC_thread me;
DCL_LOCK_STATE;
LOCK();
me=GC_lookup_thread_inner(thread_id);
UNLOCK();
return me!=NULL;
}
GC_API void GC_CALL GC_register_altstack(void*stack GC_ATTR_UNUSED,
GC_word stack_size GC_ATTR_UNUSED,
void*altstack GC_ATTR_UNUSED,
GC_word altstack_size GC_ATTR_UNUSED)
{
}
#if defined(MPROTECT_VDB)
#define UNPROTECT_THREAD(t)if (!GC_win32_dll_threads&&GC_auto_incremental&&t!=&first_thread){ GC_ASSERT(SMALL_OBJ(GC_size(t)));GC_remove_protection(HBLKPTR(t),1,FALSE);} else (void)0
#else
#define UNPROTECT_THREAD(t)(void)0
#endif
#ifdef CYGWIN32
#define GC_PTHREAD_PTRVAL(pthread_id)pthread_id
#elif defined(GC_WIN32_PTHREADS)||defined(GC_PTHREADS_PARAMARK)
#include <pthread.h>
#if defined(__WINPTHREADS_VERSION_MAJOR)
#define GC_PTHREAD_PTRVAL(pthread_id)pthread_id
#else
#define GC_PTHREAD_PTRVAL(pthread_id)pthread_id.p
#endif
#endif
STATIC void GC_delete_gc_thread_no_free(GC_vthread t)
{
#ifndef MSWINCE
CloseHandle(t->handle);
#endif
#ifndef GC_NO_THREADS_DISCOVERY
if (GC_win32_dll_threads){
t->stack_base=0;
t->id=0;
t->suspended=FALSE;
#ifdef RETRY_GET_THREAD_CONTEXT
t->context_sp=NULL;
#endif
AO_store_release(&t->tm.in_use,FALSE);
} else
#endif
{
DWORD id=((GC_thread)t)->id;
int hv=THREAD_TABLE_INDEX(id);
GC_thread p=GC_threads[hv];
GC_thread prev=NULL;
GC_ASSERT(I_HOLD_LOCK());
while (p!=(GC_thread)t){
prev=p;
p=p->tm.next;
}
if (prev==0){
GC_threads[hv]=p->tm.next;
} else {
GC_ASSERT(prev!=&first_thread);
prev->tm.next=p->tm.next;
GC_dirty(prev);
}
}
}
STATIC void GC_delete_thread(DWORD id)
{
if (GC_win32_dll_threads){
GC_vthread t=GC_lookup_thread_inner(id);
if (0==t){
WARN("Removing nonexistent thread,id=%" WARN_PRIdPTR "\n",id);
} else {
GC_delete_gc_thread_no_free(t);
}
} else {
int hv=THREAD_TABLE_INDEX(id);
GC_thread p=GC_threads[hv];
GC_thread prev=NULL;
GC_ASSERT(I_HOLD_LOCK());
while (p->id!=id){
prev=p;
p=p->tm.next;
}
#ifndef MSWINCE
CloseHandle(p->handle);
#endif
if (prev==0){
GC_threads[hv]=p->tm.next;
} else {
GC_ASSERT(prev!=&first_thread);
prev->tm.next=p->tm.next;
GC_dirty(prev);
}
if (EXPECT(p!=&first_thread,TRUE)){
GC_INTERNAL_FREE(p);
}
}
}
GC_API void GC_CALL GC_allow_register_threads(void)
{
GC_ASSERT(GC_lookup_thread_inner(GetCurrentThreadId())!=0);
#if!defined(GC_ALWAYS_MULTITHREADED)&&!defined(PARALLEL_MARK)&&!defined(GC_NO_THREADS_DISCOVERY)
parallel_initialized=TRUE;
#endif
set_need_to_lock();
}
GC_API int GC_CALL GC_register_my_thread(const struct GC_stack_base*sb)
{
GC_thread me;
DWORD thread_id=GetCurrentThreadId();
DCL_LOCK_STATE;
if (GC_need_to_lock==FALSE)
ABORT("Threads explicit registering is not previously enabled");
LOCK();
me=GC_lookup_thread_inner(thread_id);
if (me==0){
#ifdef GC_PTHREADS
me=GC_register_my_thread_inner(sb,thread_id);
#if defined(CPPCHECK)
GC_noop1(me->flags);
#endif
me->flags|=DETACHED;
#else
GC_register_my_thread_inner(sb,thread_id);
#endif
UNLOCK();
return GC_SUCCESS;
} else
#ifdef GC_PTHREADS
if ((me->flags&FINISHED)!=0){
GC_record_stack_base(me,sb);
me->flags&=~FINISHED;
#ifdef THREAD_LOCAL_ALLOC
GC_init_thread_local((GC_tlfs)(&me->tlfs));
#endif
UNLOCK();
return GC_SUCCESS;
} else
#endif
{
UNLOCK();
return GC_DUPLICATE;
}
}
STATIC void GC_wait_for_gc_completion(GC_bool wait_for_all)
{
GC_ASSERT(I_HOLD_LOCK());
if (GC_incremental&&GC_collection_in_progress()){
word old_gc_no=GC_gc_no;
do {
ENTER_GC();
GC_in_thread_creation=TRUE;
GC_collect_a_little_inner(1);
GC_in_thread_creation=FALSE;
EXIT_GC();
UNLOCK();
Sleep(0);
LOCK();
} while (GC_incremental&&GC_collection_in_progress()
&&(wait_for_all||old_gc_no==GC_gc_no));
}
}
GC_API int GC_CALL GC_unregister_my_thread(void)
{
DCL_LOCK_STATE;
#ifdef DEBUG_THREADS
GC_log_printf("Unregistering thread 0x%lx\n",(long)GetCurrentThreadId());
#endif
if (GC_win32_dll_threads){
#if defined(THREAD_LOCAL_ALLOC)
GC_ASSERT(FALSE);
#else
GC_delete_thread(GetCurrentThreadId());
#endif
} else {
#if defined(THREAD_LOCAL_ALLOC)||defined(GC_PTHREADS)
GC_thread me;
#endif
DWORD thread_id=GetCurrentThreadId();
LOCK();
GC_wait_for_gc_completion(FALSE);
#if defined(THREAD_LOCAL_ALLOC)||defined(GC_PTHREADS)
me=GC_lookup_thread_inner(thread_id);
CHECK_LOOKUP_MY_THREAD(me);
GC_ASSERT(!KNOWN_FINISHED(me));
#endif
#if defined(THREAD_LOCAL_ALLOC)
GC_ASSERT(GC_getspecific(GC_thread_key)==&me->tlfs);
GC_destroy_thread_local(&(me->tlfs));
#endif
#ifdef GC_PTHREADS
if ((me->flags&DETACHED)==0){
me->flags|=FINISHED;
} else
#endif
{
GC_delete_thread(thread_id);
}
#if defined(THREAD_LOCAL_ALLOC)
GC_remove_specific(GC_thread_key);
#endif
UNLOCK();
}
return GC_SUCCESS;
}
GC_INNER void GC_do_blocking_inner(ptr_t data,void*context GC_ATTR_UNUSED)
{
struct blocking_data*d=(struct blocking_data*)data;
DWORD thread_id=GetCurrentThreadId();
GC_thread me;
#ifdef IA64
ptr_t stack_ptr=GC_save_regs_in_stack();
#endif
DCL_LOCK_STATE;
LOCK();
me=GC_lookup_thread_inner(thread_id);
CHECK_LOOKUP_MY_THREAD(me);
GC_ASSERT(me->thread_blocked_sp==NULL);
#ifdef IA64
me->backing_store_ptr=stack_ptr;
#endif
me->thread_blocked_sp=(ptr_t)&d;
UNLOCK();
d->client_data=(d->fn)(d->client_data);
LOCK();
#if defined(CPPCHECK)
GC_noop1((word)me->thread_blocked_sp);
#endif
me->thread_blocked_sp=NULL;
UNLOCK();
}
GC_API void*GC_CALL GC_call_with_gc_active(GC_fn_type fn,
void*client_data)
{
struct GC_traced_stack_sect_s stacksect;
DWORD thread_id=GetCurrentThreadId();
GC_thread me;
DCL_LOCK_STATE;
LOCK();
me=GC_lookup_thread_inner(thread_id);
CHECK_LOOKUP_MY_THREAD(me);
GC_ASSERT(me->stack_base!=NULL);
if ((word)me->stack_base < (word)(&stacksect)){
me->stack_base=(ptr_t)(&stacksect);
#if defined(I386)
me->initial_stack_base=me->stack_base;
#endif
}
if (me->thread_blocked_sp==NULL){
UNLOCK();
client_data=fn(client_data);
GC_noop1(COVERT_DATAFLOW(&stacksect));
return client_data;
}
stacksect.saved_stack_ptr=me->thread_blocked_sp;
#ifdef IA64
stacksect.backing_store_end=GC_save_regs_in_stack();
stacksect.saved_backing_store_ptr=me->backing_store_ptr;
#endif
stacksect.prev=me->traced_stack_sect;
me->thread_blocked_sp=NULL;
me->traced_stack_sect=&stacksect;
UNLOCK();
client_data=fn(client_data);
GC_ASSERT(me->thread_blocked_sp==NULL);
GC_ASSERT(me->traced_stack_sect==&stacksect);
LOCK();
#if defined(CPPCHECK)
GC_noop1((word)me->traced_stack_sect);
#endif
me->traced_stack_sect=stacksect.prev;
#ifdef IA64
me->backing_store_ptr=stacksect.saved_backing_store_ptr;
#endif
me->thread_blocked_sp=stacksect.saved_stack_ptr;
UNLOCK();
return client_data;
}
GC_API void GC_CALL GC_set_stackbottom(void*gc_thread_handle,
const struct GC_stack_base*sb)
{
GC_thread t=(GC_thread)gc_thread_handle;
GC_ASSERT(sb->mem_base!=NULL);
if (!EXPECT(GC_is_initialized,TRUE)){
GC_ASSERT(NULL==t);
GC_stackbottom=(char*)sb->mem_base;
#ifdef IA64
GC_register_stackbottom=(ptr_t)sb->reg_base;
#endif
return;
}
GC_ASSERT(I_HOLD_LOCK());
if (NULL==t){
t=GC_lookup_thread_inner(GetCurrentThreadId());
CHECK_LOOKUP_MY_THREAD(t);
}
GC_ASSERT(!KNOWN_FINISHED(t));
GC_ASSERT(NULL==t->thread_blocked_sp
&&NULL==t->traced_stack_sect);
t->stack_base=(ptr_t)sb->mem_base;
t->last_stack_min=ADDR_LIMIT;
#ifdef IA64
t->backing_store_end=(ptr_t)sb->reg_base;
#endif
}
GC_API void*GC_CALL GC_get_my_stackbottom(struct GC_stack_base*sb)
{
DWORD thread_id=GetCurrentThreadId();
GC_thread me;
DCL_LOCK_STATE;
LOCK();
me=GC_lookup_thread_inner(thread_id);
CHECK_LOOKUP_MY_THREAD(me);
sb->mem_base=me->stack_base;
#ifdef IA64
sb->reg_base=me->backing_store_end;
#endif
UNLOCK();
return (void*)me;
}
#ifdef GC_PTHREADS
#define PTHREAD_MAP_SIZE 512
DWORD GC_pthread_map_cache[PTHREAD_MAP_SIZE]={0};
#define PTHREAD_MAP_INDEX(pthread_id)((NUMERIC_THREAD_ID(pthread_id)>>5)% PTHREAD_MAP_SIZE)
#define SET_PTHREAD_MAP_CACHE(pthread_id,win32_id)(void)(GC_pthread_map_cache[PTHREAD_MAP_INDEX(pthread_id)]=(win32_id))
#define GET_PTHREAD_MAP_CACHE(pthread_id)GC_pthread_map_cache[PTHREAD_MAP_INDEX(pthread_id)]
STATIC GC_thread GC_lookup_pthread(pthread_t id)
{
#ifndef GC_NO_THREADS_DISCOVERY
if (GC_win32_dll_threads){
int i;
LONG my_max=GC_get_max_thread_index();
for (i=0;i<=my_max&&
(!AO_load_acquire(&dll_thread_table[i].tm.in_use)
||THREAD_EQUAL(dll_thread_table[i].pthread_id,id));
i++){
}
return i<=my_max?(GC_thread)(dll_thread_table+i):NULL;
} else
#endif
{
DWORD win32_id=GET_PTHREAD_MAP_CACHE(id);
int hv_guess=THREAD_TABLE_INDEX(win32_id);
int hv;
GC_thread p;
DCL_LOCK_STATE;
LOCK();
for (p=GC_threads[hv_guess];0!=p;p=p->tm.next){
if (THREAD_EQUAL(p->pthread_id,id))
goto foundit;
}
for (hv=0;hv < THREAD_TABLE_SZ;++hv){
for (p=GC_threads[hv];0!=p;p=p->tm.next){
if (THREAD_EQUAL(p->pthread_id,id))
goto foundit;
}
}
p=0;
foundit:
UNLOCK();
return p;
}
}
#endif
#ifdef CAN_HANDLE_FORK
STATIC void GC_remove_all_threads_but_me(void)
{
int hv;
GC_thread me=NULL;
DWORD thread_id;
pthread_t pthread_id=pthread_self();
GC_ASSERT(!GC_win32_dll_threads);
for (hv=0;hv < THREAD_TABLE_SZ;++hv){
GC_thread p,next;
for (p=GC_threads[hv];0!=p;p=next){
next=p->tm.next;
if (THREAD_EQUAL(p->pthread_id,pthread_id)
&&me==NULL){
me=p;
p->tm.next=0;
} else {
#ifdef THREAD_LOCAL_ALLOC
if ((p->flags&FINISHED)==0){
GC_remove_specific_after_fork(GC_thread_key,p->pthread_id);
}
#endif
if (&first_thread!=p)
GC_INTERNAL_FREE(p);
}
}
GC_threads[hv]=NULL;
}
GC_ASSERT(me!=NULL);
thread_id=GetCurrentThreadId();
GC_threads[THREAD_TABLE_INDEX(thread_id)]=me;
me->id=thread_id;
#ifndef MSWINCE
if (!DuplicateHandle(GetCurrentProcess(),GetCurrentThread(),
GetCurrentProcess(),(HANDLE*)&me->handle,
0,FALSE,
DUPLICATE_SAME_ACCESS))
ABORT("DuplicateHandle failed");
#endif
#if defined(THREAD_LOCAL_ALLOC)&&!defined(USE_CUSTOM_SPECIFIC)
if (GC_setspecific(GC_thread_key,&me->tlfs)!=0)
ABORT("GC_setspecific failed (in child)");
#endif
}
static void fork_prepare_proc(void)
{
LOCK();
#ifdef PARALLEL_MARK
if (GC_parallel)
GC_wait_for_reclaim();
#endif
GC_wait_for_gc_completion(TRUE);
#ifdef PARALLEL_MARK
if (GC_parallel)
GC_acquire_mark_lock();
#endif
}
static void fork_parent_proc(void)
{
#ifdef PARALLEL_MARK
if (GC_parallel)
GC_release_mark_lock();
#endif
UNLOCK();
}
static void fork_child_proc(void)
{
#ifdef PARALLEL_MARK
if (GC_parallel){
GC_release_mark_lock();
GC_parallel=FALSE;
}
#endif
GC_remove_all_threads_but_me();
UNLOCK();
}
GC_API void GC_CALL GC_atfork_prepare(void)
{
if (!EXPECT(GC_is_initialized,TRUE))GC_init();
if (GC_handle_fork<=0)
fork_prepare_proc();
}
GC_API void GC_CALL GC_atfork_parent(void)
{
if (GC_handle_fork<=0)
fork_parent_proc();
}
GC_API void GC_CALL GC_atfork_child(void)
{
if (GC_handle_fork<=0)
fork_child_proc();
}
#endif
void GC_push_thread_structures(void)
{
GC_ASSERT(I_HOLD_LOCK());
#ifndef GC_NO_THREADS_DISCOVERY
if (GC_win32_dll_threads){
} else
#endif
{
GC_PUSH_ALL_SYM(GC_threads);
}
#if defined(THREAD_LOCAL_ALLOC)
GC_PUSH_ALL_SYM(GC_thread_key);
#endif
}
#ifdef WOW64_THREAD_CONTEXT_WORKAROUND
#ifndef CONTEXT_EXCEPTION_ACTIVE
#define CONTEXT_EXCEPTION_ACTIVE 0x08000000
#define CONTEXT_EXCEPTION_REQUEST 0x40000000
#define CONTEXT_EXCEPTION_REPORTING 0x80000000
#endif
static BOOL isWow64;
#define GET_THREAD_CONTEXT_FLAGS (isWow64?CONTEXT_INTEGER|CONTEXT_CONTROL|CONTEXT_EXCEPTION_REQUEST|CONTEXT_SEGMENTS:CONTEXT_INTEGER|CONTEXT_CONTROL)
#else
#define GET_THREAD_CONTEXT_FLAGS (CONTEXT_INTEGER|CONTEXT_CONTROL)
#endif
STATIC void GC_suspend(GC_thread t)
{
#ifndef MSWINCE
DWORD exitCode;
#endif
#ifdef RETRY_GET_THREAD_CONTEXT
int retry_cnt=0;
#define MAX_SUSPEND_THREAD_RETRIES (1000*1000)
#endif
#ifdef DEBUG_THREADS
GC_log_printf("Suspending 0x%x\n",(int)t->id);
#endif
UNPROTECT_THREAD(t);
GC_acquire_dirty_lock();
#ifdef MSWINCE
while (SuspendThread(THREAD_HANDLE(t))==(DWORD)-1){
GC_release_dirty_lock();
Sleep(10);
GC_acquire_dirty_lock();
}
#elif defined(RETRY_GET_THREAD_CONTEXT)
for (;;){
if (GetExitCodeThread(t->handle,&exitCode)
&&exitCode!=STILL_ACTIVE){
GC_release_dirty_lock();
#ifdef GC_PTHREADS
t->stack_base=0;
#else
GC_ASSERT(GC_win32_dll_threads);
GC_delete_gc_thread_no_free(t);
#endif
return;
}
if (SuspendThread(t->handle)!=(DWORD)-1){
CONTEXT context;
context.ContextFlags=GET_THREAD_CONTEXT_FLAGS;
if (GetThreadContext(t->handle,&context)){
t->context_sp=copy_ptr_regs(t->context_regs,&context);
break;
}
if (ResumeThread(t->handle)==(DWORD)-1)
ABORT("ResumeThread failed in suspend loop");
}
if (retry_cnt > 1){
GC_release_dirty_lock();
Sleep(0);
GC_acquire_dirty_lock();
}
if (++retry_cnt>=MAX_SUSPEND_THREAD_RETRIES)
ABORT("SuspendThread loop failed");
}
#else
if (GetExitCodeThread(t->handle,&exitCode)
&&exitCode!=STILL_ACTIVE){
GC_release_dirty_lock();
#ifdef GC_PTHREADS
t->stack_base=0;
#else
GC_ASSERT(GC_win32_dll_threads);
GC_delete_gc_thread_no_free(t);
#endif
return;
}
if (SuspendThread(t->handle)==(DWORD)-1)
ABORT("SuspendThread failed");
#endif
t->suspended=(unsigned char)TRUE;
GC_release_dirty_lock();
if (GC_on_thread_event)
GC_on_thread_event(GC_EVENT_THREAD_SUSPENDED,THREAD_HANDLE(t));
}
#if defined(GC_ASSERTIONS)&&((defined(MSWIN32)&&!defined(CONSOLE_LOG))||defined(MSWINCE))
GC_INNER GC_bool GC_write_disabled=FALSE;
#endif
GC_INNER void GC_stop_world(void)
{
DWORD thread_id=GetCurrentThreadId();
if (!GC_thr_initialized)
ABORT("GC_stop_world()called before GC_thr_init()");
GC_ASSERT(I_HOLD_LOCK());
#ifdef PARALLEL_MARK
if (GC_parallel){
GC_acquire_mark_lock();
GC_ASSERT(GC_fl_builder_count==0);
}
#endif
#if!defined(GC_NO_THREADS_DISCOVERY)||defined(GC_ASSERTIONS)
GC_please_stop=TRUE;
#endif
#if (defined(MSWIN32)&&!defined(CONSOLE_LOG))||defined(MSWINCE)
GC_ASSERT(!GC_write_disabled);
EnterCriticalSection(&GC_write_cs);
#ifdef GC_ASSERTIONS
GC_write_disabled=TRUE;
#endif
#endif
#ifndef GC_NO_THREADS_DISCOVERY
if (GC_win32_dll_threads){
int i;
int my_max;
AO_store(&GC_attached_thread,FALSE);
my_max=(int)GC_get_max_thread_index();
for (i=0;i<=my_max;i++){
GC_vthread t=dll_thread_table+i;
if (t->stack_base!=0&&t->thread_blocked_sp==NULL
&&t->id!=thread_id){
GC_suspend((GC_thread)t);
}
}
} else
#endif
{
GC_thread t;
int i;
for (i=0;i < THREAD_TABLE_SZ;i++){
for (t=GC_threads[i];t!=0;t=t->tm.next){
if (t->stack_base!=0&&t->thread_blocked_sp==NULL
&&!KNOWN_FINISHED(t)&&t->id!=thread_id){
GC_suspend(t);
}
}
}
}
#if (defined(MSWIN32)&&!defined(CONSOLE_LOG))||defined(MSWINCE)
#ifdef GC_ASSERTIONS
GC_write_disabled=FALSE;
#endif
LeaveCriticalSection(&GC_write_cs);
#endif
#ifdef PARALLEL_MARK
if (GC_parallel)
GC_release_mark_lock();
#endif
}
GC_INNER void GC_start_world(void)
{
#ifdef GC_ASSERTIONS
DWORD thread_id=GetCurrentThreadId();
#endif
GC_ASSERT(I_HOLD_LOCK());
if (GC_win32_dll_threads){
LONG my_max=GC_get_max_thread_index();
int i;
for (i=0;i<=my_max;i++){
GC_thread t=(GC_thread)(dll_thread_table+i);
if (t->suspended){
#ifdef DEBUG_THREADS
GC_log_printf("Resuming 0x%x\n",(int)t->id);
#endif
GC_ASSERT(t->stack_base!=0&&t->id!=thread_id);
if (ResumeThread(THREAD_HANDLE(t))==(DWORD)-1)
ABORT("ResumeThread failed");
t->suspended=FALSE;
if (GC_on_thread_event)
GC_on_thread_event(GC_EVENT_THREAD_UNSUSPENDED,THREAD_HANDLE(t));
}
}
} else {
GC_thread t;
int i;
for (i=0;i < THREAD_TABLE_SZ;i++){
for (t=GC_threads[i];t!=0;t=t->tm.next){
if (t->suspended){
#ifdef DEBUG_THREADS
GC_log_printf("Resuming 0x%x\n",(int)t->id);
#endif
GC_ASSERT(t->stack_base!=0&&t->id!=thread_id);
if (ResumeThread(THREAD_HANDLE(t))==(DWORD)-1)
ABORT("ResumeThread failed");
UNPROTECT_THREAD(t);
t->suspended=FALSE;
if (GC_on_thread_event)
GC_on_thread_event(GC_EVENT_THREAD_UNSUSPENDED,THREAD_HANDLE(t));
} else {
#ifdef DEBUG_THREADS
GC_log_printf("Not resuming thread 0x%x as it is not suspended\n",
(int)t->id);
#endif
}
}
}
}
#if!defined(GC_NO_THREADS_DISCOVERY)||defined(GC_ASSERTIONS)
GC_please_stop=FALSE;
#endif
}
#ifdef MSWINCE
#define GC_wince_evaluate_stack_min(s)(ptr_t)(((word)(s)- 1)&~(word)0xFFFF)
#elif defined(GC_ASSERTIONS)
#define GC_dont_query_stack_min FALSE
#endif
static ptr_t last_address=0;
static MEMORY_BASIC_INFORMATION last_info;
STATIC ptr_t GC_get_stack_min(ptr_t s)
{
ptr_t bottom;
GC_ASSERT(I_HOLD_LOCK());
if (s!=last_address){
VirtualQuery(s,&last_info,sizeof(last_info));
last_address=s;
}
do {
bottom=(ptr_t)last_info.BaseAddress;
VirtualQuery(bottom - 1,&last_info,sizeof(last_info));
last_address=bottom - 1;
} while ((last_info.Protect&PAGE_READWRITE)
&&!(last_info.Protect&PAGE_GUARD));
return(bottom);
}
static GC_bool may_be_in_stack(ptr_t s)
{
GC_ASSERT(I_HOLD_LOCK());
if (s!=last_address){
VirtualQuery(s,&last_info,sizeof(last_info));
last_address=s;
}
return (last_info.Protect&PAGE_READWRITE)
&&!(last_info.Protect&PAGE_GUARD);
}
static ptr_t copy_ptr_regs(word*regs,const CONTEXT*pcontext){
ptr_t sp;
int cnt=0;
#define context (*pcontext)
#define PUSH1(reg)(regs[cnt++]=(word)pcontext->reg)
#define PUSH2(r1,r2)(PUSH1(r1),PUSH1(r2))
#define PUSH4(r1,r2,r3,r4)(PUSH2(r1,r2),PUSH2(r3,r4))
#if defined(I386)
#ifdef WOW64_THREAD_CONTEXT_WORKAROUND
PUSH2(ContextFlags,SegFs);
#endif
PUSH4(Edi,Esi,Ebx,Edx),PUSH2(Ecx,Eax),PUSH1(Ebp);
sp=(ptr_t)context.Esp;
#elif defined(X86_64)
PUSH4(Rax,Rcx,Rdx,Rbx);PUSH2(Rbp,Rsi);PUSH1(Rdi);
PUSH4(R8,R9,R10,R11);PUSH4(R12,R13,R14,R15);
sp=(ptr_t)context.Rsp;
#elif defined(ARM32)
PUSH4(R0,R1,R2,R3),PUSH4(R4,R5,R6,R7),PUSH4(R8,R9,R10,R11);
PUSH1(R12);
sp=(ptr_t)context.Sp;
#elif defined(AARCH64)
PUSH4(X0,X1,X2,X3),PUSH4(X4,X5,X6,X7),PUSH4(X8,X9,X10,X11);
PUSH4(X12,X13,X14,X15),PUSH4(X16,X17,X18,X19),PUSH4(X20,X21,X22,X23);
PUSH4(X24,X25,X26,X27),PUSH1(X28);
PUSH1(Lr);
sp=(ptr_t)context.Sp;
#elif defined(SHx)
PUSH4(R0,R1,R2,R3),PUSH4(R4,R5,R6,R7),PUSH4(R8,R9,R10,R11);
PUSH2(R12,R13),PUSH1(R14);
sp=(ptr_t)context.R15;
#elif defined(MIPS)
PUSH4(IntAt,IntV0,IntV1,IntA0),PUSH4(IntA1,IntA2,IntA3,IntT0);
PUSH4(IntT1,IntT2,IntT3,IntT4),PUSH4(IntT5,IntT6,IntT7,IntS0);
PUSH4(IntS1,IntS2,IntS3,IntS4),PUSH4(IntS5,IntS6,IntS7,IntT8);
PUSH4(IntT9,IntK0,IntK1,IntS8);
sp=(ptr_t)context.IntSp;
#elif defined(PPC)
PUSH4(Gpr0,Gpr3,Gpr4,Gpr5),PUSH4(Gpr6,Gpr7,Gpr8,Gpr9);
PUSH4(Gpr10,Gpr11,Gpr12,Gpr14),PUSH4(Gpr15,Gpr16,Gpr17,Gpr18);
PUSH4(Gpr19,Gpr20,Gpr21,Gpr22),PUSH4(Gpr23,Gpr24,Gpr25,Gpr26);
PUSH4(Gpr27,Gpr28,Gpr29,Gpr30),PUSH1(Gpr31);
sp=(ptr_t)context.Gpr1;
#elif defined(ALPHA)
PUSH4(IntV0,IntT0,IntT1,IntT2),PUSH4(IntT3,IntT4,IntT5,IntT6);
PUSH4(IntT7,IntS0,IntS1,IntS2),PUSH4(IntS3,IntS4,IntS5,IntFp);
PUSH4(IntA0,IntA1,IntA2,IntA3),PUSH4(IntA4,IntA5,IntT8,IntT9);
PUSH4(IntT10,IntT11,IntT12,IntAt);
sp=(ptr_t)context.IntSp;
#elif defined(CPPCHECK)
sp=(ptr_t)(word)cnt;
#else
#error Architecture is not supported
#endif
#undef context
GC_ASSERT(cnt==PUSHED_REGS_COUNT);
return sp;
}
STATIC word GC_push_stack_for(GC_thread thread,DWORD me)
{
ptr_t sp,stack_min;
struct GC_traced_stack_sect_s*traced_stack_sect=
thread->traced_stack_sect;
if (thread->id==me){
GC_ASSERT(thread->thread_blocked_sp==NULL);
sp=GC_approx_sp();
} else if ((sp=thread->thread_blocked_sp)==NULL){
#ifdef RETRY_GET_THREAD_CONTEXT
word*regs=thread->context_regs;
if (thread->suspended){
sp=thread->context_sp;
} else
#else
word regs[PUSHED_REGS_COUNT];
#endif
{
CONTEXT context;
context.ContextFlags=GET_THREAD_CONTEXT_FLAGS;
if (GetThreadContext(THREAD_HANDLE(thread),&context)){
sp=copy_ptr_regs(regs,&context);
} else {
#ifdef RETRY_GET_THREAD_CONTEXT
sp=thread->context_sp;
if (NULL==sp){
return 0;
}
#else
ABORT("GetThreadContext failed");
#endif
}
}
#ifdef THREAD_LOCAL_ALLOC
GC_ASSERT(thread->suspended||!GC_world_stopped);
#endif
#ifndef WOW64_THREAD_CONTEXT_WORKAROUND
GC_push_many_regs(regs,PUSHED_REGS_COUNT);
#else
GC_push_many_regs(regs+2,PUSHED_REGS_COUNT - 2);
if (isWow64){
DWORD ContextFlags=(DWORD)regs[0];
WORD SegFs=(WORD)regs[1];
if ((ContextFlags&CONTEXT_EXCEPTION_REPORTING)!=0
&&(ContextFlags&(CONTEXT_EXCEPTION_ACTIVE
))!=0){
LDT_ENTRY selector;
PNT_TIB tib;
if (!GetThreadSelectorEntry(THREAD_HANDLE(thread),SegFs,&selector))
ABORT("GetThreadSelectorEntry failed");
tib=(PNT_TIB)(selector.BaseLow
|(selector.HighWord.Bits.BaseMid<<16)
|(selector.HighWord.Bits.BaseHi<<24));
#ifdef DEBUG_THREADS
GC_log_printf("TIB stack limit/base:%p .. %p\n",
(void*)tib->StackLimit,(void*)tib->StackBase);
#endif
GC_ASSERT(!((word)thread->stack_base
COOLER_THAN (word)tib->StackBase));
if (thread->stack_base!=thread->initial_stack_base
&&((word)thread->stack_base<=(word)tib->StackLimit
||(word)tib->StackBase < (word)thread->stack_base)){
WARN("GetThreadContext might return stale register values"
" including ESP=%p\n",sp);
} else {
sp=(ptr_t)tib->StackLimit;
}
}
#ifdef DEBUG_THREADS
else {
static GC_bool logged;
if (!logged
&&(ContextFlags&CONTEXT_EXCEPTION_REPORTING)==0){
GC_log_printf("CONTEXT_EXCEPTION_REQUEST not supported\n");
logged=TRUE;
}
}
#endif
}
#endif
}
if (thread->last_stack_min==ADDR_LIMIT){
#ifdef MSWINCE
if (GC_dont_query_stack_min){
stack_min=GC_wince_evaluate_stack_min(traced_stack_sect!=NULL?
(ptr_t)traced_stack_sect:thread->stack_base);
} else
#endif
{
stack_min=GC_get_stack_min(traced_stack_sect!=NULL?
(ptr_t)traced_stack_sect:thread->stack_base);
UNPROTECT_THREAD(thread);
thread->last_stack_min=stack_min;
}
} else {
if (traced_stack_sect!=NULL&&
(word)thread->last_stack_min > (word)traced_stack_sect){
UNPROTECT_THREAD(thread);
thread->last_stack_min=(ptr_t)traced_stack_sect;
}
if ((word)sp < (word)thread->stack_base
&&(word)sp>=(word)thread->last_stack_min){
stack_min=sp;
} else {
if (may_be_in_stack(thread->id==me&&
(word)sp < (word)thread->last_stack_min?
sp:thread->last_stack_min)){
stack_min=(ptr_t)last_info.BaseAddress;
if ((word)sp < (word)stack_min
||(word)sp>=(word)thread->stack_base)
stack_min=GC_get_stack_min(thread->last_stack_min);
} else {
stack_min=GC_get_stack_min(thread->stack_base);
}
UNPROTECT_THREAD(thread);
thread->last_stack_min=stack_min;
}
}
GC_ASSERT(GC_dont_query_stack_min
||stack_min==GC_get_stack_min(thread->stack_base)
||((word)sp>=(word)stack_min
&&(word)stack_min < (word)thread->stack_base
&&(word)stack_min
> (word)GC_get_stack_min(thread->stack_base)));
if ((word)sp>=(word)stack_min&&(word)sp < (word)thread->stack_base){
#ifdef DEBUG_THREADS
GC_log_printf("Pushing stack for 0x%x from sp %p to %p from 0x%x\n",
(int)thread->id,(void*)sp,(void*)thread->stack_base,
(int)me);
#endif
GC_push_all_stack_sections(sp,thread->stack_base,traced_stack_sect);
} else {
if (thread->id==me||(word)sp>=(word)thread->stack_base
||(word)(sp+GC_page_size)< (word)stack_min)
WARN("Thread stack pointer %p out of range,pushing everything\n",
sp);
#ifdef DEBUG_THREADS
GC_log_printf("Pushing stack for 0x%x from (min)%p to %p from 0x%x\n",
(int)thread->id,(void*)stack_min,
(void*)thread->stack_base,(int)me);
#endif
GC_push_all_stack(stack_min,thread->stack_base);
}
return thread->stack_base - sp;
}
GC_INNER void GC_push_all_stacks(void)
{
DWORD thread_id=GetCurrentThreadId();
GC_bool found_me=FALSE;
#ifndef SMALL_CONFIG
unsigned nthreads=0;
#endif
word total_size=0;
#ifndef GC_NO_THREADS_DISCOVERY
if (GC_win32_dll_threads){
int i;
LONG my_max=GC_get_max_thread_index();
for (i=0;i<=my_max;i++){
GC_thread t=(GC_thread)(dll_thread_table+i);
if (t->tm.in_use&&t->stack_base){
#ifndef SMALL_CONFIG
++nthreads;
#endif
total_size+=GC_push_stack_for(t,thread_id);
if (t->id==thread_id)found_me=TRUE;
}
}
} else
#endif
{
int i;
for (i=0;i < THREAD_TABLE_SZ;i++){
GC_thread t;
for (t=GC_threads[i];t!=0;t=t->tm.next){
if (!KNOWN_FINISHED(t)&&t->stack_base){
#ifndef SMALL_CONFIG
++nthreads;
#endif
total_size+=GC_push_stack_for(t,thread_id);
if (t->id==thread_id)found_me=TRUE;
}
}
}
}
#ifndef SMALL_CONFIG
GC_VERBOSE_LOG_PRINTF("Pushed %d thread stacks%s\n",nthreads,
GC_win32_dll_threads?
" based on DllMain thread tracking":"");
#endif
if (!found_me&&!GC_in_thread_creation)
ABORT("Collecting from unknown thread");
GC_total_stacksize=total_size;
}
#ifdef PARALLEL_MARK
#ifndef MAX_MARKERS
#define MAX_MARKERS 16
#endif
static ptr_t marker_sp[MAX_MARKERS - 1];
#ifdef IA64
static ptr_t marker_bsp[MAX_MARKERS - 1];
#endif
static ptr_t marker_last_stack_min[MAX_MARKERS - 1];
#endif
GC_INNER void GC_get_next_stack(char*start,char*limit,
char**lo,char**hi)
{
int i;
char*current_min=ADDR_LIMIT;
ptr_t*plast_stack_min=NULL;
GC_thread thread=NULL;
if (GC_win32_dll_threads){
LONG my_max=GC_get_max_thread_index();
for (i=0;i<=my_max;i++){
ptr_t s=(ptr_t)(dll_thread_table[i].stack_base);
if ((word)s > (word)start&&(word)s < (word)current_min){
plast_stack_min=(ptr_t*)
&dll_thread_table[i].last_stack_min;
current_min=s;
#if defined(CPPCHECK)
thread=(GC_thread)&dll_thread_table[i];
#endif
}
}
} else {
for (i=0;i < THREAD_TABLE_SZ;i++){
GC_thread t;
for (t=GC_threads[i];t!=0;t=t->tm.next){
ptr_t s=t->stack_base;
if ((word)s > (word)start&&(word)s < (word)current_min){
plast_stack_min=&t->last_stack_min;
thread=t;
current_min=s;
}
}
}
#ifdef PARALLEL_MARK
for (i=0;i < GC_markers_m1;++i){
ptr_t s=marker_sp[i];
#ifdef IA64
#endif
if ((word)s > (word)start&&(word)s < (word)current_min){
GC_ASSERT(marker_last_stack_min[i]!=NULL);
plast_stack_min=&marker_last_stack_min[i];
current_min=s;
thread=NULL;
}
}
#endif
}
*hi=current_min;
if (current_min==ADDR_LIMIT){
*lo=ADDR_LIMIT;
return;
}
GC_ASSERT((word)current_min > (word)start&&plast_stack_min!=NULL);
#ifdef MSWINCE
if (GC_dont_query_stack_min){
*lo=GC_wince_evaluate_stack_min(current_min);
return;
}
#endif
if ((word)current_min > (word)limit&&!may_be_in_stack(limit)){
*lo=ADDR_LIMIT;
return;
}
if (*plast_stack_min==ADDR_LIMIT
||!may_be_in_stack(*plast_stack_min)){
*lo=GC_get_stack_min(current_min);
} else {
*lo=GC_get_stack_min(*plast_stack_min);
}
if (thread!=NULL){
UNPROTECT_THREAD(thread);
}
*plast_stack_min=*lo;
}
#ifdef PARALLEL_MARK
#if defined(GC_PTHREADS)&&!defined(GC_PTHREADS_PARAMARK)
#if!defined(__MINGW32__)
#define GC_PTHREADS_PARAMARK
#endif
#endif
#if!defined(GC_PTHREADS_PARAMARK)
STATIC HANDLE GC_marker_cv[MAX_MARKERS - 1]={0};
STATIC DWORD GC_marker_Id[MAX_MARKERS - 1]={0};
#endif
#if defined(GC_PTHREADS)&&defined(HAVE_PTHREAD_SETNAME_NP_WITH_TID)
static void set_marker_thread_name(unsigned id)
{
char name_buf[16];
int len=sizeof("GC-marker-")- 1;
BCOPY("GC-marker-",name_buf,len);
if (id>=10)
name_buf[len++]=(char)('0'+(id/10)% 10);
name_buf[len]=(char)('0'+id % 10);
name_buf[len+1]='\0';
if (pthread_setname_np(pthread_self(),name_buf)!=0)
WARN("pthread_setname_np failed\n",0);
}
#elif!defined(MSWINCE)
static FARPROC setThreadDescription_fn;
static void set_marker_thread_name(unsigned id)
{
WCHAR name_buf[16];
int len=sizeof(L"GC-marker-")/sizeof(WCHAR)- 1;
HRESULT hr;
if (!setThreadDescription_fn)return;
BCOPY(L"GC-marker-",name_buf,len*sizeof(WCHAR));
if (id>=10)
name_buf[len++]=(WCHAR)('0'+(id/10)% 10);
name_buf[len]=(WCHAR)('0'+id % 10);
name_buf[len+1]=0;
hr=(*(HRESULT (WINAPI*)(HANDLE,const WCHAR*))
(word)setThreadDescription_fn)(GetCurrentThread(),name_buf);
if (FAILED(hr))
WARN("SetThreadDescription failed\n",0);
}
#else
#define set_marker_thread_name(id)(void)(id)
#endif
#ifdef GC_PTHREADS_PARAMARK
STATIC void*GC_mark_thread(void*id)
#elif defined(MSWINCE)
STATIC DWORD WINAPI GC_mark_thread(LPVOID id)
#else
STATIC unsigned __stdcall GC_mark_thread(void*id)
#endif
{
word my_mark_no=0;
if ((word)id==GC_WORD_MAX)return 0;
set_marker_thread_name((unsigned)(word)id);
marker_sp[(word)id]=GC_approx_sp();
#ifdef IA64
marker_bsp[(word)id]=GC_save_regs_in_stack();
#endif
#if!defined(GC_PTHREADS_PARAMARK)
GC_marker_Id[(word)id]=GetCurrentThreadId();
#endif
GC_acquire_mark_lock();
if (0==--GC_fl_builder_count)
GC_notify_all_builder();
for (;;++my_mark_no){
if (my_mark_no - GC_mark_no > (word)2){
my_mark_no=GC_mark_no;
}
#ifdef DEBUG_THREADS
GC_log_printf("Starting mark helper for mark number %lu\n",
(unsigned long)my_mark_no);
#endif
GC_help_marker(my_mark_no);
}
}
#ifndef GC_ASSERTIONS
#define SET_MARK_LOCK_HOLDER (void)0
#define UNSET_MARK_LOCK_HOLDER (void)0
#endif
#ifdef CAN_HANDLE_FORK
static int available_markers_m1=0;
#else
#define available_markers_m1 GC_markers_m1
#endif
#ifdef GC_PTHREADS_PARAMARK
#include <pthread.h>
#if defined(GC_ASSERTIONS)&&!defined(NUMERIC_THREAD_ID)
#define NUMERIC_THREAD_ID(id)(unsigned long)(word)GC_PTHREAD_PTRVAL(id)
#endif
#ifdef CAN_HANDLE_FORK
static pthread_cond_t mark_cv;
#else
static pthread_cond_t mark_cv=PTHREAD_COND_INITIALIZER;
#endif
GC_INNER void GC_start_mark_threads_inner(void)
{
int i;
pthread_attr_t attr;
pthread_t new_thread;
#ifndef NO_MARKER_SPECIAL_SIGMASK
sigset_t set,oldset;
#endif
GC_ASSERT(I_DONT_HOLD_LOCK());
if (available_markers_m1<=0)return;
#ifdef CAN_HANDLE_FORK
if (GC_parallel)return;
{
pthread_cond_t mark_cv_local=PTHREAD_COND_INITIALIZER;
BCOPY(&mark_cv_local,&mark_cv,sizeof(mark_cv));
}
#endif
GC_ASSERT(GC_fl_builder_count==0);
if (0!=pthread_attr_init(&attr))ABORT("pthread_attr_init failed");
if (0!=pthread_attr_setdetachstate(&attr,PTHREAD_CREATE_DETACHED))
ABORT("pthread_attr_setdetachstate failed");
#ifndef NO_MARKER_SPECIAL_SIGMASK
if (sigfillset(&set)!=0)
ABORT("sigfillset failed");
if (pthread_sigmask(SIG_BLOCK,&set,&oldset)< 0){
WARN("pthread_sigmask set failed,no markers started,"
" errno=%" WARN_PRIdPTR "\n",errno);
GC_markers_m1=0;
(void)pthread_attr_destroy(&attr);
return;
}
#endif
#ifdef CAN_HANDLE_FORK
GC_markers_m1=available_markers_m1;
#endif
for (i=0;i < available_markers_m1;++i){
marker_last_stack_min[i]=ADDR_LIMIT;
if (0!=pthread_create(&new_thread,&attr,
GC_mark_thread,(void*)(word)i)){
WARN("Marker thread creation failed\n",0);
GC_markers_m1=i;
break;
}
}
#ifndef NO_MARKER_SPECIAL_SIGMASK
if (pthread_sigmask(SIG_SETMASK,&oldset,NULL)< 0){
WARN("pthread_sigmask restore failed,errno=%" WARN_PRIdPTR "\n",
errno);
}
#endif
(void)pthread_attr_destroy(&attr);
GC_wait_for_markers_init();
GC_COND_LOG_PRINTF("Started %d mark helper threads\n",GC_markers_m1);
}
#ifdef GC_ASSERTIONS
STATIC unsigned long GC_mark_lock_holder=NO_THREAD;
#define SET_MARK_LOCK_HOLDER (void)(GC_mark_lock_holder=NUMERIC_THREAD_ID(pthread_self()))
#define UNSET_MARK_LOCK_HOLDER do { GC_ASSERT(GC_mark_lock_holder==NUMERIC_THREAD_ID(pthread_self()));GC_mark_lock_holder=NO_THREAD;} while (0)
#endif
static pthread_mutex_t mark_mutex=PTHREAD_MUTEX_INITIALIZER;
static pthread_cond_t builder_cv=PTHREAD_COND_INITIALIZER;
#ifdef LOCK_STATS
volatile AO_t GC_block_count=0;
#endif
GC_INNER void GC_acquire_mark_lock(void)
{
#if defined(NUMERIC_THREAD_ID_UNIQUE)&&!defined(THREAD_SANITIZER)
GC_ASSERT(GC_mark_lock_holder!=NUMERIC_THREAD_ID(pthread_self()));
#endif
if (pthread_mutex_lock(&mark_mutex)!=0){
ABORT("pthread_mutex_lock failed");
}
#ifdef LOCK_STATS
(void)AO_fetch_and_add1(&GC_block_count);
#endif
SET_MARK_LOCK_HOLDER;
}
GC_INNER void GC_release_mark_lock(void)
{
UNSET_MARK_LOCK_HOLDER;
if (pthread_mutex_unlock(&mark_mutex)!=0){
ABORT("pthread_mutex_unlock failed");
}
}
STATIC void GC_wait_builder(void)
{
UNSET_MARK_LOCK_HOLDER;
if (pthread_cond_wait(&builder_cv,&mark_mutex)!=0){
ABORT("pthread_cond_wait failed");
}
GC_ASSERT(GC_mark_lock_holder==NO_THREAD);
SET_MARK_LOCK_HOLDER;
}
GC_INNER void GC_wait_for_reclaim(void)
{
GC_acquire_mark_lock();
while (GC_fl_builder_count > 0){
GC_wait_builder();
}
GC_release_mark_lock();
}
GC_INNER void GC_notify_all_builder(void)
{
GC_ASSERT(GC_mark_lock_holder==NUMERIC_THREAD_ID(pthread_self()));
if (pthread_cond_broadcast(&builder_cv)!=0){
ABORT("pthread_cond_broadcast failed");
}
}
GC_INNER void GC_wait_marker(void)
{
GC_ASSERT(GC_parallel);
UNSET_MARK_LOCK_HOLDER;
if (pthread_cond_wait(&mark_cv,&mark_mutex)!=0){
ABORT("pthread_cond_wait failed");
}
GC_ASSERT(GC_mark_lock_holder==NO_THREAD);
SET_MARK_LOCK_HOLDER;
}
GC_INNER void GC_notify_all_marker(void)
{
GC_ASSERT(GC_parallel);
if (pthread_cond_broadcast(&mark_cv)!=0){
ABORT("pthread_cond_broadcast failed");
}
}
#else
#ifndef MARK_THREAD_STACK_SIZE
#define MARK_THREAD_STACK_SIZE 0
#endif
static HANDLE mark_mutex_event=(HANDLE)0;
static HANDLE builder_cv=(HANDLE)0;
static HANDLE mark_cv=(HANDLE)0;
GC_INNER void GC_start_mark_threads_inner(void)
{
int i;
GC_ASSERT(I_DONT_HOLD_LOCK());
if (available_markers_m1<=0)return;
GC_ASSERT(GC_fl_builder_count==0);
for (i=0;i < GC_markers_m1;++i){
if ((GC_marker_cv[i]=CreateEvent(NULL,
TRUE,
FALSE,
NULL))==(HANDLE)0)
ABORT("CreateEvent failed");
}
for (i=0;i < GC_markers_m1;++i){
#if defined(MSWINCE)||defined(MSWIN_XBOX1)
HANDLE handle;
DWORD thread_id;
marker_last_stack_min[i]=ADDR_LIMIT;
handle=CreateThread(NULL,
MARK_THREAD_STACK_SIZE,
GC_mark_thread,(LPVOID)(word)i,
0,&thread_id);
if (handle==NULL){
WARN("Marker thread creation failed\n",0);
break;
} else {
CloseHandle(handle);
}
#else
GC_uintptr_t handle;
unsigned thread_id;
marker_last_stack_min[i]=ADDR_LIMIT;
handle=_beginthreadex(NULL,
MARK_THREAD_STACK_SIZE,GC_mark_thread,
(void*)(word)i,0,&thread_id);
if (!handle||handle==(GC_uintptr_t)-1L){
WARN("Marker thread creation failed\n",0);
break;
} else {
}
#endif
}
while (GC_markers_m1 > i){
GC_markers_m1--;
CloseHandle(GC_marker_cv[GC_markers_m1]);
}
GC_wait_for_markers_init();
GC_COND_LOG_PRINTF("Started %d mark helper threads\n",GC_markers_m1);
if (i==0){
CloseHandle(mark_cv);
CloseHandle(builder_cv);
CloseHandle(mark_mutex_event);
}
}
#ifdef GC_ASSERTIONS
STATIC DWORD GC_mark_lock_holder=NO_THREAD;
#define SET_MARK_LOCK_HOLDER (void)(GC_mark_lock_holder=GetCurrentThreadId())
#define UNSET_MARK_LOCK_HOLDER do { GC_ASSERT(GC_mark_lock_holder==GetCurrentThreadId());GC_mark_lock_holder=NO_THREAD;} while (0)
#endif
STATIC LONG GC_mark_mutex_state=0;
#ifdef LOCK_STATS
volatile AO_t GC_block_count=0;
volatile AO_t GC_unlocked_count=0;
#endif
GC_INNER void GC_acquire_mark_lock(void)
{
#ifndef THREAD_SANITIZER
GC_ASSERT(GC_mark_lock_holder!=GetCurrentThreadId());
#endif
if (InterlockedExchange(&GC_mark_mutex_state,1)!=0){
#ifdef LOCK_STATS
(void)AO_fetch_and_add1(&GC_block_count);
#endif
while (InterlockedExchange(&GC_mark_mutex_state,
-1)!=0){
if (WaitForSingleObject(mark_mutex_event,INFINITE)==WAIT_FAILED)
ABORT("WaitForSingleObject failed");
}
}
#ifdef LOCK_STATS
else {
(void)AO_fetch_and_add1(&GC_unlocked_count);
}
#endif
GC_ASSERT(GC_mark_lock_holder==NO_THREAD);
SET_MARK_LOCK_HOLDER;
}
GC_INNER void GC_release_mark_lock(void)
{
UNSET_MARK_LOCK_HOLDER;
if (InterlockedExchange(&GC_mark_mutex_state,0)< 0){
if (SetEvent(mark_mutex_event)==FALSE)
ABORT("SetEvent failed");
}
}
GC_INNER void GC_wait_for_reclaim(void)
{
GC_ASSERT(builder_cv!=0);
for (;;){
GC_acquire_mark_lock();
if (GC_fl_builder_count==0)
break;
if (ResetEvent(builder_cv)==FALSE)
ABORT("ResetEvent failed");
GC_release_mark_lock();
if (WaitForSingleObject(builder_cv,INFINITE)==WAIT_FAILED)
ABORT("WaitForSingleObject failed");
}
GC_release_mark_lock();
}
GC_INNER void GC_notify_all_builder(void)
{
GC_ASSERT(GC_mark_lock_holder==GetCurrentThreadId());
GC_ASSERT(builder_cv!=0);
GC_ASSERT(GC_fl_builder_count==0);
if (SetEvent(builder_cv)==FALSE)
ABORT("SetEvent failed");
}
GC_INNER void GC_wait_marker(void)
{
HANDLE event=mark_cv;
DWORD thread_id=GetCurrentThreadId();
int i=GC_markers_m1;
while (i--> 0){
if (GC_marker_Id[i]==thread_id){
event=GC_marker_cv[i];
break;
}
}
if (ResetEvent(event)==FALSE)
ABORT("ResetEvent failed");
GC_release_mark_lock();
if (WaitForSingleObject(event,INFINITE)==WAIT_FAILED)
ABORT("WaitForSingleObject failed");
GC_acquire_mark_lock();
}
GC_INNER void GC_notify_all_marker(void)
{
DWORD thread_id=GetCurrentThreadId();
int i=GC_markers_m1;
while (i--> 0){
if (SetEvent(GC_marker_Id[i]!=thread_id?GC_marker_cv[i]:
mark_cv)==FALSE)
ABORT("SetEvent failed");
}
}
#endif
static unsigned required_markers_cnt=0;
#endif
typedef struct {
LPTHREAD_START_ROUTINE start;
LPVOID param;
} thread_args;
STATIC void*GC_CALLBACK GC_win32_start_inner(struct GC_stack_base*sb,
void*arg)
{
void*ret;
LPTHREAD_START_ROUTINE start=((thread_args*)arg)->start;
LPVOID param=((thread_args*)arg)->param;
GC_register_my_thread(sb);
#ifdef DEBUG_THREADS
GC_log_printf("thread 0x%lx starting...\n",(long)GetCurrentThreadId());
#endif
GC_free(arg);
#if!defined(__GNUC__)&&!defined(NO_CRT)
ret=NULL;
__try
#endif
{
ret=(void*)(word)(*start)(param);
}
#if!defined(__GNUC__)&&!defined(NO_CRT)
__finally
#endif
{
GC_unregister_my_thread();
}
#ifdef DEBUG_THREADS
GC_log_printf("thread 0x%lx returned from start routine\n",
(long)GetCurrentThreadId());
#endif
return ret;
}
STATIC DWORD WINAPI GC_win32_start(LPVOID arg)
{
return (DWORD)(word)GC_call_with_stack_base(GC_win32_start_inner,arg);
}
GC_API HANDLE WINAPI GC_CreateThread(
LPSECURITY_ATTRIBUTES lpThreadAttributes,
GC_WIN32_SIZE_T dwStackSize,
LPTHREAD_START_ROUTINE lpStartAddress,
LPVOID lpParameter,DWORD dwCreationFlags,
LPDWORD lpThreadId)
{
if (!EXPECT(parallel_initialized,TRUE))
GC_init_parallel();
#ifdef DEBUG_THREADS
GC_log_printf("About to create a thread from 0x%lx\n",
(long)GetCurrentThreadId());
#endif
if (GC_win32_dll_threads){
return CreateThread(lpThreadAttributes,dwStackSize,lpStartAddress,
lpParameter,dwCreationFlags,lpThreadId);
} else {
thread_args*args=
(thread_args*)GC_malloc_uncollectable(sizeof(thread_args));
HANDLE thread_h;
if (NULL==args){
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
return NULL;
}
args->start=lpStartAddress;
args->param=lpParameter;
GC_dirty(args);
REACHABLE_AFTER_DIRTY(lpParameter);
set_need_to_lock();
thread_h=CreateThread(lpThreadAttributes,dwStackSize,GC_win32_start,
args,dwCreationFlags,lpThreadId);
if (thread_h==0)GC_free(args);
return thread_h;
}
}
GC_API DECLSPEC_NORETURN void WINAPI GC_ExitThread(DWORD dwExitCode)
{
GC_unregister_my_thread();
ExitThread(dwExitCode);
}
#if!defined(CYGWIN32)&&!defined(MSWINCE)&&!defined(MSWIN_XBOX1)&&!defined(NO_CRT)
GC_API GC_uintptr_t GC_CALL GC_beginthreadex(
void*security,unsigned stack_size,
unsigned (__stdcall*start_address)(void*),
void*arglist,unsigned initflag,
unsigned*thrdaddr)
{
if (!EXPECT(parallel_initialized,TRUE))
GC_init_parallel();
#ifdef DEBUG_THREADS
GC_log_printf("About to create a thread from 0x%lx\n",
(long)GetCurrentThreadId());
#endif
if (GC_win32_dll_threads){
return _beginthreadex(security,stack_size,start_address,
arglist,initflag,thrdaddr);
} else {
GC_uintptr_t thread_h;
thread_args*args=
(thread_args*)GC_malloc_uncollectable(sizeof(thread_args));
if (NULL==args){
errno=EAGAIN;
return 0;
}
args->start=(LPTHREAD_START_ROUTINE)start_address;
args->param=arglist;
GC_dirty(args);
REACHABLE_AFTER_DIRTY(arglist);
set_need_to_lock();
thread_h=_beginthreadex(security,stack_size,
(unsigned (__stdcall*)(void*))GC_win32_start,
args,initflag,thrdaddr);
if (thread_h==0)GC_free(args);
return thread_h;
}
}
GC_API void GC_CALL GC_endthreadex(unsigned retval)
{
GC_unregister_my_thread();
_endthreadex(retval);
}
#endif
#ifdef GC_WINMAIN_REDIRECT
#if defined(MSWINCE)&&defined(UNDER_CE)
#define WINMAIN_LPTSTR LPWSTR
#else
#define WINMAIN_LPTSTR LPSTR
#endif
#undef WinMain
int WINAPI GC_WinMain(HINSTANCE,HINSTANCE,WINMAIN_LPTSTR,int);
typedef struct {
HINSTANCE hInstance;
HINSTANCE hPrevInstance;
WINMAIN_LPTSTR lpCmdLine;
int nShowCmd;
} main_thread_args;
static DWORD WINAPI main_thread_start(LPVOID arg)
{
main_thread_args*args=(main_thread_args*)arg;
return (DWORD)GC_WinMain(args->hInstance,args->hPrevInstance,
args->lpCmdLine,args->nShowCmd);
}
STATIC void*GC_waitForSingleObjectInfinite(void*handle)
{
return (void*)(word)WaitForSingleObject((HANDLE)handle,INFINITE);
}
#ifndef WINMAIN_THREAD_STACK_SIZE
#define WINMAIN_THREAD_STACK_SIZE 0
#endif
int WINAPI WinMain(HINSTANCE hInstance,HINSTANCE hPrevInstance,
WINMAIN_LPTSTR lpCmdLine,int nShowCmd)
{
DWORD exit_code=1;
main_thread_args args={
hInstance,hPrevInstance,lpCmdLine,nShowCmd
};
HANDLE thread_h;
DWORD thread_id;
GC_INIT();
thread_h=GC_CreateThread(NULL,
WINMAIN_THREAD_STACK_SIZE,
main_thread_start,&args,0,
&thread_id);
if (thread_h!=NULL){
if ((DWORD)(word)GC_do_blocking(GC_waitForSingleObjectInfinite,
(void*)thread_h)==WAIT_FAILED)
ABORT("WaitForSingleObject(main_thread)failed");
GetExitCodeThread (thread_h,&exit_code);
CloseHandle (thread_h);
} else {
ABORT("GC_CreateThread(main_thread)failed");
}
#ifdef MSWINCE
GC_deinit();
#endif
return (int)exit_code;
}
#endif
GC_API void GC_CALL GC_set_markers_count(unsigned markers GC_ATTR_UNUSED)
{
#ifdef PARALLEL_MARK
required_markers_cnt=markers < MAX_MARKERS?markers:MAX_MARKERS;
#endif
}
GC_INNER void GC_thr_init(void)
{
struct GC_stack_base sb;
#if (!defined(HAVE_PTHREAD_SETNAME_NP_WITH_TID)&&!defined(MSWINCE)&&defined(PARALLEL_MARK))||defined(WOW64_THREAD_CONTEXT_WORKAROUND)
HMODULE hK32=GetModuleHandle(TEXT("kernel32.dll"));
#endif
GC_ASSERT(I_HOLD_LOCK());
if (GC_thr_initialized)return;
GC_ASSERT((word)&GC_threads % sizeof(word)==0);
#ifdef GC_NO_THREADS_DISCOVERY
#define GC_main_thread GetCurrentThreadId()
#else
GC_main_thread=GetCurrentThreadId();
#endif
GC_thr_initialized=TRUE;
#ifdef CAN_HANDLE_FORK
if (GC_handle_fork){
#ifdef CAN_CALL_ATFORK
if (pthread_atfork(fork_prepare_proc,fork_parent_proc,
fork_child_proc)==0){
GC_handle_fork=1;
} else
#endif
if (GC_handle_fork!=-1)
ABORT("pthread_atfork failed");
}
#endif
#ifdef WOW64_THREAD_CONTEXT_WORKAROUND
if (hK32){
FARPROC pfn=GetProcAddress(hK32,"IsWow64Process");
if (pfn
&&!(*(BOOL (WINAPI*)(HANDLE,BOOL*))(word)pfn)(
GetCurrentProcess(),&isWow64))
isWow64=FALSE;
}
#endif
sb.mem_base=GC_stackbottom;
GC_ASSERT(sb.mem_base!=NULL);
#ifdef IA64
sb.reg_base=GC_register_stackbottom;
#endif
#if defined(PARALLEL_MARK)
{
char*markers_string=GETENV("GC_MARKERS");
int markers=required_markers_cnt;
if (markers_string!=NULL){
markers=atoi(markers_string);
if (markers<=0||markers > MAX_MARKERS){
WARN("Too big or invalid number of mark threads:%" WARN_PRIdPTR
";using maximum threads\n",(signed_word)markers);
markers=MAX_MARKERS;
}
} else if (0==markers){
#ifdef MSWINCE
markers=(int)GC_sysinfo.dwNumberOfProcessors;
#else
#ifdef _WIN64
DWORD_PTR procMask=0;
DWORD_PTR sysMask;
#else
DWORD procMask=0;
DWORD sysMask;
#endif
int ncpu=0;
if (
#ifdef __cplusplus
GetProcessAffinityMask(GetCurrentProcess(),&procMask,&sysMask)
#else
GetProcessAffinityMask(GetCurrentProcess(),
(void*)&procMask,(void*)&sysMask)
#endif
&&procMask){
do {
ncpu++;
} while ((procMask&=procMask - 1)!=0);
}
markers=ncpu;
#endif
#if defined(GC_MIN_MARKERS)&&!defined(CPPCHECK)
if (markers < GC_MIN_MARKERS)
markers=GC_MIN_MARKERS;
#endif
if (markers > MAX_MARKERS)
markers=MAX_MARKERS;
}
available_markers_m1=markers - 1;
}
if (GC_win32_dll_threads||available_markers_m1<=0){
GC_parallel=FALSE;
GC_COND_LOG_PRINTF(
"Single marker thread,turning off parallel marking\n");
} else {
#ifndef GC_PTHREADS_PARAMARK
mark_mutex_event=CreateEvent(NULL,
FALSE,
FALSE,NULL);
builder_cv=CreateEvent(NULL,
TRUE,
FALSE,NULL);
mark_cv=CreateEvent(NULL,TRUE,
FALSE,NULL);
if (mark_mutex_event==(HANDLE)0||builder_cv==(HANDLE)0
||mark_cv==(HANDLE)0)
ABORT("CreateEvent failed");
#endif
#if!defined(HAVE_PTHREAD_SETNAME_NP_WITH_TID)&&!defined(MSWINCE)
if (hK32)
setThreadDescription_fn=GetProcAddress(hK32,
"SetThreadDescription");
#endif
}
#endif
GC_ASSERT(0==GC_lookup_thread_inner(GC_main_thread));
GC_register_my_thread_inner(&sb,GC_main_thread);
#undef GC_main_thread
}
#ifdef GC_PTHREADS
struct start_info {
void*(*start_routine)(void*);
void*arg;
GC_bool detached;
};
GC_API int GC_pthread_join(pthread_t pthread_id,void**retval)
{
int result;
#ifndef GC_WIN32_PTHREADS
GC_thread t;
#endif
DCL_LOCK_STATE;
GC_ASSERT(!GC_win32_dll_threads);
#ifdef DEBUG_THREADS
GC_log_printf("thread %p(0x%lx)is joining thread %p\n",
(void*)GC_PTHREAD_PTRVAL(pthread_self()),
(long)GetCurrentThreadId(),
(void*)GC_PTHREAD_PTRVAL(pthread_id));
#endif
#ifndef GC_WIN32_PTHREADS
while ((t=GC_lookup_pthread(pthread_id))==0)
Sleep(10);
#endif
result=pthread_join(pthread_id,retval);
if (0==result){
#ifdef GC_WIN32_PTHREADS
GC_thread t=GC_lookup_pthread(pthread_id);
if (NULL==t)ABORT("Thread not registered");
#endif
LOCK();
if ((t->flags&FINISHED)!=0){
GC_delete_gc_thread_no_free(t);
GC_INTERNAL_FREE(t);
}
UNLOCK();
}
#ifdef DEBUG_THREADS
GC_log_printf("thread %p(0x%lx)join with thread %p %s\n",
(void*)GC_PTHREAD_PTRVAL(pthread_self()),
(long)GetCurrentThreadId(),
(void*)GC_PTHREAD_PTRVAL(pthread_id),
result!=0?"failed":"succeeded");
#endif
return result;
}
GC_API int GC_pthread_create(pthread_t*new_thread,
GC_PTHREAD_CREATE_CONST pthread_attr_t*attr,
void*(*start_routine)(void*),void*arg)
{
int result;
struct start_info*si;
if (!EXPECT(parallel_initialized,TRUE))
GC_init_parallel();
GC_ASSERT(!GC_win32_dll_threads);
si=(struct start_info*)GC_malloc_uncollectable(
sizeof(struct start_info));
if (NULL==si)
return EAGAIN;
si->start_routine=start_routine;
si->arg=arg;
GC_dirty(si);
REACHABLE_AFTER_DIRTY(arg);
if (attr!=0&&
pthread_attr_getdetachstate(attr,&si->detached)
==PTHREAD_CREATE_DETACHED){
si->detached=TRUE;
}
#ifdef DEBUG_THREADS
GC_log_printf("About to create a thread from %p(0x%lx)\n",
(void*)GC_PTHREAD_PTRVAL(pthread_self()),
(long)GetCurrentThreadId());
#endif
set_need_to_lock();
result=pthread_create(new_thread,attr,GC_pthread_start,si);
if (result){
GC_free(si);
}
return(result);
}
STATIC void*GC_CALLBACK GC_pthread_start_inner(struct GC_stack_base*sb,
void*arg)
{
struct start_info*si=(struct start_info*)arg;
void*result;
void*(*start)(void*);
void*start_arg;
DWORD thread_id=GetCurrentThreadId();
pthread_t pthread_id=pthread_self();
GC_thread me;
DCL_LOCK_STATE;
#ifdef DEBUG_THREADS
GC_log_printf("thread %p(0x%x)starting...\n",
(void*)GC_PTHREAD_PTRVAL(pthread_id),(int)thread_id);
#endif
GC_ASSERT(!GC_win32_dll_threads);
LOCK();
me=GC_register_my_thread_inner(sb,thread_id);
SET_PTHREAD_MAP_CACHE(pthread_id,thread_id);
GC_ASSERT(me!=&first_thread);
me->pthread_id=pthread_id;
if (si->detached)me->flags|=DETACHED;
UNLOCK();
start=si->start_routine;
start_arg=si->arg;
GC_free(si);
pthread_cleanup_push(GC_thread_exit_proc,(void*)me);
result=(*start)(start_arg);
me->status=result;
GC_dirty(me);
pthread_cleanup_pop(1);
#ifdef DEBUG_THREADS
GC_log_printf("thread %p(0x%x)returned from start routine\n",
(void*)GC_PTHREAD_PTRVAL(pthread_id),(int)thread_id);
#endif
return(result);
}
STATIC void*GC_pthread_start(void*arg)
{
return GC_call_with_stack_base(GC_pthread_start_inner,arg);
}
STATIC void GC_thread_exit_proc(void*arg)
{
GC_thread me=(GC_thread)arg;
DCL_LOCK_STATE;
GC_ASSERT(!GC_win32_dll_threads);
#ifdef DEBUG_THREADS
GC_log_printf("thread %p(0x%lx)called pthread_exit()\n",
(void*)GC_PTHREAD_PTRVAL(pthread_self()),
(long)GetCurrentThreadId());
#endif
LOCK();
GC_wait_for_gc_completion(FALSE);
#if defined(THREAD_LOCAL_ALLOC)
GC_ASSERT(GC_getspecific(GC_thread_key)==&me->tlfs);
GC_destroy_thread_local(&(me->tlfs));
#endif
if (me->flags&DETACHED){
GC_delete_thread(GetCurrentThreadId());
} else {
me->flags|=FINISHED;
}
#if defined(THREAD_LOCAL_ALLOC)
GC_remove_specific(GC_thread_key);
#endif
UNLOCK();
}
#ifndef GC_NO_PTHREAD_SIGMASK
GC_API int GC_pthread_sigmask(int how,const sigset_t*set,
sigset_t*oset)
{
return pthread_sigmask(how,set,oset);
}
#endif
GC_API int GC_pthread_detach(pthread_t thread)
{
int result;
GC_thread t;
DCL_LOCK_STATE;
GC_ASSERT(!GC_win32_dll_threads);
while ((t=GC_lookup_pthread(thread))==NULL)
Sleep(10);
result=pthread_detach(thread);
if (result==0){
LOCK();
t->flags|=DETACHED;
if ((t->flags&FINISHED)!=0){
GC_delete_gc_thread_no_free(t);
GC_INTERNAL_FREE(t);
}
UNLOCK();
}
return result;
}
#elif!defined(GC_NO_THREADS_DISCOVERY)
#ifdef GC_INSIDE_DLL
GC_API
#else
#define GC_DllMain DllMain
#endif
BOOL WINAPI GC_DllMain(HINSTANCE inst GC_ATTR_UNUSED,ULONG reason,
LPVOID reserved GC_ATTR_UNUSED)
{
DWORD thread_id;
if (!GC_win32_dll_threads&&parallel_initialized)return TRUE;
switch (reason){
case DLL_THREAD_ATTACH:
#ifdef PARALLEL_MARK
if (GC_parallel){
break;
}
#endif
case DLL_PROCESS_ATTACH:
thread_id=GetCurrentThreadId();
if (parallel_initialized&&GC_main_thread!=thread_id){
#ifdef PARALLEL_MARK
ABORT("Cannot initialize parallel marker from DllMain");
#else
struct GC_stack_base sb;
#ifdef GC_ASSERTIONS
int sb_result=
#endif
GC_get_stack_base(&sb);
GC_ASSERT(sb_result==GC_SUCCESS);
GC_register_my_thread_inner(&sb,thread_id);
#endif
}
break;
case DLL_THREAD_DETACH:
GC_ASSERT(parallel_initialized);
if (GC_win32_dll_threads){
GC_delete_thread(GetCurrentThreadId());
}
break;
case DLL_PROCESS_DETACH:
if (GC_win32_dll_threads){
int i;
int my_max=(int)GC_get_max_thread_index();
for (i=0;i<=my_max;++i){
if (AO_load(&(dll_thread_table[i].tm.in_use)))
GC_delete_gc_thread_no_free(&dll_thread_table[i]);
}
GC_deinit();
}
break;
}
return TRUE;
}
#endif
GC_INNER void GC_init_parallel(void)
{
#if defined(THREAD_LOCAL_ALLOC)
GC_thread me;
DCL_LOCK_STATE;
#endif
if (parallel_initialized)return;
parallel_initialized=TRUE;
if (!GC_is_initialized)GC_init();
#if defined(CPPCHECK)&&!defined(GC_NO_THREADS_DISCOVERY)
GC_noop1((word)&GC_DllMain);
#endif
if (GC_win32_dll_threads){
set_need_to_lock();
}
#if defined(THREAD_LOCAL_ALLOC)
LOCK();
me=GC_lookup_thread_inner(GetCurrentThreadId());
CHECK_LOOKUP_MY_THREAD(me);
GC_init_thread_local(&me->tlfs);
UNLOCK();
#endif
}
#if defined(USE_PTHREAD_LOCKS)
GC_INNER void GC_lock(void)
{
pthread_mutex_lock(&GC_allocate_ml);
}
#endif
#if defined(THREAD_LOCAL_ALLOC)
GC_INNER void GC_mark_thread_local_free_lists(void)
{
int i;
GC_thread p;
for (i=0;i < THREAD_TABLE_SZ;++i){
for (p=GC_threads[i];0!=p;p=p->tm.next){
if (!KNOWN_FINISHED(p)){
#ifdef DEBUG_THREADS
GC_log_printf("Marking thread locals for 0x%x\n",(int)p->id);
#endif
GC_mark_thread_local_fls_for(&(p->tlfs));
}
}
}
}
#if defined(GC_ASSERTIONS)
void GC_check_tls(void)
{
int i;
GC_thread p;
for (i=0;i < THREAD_TABLE_SZ;++i){
for (p=GC_threads[i];0!=p;p=p->tm.next){
if (!KNOWN_FINISHED(p))
GC_check_tls_for(&(p->tlfs));
}
}
#if defined(USE_CUSTOM_SPECIFIC)
if (GC_thread_key!=0)
GC_check_tsd_marks(GC_thread_key);
#endif
}
#endif
#endif
#ifndef GC_NO_THREAD_REDIRECTS
#define CreateThread GC_CreateThread
#define ExitThread GC_ExitThread
#undef _beginthreadex
#define _beginthreadex GC_beginthreadex
#undef _endthreadex
#define _endthreadex GC_endthreadex
#endif
#endif
#ifndef GC_PTHREAD_START_STANDALONE
#if defined(__GNUC__)&&defined(__linux__)
#undef __EXCEPTIONS
#endif
#if defined(GC_PTHREADS)&&!defined(GC_WIN32_THREADS)
#include <pthread.h>
#include <sched.h>
GC_INNER_PTHRSTART void*GC_CALLBACK GC_inner_start_routine(
struct GC_stack_base*sb,void*arg)
{
void*(*start)(void*);
void*start_arg;
void*result;
volatile GC_thread me=
GC_start_rtn_prepare_thread(&start,&start_arg,sb,arg);
#ifndef NACL
pthread_cleanup_push(GC_thread_exit_proc,me);
#endif
result=(*start)(start_arg);
#if defined(DEBUG_THREADS)&&!defined(GC_PTHREAD_START_STANDALONE)
GC_log_printf("Finishing thread %p\n",(void*)pthread_self());
#endif
me->status=result;
GC_end_stubborn_change(me);
#ifndef NACL
pthread_cleanup_pop(1);
#endif
return result;
}
#endif
#endif
#ifndef GC_NO_THREAD_REDIRECTS
#define GC_PTHREAD_REDIRECTS_ONLY
#ifndef GC_PTHREAD_REDIRECTS_H
#define GC_PTHREAD_REDIRECTS_H
#if defined(GC_H)&&defined(GC_PTHREADS)
#ifndef GC_PTHREAD_REDIRECTS_ONLY
#include <pthread.h>
#ifndef GC_NO_DLOPEN
#include <dlfcn.h>
#endif
#ifndef GC_NO_PTHREAD_SIGMASK
#include <signal.h>
#endif
#ifdef __cplusplus
extern "C" {
#endif
#ifndef GC_SUSPEND_THREAD_ID
#define GC_SUSPEND_THREAD_ID pthread_t
#endif
#ifndef GC_NO_DLOPEN
GC_API void*GC_dlopen(const char*,int);
#endif
#ifndef GC_NO_PTHREAD_SIGMASK
#if defined(GC_PTHREAD_SIGMASK_NEEDED)||defined(_BSD_SOURCE)||defined(_GNU_SOURCE)||(_POSIX_C_SOURCE>=199506L)||(_XOPEN_SOURCE>=500)
GC_API int GC_pthread_sigmask(int,const sigset_t*,
sigset_t*);
#endif
#endif
#ifndef GC_PTHREAD_CREATE_CONST
#define GC_PTHREAD_CREATE_CONST const
#endif
GC_API int GC_pthread_create(pthread_t*,
GC_PTHREAD_CREATE_CONST pthread_attr_t*,
void*(*)(void*),void*);
GC_API int GC_pthread_join(pthread_t,void**);
GC_API int GC_pthread_detach(pthread_t);
#ifndef GC_NO_PTHREAD_CANCEL
GC_API int GC_pthread_cancel(pthread_t);
#endif
#if defined(GC_HAVE_PTHREAD_EXIT)&&!defined(GC_PTHREAD_EXIT_DECLARED)
#define GC_PTHREAD_EXIT_DECLARED
GC_API void GC_pthread_exit(void*)GC_PTHREAD_EXIT_ATTRIBUTE;
#endif
#ifdef __cplusplus
}
#endif
#endif
#if!defined(GC_NO_THREAD_REDIRECTS)&&!defined(GC_USE_LD_WRAP)
#undef pthread_create
#undef pthread_join
#undef pthread_detach
#define pthread_create GC_pthread_create
#define pthread_join GC_pthread_join
#define pthread_detach GC_pthread_detach
#ifndef GC_NO_PTHREAD_SIGMASK
#undef pthread_sigmask
#define pthread_sigmask GC_pthread_sigmask
#endif
#ifndef GC_NO_DLOPEN
#undef dlopen
#define dlopen GC_dlopen
#endif
#ifndef GC_NO_PTHREAD_CANCEL
#undef pthread_cancel
#define pthread_cancel GC_pthread_cancel
#endif
#ifdef GC_HAVE_PTHREAD_EXIT
#undef pthread_exit
#define pthread_exit GC_pthread_exit
#endif
#endif
#endif
#endif
#endif