mirror of git://gcc.gnu.org/git/gcc.git
174 lines
4.2 KiB
D
174 lines
4.2 KiB
D
/**
|
|
* Written in the D programming language.
|
|
* This module provides Win32-specific support for sections.
|
|
*
|
|
* Copyright: Copyright Digital Mars 2008 - 2012.
|
|
* License: Distributed under the
|
|
* $(LINK2 http://www.boost.org/LICENSE_1_0.txt, Boost Software License 1.0).
|
|
* (See accompanying file LICENSE)
|
|
* Authors: Walter Bright, Sean Kelly, Martin Nowak
|
|
* Source: $(DRUNTIMESRC src/rt/_sections_win32.d)
|
|
*/
|
|
|
|
module rt.sections_win32;
|
|
|
|
version (CRuntime_DigitalMars):
|
|
|
|
// debug = PRINTF;
|
|
debug(PRINTF) import core.stdc.stdio;
|
|
import rt.minfo;
|
|
import core.stdc.stdlib : malloc, free;
|
|
|
|
struct SectionGroup
|
|
{
|
|
static int opApply(scope int delegate(ref SectionGroup) dg)
|
|
{
|
|
return dg(_sections);
|
|
}
|
|
|
|
static int opApplyReverse(scope int delegate(ref SectionGroup) dg)
|
|
{
|
|
return dg(_sections);
|
|
}
|
|
|
|
@property immutable(ModuleInfo*)[] modules() const
|
|
{
|
|
return _moduleGroup.modules;
|
|
}
|
|
|
|
@property ref inout(ModuleGroup) moduleGroup() inout
|
|
{
|
|
return _moduleGroup;
|
|
}
|
|
|
|
@property inout(void[])[] gcRanges() inout
|
|
{
|
|
return _gcRanges[];
|
|
}
|
|
|
|
private:
|
|
ModuleGroup _moduleGroup;
|
|
void[][] _gcRanges;
|
|
}
|
|
|
|
shared(bool) conservative;
|
|
|
|
void initSections() nothrow @nogc
|
|
{
|
|
_sections._moduleGroup = ModuleGroup(getModuleInfos());
|
|
|
|
import rt.sections;
|
|
conservative = !scanDataSegPrecisely();
|
|
|
|
if (conservative)
|
|
{
|
|
_sections._gcRanges = (cast(void[]*) malloc(2 * (void[]).sizeof))[0..2];
|
|
|
|
auto databeg = cast(void*)&_xi_a;
|
|
auto dataend = cast(void*)_moduleinfo_array.ptr;
|
|
_sections._gcRanges[0] = databeg[0 .. dataend - databeg];
|
|
|
|
// skip module info and CONST segment
|
|
auto bssbeg = cast(void*)&_edata;
|
|
auto bssend = cast(void*)&_end;
|
|
_sections._gcRanges[1] = bssbeg[0 .. bssend - bssbeg];
|
|
}
|
|
else
|
|
{
|
|
size_t count = &_DPend - &_DPbegin;
|
|
auto ranges = cast(void[]*) malloc(count * (void[]).sizeof);
|
|
size_t r = 0;
|
|
void* prev = null;
|
|
for (size_t i = 0; i < count; i++)
|
|
{
|
|
void* addr = (&_DPbegin)[i];
|
|
if (prev + (void*).sizeof == addr)
|
|
ranges[r-1] = ranges[r-1].ptr[0 .. ranges[r-1].length + (void*).sizeof];
|
|
else
|
|
ranges[r++] = (cast(void**)addr)[0..1];
|
|
prev = addr;
|
|
}
|
|
_sections._gcRanges = ranges[0..r];
|
|
}
|
|
}
|
|
|
|
void finiSections() nothrow @nogc
|
|
{
|
|
free(_sections._gcRanges.ptr);
|
|
}
|
|
|
|
void[] initTLSRanges() nothrow @nogc
|
|
{
|
|
auto pbeg = cast(void*)&_tlsstart;
|
|
auto pend = cast(void*)&_tlsend;
|
|
return pbeg[0 .. pend - pbeg];
|
|
}
|
|
|
|
void finiTLSRanges(void[] rng) nothrow @nogc
|
|
{
|
|
}
|
|
|
|
void scanTLSRanges(void[] rng, scope void delegate(void* pbeg, void* pend) nothrow dg) nothrow
|
|
{
|
|
if (conservative)
|
|
{
|
|
dg(rng.ptr, rng.ptr + rng.length);
|
|
}
|
|
else
|
|
{
|
|
for (auto p = &_TPbegin; p < &_TPend; )
|
|
{
|
|
uint beg = *p++;
|
|
uint end = beg + cast(uint)((void*).sizeof);
|
|
while (p < &_TPend && *p == end)
|
|
{
|
|
end += (void*).sizeof;
|
|
p++;
|
|
}
|
|
dg(rng.ptr + beg, rng.ptr + end);
|
|
}
|
|
}
|
|
}
|
|
|
|
private:
|
|
|
|
__gshared SectionGroup _sections;
|
|
|
|
// Windows: this gets initialized by minit.asm
|
|
extern(C) __gshared immutable(ModuleInfo*)[] _moduleinfo_array;
|
|
extern(C) void _minit() nothrow @nogc;
|
|
|
|
immutable(ModuleInfo*)[] getModuleInfos() nothrow @nogc
|
|
out (result)
|
|
{
|
|
foreach (m; result)
|
|
assert(m !is null);
|
|
}
|
|
body
|
|
{
|
|
// _minit directly alters the global _moduleinfo_array
|
|
_minit();
|
|
return _moduleinfo_array;
|
|
}
|
|
|
|
extern(C)
|
|
{
|
|
extern __gshared
|
|
{
|
|
int _xi_a; // &_xi_a just happens to be start of data segment
|
|
int _edata; // &_edata is start of BSS segment
|
|
int _end; // &_end is past end of BSS
|
|
|
|
void* _DPbegin; // first entry in the array of pointers addresses
|
|
void* _DPend; // &_DPend points after last entry of array
|
|
uint _TPbegin; // first entry in the array of TLS offsets of pointers
|
|
uint _TPend; // &_DPend points after last entry of array
|
|
}
|
|
|
|
extern
|
|
{
|
|
int _tlsstart;
|
|
int _tlsend;
|
|
}
|
|
}
|