Home | History | Annotate | Line # | Download | only in ld.elf_so
t_ifunc.c revision 1.7
      1 /*	$NetBSD: t_ifunc.c,v 1.7 2018/01/01 06:34:13 maya Exp $	*/
      2 
      3 /*
      4  * Copyright (c) 2014 The NetBSD Foundation, Inc.
      5  * All rights reserved.
      6  *
      7  * Redistribution and use in source and binary forms, with or without
      8  * modification, are permitted provided that the following conditions
      9  * are met:
     10  * 1. Redistributions of source code must retain the above copyright
     11  *    notice, this list of conditions and the following disclaimer.
     12  * 2. Redistributions in binary form must reproduce the above copyright
     13  *    notice, this list of conditions and the following disclaimer in the
     14  *    documentation and/or other materials provided with the distribution.
     15  *
     16  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND
     17  * CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
     18  * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
     19  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
     20  * IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS BE LIABLE FOR ANY
     21  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
     22  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE
     23  * GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
     24  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER
     25  * IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
     26  * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
     27  * IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
     28  */
     29 
     30 #include <sys/types.h>
     31 
     32 #include <atf-c.h>
     33 #include <dlfcn.h>
     34 #include <util.h>
     35 
     36 #include "h_macros.h"
     37 
     38 #if defined( __arm__) || defined(__i386__) || defined(__x86_64__) || defined(__powerpc__) || defined(__sparc__)
     39 #define	LINKER_SUPPORT 1
     40 #else
     41 #define	LINKER_SUPPORT 0
     42 #endif
     43 
     44 ATF_TC(rtld_ifunc);
     45 
     46 ATF_TC_HEAD(rtld_ifunc, tc)
     47 {
     48 	atf_tc_set_md_var(tc, "descr", "ifunc functions are resolved");
     49 }
     50 
     51 ATF_TC_BODY(rtld_ifunc, tc)
     52 {
     53 	const char *envstr[] = {
     54 	    "0", "1"
     55 	};
     56 	int expected_result[] = {
     57 	    0xdeadbeef, 0xbeefdead
     58 	};
     59 	void *handle;
     60 	int (*sym)(void);
     61 	int result;
     62 	const char *error;
     63 	size_t i;
     64 
     65 	if (!LINKER_SUPPORT)
     66 		atf_tc_skip("Missing linker support for ifunc relocations");
     67 
     68 	for (i = 0; i < __arraycount(envstr); ++i) {
     69 		setenv("USE_IFUNC2", envstr[i], 1);
     70 
     71 		handle = dlopen("libh_helper_ifunc_dso.so", RTLD_LAZY);
     72 		error = dlerror();
     73 		ATF_CHECK(error == NULL);
     74 		ATF_CHECK(handle != NULL);
     75 
     76 		sym = dlsym(handle, "ifunc");
     77 		error = dlerror();
     78 		ATF_CHECK(error == NULL);
     79 		ATF_CHECK(sym != NULL);
     80 
     81 		result = (*sym)();
     82 		ATF_CHECK(result == expected_result[i]);
     83 
     84 		dlclose(handle);
     85 		error = dlerror();
     86 		ATF_CHECK(error == NULL);
     87 
     88 		char *command;
     89 		easprintf(&command, "%s/h_ifunc %d",
     90 		    atf_tc_get_config_var(tc, "srcdir"), expected_result[i]);
     91 		if (system(command) != EXIT_SUCCESS)
     92 			atf_tc_fail("Test failed; see output for details");
     93 		free(command);
     94 	}
     95 }
     96 
     97 ATF_TC(rtld_hidden_ifunc);
     98 
     99 ATF_TC_HEAD(rtld_hidden_ifunc, tc)
    100 {
    101 	atf_tc_set_md_var(tc, "descr", "hidden ifunc functions are resolved");
    102 }
    103 
    104 ATF_TC_BODY(rtld_hidden_ifunc, tc)
    105 {
    106 	const char *envstr[] = {
    107 	    "0", "1"
    108 	};
    109 	int expected_result[] = {
    110 	    0xdeadbeef, 0xbeefdead
    111 	};
    112 	void *handle;
    113 	int (*sym)(void);
    114 	int (*(*sym2)(void))(void);
    115 	int result;
    116 	const char *error;
    117 	size_t i;
    118 
    119 	if (!LINKER_SUPPORT)
    120 		atf_tc_skip("Missing linker support for ifunc relocations");
    121 
    122 	for (i = 0; i < __arraycount(envstr); ++i) {
    123 		setenv("USE_IFUNC2", envstr[i], 1);
    124 
    125 		handle = dlopen("libh_helper_ifunc_dso.so", RTLD_LAZY);
    126 		error = dlerror();
    127 		ATF_CHECK(error == NULL);
    128 		ATF_CHECK(handle != NULL);
    129 
    130 		sym = dlsym(handle, "ifunc_plt");
    131 		error = dlerror();
    132 		ATF_CHECK(error == NULL);
    133 		ATF_CHECK(sym != NULL);
    134 
    135 		result = (*sym)();
    136 		ATF_CHECK(result == expected_result[!i]);
    137 
    138 		sym2 = dlsym(handle, "ifunc_indirect");
    139 		error = dlerror();
    140 		ATF_CHECK(error == NULL);
    141 		ATF_CHECK(sym2 != NULL);
    142 
    143 		sym = (*sym2)();
    144 		result = (*sym)();
    145 		ATF_CHECK(result == expected_result[!i]);
    146 
    147 		dlclose(handle);
    148 		error = dlerror();
    149 		ATF_CHECK(error == NULL);
    150 
    151 		char *command;
    152 		easprintf(&command, "%s/h_ifunc %d",
    153 		    atf_tc_get_config_var(tc, "srcdir"), expected_result[i]);
    154 		if (system(command) != EXIT_SUCCESS)
    155 			atf_tc_fail("Test failed; see output for details");
    156 		free(command);
    157 	}
    158 }
    159 
    160 ATF_TC(rtld_main_ifunc);
    161 ATF_TC_HEAD(rtld_main_ifunc, tc)
    162 {
    163 	atf_tc_set_md_var(tc, "descr",
    164 	    "ifunc functions are resolved in the executable");
    165 }
    166 
    167 #if LINKER_SUPPORT
    168 static unsigned int
    169 ifunc_helper(void)
    170 {
    171 	return 0xdeadbeef;
    172 }
    173 
    174 static __attribute__((used))
    175 unsigned int (*resolve_ifunc(void))(void)
    176 {
    177 	return ifunc_helper;
    178 }
    179 __hidden_ifunc(ifunc, resolve_ifunc);
    180 #endif
    181 unsigned int ifunc(void);
    182 
    183 ATF_TC_BODY(rtld_main_ifunc, tc)
    184 {
    185 	if (!LINKER_SUPPORT)
    186 		atf_tc_skip("Missing linker support for ifunc relocations");
    187 	ATF_CHECK(ifunc() == 0xdeadbeef);
    188 }
    189 
    190 ATF_TP_ADD_TCS(tp)
    191 {
    192 	ATF_TP_ADD_TC(tp, rtld_ifunc);
    193 	ATF_TP_ADD_TC(tp, rtld_hidden_ifunc);
    194 	ATF_TP_ADD_TC(tp, rtld_main_ifunc);
    195 	return 0;
    196 }
    197