blob: db7347fbe155d1c98d6606d63dc14fa9d630494e [file] [log] [blame]
/**
*
* 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);
}