/** * * Copyright: Copyright Digital Mars 2011 - 2012. * License: $(WEB www.boost.org/LICENSE_1_0.txt, Boost License 1.0). * Authors: Martin Nowak */ /* Copyright Digital Mars 2011. * Distributed under the Boost Software License, Version 1.0. * (See accompanying file LICENSE or copy at * http://www.boost.org/LICENSE_1_0.txt) */ module rt.tlsgc; import core.stdc.stdlib; static import rt.lifetime, rt.sections; /** * Per thread record to store thread associated data for garbage collection. */ struct Data { typeof(rt.sections.initTLSRanges()) tlsRanges; rt.lifetime.BlkInfo** blockInfoCache; } /** * Initialization hook, called FROM each thread. No assumptions about * module initialization state should be made. */ void* init() nothrow @nogc { auto data = cast(Data*).malloc(Data.sizeof); import core.exception; if ( data is null ) core.exception.onOutOfMemoryError(); *data = Data.init; // do module specific initialization data.tlsRanges = rt.sections.initTLSRanges(); data.blockInfoCache = &rt.lifetime.__blkcache_storage; return data; } /** * Finalization hook, called FOR each thread. No assumptions about * module initialization state should be made. */ void destroy(void* data) nothrow @nogc { // do module specific finalization rt.sections.finiTLSRanges((cast(Data*)data).tlsRanges); .free(data); } alias void delegate(void* pstart, void* pend) nothrow ScanDg; /** * GC scan hook, called FOR each thread. Can be used to scan * additional thread local memory. */ void scan(void* data, scope ScanDg dg) nothrow { // do module specific marking rt.sections.scanTLSRanges((cast(Data*)data).tlsRanges, dg); } alias int delegate(void* addr) nothrow IsMarkedDg; /** * GC sweep hook, called FOR each thread. Can be used to free * additional thread local memory or associated data structures. Note * that only memory allocated from the GC can have marks. */ void processGCMarks(void* data, scope IsMarkedDg dg) nothrow { // do module specific sweeping rt.lifetime.processGCMarks(*(cast(Data*)data).blockInfoCache, dg); } |