|  | /* Test STT_GNU_IFUNC symbols with dlopen: | 
|  |  | 
|  | 1. Direct function call. | 
|  | 2. Function pointer. | 
|  | 3. Visibility with override. | 
|  | */ | 
|  |  | 
|  | #include <dlfcn.h> | 
|  | #include <stdlib.h> | 
|  | #include <stdio.h> | 
|  |  | 
|  | extern int __attribute__ ((noinline)) foo (void); | 
|  | extern int __attribute__ ((noinline)) foo_hidden (void); | 
|  | extern int __attribute__ ((noinline)) foo_protected (void); | 
|  |  | 
|  | typedef int (*foo_p) (void); | 
|  |  | 
|  | int | 
|  | __attribute__ ((noinline)) | 
|  | foo (void) | 
|  | { | 
|  | return -30; | 
|  | } | 
|  |  | 
|  | int | 
|  | __attribute__ ((noinline)) | 
|  | foo_hidden (void) | 
|  | { | 
|  | return -20; | 
|  | } | 
|  |  | 
|  | int | 
|  | __attribute__ ((noinline)) | 
|  | foo_protected (void) | 
|  | { | 
|  | return -40; | 
|  | } | 
|  |  | 
|  | int | 
|  | main (void) | 
|  | { | 
|  | foo_p p; | 
|  | foo_p (*f) (void); | 
|  | int *ret; | 
|  |  | 
|  | void *h = dlopen ("ifuncmod3.so", RTLD_LAZY); | 
|  | if (h == NULL) | 
|  | { | 
|  | printf ("cannot load: %s\n", dlerror ()); | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | p = dlsym (h, "foo"); | 
|  | if (p == NULL) | 
|  | { | 
|  | printf ("symbol not found: %s\n", dlerror ()); | 
|  | return 1; | 
|  | } | 
|  | if ((*p) () != -1) | 
|  | abort (); | 
|  |  | 
|  | f = dlsym (h, "get_foo_p"); | 
|  | if (f == NULL) | 
|  | { | 
|  | printf ("symbol not found: %s\n", dlerror ()); | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | ret = dlsym (h, "ret_foo"); | 
|  | if (ret == NULL) | 
|  | { | 
|  | printf ("symbol not found: %s\n", dlerror ()); | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | p = (*f) (); | 
|  | if (p != foo) | 
|  | abort (); | 
|  | if (foo () != -30) | 
|  | abort (); | 
|  | if (*ret != -30 || (*p) () != *ret) | 
|  | abort (); | 
|  |  | 
|  | f = dlsym (h, "get_foo_hidden_p"); | 
|  | if (f == NULL) | 
|  | { | 
|  | printf ("symbol not found: %s\n", dlerror ()); | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | ret = dlsym (h, "ret_foo_hidden"); | 
|  | if (ret == NULL) | 
|  | { | 
|  | printf ("symbol not found: %s\n", dlerror ()); | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | p = (*f) (); | 
|  | if (foo_hidden () != -20) | 
|  | abort (); | 
|  | if (*ret != 1 || (*p) () != *ret) | 
|  | abort (); | 
|  |  | 
|  | f = dlsym (h, "get_foo_protected_p"); | 
|  | if (f == NULL) | 
|  | { | 
|  | printf ("symbol not found: %s\n", dlerror ()); | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | ret = dlsym (h, "ret_foo_protected"); | 
|  | if (ret == NULL) | 
|  | { | 
|  | printf ("symbol not found: %s\n", dlerror ()); | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | p = (*f) (); | 
|  | if (p == foo_protected) | 
|  | abort (); | 
|  | if (foo_protected () != -40) | 
|  | abort (); | 
|  | if (*ret != 0 || (*p) () != *ret) | 
|  | abort (); | 
|  |  | 
|  | if (dlclose (h) != 0) | 
|  | { | 
|  | printf ("cannot close: %s\n", dlerror ()); | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } |