blob: 869e0c191ed4101504c09691ae98769e6d09276b [file] [log] [blame]
Elliott Hughes3b297c42012-10-11 16:08:51 -07001/*
2 * Copyright (C) 2007 The Android Open Source Project
Dimitry Ivanovbcc4da92017-02-15 15:31:13 -08003 * All rights reserved.
Elliott Hughes3b297c42012-10-11 16:08:51 -07004 *
Dimitry Ivanovbcc4da92017-02-15 15:31:13 -08005 * 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.
Elliott Hughes3b297c42012-10-11 16:08:51 -070014 *
Dimitry Ivanovbcc4da92017-02-15 15:31:13 -080015 * 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.
Elliott Hughes3b297c42012-10-11 16:08:51 -070027 */
28
Elliott Hughes5419b942012-10-16 15:54:46 -070029#include "linker.h"
Evgenii Stepanov0a3637d2016-07-06 13:20:59 -070030#include "linker_cfi.h"
Dimitry Ivanov48ec2882016-08-04 11:50:36 -070031#include "linker_globals.h"
Dimitry Ivanov769b33f2016-07-21 11:33:40 -070032#include "linker_dlwarning.h"
Elliott Hughes5419b942012-10-16 15:54:46 -070033
Elliott Hughes3b297c42012-10-11 16:08:51 -070034#include <pthread.h>
35#include <stdio.h>
Elliott Hughes5419b942012-10-16 15:54:46 -070036#include <stdlib.h>
Elliott Hughes05fc1d72015-01-28 18:02:33 -080037#include <string.h>
Dmitriy Ivanov19133522015-06-02 17:36:54 -070038#include <android/api-level.h>
Elliott Hughes3b297c42012-10-11 16:08:51 -070039
Elliott Hughes5419b942012-10-16 15:54:46 -070040#include <bionic/pthread_internal.h>
Elliott Hugheseb847bc2013-10-09 15:50:50 -070041#include "private/bionic_tls.h"
42#include "private/ScopedPthreadMutexLocker.h"
Elliott Hughes3b297c42012-10-11 16:08:51 -070043
dimitry7abea572017-08-29 18:14:49 +020044#define __LINKER_PUBLIC__ __attribute__((visibility("default")))
45
46extern "C" {
47
48android_namespace_t* __loader_android_create_namespace(const char* name,
49 const char* ld_library_path,
50 const char* default_library_path,
51 uint64_t type,
52 const char* permitted_when_isolated_path,
53 android_namespace_t* parent_namespace,
54 const void* caller_addr) __LINKER_PUBLIC__;
55void* __loader_android_dlopen_ext(const char* filename,
56 int flags,
57 const android_dlextinfo* extinfo,
58 const void* caller_addr) __LINKER_PUBLIC__;
59void __loader_android_dlwarning(void* obj, void (*f)(void*, const char*)) __LINKER_PUBLIC__;
60uint32_t __loader_android_get_application_target_sdk_version() __LINKER_PUBLIC__;
61void __loader_android_get_LD_LIBRARY_PATH(char* buffer, size_t buffer_size) __LINKER_PUBLIC__;
62android_namespace_t* __loader_android_get_exported_namespace(const char* name) __LINKER_PUBLIC__;
63bool __loader_android_init_anonymous_namespace(const char* shared_libs_sonames,
64 const char* library_search_path) __LINKER_PUBLIC__;
65bool __loader_android_link_namespaces(android_namespace_t* namespace_from,
66 android_namespace_t* namespace_to,
67 const char* shared_libs_sonames) __LINKER_PUBLIC__;
68void __loader_android_set_application_target_sdk_version(uint32_t target) __LINKER_PUBLIC__;
69void __loader_android_update_LD_LIBRARY_PATH(const char* ld_library_path) __LINKER_PUBLIC__;
70void __loader_cfi_fail(uint64_t CallSiteTypeId,
71 void* Ptr,
72 void *DiagData,
73 void *CallerPc) __LINKER_PUBLIC__;
74int __loader_dl_iterate_phdr(int (*cb)(dl_phdr_info* info, size_t size, void* data),
75 void* data) __LINKER_PUBLIC__;
76int __loader_dladdr(const void* addr, Dl_info* info) __LINKER_PUBLIC__;
77int __loader_dlclose(void* handle) __LINKER_PUBLIC__;
78char* __loader_dlerror() __LINKER_PUBLIC__;
79void* __loader_dlopen(const char* filename, int flags, const void* caller_addr) __LINKER_PUBLIC__;
80void* __loader_dlsym(void* handle, const char* symbol, const void* caller_addr) __LINKER_PUBLIC__;
81void* __loader_dlvsym(void* handle,
82 const char* symbol,
83 const char* version,
84 const void* caller_addr) __LINKER_PUBLIC__;
85#if defined(__arm__)
86_Unwind_Ptr __loader_dl_unwind_find_exidx(_Unwind_Ptr pc, int* pcount) __LINKER_PUBLIC__;
87#endif
88}
89
Elliott Hughes212e0e32014-12-01 16:43:51 -080090static pthread_mutex_t g_dl_mutex = PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP;
Elliott Hughes3b297c42012-10-11 16:08:51 -070091
Elliott Hughes5e071a12016-08-11 15:02:45 -070092static char* __bionic_set_dlerror(char* new_value) {
Elliott Hughes2a0b8732013-10-08 18:50:24 -070093 char** dlerror_slot = &reinterpret_cast<char**>(__get_tls())[TLS_SLOT_DLERROR];
Elliott Hughes5419b942012-10-16 15:54:46 -070094
Elliott Hughes5e071a12016-08-11 15:02:45 -070095 char* old_value = *dlerror_slot;
Elliott Hughes5419b942012-10-16 15:54:46 -070096 *dlerror_slot = new_value;
Elliott Hughes295082b2017-04-15 09:11:15 -070097 if (new_value != nullptr) LD_LOG(kLogErrors, "dlerror set to \"%s\"", new_value);
Elliott Hughes5419b942012-10-16 15:54:46 -070098 return old_value;
Elliott Hughes3b297c42012-10-11 16:08:51 -070099}
100
Elliott Hughes5419b942012-10-16 15:54:46 -0700101static void __bionic_format_dlerror(const char* msg, const char* detail) {
102 char* buffer = __get_thread()->dlerror_buffer;
103 strlcpy(buffer, msg, __BIONIC_DLERROR_BUFFER_SIZE);
Dmitriy Ivanov851135b2014-08-29 12:02:36 -0700104 if (detail != nullptr) {
Elliott Hughes5419b942012-10-16 15:54:46 -0700105 strlcat(buffer, ": ", __BIONIC_DLERROR_BUFFER_SIZE);
106 strlcat(buffer, detail, __BIONIC_DLERROR_BUFFER_SIZE);
107 }
108
109 __bionic_set_dlerror(buffer);
110}
111
dimitry7abea572017-08-29 18:14:49 +0200112char* __loader_dlerror() {
Elliott Hughes5e071a12016-08-11 15:02:45 -0700113 char* old_value = __bionic_set_dlerror(nullptr);
Elliott Hughes5419b942012-10-16 15:54:46 -0700114 return old_value;
115}
116
dimitry7abea572017-08-29 18:14:49 +0200117void __loader_android_get_LD_LIBRARY_PATH(char* buffer, size_t buffer_size) {
Elliott Hughes1728b232014-05-14 10:02:03 -0700118 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Elliott Hughesa4aafd12014-01-13 16:37:47 -0800119 do_android_get_LD_LIBRARY_PATH(buffer, buffer_size);
120}
121
dimitry7abea572017-08-29 18:14:49 +0200122void __loader_android_update_LD_LIBRARY_PATH(const char* ld_library_path) {
Elliott Hughes1728b232014-05-14 10:02:03 -0700123 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Elliott Hughescade4c32012-12-20 14:42:14 -0800124 do_android_update_LD_LIBRARY_PATH(ld_library_path);
125}
126
Dimitry Ivanovd9e427c2016-11-22 16:55:25 -0800127static void* dlopen_ext(const char* filename,
128 int flags,
129 const android_dlextinfo* extinfo,
130 const void* caller_addr) {
Elliott Hughes1728b232014-05-14 10:02:03 -0700131 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Dimitry Ivanovb996d602016-07-11 18:11:39 -0700132 g_linker_logger.ResetState();
Dimitry Ivanovd88e1f32016-03-24 15:30:30 -0700133 void* result = do_dlopen(filename, flags, extinfo, caller_addr);
Dmitriy Ivanov851135b2014-08-29 12:02:36 -0700134 if (result == nullptr) {
Elliott Hughes650be4e2013-03-05 18:47:58 -0800135 __bionic_format_dlerror("dlopen failed", linker_get_error_buffer());
Dmitriy Ivanov851135b2014-08-29 12:02:36 -0700136 return nullptr;
Elliott Hughes3b297c42012-10-11 16:08:51 -0700137 }
Elliott Hughes3b297c42012-10-11 16:08:51 -0700138 return result;
139}
140
dimitry7abea572017-08-29 18:14:49 +0200141void* __loader_android_dlopen_ext(const char* filename,
Dimitry Ivanovd9e427c2016-11-22 16:55:25 -0800142 int flags,
143 const android_dlextinfo* extinfo,
144 const void* caller_addr) {
Dmitriy Ivanove5cfafe2015-07-17 10:36:10 -0700145 return dlopen_ext(filename, flags, extinfo, caller_addr);
Dmitriy Ivanovb648a8a2014-05-19 15:06:58 -0700146}
147
dimitry7abea572017-08-29 18:14:49 +0200148void* __loader_dlopen(const char* filename, int flags, const void* caller_addr) {
Dmitriy Ivanove5cfafe2015-07-17 10:36:10 -0700149 return dlopen_ext(filename, flags, nullptr, caller_addr);
Torne (Richard Coles)012cb452014-02-06 14:34:21 +0000150}
151
Dimitry Ivanovd9e427c2016-11-22 16:55:25 -0800152void* dlsym_impl(void* handle, const char* symbol, const char* version, const void* caller_addr) {
Dimitry Ivanov4a2c5aa2015-12-10 16:08:14 -0800153 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Dimitry Ivanovb996d602016-07-11 18:11:39 -0700154 g_linker_logger.ResetState();
Dimitry Ivanov4a2c5aa2015-12-10 16:08:14 -0800155 void* result;
Dimitry Ivanov9cf99cb2015-12-11 14:22:24 -0800156 if (!do_dlsym(handle, symbol, version, caller_addr, &result)) {
Dimitry Ivanov4a2c5aa2015-12-10 16:08:14 -0800157 __bionic_format_dlerror(linker_get_error_buffer(), nullptr);
Dmitriy Ivanov851135b2014-08-29 12:02:36 -0700158 return nullptr;
Elliott Hughes3b297c42012-10-11 16:08:51 -0700159 }
Dimitry Ivanov4a2c5aa2015-12-10 16:08:14 -0800160
161 return result;
Elliott Hughes3b297c42012-10-11 16:08:51 -0700162}
163
dimitry7abea572017-08-29 18:14:49 +0200164void* __loader_dlsym(void* handle, const char* symbol, const void* caller_addr) {
Dimitry Ivanov9cf99cb2015-12-11 14:22:24 -0800165 return dlsym_impl(handle, symbol, nullptr, caller_addr);
166}
167
dimitry7abea572017-08-29 18:14:49 +0200168void* __loader_dlvsym(void* handle, const char* symbol, const char* version, const void* caller_addr) {
Dimitry Ivanov9cf99cb2015-12-11 14:22:24 -0800169 return dlsym_impl(handle, symbol, version, caller_addr);
170}
171
dimitry7abea572017-08-29 18:14:49 +0200172int __loader_dladdr(const void* addr, Dl_info* info) {
Elliott Hughes1728b232014-05-14 10:02:03 -0700173 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Dimitry Ivanov4a2c5aa2015-12-10 16:08:14 -0800174 return do_dladdr(addr, info);
Elliott Hughes3b297c42012-10-11 16:08:51 -0700175}
176
dimitry7abea572017-08-29 18:14:49 +0200177int __loader_dlclose(void* handle) {
Elliott Hughes1728b232014-05-14 10:02:03 -0700178 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Dimitry Ivanovd88e1f32016-03-24 15:30:30 -0700179 int result = do_dlclose(handle);
180 if (result != 0) {
181 __bionic_format_dlerror("dlclose failed", linker_get_error_buffer());
182 }
183 return result;
Elliott Hughes3b297c42012-10-11 16:08:51 -0700184}
185
dimitry7abea572017-08-29 18:14:49 +0200186int __loader_dl_iterate_phdr(int (*cb)(dl_phdr_info* info, size_t size, void* data), void* data) {
Dmitriy Ivanov7271caf2015-06-29 14:48:25 -0700187 ScopedPthreadMutexLocker locker(&g_dl_mutex);
188 return do_dl_iterate_phdr(cb, data);
189}
190
dimitry7abea572017-08-29 18:14:49 +0200191// This function is needed by libgcc.a
192int dl_iterate_phdr(int (*cb)(dl_phdr_info* info, size_t size, void* data), void* data) {
193 return __loader_dl_iterate_phdr(cb, data);
194}
195
Dimitry Ivanovd9e427c2016-11-22 16:55:25 -0800196#if defined(__arm__)
dimitry7abea572017-08-29 18:14:49 +0200197_Unwind_Ptr __loader_dl_unwind_find_exidx(_Unwind_Ptr pc, int* pcount) {
Dimitry Ivanovd9e427c2016-11-22 16:55:25 -0800198 ScopedPthreadMutexLocker locker(&g_dl_mutex);
199 return do_dl_unwind_find_exidx(pc, pcount);
200}
201#endif
202
dimitry7abea572017-08-29 18:14:49 +0200203void __loader_android_set_application_target_sdk_version(uint32_t target) {
Dmitriy Ivanovd974e882015-05-27 18:29:41 -0700204 // lock to avoid modification in the middle of dlopen.
205 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Dmitriy Ivanov79fd6682015-05-21 17:43:49 -0700206 set_application_target_sdk_version(target);
207}
208
dimitry7abea572017-08-29 18:14:49 +0200209uint32_t __loader_android_get_application_target_sdk_version() {
Dmitriy Ivanov79fd6682015-05-21 17:43:49 -0700210 return get_application_target_sdk_version();
211}
212
dimitry7abea572017-08-29 18:14:49 +0200213void __loader_android_dlwarning(void* obj, void (*f)(void*, const char*)) {
Dimitry Ivanov769b33f2016-07-21 11:33:40 -0700214 ScopedPthreadMutexLocker locker(&g_dl_mutex);
215 get_dlwarning(obj, f);
216}
217
dimitry7abea572017-08-29 18:14:49 +0200218bool __loader_android_init_anonymous_namespace(const char* shared_libs_sonames,
219 const char* library_search_path) {
Dmitriy Ivanov42d5fcb2015-10-29 17:01:24 -0700220 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Dimitry Ivanov7a34b9d2017-02-03 14:07:34 -0800221 bool success = init_anonymous_namespace(shared_libs_sonames, library_search_path);
Dmitriy Ivanov42d5fcb2015-10-29 17:01:24 -0700222 if (!success) {
Dimitry Ivanov7a34b9d2017-02-03 14:07:34 -0800223 __bionic_format_dlerror("android_init_anonymous_namespace failed", linker_get_error_buffer());
Dmitriy Ivanov42d5fcb2015-10-29 17:01:24 -0700224 }
225
226 return success;
227}
228
dimitry7abea572017-08-29 18:14:49 +0200229android_namespace_t* __loader_android_create_namespace(const char* name,
Dimitry Ivanovd9e427c2016-11-22 16:55:25 -0800230 const char* ld_library_path,
231 const char* default_library_path,
232 uint64_t type,
233 const char* permitted_when_isolated_path,
234 android_namespace_t* parent_namespace,
235 const void* caller_addr) {
Dmitriy Ivanov42d5fcb2015-10-29 17:01:24 -0700236 ScopedPthreadMutexLocker locker(&g_dl_mutex);
237
Dimitry Ivanovfc2da532016-05-12 15:20:21 -0700238 android_namespace_t* result = create_namespace(caller_addr,
239 name,
240 ld_library_path,
241 default_library_path,
242 type,
243 permitted_when_isolated_path,
244 parent_namespace);
Dmitriy Ivanov42d5fcb2015-10-29 17:01:24 -0700245
246 if (result == nullptr) {
247 __bionic_format_dlerror("android_create_namespace failed", linker_get_error_buffer());
248 }
249
250 return result;
251}
252
dimitry7abea572017-08-29 18:14:49 +0200253bool __loader_android_link_namespaces(android_namespace_t* namespace_from,
254 android_namespace_t* namespace_to,
255 const char* shared_libs_sonames) {
Dimitry Ivanov7a34b9d2017-02-03 14:07:34 -0800256 ScopedPthreadMutexLocker locker(&g_dl_mutex);
257
258 bool success = link_namespaces(namespace_from, namespace_to, shared_libs_sonames);
259
260 if (!success) {
261 __bionic_format_dlerror("android_link_namespaces failed", linker_get_error_buffer());
262 }
263
264 return success;
265}
266
dimitry7abea572017-08-29 18:14:49 +0200267android_namespace_t* __loader_android_get_exported_namespace(const char* name) {
Jiyong Park01de74e2017-04-03 23:10:37 +0900268 return get_exported_namespace(name);
269}
270
dimitry7abea572017-08-29 18:14:49 +0200271void __loader_cfi_fail(uint64_t CallSiteTypeId, void* Ptr, void *DiagData, void *CallerPc) {
Evgenii Stepanov0a3637d2016-07-06 13:20:59 -0700272 CFIShadowWriter::CfiFail(CallSiteTypeId, Ptr, DiagData, CallerPc);
273}
274
Dmitriy Ivanovaae859c2015-03-31 11:14:03 -0700275static uint8_t __libdl_info_buf[sizeof(soinfo)] __attribute__((aligned(8)));
276static soinfo* __libdl_info = nullptr;
Dmitriy Ivanovd59e5002014-05-09 09:10:14 -0700277
Elliott Hughes22d62922012-10-12 10:50:21 -0700278// This is used by the dynamic linker. Every process gets these symbols for free.
dimitry7abea572017-08-29 18:14:49 +0200279soinfo* get_libdl_info(const char* linker_path,
280 const soinfo& linker_si,
281 const link_map& linker_map) {
282 CHECK((linker_si.flags_ & FLAG_GNU_HASH) != 0);
283
Dmitriy Ivanovaae859c2015-03-31 11:14:03 -0700284 if (__libdl_info == nullptr) {
Dimitry Ivanovd9e427c2016-11-22 16:55:25 -0800285 __libdl_info = new (__libdl_info_buf) soinfo(&g_default_namespace, linker_path, nullptr, 0, 0);
dimitry7abea572017-08-29 18:14:49 +0200286 __libdl_info->flags_ |= (FLAG_LINKED | FLAG_GNU_HASH);
287 __libdl_info->strtab_ = linker_si.strtab_;
288 __libdl_info->symtab_ = linker_si.symtab_;
289 __libdl_info->load_bias = linker_si.load_bias;
290
291 __libdl_info->gnu_nbucket_ = linker_si.gnu_nbucket_;
292 __libdl_info->gnu_maskwords_ = linker_si.gnu_maskwords_;
293 __libdl_info->gnu_shift2_ = linker_si.gnu_shift2_;
294 __libdl_info->gnu_bloom_filter_ = linker_si.gnu_bloom_filter_;
295 __libdl_info->gnu_bucket_ = linker_si.gnu_bucket_;
296 __libdl_info->gnu_chain_ = linker_si.gnu_chain_;
297
Dmitriy Ivanovaae859c2015-03-31 11:14:03 -0700298 __libdl_info->ref_count_ = 1;
dimitry7abea572017-08-29 18:14:49 +0200299 __libdl_info->strtab_size_ = linker_si.strtab_size_;
Dmitriy Ivanovaae859c2015-03-31 11:14:03 -0700300 __libdl_info->local_group_root_ = __libdl_info;
dimitry8e8c2c02018-01-04 12:08:32 +0100301 __libdl_info->soname_ = linker_si.soname_;
Dmitriy Ivanov19133522015-06-02 17:36:54 -0700302 __libdl_info->target_sdk_version_ = __ANDROID_API__;
Dimitry Ivanovd88e1f32016-03-24 15:30:30 -0700303 __libdl_info->generate_handle();
Dimitry Ivanovcd510cb2017-05-31 15:07:41 -0700304 __libdl_info->link_map_head.l_addr = linker_map.l_addr;
305 __libdl_info->link_map_head.l_name = linker_map.l_name;
306 __libdl_info->link_map_head.l_ld = linker_map.l_ld;
Mike Frysinger747d30e2015-10-20 14:06:25 -0400307#if defined(__work_around_b_24465209__)
Dmitriy Ivanovaae859c2015-03-31 11:14:03 -0700308 strlcpy(__libdl_info->old_name_, __libdl_info->soname_, sizeof(__libdl_info->old_name_));
309#endif
Dmitriy Ivanovd59e5002014-05-09 09:10:14 -0700310 }
Elliott Hughes3b297c42012-10-11 16:08:51 -0700311
Dmitriy Ivanovaae859c2015-03-31 11:14:03 -0700312 return __libdl_info;
Dmitriy Ivanovd59e5002014-05-09 09:10:14 -0700313}