|  | /* | 
|  | * Copyright (C) 2019 The Android Open Source Project | 
|  | * All rights reserved. | 
|  | * | 
|  | * Redistribution and use in source and binary forms, with or without | 
|  | * modification, are permitted provided that the following conditions | 
|  | * are met: | 
|  | *  * Redistributions of source code must retain the above copyright | 
|  | *    notice, this list of conditions and the following disclaimer. | 
|  | *  * Redistributions in binary form must reproduce the above copyright | 
|  | *    notice, this list of conditions and the following disclaimer in | 
|  | *    the documentation and/or other materials provided with the | 
|  | *    distribution. | 
|  | * | 
|  | * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | 
|  | * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | 
|  | * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS | 
|  | * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE | 
|  | * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, | 
|  | * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, | 
|  | * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS | 
|  | * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED | 
|  | * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | 
|  | * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT | 
|  | * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | 
|  | * SUCH DAMAGE. | 
|  | */ | 
|  |  | 
|  | #include "linker_tls.h" | 
|  |  | 
|  | #include <vector> | 
|  |  | 
|  | #include "async_safe/CHECK.h" | 
|  | #include "private/ScopedRWLock.h" | 
|  | #include "private/ScopedSignalBlocker.h" | 
|  | #include "private/bionic_defs.h" | 
|  | #include "private/bionic_elf_tls.h" | 
|  | #include "private/bionic_globals.h" | 
|  | #include "private/linker_native_bridge.h" | 
|  | #include "linker_main.h" | 
|  | #include "linker_soinfo.h" | 
|  |  | 
|  | static bool g_static_tls_finished; | 
|  | static std::vector<TlsModule> g_tls_modules; | 
|  |  | 
|  | static size_t get_unused_module_index() { | 
|  | for (size_t i = 0; i < g_tls_modules.size(); ++i) { | 
|  | if (g_tls_modules[i].soinfo_ptr == nullptr) { | 
|  | return i; | 
|  | } | 
|  | } | 
|  | g_tls_modules.push_back({}); | 
|  | __libc_shared_globals()->tls_modules.module_count = g_tls_modules.size(); | 
|  | __libc_shared_globals()->tls_modules.module_table = g_tls_modules.data(); | 
|  | return g_tls_modules.size() - 1; | 
|  | } | 
|  |  | 
|  | static void register_tls_module(soinfo* si, size_t static_offset) { | 
|  | TlsModules& libc_modules = __libc_shared_globals()->tls_modules; | 
|  |  | 
|  | // The global TLS module table points at the std::vector of modules declared | 
|  | // in this file, so acquire a write lock before modifying the std::vector. | 
|  | ScopedSignalBlocker ssb; | 
|  | ScopedWriteLock locker(&libc_modules.rwlock); | 
|  |  | 
|  | size_t module_idx = get_unused_module_index(); | 
|  |  | 
|  | soinfo_tls* si_tls = si->get_tls(); | 
|  | si_tls->module_id = __tls_module_idx_to_id(module_idx); | 
|  |  | 
|  | const size_t new_generation = ++libc_modules.generation; | 
|  | __libc_tls_generation_copy = new_generation; | 
|  | if (libc_modules.generation_libc_so != nullptr) { | 
|  | *libc_modules.generation_libc_so = new_generation; | 
|  | } | 
|  |  | 
|  | g_tls_modules[module_idx] = { | 
|  | .segment = si_tls->segment, | 
|  | .static_offset = static_offset, | 
|  | .first_generation = new_generation, | 
|  | .soinfo_ptr = si, | 
|  | }; | 
|  | } | 
|  |  | 
|  | static void unregister_tls_module(soinfo* si) { | 
|  | ScopedSignalBlocker ssb; | 
|  | ScopedWriteLock locker(&__libc_shared_globals()->tls_modules.rwlock); | 
|  |  | 
|  | soinfo_tls* si_tls = si->get_tls(); | 
|  | TlsModule& mod = g_tls_modules[__tls_module_id_to_idx(si_tls->module_id)]; | 
|  | CHECK(mod.static_offset == SIZE_MAX); | 
|  | CHECK(mod.soinfo_ptr == si); | 
|  | mod = {}; | 
|  | si_tls->module_id = kTlsUninitializedModuleId; | 
|  | } | 
|  |  | 
|  | // The reference is valid until a TLS module is registered or unregistered. | 
|  | const TlsModule& get_tls_module(size_t module_id) { | 
|  | size_t module_idx = __tls_module_id_to_idx(module_id); | 
|  | CHECK(module_idx < g_tls_modules.size()); | 
|  | return g_tls_modules[module_idx]; | 
|  | } | 
|  |  | 
|  | __BIONIC_WEAK_FOR_NATIVE_BRIDGE | 
|  | extern "C" void __linker_reserve_bionic_tls_in_static_tls() { | 
|  | __libc_shared_globals()->static_tls_layout.reserve_bionic_tls(); | 
|  | } | 
|  |  | 
|  | void linker_setup_exe_static_tls(const char* progname) { | 
|  | soinfo* somain = solist_get_somain(); | 
|  | StaticTlsLayout& layout = __libc_shared_globals()->static_tls_layout; | 
|  | if (somain->get_tls() == nullptr) { | 
|  | layout.reserve_exe_segment_and_tcb(nullptr, progname); | 
|  | } else { | 
|  | register_tls_module(somain, layout.reserve_exe_segment_and_tcb(&somain->get_tls()->segment, progname)); | 
|  | } | 
|  |  | 
|  | // The pthread key data is located at the very front of bionic_tls. As a | 
|  | // temporary workaround, allocate bionic_tls just after the thread pointer so | 
|  | // Golang can find its pthread key, as long as the executable's TLS segment is | 
|  | // small enough. Specifically, Golang scans forward 384 words from the TP on | 
|  | // ARM. | 
|  | //  - http://b/118381796 | 
|  | //  - https://github.com/golang/go/issues/29674 | 
|  | __linker_reserve_bionic_tls_in_static_tls(); | 
|  | } | 
|  |  | 
|  | void linker_finalize_static_tls() { | 
|  | g_static_tls_finished = true; | 
|  | __libc_shared_globals()->static_tls_layout.finish_layout(); | 
|  | } | 
|  |  | 
|  | void register_soinfo_tls(soinfo* si) { | 
|  | soinfo_tls* si_tls = si->get_tls(); | 
|  | if (si_tls == nullptr || si_tls->module_id != kTlsUninitializedModuleId) { | 
|  | return; | 
|  | } | 
|  | size_t static_offset = SIZE_MAX; | 
|  | if (!g_static_tls_finished) { | 
|  | StaticTlsLayout& layout = __libc_shared_globals()->static_tls_layout; | 
|  | static_offset = layout.reserve_solib_segment(si_tls->segment); | 
|  | } | 
|  | register_tls_module(si, static_offset); | 
|  | } | 
|  |  | 
|  | void unregister_soinfo_tls(soinfo* si) { | 
|  | soinfo_tls* si_tls = si->get_tls(); | 
|  | if (si_tls == nullptr || si_tls->module_id == kTlsUninitializedModuleId) { | 
|  | return; | 
|  | } | 
|  | return unregister_tls_module(si); | 
|  | } |