t_ifunc.c revision 1.3 1 /* $NetBSD: t_ifunc.c,v 1.3 2017/08/10 19:03:27 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__)
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 result;
110 const char *error;
111 size_t i;
112
113 for (i = 0; i < __arraycount(envstr); ++i) {
114 setenv("USE_IFUNC2", envstr[i], 1);
115
116 handle = dlopen("libh_helper_ifunc_dso.so", RTLD_LAZY);
117 error = dlerror();
118 ATF_CHECK(error == NULL);
119 ATF_CHECK(handle != NULL);
120
121 sym = dlsym(handle, "ifunc_plt");
122 error = dlerror();
123 ATF_CHECK(error == NULL);
124 ATF_CHECK(sym != NULL);
125
126 result = (*sym)();
127 ATF_CHECK(result == expected_result[!i]);
128
129 dlclose(handle);
130 error = dlerror();
131 ATF_CHECK(error == NULL);
132
133 char *command;
134 easprintf(&command, "%s/h_ifunc %d",
135 atf_tc_get_config_var(tc, "srcdir"), expected_result[i]);
136 if (system(command) != EXIT_SUCCESS)
137 atf_tc_fail("Test failed; see output for details");
138 free(command);
139 }
140 }
141
142 ATF_TP_ADD_TCS(tp)
143 {
144 ATF_TP_ADD_TC(tp, rtld_ifunc);
145 ATF_TP_ADD_TC(tp, rtld_hidden_ifunc);
146 return 0;
147 }
148