Blame resolv/tst-resolv-threads.c

Packit 6c4009
/* Test basic nss_dns functionality with multiple threads.
Packit 6c4009
   Copyright (C) 2016-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
Packit 6c4009
   The GNU C Library is free software; you can redistribute it and/or
Packit 6c4009
   modify it under the terms of the GNU Lesser General Public
Packit 6c4009
   License as published by the Free Software Foundation; either
Packit 6c4009
   version 2.1 of the License, or (at your option) any later version.
Packit 6c4009
Packit 6c4009
   The GNU C Library is distributed in the hope that it will be useful,
Packit 6c4009
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 6c4009
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 6c4009
   Lesser General Public License for more details.
Packit 6c4009
Packit 6c4009
   You should have received a copy of the GNU Lesser General Public
Packit 6c4009
   License along with the GNU C Library; if not, see
Packit 6c4009
   <http://www.gnu.org/licenses/>.  */
Packit 6c4009
Packit 6c4009
/* Unlike tst-resolv-basic, this test does not overwrite the _res
Packit 6c4009
   structure and relies on namespaces to achieve the redirection to
Packit 6c4009
   the test servers with a custom /etc/resolv.conf file.  */
Packit 6c4009
Packit 6c4009
#include <dlfcn.h>
Packit 6c4009
#include <errno.h>
Packit 6c4009
#include <gnu/lib-names.h>
Packit 6c4009
#include <netdb.h>
Packit 6c4009
#include <resolv/resolv-internal.h>
Packit 6c4009
#include <resolv/resolv_context.h>
Packit 6c4009
#include <stdio.h>
Packit 6c4009
#include <stdlib.h>
Packit 6c4009
#include <string.h>
Packit 6c4009
#include <support/check.h>
Packit 6c4009
#include <support/namespace.h>
Packit 6c4009
#include <support/resolv_test.h>
Packit 6c4009
#include <support/support.h>
Packit 6c4009
#include <support/temp_file.h>
Packit 6c4009
#include <support/test-driver.h>
Packit 6c4009
#include <support/xthread.h>
Packit 6c4009
#include <support/xunistd.h>
Packit 6c4009
Packit 6c4009
/* Each client thread sends this many queries.  */
Packit 6c4009
enum { queries_per_thread = 500 };
Packit 6c4009
Packit 6c4009
/* Return a small positive number identifying this thread.  */
Packit 6c4009
static int
Packit 6c4009
get_thread_number (void)
Packit 6c4009
{
Packit 6c4009
  static int __thread local;
Packit 6c4009
  if (local != 0)
Packit 6c4009
    return local;
Packit 6c4009
  static int global = 1;
Packit 6c4009
  local = __atomic_fetch_add (&global, 1, __ATOMIC_RELAXED);
Packit 6c4009
  return local;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
static void
Packit 6c4009
response (const struct resolv_response_context *ctx,
Packit 6c4009
          struct resolv_response_builder *b,
Packit 6c4009
          const char *qname, uint16_t qclass, uint16_t qtype)
Packit 6c4009
{
Packit 6c4009
  TEST_VERIFY_EXIT (qname != NULL);
Packit 6c4009
Packit 6c4009
  int counter = 0;
Packit 6c4009
  int thread = 0;
Packit 6c4009
  int dummy = 0;
Packit 6c4009
  TEST_VERIFY (sscanf (qname, "counter%d.thread%d.example.com%n",
Packit 6c4009
                       &counter, &thread, &dummy) == 2);
Packit 6c4009
  TEST_VERIFY (dummy > 0);
Packit 6c4009
Packit 6c4009
  struct resolv_response_flags flags = { 0 };
Packit 6c4009
  resolv_response_init (b, flags);
Packit 6c4009
  resolv_response_add_question (b, qname, qclass, qtype);
Packit 6c4009
Packit 6c4009
  resolv_response_section (b, ns_s_an);
Packit 6c4009
  resolv_response_open_record (b, qname, qclass, qtype, 0);
Packit 6c4009
  switch (qtype)
Packit 6c4009
    {
Packit 6c4009
    case T_A:
Packit 6c4009
      {
Packit 6c4009
        char ipv4[4] = {10, 0, counter, thread};
Packit 6c4009
        resolv_response_add_data (b, &ipv4, sizeof (ipv4));
Packit 6c4009
      }
Packit 6c4009
      break;
Packit 6c4009
    case T_AAAA:
Packit 6c4009
      {
Packit 6c4009
        char ipv6[16]
Packit 6c4009
          = {0x20, 0x01, 0xd, 0xb8, 0, 0, 0, 0, 0, 0, 0,
Packit 6c4009
             counter, 0, thread, 0, 0};
Packit 6c4009
        resolv_response_add_data (b, &ipv6, sizeof (ipv6));
Packit 6c4009
      }
Packit 6c4009
      break;
Packit 6c4009
    default:
Packit 6c4009
      support_record_failure ();
Packit 6c4009
      printf ("error: unexpected QTYPE: %s/%u/%u\n",
Packit 6c4009
              qname, qclass, qtype);
Packit 6c4009
    }
Packit 6c4009
  resolv_response_close_record (b);
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Check that the resolver configuration for this thread has an
Packit 6c4009
   extended resolver configuration.  */
Packit 6c4009
static void
Packit 6c4009
check_have_conf (void)
Packit 6c4009
{
Packit 6c4009
  struct resolv_context *ctx = __resolv_context_get ();
Packit 6c4009
  TEST_VERIFY_EXIT (ctx != NULL);
Packit 6c4009
  TEST_VERIFY (ctx->conf != NULL);
Packit 6c4009
  __resolv_context_put (ctx);
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Verify that E matches the expected response for FAMILY and
Packit 6c4009
   COUNTER.  */
Packit 6c4009
static void
Packit 6c4009
check_hostent (const char *caller, const char *function, const char *qname,
Packit 6c4009
               int ret, struct hostent *e, int family, int counter)
Packit 6c4009
{
Packit 6c4009
  if (ret != 0)
Packit 6c4009
    {
Packit 6c4009
      errno = ret;
Packit 6c4009
      support_record_failure ();
Packit 6c4009
      printf ("error: %s: %s for %s failed: %m\n", caller, function, qname);
Packit 6c4009
      return;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  TEST_VERIFY_EXIT (e != NULL);
Packit 6c4009
  TEST_VERIFY (strcmp (qname, e->h_name) == 0);
Packit 6c4009
  TEST_VERIFY (e->h_addrtype == family);
Packit 6c4009
  TEST_VERIFY_EXIT (e->h_addr_list[0] != NULL);
Packit 6c4009
  TEST_VERIFY (e->h_addr_list[1] == NULL);
Packit 6c4009
  switch (family)
Packit 6c4009
    {
Packit 6c4009
    case AF_INET:
Packit 6c4009
      {
Packit 6c4009
        char addr[4] = {10, 0, counter, get_thread_number ()};
Packit 6c4009
        TEST_VERIFY (e->h_length == sizeof (addr));
Packit 6c4009
        TEST_VERIFY (memcmp (e->h_addr_list[0], addr, sizeof (addr)) == 0);
Packit 6c4009
      }
Packit 6c4009
      break;
Packit 6c4009
    case AF_INET6:
Packit 6c4009
      {
Packit 6c4009
      char addr[16]
Packit 6c4009
        = {0x20, 0x01, 0xd, 0xb8, 0, 0, 0, 0, 0, 0,
Packit 6c4009
           0, counter, 0, get_thread_number (), 0, 0};
Packit 6c4009
      TEST_VERIFY (e->h_length == sizeof (addr));
Packit 6c4009
      TEST_VERIFY (memcmp (e->h_addr_list[0], addr, sizeof (addr)) == 0);
Packit 6c4009
      }
Packit 6c4009
      break;
Packit 6c4009
    default:
Packit 6c4009
      FAIL_EXIT1 ("%s: invalid address family %d", caller, family);
Packit 6c4009
    }
Packit 6c4009
  check_have_conf ();
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Check a getaddrinfo result.  */
Packit 6c4009
static void
Packit 6c4009
check_addrinfo (const char *caller, const char *qname,
Packit 6c4009
                int ret, struct addrinfo *ai, int family, int counter)
Packit 6c4009
{
Packit 6c4009
  if (ret != 0)
Packit 6c4009
    {
Packit 6c4009
      support_record_failure ();
Packit 6c4009
      printf ("error: %s: getaddrinfo for %s failed: %s\n",
Packit 6c4009
              caller, qname, gai_strerror (ret));
Packit 6c4009
      return;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  TEST_VERIFY_EXIT (ai != NULL);
Packit 6c4009
Packit 6c4009
  /* Check that available data matches the requirements.  */
Packit 6c4009
  bool have_ipv4 = false;
Packit 6c4009
  bool have_ipv6 = false;
Packit 6c4009
  for (struct addrinfo *p = ai; p != NULL; p = p->ai_next)
Packit 6c4009
    {
Packit 6c4009
      TEST_VERIFY (p->ai_socktype == SOCK_STREAM);
Packit 6c4009
      TEST_VERIFY (p->ai_protocol == IPPROTO_TCP);
Packit 6c4009
      TEST_VERIFY_EXIT (p->ai_addr != NULL);
Packit 6c4009
      TEST_VERIFY (p->ai_addr->sa_family == p->ai_family);
Packit 6c4009
Packit 6c4009
      switch (p->ai_family)
Packit 6c4009
        {
Packit 6c4009
        case AF_INET:
Packit 6c4009
          {
Packit 6c4009
            TEST_VERIFY (!have_ipv4);
Packit 6c4009
            have_ipv4 = true;
Packit 6c4009
            struct sockaddr_in *sa = (struct sockaddr_in *) p->ai_addr;
Packit 6c4009
            TEST_VERIFY (p->ai_addrlen == sizeof (*sa));
Packit 6c4009
            char addr[4] = {10, 0, counter, get_thread_number ()};
Packit 6c4009
            TEST_VERIFY (memcmp (&sa->sin_addr, addr, sizeof (addr)) == 0);
Packit 6c4009
            TEST_VERIFY (ntohs (sa->sin_port) == 80);
Packit 6c4009
          }
Packit 6c4009
          break;
Packit 6c4009
        case AF_INET6:
Packit 6c4009
          {
Packit 6c4009
            TEST_VERIFY (!have_ipv6);
Packit 6c4009
            have_ipv6 = true;
Packit 6c4009
            struct sockaddr_in6 *sa = (struct sockaddr_in6 *) p->ai_addr;
Packit 6c4009
            TEST_VERIFY (p->ai_addrlen == sizeof (*sa));
Packit 6c4009
            char addr[16]
Packit 6c4009
              = {0x20, 0x01, 0xd, 0xb8, 0, 0, 0, 0, 0, 0,
Packit 6c4009
                 0, counter, 0, get_thread_number (), 0, 0};
Packit 6c4009
            TEST_VERIFY (memcmp (&sa->sin6_addr, addr, sizeof (addr)) == 0);
Packit 6c4009
            TEST_VERIFY (ntohs (sa->sin6_port) == 80);
Packit 6c4009
          }
Packit 6c4009
          break;
Packit 6c4009
        default:
Packit 6c4009
          FAIL_EXIT1 ("%s: invalid address family %d", caller, family);
Packit 6c4009
        }
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  switch (family)
Packit 6c4009
    {
Packit 6c4009
      case AF_INET:
Packit 6c4009
        TEST_VERIFY (have_ipv4);
Packit 6c4009
        TEST_VERIFY (!have_ipv6);
Packit 6c4009
        break;
Packit 6c4009
      case AF_INET6:
Packit 6c4009
        TEST_VERIFY (!have_ipv4);
Packit 6c4009
        TEST_VERIFY (have_ipv6);
Packit 6c4009
        break;
Packit 6c4009
      case AF_UNSPEC:
Packit 6c4009
        TEST_VERIFY (have_ipv4);
Packit 6c4009
        TEST_VERIFY (have_ipv6);
Packit 6c4009
        break;
Packit 6c4009
    default:
Packit 6c4009
      FAIL_EXIT1 ("%s: invalid address family %d", caller, family);
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  check_have_conf ();
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* This barrier ensures that all test threads begin their work
Packit 6c4009
   simultaneously.  */
Packit 6c4009
static pthread_barrier_t barrier;
Packit 6c4009
Packit 6c4009
/* Test gethostbyname2_r (if do_2 is false) or gethostbyname2_r with
Packit 6c4009
   AF_INET (if do_2 is true).  */
Packit 6c4009
static void *
Packit 6c4009
byname (bool do_2)
Packit 6c4009
{
Packit 6c4009
  int this_thread = get_thread_number ();
Packit 6c4009
  xpthread_barrier_wait (&barrier);
Packit 6c4009
  for (int i = 0; i < queries_per_thread; ++i)
Packit 6c4009
    {
Packit 6c4009
      char qname[100];
Packit 6c4009
      snprintf (qname, sizeof (qname), "counter%d.thread%d.example.com",
Packit 6c4009
                i, this_thread);
Packit 6c4009
      struct hostent storage;
Packit 6c4009
      char buf[1000];
Packit 6c4009
      struct hostent *e = NULL;
Packit 6c4009
      int herrno;
Packit 6c4009
      int ret;
Packit 6c4009
      if (do_2)
Packit 6c4009
        ret = gethostbyname_r (qname, &storage, buf, sizeof (buf),
Packit 6c4009
                               &e, &herrno);
Packit 6c4009
      else
Packit 6c4009
        ret = gethostbyname2_r (qname, AF_INET, &storage, buf, sizeof (buf),
Packit 6c4009
                                &e, &herrno);
Packit 6c4009
      check_hostent (__func__, do_2 ? "gethostbyname2_r" : "gethostbyname_r",
Packit 6c4009
                     qname, ret, e, AF_INET, i);
Packit 6c4009
    }
Packit 6c4009
  check_have_conf ();
Packit 6c4009
  return NULL;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Test gethostbyname_r.  */
Packit 6c4009
static void *
Packit 6c4009
thread_byname (void *closure)
Packit 6c4009
{
Packit 6c4009
  return byname (false);
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Test gethostbyname2_r with AF_INET.  */
Packit 6c4009
static void *
Packit 6c4009
thread_byname2 (void *closure)
Packit 6c4009
{
Packit 6c4009
  return byname (true);
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Call gethostbyname_r with RES_USE_INET6 (if do_2 is false), or
Packit 6c4009
   gethostbyname_r with AF_INET6 (if do_2 is true).  */
Packit 6c4009
static void *
Packit 6c4009
byname_inet6 (bool do_2)
Packit 6c4009
{
Packit 6c4009
  int this_thread = get_thread_number ();
Packit 6c4009
  xpthread_barrier_wait (&barrier);
Packit 6c4009
  if (!do_2)
Packit 6c4009
    {
Packit 6c4009
      res_init ();
Packit 6c4009
      _res.options |= DEPRECATED_RES_USE_INET6;
Packit 6c4009
      TEST_VERIFY (strcmp (_res.defdname, "example.com") == 0);
Packit 6c4009
    }
Packit 6c4009
  for (int i = 0; i < queries_per_thread; ++i)
Packit 6c4009
    {
Packit 6c4009
      char qname[100];
Packit 6c4009
      snprintf (qname, sizeof (qname), "counter%d.thread%d.example.com",
Packit 6c4009
                i, this_thread);
Packit 6c4009
      struct hostent storage;
Packit 6c4009
      char buf[1000];
Packit 6c4009
      struct hostent *e = NULL;
Packit 6c4009
      int herrno;
Packit 6c4009
      int ret;
Packit 6c4009
      if (do_2)
Packit 6c4009
        ret = gethostbyname2_r (qname, AF_INET6, &storage, buf, sizeof (buf),
Packit 6c4009
                                &e, &herrno);
Packit 6c4009
      else
Packit 6c4009
        ret = gethostbyname_r (qname, &storage, buf, sizeof (buf),
Packit 6c4009
                               &e, &herrno);
Packit 6c4009
      check_hostent (__func__,
Packit 6c4009
                     do_2 ? "gethostbyname2_r" : "gethostbyname_r",
Packit 6c4009
                     qname, ret, e, AF_INET6, i);
Packit 6c4009
    }
Packit 6c4009
  return NULL;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Test gethostbyname_r with AF_INET6.  */
Packit 6c4009
static void *
Packit 6c4009
thread_byname_inet6 (void *closure)
Packit 6c4009
{
Packit 6c4009
  return byname_inet6 (false);
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Test gethostbyname2_r with AF_INET6.  */
Packit 6c4009
static void *
Packit 6c4009
thread_byname2_af_inet6 (void *closure)
Packit 6c4009
{
Packit 6c4009
  return byname_inet6 (true);
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Run getaddrinfo tests for FAMILY.  */
Packit 6c4009
static void *
Packit 6c4009
gai (int family, bool do_inet6)
Packit 6c4009
{
Packit 6c4009
  int this_thread = get_thread_number ();
Packit 6c4009
  xpthread_barrier_wait (&barrier);
Packit 6c4009
  if (do_inet6)
Packit 6c4009
    {
Packit 6c4009
      res_init ();
Packit 6c4009
      _res.options |= DEPRECATED_RES_USE_INET6;
Packit 6c4009
      check_have_conf ();
Packit 6c4009
    }
Packit 6c4009
  for (int i = 0; i < queries_per_thread; ++i)
Packit 6c4009
    {
Packit 6c4009
      char qname[100];
Packit 6c4009
      snprintf (qname, sizeof (qname), "counter%d.thread%d.example.com",
Packit 6c4009
                i, this_thread);
Packit 6c4009
      struct addrinfo hints =
Packit 6c4009
        {
Packit 6c4009
          .ai_family = family,
Packit 6c4009
          .ai_socktype = SOCK_STREAM,
Packit 6c4009
          .ai_protocol = IPPROTO_TCP,
Packit 6c4009
        };
Packit 6c4009
      struct addrinfo *ai;
Packit 6c4009
      int ret = getaddrinfo (qname, "80", &hints, &ai;;
Packit 6c4009
      check_addrinfo (__func__, qname, ret, ai, family, i);
Packit 6c4009
      if (ret == 0)
Packit 6c4009
        freeaddrinfo (ai);
Packit 6c4009
    }
Packit 6c4009
  return NULL;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Test getaddrinfo with AF_INET.  */
Packit 6c4009
static void *
Packit 6c4009
thread_gai_inet (void *closure)
Packit 6c4009
{
Packit 6c4009
  return gai (AF_INET, false);
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Test getaddrinfo with AF_INET6.  */
Packit 6c4009
static void *
Packit 6c4009
thread_gai_inet6 (void *closure)
Packit 6c4009
{
Packit 6c4009
  return gai (AF_INET6, false);
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Test getaddrinfo with AF_UNSPEC.  */
Packit 6c4009
static void *
Packit 6c4009
thread_gai_unspec (void *closure)
Packit 6c4009
{
Packit 6c4009
  return gai (AF_UNSPEC, false);
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Test getaddrinfo with AF_INET.  */
Packit 6c4009
static void *
Packit 6c4009
thread_gai_inet_inet6 (void *closure)
Packit 6c4009
{
Packit 6c4009
  return gai (AF_INET, true);
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Test getaddrinfo with AF_INET6.  */
Packit 6c4009
static void *
Packit 6c4009
thread_gai_inet6_inet6 (void *closure)
Packit 6c4009
{
Packit 6c4009
  return gai (AF_INET6, true);
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Test getaddrinfo with AF_UNSPEC.  */
Packit 6c4009
static void *
Packit 6c4009
thread_gai_unspec_inet6 (void *closure)
Packit 6c4009
{
Packit 6c4009
  return gai (AF_UNSPEC, true);
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Description of the chroot environment used to run the tests.  */
Packit 6c4009
static struct support_chroot *chroot_env;
Packit 6c4009
Packit 6c4009
/* Set up the chroot environment.  */
Packit 6c4009
static void
Packit 6c4009
prepare (int argc, char **argv)
Packit 6c4009
{
Packit 6c4009
  chroot_env = support_chroot_create
Packit 6c4009
    ((struct support_chroot_configuration)
Packit 6c4009
     {
Packit 6c4009
       .resolv_conf =
Packit 6c4009
         "search example.com\n"
Packit 6c4009
         "nameserver 127.0.0.1\n"
Packit 6c4009
         "nameserver 127.0.0.2\n"
Packit 6c4009
         "nameserver 127.0.0.3\n",
Packit 6c4009
     });
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
static int
Packit 6c4009
do_test (void)
Packit 6c4009
{
Packit 6c4009
  support_become_root ();
Packit 6c4009
  if (!support_enter_network_namespace ())
Packit 6c4009
    return EXIT_UNSUPPORTED;
Packit 6c4009
  if (!support_can_chroot ())
Packit 6c4009
    return EXIT_UNSUPPORTED;
Packit 6c4009
Packit 6c4009
  /* Load the shared object outside of the chroot.  */
Packit 6c4009
  TEST_VERIFY (dlopen (LIBNSS_DNS_SO, RTLD_LAZY) != NULL);
Packit 6c4009
Packit 6c4009
  xchroot (chroot_env->path_chroot);
Packit 6c4009
  TEST_VERIFY_EXIT (chdir ("/") == 0);
Packit 6c4009
Packit 6c4009
  struct sockaddr_in server_address =
Packit 6c4009
    {
Packit 6c4009
      .sin_family = AF_INET,
Packit 6c4009
      .sin_addr = { .s_addr = htonl (INADDR_LOOPBACK) },
Packit 6c4009
      .sin_port = htons (53)
Packit 6c4009
    };
Packit 6c4009
  const struct sockaddr *server_addresses[1] =
Packit 6c4009
    { (const struct sockaddr *) &server_address };
Packit 6c4009
Packit 6c4009
  struct resolv_test *aux = resolv_test_start
Packit 6c4009
    ((struct resolv_redirect_config)
Packit 6c4009
     {
Packit 6c4009
       .response_callback = response,
Packit 6c4009
       .nscount = 1,
Packit 6c4009
       .disable_redirect = true,
Packit 6c4009
       .server_address_overrides = server_addresses,
Packit 6c4009
     });
Packit 6c4009
Packit 6c4009
  enum { thread_count = 10 };
Packit 6c4009
  xpthread_barrier_init (&barrier, NULL, thread_count + 1);
Packit 6c4009
  pthread_t threads[thread_count];
Packit 6c4009
  typedef void *(*thread_func) (void *);
Packit 6c4009
  thread_func thread_funcs[thread_count] =
Packit 6c4009
    {
Packit 6c4009
      thread_byname,
Packit 6c4009
      thread_byname2,
Packit 6c4009
      thread_byname_inet6,
Packit 6c4009
      thread_byname2_af_inet6,
Packit 6c4009
      thread_gai_inet,
Packit 6c4009
      thread_gai_inet6,
Packit 6c4009
      thread_gai_unspec,
Packit 6c4009
      thread_gai_inet_inet6,
Packit 6c4009
      thread_gai_inet6_inet6,
Packit 6c4009
      thread_gai_unspec_inet6,
Packit 6c4009
    };
Packit 6c4009
  for (int i = 0; i < thread_count; ++i)
Packit 6c4009
    threads[i] = xpthread_create (NULL, thread_funcs[i], NULL);
Packit 6c4009
  xpthread_barrier_wait (&barrier); /* Start the test threads.  */
Packit 6c4009
  for (int i = 0; i < thread_count; ++i)
Packit 6c4009
    xpthread_join (threads[i]);
Packit 6c4009
Packit 6c4009
  resolv_test_end (aux);
Packit 6c4009
  support_chroot_free (chroot_env);
Packit 6c4009
Packit 6c4009
  return 0;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
#define PREPARE prepare
Packit 6c4009
#include <support/test-driver.c>