2014-11-04 17:38:18 +00:00
/*
* Copyright ( C ) 2012 The Android Open Source Project
*
* Licensed under the Apache License , Version 2.0 ( the " License " ) ;
* you may not use this file except in compliance with the License .
* You may obtain a copy of the License at
*
* http : //www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing , software
* distributed under the License is distributed on an " AS IS " BASIS ,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND , either express or implied .
* See the License for the specific language governing permissions and
* limitations under the License .
*/
# include <gtest/gtest.h>
2017-08-30 02:30:53 +00:00
# if defined(__BIONIC__)
# include <android-base/properties.h>
# endif
2014-11-04 17:38:18 +00:00
# include <dlfcn.h>
# include <libgen.h>
# include <limits.h>
# include <stdio.h>
# include <stdint.h>
# include <string>
2017-05-19 16:01:24 +00:00
# include <iostream>
# include <fstream>
2014-11-04 17:38:18 +00:00
2017-03-29 20:48:02 +00:00
# include "gtest_globals.h"
2018-11-13 15:35:21 +00:00
# include <android-base/file.h>
2017-02-23 19:53:43 +00:00
# include "utils.h"
2014-11-04 17:38:18 +00:00
extern " C " int main_global_default_serial ( ) {
return 3370318 ;
}
extern " C " int main_global_protected_serial ( ) {
return 2716057 ;
}
// The following functions are defined in DT_NEEDED
// libdl_preempt_test.so library.
// This one calls main_global_default_serial
extern " C " int main_global_default_get_serial ( ) ;
// This one calls main_global_protected_serial
extern " C " int main_global_protected_get_serial ( ) ;
// This one calls lib_global_default_serial
extern " C " int lib_global_default_get_serial ( ) ;
// This one calls lib_global_protected_serial
extern " C " int lib_global_protected_get_serial ( ) ;
// This test verifies that the global default function
// main_global_default_serial() is preempted by
// the function defined above.
TEST ( dl , main_preempts_global_default ) {
ASSERT_EQ ( 3370318 , main_global_default_get_serial ( ) ) ;
}
// This one makes sure that the global protected
// symbols do not get preempted
TEST ( dl , main_does_not_preempt_global_protected ) {
ASSERT_EQ ( 3370318 , main_global_protected_get_serial ( ) ) ;
}
// check same things for lib
TEST ( dl , lib_preempts_global_default ) {
ASSERT_EQ ( 3370318 , lib_global_default_get_serial ( ) ) ;
}
TEST ( dl , lib_does_not_preempt_global_protected ) {
ASSERT_EQ ( 3370318 , lib_global_protected_get_serial ( ) ) ;
}
2017-02-23 19:53:43 +00:00
# if defined(__BIONIC__)
# if defined(__LP64__)
static constexpr const char * kPathToLinker = " /system/bin/linker64 " ;
# else
static constexpr const char * kPathToLinker = " /system/bin/linker " ;
# endif
2018-10-02 06:10:05 +00:00
# endif
TEST ( dl , exec_linker ) {
# if defined(__BIONIC__)
std : : string usage_prefix = std : : string ( " Usage: " ) + kPathToLinker ;
ExecTestHelper eth ;
eth . SetArgs ( { kPathToLinker , nullptr } ) ;
eth . Run ( [ & ] ( ) { execve ( kPathToLinker , eth . GetArgs ( ) , eth . GetEnv ( ) ) ; } , 0 , nullptr ) ;
ASSERT_EQ ( 0u , eth . GetOutput ( ) . find ( usage_prefix ) ) < < " Test output: \n " < < eth . GetOutput ( ) ;
# endif
}
TEST ( dl , exec_linker_load_file ) {
# if defined(__BIONIC__)
std : : string helper = GetTestlibRoot ( ) +
" /exec_linker_helper/exec_linker_helper " ;
std : : string expected_output =
" ctor: argc=1 argv[0]= " + helper + " \n " +
" main: argc=1 argv[0]= " + helper + " \n " +
2018-11-22 10:41:36 +00:00
" __progname= " + helper + " \n " +
2018-10-02 06:10:05 +00:00
" helper_func called \n " ;
2017-02-23 19:53:43 +00:00
ExecTestHelper eth ;
2018-10-02 06:10:05 +00:00
eth . SetArgs ( { kPathToLinker , helper . c_str ( ) , nullptr } ) ;
2017-02-23 19:53:43 +00:00
eth . Run ( [ & ] ( ) { execve ( kPathToLinker , eth . GetArgs ( ) , eth . GetEnv ( ) ) ; } , 0 , expected_output . c_str ( ) ) ;
# endif
}
2018-10-02 06:10:05 +00:00
TEST ( dl , exec_linker_load_from_zip ) {
# if defined(__BIONIC__)
std : : string helper = GetTestlibRoot ( ) +
" /libdlext_test_zip/libdlext_test_zip_zipaligned.zip!/libdir/exec_linker_helper " ;
std : : string expected_output =
" ctor: argc=1 argv[0]= " + helper + " \n " +
" main: argc=1 argv[0]= " + helper + " \n " +
2018-11-22 10:41:36 +00:00
" __progname= " + helper + " \n " +
2018-10-02 06:10:05 +00:00
" helper_func called \n " ;
ExecTestHelper eth ;
eth . SetArgs ( { kPathToLinker , helper . c_str ( ) , nullptr } ) ;
eth . Run ( [ & ] ( ) { execve ( kPathToLinker , eth . GetArgs ( ) , eth . GetEnv ( ) ) ; } , 0 , expected_output . c_str ( ) ) ;
# endif
}
TEST ( dl , exec_linker_load_self ) {
# if defined(__BIONIC__)
std : : string error_message = " error: linker cannot load itself \n " ;
ExecTestHelper eth ;
eth . SetArgs ( { kPathToLinker , kPathToLinker , nullptr } ) ;
eth . Run ( [ & ] ( ) { execve ( kPathToLinker , eth . GetArgs ( ) , eth . GetEnv ( ) ) ; } , EXIT_FAILURE , error_message . c_str ( ) ) ;
# endif
}
2017-03-29 20:48:02 +00:00
TEST ( dl , preinit_system_calls ) {
# if defined(__BIONIC__)
2019-03-08 23:20:23 +00:00
SKIP_WITH_HWASAN < < " hwasan not initialized in preinit_array, b/124007027 " ;
2018-08-22 01:13:10 +00:00
std : : string helper = GetTestlibRoot ( ) +
2017-03-29 20:48:02 +00:00
" /preinit_syscall_test_helper/preinit_syscall_test_helper " ;
chmod ( helper . c_str ( ) , 0755 ) ; // TODO: "x" lost in CTS, b/34945607
ExecTestHelper eth ;
eth . SetArgs ( { helper . c_str ( ) , nullptr } ) ;
eth . Run ( [ & ] ( ) { execve ( helper . c_str ( ) , eth . GetArgs ( ) , eth . GetEnv ( ) ) ; } , 0 , nullptr ) ;
# endif
}
2018-11-22 10:14:14 +00:00
TEST ( dl , preinit_getauxval ) {
2017-03-29 20:48:02 +00:00
# if defined(__BIONIC__)
2019-03-08 23:20:23 +00:00
SKIP_WITH_HWASAN < < " hwasan not initialized in preinit_array, b/124007027 " ;
2018-08-22 01:13:10 +00:00
std : : string helper = GetTestlibRoot ( ) +
2017-03-29 20:48:02 +00:00
" /preinit_getauxval_test_helper/preinit_getauxval_test_helper " ;
chmod ( helper . c_str ( ) , 0755 ) ; // TODO: "x" lost in CTS, b/34945607
ExecTestHelper eth ;
eth . SetArgs ( { helper . c_str ( ) , nullptr } ) ;
eth . Run ( [ & ] ( ) { execve ( helper . c_str ( ) , eth . GetArgs ( ) , eth . GetEnv ( ) ) ; } , 0 , nullptr ) ;
2018-11-07 22:39:43 +00:00
# else
// Force a failure when not compiled for bionic so the test is considered a pass.
ASSERT_TRUE ( false ) ;
2017-03-29 20:48:02 +00:00
# endif
}
2017-05-19 16:01:24 +00:00
TEST ( dl , exec_without_ld_preload ) {
# if defined(__BIONIC__)
2018-08-22 01:13:10 +00:00
std : : string helper = GetTestlibRoot ( ) +
2017-05-19 16:01:24 +00:00
" /ld_preload_test_helper/ld_preload_test_helper " ;
chmod ( helper . c_str ( ) , 0755 ) ;
ExecTestHelper eth ;
eth . SetArgs ( { helper . c_str ( ) , nullptr } ) ;
eth . Run ( [ & ] ( ) { execve ( helper . c_str ( ) , eth . GetArgs ( ) , eth . GetEnv ( ) ) ; } , 0 , " 12345 " ) ;
# endif
}
TEST ( dl , exec_with_ld_preload ) {
# if defined(__BIONIC__)
2018-08-22 01:13:10 +00:00
std : : string helper = GetTestlibRoot ( ) +
2017-05-19 16:01:24 +00:00
" /ld_preload_test_helper/ld_preload_test_helper " ;
2018-08-22 01:13:10 +00:00
std : : string env = std : : string ( " LD_PRELOAD= " ) + GetTestlibRoot ( ) + " /ld_preload_test_helper_lib2.so " ;
2017-05-19 16:01:24 +00:00
chmod ( helper . c_str ( ) , 0755 ) ;
ExecTestHelper eth ;
eth . SetArgs ( { helper . c_str ( ) , nullptr } ) ;
eth . SetEnv ( { env . c_str ( ) , nullptr } ) ;
// ld_preload_test_helper calls get_value_from_lib() and returns the value.
// The symbol is defined by two libs: ld_preload_test_helper_lib.so and
// ld_preloaded_lib.so. The former is DT_NEEDED and the latter is LD_PRELOADED
// via this execution. The main executable is linked to the LD_PRELOADED lib
// and the value given from the lib is returned.
eth . Run ( [ & ] ( ) { execve ( helper . c_str ( ) , eth . GetArgs ( ) , eth . GetEnv ( ) ) ; } , 0 , " 54321 " ) ;
# endif
}
// ld_config_test_helper must fail because it is depending on a lib which is not
// in the search path
//
// Call sequence is...
// _helper -- (get_value_from_lib()) -->
// _lib1.so -- (get_value_from_another_lib()) -->
// _lib2.so (returns 12345)
// The two libs are in ns2/ subdir.
TEST ( dl , exec_without_ld_config_file ) {
# if defined(__BIONIC__)
2019-10-29 20:41:00 +00:00
std : : string error_message =
" CANNOT LINK EXECUTABLE \" " + GetTestlibRoot ( ) +
" /ld_config_test_helper/ld_config_test_helper \" : library \" ld_config_test_helper_lib1.so \" "
" not found: needed by main executable \n " ;
2018-08-22 01:13:10 +00:00
std : : string helper = GetTestlibRoot ( ) +
2017-05-19 16:01:24 +00:00
" /ld_config_test_helper/ld_config_test_helper " ;
chmod ( helper . c_str ( ) , 0755 ) ;
ExecTestHelper eth ;
eth . SetArgs ( { helper . c_str ( ) , nullptr } ) ;
2017-09-29 09:52:17 +00:00
eth . Run ( [ & ] ( ) { execve ( helper . c_str ( ) , eth . GetArgs ( ) , eth . GetEnv ( ) ) ; } , EXIT_FAILURE , error_message . c_str ( ) ) ;
2017-05-19 16:01:24 +00:00
# endif
}
# if defined(__BIONIC__)
2018-05-09 12:37:47 +00:00
extern " C " void android_get_LD_LIBRARY_PATH ( char * , size_t ) ;
2018-04-04 03:03:12 +00:00
static void create_ld_config_file ( const char * config_file ) {
2018-05-09 12:37:47 +00:00
char default_search_paths [ PATH_MAX ] ;
android_get_LD_LIBRARY_PATH ( default_search_paths , sizeof ( default_search_paths ) ) ;
2018-04-04 03:03:12 +00:00
std : : ofstream fout ( config_file , std : : ios : : out ) ;
2018-08-22 01:13:10 +00:00
fout < < " dir.test = " < < GetTestlibRoot ( ) < < " /ld_config_test_helper/ " < < std : : endl
2017-05-19 16:01:24 +00:00
< < " [test] " < < std : : endl
< < " additional.namespaces = ns2 " < < std : : endl
2018-08-22 01:13:10 +00:00
< < " namespace.default.search.paths = " < < GetTestlibRoot ( ) < < std : : endl
2017-05-19 16:01:24 +00:00
< < " namespace.default.links = ns2 " < < std : : endl
< < " namespace.default.link.ns2.shared_libs = libc.so:libm.so:libdl.so:ld_config_test_helper_lib1.so " < < std : : endl
2018-08-22 01:13:10 +00:00
< < " namespace.ns2.search.paths = " < < default_search_paths < < " : " < < GetTestlibRoot ( ) < < " /ns2 " < < std : : endl ;
2017-05-19 16:01:24 +00:00
fout . close ( ) ;
}
# endif
2017-09-19 00:48:07 +00:00
# if defined(__BIONIC__)
2017-10-18 11:23:08 +00:00
static bool is_debuggable_build ( ) {
return android : : base : : GetBoolProperty ( " ro.debuggable " , false ) ;
2017-09-19 00:48:07 +00:00
}
# endif
2017-05-19 16:01:24 +00:00
// _lib1.so and _lib2.so are now searchable by having another namespace 'ns2'
// whose search paths include the 'ns2/' subdir.
TEST ( dl , exec_with_ld_config_file ) {
# if defined(__BIONIC__)
2019-03-08 23:20:23 +00:00
SKIP_WITH_HWASAN < < " libclang_rt.hwasan is not found with custom ld config " ;
2017-10-18 11:23:08 +00:00
if ( ! is_debuggable_build ( ) ) {
2017-09-19 00:48:07 +00:00
// LD_CONFIG_FILE is not supported on user build
return ;
}
2018-08-22 01:13:10 +00:00
std : : string helper = GetTestlibRoot ( ) +
2017-05-19 16:01:24 +00:00
" /ld_config_test_helper/ld_config_test_helper " ;
2018-04-04 03:03:12 +00:00
TemporaryFile config_file ;
2018-11-13 15:35:21 +00:00
create_ld_config_file ( config_file . path ) ;
std : : string env = std : : string ( " LD_CONFIG_FILE= " ) + config_file . path ;
2017-05-19 16:01:24 +00:00
chmod ( helper . c_str ( ) , 0755 ) ;
ExecTestHelper eth ;
eth . SetArgs ( { helper . c_str ( ) , nullptr } ) ;
eth . SetEnv ( { env . c_str ( ) , nullptr } ) ;
eth . Run ( [ & ] ( ) { execve ( helper . c_str ( ) , eth . GetArgs ( ) , eth . GetEnv ( ) ) ; } , 0 , " 12345 " ) ;
# endif
}
// _lib3.so has same symbol as lib2.so but returns 54321. _lib3.so is
// LD_PRELOADed. This test is to ensure LD_PRELOADed libs are available to
// additional namespaces other than the default namespace.
TEST ( dl , exec_with_ld_config_file_with_ld_preload ) {
# if defined(__BIONIC__)
2019-03-08 23:20:23 +00:00
SKIP_WITH_HWASAN < < " libclang_rt.hwasan is not found with custom ld config " ;
2017-10-18 11:23:08 +00:00
if ( ! is_debuggable_build ( ) ) {
2017-09-19 00:48:07 +00:00
// LD_CONFIG_FILE is not supported on user build
return ;
}
2018-08-22 01:13:10 +00:00
std : : string helper = GetTestlibRoot ( ) +
2017-05-19 16:01:24 +00:00
" /ld_config_test_helper/ld_config_test_helper " ;
2018-04-04 03:03:12 +00:00
TemporaryFile config_file ;
2018-11-13 15:35:21 +00:00
create_ld_config_file ( config_file . path ) ;
std : : string env = std : : string ( " LD_CONFIG_FILE= " ) + config_file . path ;
2018-08-22 01:13:10 +00:00
std : : string env2 = std : : string ( " LD_PRELOAD= " ) + GetTestlibRoot ( ) + " /ld_config_test_helper_lib3.so " ;
2017-05-19 16:01:24 +00:00
chmod ( helper . c_str ( ) , 0755 ) ;
ExecTestHelper eth ;
eth . SetArgs ( { helper . c_str ( ) , nullptr } ) ;
eth . SetEnv ( { env . c_str ( ) , env2 . c_str ( ) , nullptr } ) ;
eth . Run ( [ & ] ( ) { execve ( helper . c_str ( ) , eth . GetArgs ( ) , eth . GetEnv ( ) ) ; } , 0 , " 54321 " ) ;
# endif
}
// ensures that LD_CONFIG_FILE env var does not work for production builds.
// The test input is the same as exec_with_ld_config_file, but it must fail in
// this case.
TEST ( dl , disable_ld_config_file ) {
# if defined(__BIONIC__)
if ( getuid ( ) = = 0 ) {
// when executed from the shell (e.g. not as part of CTS), skip the test.
// This test is only for CTS.
return ;
}
2017-10-18 11:23:08 +00:00
if ( is_debuggable_build ( ) ) {
2017-08-30 02:30:53 +00:00
// Skip the test for non production devices
return ;
}
2018-08-22 01:13:10 +00:00
std : : string error_message = " CANNOT LINK EXECUTABLE \" " + GetTestlibRoot ( ) + " /ld_config_test_helper/ld_config_test_helper \" : library \" ld_config_test_helper_lib1.so \" not found \n " ;
std : : string helper = GetTestlibRoot ( ) +
2017-05-19 16:01:24 +00:00
" /ld_config_test_helper/ld_config_test_helper " ;
2018-04-04 03:03:12 +00:00
TemporaryFile config_file ;
2018-11-13 15:35:21 +00:00
create_ld_config_file ( config_file . path ) ;
std : : string env = std : : string ( " LD_CONFIG_FILE= " ) + config_file . path ;
2017-05-19 16:01:24 +00:00
chmod ( helper . c_str ( ) , 0755 ) ;
ExecTestHelper eth ;
eth . SetArgs ( { helper . c_str ( ) , nullptr } ) ;
eth . SetEnv ( { env . c_str ( ) , nullptr } ) ;
2017-11-28 04:37:03 +00:00
eth . Run ( [ & ] ( ) { execve ( helper . c_str ( ) , eth . GetArgs ( ) , eth . GetEnv ( ) ) ; } , EXIT_FAILURE , error_message . c_str ( ) ) ;
2017-05-19 16:01:24 +00:00
# endif
}