mirror of
https://github.com/bitwiseworks/gcc-os2.git
synced 2026-02-13 21:54:40 +00:00
Source URL: git://gcc.gnu.org/git/gcc.git Source Commit: 3e7b85061947bdc7c7465743ba90734566860821
185 lines
5.1 KiB
D
185 lines
5.1 KiB
D
// Bionic-specific support for sections.
|
|
// Copyright (C) 2019 Free Software Foundation, Inc.
|
|
|
|
// GCC is free software; you can redistribute it and/or modify it under
|
|
// the terms of the GNU General Public License as published by the Free
|
|
// Software Foundation; either version 3, or (at your option) any later
|
|
// version.
|
|
|
|
// GCC is distributed in the hope that it will be useful, but WITHOUT ANY
|
|
// WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
// FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
// for more details.
|
|
|
|
// Under Section 7 of GPL version 3, you are granted additional
|
|
// permissions described in the GCC Runtime Library Exception, version
|
|
// 3.1, as published by the Free Software Foundation.
|
|
|
|
// You should have received a copy of the GNU General Public License and
|
|
// a copy of the GCC Runtime Library Exception along with this program;
|
|
// see the files COPYING3 and COPYING.RUNTIME respectively. If not, see
|
|
// <http://www.gnu.org/licenses/>.
|
|
|
|
module gcc.sections.android;
|
|
|
|
version (CRuntime_Bionic):
|
|
|
|
// debug = PRINTF;
|
|
debug(PRINTF) import core.stdc.stdio;
|
|
import core.stdc.stdlib : malloc, free;
|
|
import rt.deh, rt.minfo;
|
|
import core.sys.posix.pthread;
|
|
import core.stdc.stdlib : calloc;
|
|
import core.stdc.string : memcpy;
|
|
|
|
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 nothrow @nogc
|
|
{
|
|
return _moduleGroup.modules;
|
|
}
|
|
|
|
@property ref inout(ModuleGroup) moduleGroup() inout nothrow @nogc
|
|
{
|
|
return _moduleGroup;
|
|
}
|
|
|
|
@property immutable(FuncTable)[] ehTables() const nothrow @nogc
|
|
{
|
|
auto pbeg = cast(immutable(FuncTable)*)&__start_deh;
|
|
auto pend = cast(immutable(FuncTable)*)&__stop_deh;
|
|
return pbeg[0 .. pend - pbeg];
|
|
}
|
|
|
|
@property inout(void[])[] gcRanges() inout nothrow @nogc
|
|
{
|
|
return _gcRanges[];
|
|
}
|
|
|
|
private:
|
|
ModuleGroup _moduleGroup;
|
|
void[][1] _gcRanges;
|
|
}
|
|
|
|
void initSections() nothrow @nogc
|
|
{
|
|
pthread_key_create(&_tlsKey, null);
|
|
|
|
auto mbeg = cast(immutable ModuleInfo**)&__start_minfo;
|
|
auto mend = cast(immutable ModuleInfo**)&__stop_minfo;
|
|
_sections.moduleGroup = ModuleGroup(mbeg[0 .. mend - mbeg]);
|
|
|
|
auto pbeg = cast(void*)&_tlsend;
|
|
auto pend = cast(void*)&__bss_end__;
|
|
// _tlsend is a 32-bit int and may not be 64-bit void*-aligned, so align pbeg.
|
|
version (D_LP64) pbeg = cast(void*)(cast(size_t)(pbeg + 7) & ~cast(size_t)7);
|
|
_sections._gcRanges[0] = pbeg[0 .. pend - pbeg];
|
|
}
|
|
|
|
void finiSections() nothrow @nogc
|
|
{
|
|
pthread_key_delete(_tlsKey);
|
|
}
|
|
|
|
void[]* initTLSRanges() nothrow @nogc
|
|
{
|
|
return &getTLSBlock();
|
|
}
|
|
|
|
void finiTLSRanges(void[]* rng) nothrow @nogc
|
|
{
|
|
.free(rng.ptr);
|
|
.free(rng);
|
|
}
|
|
|
|
void scanTLSRanges(void[]* rng, scope void delegate(void* pbeg, void* pend) nothrow dg) nothrow
|
|
{
|
|
dg(rng.ptr, rng.ptr + rng.length);
|
|
}
|
|
|
|
/* NOTE: The Bionic C library ignores thread-local data stored in the normal
|
|
* .tbss/.tdata ELF sections, which are marked with the SHF_TLS/STT_TLS
|
|
* flags. So instead we roll our own by keeping TLS data in the
|
|
* .tdata/.tbss sections but removing the SHF_TLS/STT_TLS flags, and
|
|
* access the TLS data using this function and the _tlsstart/_tlsend
|
|
* symbols as delimiters.
|
|
*
|
|
* This function is called by the code emitted by the compiler. It
|
|
* is expected to translate an address in the TLS static data to
|
|
* the corresponding address in the TLS dynamic per-thread data.
|
|
*/
|
|
|
|
extern(C) void* __tls_get_addr( void* p ) nothrow @nogc
|
|
{
|
|
debug(PRINTF) printf(" __tls_get_addr input - %p\n", p);
|
|
immutable offset = cast(size_t)(p - cast(void*)&_tlsstart);
|
|
auto tls = getTLSBlockAlloc();
|
|
assert(offset < tls.length);
|
|
return tls.ptr + offset;
|
|
}
|
|
|
|
private:
|
|
|
|
__gshared pthread_key_t _tlsKey;
|
|
|
|
ref void[] getTLSBlock() nothrow @nogc
|
|
{
|
|
auto pary = cast(void[]*)pthread_getspecific(_tlsKey);
|
|
if (pary is null)
|
|
{
|
|
pary = cast(void[]*).calloc(1, (void[]).sizeof);
|
|
if (pthread_setspecific(_tlsKey, pary) != 0)
|
|
{
|
|
import core.stdc.stdio;
|
|
perror("pthread_setspecific failed with");
|
|
assert(0);
|
|
}
|
|
}
|
|
return *pary;
|
|
}
|
|
|
|
ref void[] getTLSBlockAlloc() nothrow @nogc
|
|
{
|
|
auto pary = &getTLSBlock();
|
|
if (!pary.length)
|
|
{
|
|
auto pbeg = cast(void*)&_tlsstart;
|
|
auto pend = cast(void*)&_tlsend;
|
|
auto p = .malloc(pend - pbeg);
|
|
memcpy(p, pbeg, pend - pbeg);
|
|
*pary = p[0 .. pend - pbeg];
|
|
}
|
|
return *pary;
|
|
}
|
|
|
|
__gshared SectionGroup _sections;
|
|
|
|
extern(C)
|
|
{
|
|
/* Symbols created by the compiler/linker and inserted into the
|
|
* object file that 'bracket' sections.
|
|
*/
|
|
extern __gshared
|
|
{
|
|
void* __start_deh;
|
|
void* __stop_deh;
|
|
void* __start_minfo;
|
|
void* __stop_minfo;
|
|
|
|
size_t __bss_end__;
|
|
|
|
int _tlsstart;
|
|
int _tlsend;
|
|
}
|
|
}
|