Blame misc/tst-allocate_once.c

Packit Service 82fcde
/* Test the allocate_once function.
Packit Service 82fcde
   Copyright (C) 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
#include <allocate_once.h>
Packit Service 82fcde
#include <mcheck.h>
Packit Service 82fcde
#include <string.h>
Packit Service 82fcde
#include <support/check.h>
Packit Service 82fcde
#include <support/support.h>
Packit Service 82fcde
Packit Service 82fcde
/* Allocate a new string.  */
Packit Service 82fcde
static void *
Packit Service 82fcde
allocate_string (void *closure)
Packit Service 82fcde
{
Packit Service 82fcde
  return xstrdup (closure);
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
/* Allocation and deallocation functions which are not expected to be
Packit Service 82fcde
   called.  */
Packit Service 82fcde
Packit Service 82fcde
static void *
Packit Service 82fcde
allocate_not_called (void *closure)
Packit Service 82fcde
{
Packit Service 82fcde
  FAIL_EXIT1 ("allocation function called unexpectedly (%p)", closure);
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
static void
Packit Service 82fcde
deallocate_not_called (void *closure, void *ptr)
Packit Service 82fcde
{
Packit Service 82fcde
  FAIL_EXIT1 ("deallocate function called unexpectedly (%p, %p)",
Packit Service 82fcde
              closure, ptr);
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
/* Counter for various function calls.  */
Packit Service 82fcde
static int function_called;
Packit Service 82fcde
Packit Service 82fcde
/* An allocation function which returns NULL and records that it has
Packit Service 82fcde
   been called.  */
Packit Service 82fcde
static void *
Packit Service 82fcde
allocate_return_null (void *closure)
Packit Service 82fcde
{
Packit Service 82fcde
  /* The function should only be called once.  */
Packit Service 82fcde
  TEST_COMPARE (function_called, 0);
Packit Service 82fcde
  ++function_called;
Packit Service 82fcde
  return NULL;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
/* The following is used to check the retry logic, by causing a fake
Packit Service 82fcde
   race condition.  */
Packit Service 82fcde
static void *fake_race_place;
Packit Service 82fcde
static char fake_race_region[3]; /* To obtain unique addresses.  */
Packit Service 82fcde
Packit Service 82fcde
static void *
Packit Service 82fcde
fake_race_allocate (void *closure)
Packit Service 82fcde
{
Packit Service 82fcde
  TEST_VERIFY (closure == &fake_race_region[0]);
Packit Service 82fcde
  TEST_COMPARE (function_called, 0);
Packit Service 82fcde
  ++function_called;
Packit Service 82fcde
  /* Fake allocation by another thread.  */
Packit Service 82fcde
  fake_race_place = &fake_race_region[1];
Packit Service 82fcde
  return &fake_race_region[2];
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
static void
Packit Service 82fcde
fake_race_deallocate (void *closure, void *ptr)
Packit Service 82fcde
{
Packit Service 82fcde
  /* Check that the pointer returned from fake_race_allocate is
Packit Service 82fcde
     deallocated (and not the one stored in fake_race_place).  */
Packit Service 82fcde
  TEST_VERIFY (ptr == &fake_race_region[2]);
Packit Service 82fcde
Packit Service 82fcde
  TEST_VERIFY (fake_race_place == &fake_race_region[1]);
Packit Service 82fcde
  TEST_VERIFY (closure == &fake_race_region[0]);
Packit Service 82fcde
  TEST_COMPARE (function_called, 1);
Packit Service 82fcde
  ++function_called;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
/* Similar to fake_race_allocate, but expects to be paired with free
Packit Service 82fcde
   as the deallocation function.  */
Packit Service 82fcde
static void *
Packit Service 82fcde
fake_race_allocate_for_free (void *closure)
Packit Service 82fcde
{
Packit Service 82fcde
  TEST_VERIFY (closure == &fake_race_region[0]);
Packit Service 82fcde
  TEST_COMPARE (function_called, 0);
Packit Service 82fcde
  ++function_called;
Packit Service 82fcde
  /* Fake allocation by another thread.  */
Packit Service 82fcde
  fake_race_place = &fake_race_region[1];
Packit Service 82fcde
  return xstrdup ("to be freed");
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
static int
Packit Service 82fcde
do_test (void)
Packit Service 82fcde
{
Packit Service 82fcde
  mtrace ();
Packit Service 82fcde
Packit Service 82fcde
  /* Simple allocation.  */
Packit Service 82fcde
  void *place1 = NULL;
Packit Service 82fcde
  char *string1 = allocate_once (&place1, allocate_string,
Packit Service 82fcde
                                   deallocate_not_called,
Packit Service 82fcde
                                   (char *) "test string 1");
Packit Service 82fcde
  TEST_VERIFY_EXIT (string1 != NULL);
Packit Service 82fcde
  TEST_VERIFY (strcmp ("test string 1", string1) == 0);
Packit Service 82fcde
  /* Second call returns the first pointer, without calling any
Packit Service 82fcde
     callbacks.  */
Packit Service 82fcde
  TEST_VERIFY (string1
Packit Service 82fcde
               == allocate_once (&place1, allocate_not_called,
Packit Service 82fcde
                                 deallocate_not_called,
Packit Service 82fcde
                                 (char *) "test string 1a"));
Packit Service 82fcde
Packit Service 82fcde
  /* Different place should result in another call.  */
Packit Service 82fcde
  void *place2 = NULL;
Packit Service 82fcde
  char *string2 = allocate_once (&place2, allocate_string,
Packit Service 82fcde
                                 deallocate_not_called,
Packit Service 82fcde
                                 (char *) "test string 2");
Packit Service 82fcde
  TEST_VERIFY_EXIT (string2 != NULL);
Packit Service 82fcde
  TEST_VERIFY (strcmp ("test string 2", string2) == 0);
Packit Service 82fcde
  TEST_VERIFY (string1 != string2);
Packit Service 82fcde
Packit Service 82fcde
  /* Check error reporting (NULL return value from the allocation
Packit Service 82fcde
     function).  */
Packit Service 82fcde
  void *place3 = NULL;
Packit Service 82fcde
  char *string3 = allocate_once (&place3, allocate_return_null,
Packit Service 82fcde
                                 deallocate_not_called, NULL);
Packit Service 82fcde
  TEST_VERIFY (string3 == NULL);
Packit Service 82fcde
  TEST_COMPARE (function_called, 1);
Packit Service 82fcde
Packit Service 82fcde
  /* Check that the deallocation function is called if the race is
Packit Service 82fcde
     lost.  */
Packit Service 82fcde
  function_called = 0;
Packit Service 82fcde
  TEST_VERIFY (allocate_once (&fake_race_place,
Packit Service 82fcde
                              fake_race_allocate,
Packit Service 82fcde
                              fake_race_deallocate,
Packit Service 82fcde
                              &fake_race_region[0])
Packit Service 82fcde
               == &fake_race_region[1]);
Packit Service 82fcde
  TEST_COMPARE (function_called, 2);
Packit Service 82fcde
  function_called = 3;
Packit Service 82fcde
  TEST_VERIFY (allocate_once (&fake_race_place,
Packit Service 82fcde
                              fake_race_allocate,
Packit Service 82fcde
                              fake_race_deallocate,
Packit Service 82fcde
                              &fake_race_region[0])
Packit Service 82fcde
               == &fake_race_region[1]);
Packit Service 82fcde
  TEST_COMPARE (function_called, 3);
Packit Service 82fcde
Packit Service 82fcde
  /* Similar, but this time rely on that free is called.  */
Packit Service 82fcde
  function_called = 0;
Packit Service 82fcde
  fake_race_place = NULL;
Packit Service 82fcde
  TEST_VERIFY (allocate_once (&fake_race_place,
Packit Service 82fcde
                                fake_race_allocate_for_free,
Packit Service 82fcde
                                NULL,
Packit Service 82fcde
                                &fake_race_region[0])
Packit Service 82fcde
               == &fake_race_region[1]);
Packit Service 82fcde
  TEST_COMPARE (function_called, 1);
Packit Service 82fcde
  function_called = 3;
Packit Service 82fcde
  TEST_VERIFY (allocate_once (&fake_race_place,
Packit Service 82fcde
                              fake_race_allocate_for_free,
Packit Service 82fcde
                              NULL,
Packit Service 82fcde
                              &fake_race_region[0])
Packit Service 82fcde
               == &fake_race_region[1]);
Packit Service 82fcde
  TEST_COMPARE (function_called, 3);
Packit Service 82fcde
Packit Service 82fcde
  free (place2);
Packit Service 82fcde
  free (place1);
Packit Service 82fcde
Packit Service 82fcde
  return 0;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
#include <support/test-driver.c>