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