aboutsummaryrefslogtreecommitdiff
path: root/libphobos/libdruntime/gcc/sections/android.d
diff options
context:
space:
mode:
authorIain Buclaw <ibuclaw@gdcproject.org>2019-04-13 15:29:15 +0000
committerIain Buclaw <ibuclaw@gcc.gnu.org>2019-04-13 15:29:15 +0000
commit8b6518285b87fd282573fa5d7d6f7414d58dc96b (patch)
tree513a3346a5fe38d485ac1a2ca28fc7336fcf3e33 /libphobos/libdruntime/gcc/sections/android.d
parent151c5c0b80a30ba4316c86fcd0a7bf232a4ff127 (diff)
downloadgcc-8b6518285b87fd282573fa5d7d6f7414d58dc96b.zip
gcc-8b6518285b87fd282573fa5d7d6f7414d58dc96b.tar.gz
gcc-8b6518285b87fd282573fa5d7d6f7414d58dc96b.tar.bz2
libphobos: Move rt.sections modules to gcc.sections
These modules depend on a mixture between how the compiler emits run-time module information, and what functions are exposed by the platform to inquire about loaded global and thread-local data sections. As the upstream implementation is written to work only with how the reference D compiler writes out data, much of what is present does not apply to the GCC D front-end. So it has been moved to a non-upstream location in the source tree, where most of it will be rewritten once each port has been completed. The only tested module sections/elf_shared.d has been cleaned up so that all deprecated declarations have been removed, as well as the brittle module collision checking, which required bss_sections.c. All other ports have been left unchanged apart from a commonizing of attributes. libphobos/ChangeLog: 2019-04-13 Iain Buclaw <ibuclaw@gdcproject.org> * libdruntime/Makefile.am (DRUNTIME_CSOURCES): Remove bss_sections.c. (DRUNTIME_DSOURCES): Rename rt/sections_* modules to gcc/sections/*. * libdruntime/Makefile.in: Regenerate. * libdruntime/gcc/sections/android.d: New file. * libdruntime/gcc/sections/elf_shared.d: New file. * libdruntime/gcc/sections/osx.d: New file. * libdruntime/gcc/sections/package.d: New file. * libdruntime/gcc/sections/solaris.d: New file. * libdruntime/gcc/sections/win32.d: New file. * libdruntime/gcc/sections/win64.d: New file. * libdruntime/rt/bss_section.c: Remove. * libdruntime/rt/sections.d: Publicly import gcc.sections. * libdruntime/rt/sections_android.d: Remove. * libdruntime/rt/sections_elf_shared.d: Remove. * libdruntime/rt/sections_osx.d: Remove. * libdruntime/rt/sections_solaris.d: Remove. * libdruntime/rt/sections_win32.d: Remove. * libdruntime/rt/sections_win64.d: Remove. From-SVN: r270341
Diffstat (limited to 'libphobos/libdruntime/gcc/sections/android.d')
-rw-r--r--libphobos/libdruntime/gcc/sections/android.d184
1 files changed, 184 insertions, 0 deletions
diff --git a/libphobos/libdruntime/gcc/sections/android.d b/libphobos/libdruntime/gcc/sections/android.d
new file mode 100644
index 0000000..f00bb89
--- /dev/null
+++ b/libphobos/libdruntime/gcc/sections/android.d
@@ -0,0 +1,184 @@
+// 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;
+ }
+}