ed225df3ad
the initialization routine to exit by throwing an exception. Such an execution, termed exceptional, requires call_once to propagate the exception to its caller. A program may contain any number of exceptional executions but only one returning execution (which, if it exists, must be the last execution with the same once flag). On POSIX systems such as Linux, std::call_once is implemented in terms of pthread_once. However, as discussed in libstdc++ bug 66146 - "call_once not C++11-compliant on ppc64le," GLIBC's pthread_once hangs when the initialization function exits by throwing an exception on at least arm and ppc64 (though apparently not on x86_64). This effectively prevents call_once from conforming to the C++ requirements since there doesn't appear to be a thread-safe way to work around this problem in libstdc++. This patch changes pthread_once to handle gracefully init functions that exit by throwing exceptions. It was successfully tested on ppc64, ppc64le, and x86_64. [BZ #18435] * nptl/Makefile: Add tst-once5.cc. * nptl/pthreadP.h (pthread_cleanup_push, pthread_cleanup_pop): Remove macro redefinitions. * nptl/tst-once5.cc: New test.
81 lines
2.2 KiB
C++
81 lines
2.2 KiB
C++
/* Copyright (C) 2015 Free Software Foundation, Inc.
|
|
This file is part of the GNU C Library.
|
|
Contributed by Ulrich Drepper <drepper@redhat.com>, 2002.
|
|
|
|
The GNU C Library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Lesser General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 2.1 of the License, or (at your option) any later version.
|
|
|
|
The GNU C Library is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
License along with the GNU C Library; if not, see
|
|
<http://www.gnu.org/licenses/>. */
|
|
|
|
#include <errno.h>
|
|
#include <pthread.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
|
|
|
|
static pthread_once_t once = PTHREAD_ONCE_INIT;
|
|
|
|
// Exception type thrown from the pthread_once init routine.
|
|
struct OnceException { };
|
|
|
|
// Test iteration counter.
|
|
static int niter;
|
|
|
|
static void
|
|
init_routine (void)
|
|
{
|
|
if (niter < 2)
|
|
throw OnceException ();
|
|
}
|
|
|
|
// Verify that an exception thrown from the pthread_once init routine
|
|
// is propagated to the pthread_once caller and that the function can
|
|
// be subsequently invoked to attempt the initialization again.
|
|
static int
|
|
do_test (void)
|
|
{
|
|
int result = 1;
|
|
|
|
// Repeat three times, having the init routine throw the first two
|
|
// times and succeed on the final attempt.
|
|
for (niter = 0; niter != 3; ++niter) {
|
|
|
|
try {
|
|
int rc = pthread_once (&once, init_routine);
|
|
if (rc)
|
|
fprintf (stderr, "pthread_once failed: %i (%s)\n",
|
|
rc, strerror (rc));
|
|
|
|
if (niter < 2)
|
|
fputs ("pthread_once unexpectedly returned without"
|
|
" throwing an exception", stderr);
|
|
}
|
|
catch (OnceException) {
|
|
if (1 < niter)
|
|
fputs ("pthread_once unexpectedly threw", stderr);
|
|
result = 0;
|
|
}
|
|
catch (...) {
|
|
fputs ("pthread_once threw an unknown exception", stderr);
|
|
}
|
|
|
|
// Abort the test on the first failure.
|
|
if (result)
|
|
break;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
#define TEST_FUNCTION do_test ()
|
|
#include "../test-skeleton.c"
|