1 /* Test STT_GNU_IFUNC symbols with dlopen:
2
3 1. Direct function call.
4 2. Function pointer.
5 3. Visibility with override.
6 */
7
8 #include <dlfcn.h>
9 #include <stdlib.h>
10 #include <stdio.h>
11
12 extern int __attribute__ ((noinline)) foo (void);
13 extern int __attribute__ ((noinline)) foo_hidden (void);
14 extern int __attribute__ ((noinline)) foo_protected (void);
15
16 typedef int (*foo_p) (void);
17
18 int
19 __attribute__ ((noinline))
foo(void)20 foo (void)
21 {
22 return -30;
23 }
24
25 int
26 __attribute__ ((noinline))
foo_hidden(void)27 foo_hidden (void)
28 {
29 return -20;
30 }
31
32 int
33 __attribute__ ((noinline))
foo_protected(void)34 foo_protected (void)
35 {
36 return -40;
37 }
38
39 int
main(void)40 main (void)
41 {
42 foo_p p;
43 foo_p (*f) (void);
44 int *ret;
45
46 void *h = dlopen ("ifuncmod3.so", RTLD_LAZY);
47 if (h == NULL)
48 {
49 printf ("cannot load: %s\n", dlerror ());
50 return 1;
51 }
52
53 p = dlsym (h, "foo");
54 if (p == NULL)
55 {
56 printf ("symbol not found: %s\n", dlerror ());
57 return 1;
58 }
59 if ((*p) () != -1)
60 abort ();
61
62 f = dlsym (h, "get_foo_p");
63 if (f == NULL)
64 {
65 printf ("symbol not found: %s\n", dlerror ());
66 return 1;
67 }
68
69 ret = dlsym (h, "ret_foo");
70 if (ret == NULL)
71 {
72 printf ("symbol not found: %s\n", dlerror ());
73 return 1;
74 }
75
76 p = (*f) ();
77 if (p != foo)
78 abort ();
79 if (foo () != -30)
80 abort ();
81 if (*ret != -30 || (*p) () != *ret)
82 abort ();
83
84 f = dlsym (h, "get_foo_hidden_p");
85 if (f == NULL)
86 {
87 printf ("symbol not found: %s\n", dlerror ());
88 return 1;
89 }
90
91 ret = dlsym (h, "ret_foo_hidden");
92 if (ret == NULL)
93 {
94 printf ("symbol not found: %s\n", dlerror ());
95 return 1;
96 }
97
98 p = (*f) ();
99 if (foo_hidden () != -20)
100 abort ();
101 if (*ret != 1 || (*p) () != *ret)
102 abort ();
103
104 f = dlsym (h, "get_foo_protected_p");
105 if (f == NULL)
106 {
107 printf ("symbol not found: %s\n", dlerror ());
108 return 1;
109 }
110
111 ret = dlsym (h, "ret_foo_protected");
112 if (ret == NULL)
113 {
114 printf ("symbol not found: %s\n", dlerror ());
115 return 1;
116 }
117
118 p = (*f) ();
119 if (p == foo_protected)
120 abort ();
121 if (foo_protected () != -40)
122 abort ();
123 if (*ret != 0 || (*p) () != *ret)
124 abort ();
125
126 if (dlclose (h) != 0)
127 {
128 printf ("cannot close: %s\n", dlerror ());
129 return 1;
130 }
131
132 return 0;
133 }
134