| The Android Open Source Project | 1dc9e47 | 2009-03-03 19:28:35 -0800 | [diff] [blame] | 1 | /* | 
|  | 2 | * Copyright (C) 2008 The Android Open Source Project | 
|  | 3 | * All rights reserved. | 
|  | 4 | * | 
|  | 5 | * Redistribution and use in source and binary forms, with or without | 
|  | 6 | * modification, are permitted provided that the following conditions | 
|  | 7 | * are met: | 
|  | 8 | *  * Redistributions of source code must retain the above copyright | 
|  | 9 | *    notice, this list of conditions and the following disclaimer. | 
|  | 10 | *  * Redistributions in binary form must reproduce the above copyright | 
|  | 11 | *    notice, this list of conditions and the following disclaimer in | 
|  | 12 | *    the documentation and/or other materials provided with the | 
|  | 13 | *    distribution. | 
|  | 14 | * | 
|  | 15 | * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | 
|  | 16 | * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | 
|  | 17 | * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS | 
|  | 18 | * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE | 
|  | 19 | * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, | 
|  | 20 | * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, | 
|  | 21 | * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS | 
|  | 22 | * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED | 
|  | 23 | * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | 
|  | 24 | * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT | 
|  | 25 | * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | 
|  | 26 | * SUCH DAMAGE. | 
|  | 27 | */ | 
|  | 28 |  | 
| Elliott Hughes | cbc80ba | 2018-02-13 14:26:29 -0800 | [diff] [blame] | 29 | #pragma once | 
| The Android Open Source Project | 1dc9e47 | 2009-03-03 19:28:35 -0800 | [diff] [blame] | 30 |  | 
| Dimitry Ivanov | 4a2c5aa | 2015-12-10 16:08:14 -0800 | [diff] [blame] | 31 | #include <dlfcn.h> | 
| Elliott Hughes | afab3ff | 2015-07-28 14:58:37 -0700 | [diff] [blame] | 32 | #include <android/dlext.h> | 
| Nick Kralevich | 9ec0f03 | 2012-02-28 10:40:00 -0800 | [diff] [blame] | 33 | #include <elf.h> | 
| Dmitriy Ivanov | 0d15094 | 2014-08-22 12:25:04 -0700 | [diff] [blame] | 34 | #include <inttypes.h> | 
| Pavel Chupin | b7beb69 | 2012-08-17 12:53:29 +0400 | [diff] [blame] | 35 | #include <link.h> | 
| Dmitriy Ivanov | d59e500 | 2014-05-09 09:10:14 -0700 | [diff] [blame] | 36 | #include <sys/stat.h> | 
| Elliott Hughes | afab3ff | 2015-07-28 14:58:37 -0700 | [diff] [blame] | 37 | #include <unistd.h> | 
| Elliott Hughes | 4688279 | 2012-08-03 16:49:39 -0700 | [diff] [blame] | 38 |  | 
| Elliott Hughes | afab3ff | 2015-07-28 14:58:37 -0700 | [diff] [blame] | 39 | #include "private/bionic_page.h" | 
| Dmitriy Ivanov | d59e500 | 2014-05-09 09:10:14 -0700 | [diff] [blame] | 40 | #include "linked_list.h" | 
| Dimitry Ivanov | b943f30 | 2016-08-03 16:00:10 -0700 | [diff] [blame] | 41 | #include "linker_common_types.h" | 
| Dimitry Ivanov | b996d60 | 2016-07-11 18:11:39 -0700 | [diff] [blame] | 42 | #include "linker_logger.h" | 
| Dimitry Ivanov | 48ec288 | 2016-08-04 11:50:36 -0700 | [diff] [blame] | 43 | #include "linker_soinfo.h" | 
| Elliott Hughes | 650be4e | 2013-03-05 18:47:58 -0800 | [diff] [blame] | 44 |  | 
| Dmitriy Ivanov | aae859c | 2015-03-31 11:14:03 -0700 | [diff] [blame] | 45 | #include <string> | 
| Dmitriy Ivanov | 2a81536 | 2015-04-09 13:42:33 -0700 | [diff] [blame] | 46 | #include <vector> | 
| Dmitriy Ivanov | aae859c | 2015-03-31 11:14:03 -0700 | [diff] [blame] | 47 |  | 
| Elliott Hughes | 0266ae5 | 2014-02-10 17:46:57 -0800 | [diff] [blame] | 48 | #if defined(__LP64__) | 
|  | 49 | #define ELFW(what) ELF64_ ## what | 
|  | 50 | #else | 
|  | 51 | #define ELFW(what) ELF32_ ## what | 
|  | 52 | #endif | 
| Elliott Hughes | 650be4e | 2013-03-05 18:47:58 -0800 | [diff] [blame] | 53 |  | 
| Chris Dearman | 9918665 | 2014-02-06 20:36:51 -0800 | [diff] [blame] | 54 | // mips64 interprets Elf64_Rel structures' r_info field differently. | 
|  | 55 | // bionic (like other C libraries) has macros that assume regular ELF files, | 
|  | 56 | // but the dynamic linker needs to be able to load mips64 ELF files. | 
|  | 57 | #if defined(__mips__) && defined(__LP64__) | 
|  | 58 | #undef ELF64_R_SYM | 
|  | 59 | #undef ELF64_R_TYPE | 
|  | 60 | #undef ELF64_R_INFO | 
|  | 61 | #define ELF64_R_SYM(info)   (((info) >> 0) & 0xffffffff) | 
|  | 62 | #define ELF64_R_SSYM(info)  (((info) >> 32) & 0xff) | 
|  | 63 | #define ELF64_R_TYPE3(info) (((info) >> 40) & 0xff) | 
|  | 64 | #define ELF64_R_TYPE2(info) (((info) >> 48) & 0xff) | 
|  | 65 | #define ELF64_R_TYPE(info)  (((info) >> 56) & 0xff) | 
|  | 66 | #endif | 
|  | 67 |  | 
| Bernhard Rosenkränzer | 59ba65e | 2016-08-19 21:57:12 +0200 | [diff] [blame] | 68 | #define SUPPORTED_DT_FLAGS_1 (DF_1_NOW | DF_1_GLOBAL | DF_1_NODELETE | DF_1_PIE) | 
| Dmitriy Ivanov | d225a5e | 2014-08-28 14:12:12 -0700 | [diff] [blame] | 69 |  | 
| Dmitriy Ivanov | 2a81536 | 2015-04-09 13:42:33 -0700 | [diff] [blame] | 70 | // Class used construct version dependency graph. | 
|  | 71 | class VersionTracker { | 
|  | 72 | public: | 
|  | 73 | VersionTracker() = default; | 
|  | 74 | bool init(const soinfo* si_from); | 
|  | 75 |  | 
|  | 76 | const version_info* get_version_info(ElfW(Versym) source_symver) const; | 
|  | 77 | private: | 
|  | 78 | bool init_verneed(const soinfo* si_from); | 
|  | 79 | bool init_verdef(const soinfo* si_from); | 
|  | 80 | void add_version_info(size_t source_index, ElfW(Word) elf_hash, | 
|  | 81 | const char* ver_name, const soinfo* target_si); | 
|  | 82 |  | 
|  | 83 | std::vector<version_info> version_infos; | 
|  | 84 |  | 
|  | 85 | DISALLOW_COPY_AND_ASSIGN(VersionTracker); | 
|  | 86 | }; | 
|  | 87 |  | 
| Dmitriy Ivanov | 2a81536 | 2015-04-09 13:42:33 -0700 | [diff] [blame] | 88 | bool soinfo_do_lookup(soinfo* si_from, const char* name, const version_info* vi, | 
| Dimitry Ivanov | b943f30 | 2016-08-03 16:00:10 -0700 | [diff] [blame] | 89 | soinfo** si_found_in, const soinfo_list_t& global_group, | 
|  | 90 | const soinfo_list_t& local_group, const ElfW(Sym)** symbol); | 
| Dmitriy Ivanov | 114ff69 | 2015-01-14 11:36:38 -0800 | [diff] [blame] | 91 |  | 
|  | 92 | enum RelocationKind { | 
|  | 93 | kRelocAbsolute = 0, | 
|  | 94 | kRelocRelative, | 
|  | 95 | kRelocCopy, | 
|  | 96 | kRelocSymbol, | 
| Vic Yang | 521ad07 | 2019-07-25 11:05:33 -0700 | [diff] [blame] | 97 | kRelocSymbolCached, | 
| Dmitriy Ivanov | 114ff69 | 2015-01-14 11:36:38 -0800 | [diff] [blame] | 98 | kRelocMax | 
|  | 99 | }; | 
|  | 100 |  | 
|  | 101 | void count_relocation(RelocationKind kind); | 
|  | 102 |  | 
| Vic Yang | 542db79 | 2019-07-25 10:39:27 -0700 | [diff] [blame] | 103 | void print_linker_stats(); | 
|  | 104 |  | 
| Ryan Prichard | cf9ed12 | 2019-06-04 20:56:56 -0700 | [diff] [blame] | 105 | soinfo* get_libdl_info(const soinfo& linker_si); | 
| The Android Open Source Project | 1dc9e47 | 2009-03-03 19:28:35 -0800 | [diff] [blame] | 106 |  | 
| Evgenii Stepanov | 0a3637d | 2016-07-06 13:20:59 -0700 | [diff] [blame] | 107 | soinfo* find_containing_library(const void* p); | 
|  | 108 |  | 
| Ryan Prichard | 8f639a4 | 2018-10-01 23:10:05 -0700 | [diff] [blame] | 109 | int open_executable(const char* path, off64_t* file_offset, std::string* realpath); | 
|  | 110 |  | 
| Elliott Hughes | a4aafd1 | 2014-01-13 16:37:47 -0800 | [diff] [blame] | 111 | void do_android_get_LD_LIBRARY_PATH(char*, size_t); | 
| Elliott Hughes | cade4c3 | 2012-12-20 14:42:14 -0800 | [diff] [blame] | 112 | void do_android_update_LD_LIBRARY_PATH(const char* ld_library_path); | 
| Dimitry Ivanov | d9e427c | 2016-11-22 16:55:25 -0800 | [diff] [blame] | 113 | void* do_dlopen(const char* name, | 
|  | 114 | int flags, | 
|  | 115 | const android_dlextinfo* extinfo, | 
|  | 116 | const void* caller_addr); | 
|  | 117 |  | 
| Dimitry Ivanov | d88e1f3 | 2016-03-24 15:30:30 -0700 | [diff] [blame] | 118 | int do_dlclose(void* handle); | 
| David 'Digit' Turner | 1608416 | 2012-06-12 16:25:37 +0200 | [diff] [blame] | 119 |  | 
| Dmitriy Ivanov | 7271caf | 2015-06-29 14:48:25 -0700 | [diff] [blame] | 120 | int do_dl_iterate_phdr(int (*cb)(dl_phdr_info* info, size_t size, void* data), void* data); | 
|  | 121 |  | 
| Dimitry Ivanov | d9e427c | 2016-11-22 16:55:25 -0800 | [diff] [blame] | 122 | #if defined(__arm__) | 
|  | 123 | _Unwind_Ptr do_dl_unwind_find_exidx(_Unwind_Ptr pc, int* pcount); | 
|  | 124 | #endif | 
|  | 125 |  | 
|  | 126 | bool do_dlsym(void* handle, const char* sym_name, | 
|  | 127 | const char* sym_ver, | 
|  | 128 | const void* caller_addr, | 
|  | 129 | void** symbol); | 
| Dmitriy Ivanov | 42d5fcb | 2015-10-29 17:01:24 -0700 | [diff] [blame] | 130 |  | 
| Dimitry Ivanov | 4a2c5aa | 2015-12-10 16:08:14 -0800 | [diff] [blame] | 131 | int do_dladdr(const void* addr, Dl_info* info); | 
| Elliott Hughes | d23736e | 2012-11-01 15:16:56 -0700 | [diff] [blame] | 132 |  | 
| Evgenii Stepanov | 0a3637d | 2016-07-06 13:20:59 -0700 | [diff] [blame] | 133 | // void ___cfi_slowpath(uint64_t CallSiteTypeId, void *Ptr, void *Ret); | 
|  | 134 | // void ___cfi_slowpath_diag(uint64_t CallSiteTypeId, void *Ptr, void *DiagData, void *Ret); | 
|  | 135 | void ___cfi_fail(uint64_t CallSiteTypeId, void* Ptr, void *DiagData, void *Ret); | 
|  | 136 |  | 
| Elliott Hughes | ff1428a | 2018-11-12 16:01:37 -0800 | [diff] [blame] | 137 | void set_application_target_sdk_version(int target); | 
|  | 138 | int get_application_target_sdk_version(); | 
| Dmitriy Ivanov | 79fd668 | 2015-05-21 17:43:49 -0700 | [diff] [blame] | 139 |  | 
| Dimitry Ivanov | 41fd295 | 2016-05-09 17:37:39 -0700 | [diff] [blame] | 140 | enum { | 
|  | 141 | /* A regular namespace is the namespace with a custom search path that does | 
|  | 142 | * not impose any restrictions on the location of native libraries. | 
|  | 143 | */ | 
|  | 144 | ANDROID_NAMESPACE_TYPE_REGULAR = 0, | 
|  | 145 |  | 
|  | 146 | /* An isolated namespace requires all the libraries to be on the search path | 
|  | 147 | * or under permitted_when_isolated_path. The search path is the union of | 
|  | 148 | * ld_library_path and default_library_path. | 
|  | 149 | */ | 
|  | 150 | ANDROID_NAMESPACE_TYPE_ISOLATED = 1, | 
|  | 151 |  | 
|  | 152 | /* The shared namespace clones the list of libraries of the caller namespace upon creation | 
|  | 153 | * which means that they are shared between namespaces - the caller namespace and the new one | 
|  | 154 | * will use the same copy of a library if it was loaded prior to android_create_namespace call. | 
|  | 155 | * | 
|  | 156 | * Note that libraries loaded after the namespace is created will not be shared. | 
|  | 157 | * | 
|  | 158 | * Shared namespaces can be isolated or regular. Note that they do not inherit the search path nor | 
|  | 159 | * permitted_path from the caller's namespace. | 
|  | 160 | */ | 
|  | 161 | ANDROID_NAMESPACE_TYPE_SHARED = 2, | 
| Jiyong Park | 37b91af | 2017-05-05 22:07:05 +0900 | [diff] [blame] | 162 |  | 
|  | 163 | /* This flag instructs linker to enable grey-list workaround for the namespace. | 
|  | 164 | * See http://b/26394120 for details. | 
|  | 165 | */ | 
|  | 166 | ANDROID_NAMESPACE_TYPE_GREYLIST_ENABLED = 0x08000000, | 
|  | 167 |  | 
| Dimitry Ivanov | 41fd295 | 2016-05-09 17:37:39 -0700 | [diff] [blame] | 168 | ANDROID_NAMESPACE_TYPE_SHARED_ISOLATED = ANDROID_NAMESPACE_TYPE_SHARED | | 
|  | 169 | ANDROID_NAMESPACE_TYPE_ISOLATED, | 
|  | 170 | }; | 
|  | 171 |  | 
| Dimitry Ivanov | 7a34b9d | 2017-02-03 14:07:34 -0800 | [diff] [blame] | 172 | bool init_anonymous_namespace(const char* shared_lib_sonames, const char* library_search_path); | 
| Dimitry Ivanov | fc2da53 | 2016-05-12 15:20:21 -0700 | [diff] [blame] | 173 | android_namespace_t* create_namespace(const void* caller_addr, | 
|  | 174 | const char* name, | 
|  | 175 | const char* ld_library_path, | 
|  | 176 | const char* default_library_path, | 
|  | 177 | uint64_t type, | 
|  | 178 | const char* permitted_when_isolated_path, | 
|  | 179 | android_namespace_t* parent_namespace); | 
| Dmitriy Ivanov | 42d5fcb | 2015-10-29 17:01:24 -0700 | [diff] [blame] | 180 |  | 
| Dimitry Ivanov | 7a34b9d | 2017-02-03 14:07:34 -0800 | [diff] [blame] | 181 | bool link_namespaces(android_namespace_t* namespace_from, | 
|  | 182 | android_namespace_t* namespace_to, | 
|  | 183 | const char* shared_lib_sonames); | 
|  | 184 |  | 
| Logan Chien | 9ee4591 | 2018-01-18 12:05:09 +0800 | [diff] [blame] | 185 | bool link_namespaces_all_libs(android_namespace_t* namespace_from, | 
|  | 186 | android_namespace_t* namespace_to); | 
|  | 187 |  | 
| Jiyong Park | 01de74e | 2017-04-03 23:10:37 +0900 | [diff] [blame] | 188 | android_namespace_t* get_exported_namespace(const char* name); | 
|  | 189 |  | 
| dimitry | 06016f2 | 2018-01-05 11:39:28 +0100 | [diff] [blame] | 190 | void increment_dso_handle_reference_counter(void* dso_handle); | 
|  | 191 | void decrement_dso_handle_reference_counter(void* dso_handle); | 
| Vic Yang | bb7e123 | 2019-01-29 20:23:16 -0800 | [diff] [blame] | 192 |  | 
|  | 193 | void purge_unused_memory(); | 
| Torne (Richard Coles) | efbe9a5 | 2018-10-17 15:59:38 -0400 | [diff] [blame] | 194 |  | 
|  | 195 | struct address_space_params { | 
|  | 196 | void* start_addr = nullptr; | 
|  | 197 | size_t reserved_size = 0; | 
|  | 198 | bool must_use_address = false; | 
|  | 199 | }; |