2016-12-23 13:39:23 -05:00
|
|
|
/* Test cancellation of getpwuid_r.
|
2017-01-01 00:14:16 +00:00
|
|
|
Copyright (C) 2016-2017 Free Software Foundation, Inc.
|
2016-12-23 13:39:23 -05:00
|
|
|
This file is part of the GNU C Library.
|
|
|
|
|
|
|
|
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/>. */
|
|
|
|
|
|
|
|
/* Test if cancellation of getpwuid_r incorrectly leaves internal
|
|
|
|
function state locked resulting in hang of subsequent calls to
|
|
|
|
getpwuid_r. The main thread creates a second thread which will do
|
|
|
|
the calls to getpwuid_r. A semaphore is used by the second thread to
|
|
|
|
signal to the main thread that it is as close as it can be to the
|
|
|
|
call site of getpwuid_r. The goal of the semaphore is to avoid any
|
|
|
|
cancellable function calls between the sem_post and the call to
|
|
|
|
getpwuid_r. The main thread then attempts to cancel the second
|
|
|
|
thread. Without the fixes the cancellation happens at any number of
|
|
|
|
calls to cancellable functions in getpuid_r, but with the fix the
|
|
|
|
cancellation either does not happen or happens only at expected
|
|
|
|
points where the internal state is consistent. We use an explicit
|
|
|
|
pthread_testcancel call to terminate the loop in a timely fashion
|
|
|
|
if the implementation does not have a cancellation point. */
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <pthread.h>
|
|
|
|
#include <pwd.h>
|
Miscellaneous low-risk changes preparing for _ISOMAC testsuite.
These are a grab bag of changes where the testsuite was using internal
symbols of some variety, but this was straightforward to fix, and the
fixed code should work with or without the change to compile the
testsuite under _ISOMAC.
Four of these are just more #include adjustments, but I want to highlight
sysdeps/powerpc/fpu/tst-setcontext-fpscr.c, which appears to have been
written before the advent of sys/auxv.h. I think a big chunk of this file
could be replaced by a simple call to getauxval, but I'll let someone who
actually has a powerpc machine to test on do that.
dlfcn/tst-dladdr.c was including ldsodefs.h just so it could use
DL_LOOKUP_ADDRESS to print an additional diagnostic; as requested by Carlos,
I have removed this.
math/test-misc.c was using #ifndef NO_LONG_DOUBLE, which is an internal
configuration macro, to decide whether to do certain tests involving
'long double'. I changed the test to #if LDBL_MANT_DIG > DBL_MANT_DIG
instead, which uses only public float.h macros and is equivalent on
all supported platforms. (Note that NO_LONG_DOUBLE doesn't mean 'the
compiler doesn't support long double', it means 'long double is the
same as double'.)
tst-writev.c has a configuration macro 'ARTIFICIAL_LIMIT' that the
Makefiles are expected to define, and sysdeps/unix/sysv/linux/Makefile
was using the internal __getpagesize in the definition; changed to
sysconf(_SC_PAGESIZE) which is the POSIX equivalent.
ia64-linux doesn't supply 'clone', only '__clone2', which is not
defined in the public headers(!) All the other clone tests have local
extern declarations of __clone2, but tst-clone.c doesn't; it was
getting away with this because include/sched.h does declare __clone2.
* nss/tst-cancel-getpwuid_r.c: Include nss.h.
* string/strcasestr.c: No need to include config.h.
* sysdeps/powerpc/fpu/tst-setcontext-fpscr.c: Include
sys/auxv.h. Don't include sysdep.h.
* sysdeps/powerpc/tst-set_ppr.c: Don't include dl-procinfo.h.
* dlfcn/tst-dladdr.c: Don't include ldsodefs.h. Don't use
DL_LOOKUP_ADDRESS.
* math/test-misc.c: Instead of testing NO_LONG_DOUBLE, test whether
LDBL_MANT_DIG is greater than DBL_MANT_DIG.
* sysdeps/unix/sysv/linux/Makefile (CFLAGS-tst-writev.c): Use
sysconf (_SC_PAGESIZE) instead of __getpagesize in definition
of ARTIFICIAL_LIMIT.
* sysdeps/unix/sysv/linux/tst-clone.c [__ia64__]: Add extern
declaration of __clone2.
2016-11-20 20:46:30 -05:00
|
|
|
#include <nss.h>
|
2016-12-23 13:39:23 -05:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <semaphore.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <support/support.h>
|
|
|
|
|
|
|
|
sem_t started;
|
|
|
|
char *wbuf;
|
|
|
|
long wbufsz;
|
|
|
|
|
|
|
|
void
|
|
|
|
worker_free (void *arg)
|
|
|
|
{
|
|
|
|
free (arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void *
|
|
|
|
worker (void *arg)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
unsigned int iter = 0;
|
|
|
|
struct passwd pwbuf, *pw;
|
|
|
|
uid_t uid;
|
|
|
|
|
|
|
|
uid = geteuid ();
|
|
|
|
|
|
|
|
/* Use a reasonable sized buffer. Note that _SC_GETPW_R_SIZE_MAX is
|
|
|
|
just a hint and not any kind of maximum value. */
|
|
|
|
wbufsz = sysconf (_SC_GETPW_R_SIZE_MAX);
|
|
|
|
if (wbufsz == -1)
|
|
|
|
wbufsz = 1024;
|
|
|
|
wbuf = xmalloc (wbufsz);
|
|
|
|
|
|
|
|
pthread_cleanup_push (worker_free, wbuf);
|
|
|
|
sem_post (&started);
|
|
|
|
while (1)
|
|
|
|
{
|
|
|
|
iter++;
|
|
|
|
|
|
|
|
ret = getpwuid_r (uid, &pwbuf, wbuf, wbufsz, &pw);
|
|
|
|
|
|
|
|
/* The call to getpwuid_r may not cancel so we need to test
|
|
|
|
for cancellation after some number of iterations of the
|
|
|
|
function. Choose an arbitrary 100,000 iterations of running
|
|
|
|
getpwuid_r in a tight cancellation loop before testing for
|
|
|
|
cancellation. */
|
|
|
|
if (iter > 100000)
|
|
|
|
pthread_testcancel ();
|
|
|
|
|
|
|
|
if (ret == ERANGE)
|
|
|
|
{
|
|
|
|
/* Increase the buffer size. */
|
|
|
|
free (wbuf);
|
|
|
|
wbufsz = wbufsz * 2;
|
|
|
|
wbuf = xmalloc (wbufsz);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
pthread_cleanup_pop (1);
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
do_test (void)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
char *buf;
|
|
|
|
long bufsz;
|
|
|
|
void *retval;
|
|
|
|
struct passwd pwbuf, *pw;
|
|
|
|
pthread_t thread;
|
|
|
|
|
|
|
|
/* Configure the test to only use files. We control the files plugin
|
|
|
|
as part of glibc so we assert that it should be deferred
|
|
|
|
cancellation safe. */
|
|
|
|
__nss_configure_lookup ("passwd", "files");
|
|
|
|
|
|
|
|
/* Use a reasonable sized buffer. Note that _SC_GETPW_R_SIZE_MAX is
|
|
|
|
just a hint and not any kind of maximum value. */
|
|
|
|
bufsz = sysconf (_SC_GETPW_R_SIZE_MAX);
|
|
|
|
if (bufsz == -1)
|
|
|
|
bufsz = 1024;
|
|
|
|
buf = xmalloc (bufsz);
|
|
|
|
|
|
|
|
sem_init (&started, 0, 0);
|
|
|
|
|
|
|
|
pthread_create (&thread, NULL, worker, NULL);
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
ret = sem_wait (&started);
|
|
|
|
if (ret == -1 && errno != EINTR)
|
|
|
|
{
|
|
|
|
printf ("FAIL: Failed to wait for second thread to start.\n");
|
|
|
|
exit (EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
while (ret != 0);
|
|
|
|
|
|
|
|
printf ("INFO: Cancelling thread\n");
|
|
|
|
if ((ret = pthread_cancel (thread)) != 0)
|
|
|
|
{
|
|
|
|
printf ("FAIL: Failed to cancel thread. Returned %d\n", ret);
|
|
|
|
exit (EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
printf ("INFO: Joining...\n");
|
|
|
|
pthread_join (thread, &retval);
|
|
|
|
if (retval != PTHREAD_CANCELED)
|
|
|
|
{
|
|
|
|
printf ("FAIL: Thread was not cancelled.\n");
|
|
|
|
exit (EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
printf ("INFO: Joined, trying getpwuid_r call\n");
|
|
|
|
|
|
|
|
/* Before the fix in 312be3f9f5eab1643d7dcc7728c76d413d4f2640 for this
|
|
|
|
issue the cancellation point could happen in any number of internal
|
|
|
|
calls, and therefore locks would be left held and the following
|
|
|
|
call to getpwuid_r would block and the test would time out. */
|
|
|
|
do
|
|
|
|
{
|
|
|
|
ret = getpwuid_r (geteuid (), &pwbuf, buf, bufsz, &pw);
|
|
|
|
if (ret == ERANGE)
|
|
|
|
{
|
|
|
|
/* Increase the buffer size. */
|
|
|
|
free (buf);
|
|
|
|
bufsz = bufsz * 2;
|
|
|
|
buf = xmalloc (bufsz);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
while (ret == ERANGE);
|
|
|
|
|
|
|
|
free (buf);
|
|
|
|
|
|
|
|
/* Before the fix we would never get here. */
|
|
|
|
printf ("PASS: Canceled getpwuid_r successfully"
|
|
|
|
" and called it again without blocking.\n");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-01-05 17:39:38 +00:00
|
|
|
#define TIMEOUT 900
|
2016-12-23 13:39:23 -05:00
|
|
|
#include <support/test-driver.c>
|