blob: 7f40f9034e1c5ecf923297b1ccb1cf70fb707579 [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__;
dimitry06016f22018-01-05 11:39:28 +010085void __loader_add_thread_local_dtor(void* dso_handle) __LINKER_PUBLIC__;
86void __loader_remove_thread_local_dtor(void* dso_handle) __LINKER_PUBLIC__;
dimitry7abea572017-08-29 18:14:49 +020087#if defined(__arm__)
88_Unwind_Ptr __loader_dl_unwind_find_exidx(_Unwind_Ptr pc, int* pcount) __LINKER_PUBLIC__;
89#endif
90}
91
Elliott Hughes212e0e32014-12-01 16:43:51 -080092static pthread_mutex_t g_dl_mutex = PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP;
Elliott Hughes3b297c42012-10-11 16:08:51 -070093
Elliott Hughes5e071a12016-08-11 15:02:45 -070094static char* __bionic_set_dlerror(char* new_value) {
Elliott Hughes2a0b8732013-10-08 18:50:24 -070095 char** dlerror_slot = &reinterpret_cast<char**>(__get_tls())[TLS_SLOT_DLERROR];
Elliott Hughes5419b942012-10-16 15:54:46 -070096
Elliott Hughes5e071a12016-08-11 15:02:45 -070097 char* old_value = *dlerror_slot;
Elliott Hughes5419b942012-10-16 15:54:46 -070098 *dlerror_slot = new_value;
Elliott Hughes295082b2017-04-15 09:11:15 -070099 if (new_value != nullptr) LD_LOG(kLogErrors, "dlerror set to \"%s\"", new_value);
Elliott Hughes5419b942012-10-16 15:54:46 -0700100 return old_value;
Elliott Hughes3b297c42012-10-11 16:08:51 -0700101}
102
Elliott Hughes5419b942012-10-16 15:54:46 -0700103static void __bionic_format_dlerror(const char* msg, const char* detail) {
104 char* buffer = __get_thread()->dlerror_buffer;
105 strlcpy(buffer, msg, __BIONIC_DLERROR_BUFFER_SIZE);
Dmitriy Ivanov851135b2014-08-29 12:02:36 -0700106 if (detail != nullptr) {
Elliott Hughes5419b942012-10-16 15:54:46 -0700107 strlcat(buffer, ": ", __BIONIC_DLERROR_BUFFER_SIZE);
108 strlcat(buffer, detail, __BIONIC_DLERROR_BUFFER_SIZE);
109 }
110
111 __bionic_set_dlerror(buffer);
112}
113
dimitry7abea572017-08-29 18:14:49 +0200114char* __loader_dlerror() {
Elliott Hughes5e071a12016-08-11 15:02:45 -0700115 char* old_value = __bionic_set_dlerror(nullptr);
Elliott Hughes5419b942012-10-16 15:54:46 -0700116 return old_value;
117}
118
dimitry7abea572017-08-29 18:14:49 +0200119void __loader_android_get_LD_LIBRARY_PATH(char* buffer, size_t buffer_size) {
Elliott Hughes1728b232014-05-14 10:02:03 -0700120 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Elliott Hughesa4aafd12014-01-13 16:37:47 -0800121 do_android_get_LD_LIBRARY_PATH(buffer, buffer_size);
122}
123
dimitry7abea572017-08-29 18:14:49 +0200124void __loader_android_update_LD_LIBRARY_PATH(const char* ld_library_path) {
Elliott Hughes1728b232014-05-14 10:02:03 -0700125 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Elliott Hughescade4c32012-12-20 14:42:14 -0800126 do_android_update_LD_LIBRARY_PATH(ld_library_path);
127}
128
Dimitry Ivanovd9e427c2016-11-22 16:55:25 -0800129static void* dlopen_ext(const char* filename,
130 int flags,
131 const android_dlextinfo* extinfo,
132 const void* caller_addr) {
Elliott Hughes1728b232014-05-14 10:02:03 -0700133 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Dimitry Ivanovb996d602016-07-11 18:11:39 -0700134 g_linker_logger.ResetState();
Dimitry Ivanovd88e1f32016-03-24 15:30:30 -0700135 void* result = do_dlopen(filename, flags, extinfo, caller_addr);
Dmitriy Ivanov851135b2014-08-29 12:02:36 -0700136 if (result == nullptr) {
Elliott Hughes650be4e2013-03-05 18:47:58 -0800137 __bionic_format_dlerror("dlopen failed", linker_get_error_buffer());
Dmitriy Ivanov851135b2014-08-29 12:02:36 -0700138 return nullptr;
Elliott Hughes3b297c42012-10-11 16:08:51 -0700139 }
Elliott Hughes3b297c42012-10-11 16:08:51 -0700140 return result;
141}
142
dimitry7abea572017-08-29 18:14:49 +0200143void* __loader_android_dlopen_ext(const char* filename,
Dimitry Ivanovd9e427c2016-11-22 16:55:25 -0800144 int flags,
145 const android_dlextinfo* extinfo,
146 const void* caller_addr) {
Dmitriy Ivanove5cfafe2015-07-17 10:36:10 -0700147 return dlopen_ext(filename, flags, extinfo, caller_addr);
Dmitriy Ivanovb648a8a2014-05-19 15:06:58 -0700148}
149
dimitry7abea572017-08-29 18:14:49 +0200150void* __loader_dlopen(const char* filename, int flags, const void* caller_addr) {
Dmitriy Ivanove5cfafe2015-07-17 10:36:10 -0700151 return dlopen_ext(filename, flags, nullptr, caller_addr);
Torne (Richard Coles)012cb452014-02-06 14:34:21 +0000152}
153
Dimitry Ivanovd9e427c2016-11-22 16:55:25 -0800154void* dlsym_impl(void* handle, const char* symbol, const char* version, const void* caller_addr) {
Dimitry Ivanov4a2c5aa2015-12-10 16:08:14 -0800155 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Dimitry Ivanovb996d602016-07-11 18:11:39 -0700156 g_linker_logger.ResetState();
Dimitry Ivanov4a2c5aa2015-12-10 16:08:14 -0800157 void* result;
Dimitry Ivanov9cf99cb2015-12-11 14:22:24 -0800158 if (!do_dlsym(handle, symbol, version, caller_addr, &result)) {
Dimitry Ivanov4a2c5aa2015-12-10 16:08:14 -0800159 __bionic_format_dlerror(linker_get_error_buffer(), nullptr);
Dmitriy Ivanov851135b2014-08-29 12:02:36 -0700160 return nullptr;
Elliott Hughes3b297c42012-10-11 16:08:51 -0700161 }
Dimitry Ivanov4a2c5aa2015-12-10 16:08:14 -0800162
163 return result;
Elliott Hughes3b297c42012-10-11 16:08:51 -0700164}
165
dimitry7abea572017-08-29 18:14:49 +0200166void* __loader_dlsym(void* handle, const char* symbol, const void* caller_addr) {
Dimitry Ivanov9cf99cb2015-12-11 14:22:24 -0800167 return dlsym_impl(handle, symbol, nullptr, caller_addr);
168}
169
dimitry7abea572017-08-29 18:14:49 +0200170void* __loader_dlvsym(void* handle, const char* symbol, const char* version, const void* caller_addr) {
Dimitry Ivanov9cf99cb2015-12-11 14:22:24 -0800171 return dlsym_impl(handle, symbol, version, caller_addr);
172}
173
dimitry7abea572017-08-29 18:14:49 +0200174int __loader_dladdr(const void* addr, Dl_info* info) {
Elliott Hughes1728b232014-05-14 10:02:03 -0700175 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Dimitry Ivanov4a2c5aa2015-12-10 16:08:14 -0800176 return do_dladdr(addr, info);
Elliott Hughes3b297c42012-10-11 16:08:51 -0700177}
178
dimitry7abea572017-08-29 18:14:49 +0200179int __loader_dlclose(void* handle) {
Elliott Hughes1728b232014-05-14 10:02:03 -0700180 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Dimitry Ivanovd88e1f32016-03-24 15:30:30 -0700181 int result = do_dlclose(handle);
182 if (result != 0) {
183 __bionic_format_dlerror("dlclose failed", linker_get_error_buffer());
184 }
185 return result;
Elliott Hughes3b297c42012-10-11 16:08:51 -0700186}
187
dimitry7abea572017-08-29 18:14:49 +0200188int __loader_dl_iterate_phdr(int (*cb)(dl_phdr_info* info, size_t size, void* data), void* data) {
Dmitriy Ivanov7271caf2015-06-29 14:48:25 -0700189 ScopedPthreadMutexLocker locker(&g_dl_mutex);
190 return do_dl_iterate_phdr(cb, data);
191}
192
dimitry7abea572017-08-29 18:14:49 +0200193// This function is needed by libgcc.a
194int dl_iterate_phdr(int (*cb)(dl_phdr_info* info, size_t size, void* data), void* data) {
195 return __loader_dl_iterate_phdr(cb, data);
196}
197
Dimitry Ivanovd9e427c2016-11-22 16:55:25 -0800198#if defined(__arm__)
dimitry7abea572017-08-29 18:14:49 +0200199_Unwind_Ptr __loader_dl_unwind_find_exidx(_Unwind_Ptr pc, int* pcount) {
Dimitry Ivanovd9e427c2016-11-22 16:55:25 -0800200 ScopedPthreadMutexLocker locker(&g_dl_mutex);
201 return do_dl_unwind_find_exidx(pc, pcount);
202}
203#endif
204
dimitry7abea572017-08-29 18:14:49 +0200205void __loader_android_set_application_target_sdk_version(uint32_t target) {
Dmitriy Ivanovd974e882015-05-27 18:29:41 -0700206 // lock to avoid modification in the middle of dlopen.
207 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Dmitriy Ivanov79fd6682015-05-21 17:43:49 -0700208 set_application_target_sdk_version(target);
209}
210
dimitry7abea572017-08-29 18:14:49 +0200211uint32_t __loader_android_get_application_target_sdk_version() {
Dmitriy Ivanov79fd6682015-05-21 17:43:49 -0700212 return get_application_target_sdk_version();
213}
214
dimitry7abea572017-08-29 18:14:49 +0200215void __loader_android_dlwarning(void* obj, void (*f)(void*, const char*)) {
Dimitry Ivanov769b33f2016-07-21 11:33:40 -0700216 ScopedPthreadMutexLocker locker(&g_dl_mutex);
217 get_dlwarning(obj, f);
218}
219
dimitry7abea572017-08-29 18:14:49 +0200220bool __loader_android_init_anonymous_namespace(const char* shared_libs_sonames,
221 const char* library_search_path) {
Dmitriy Ivanov42d5fcb2015-10-29 17:01:24 -0700222 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Dimitry Ivanov7a34b9d2017-02-03 14:07:34 -0800223 bool success = init_anonymous_namespace(shared_libs_sonames, library_search_path);
Dmitriy Ivanov42d5fcb2015-10-29 17:01:24 -0700224 if (!success) {
Dimitry Ivanov7a34b9d2017-02-03 14:07:34 -0800225 __bionic_format_dlerror("android_init_anonymous_namespace failed", linker_get_error_buffer());
Dmitriy Ivanov42d5fcb2015-10-29 17:01:24 -0700226 }
227
228 return success;
229}
230
dimitry7abea572017-08-29 18:14:49 +0200231android_namespace_t* __loader_android_create_namespace(const char* name,
Dimitry Ivanovd9e427c2016-11-22 16:55:25 -0800232 const char* ld_library_path,
233 const char* default_library_path,
234 uint64_t type,
235 const char* permitted_when_isolated_path,
236 android_namespace_t* parent_namespace,
237 const void* caller_addr) {
Dmitriy Ivanov42d5fcb2015-10-29 17:01:24 -0700238 ScopedPthreadMutexLocker locker(&g_dl_mutex);
239
Dimitry Ivanovfc2da532016-05-12 15:20:21 -0700240 android_namespace_t* result = create_namespace(caller_addr,
241 name,
242 ld_library_path,
243 default_library_path,
244 type,
245 permitted_when_isolated_path,
246 parent_namespace);
Dmitriy Ivanov42d5fcb2015-10-29 17:01:24 -0700247
248 if (result == nullptr) {
249 __bionic_format_dlerror("android_create_namespace failed", linker_get_error_buffer());
250 }
251
252 return result;
253}
254
dimitry7abea572017-08-29 18:14:49 +0200255bool __loader_android_link_namespaces(android_namespace_t* namespace_from,
256 android_namespace_t* namespace_to,
257 const char* shared_libs_sonames) {
Dimitry Ivanov7a34b9d2017-02-03 14:07:34 -0800258 ScopedPthreadMutexLocker locker(&g_dl_mutex);
259
260 bool success = link_namespaces(namespace_from, namespace_to, shared_libs_sonames);
261
262 if (!success) {
263 __bionic_format_dlerror("android_link_namespaces failed", linker_get_error_buffer());
264 }
265
266 return success;
267}
268
dimitry7abea572017-08-29 18:14:49 +0200269android_namespace_t* __loader_android_get_exported_namespace(const char* name) {
Jiyong Park01de74e2017-04-03 23:10:37 +0900270 return get_exported_namespace(name);
271}
272
dimitry7abea572017-08-29 18:14:49 +0200273void __loader_cfi_fail(uint64_t CallSiteTypeId, void* Ptr, void *DiagData, void *CallerPc) {
Evgenii Stepanov0a3637d2016-07-06 13:20:59 -0700274 CFIShadowWriter::CfiFail(CallSiteTypeId, Ptr, DiagData, CallerPc);
275}
276
dimitry06016f22018-01-05 11:39:28 +0100277void __loader_add_thread_local_dtor(void* dso_handle) {
278 ScopedPthreadMutexLocker locker(&g_dl_mutex);
279 increment_dso_handle_reference_counter(dso_handle);
280}
281
282void __loader_remove_thread_local_dtor(void* dso_handle) {
283 ScopedPthreadMutexLocker locker(&g_dl_mutex);
284 decrement_dso_handle_reference_counter(dso_handle);
285}
286
Dmitriy Ivanovaae859c2015-03-31 11:14:03 -0700287static uint8_t __libdl_info_buf[sizeof(soinfo)] __attribute__((aligned(8)));
288static soinfo* __libdl_info = nullptr;
Dmitriy Ivanovd59e5002014-05-09 09:10:14 -0700289
Elliott Hughes22d62922012-10-12 10:50:21 -0700290// This is used by the dynamic linker. Every process gets these symbols for free.
dimitry7abea572017-08-29 18:14:49 +0200291soinfo* get_libdl_info(const char* linker_path,
292 const soinfo& linker_si,
293 const link_map& linker_map) {
294 CHECK((linker_si.flags_ & FLAG_GNU_HASH) != 0);
295
Dmitriy Ivanovaae859c2015-03-31 11:14:03 -0700296 if (__libdl_info == nullptr) {
Dimitry Ivanovd9e427c2016-11-22 16:55:25 -0800297 __libdl_info = new (__libdl_info_buf) soinfo(&g_default_namespace, linker_path, nullptr, 0, 0);
dimitry7abea572017-08-29 18:14:49 +0200298 __libdl_info->flags_ |= (FLAG_LINKED | FLAG_GNU_HASH);
299 __libdl_info->strtab_ = linker_si.strtab_;
300 __libdl_info->symtab_ = linker_si.symtab_;
301 __libdl_info->load_bias = linker_si.load_bias;
302
303 __libdl_info->gnu_nbucket_ = linker_si.gnu_nbucket_;
304 __libdl_info->gnu_maskwords_ = linker_si.gnu_maskwords_;
305 __libdl_info->gnu_shift2_ = linker_si.gnu_shift2_;
306 __libdl_info->gnu_bloom_filter_ = linker_si.gnu_bloom_filter_;
307 __libdl_info->gnu_bucket_ = linker_si.gnu_bucket_;
308 __libdl_info->gnu_chain_ = linker_si.gnu_chain_;
309
Dmitriy Ivanovaae859c2015-03-31 11:14:03 -0700310 __libdl_info->ref_count_ = 1;
dimitry7abea572017-08-29 18:14:49 +0200311 __libdl_info->strtab_size_ = linker_si.strtab_size_;
Dmitriy Ivanovaae859c2015-03-31 11:14:03 -0700312 __libdl_info->local_group_root_ = __libdl_info;
dimitry8e8c2c02018-01-04 12:08:32 +0100313 __libdl_info->soname_ = linker_si.soname_;
Dmitriy Ivanov19133522015-06-02 17:36:54 -0700314 __libdl_info->target_sdk_version_ = __ANDROID_API__;
Dimitry Ivanovd88e1f32016-03-24 15:30:30 -0700315 __libdl_info->generate_handle();
Dimitry Ivanovcd510cb2017-05-31 15:07:41 -0700316 __libdl_info->link_map_head.l_addr = linker_map.l_addr;
317 __libdl_info->link_map_head.l_name = linker_map.l_name;
318 __libdl_info->link_map_head.l_ld = linker_map.l_ld;
Mike Frysinger747d30e2015-10-20 14:06:25 -0400319#if defined(__work_around_b_24465209__)
Dmitriy Ivanovaae859c2015-03-31 11:14:03 -0700320 strlcpy(__libdl_info->old_name_, __libdl_info->soname_, sizeof(__libdl_info->old_name_));
321#endif
Dmitriy Ivanovd59e5002014-05-09 09:10:14 -0700322 }
Elliott Hughes3b297c42012-10-11 16:08:51 -0700323
Dmitriy Ivanovaae859c2015-03-31 11:14:03 -0700324 return __libdl_info;
Dmitriy Ivanovd59e5002014-05-09 09:10:14 -0700325}