mirror of
https://sourceware.org/git/glibc.git
synced 2025-08-01 10:06:57 +03:00
pthread: Move most once tests from nptl to sysdeps/pthread
So they can be checked with htl too.
This commit is contained in:
@ -51,11 +51,17 @@ tests += tst-cnd-basic tst-mtx-trylock tst-cnd-broadcast \
|
||||
tst-cond23 tst-cond24 tst-cond25 \
|
||||
tst-cond-except \
|
||||
tst-mutex1 tst-mutex2 tst-mutex3 tst-mutex4 tst-mutex6 tst-mutex10 \
|
||||
tst-once1 tst-once2 tst-once3 tst-once4 \
|
||||
tst-spin1 tst-spin2 tst-spin3 tst-spin4
|
||||
|
||||
tests += tst-oncex3 tst-oncex4
|
||||
|
||||
ifeq ($(build-shared),yes)
|
||||
tests-static += tst-cond8-static
|
||||
tests += tst-cond8-static
|
||||
endif
|
||||
|
||||
CFLAGS-tst-oncex3.c += -fexceptions
|
||||
CFLAGS-tst-oncex4.c += -fexceptions
|
||||
|
||||
endif
|
||||
|
50
sysdeps/pthread/tst-once1.c
Normal file
50
sysdeps/pthread/tst-once1.c
Normal file
@ -0,0 +1,50 @@
|
||||
/* Copyright (C) 2002-2020 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
|
||||
<https://www.gnu.org/licenses/>. */
|
||||
|
||||
#include <pthread.h>
|
||||
#include <stdio.h>
|
||||
|
||||
|
||||
static pthread_once_t once = PTHREAD_ONCE_INIT;
|
||||
|
||||
static int global;
|
||||
|
||||
static void
|
||||
once_handler (void)
|
||||
{
|
||||
++global;
|
||||
}
|
||||
|
||||
|
||||
static int
|
||||
do_test (void)
|
||||
{
|
||||
pthread_once (&once, once_handler);
|
||||
pthread_once (&once, once_handler);
|
||||
|
||||
if (global != 1)
|
||||
{
|
||||
printf ("global = %d, expected 1\n", global);
|
||||
return 1;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
#define TEST_FUNCTION do_test ()
|
||||
#include "../test-skeleton.c"
|
102
sysdeps/pthread/tst-once2.c
Normal file
102
sysdeps/pthread/tst-once2.c
Normal file
@ -0,0 +1,102 @@
|
||||
/* Copyright (C) 2002-2020 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
|
||||
<https://www.gnu.org/licenses/>. */
|
||||
|
||||
#include <pthread.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <time.h>
|
||||
|
||||
|
||||
#define N 100
|
||||
|
||||
static pthread_once_t once = PTHREAD_ONCE_INIT;
|
||||
|
||||
static int global;
|
||||
|
||||
static void
|
||||
once_handler (void)
|
||||
{
|
||||
struct timespec ts;
|
||||
|
||||
++global;
|
||||
|
||||
ts.tv_sec = 2;
|
||||
ts.tv_nsec = 0;
|
||||
nanosleep (&ts, NULL);
|
||||
}
|
||||
|
||||
|
||||
static void *
|
||||
tf (void *arg)
|
||||
{
|
||||
pthread_once (&once, once_handler);
|
||||
|
||||
if (global != 1)
|
||||
{
|
||||
printf ("thread %ld: global == %d\n", (long int) arg, global);
|
||||
exit (1);
|
||||
}
|
||||
|
||||
return NULL;
|
||||
}
|
||||
|
||||
|
||||
static int
|
||||
do_test (void)
|
||||
{
|
||||
pthread_attr_t at;
|
||||
pthread_t th[N];
|
||||
int cnt;
|
||||
|
||||
if (pthread_attr_init (&at) != 0)
|
||||
{
|
||||
puts ("attr_init failed");
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (pthread_attr_setstacksize (&at, 1 * 1024 * 1024) != 0)
|
||||
{
|
||||
puts ("attr_setstacksize failed");
|
||||
return 1;
|
||||
}
|
||||
|
||||
for (cnt = 0; cnt < N; ++cnt)
|
||||
if (pthread_create (&th[cnt], &at, tf, (void *) (long int) cnt) != 0)
|
||||
{
|
||||
printf ("creation of thread %d failed\n", cnt);
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (pthread_attr_destroy (&at) != 0)
|
||||
{
|
||||
puts ("attr_destroy failed");
|
||||
return 1;
|
||||
}
|
||||
|
||||
for (cnt = 0; cnt < N; ++cnt)
|
||||
if (pthread_join (th[cnt], NULL) != 0)
|
||||
{
|
||||
printf ("join of thread %d failed\n", cnt);
|
||||
return 1;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
#define TEST_FUNCTION do_test ()
|
||||
#include "../test-skeleton.c"
|
165
sysdeps/pthread/tst-once3.c
Normal file
165
sysdeps/pthread/tst-once3.c
Normal file
@ -0,0 +1,165 @@
|
||||
/* Copyright (C) 2002-2020 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
|
||||
<https://www.gnu.org/licenses/>. */
|
||||
|
||||
#include <pthread.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <time.h>
|
||||
|
||||
|
||||
#define N 100
|
||||
|
||||
static pthread_once_t once = PTHREAD_ONCE_INIT;
|
||||
|
||||
static pthread_cond_t cond = PTHREAD_COND_INITIALIZER;
|
||||
static pthread_mutex_t mut = PTHREAD_MUTEX_INITIALIZER;
|
||||
|
||||
static pthread_barrier_t bar;
|
||||
|
||||
static int global;
|
||||
static int cl_called;
|
||||
|
||||
static void
|
||||
once_handler1 (void)
|
||||
{
|
||||
if (pthread_mutex_lock (&mut) != 0)
|
||||
{
|
||||
puts ("once_handler1: mutex_lock failed");
|
||||
exit (1);
|
||||
}
|
||||
puts ("once_handler1: locked");
|
||||
|
||||
int r = pthread_barrier_wait (&bar);
|
||||
if (r != 0 && r!= PTHREAD_BARRIER_SERIAL_THREAD)
|
||||
{
|
||||
puts ("once_handler1: barrier_wait failed");
|
||||
exit (1);
|
||||
}
|
||||
|
||||
puts ("once_handler1: going to wait on cond");
|
||||
|
||||
pthread_cond_wait (&cond, &mut);
|
||||
|
||||
/* We should never get here. */
|
||||
exit (42);
|
||||
}
|
||||
|
||||
static void
|
||||
once_handler2 (void)
|
||||
{
|
||||
global = 1;
|
||||
}
|
||||
|
||||
|
||||
static void
|
||||
cl (void *arg)
|
||||
{
|
||||
cl_called = 1;
|
||||
}
|
||||
|
||||
|
||||
static void *
|
||||
tf (void *arg)
|
||||
{
|
||||
pthread_cleanup_push (cl, NULL)
|
||||
|
||||
pthread_once (&once, once_handler1);
|
||||
|
||||
pthread_cleanup_pop (0);
|
||||
|
||||
/* We should never get here. */
|
||||
puts ("pthread_once in tf returned");
|
||||
exit (1);
|
||||
}
|
||||
|
||||
|
||||
static int
|
||||
do_test (void)
|
||||
{
|
||||
pthread_t th;
|
||||
|
||||
if (pthread_barrier_init (&bar, NULL, 2) != 0)
|
||||
{
|
||||
puts ("barrier_init failed");
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (pthread_create (&th, NULL, tf, NULL) != 0)
|
||||
{
|
||||
puts ("first create failed");
|
||||
return 1;
|
||||
}
|
||||
|
||||
int r = pthread_barrier_wait (&bar);
|
||||
if (r != 0 && r!= PTHREAD_BARRIER_SERIAL_THREAD)
|
||||
{
|
||||
puts ("barrier_wait failed");
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (pthread_mutex_lock (&mut) != 0)
|
||||
{
|
||||
puts ("mutex_lock failed");
|
||||
return 1;
|
||||
}
|
||||
/* We unlock the mutex so that we catch the case where the pthread_cond_wait
|
||||
call incorrectly resumes and tries to get the mutex. */
|
||||
if (pthread_mutex_unlock (&mut) != 0)
|
||||
{
|
||||
puts ("mutex_unlock failed");
|
||||
return 1;
|
||||
}
|
||||
|
||||
/* Cancel the thread. */
|
||||
puts ("going to cancel");
|
||||
if (pthread_cancel (th) != 0)
|
||||
{
|
||||
puts ("cancel failed");
|
||||
return 1;
|
||||
}
|
||||
|
||||
void *result;
|
||||
pthread_join (th, &result);
|
||||
if (result != PTHREAD_CANCELED)
|
||||
{
|
||||
puts ("join didn't return PTHREAD_CANCELED");
|
||||
return 1;
|
||||
}
|
||||
puts ("joined successfully");
|
||||
|
||||
printf ("once = %d\n", *(int *) &once);
|
||||
|
||||
if (cl_called != 1)
|
||||
{
|
||||
puts ("cleanup handler not called");
|
||||
return 1;
|
||||
}
|
||||
|
||||
pthread_once (&once, once_handler2);
|
||||
|
||||
if (global != 1)
|
||||
{
|
||||
puts ("global still 0");
|
||||
return 1;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
#define TEST_FUNCTION do_test ()
|
||||
#include "../test-skeleton.c"
|
200
sysdeps/pthread/tst-once4.c
Normal file
200
sysdeps/pthread/tst-once4.c
Normal file
@ -0,0 +1,200 @@
|
||||
/* Copyright (C) 2003-2020 Free Software Foundation, Inc.
|
||||
This file is part of the GNU C Library.
|
||||
Contributed by Ulrich Drepper <drepper@redhat.com>, 2003.
|
||||
|
||||
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
|
||||
<https://www.gnu.org/licenses/>. */
|
||||
|
||||
#include <pthread.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <time.h>
|
||||
#include <unistd.h>
|
||||
|
||||
|
||||
static pthread_once_t once = PTHREAD_ONCE_INIT;
|
||||
|
||||
static pthread_cond_t cond = PTHREAD_COND_INITIALIZER;
|
||||
static pthread_mutex_t mut = PTHREAD_MUTEX_INITIALIZER;
|
||||
|
||||
static pthread_barrier_t bar;
|
||||
|
||||
static int global;
|
||||
static int cl_called;
|
||||
|
||||
static void
|
||||
once_handler1 (void)
|
||||
{
|
||||
if (pthread_mutex_lock (&mut) != 0)
|
||||
{
|
||||
puts ("once_handler1: mutex_lock failed");
|
||||
exit (1);
|
||||
}
|
||||
|
||||
int r = pthread_barrier_wait (&bar);
|
||||
if (r != 0 && r!= PTHREAD_BARRIER_SERIAL_THREAD)
|
||||
{
|
||||
puts ("once_handler1: barrier_wait failed");
|
||||
exit (1);
|
||||
}
|
||||
|
||||
pthread_cond_wait (&cond, &mut);
|
||||
|
||||
/* We should never get here. */
|
||||
}
|
||||
|
||||
|
||||
static void
|
||||
once_handler2 (void)
|
||||
{
|
||||
global = 1;
|
||||
}
|
||||
|
||||
|
||||
static void
|
||||
cl (void *arg)
|
||||
{
|
||||
++cl_called;
|
||||
}
|
||||
|
||||
|
||||
static void *
|
||||
tf1 (void *arg)
|
||||
{
|
||||
pthread_cleanup_push (cl, NULL);
|
||||
|
||||
pthread_once (&once, once_handler1);
|
||||
|
||||
pthread_cleanup_pop (0);
|
||||
|
||||
/* We should never get here. */
|
||||
puts ("pthread_once in tf returned");
|
||||
exit (1);
|
||||
}
|
||||
|
||||
|
||||
static void *
|
||||
tf2 (void *arg)
|
||||
{
|
||||
pthread_cleanup_push (cl, NULL);
|
||||
|
||||
int r = pthread_barrier_wait (&bar);
|
||||
if (r != 0 && r!= PTHREAD_BARRIER_SERIAL_THREAD)
|
||||
{
|
||||
puts ("once_handler2: barrier_wait failed");
|
||||
exit (1);
|
||||
}
|
||||
|
||||
pthread_cleanup_pop (0);
|
||||
|
||||
pthread_once (&once, once_handler2);
|
||||
|
||||
return NULL;
|
||||
}
|
||||
|
||||
|
||||
static int
|
||||
do_test (void)
|
||||
{
|
||||
pthread_t th[2];
|
||||
|
||||
if (pthread_barrier_init (&bar, NULL, 2) != 0)
|
||||
{
|
||||
puts ("barrier_init failed");
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (pthread_create (&th[0], NULL, tf1, NULL) != 0)
|
||||
{
|
||||
puts ("first create failed");
|
||||
return 1;
|
||||
}
|
||||
|
||||
int r = pthread_barrier_wait (&bar);
|
||||
if (r != 0 && r!= PTHREAD_BARRIER_SERIAL_THREAD)
|
||||
{
|
||||
puts ("first barrier_wait failed");
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (pthread_mutex_lock (&mut) != 0)
|
||||
{
|
||||
puts ("mutex_lock failed");
|
||||
return 1;
|
||||
}
|
||||
/* We unlock the mutex so that we catch the case where the pthread_cond_wait
|
||||
call incorrectly resumes and tries to get the mutex. */
|
||||
if (pthread_mutex_unlock (&mut) != 0)
|
||||
{
|
||||
puts ("mutex_unlock failed");
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (pthread_create (&th[1], NULL, tf2, NULL) != 0)
|
||||
{
|
||||
puts ("second create failed");
|
||||
return 1;
|
||||
}
|
||||
|
||||
r = pthread_barrier_wait (&bar);
|
||||
if (r != 0 && r!= PTHREAD_BARRIER_SERIAL_THREAD)
|
||||
{
|
||||
puts ("second barrier_wait failed");
|
||||
return 1;
|
||||
}
|
||||
|
||||
/* Give the second thread a chance to reach the pthread_once call. */
|
||||
sleep (2);
|
||||
|
||||
/* Cancel the thread. */
|
||||
if (pthread_cancel (th[0]) != 0)
|
||||
{
|
||||
puts ("cancel failed");
|
||||
return 1;
|
||||
}
|
||||
|
||||
void *result;
|
||||
pthread_join (th[0], &result);
|
||||
if (result != PTHREAD_CANCELED)
|
||||
{
|
||||
puts ("first join didn't return PTHREAD_CANCELED");
|
||||
return 1;
|
||||
}
|
||||
|
||||
puts ("joined first thread");
|
||||
|
||||
pthread_join (th[1], &result);
|
||||
if (result != NULL)
|
||||
{
|
||||
puts ("second join didn't return PTHREAD_CANCELED");
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (global != 1)
|
||||
{
|
||||
puts ("global still 0");
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (cl_called != 1)
|
||||
{
|
||||
printf ("cl_called = %d\n", cl_called);
|
||||
return 1;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
#define TEST_FUNCTION do_test ()
|
||||
#include "../test-skeleton.c"
|
1
sysdeps/pthread/tst-oncex3.c
Normal file
1
sysdeps/pthread/tst-oncex3.c
Normal file
@ -0,0 +1 @@
|
||||
#include "tst-once3.c"
|
1
sysdeps/pthread/tst-oncex4.c
Normal file
1
sysdeps/pthread/tst-oncex4.c
Normal file
@ -0,0 +1 @@
|
||||
#include "tst-once4.c"
|
Reference in New Issue
Block a user