Blame elf/tst-tls-manydynamic.c

Packit Service 82fcde
/* Test with many dynamic TLS variables.
Packit Service 82fcde
   Copyright (C) 2016-2018 Free Software Foundation, Inc.
Packit Service 82fcde
   This file is part of the GNU C Library.
Packit Service 82fcde
Packit Service 82fcde
   The GNU C Library is free software; you can redistribute it and/or
Packit Service 82fcde
   modify it under the terms of the GNU Lesser General Public
Packit Service 82fcde
   License as published by the Free Software Foundation; either
Packit Service 82fcde
   version 2.1 of the License, or (at your option) any later version.
Packit Service 82fcde
Packit Service 82fcde
   The GNU C Library is distributed in the hope that it will be useful,
Packit Service 82fcde
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 82fcde
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service 82fcde
   Lesser General Public License for more details.
Packit Service 82fcde
Packit Service 82fcde
   You should have received a copy of the GNU Lesser General Public
Packit Service 82fcde
   License along with the GNU C Library; if not, see
Packit Service 82fcde
   <http://www.gnu.org/licenses/>.  */
Packit Service 82fcde
Packit Service 82fcde
/* This test intends to exercise dynamic TLS variable allocation.  It
Packit Service 82fcde
   achieves this by combining dlopen (to avoid static TLS allocation
Packit Service 82fcde
   after static TLS resizing), many DSOs with a large variable (to
Packit Service 82fcde
   exceed the static TLS reserve), and an already-running thread (to
Packit Service 82fcde
   force full dynamic TLS initialization).  */
Packit Service 82fcde
Packit Service 82fcde
#include "tst-tls-manydynamic.h"
Packit Service 82fcde
Packit Service 82fcde
#include <errno.h>
Packit Service 82fcde
#include <dlfcn.h>
Packit Service 82fcde
#include <pthread.h>
Packit Service 82fcde
#include <stdio.h>
Packit Service 82fcde
#include <stdlib.h>
Packit Service 82fcde
#include <string.h>
Packit Service 82fcde
Packit Service 82fcde
static int do_test (void);
Packit Service 82fcde
#include <support/xthread.h>
Packit Service 82fcde
#include <support/test-driver.c>
Packit Service 82fcde
Packit Service 82fcde
void *handles[COUNT];
Packit Service 82fcde
set_value_func set_value_funcs[COUNT];
Packit Service 82fcde
get_value_func get_value_funcs[COUNT];
Packit Service 82fcde
Packit Service 82fcde
static void
Packit Service 82fcde
init_functions (void)
Packit Service 82fcde
{
Packit Service 82fcde
  for (int i = 0; i < COUNT; ++i)
Packit Service 82fcde
    {
Packit Service 82fcde
      /* Open the module.  */
Packit Service 82fcde
      {
Packit Service 82fcde
        char soname[100];
Packit Service 82fcde
        snprintf (soname, sizeof (soname), "tst-tls-manydynamic%02dmod.so", i);
Packit Service 82fcde
        handles[i] = dlopen (soname, RTLD_LAZY);
Packit Service 82fcde
        if (handles[i] == NULL)
Packit Service 82fcde
          {
Packit Service 82fcde
            printf ("error: dlopen failed: %s\n", dlerror ());
Packit Service 82fcde
            exit (1);
Packit Service 82fcde
          }
Packit Service 82fcde
      }
Packit Service 82fcde
Packit Service 82fcde
      /* Obtain the setter function.  */
Packit Service 82fcde
      {
Packit Service 82fcde
        char fname[100];
Packit Service 82fcde
        snprintf (fname, sizeof (fname), "set_value_%02d", i);
Packit Service 82fcde
        void *func = dlsym (handles[i], fname);
Packit Service 82fcde
        if (func == NULL)
Packit Service 82fcde
          {
Packit Service 82fcde
            printf ("error: dlsym: %s\n", dlerror ());
Packit Service 82fcde
            exit (1);
Packit Service 82fcde
          }
Packit Service 82fcde
        set_value_funcs[i] = func;
Packit Service 82fcde
      }
Packit Service 82fcde
Packit Service 82fcde
      /* Obtain the getter function.  */
Packit Service 82fcde
      {
Packit Service 82fcde
        char fname[100];
Packit Service 82fcde
        snprintf (fname, sizeof (fname), "get_value_%02d", i);
Packit Service 82fcde
        void *func = dlsym (handles[i], fname);
Packit Service 82fcde
        if (func == NULL)
Packit Service 82fcde
          {
Packit Service 82fcde
            printf ("error: dlsym: %s\n", dlerror ());
Packit Service 82fcde
            exit (1);
Packit Service 82fcde
          }
Packit Service 82fcde
        get_value_funcs[i] = func;
Packit Service 82fcde
      }
Packit Service 82fcde
    }
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
static pthread_barrier_t barrier;
Packit Service 82fcde
Packit Service 82fcde
/* Running thread which forces real TLS initialization.  */
Packit Service 82fcde
static void *
Packit Service 82fcde
blocked_thread_func (void *closure)
Packit Service 82fcde
{
Packit Service 82fcde
  xpthread_barrier_wait (&barrier);
Packit Service 82fcde
Packit Service 82fcde
  /* TLS test runs here in the main thread.  */
Packit Service 82fcde
Packit Service 82fcde
  xpthread_barrier_wait (&barrier);
Packit Service 82fcde
  return NULL;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
static int
Packit Service 82fcde
do_test (void)
Packit Service 82fcde
{
Packit Service 82fcde
  {
Packit Service 82fcde
    int ret = pthread_barrier_init (&barrier, NULL, 2);
Packit Service 82fcde
    if (ret != 0)
Packit Service 82fcde
      {
Packit Service 82fcde
        errno = ret;
Packit Service 82fcde
        printf ("error: pthread_barrier_init: %m\n");
Packit Service 82fcde
        exit (1);
Packit Service 82fcde
      }
Packit Service 82fcde
  }
Packit Service 82fcde
Packit Service 82fcde
  pthread_t blocked_thread = xpthread_create (NULL, blocked_thread_func, NULL);
Packit Service 82fcde
  xpthread_barrier_wait (&barrier);
Packit Service 82fcde
Packit Service 82fcde
  init_functions ();
Packit Service 82fcde
Packit Service 82fcde
  struct value values[COUNT];
Packit Service 82fcde
  /* Initialze the TLS variables.  */
Packit Service 82fcde
  for (int i = 0; i < COUNT; ++i)
Packit Service 82fcde
    {
Packit Service 82fcde
      for (int j = 0; j < PER_VALUE_COUNT; ++j)
Packit Service 82fcde
        values[i].num[j] = rand ();
Packit Service 82fcde
      set_value_funcs[i] (&values[i]);
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  /* Read back their values to check that they do not overlap.  */
Packit Service 82fcde
  for (int i = 0; i < COUNT; ++i)
Packit Service 82fcde
    {
Packit Service 82fcde
      struct value actual;
Packit Service 82fcde
      get_value_funcs[i] (&actual);
Packit Service 82fcde
Packit Service 82fcde
      for (int j = 0; j < PER_VALUE_COUNT; ++j)
Packit Service 82fcde
        if (actual.num[j] != values[i].num[j])
Packit Service 82fcde
        {
Packit Service 82fcde
          printf ("error: mismatch at variable %d/%d: %d != %d\n",
Packit Service 82fcde
                  i, j, actual.num[j], values[i].num[j]);
Packit Service 82fcde
          exit (1);
Packit Service 82fcde
        }
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  xpthread_barrier_wait (&barrier);
Packit Service 82fcde
  xpthread_join (blocked_thread);
Packit Service 82fcde
Packit Service 82fcde
  /* Close the modules.  */
Packit Service 82fcde
  for (int i = 0; i < COUNT; ++i)
Packit Service 82fcde
    dlclose (handles[i]);
Packit Service 82fcde
Packit Service 82fcde
  return 0;
Packit Service 82fcde
}