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