blob: 4b84537daa9d4c331a55c2429399d23398e65ddd [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__;
Logan Chien9ee45912018-01-18 12:05:09 +080068bool __loader_android_link_namespaces_all_libs(android_namespace_t* namespace_from,
69 android_namespace_t* namespace_to) __LINKER_PUBLIC__;
dimitry7abea572017-08-29 18:14:49 +020070void __loader_android_set_application_target_sdk_version(uint32_t target) __LINKER_PUBLIC__;
71void __loader_android_update_LD_LIBRARY_PATH(const char* ld_library_path) __LINKER_PUBLIC__;
72void __loader_cfi_fail(uint64_t CallSiteTypeId,
73 void* Ptr,
74 void *DiagData,
75 void *CallerPc) __LINKER_PUBLIC__;
76int __loader_dl_iterate_phdr(int (*cb)(dl_phdr_info* info, size_t size, void* data),
77 void* data) __LINKER_PUBLIC__;
78int __loader_dladdr(const void* addr, Dl_info* info) __LINKER_PUBLIC__;
79int __loader_dlclose(void* handle) __LINKER_PUBLIC__;
80char* __loader_dlerror() __LINKER_PUBLIC__;
81void* __loader_dlopen(const char* filename, int flags, const void* caller_addr) __LINKER_PUBLIC__;
82void* __loader_dlsym(void* handle, const char* symbol, const void* caller_addr) __LINKER_PUBLIC__;
83void* __loader_dlvsym(void* handle,
84 const char* symbol,
85 const char* version,
86 const void* caller_addr) __LINKER_PUBLIC__;
dimitry06016f22018-01-05 11:39:28 +010087void __loader_add_thread_local_dtor(void* dso_handle) __LINKER_PUBLIC__;
88void __loader_remove_thread_local_dtor(void* dso_handle) __LINKER_PUBLIC__;
dimitry7abea572017-08-29 18:14:49 +020089#if defined(__arm__)
90_Unwind_Ptr __loader_dl_unwind_find_exidx(_Unwind_Ptr pc, int* pcount) __LINKER_PUBLIC__;
91#endif
92}
93
Elliott Hughes212e0e32014-12-01 16:43:51 -080094static pthread_mutex_t g_dl_mutex = PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP;
Elliott Hughes3b297c42012-10-11 16:08:51 -070095
Elliott Hughes5e071a12016-08-11 15:02:45 -070096static char* __bionic_set_dlerror(char* new_value) {
Elliott Hughes2a0b8732013-10-08 18:50:24 -070097 char** dlerror_slot = &reinterpret_cast<char**>(__get_tls())[TLS_SLOT_DLERROR];
Elliott Hughes5419b942012-10-16 15:54:46 -070098
Elliott Hughes5e071a12016-08-11 15:02:45 -070099 char* old_value = *dlerror_slot;
Elliott Hughes5419b942012-10-16 15:54:46 -0700100 *dlerror_slot = new_value;
Elliott Hughes295082b2017-04-15 09:11:15 -0700101 if (new_value != nullptr) LD_LOG(kLogErrors, "dlerror set to \"%s\"", new_value);
Elliott Hughes5419b942012-10-16 15:54:46 -0700102 return old_value;
Elliott Hughes3b297c42012-10-11 16:08:51 -0700103}
104
Elliott Hughes5419b942012-10-16 15:54:46 -0700105static void __bionic_format_dlerror(const char* msg, const char* detail) {
106 char* buffer = __get_thread()->dlerror_buffer;
107 strlcpy(buffer, msg, __BIONIC_DLERROR_BUFFER_SIZE);
Dmitriy Ivanov851135b2014-08-29 12:02:36 -0700108 if (detail != nullptr) {
Elliott Hughes5419b942012-10-16 15:54:46 -0700109 strlcat(buffer, ": ", __BIONIC_DLERROR_BUFFER_SIZE);
110 strlcat(buffer, detail, __BIONIC_DLERROR_BUFFER_SIZE);
111 }
112
113 __bionic_set_dlerror(buffer);
114}
115
dimitry7abea572017-08-29 18:14:49 +0200116char* __loader_dlerror() {
Elliott Hughes5e071a12016-08-11 15:02:45 -0700117 char* old_value = __bionic_set_dlerror(nullptr);
Elliott Hughes5419b942012-10-16 15:54:46 -0700118 return old_value;
119}
120
dimitry7abea572017-08-29 18:14:49 +0200121void __loader_android_get_LD_LIBRARY_PATH(char* buffer, size_t buffer_size) {
Elliott Hughes1728b232014-05-14 10:02:03 -0700122 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Elliott Hughesa4aafd12014-01-13 16:37:47 -0800123 do_android_get_LD_LIBRARY_PATH(buffer, buffer_size);
124}
125
dimitry7abea572017-08-29 18:14:49 +0200126void __loader_android_update_LD_LIBRARY_PATH(const char* ld_library_path) {
Elliott Hughes1728b232014-05-14 10:02:03 -0700127 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Elliott Hughescade4c32012-12-20 14:42:14 -0800128 do_android_update_LD_LIBRARY_PATH(ld_library_path);
129}
130
Dimitry Ivanovd9e427c2016-11-22 16:55:25 -0800131static void* dlopen_ext(const char* filename,
132 int flags,
133 const android_dlextinfo* extinfo,
134 const void* caller_addr) {
Elliott Hughes1728b232014-05-14 10:02:03 -0700135 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Dimitry Ivanovb996d602016-07-11 18:11:39 -0700136 g_linker_logger.ResetState();
Dimitry Ivanovd88e1f32016-03-24 15:30:30 -0700137 void* result = do_dlopen(filename, flags, extinfo, caller_addr);
Dmitriy Ivanov851135b2014-08-29 12:02:36 -0700138 if (result == nullptr) {
Elliott Hughes650be4e2013-03-05 18:47:58 -0800139 __bionic_format_dlerror("dlopen failed", linker_get_error_buffer());
Dmitriy Ivanov851135b2014-08-29 12:02:36 -0700140 return nullptr;
Elliott Hughes3b297c42012-10-11 16:08:51 -0700141 }
Elliott Hughes3b297c42012-10-11 16:08:51 -0700142 return result;
143}
144
dimitry7abea572017-08-29 18:14:49 +0200145void* __loader_android_dlopen_ext(const char* filename,
Dimitry Ivanovd9e427c2016-11-22 16:55:25 -0800146 int flags,
147 const android_dlextinfo* extinfo,
148 const void* caller_addr) {
Dmitriy Ivanove5cfafe2015-07-17 10:36:10 -0700149 return dlopen_ext(filename, flags, extinfo, caller_addr);
Dmitriy Ivanovb648a8a2014-05-19 15:06:58 -0700150}
151
dimitry7abea572017-08-29 18:14:49 +0200152void* __loader_dlopen(const char* filename, int flags, const void* caller_addr) {
Dmitriy Ivanove5cfafe2015-07-17 10:36:10 -0700153 return dlopen_ext(filename, flags, nullptr, caller_addr);
Torne (Richard Coles)012cb452014-02-06 14:34:21 +0000154}
155
Dimitry Ivanovd9e427c2016-11-22 16:55:25 -0800156void* dlsym_impl(void* handle, const char* symbol, const char* version, const void* caller_addr) {
Dimitry Ivanov4a2c5aa2015-12-10 16:08:14 -0800157 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Dimitry Ivanovb996d602016-07-11 18:11:39 -0700158 g_linker_logger.ResetState();
Dimitry Ivanov4a2c5aa2015-12-10 16:08:14 -0800159 void* result;
Dimitry Ivanov9cf99cb2015-12-11 14:22:24 -0800160 if (!do_dlsym(handle, symbol, version, caller_addr, &result)) {
Dimitry Ivanov4a2c5aa2015-12-10 16:08:14 -0800161 __bionic_format_dlerror(linker_get_error_buffer(), nullptr);
Dmitriy Ivanov851135b2014-08-29 12:02:36 -0700162 return nullptr;
Elliott Hughes3b297c42012-10-11 16:08:51 -0700163 }
Dimitry Ivanov4a2c5aa2015-12-10 16:08:14 -0800164
165 return result;
Elliott Hughes3b297c42012-10-11 16:08:51 -0700166}
167
dimitry7abea572017-08-29 18:14:49 +0200168void* __loader_dlsym(void* handle, const char* symbol, const void* caller_addr) {
Dimitry Ivanov9cf99cb2015-12-11 14:22:24 -0800169 return dlsym_impl(handle, symbol, nullptr, caller_addr);
170}
171
dimitry7abea572017-08-29 18:14:49 +0200172void* __loader_dlvsym(void* handle, const char* symbol, const char* version, const void* caller_addr) {
Dimitry Ivanov9cf99cb2015-12-11 14:22:24 -0800173 return dlsym_impl(handle, symbol, version, caller_addr);
174}
175
dimitry7abea572017-08-29 18:14:49 +0200176int __loader_dladdr(const void* addr, Dl_info* info) {
Elliott Hughes1728b232014-05-14 10:02:03 -0700177 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Dimitry Ivanov4a2c5aa2015-12-10 16:08:14 -0800178 return do_dladdr(addr, info);
Elliott Hughes3b297c42012-10-11 16:08:51 -0700179}
180
dimitry7abea572017-08-29 18:14:49 +0200181int __loader_dlclose(void* handle) {
Elliott Hughes1728b232014-05-14 10:02:03 -0700182 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Dimitry Ivanovd88e1f32016-03-24 15:30:30 -0700183 int result = do_dlclose(handle);
184 if (result != 0) {
185 __bionic_format_dlerror("dlclose failed", linker_get_error_buffer());
186 }
187 return result;
Elliott Hughes3b297c42012-10-11 16:08:51 -0700188}
189
dimitry7abea572017-08-29 18:14:49 +0200190int __loader_dl_iterate_phdr(int (*cb)(dl_phdr_info* info, size_t size, void* data), void* data) {
Dmitriy Ivanov7271caf2015-06-29 14:48:25 -0700191 ScopedPthreadMutexLocker locker(&g_dl_mutex);
192 return do_dl_iterate_phdr(cb, data);
193}
194
dimitry7abea572017-08-29 18:14:49 +0200195// This function is needed by libgcc.a
196int dl_iterate_phdr(int (*cb)(dl_phdr_info* info, size_t size, void* data), void* data) {
197 return __loader_dl_iterate_phdr(cb, data);
198}
199
Dimitry Ivanovd9e427c2016-11-22 16:55:25 -0800200#if defined(__arm__)
dimitry7abea572017-08-29 18:14:49 +0200201_Unwind_Ptr __loader_dl_unwind_find_exidx(_Unwind_Ptr pc, int* pcount) {
Dimitry Ivanovd9e427c2016-11-22 16:55:25 -0800202 ScopedPthreadMutexLocker locker(&g_dl_mutex);
203 return do_dl_unwind_find_exidx(pc, pcount);
204}
205#endif
206
dimitry7abea572017-08-29 18:14:49 +0200207void __loader_android_set_application_target_sdk_version(uint32_t target) {
Dmitriy Ivanovd974e882015-05-27 18:29:41 -0700208 // lock to avoid modification in the middle of dlopen.
209 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Dmitriy Ivanov79fd6682015-05-21 17:43:49 -0700210 set_application_target_sdk_version(target);
211}
212
dimitry7abea572017-08-29 18:14:49 +0200213uint32_t __loader_android_get_application_target_sdk_version() {
Dmitriy Ivanov79fd6682015-05-21 17:43:49 -0700214 return get_application_target_sdk_version();
215}
216
dimitry7abea572017-08-29 18:14:49 +0200217void __loader_android_dlwarning(void* obj, void (*f)(void*, const char*)) {
Dimitry Ivanov769b33f2016-07-21 11:33:40 -0700218 ScopedPthreadMutexLocker locker(&g_dl_mutex);
219 get_dlwarning(obj, f);
220}
221
dimitry7abea572017-08-29 18:14:49 +0200222bool __loader_android_init_anonymous_namespace(const char* shared_libs_sonames,
223 const char* library_search_path) {
Dmitriy Ivanov42d5fcb2015-10-29 17:01:24 -0700224 ScopedPthreadMutexLocker locker(&g_dl_mutex);
Dimitry Ivanov7a34b9d2017-02-03 14:07:34 -0800225 bool success = init_anonymous_namespace(shared_libs_sonames, library_search_path);
Dmitriy Ivanov42d5fcb2015-10-29 17:01:24 -0700226 if (!success) {
Dimitry Ivanov7a34b9d2017-02-03 14:07:34 -0800227 __bionic_format_dlerror("android_init_anonymous_namespace failed", linker_get_error_buffer());
Dmitriy Ivanov42d5fcb2015-10-29 17:01:24 -0700228 }
229
230 return success;
231}
232
dimitry7abea572017-08-29 18:14:49 +0200233android_namespace_t* __loader_android_create_namespace(const char* name,
Dimitry Ivanovd9e427c2016-11-22 16:55:25 -0800234 const char* ld_library_path,
235 const char* default_library_path,
236 uint64_t type,
237 const char* permitted_when_isolated_path,
238 android_namespace_t* parent_namespace,
239 const void* caller_addr) {
Dmitriy Ivanov42d5fcb2015-10-29 17:01:24 -0700240 ScopedPthreadMutexLocker locker(&g_dl_mutex);
241
Dimitry Ivanovfc2da532016-05-12 15:20:21 -0700242 android_namespace_t* result = create_namespace(caller_addr,
243 name,
244 ld_library_path,
245 default_library_path,
246 type,
247 permitted_when_isolated_path,
248 parent_namespace);
Dmitriy Ivanov42d5fcb2015-10-29 17:01:24 -0700249
250 if (result == nullptr) {
251 __bionic_format_dlerror("android_create_namespace failed", linker_get_error_buffer());
252 }
253
254 return result;
255}
256
dimitry7abea572017-08-29 18:14:49 +0200257bool __loader_android_link_namespaces(android_namespace_t* namespace_from,
258 android_namespace_t* namespace_to,
259 const char* shared_libs_sonames) {
Dimitry Ivanov7a34b9d2017-02-03 14:07:34 -0800260 ScopedPthreadMutexLocker locker(&g_dl_mutex);
261
262 bool success = link_namespaces(namespace_from, namespace_to, shared_libs_sonames);
263
264 if (!success) {
265 __bionic_format_dlerror("android_link_namespaces failed", linker_get_error_buffer());
266 }
267
268 return success;
269}
270
Logan Chien9ee45912018-01-18 12:05:09 +0800271bool __loader_android_link_namespaces_all_libs(android_namespace_t* namespace_from,
272 android_namespace_t* namespace_to) {
273 ScopedPthreadMutexLocker locker(&g_dl_mutex);
274
275 bool success = link_namespaces_all_libs(namespace_from, namespace_to);
276
277 if (!success) {
278 __bionic_format_dlerror("android_link_namespaces_all_libs failed", linker_get_error_buffer());
279 }
280
281 return success;
282}
283
dimitry7abea572017-08-29 18:14:49 +0200284android_namespace_t* __loader_android_get_exported_namespace(const char* name) {
Jiyong Park01de74e2017-04-03 23:10:37 +0900285 return get_exported_namespace(name);
286}
287
dimitry7abea572017-08-29 18:14:49 +0200288void __loader_cfi_fail(uint64_t CallSiteTypeId, void* Ptr, void *DiagData, void *CallerPc) {
Evgenii Stepanov0a3637d2016-07-06 13:20:59 -0700289 CFIShadowWriter::CfiFail(CallSiteTypeId, Ptr, DiagData, CallerPc);
290}
291
dimitry06016f22018-01-05 11:39:28 +0100292void __loader_add_thread_local_dtor(void* dso_handle) {
293 ScopedPthreadMutexLocker locker(&g_dl_mutex);
294 increment_dso_handle_reference_counter(dso_handle);
295}
296
297void __loader_remove_thread_local_dtor(void* dso_handle) {
298 ScopedPthreadMutexLocker locker(&g_dl_mutex);
299 decrement_dso_handle_reference_counter(dso_handle);
300}
301
Dmitriy Ivanovaae859c2015-03-31 11:14:03 -0700302static uint8_t __libdl_info_buf[sizeof(soinfo)] __attribute__((aligned(8)));
303static soinfo* __libdl_info = nullptr;
Dmitriy Ivanovd59e5002014-05-09 09:10:14 -0700304
Elliott Hughes22d62922012-10-12 10:50:21 -0700305// This is used by the dynamic linker. Every process gets these symbols for free.
dimitry7abea572017-08-29 18:14:49 +0200306soinfo* get_libdl_info(const char* linker_path,
307 const soinfo& linker_si,
308 const link_map& linker_map) {
309 CHECK((linker_si.flags_ & FLAG_GNU_HASH) != 0);
310
Dmitriy Ivanovaae859c2015-03-31 11:14:03 -0700311 if (__libdl_info == nullptr) {
Dimitry Ivanovd9e427c2016-11-22 16:55:25 -0800312 __libdl_info = new (__libdl_info_buf) soinfo(&g_default_namespace, linker_path, nullptr, 0, 0);
dimitry7abea572017-08-29 18:14:49 +0200313 __libdl_info->flags_ |= (FLAG_LINKED | FLAG_GNU_HASH);
314 __libdl_info->strtab_ = linker_si.strtab_;
315 __libdl_info->symtab_ = linker_si.symtab_;
316 __libdl_info->load_bias = linker_si.load_bias;
317
318 __libdl_info->gnu_nbucket_ = linker_si.gnu_nbucket_;
319 __libdl_info->gnu_maskwords_ = linker_si.gnu_maskwords_;
320 __libdl_info->gnu_shift2_ = linker_si.gnu_shift2_;
321 __libdl_info->gnu_bloom_filter_ = linker_si.gnu_bloom_filter_;
322 __libdl_info->gnu_bucket_ = linker_si.gnu_bucket_;
323 __libdl_info->gnu_chain_ = linker_si.gnu_chain_;
324
Dmitriy Ivanovaae859c2015-03-31 11:14:03 -0700325 __libdl_info->ref_count_ = 1;
dimitry7abea572017-08-29 18:14:49 +0200326 __libdl_info->strtab_size_ = linker_si.strtab_size_;
Dmitriy Ivanovaae859c2015-03-31 11:14:03 -0700327 __libdl_info->local_group_root_ = __libdl_info;
dimitry8e8c2c02018-01-04 12:08:32 +0100328 __libdl_info->soname_ = linker_si.soname_;
Dmitriy Ivanov19133522015-06-02 17:36:54 -0700329 __libdl_info->target_sdk_version_ = __ANDROID_API__;
Dimitry Ivanovd88e1f32016-03-24 15:30:30 -0700330 __libdl_info->generate_handle();
Dimitry Ivanovcd510cb2017-05-31 15:07:41 -0700331 __libdl_info->link_map_head.l_addr = linker_map.l_addr;
332 __libdl_info->link_map_head.l_name = linker_map.l_name;
333 __libdl_info->link_map_head.l_ld = linker_map.l_ld;
Mike Frysinger747d30e2015-10-20 14:06:25 -0400334#if defined(__work_around_b_24465209__)
Dmitriy Ivanovaae859c2015-03-31 11:14:03 -0700335 strlcpy(__libdl_info->old_name_, __libdl_info->soname_, sizeof(__libdl_info->old_name_));
336#endif
Dmitriy Ivanovd59e5002014-05-09 09:10:14 -0700337 }
Elliott Hughes3b297c42012-10-11 16:08:51 -0700338
Dmitriy Ivanovaae859c2015-03-31 11:14:03 -0700339 return __libdl_info;
Dmitriy Ivanovd59e5002014-05-09 09:10:14 -0700340}