hjl / source-git / glibc

Forked from source-git/glibc 3 years ago
Clone

Blame elf/ifuncmain3.c

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