Home | History | Annotate | Line # | Download | only in ld.elf_so
t_ifunc.c revision 1.7.2.1
      1  1.7.2.1  pgoyette /*	$NetBSD: t_ifunc.c,v 1.7.2.1 2018/03/15 09:12:07 pgoyette 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.7      maya #if defined( __arm__) || defined(__i386__) || defined(__x86_64__) || defined(__powerpc__) || defined(__sparc__)
     39      1.7      maya #define	LINKER_SUPPORT 1
     40      1.6    martin #else
     41      1.7      maya #define	LINKER_SUPPORT 0
     42      1.6    martin #endif
     43      1.6    martin 
     44      1.1     joerg ATF_TC(rtld_ifunc);
     45      1.1     joerg 
     46      1.1     joerg ATF_TC_HEAD(rtld_ifunc, tc)
     47      1.1     joerg {
     48      1.1     joerg 	atf_tc_set_md_var(tc, "descr", "ifunc functions are resolved");
     49      1.1     joerg }
     50      1.1     joerg 
     51      1.1     joerg ATF_TC_BODY(rtld_ifunc, tc)
     52      1.1     joerg {
     53      1.1     joerg 	const char *envstr[] = {
     54      1.1     joerg 	    "0", "1"
     55      1.1     joerg 	};
     56  1.7.2.1  pgoyette 	long long expected_result[] = {
     57  1.7.2.1  pgoyette 	    0xdeadbeefll, 0xbeefdeadll
     58      1.1     joerg 	};
     59      1.1     joerg 	void *handle;
     60  1.7.2.1  pgoyette 	long long (*sym)(void);
     61  1.7.2.1  pgoyette 	long long result;
     62      1.1     joerg 	const char *error;
     63      1.1     joerg 	size_t i;
     64      1.1     joerg 
     65      1.7      maya 	if (!LINKER_SUPPORT)
     66      1.7      maya 		atf_tc_skip("Missing linker support for ifunc relocations");
     67      1.3     joerg 
     68      1.1     joerg 	for (i = 0; i < __arraycount(envstr); ++i) {
     69      1.1     joerg 		setenv("USE_IFUNC2", envstr[i], 1);
     70      1.1     joerg 
     71      1.1     joerg 		handle = dlopen("libh_helper_ifunc_dso.so", RTLD_LAZY);
     72      1.1     joerg 		error = dlerror();
     73      1.1     joerg 		ATF_CHECK(error == NULL);
     74      1.1     joerg 		ATF_CHECK(handle != NULL);
     75      1.1     joerg 
     76      1.1     joerg 		sym = dlsym(handle, "ifunc");
     77      1.1     joerg 		error = dlerror();
     78      1.1     joerg 		ATF_CHECK(error == NULL);
     79      1.1     joerg 		ATF_CHECK(sym != NULL);
     80      1.1     joerg 
     81      1.1     joerg 		result = (*sym)();
     82      1.1     joerg 		ATF_CHECK(result == expected_result[i]);
     83      1.1     joerg 
     84      1.1     joerg 		dlclose(handle);
     85      1.1     joerg 		error = dlerror();
     86      1.1     joerg 		ATF_CHECK(error == NULL);
     87      1.1     joerg 
     88      1.1     joerg 		char *command;
     89  1.7.2.1  pgoyette 		easprintf(&command, "%s/h_ifunc %lld",
     90      1.1     joerg 		    atf_tc_get_config_var(tc, "srcdir"), expected_result[i]);
     91      1.1     joerg 		if (system(command) != EXIT_SUCCESS)
     92      1.1     joerg 			atf_tc_fail("Test failed; see output for details");
     93      1.1     joerg 		free(command);
     94      1.1     joerg 	}
     95      1.1     joerg }
     96      1.1     joerg 
     97      1.3     joerg ATF_TC(rtld_hidden_ifunc);
     98      1.3     joerg 
     99      1.3     joerg ATF_TC_HEAD(rtld_hidden_ifunc, tc)
    100      1.3     joerg {
    101      1.3     joerg 	atf_tc_set_md_var(tc, "descr", "hidden ifunc functions are resolved");
    102      1.3     joerg }
    103      1.3     joerg 
    104      1.3     joerg ATF_TC_BODY(rtld_hidden_ifunc, tc)
    105      1.3     joerg {
    106      1.3     joerg 	const char *envstr[] = {
    107      1.3     joerg 	    "0", "1"
    108      1.3     joerg 	};
    109  1.7.2.1  pgoyette 	long long expected_result[] = {
    110  1.7.2.1  pgoyette 	    0xdeadbeefll, 0xbeefdeadll
    111      1.3     joerg 	};
    112      1.3     joerg 	void *handle;
    113  1.7.2.1  pgoyette 	long long (*sym)(void);
    114  1.7.2.1  pgoyette 	long long (*(*sym2)(void))(void);
    115  1.7.2.1  pgoyette 	long long result;
    116      1.3     joerg 	const char *error;
    117      1.3     joerg 	size_t i;
    118      1.3     joerg 
    119      1.7      maya 	if (!LINKER_SUPPORT)
    120      1.7      maya 		atf_tc_skip("Missing linker support for ifunc relocations");
    121      1.6    martin 
    122      1.3     joerg 	for (i = 0; i < __arraycount(envstr); ++i) {
    123      1.3     joerg 		setenv("USE_IFUNC2", envstr[i], 1);
    124      1.3     joerg 
    125      1.3     joerg 		handle = dlopen("libh_helper_ifunc_dso.so", RTLD_LAZY);
    126      1.3     joerg 		error = dlerror();
    127      1.3     joerg 		ATF_CHECK(error == NULL);
    128      1.3     joerg 		ATF_CHECK(handle != NULL);
    129      1.3     joerg 
    130      1.3     joerg 		sym = dlsym(handle, "ifunc_plt");
    131      1.3     joerg 		error = dlerror();
    132      1.3     joerg 		ATF_CHECK(error == NULL);
    133      1.3     joerg 		ATF_CHECK(sym != NULL);
    134      1.3     joerg 
    135      1.3     joerg 		result = (*sym)();
    136      1.3     joerg 		ATF_CHECK(result == expected_result[!i]);
    137      1.3     joerg 
    138      1.4     joerg 		sym2 = dlsym(handle, "ifunc_indirect");
    139      1.4     joerg 		error = dlerror();
    140      1.4     joerg 		ATF_CHECK(error == NULL);
    141      1.4     joerg 		ATF_CHECK(sym2 != NULL);
    142      1.4     joerg 
    143      1.4     joerg 		sym = (*sym2)();
    144      1.4     joerg 		result = (*sym)();
    145      1.4     joerg 		ATF_CHECK(result == expected_result[!i]);
    146      1.4     joerg 
    147      1.3     joerg 		dlclose(handle);
    148      1.3     joerg 		error = dlerror();
    149      1.3     joerg 		ATF_CHECK(error == NULL);
    150      1.3     joerg 
    151      1.3     joerg 		char *command;
    152  1.7.2.1  pgoyette 		easprintf(&command, "%s/h_ifunc %lld",
    153      1.3     joerg 		    atf_tc_get_config_var(tc, "srcdir"), expected_result[i]);
    154      1.3     joerg 		if (system(command) != EXIT_SUCCESS)
    155      1.3     joerg 			atf_tc_fail("Test failed; see output for details");
    156      1.3     joerg 		free(command);
    157      1.3     joerg 	}
    158      1.3     joerg }
    159      1.3     joerg 
    160      1.5     joerg ATF_TC(rtld_main_ifunc);
    161      1.5     joerg ATF_TC_HEAD(rtld_main_ifunc, tc)
    162      1.5     joerg {
    163      1.5     joerg 	atf_tc_set_md_var(tc, "descr",
    164      1.5     joerg 	    "ifunc functions are resolved in the executable");
    165      1.5     joerg }
    166      1.5     joerg 
    167      1.7      maya #if LINKER_SUPPORT
    168  1.7.2.1  pgoyette static long long
    169      1.5     joerg ifunc_helper(void)
    170      1.5     joerg {
    171  1.7.2.1  pgoyette 	return 0xdeadbeefll;
    172      1.5     joerg }
    173      1.5     joerg 
    174      1.5     joerg static __attribute__((used))
    175  1.7.2.1  pgoyette long long (*resolve_ifunc(void))(void)
    176      1.5     joerg {
    177      1.5     joerg 	return ifunc_helper;
    178      1.5     joerg }
    179      1.5     joerg __hidden_ifunc(ifunc, resolve_ifunc);
    180      1.7      maya #endif
    181  1.7.2.1  pgoyette long long ifunc(void);
    182      1.5     joerg 
    183      1.5     joerg ATF_TC_BODY(rtld_main_ifunc, tc)
    184      1.5     joerg {
    185      1.7      maya 	if (!LINKER_SUPPORT)
    186      1.7      maya 		atf_tc_skip("Missing linker support for ifunc relocations");
    187  1.7.2.1  pgoyette 	ATF_CHECK(ifunc() == 0xdeadbeefll);
    188      1.5     joerg }
    189      1.5     joerg 
    190      1.1     joerg ATF_TP_ADD_TCS(tp)
    191      1.1     joerg {
    192      1.1     joerg 	ATF_TP_ADD_TC(tp, rtld_ifunc);
    193      1.3     joerg 	ATF_TP_ADD_TC(tp, rtld_hidden_ifunc);
    194      1.5     joerg 	ATF_TP_ADD_TC(tp, rtld_main_ifunc);
    195      1.1     joerg 	return 0;
    196      1.1     joerg }
    197